Новое…

Удобный и безопасный torrent-клиент Deluge. Установка на Ubuntu Server.

Торренты прочно вошли в нашу жизнь как удобное средство обмена файлами через сеть интернет. Однако, рано или поздно встаёт вопрос, насколько безопасно это использование и рационально ли мы используем для этого компьютер. Торрент-обмен строится на базе пиринговой сети, о которой я упоминал в другой своей статье – о клиенте и сервере, а пользователь использует клиентское приложение, которое и выполняет всю работу по файловому обмену. Весь вопрос безопасности сводится к тому, кем изготовлено это приложение и насколько качественно оно сделано. Если на компьютере, где установлен торрент-клиент, находится какая-либо конфиденциальная информация (личные файлы, семейные фотографии, сканы документов), то я не считаю установку торрент-клиента на такой компьютер хотя бы сколько-нибудь безопасной. Гораздо было бы удобнее разместить торрент-клиента на отдельном компьютере или виртуальной машине, давать удаленно задачи на файловый обмен и забирать готовые файлы по мере выполнения. Об этом речь в этой статье, а устанавливать будем софт на Ubuntu 22.04 Server. В качестве клиента будет использоваться веб-интерфейс.

Сначала устанавливаем сам сервер Deluge и его вспомогательный веб-сервис:

sudo apt-get install deluged deluge-web

В Ubuntu 22.04 вместе с установкой Deluge не происходит автоматического формирования файлов настройки сервиса, по этому их нужно создать вручную. Для этого нужно открыть необходимый каталог:

cd /etc/systemd/system

Теперь создадим сначала файла deluged.service:

sudo nano deluged.service

и наполним его следующим содержимым:

[Unit]
Description=Deluge Bittorrent Client Daemon
Documentation=man:deluged
After=network-online.target

[Service]
Type=simple
User=deluge
Group=deluge
UMask=000
ExecStart=/usr/bin/deluged -d
Restart=on-failure
TimeoutStopSec=300

[Install]
WantedBy=multi-user.target

После этого создадим файл deluge-web.service:

sudo nano deluge-web.service

Со следующим содержимым:

[Unit]
Description=Deluge Bittorrent Client Web Interface
Documentation=man:deluge-web
After=network-online.target deluged.service
Wants=deluged.service

[Service]
Type=simple
User=deluge
Group=deluge
UMask=022
ExecStart=/usr/bin/deluge-web -d
Restart=on-failure

[Install]
WantedBy=multi-user.target

Проверьте, действительно ли исполняемые файлы находятся по этому пути (ну кто его знает куда их в следующий раз запихнут):


/usr/bin/deluged

/usr/bin/deluge-web

Если вдруг оказалось, что там их нет, то найдём их, для этого сначала возвратимся в корневой каталог и запустим поиск:

cd /

find / -name deluged

Лично у меня с этим проблем не возникло:

Теперь нужно создать пользователя, того самого, которого указали в настройках сервиса (это был пользователь deluge). От его имени будет выполняться сервис. Создаётся он просто:

sudo adduser deluge

Пароль вводится любой, он всё равно нам вряд ли понадобится, в конце выражаем согласие игреком:

Далее включаем и запускаем работу сервисов:

sudo systemctl enable deluged.service

sudo systemctl enable deluge-web.service

sudo systemctl start deluged.service

sudo systemctl start deluge-web.service

Теперь открываем браузер и вводим путь к вашему Ubuntu серверу с Deluge. У меня это 192.168.1.24 (не забываем указать специальный порт интерфейса):

Пароль входа – это НЕ тот пароль, который вы ставили для пользователя deluge. Это пароль по-умолчанию, который тоже deluge.

На вопрос не хочу ли я сменить этот пароль, лично я отвечаю “Нет”, т.к. кроме меня в сети никто файлы качать не будет, и админить их тоже:

Далее появится окно выбора сервера. Нужно кликнуть (выбрать) по одной единственной строке и далее по кнопке “Start Daemon”:

Сервис веб состыкуется с сервисом deluged (это почему-то обозначается как старт демона) и теперь можно подключиться (нажать кнопку Connect – она станет доступна):

Появится рабочее окно списка торрентов:

Теперь, когда мы убедились, что всё работает, нужно донастроить сервер. Добавлять вручную торренты дело заморочное, по этому хорошо бы каталог, куда мы сразу записывали торрент-файлы (файлы с расширением .torrent) и хорошо бы иметь доступ к папке с уже скачанными файлами, но об этом позже. А пока создадим два каталога внутри домашнего каталога deluge:

sudo mkdir /home/deluge/data

sudo mkdir /home/deluge/torrents

Первая – для данных, вторая для торрент-файлов

Установим хозяина владельца на папки:

sudo chown deluge:deluge /home/deluge/data

sudo chown deluge:deluge /home/deluge/torrents

и проверим, всё ли норм:

Теперь объясню зачем это сделано именно так. Дело в том, что Deluge я устанавливаю на чистый виртуальный свежеустановленный Ubuntu server с минимальными настройками, в частности – 16 ГБ жесткого диска. Конечно, такого размера не хватит практически ни на что. Для того чтобы иметь много пространства, я подключу физический старый дряхлых жесткий диск и прокину его внутрь этой виртуальной. Делаю я так, потому что мне для торрентов жалко нового диска, и рано или поздно он своё отслужит, и когда накроется, я не хочу снова устанавливать Ubuntu, Deluge, Samba и т.п., я просто хочу прокинуть новый такой же старый и дряхлых (извиняюсь за каламбур) жесткий диск и продолжить сразу же работу. Вопрос будет только в том, чтобы перемонтировать этот диск в папку /home/deluge/data и проблема будет решена. Вот для чего я создал папку data – туда будет монтироваться диск большого объёма. Для торрент-файлов такой подход будет избыточным – это небольшие файлики, содержащие только управляющую информацию на закачку.

После подключения диска для готовых файлов к виртуальной (или физической) машине (это не имеет значения), на нём хорошо бы переопределить разделы и отформатировать, но сначала я его найду в списке устройств:

cd /dev

ls -al | grep sd

В результате система покажет какие диски у нас присутствуют:

Вот этот sdb без номера и есть наш новый старый дряхлый диск…хотя вру, у меня это новый виртуальный, новее некуда…на старом дряхлом физическом)))

Я дам ему один единственный раздел всего объёма:

sudo fdisk /dev/sdb

Введем сначала команду n и всё “отэнтерим”, проверим разделы командой p, а затем введем команду w для сохранения и выхода:

Да, диск я взял новый не такой уж и большой – 64 ГБ (не оказалось свободного места на датасторе перед написанием статьи), но это не так критично )).

Проверим, что нового появилось в дисках:

Всё верно – появился диск sdb1. Это как раз и есть наш раздел, но прежде чем сможем его использовать, нужно его отформатировать. Для этого я буду использовать следующую команду (опять отэнтерив вопросы):

sudo mkfs.ext4 /dev/sdb1

Получим результат:

Убедимся, что всё прошло ровно командой blkid:

Видим, что диск sdb1 имеет идентификатор и признак файловой системы ext4.

Создадим на диск папку, куда будем при загрузке системы монтировать этот диск:

sudo mkdir /torrents-disk

Добавим в файл /etc/fstab новую строку:

UUID=51a19ac1-7b9e-4553-ab29-9c99eae704f9 /torrents-disk ext4

Должно получиться как-то так:

Сохраним файл в новом виде и перезагрузим машину командой reboot.

После загрузки проверим всё ли мы сделали правильно командой:

df -h /torrents-disk

Должно получиться как-то так:

Информация свидетельствует о том, что при загрузке системы диск автоматически примонтировался успешно.

Создадим внутри каталога /torrents-disk каталог data:

sudo mkdir /torrents-disk/data

Примонтируем его в папку /home/deluge/data (если вы ещё не забыли зачем мы его вообще в систему добавили):

sudo mount -o bind /torrents-disk/data /home/deluge/data

Убеждаемся, что и папка /home/deluge/data ссылается теперь на наш “большой” диск:

Теперь, возможно у вас возник вопрос,- почему я прямо в fstab не примонтировал диск внутрь /home/deluge. Ответ прост – мне так удобнее в обслуживании, и ещё диск этот может мне понадобиться не только для торрентов, и захочу отдельную папку его примонтировать в другую совсем не связанную с торрент-сервисом. Если 2-3 ТБ, пусть он старый, но использовать его для торрентов, как-то неправдоподобно и бессмысленно. Я такой объем торрентов точно не буду вводить в обмен, по этому буду разделять диск с ещё другими сервисами, какими – потом решу.

Вторую команду монтирования тоже хорошо бы запихнуть куда-нибудь для автоматического выполнения при загрузке системы. Запихну-ка я её в rc.local. Однако, если мы посмотрим внимательно на свежий Ubuntu 22.04, там этого файла нет, а сервис находится в полной отключке))). Исправим это недоразумение)).

Создадим файл /etc/rc.local:

sudo nano /etc/rc.local

Содержимое его будет такое:

#!/bin/sh -e

mount -o bind /torrents-disk/data /home/deluge/data

exit

Сохраним его и дадим права и установим как исполняемый:

chmod 0744 /etc/rc.local

Создадим файл настройки сервиса:

sudo nano /etc/systemd/system/rc-local.service

Должно получиться так:

Включим, стартанём и проверим сервис rc-local.service:

sudo systemctl enable rc-local.service

sudo systemctl start rc-local.service

sudo systemctl status rc-local.service

Должно быть так:

Давайте перезагрузим наш сервачок и убедимся, что после ребута он будет делать ровно то, что должен. Проверим папку /home/deluge/data:

Теперь создадим три каталога внутри /torrents-disk/data.

sudo mkdir /torrents-disk/data/downloads

sudo mkdir /torrents-disk/data/downloaded

sudo mkdir /torrents-disk/data/autouploads

downloads будет хранить торренты пока они находятся в процессе закачки (до конца незакачанные торренты)

downloaded будет хранить файлы, которые полностью закачены и готовы к употреблению, так сказать

autouploads будет хранить торрент файлы .torrents. В эту папку мы будем сохранять задания на закачку, т.е. сами торренты.

На каталог autouploads нужно дать все права, чтобы гостевой пользователь мог туда сохранять торренты:

sudo chmod 0777 /torrents-disk/data/autouploads

На папку /torrents-disk/data и всех, которые внутри устанавливаем владельца:

sudo chown -R deluge:deluge /torrents-disk/data

Теперь организуем сетевой доступ к папке с торрентами и к папке с готовыми загруженными файлами. Делать я это буду путём установки в систему сервиса Samba:

sudo apt-get install samba

Настроим чтобы сервис мог видеть наши папки, для этого отредактируем необходимый файл:

sudo nano /etc/samba/smb.conf

В конец файла допишем:

[autouploads]
comment = Auto uploads
browseable = yes
path = /torrents-disk/data/autouploads
guest ok = yes
read only = no
create mask = 0777

[downloaded]
comment = Downloaded
browseable = yes
path = /torrents-disk/data/downloaded
guest ok = yes
read only = yes
create mask = 0700

Должно получиться так:

В папку autouploads можно будет писать файлы, а из папки downloaded можно будет только читать файлы (оно и логично).

Перезапустим сервис Samba:

sudo /etc/init.d/smbd restart

Проверим, что по сети стал доступен сам сервер \\192.168.1.24\ (вместо моего IP тут будет IP вашего), а папка autouploads доступна для записи (я просто создал тестовый файл):

Попробуем сохранить туда торрент файл:

Файл торрента действительно сохранился в сетевой папке, однако, ожидать его закачки ещё рановато. Нужно настроить сам Deluge и привязать его к нашим каталогам. Откроем эти настройки в веб-интерфейсе:

Сразу же видим, что папка загрузки указано неверно. В Линуксе имеет значение заглавная буква имен или нет. Правильный каталог не Downloads, а downloads. Также сами каталоги находятся внутри /home/deluge/data. Укажем и остальные каталоги и нажмем кнопку Apply:

Переходим в раздел Plugins и добавляем новый плагин AutoAdd. Он нужен чтобы Deluge автоматически забирал файлы из каталога с торрентами (где у нас уже ожидает по крайней мере один файл с торрентом), потом также нажимаем Apply:

Переходим во вновь появившийся раздел AutoAdd и добавляем папку слежения за торрентами, а затем кнопку Add и опять-таки кнопку Apply:

После этого окно закрывается и торрент сразу же начинает скачиваться:

Файл скачался и мы можем его теперь скопировать из папки (по факту это оказался не просто один файл, а сразу каталог с несколькими файлами):

Статья создана El Vinto, 2023 (Исходник на сайте https://elvinto.ru)

Коллекторы против программиста

Ранее я уже рассказывал, как банк на меня подавал в суд и по результатам нескольких судебных заседаний дело было оставлено без рассмотрения ввиду отсутствия достаточных доказательств получения кредита. По прошествии года как дело было передано в архив, банк, совершив ещё около сотни телефонных звонков с требованием отдать деньги, внезапно отменил свои требования и закрыл производство по взыскании задолженности. Однако, вместе ещё с другими делами по взысканию других клиентов, на помойку выброшено не было, а было продано коллекторам. Сумма сделки составила около 70 млн.рублей (эта информация была в материалах дела, о чем далее). Я представляю сколько тысяч долгов было перекуплено у банка. Естественно, коллекторы хотят на покупке долгов сделать деньги и рассматривают это как надёжное вложение капитала. Ну, кто-то в производство российских танков и самолётов деньги вкладывает, а кто-то в долги российских граждан. Лично для меня сразу становится понятно куда идут от такого бизнеса доходы и на чьей стороне эти бизнесмены. И ладно я, и не понятно как на меня “повешенными” банком 35 тыс. руб, а кто-то действительно брал чтобы прокормить семью, вырастить детей, дать им образование, но не рассчитал и не смог отдать. Какая бизнесу разница, это очередной источник прибыли. Даже если у тебя нет денег, ты всё равно источник денег. Деньги не пахнут. Ну да ладно философии, ближе к делу…причем гражданскому делу…В связи с этим, мониторя в очередной раз сайт своего судебного участка, я обнаруживаю, что некая организация, общество с ограниченной ответственностью подало на меня в суд по взысканию долга ссылаясь на договор цессии как на доказательство переуступки прав требований (именно его копия с суммой в 70 млн. и была в деле). О судебном заседании я узнал буквально за три дня, по этому пришёл пораньше, за полчаса и попросил ознакомиться с материалами дела, так и объяснив, что только в пятницу узнал, что в понедельник у меня заседание. Когда я стал фотографировать листы, они мне показались удивительно знакомыми, я ведь видел их уже тогда, когда ознакамливался в прошлый раз, только в тогда сам банк на меня подавал в суд. В связи с этим, я на судебном заседании так и заявил, что все доказательство предоставленные суду не могут быть приобщены к делу как доказательства, т.к. представляют из себя просто скопированные листы бумаги без печатей и подписей и при этом не являются подлинниками. Однако, в этот раз та же самая судья была совершенно другого мнения, и сказала, что она назначает следующее заседание через 20 дней, но при этом так и сказала, что скорее всего решение будет не в мою пользу. Как же так??? Где эта Фемида? В прошлый раз не было достаточных доказательств, а в этот раз эти же доказательства уже достаточны для вынесения решения не мою пользу! Я с этим не согласился и стал готовить апелляцию.

На само судебное заседание я не пошёл. И правильно сделал, т.к. если бы пошёл, то времени на составления и подачу заявления на выдачу мотивированного решения у меня было бы 3 дня, а так получилось 15 дней. Тут, должен сказать, очередность такая:

  • Судья выносит решение, но оно неполное, ну там написано мол так и так, суд решил и текст что решил. Нет никакой описательной части, почему суд так решил, чем руководствовался, на какие факты опирался. С таким решением апелляцию не подать.
  • По этому нужно теперь сделать запрос на составление мотивированного решения. И вот подать его можно в течение 3 дней со дня принятия решения по делу, если заявитель был на судебном заседании, либо в течение 15 дней, если не был. Я не был, по этому неспеша всё подготовил и направил. Такое мотивированное решение ещё называется решением в окончательном виде.
  • Далее составляется апелляционная жалоба, в которой в частности указывается дата когда же было вынесено решение в окончательном виде. Потому что через 30 дней подать апелляционную жалобу будет возможно только если есть существенные обоснования почему не подал во время, в ином случае подать её будет уже невозможно и через месяц решение вступит в законную силу.

Однако, по прошествии месяца со дня подачи заявления, я мотивированного решения так и не получил. Я позвонил в суд и объяснил ситуацию, мол понимаете если вы через неделю мне его отправите, а там будет стоять дата мотивированного решения такая же как дата подачи заявления, то получится что у меня уже будет просрочка подачи апелляционной жалобы. Уверованиям секретаря, что всё будет нормально, я не доверил, по этому подал, не дожидаясь мотивированного решения, краткую апелляционную жалобу. Формально такого документа не существует, по этому суд принимает его как обычную апелляционную жалобу, регистрирует, что она подана во время, смотрит что в ней нет существенно важных описательных моментов и возвращает заявителю на доработку. Я подал с таким содержанием:

В ****** районный суд
Санкт-Петербурга
тут адрес суда

Заявитель (Ответчик): Мои ФИО
Адрес: Мой адрес
Тел.: Мой моб.телефон

Истец: ООО ********

Гражданское дело № *******

Апелляционная жалоба на решение мирового судьи

Решением мирового судьи судебного участка № *** по ******* району Санкт-Петербурга *** 2023 г. вынесено решение по гражданскому делу № ********* по иску ООО ******** к (Мои ФИО) о взыскании задолженности по кредитному договору, судебных расходов. Исковые требования были удовлетворены полностью. Считаю решение мирового судьи по делу № ***** о взыскании задолженности по кредитному договору, судебных расходов незаконным и необоснованным по следующим основаниям:
Согласно ч.2 ст.71 Гражданского процессуального кодекса письменные доказательства предоставляются в подлиннике или надлежащим образом заверенной копии. Однако, график платежей, выписка по счету, заявление заемщика, заявление на получение кредита, предоставленные Истцом суду, надлежащим образом не заверены и не являются подлинниками.
Истец требует взыскать задолженность в рамках кредитного договора, однако, кредитный договор в качестве доказательства суду предоставлен не был.
В судебном заседании ***** 2023 г. судьей была оглашена лишь резолютивная часть решения.
С решением суда не согласен, считаю, что суд дал неверную оценку представленным доказательствам, неправильно применил закон, не выяснил обстоятельства, имеющие существенное значение для рассмотрения дела.
*** 2023 г., по истечении установленного законом срока для составления мотивированного решения, мотивированного решения в канцелярии не было. В этот же день секретарь *** судебного участка (тут фамилия секретаря) сообщила мне, что решение не готово, а готово будет только через неделю -**** 2023 г.
Настоящая апелляционная жалоба является предварительной, в виду того, что не получено решение в окончательной форме.
На основании вышеизложенного и руководствуясь действующим законодательством,


Прошу:


Отменить решение о взыскании задолженности по кредитному договору, судебных расходов и принять новое решение, которым отказать в удовлетворении исковых требований ООО *** в полном объеме.
Мотивированная жалоба будет подана после вынесения решения в окончательной форме.

Приложение:
Копия подтверждающая оплату государственной пошлины.
Копия описи письма подтверждающего запрос на составление мотивированного решения.
Копия апелляционной жалобы для Истца.

*** 2023 года ______ / Моя фамилия и инициалы/

Т.е. я не указывал, что это краткая/не краткая, просто апелляционная жалоба. Пошлина составляет 150 руб. Реквизиты я выписал с сайта районного суда, где находится апелляционный суд (у нас это располагается в районном суде), пришёл с ними в банк и предъявил к оплате. Девочка-операционист сама их ввела как надо и отдала мне чек. Этот чек я отсканил и приложил в письме к этой жалобе. Вложил туда копию для истца. Копия описи, т.е. письмо отправлял Почтой России с трек-номером, заказное с описью вложения. На почте знают.

Через неделю мне Почтой России пришло наконец-таки мотивированное решение. К слову сказать, дата там стояла по честному, зря не доверял.

Изучая мысли судьи при принятии решения, я читал и удивлялся. И чем больше читал, тем большее удивление у меня это вызывало. Скажу проще, если зеркалирую её работу и мысли на мою специальность:

Приходит ко мне начальник и говорит, вот Сергей тебе задание сделать отчет по номенклатурным остаткам, даю тебе 1 час. Ты приносишь через неделю ему отчет, в котором действительно есть номенклатура, есть остатки, но нет между ними соответствия, и удивлённо так говоришь, а какая разница, у нас 5 тонн яблок и 20 тонн топинамбура или 5 тонн топинамбура и 20 тонн яблок, всего-то 25 тонн, значит всё норм!

Вот именно так и выглядело мотивированное решение судьи…По результатам ознакомления я составил уже полную апелляционную жалобу:

В ****** районный суд
Санкт-Петербурга
тут адрес суда

Заявитель (Ответчик): Мои ФИО
Адрес: Мой адрес
Тел.: Мой моб.телефон

Истец: ООО ********

Гражданское дело № *******

Апелляционная жалоба на решение мирового судьи

Решением мирового судьи судебного участка № *** по ******* району Санкт-Петербурга *** 2023 г. вынесено решение по гражданскому делу № ********* по иску ООО ******** к (Мои ФИО) о взыскании задолженности по кредитному договору, судебных расходов. Исковые требования были удовлетворены полностью.

Данное решение было изготовлено в окончательной форме ***2023 г.
Считаю решение мирового судьи по делу № ****** о взыскании задолженности по кредитному договору № **** от ****.2018, судебных расходов незаконным и необоснованным по следующим основаниям.
Согласно ч.2 ст.71 Гражданского процессуального кодекса письменные доказательства предоставляются в подлиннике или надлежащим образом заверенной копии. Однако, график платежей, выписка по счету, заявление заемщика, заявление на получение кредита, предоставленные Истцом суду, надлежащим образом не заверены и не являются подлинниками, однако суд счел эти факты несущественными и данные документы были приняты судом в качестве доказательства.
Истец требует взыскать задолженность в рамках кредитного договора. Согласно ч.2 ст.820 Гражданского кодекса, кредитный договор должен быть заключен в письменном виде. Истец кредитный договор в качестве доказательства суду не предоставил ни в качестве надлежащим образом заверенной копии, ни в подлиннике. Однако суд счёл эти факты несущественными.
В мотивированном решении суд указывает доводы при принятии решения:

  1. Из материалов дела следует, что мной представлено заявление, подтверждающее согласие с условиями договора о комплексном банковском облуживании физических лиц с индивидуальными и общими условиями, подписанное простой цифровой подписью. Считаю этот довод суда ошибочным, так как согласие с условиями комплексного банковского обслуживания носит общих характер, было заключено в рамках зарплатного проекта, и не имеет отношение к данному исковому требованию, является лишь волеизъявлением о намерении открытия зарплатного счета, обслуживания его банком и получения предложений на банковские продукты, в том числе иметь возможность узнавать о кредитных предложениях, их условиях и возможности их предоставления, но не является волеизъявлением заключения какой-либо определенной кредитной сделки. Согласно ст.60 ГПК РФ, обстоятельства дела, которые в соответствии с законом должны быть подтверждены определенными средствами доказывания, не могут подтверждаться никакими другими доказательствами, в следствии этого считаю, что если Истец утверждает о заключении кредитного договора, то суду должен быть предоставлен в качестве доказательства именно кредитный договор, а не какой-либо иной документ. Считаю данное заявление, предоставленное Истцом суду, недопустимым доказательством.
  2. Суд принимает во внимание довод, что между мной и Истцом документы подписывались простой электронной подписью. Считаю этот довод ошибочным, так как Истец не предоставил суду договора между мной и центром сертификации об изготовлении такой электронной подписи, которой и были якобы подписаны указанные Истцом документы. Иные цифровые ключи, логины и пароли, в том числе создаваемые для своих клиентов самим банком при помощи собственных технических средств, считаю ничтожными, так как в качестве доказательства не было предоставлено лицензии, по которой Истец имеет законное право выпускать электронные подписи, имеющие юридическую силу. Истец не предоставил доказательств, по которым гарантируется, что электронный ключ этой подписи исключает возможность использования её самим банком или третьими лицами, и однозначно определяет, что она может быть использована только её владельцем.
  3. Суд в своем доводе применяет п.1 ст.160 ГК РФ и считает, что по данному кредитному договору сделку заключенную в электронном виде, можно считать совершенной. Считаю этот довод ошибочным, поскольку в этой же статье указано: «позволяющих воспроизвести на материальном носителе в неизменном виде содержание сделки». Предоставленные в качестве доказательства Истцом суду сведения на бумажном носителе невозможно характеризовать как доказательства «в неизменном виде», так как являются обычными скопированными листами бумаги, и невозможно однозначно определить действительно ли они находятся в неизменном виде.
  4. Суд в своем решении принимает норму закона, по которой законодательство Российской Федерации допускает заключение кредитного договора путем использования кодов, паролей или иных средств подтверждения факта формирования электронной подписи при условии соглашения сторон о специальном способе определения лица, выразившего волю на заключение договора. Считаю этот довод суда ошибочным, поскольку в нем не отражена статья Гражданского кодекса, по которой законодательство допускает заключение в таком виде именно кредитного договора. Так же суд не принял во внимание, что такое соглашение сторон между мной и Истцом, по которому данный кредитный договор мог бы быть изготовлен не в письменном виде и подписан с использованием кодов, паролей или иных средств, Истцом суду предоставлено не было.
  5. Суд принял в качестве доказательства выписку банка по счету и на её основании сделал вывод, что перечисленные средства были именно суммой по кредиту. Считаю этот довод ошибочным, поскольку согласно ст.26 №395-1 ФЗ, кредитная организация, Банк России, организация, осуществляющая функции по обязательному страхованию вкладов, гарантируют тайну об операциях, о счетах и вкладах своих клиентов и корреспондентов. Истец не предоставил суду соглашения между мной и банком, что я даю согласие на разглашение данной информации и предоставлении её третьим лицам, в том числе по договору цессии. В связи с этим считаю, что эти сведения получены Истцом незаконно и согласно п.2 ст.55 ГПК РФ не могут быть положены в основу решения суда. Также суд принял этот документ в материалы дела в качестве доказательства, хотя это внутренний документ банка и не является документом двухстороннего характера, отражающим волеизъявление всех участвующих в нем сторон.
    С решением суда не согласен, считаю, что суд дал неверную оценку представленным доказательствам, неправильно применил закон, не выяснил обстоятельства, имеющие существенное значение для рассмотрения дела.
    На основании вышеизложенного, основываясь на ст.320-322, 328 Гражданского процессуального кодекса РФ, и руководствуясь действующим законодательством,

Прошу:

Отменить решение о взыскании задолженности по кредитному договору, судебных расходов и принять новое решение, которым отказать в удовлетворении исковых требований ООО *** в полном объеме.
Мотивированная жалоба будет подана после вынесения решения в окончательной форме.

Приложение:
Копия подтверждающая оплату государственной пошлины.
Копия описи письма подтверждающего запрос на составление мотивированного решения.
Копия апелляционной жалобы для Истца.

*** 2023 года ______ / Моя фамилия и инициалы/

Районный суд получил мою жалобу и переслал её мировому судье. Я только потом узнал, что можно было сразу мировому судье отправлять, а не в районный суд. Они подготовили, провели необходимые процедуры и сами обратно вернули в районный суд. После некоторого времени, мониторя уже сайт районного суда я увидел следующее:

Если честно, то я не ожидал такого поворота, что значит снято? Я думал, что увижу какое-то решение, а как это так, просто взяли и сняли. На самом деле, тут может быть несколько причин, и погуглив я выяснил, что так делается очень часто, особенно если у судьи нет времени или желания этим заниматься. По этому она находит причину и возвращает дело обратно мировому судье.

Я звоню мировому судье, мол какова причина возврата моей апелляционной жалобы. Секретарь ответила, что просто забыли направить копию истцу, по этому суд вернул на доработку. Доработка длилась 5 месяцев, после чего мировой суд вновь отправил и через ещё месяц было вынесено определение по апелляционной жалобе:

Иными словами, суд апелляционной инстанции тоже счел, что обычные бумажки формата А4 напечатанные на офисном принтере можно приобщать в качестве доказательства и выносить по ним решение. Суд счел, что банк вправе выпускать электронные подписи без какой либо лицензии от ФСБ, раздавать их клиентам, а потом призывать к ответу за действия. Суд счел, что нет разницы, зарплатный у тебя договор с банком или кредитный, всё одно и тоже, и если какая ошибка банка – будешь отвечать!..Не, как-то это нелогично, а зачем тогда вообще был бы нужен суд? Я не верю, что так оно было на самом деле, и тут могу только догадываться. Полагаю, судья увидела апелляцию по делу о взыскании, и не заглядывая даже во внутрь текста, так как это скукота нечеловеческая, да ещё и поданная не юристом, а ответчиком лично, да мало ли какую он там ересь, терпила, написал, да и по-быстрому – КАЗНИТЬ короче, чем ПОМИЛОВАТЬ. И сколько таких дел проходит за год…- Да, дел таких, видимо, много. Но, эти нюансы уже не моё дело, а дело судейской коллегии. Кстати, должен признать, какое-никакое, а решение есть. Это теперь даёт возможность в 3-х месячный срок подать кассационную жалобу, но для этого я должен получить определение на апелляционную жалобу. А сама кассационная жалоба ещё только предстоит, и это уже совсем другая история.

«Кроссворд» (Жанр: Комедия)

Включив телевизор для вида, он устроился перед журнальным столиком, на котором стояла тарелка с едой, и почесал шею под подбородком.

– Ты забыла принести мне кроксворд,- демонстративно искривив названия слова «кроссворд», сказал Валентин.
– Ну да, забыла. Попробуй всё упомни,- ответила она, немного подняв взгляд кверху.
– За это тебе штраф пятьсот рублей! – пошутил он.
– Ага, сейчас только отксерю.
– Если бы ты умела ксерить, то сейчас работала бы в офисе с солидной зарплатой, а то всё на меня взвалила.
– Ты думаешь в офисе работники только ксерят? Там наверно ещё чего-нибудь делать нужно.
– Нет, в офисе даже должность такая есть, ксерун называется, а если это девушка, то ксерунья или ксерульщица, в зависимости от квалификации.
– Да нет, я была у нас в завод управлении в офисе, нет там должностей таких.
– Откуда тебе известно, что нет; ты же туда только за зарплатой ходишь. Надо было тебе не заклёпщицей идти, хоть бы на новые обои денег насобирали.
– Валентин, ну хватит, на вот лучше принесла кроссворд тебе, поодгадывай, свежий; и ещё вот котлеты поешь.
– Что свежий, кроссворд или котлеты, не понял.
– И то и другое.- Немного улыбнувшись произнесла она.
– Ну ладно, давай посмотрим что тут.- Он сдвинул старые очки на нос и стал вчитываться.
– Гуля, вот смотри, пять по горизонтали – наука по разведению собак, слово из девяти букв, что это?
– Ой, слушай, на языке вертится. А, может это слово «собачатье», собак же разводят?
– Ну-ка, сейчас проверим. Ага, подходит!
– Так, а три по вертикали – мужчина, намеревающийся жениться, пять букв, кто это?
– Ну, Валентин, откуда мне знать. Вон, у нашей соседки сын женился, так она ему с первого дня рога ставит. Так кто он? Рогач, получается.
– Так-так, погоди. А что, подходит. Вот, Гуля, тебе нужно было кроссворд отгадывать, а не мне!
– Давай ещё, семь по горизонтали – человек, управляющий транспортным средством, восемь букв, а?
– Ну, сложные вопросы пошли. Леха – брат Светы, ну кудрявый такой, на троллейбусе работает, так кто он?
– Ну, Гуля, кто он, его все кудрявым так и зовут; дай-ка проверю. Слушай, а подходит, да, «кудрявый»!
– Ну вот и пиши! Сейчас мы с тобой всё отгадаем!
Гуля улыбаясь заинтересованно села рядом с Валентином и стала тоже читать вопросы. Её охватил задор, и она стала уже предвкушать вид полностью отгаданного кроссворда.
– А вот, смотри, что такое слово двенадцать по вертикали, пять букв? Сейчас точно отгадаем!
– Это, это, сейчас посмотрим…Так,- Квантовая частица, из которой состоит конденсат Бозе-Эйнштейна.
Валентин медленно стал надувать щеки, похлопал авторучкой себе по подбородку несколько раз, перевёл взгляд на Гулю, которая уже наверно в шестой раз пыталась прочитать правильно вопрос и понять в нём хотя бы одно слово, ещё раз взглянул на кроссворд. Он ждал когда нужная мысль его посетит, но она никак не посещала его. Он медленно нашпиговал на вилку часть котлеты, машинально поместил её в рот, не отрывая взгляда от вопроса, поднёс было ручку к газете чтобы что-то написать. В этот момент Гуля уже в надежде, что он знает ответ немного улыбнулась и даже открыла рот, но Валентин отвел руку и снова задумался, одновременно с этим Гуля рот обратно закрыла и опять стала серьезно вчитываться в вопрос. Её длинный и слегка задранный нос ходил вслед за её взглядом то вправо, то влево, и лишь иногда маячил вверх-вниз, когда она вчитывалась в очередное слово.
– Хрень какая-то! – Наконец прервав тишину вспылил Валентин.- Гуля, вот какого чёрта ты мне такие кроссворды на ночь глядя даёшь, да ещё за едой, а?!
– Ну, Валя, ну что ты так разошёлся? Да выбрось ты этот кроссворд! Давай другое что-нибудь посмотрим! А гляди, «хрень» подходит для ответа!
– Да мне просто интересно, какого лешего ты мне решила перед сном настроение испортить?! Что подходит? Ты видишь вторая буква «у» от слова «кудрявый», а в хрени вторая буква «р»!
– Ну, Валя, ну ошиблись они значит. Знаешь бездари какие кроссворды составляют? ЕГЭ сдаст и думает уже профессор, и идёт в журнал работать. А ошибок поналепит, что потом у людей настроение пропадает. Выбрось ты этот кроссворд в мусорку! Знала бы, не дала тебе его отгадывать. Давай вон лучше телик посмотрим, там говорят Илон Маск свою бандуру опять поставил, запускать будет, людей там тьма собралось…
– Да на кой ляд мне смотреть на бандуру Маска, лучше котлет ещё принеси, надо срочно едой стресс снять…

El Vinto, 2023

Java Android. Урок 20 (“Делаем игру Salvador”)

На этом уроке цель будет сделать как-то так:

Т.е. как можно понять, будем разрабатывать отображение сканера (масштабную карту местности) игрового пространства. Соответственно и все условные объекты на ней тоже должны отображаться.

Отображение ландшафта на сканере я делаю так:

Метод получения координат похож на тот, что используется для отображения на игровом поле, только в конце формулы я масштабирую к размеру topper и добавляю константу его центра.

Задание по уроку:

1. Аналогично отображению рельефа, сделайте отображение на сканере игрока и всех инопланетян

2. Сделайте так, чтобы заставка циклически повторялась, а по клику завершалась и начиналась игра

3. Добавьте на отображением суммы бонусов отображение времени раунда.

Если всё получилось как нужно – молодцы! Если нет, давайте разберёмся что не так.

Урок 19 Урок 21

Java Android. Урок 19 (“Делаем игру Salvador”)

Продолжаем практическое изучение Java Android на примере игры Salvador. На этом уроке и в его домашнем задании будем делать следующее, т.е. заставку к игре:

Сделать её можно используя тысячи методов, но нужно было что-то выбрать и я решил попробовать тот, который ещё ни разу не проверял.

Для начала добавим переменную roundTime класса GameEngine, которая будет фиксировать прошедшее в раунде время. Также добавим переменную mainGameLogic, которая будет определять логику игры, т.к. что сейчас у нас на экране, игровой процесс, заставка или ещё что-то иное:

Объект GameObject теперь хранит много самостоятельной информации общей для всех его дочерних классов, а некоторые функции и переменные из дочерних вынесены в этот:

Класс Alien теперь умеет работать с несколькими подтипами:

Сам метод поведения Alien тоже изменился:

Начальная инициализация объектов теперь начинается с группировки корабля игрока, а логика игры – заставка:

Заметьте, теперь информация о будущем объекте после группировки передаётся прямо в конструктор GroupingParts.

Основной код метода 1/25 секунды тоже поменялся, и чтобы уже определиться, переименовал его в mainStrob():

тут дальше следует череда разного рода действий заставки возникающих как только время раунда достигает нового значения 2000 мс, 3000 мс и т.д., тут то группируется корабль игрока, потом группируется инопланетянин, через 4 сек это стреляет в того ракетой, через 4.4 сек наш стреляет бластером в этого и т.д.

Ключевой код работы с объектами теперь упростился, т.к. часть функционала взял на себя GameObject и его дочерние классы:

Теперь посмотрим зачем в сначала вводил переменную позиция по X корабля игрока на экране. До этого момента это была лишняя переменная, т.к. корабль всегда находился по горизонтали в центре, так спрашивается, а зачем тогда она нужна. Нужна она вот для чего:

Правильно, чтобы расширить возможности игрового пространства. Когда корабль игрока летит по рельефу вправо, то его корабль отодвигается влево, чтобы улучшить реакцию на инопланетян появляющихся по ходу движения. При этом и игровая ситуация для манёвров более предсказуема. Как можно заметить, это не влияет на реальную скорость, т.к. определяет только визуальное представление, но не координату в игровом пространстве. Однако, учитывая как у нас организован функционал, сделать это оказалось крайне просто:

А в стробе просто делаю так:

“300” – это та самая граница в которую корабль упирается слева или справа.

onScreenVX – определяет с какой скоростью он будет перемещаться по экрану в эту граничную область.

Задание по уроку:

1. Сделайте другие подюниты класса Alien

2. Сделайте так, чтобы заставка циклически повторялась, а по клику завершалась и начиналась игра

3. Добавьте на отображением суммы бонусов отображение времени раунда.

Если всё получилось как нужно – молодцы! Если нет, давайте разберёмся что не так.

Урок 18 Урок 20

Шаверма по-домашнему. Как я делаю

Это блюдо, которое наверно сможет сделать каждый, если конечно у него нет желания выпендриваться, вертеть шаверму по столу, шоркать ножиками, а затем сдирая мясо с гриля и улыбаться сквозь густую чёрную бороду)).

Для шавухи потребуется следующее:

  • 500 гр. куриного филе
  • 200 гр. сметаны
  • 100 мл уксуса
  • 2 луковицы среднего размера
  • 50 мл растительного масла (обычное для жарки)
  • 4 дольки чеснока
  • Половина лимона
  • Красный молотый перец чили (по вкусу, я кладу много)
  • Обычная приправа для курицы или индейки
  • Обычный тонкий лаваш (я покупаю длинный и режу на 4 части)
  • Один большой помидор
  • Половина свежего огурца
  • Соль

Маринад:

Делается по принципу маринада для шашлыка. Вечером, купив после работы филе куры, режем её на мелкие части и кладём в посудину. Кура должна быть прохладной, в прохладной воде и её следует промыть. Заливаем всё уксусом, режем мелко лук (1 луковицу) и весь чеснок. Засыпаем перцем, приправой для курицы и солью по вкусу. Лично я кладу столовую ложку перца и столько же приправы, столько же соли, но на меня лучше не равняться)). Перемешиваем, а после этого тщательно моем руки, тщательно их ополаскиваем, засовываем в посудину, и начинаем крепко сжимать и переворачивать мясо как-будто мы месим тесто. После 5 минут жамканья уминаем всё на дно, режем 4 круглеша лимона (с кожурой) и кладём сверху. До утра помещаем в холодильник.

Жарить

Казалось бы, что тут такого пожОрить куру, ан нет. Делю это так. На обычной газовой конфорке, сколько ватт она не знаю, врубаю на полную катушку. Беру большую сковороду, подливаю 3 столовых ложки растительно масла, кладу туда всё что мариновалось, прямо всё содержимое через край и уже на сковороде начинаю перемешивать. По прошествии минут пяти, маринад начнёт шипеть и благоухать уксусом. Вытяжка не помешает тому, у кого проблемы с таким запахом или какие-то проблемы с уксусом в целом. Я обхожусь так, т.к. так легче контролировать процесс жарки. Через минут 15-20, когда кура будет уже готова, подливки должно оставаться ещё на сковороде много, теперь она там будет мешать, но(!) пригодится далее для другого:

ВНИМАНИЕ! Кура должна уже быть прожаренной, чтобы сливаемая подливка была уже полностью пригодна для употребления. По этому просто сливаем её в какую-нибудь посудину и продолжаем жарить дальше, подлив ещё немного растительного масла. Через некоторое время кура начнёт темнеть, покрываться кое где хрустящей корочкой и пойдёт запах, похожий как в шаверменной. Когда станет такой:

это значит, что она уже полностью готова. Ошмётки лимона можно вынуть, мелко порезать и замешать обратно. Жарка закончена.

Салат-соус

Соус и салат я делаю одним блюдом, примерно так, как это делается обычный помидорный салат со сметаной. Мелко режем помидор, огурец и половину луковицы. Кладём в бадейку, заливаем сметаной, заливаем примерно 30 гр. растительного масла, добавляем приправу для курицы и красный перец, солим. Перемешиваем всё до тех пор, пока на поверхности не будет плавать масло. Салат получается вот таким:

Заворачивание

Тут либо вы спец по заворачиванию шавермы, либо просто делаете как я. Отрезать от лаваша кусок примерно 30 х 20 см, разложить на столе:

Вылить на неё 2 столовые ложки подливы (которую сливали; именно по этому она уже должна быть пригодна к употреблению). Эта подлива не даст лавашу пересохнуть и сделает его ещё более мягким. Теперь у вас есть от силы минуты 3 чтобы лаваш не размяк. Быстро кладете по середине мясо:

После этого быстро салат:

После этого просто третью лаваша справа и третью лаваша слева внахлёст закрываем внутренности. С торцов лучше не закрывать, иначе если сильно не изловчиться, то всё распадётся и повываливается. По этому, чтобы не рисковать, я сразу это отправляю на плоский электрический гриль (двухстворчатый – нагрев сверху и снизу):

Но если быть честным, то сразу две делаю и отправляю на гриль)) :

Через пару минут приоткрыть гриль и снова той же подливой немного смазать (на фото я слегка переборщил):

Снова закрыть гриль, там всё сразу зашипит, а лишняя подлива начнёт стекать в сборник. Значит делаем всё правильно. Закрываем, периодически наблюдая степень прожарки и готовности. Тут кому как нравится. После этого получаем простую домашнюю шавуху:

Java Android. Урок 18 (“Делаем игру Salvador”)

Этот урок будет посвящен небольшому приведению игры в чувства, создание динамизма и зрелищности. Когда что-то много меняется, это для аркады то, что как раз и нужно. Всё внимание должно быть приковано к игре, особенно если она это внимание к себе требует. Будем сейчас делать что-то вроде этого:

Правда, не в тесте нужно будет убегать от инопланетян, избегать с ними столкновения, а не наоборот догонять. Это они будут стараться атаковать всеми своими средствами. Сейчас у них это и таран и ракеты (белые летящие точки). На нашем корабле что-то моргает, на их тоже самое. Это всё создаёт необходимый антураж аркады.

Итак, теперь для начала нужно немного изменить код отвечающий за полёт корабля игрока. Вообще, это частое явление, когда вы пишите некоторые части исключительно чтобы пройти следующую стадию разработки, а потом оптимизируете и меняете код уже под новые реалии и функционал.

Сделаем для всего, что связано с игроком отдельный класс и дадим ему имя Gamer:

Как видите, я перенес сюда и хранение координат и метод отрисовки, теперь это уже не просто круг, а что-то похожее на летательный аппарат, который умеет поворачиваться в ту сторону, куда направлен вектор тяги. Также добавлены для антуража некоторые моргалки на его поверхности, придающие живость, ну, живенько чтобы было)).

Для инопланетян отрисовка идёт тоже несколько по другому:

Код, отвечающий за движение инопланетян, и вызывающий у них броуновское движение тоже изменил, ведь в домашнем задании вам нужно было сделать чтобы на некотором расстоянии от игрока, их броуновское движение прекращалось и она начинали идти на таран. Посмотрим, как это сделал я:

Метод getR() (расчет расстояния от инопланетянина до игрока) работает на первый взгляд немного странно. Можно заметить, что dx, dy и r – это не локальные переменные метода, а всего класса, при этом метод ещё этот r и возвращает в качестве результата. Всё дело в том, что я использую его и тут и ещё далее сразу для двух функционалов.

Всплывающий текст с бонусами – это такой же игровой объект, как и инопланетяне и группирующиеся объекты:

Ракеты – тоже игровой объект:

Вызываемый метод normalizeCords() – это оптимизация расчета координат реализованная в родительском классе:

Paint для отображения бонусных очков шрифтом из прошлых циклов уроков, мы делаем так – за это отвечает Typeface:

Теперь для работы с этими новыми объектами нужно всего лишь добавить логику взаимодействия с игровым процессом, а делается это как раз в методе, который запускается 25 раз в секунду (не забываем, что ракеты и текст с бонусами нужно ставить на ожидание и последующее удаление, иначе они так и будут создаваться и загромождать интерфейс и отбирать ресурсы):

Задание по уроку:

1. Сделайте так, чтобы у корабля игрока отображался хвост

2. Добавьте кнопку стрельбы, чтобы игрок мог стрелять, причем только горизонтально и в том направлении, куда направлен корабль.

3. Сделайте функционал уничтожения инопланетян, если луч бластера примерно входит в область инопланетянина.

По результатам домашнего задания должно получиться как-то так:

Если всё получилось как нужно – молодцы! Если нет, давайте разберёмся что не так.

Урок 17 Урок 19

El Vinto, 2023 (Copyright)

Java Android. Урок 17 (“Делаем игру Salvador”)

Итак, отключаем тестовый процесс самоликвидации инопланетян, просто закомментировав эту строку (мало ли ещё пригодится, а оно так):

Перенесём парочку полезных методов генерации диапазона чисел из GroupingParts в GameObject, т.к. теперь эти метода мы будем использовать ещё как минимум и в классе Alien:

Добавим в Alien аналогичные игроку свойства для перемещения:

Реализуем ещё не задействованный от родительского класса метод run(). Если помните, он вызывается для каждого объекта 25 раз в секунду (до этого тоже вызывался для Alien, просто был “пустым” из родительского класса, заглушкой):

Теперь этот метод в Alien устраивает хаос, причём в прямом смысле этого слова. Мы получаем броуновское движение:

Добавим в класс GameObject заглушку-метод contact() который будет вызываться аналогично run() (и вместе с ней 25 раз в секунду):

Для класса Alien выполним реализацию:

Здесь всё просто – мы находим разницу между координатами игрока и объекта, и если она меньше некой допустимой, значит будем запускать цикл взрыва объекта (переменная waitToExplosion как раз и создана в Alien для этого).

По формуле тоже просто – вспоминаем школу, геометрию, теорему Пифагора:

Заметьте как я нахожу разницу координат по X – у меня стоит знак плюс. Это не ошибка, дело в том, что изначально я спозиционировал игровой процесс в относительном движении (только по X) игрока и всего остального, таким образом его координата всегда будет противоположной всем остальным объектам игры. Также необходимо пройти условия, когда объекты оказываются на границе рельефа местности, когда, например, у одного относительная координата -5000 и у другого 5000, но тут нужно просто из суммы вычесть константу.

Чтобы выполнить взрыв мне будет достаточно просто добавить это в уже существующее условие:

Результат будет очевиден, но придётся хорошенько погоняться за объектами, и это оказалось задачей не из простых))):

Приступим теперь к тому, что нужно было вам подготовить в домашнем задании – отображение текста. Начнем с того текста, который отображает бонусы при уничтожении инопланетян. Для начала добавим новый класс:

Теперь выполним то, что должно создавать текст и убирать его по прошествии некоторого времени (в данном случае 2 сек):

В результате получится как-то так:

Задание по уроку:

1. Сделайте подсчет игровых очков в верхнем левом углу как сумму показываемых бонусов при уничтожении инопланетян.

2. Импортируйте шрифт из игры “Быки и коровы”, которую мы делали и используйте его для п.1

3. Сделайте чтобы корабли инопланетян если находятся ближе определенного расстояния к игроку переставали хаотически двигаться и начинали сами на него нападать вызывая столкновение:

Если всё получилось как нужно – молодцы! Если нет, давайте разберёмся что не так.

Урок 16 Урок 18

El Vinto, 2023 (Copyright)

Java Android. Урок 16 (“Делаем игру Salvador”)

Итак, к этому уроку, опираясь на ваши домашние задания, вы уже должны быть готовы чтобы сделать это:

Что тут есть:

  • Рельеф, который при движении влево или вправо никогда не кончается; он подобно Земному шару, если идти по экватору – придёшь в ту же точку, откуда вышел.
  • Сначала генерирует точка сбора каждого инопланетянина. После его группировки он появляется, а затем через несколько секунд он самовзрывается.
  • Движение всех объектов на экране согласовано в относительном движении игрока
  • Движение игрока выполнено с учетом физики движение (есть скорость движения, есть учёт ускорения/торможения, есть учет сопротивления воздуха). Это позволяет создавать эффект инерционного движения, когда при смене направления движения корабль игрока не мгновенно начинает лететь в обратную сторону, а сначала замедляется, и лишь потом разгоняется в обратную сторону.

То, что мы в прошлых уроках использовали для тестирования геймплея (игрового процесса), было удобно и эффективно, однако, с новыми реалиями и физикой некоторые моменты нужно будет упростить, изменить и добавить новые. Начнём.

Инициализацию параметров игрока делаем так:

К координатам игрока я добавил переменные, которые будут хранить его скорость и ускорение, а также их предельные значения (у игрока не должна бесконечно возрастать скорость, иначе это будет противоречить канонам физики). Заметьте, предел ускорения по X и Y различны, оно и понятно – двигатели нашего корабля развивают разную мощность по вертикали и горизонтали))). Тут ещё не мешало бы вспомнить школьный курс физики и математики, что

, где:

L – перемещение объекта

L0 – начальная координата

V0 – начальная скорость

a – ускорение

t – время

Однако, увы, эта формула будет для нас не пригодна, т.к. у нас нет показателя времени – t. Связано это как раз с тем, что обновление экрана мы выполняем квантами времени, как помните,- 25 раз в секунду. Т.е. с точки зрения математики мы обладаем не параметром t, а параметром ∂t (дифференциал t). Исходя из этого и перемещение объекта будем определять через совершенное за единицу времени перемещение. Так и поступим. Изменим для начала метод onButtonDown() как и было это в домашнем задании – он теперь будет определять не какие-то тестовые координаты, а именно само ускорение. С точки зрения моделирования это так и есть – как в автомобиле, чем сильнее нажимаем на педаль газа, тем с большим ускорением разгоняемся. Добавим ещё один параметр с радиусом кнопки, чтобы более универсально можно было настраивать параметры размеров интерфейса.

Разумеется при вызовах тоже нужно это учесть:

Сам расчет расстояния перемещения выведен туда, где он более всего логичен – в метод вызываемый 25 раз в секунду:

Сначала мы находим текущую скорость от ускорения. Затем изменение расстояние от этой скорости, тут вроде должно быть понятно. Подчеркиваю, метод вызывается 25 раз в секунду, по этому с точки зрения динамического моделируемого процесса, мы имеем расчет параметров за единицу времени ∂t. Просто она в математике стремится к нулю, а в жизни для игрока уже всё что чаще 24 кадров в секунду уже незаметно и кажется непрерывным движением, и этого оказывается достаточным. С таким ∂t идеально точный математический расчет траектории мы конечно же не получим, но визуально мы разницы не заметим. Далее в коде идут условия, по которым я ограничиваю и текущую скорость и координаты чтобы корабль игрока не вышел за пределы игрового поля или не развил скорость, при которой уже станет не понятным что на экране вообще происходит. После этого я умножаю на числовую константу ускорение и скорость. Это таким образом я создал эффект сопротивления воздуха, т.е. если игрок отпустит кнопку управления, то через какое-то время его корабль сам плавно остановится. Если бы это был космос, где нет сопротивления воздуха, то корабль летел бы сколь угодно долго пока не столкнулся бы с каким-либо объектом, но в нашей игровой реальности, пардон, виртуальности, условия другие.

Но с этим методом ещё не всё, давайте сразу же добавим в него и ещё кое что:

Основное предназначение этой части кода – управление трансформацией инопланетных объектов. Для каждого объекта выполняется сначала метод run(), далее посмотрим что он делает. Затем определяется тип объекта, что это ещё только группирующийся/взрывающийся объект или уже готовый инопланетянин. Тут же определяется, что если группировка уже завершена, то нужно на этом месте создать инопланетянина, а сам группирующийся объект удалить из списка игровых объектов. Если инопланетянин “прожил” некоторый лимит времени, то должен удалиться из списка игровых объектов, но в этот же момент на этом же месте нужно создать взрыв ( метод explosion() ).

Заметьте, я не сразу прямо в цикле объектов gameObjects удаляю “старый” объект из списка и добавляю туда новый, а изначально создал два временных списка, один для удаления, другой для добавления, и делаю работу над ними уже после выполнения основного цикла. Делается это для того, чтобы не получилось исключительной ситуации. Если циклом перебираем список, и сам цикл зависим от этого массива, то менять состав этого списка внутри этого цикла строго нельзя (иначе результат такого функционала может стать не предсказуемым) !

Завершает метод всё тот же invalidate(), который запускает цепочку системных вызовов по обновлению экрана и запуску метода onDraw().

Претерпел изменения класс GameObject (изменил немного формат некоторых методов). Конструктор теперь не содержит начальных координат, т.к. они генерятся в burn() дочернего класса GroupingParts и больше генерить их пока негде; в конструктор добавлен тип:

Дочерние классы:

Заметьте выполнения класса explosion() также как и burn() только координаты при этом не генерятся и лимиты наоборот, а так всё тоже самое.

Инициализация местности немного изменена (понижена), а инициализация объектов теперь использует тип:

Заметьте, инициализируется группирующийся объект, который (как я вначале написал) после схлопывания удаляется и создаётся одновременно новый уже типа Alien. Сгенерировал сразу 100 объектов чтобы оценить будет ли хромать производительность.

Батон, пардон, кнопку увеличил в размерах и перенёс выше и правее, т.к. испытав на своём телефоне с разрешением 1920 х 1080 понял, что кнопка для управления для пальца слишком мала. Также изменил и настройки виртуального мобильника на аналогичные:

В изображение кнопки добавил функционал делающие её слегка прозрачной, это позволяет и видеть что находится за ней и при этом видеть саму кнопку (если делали домашнее задание в одном из уроков то поняли, что я о setAlpha():

Рельеф Земли теперь изображается с учетом того, что она якобы круглая, т.е. летим в одну сторону, вылетаем из другой (циклически повторяется). Для этого пришлось сделать её отображение с учетом этого:

Тут было два очевидных варианта решения, либо оставить всё как есть и адаптировать под существующий функционал, либо математически представить её действительно круглой (ввести полярную систему координат и измерять не координатами по x и y, а углами alpha и beta. Но расчеты в полярной системе потребовали бы вычисления косинусов и синусов в вещественных числах, а это существенно забрало бы вычислительных ресурсов. В связи с этим, если вы ещё раз посмотрите на код объектов, то увидите, что вычисление координат при отрисовке претерпело явные изменения. Для рельефа же потребовалось сделать не только цикл его отрисовки, но и два цикла дорисовки на случай, если мы подлетаем к краю рельефа (слева или справа) – визуально он не должен обрываться, а должен дорисовываться его началом до границ экрана, создавая иллюзию циклической поверхности, вроде экватора планеты. Но делать три цикла подряд тут как-то некрасиво, по этому я сделал цикл в цикле.

По факту получилось даже лучше, чем планировалось:

Задание по уроку:

1. Снова оптимизируйте отрисовку рельефа так, чтобы его куски за пределами экрана не отрисовывались. Аналогично оптимизируйте отрисовку объектов и частей группировки / взрыва.

2. Самостоятельно разберите вопрос рисования текста на холсте.

3. Сделайте чтобы корабли инопланетян случайно двигались тоже с учетом физики

Если всё получилось как нужно – молодцы! Если нет, давайте разберёмся что не так.

Урок 15 Урок 17

El Vinto, 2023 (Copyright)