This commit is contained in:
Arsen Mirzaev Tatyano-Muradovich 2025-07-06 14:09:01 +07:00
parent edf98e5ecb
commit 77f62d53af
219 changed files with 3029 additions and 173 deletions

8
.gitmodules vendored Normal file
View File

@ -0,0 +1,8 @@
[submodule "pechatalka.mjs"]
path = pechatalka.mjs
url = https://git.svoboda.works/mirzaev/pechatalka.mjs
branch = stable
[submodule "damper.mjs"]
path = damper.mjs
url = https://git.svoboda.works/mirzaev/damper.mjs
branch = stable

View File

@ -1,7 +1,7 @@
{ {
"name": "mirzaev/pechatalka", "name": "svoboda/pechatalka",
"description": "Chat-robot constructor for the Svoboda typography", "description": "Chat-robot constructor for the Svoboda typography",
"homepage": "https://git.svoboda.works/mirzaev/pechatalka", "homepage": "https://git.svoboda.works/svoboda/pechatalka",
"type": "site", "type": "site",
"keywords": [ "keywords": [
"minimal", "minimal",
@ -15,32 +15,42 @@
"name": "Arsen Mirzaev Tatyano-Muradovich", "name": "Arsen Mirzaev Tatyano-Muradovich",
"email": "arsen@mirzaev.sexy", "email": "arsen@mirzaev.sexy",
"homepage": "https://mirzaev.sexy", "homepage": "https://mirzaev.sexy",
"role": "Programmer" "role": "Creator"
} }
], ],
"support": { "support": {
"wiki": "https://git.svoboda.works/mirzaev/pechatalka/wiki", "wiki": "https://git.svoboda.works/svoboda/pechatalka/wiki",
"issues": "https://git.svoboda.works/mirzaev/pechatalka/issues" "issues": "https://git.svoboda.works/svoboda/pechatalka/issues"
}, },
"require": { "require": {
"php": "^8.4", "php": "^8.4",
"mirzaev/minimal": "^3.6", "mirzaev/minimal": "^3",
"mirzaev/baza": "^3.3", "mirzaev/baza": "^3.3",
"twig/twig": "^3.2", "twig/twig": "^3.2",
"twig/extra-bundle": "^3.7", "twig/extra-bundle": "^3.7",
"twig/intl-extra": "^3.10" "twig/intl-extra": "^3.10",
"svoboda/time": "^1.0",
"badfarm/zanzara": "^0.9.1",
"nyholm/psr7": "^1.8",
"react/filesystem": "^0.1.2"
}, },
"autoload": { "autoload": {
"psr-4": { "psr-4": {
"mirzaev\\pechatalka\\": "mirzaev/pechatalka/system" "svoboda\\pechatalka\\": "svoboda/pechatalka/system"
} }
}, },
"autoload-dev": { "autoload-dev": {
"psr-4": { "psr-4": {
"mirzaev\\pechatalka\\tests\\": "mirzaev/pechatalka/tests" "svoboda\\pechatalka\\tests\\": "svoboda/pechatalka/tests"
} }
}, },
"scripts": { "scripts": {
"pre-update-cmd": "./install.sh" "pre-update-cmd": "./install.sh"
},
"config": {
"allow-plugins": {
"php-http/discovery": true,
"wyrihaximus/composer-update-bin-autoload-path": true
}
} }
} }

1
damper.mjs Submodule

@ -0,0 +1 @@
Subproject commit 68589e968cbc043f35c2948a9c90293b6f5f9cb9

View File

@ -0,0 +1,53 @@
server {
listen 80;
listen [::]:80;
server_name pechatalka.svoboda.works;
# 301 302
return 301 https://$server_name$request_uri;
}
server {
listen 443 ssl;
listen 443 quic;
listen [::]:443 ssl;
listen [::]:443 quic;
server_name pechatalka.svoboda.works;
http2 on;
http3 on;
quic_gso on;
quic_retry on;
add_header Alt-Svc 'h3=":$server_port"; ma=86400';
add_header x-quic 'h3';
root /var/www/pechatalka.svoboda.works/svoboda/pechatalka/system/public;
index index.php;
keepalive_timeout 60;
include snippets/ssl-params.conf;
include snippets/ssl-svoboda.conf;
include snippets/php8_4.conf;
location / {
try_files $uri $uri/ /index.php?$query_string;
}
location ~* \.(?:jpg|jpeg|gif|png|ico|cur|gz|svg|svgz|mp4|mp3|ogg|ogv|webm|htc|woff2|woff)$ {
expires 1M;
access_log off;
add_header Cache-Control "max-age=2629746, public";
}
location ~* \.(?:css|js|mjs|min)$ {
expires 1y;
access_log off;
add_header Cache-Control "max-age=31556952, public";
}
}

View File

@ -0,0 +1,17 @@
[Unit]
Description=Telegram chat-robot: @domain_of_your_robot_here
Wants=network.target
After=syslog.target network-online.target
[Service]
ExecStart=sudo -u www-data /usr/bin/php /var/www/pechatalka/svoboda/pechatalka/system/public/robot.php
PIDFile=/var/run/php/pechatalka.pid
RemainAfterExit=no
RuntimeMaxSec=3600s
Restart=always
RestartSec=5s
[Install]
WantedBy=multi-user.target

View File

@ -1 +0,0 @@
pechatalka.mjs/pechatalka.mjs.min

View File

@ -1,4 +0,0 @@
@charset "UTF-8";
aside {
}

View File

@ -1,37 +0,0 @@
@charset "UTF-8";
@media (prefers-color-scheme: dark) {
:root {
--text-color: initial;
--text-color-hover: initial;
--text-color-active: initial;
--text-notice-color: initial;
--text-warning-color: initial;
--text-selected-color: initial;
--text-selected-background-color: initial;
--link-color: initial;
--link-color-hover: initial;
--link-color-active: initial;
color: var(--text-color);
}
}
@media (prefers-color-scheme: light) {
:root {
--text-color: initial;
--text-color-hover: initial;
--text-color-active: initial;
--text-notice-color: initial;
--text-warning-color: initial;
--text-selected-color: initial;
--text-selected-background-color: initial;
--link-color: initial;
--link-color-hover: initial;
--link-color-active: initial;
color: var(--text-color);
}
}

View File

@ -1,9 +0,0 @@
@import url('/css/fonts/fira.css');
@import url('/css/fonts/hack.css');
@import url('/css/fonts/dejavu.css');
@font-face {
font-family: 'Commissioner';
src: url('/fonts/commissioner.ttf');
font-weight: 400;
}

View File

@ -1,4 +0,0 @@
@charset "UTF-8";
footer {
}

View File

@ -1,4 +0,0 @@
@charset "UTF-8";
header {
}

View File

@ -1,13 +0,0 @@
@charset "UTF-8";
body {
margin: unset;
}
main {
display: flex;
flex-direction: column;
align-items: center;
gap: var(--gap);
transition: 0s;
}

View File

@ -1,34 +0,0 @@
@charset "UTF-8";
:root {
--gap: min(12px, 1rem);
/* font-family: , system-ui, sans-serif; */
font-family: "dejavu";
text-decoration: none;
outline: none;
border: none;
transition: 0.1s ease-out;
}
/* Selection */
::selection {
color: var(--text-selected-color);
background: var(--text-selected-background-color);
}
::-moz-selection {
color: var(--text-selected-color);
background: var(--text-selected-background-color);
}
.unselectable {
-webkit-touch-callout: none;
-webkit-user-select: none;
-khtml-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
}

View File

@ -1,8 +0,0 @@
<?php
return [
'endpoint' => 'unix:///var/run/arangodb3/arango.sock',
'database' => 'pechatalka',
'name' => 'pechatalka',
'password' => ''
];

View File

@ -1,5 +0,0 @@
{% block js %}
{% for element in js %}
<script {% if element.src %}src="{{ element.src }}"{% endif %} {% if element.type %}type="{{ element.type }}"{% endif %}>{{ element.innerText }}</script>
{% endfor %}
{% endblock %}

@ -1 +1 @@
Subproject commit a32ea39cf65b73016db5e8baad93bff77811075f Subproject commit b1400479c4f06b49d0a6d6d5c4496cd9b4430752

View File

@ -0,0 +1,70 @@
<?php
declare(strict_types=1);
namespace svoboda\pechatalka\controllers;
// Files of the project
use svoboda\pechatalka\controllers\core;
// Framework for PHP
use mirzaev\minimal\http\enumerations\content,
mirzaev\minimal\http\enumerations\status;
/**
* Index
*
* @package svoboda\pechatalka\constructor
*
* @param array $errors Registry of errors
*
* @method null index() Main page
*
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
*/
final class constructor extends core
{
/**
* Errors
*
* @var array $errors Registry of errors
*/
protected array $errors = [
'system' => []
];
/**
* Pin
*
* @return null
*/
public function pin(): null
{
if (str_contains($this->request->headers['accept'], content::any->value)) {
// Request for any response
// Render page
$page = $this->view->render('/constructor/pin/page.html');
// Sending response
$this->response
->start()
->clean()
->sse()
->write($page)
->validate($this->request)
?->body()
->end();
// Deinitializing rendered page
unset($page);
// Exit (success)
return null;
}
// Exit (fail)
return null;
}
}

View File

@ -2,13 +2,12 @@
declare(strict_types=1); declare(strict_types=1);
namespace mirzaev\pechatalka\controllers; namespace svoboda\pechatalka\controllers;
// Files of the project // Files of the project
use mirzaev\pechatalka\views\templater, use svoboda\pechatalka\views\templater,
mirzaev\pechatalka\models\core as models, svoboda\pechatalka\models\core as models,
mirzaev\pechatalka\models\session, svoboda\pechatalka\models\enumerations\language;
mirzaev\pechatalka\models\enumerations\language;
// Framework for PHP // Framework for PHP
use mirzaev\minimal\core as minimal, use mirzaev\minimal\core as minimal,
@ -19,7 +18,7 @@ use mirzaev\minimal\core as minimal,
/** /**
* Controllers core * Controllers core
* *
* @package mirzaev\pechatalka\controllers * @package svoboda\pechatalka\controllers
* *
* @param language $language Language * @param language $language Language
* @param response $response Response * @param response $response Response
@ -29,7 +28,6 @@ use mirzaev\minimal\core as minimal,
* *
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License * @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy> * @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
* @author mirzaev <mail@domain.zone>
*/ */
class core extends controller class core extends controller
{ {
@ -75,7 +73,7 @@ class core extends controller
if (isset($_SERVER['HTTP_USER_AGENT']) && $_SERVER['HTTP_USER_AGENT'] === 'nginx-ssl early hints') return status::bruh->label; if (isset($_SERVER['HTTP_USER_AGENT']) && $_SERVER['HTTP_USER_AGENT'] === 'nginx-ssl early hints') return status::bruh->label;
// Initializing the view template engine instance // Initializing the view template engine instance
$this->view = new templater($this->session); $this->view = new templater();
// For the extends system // For the extends system
parent::__construct(core: $core); parent::__construct(core: $core);

View File

@ -2,10 +2,10 @@
declare(strict_types=1); declare(strict_types=1);
namespace mirzaev\pechatalka\controllers; namespace svoboda\pechatalka\controllers;
// Files of the project // Files of the project
use mirzaev\pechatalka\controllers\core; use svoboda\pechatalka\controllers\core;
// Framework for PHP // Framework for PHP
use mirzaev\minimal\http\enumerations\content, use mirzaev\minimal\http\enumerations\content,
@ -14,7 +14,7 @@ use mirzaev\minimal\http\enumerations\content,
/** /**
* Index * Index
* *
* @package mirzaev\pechatalka\controllers * @package svoboda\pechatalka\controllers
* *
* @param array $errors Registry of errors * @param array $errors Registry of errors
* *
@ -22,7 +22,6 @@ use mirzaev\minimal\http\enumerations\content,
* *
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License * @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy> * @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
* @author mirzaev <mail@domain.zone>
*/ */
final class index extends core final class index extends core
{ {

View File

@ -0,0 +1,83 @@
<?php
// Exit (success)
return [
// System
'svoboda' => 'Svoboda',
'svoboda_work_union' => 'Svoboda Work Union',
'svoboda_work_union_short' => 'SWU',
'pechatalka' => 'Pechatalka',
'empty' => 'Empty',
'yes' => 'Yes',
'no' => 'No',
// Main menu
'menu_title' => 'Main menu',
'menu_accounts' => 'Accounts',
'menu_button_constructor_pin' => 'Pin',
'menu_not_syncronized' => 'The database does not synchronize with the blockchain network',
// Account
'account_title' => 'Account',
'account_export' => 'This account can be exported to the Svoboda main blockchain network in the future',
'account_data' => "⚠️ Your data may be given to the special services of Russia\nBe careful and think with your head.",
'account_security_repository' => 'Information security',
'account_security_repository_mirror_github' => 'mirror',
'account_authorized_system' => 'Access to the system',
'account_authorized_settings' => 'Access to settings',
'account_authorized_system_accounts' => 'System access to accounts management',
'account_authorized_system_settings' => 'System access to the system settings',
'account_button_localizations' => 'Localizations',
'account_localization_create_failted_to_initialize_language' => 'Failed to initialize language',
// Language setting
'settings_select_language_title' => 'Select language',
'settings_select_language_description' => 'The selected language will be writed in your account settings',
'settings_language_update_success' => 'Language replaced:',
'settings_language_update_fail' => 'Failed to replace language',
// Language selection
'select_language_title' => 'Select language',
'select_language_description' => 'The selected language will be used in the current process',
'select_language_button_add' => 'Add a language',
// Repository
'repository_title' => 'Repository',
'repository_text' => <<<TXT
Pechatalka is written in [PHP](https://www.php.net/) using [Zanzara](https://github.com/badfarm/zanzara) for Telegram,
my [MINIMAL](https://git.svoboda.works/mirzaev/minimal) framework for PHP and my [Baza](https://git.svoboda.works/mirzaev/baza) database
The code is under the [WTFPL](https://en.wikipedia.org/wiki/WTFPL) license
You can help me with the development, or use my code for free\!
TXT,
'repository_button_code' => 'The code',
'repository_button_issues' => 'Issues',
'repository_button_suggestions' => 'Suggestions',
// Author
'author_title' => 'Author',
'author_text' => <<<TXT
*Arsen Mirzaev Tatyano\-Muradovich*
Programmer, anarchist, vegetarian
TXT,
'author_button_neurojournal' => 'Neurojournal',
'author_button_projects' => 'Projects',
'author_button_telegram' => 'Telegram',
'author_button_twitter' => 'Twitter',
'author_button_bluesky' => 'Bluesky',
'author_button_bastyon' => 'Bastyon',
'author_button_youtube_english' => 'YouTube',
'author_button_youtube_russian' => 'YouTube',
'author_button_message' => 'Send a message',
// Authorization
'not_authorized_system' => 'You do not have access to the system',
'not_authorized_messages' => 'You do not have access to send messages',
'not_authorized_joins' => 'You do not have access to joins',
'not_authorized_settings' => 'You do not have access to the settings',
'not_authorized_system_settings' => 'You do not have access to the system settings',
'not_authorized_system_distributions' => 'You do not have access to distributions administration',
// Other
'why_so_shroomious' => 'why so shroomious'
];

View File

@ -0,0 +1,88 @@
<?php
// Exit (success)
return [
// System
'svoboda' => 'Свобода',
'svoboda_work_union' => 'Рабочий Союз Свободы',
'svoboda_work_union_short' => 'РСС',
'Pechatalka' => 'Печаталка',
'empty' => 'Пусто',
'yes' => 'Да',
'no' => 'Нет',
// Main menu
'menu_title' => 'Главное меню',
'menu_accounts' => 'Аккаунты',
'menu_button_constructor_pin' => 'Значок',
'menu_not_syncronized' => 'База данных не синхронизируется с блокчейн сетью',
// Аккаунт
'account_title' => 'Аккаунт',
'account_export' => 'Этот аккаунт может быть экспортирован в основную блокчейн сеть Свободы в будущем',
'account_data' => "⚠️ Твои данные могут быть выданы спецслужбам России\nБудь осторожен и думай своей головой",
'account_security_repository' => 'Информационная безопасность',
'account_security_repository_mirror_github' => 'зеркало',
'account_authorized_system' => 'Доступ к системе',
'account_authorized_messages' => 'Доступ к сообщениям',
'account_authorized_joins' => 'Доступ к вступлениям',
'account_authorized_settings' => 'Доступ к изменению настроек',
'account_authorized_system_accounts' => 'Системный доступ к управлению аккаунтами',
'account_authorized_system_distributions' => 'Системный доступ к управлению дистрибутивами',
'account_authorized_system_members' => 'Системный доступ к управлению участниками дистрибутивов',
'account_authorized_system_settings' => 'Системный доступ к системным настройкам',
'account_button_localizations' => 'Локализации',
'account_localization_create_failted_to_initialize_language' => 'Не удалось инициализировать язык',
// Настройки языка
'settings_select_language_title' => 'Выбери язык',
'settings_select_language_description' => 'Выбранный язык будет записан в настройки аккаунта',
'settings_language_update_success' => 'Язык заменён:',
'settings_language_update_fail' => 'Не удалось заменить язык',
// Выбор языка
'select_language_title' => 'Выбери язык',
'select_language_description' => 'Выбранный язык будет использован в текущем процессе',
'select_language_button_add' => 'Добавить язык',
// Репозиторий
'repository_title' => 'Репозиторий',
'repository_text' => <<<TXT
Печаталка написана на [PHP](https://www.php.net/) используя [Zanzara](https://github.com/badfarm/zanzara) для Telegram,
мой [MINIMAL](https://git.svoboda.works/mirzaev/minimal) фреймворк для PHP и моя база данных [Baza](https://git.svoboda.works/mirzaev/baza)
Код находится под лицензией [WTFPL](https://en.wikipedia.org/wiki/WTFPL)
Помогай с разработкой или используй мой код бесплатно\!
TXT,
'repository_button_code' => 'Код',
'repository_button_issues' => 'Проблемы',
'repository_button_suggestions' => 'Предложения',
// Автор
'author_title' => 'Автор',
'author_text' => <<<TXT
*Арсен Мирзаев Татьяно\-Мурадович*
Программист, анархист, вегетарианец
TXT,
'author_button_neurojournal' => 'Нейрожурнал',
'author_button_projects' => 'Проекты',
'author_button_telegram' => 'Телеграм',
'author_button_twitter' => 'Twitter',
'author_button_bluesky' => 'Bluesky',
'author_button_bastyon' => 'Bastyon',
'author_button_youtube_english' => 'YouTube',
'author_button_youtube_russian' => 'YouTube',
'author_button_message' => 'Отправить сообщение',
// Авторизация
'not_authorized_system' => 'У тебя нет доступа к системе',
'not_authorized_messages' => 'У тебя нет доступа к сообщениям',
'not_authorized_joins' => 'У тебя нет доступа к вступлениям',
'not_authorized_settings' => 'У тебя нет доступа к настройкам',
'not_authorized_system_settings' => 'У тебя нет доступа к системным настройкам',
'not_authorized_system_distributions' => 'У тебя нет доступа к администрированию дистрибутивов',
// Прочее
'why_so_shroomious' => 'почему такой грибъёзный'
];

View File

@ -0,0 +1,202 @@
<?php
declare(strict_types=1);
namespace svoboda\pechatalka\models;
// Files of the project
use svoboda\pechatalka\models\core;
// Svoboda time
use svoboda\time\statement as svoboda;
// Baza database
use mirzaev\baza\database,
mirzaev\baza\column,
mirzaev\baza\record,
mirzaev\baza\enumerations\encoding,
mirzaev\baza\enumerations\type;
// Framework for Telegram
use Zanzara\Telegram\Type\User as telegram;
// Built-in libraries
use Exception as exception,
RuntimeException as exception_runtime;
/**
* Account
*
* @package svoboda\pechatalka\models
*
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
*/
final class account extends core
{
/**
* File
*
* @var string $database Path to the database file
*/
protected string $file = DATABASES . DIRECTORY_SEPARATOR . 'accounts.baza';
/**
* Database
*
* @var database $database The database
*/
public protected(set) database $database;
/**
* Constructor
*
* @return void
*/
public function __construct()
{
// Initializing the database
$this->database = new database()
->encoding(encoding::utf8)
->columns(
new column('identifier', type::integer_unsigned),
new column('identifier_telegram', type::integer),
new column('domain', type::string, ['length' => 32]),
new column('name_first', type::string, ['length' => 64]),
new column('name_second', type::string, ['length' => 64]),
new column('language', type::string, ['length' => 2]),
new column('robot', type::char),
new column('authorized_system', type::char),
new column('authorized_settings', type::char),
new column('authorized_system_accounts', type::char),
new column('authorized_system_settings', type::char),
new column('updated', type::integer_unsigned),
new column('created', type::integer_unsigned)
)
->connect($this->file);
}
/**
* Initialize
*
* Searches for the account record in the database, and if it does not find it, it creates it
*
* @param telegram $telegram The telegram account
*
* @throws exception_runtime if update the account record in the database by the telegram account values
* @throws exception_runtime if failed to find the registered account
* @throws exception_runtime if failed to registrate the account
*
* @return record The account record from the database
*/
public function initialize(telegram $telegram): record
{
// Searching for the account in the database
$account = $this->database->read(filter: fn(record $record) => $record->identifier_telegram === $telegram->getId(), amount: 1)[0] ?? null;
if ($account instanceof record) {
// Found the account record
if (
$account->name_first !== $telegram->getFirstName() ||
$account->name_second !== $telegram->getLastName() ||
$account->domain !== $telegram->getUsername()
) {
// The telegram account was updated
// Updating the account in the database
$updated = $this->database->read(
filter: fn(record $record) => $record->identifier_telegram === $telegram->getId(),
update: function (record &$record) use ($telegram){
// Writing new values into the record
$record->name_first = $telegram->getFirstName();
$record->name_second = $telegram->getLastName();
$record->domain = $telegram->getUsername();
$record->updated = svoboda::timestamp();
},
amount: 1
)[0] ?? null;
if ($updated instanceof record && $updated->values() !== $account->values()) {
// Updated the account in the database
// Exit (success)
return $updated;
} else {
// Not updated the account in the database
// Exit (fail)
throw new exception_runtime('Failed to update the account record in the database by the telegram account values');
}
}
// Exit (success)
return $account;
} else {
// Not found the account record
if ($this->registrate($telegram)) {
// Registered the account
// Searching for the registered account in the database
$account = $this->database->read(filter: fn(record $record) => $record->identifier_telegram === $telegram->getId(), amount: 1)[0] ?? null;
if ($account instanceof record) {
// Found the registered account
// Exit (success)
return $account;
} else {
// Not found the registered account
// Exit (fail)
throw new exception_runtime('Failed to find the registered account');
}
} else {
// Not registered the account
// Exit (fail)
throw new exception_runtime('Failed to registrate the account');
}
}
}
/**
* Registrate
*
* Creates the account record in the database
*
* @param telegram $telegram The telegram account
*
* @return int|false The record identifier, if created
*/
public function registrate(telegram $telegram): int|false
{
// Initializing the identifier
$identifier = $this->database->count() + 1;
// Initializing the record
$record = $this->database->record(
$identifier,
(int) $telegram->getId(),
$telegram->getFirstName(),
$telegram->getLastName(),
$telegram->getUsername(),
$telegram->getLanguageCode(),
(int) $telegram->isBot(),
1,
1,
0,
0,
svoboda::timestamp(),
svoboda::timestamp()
);
// Creating the record in the database
$created = $this->database->write($record);
// Exit (success)
return $created ? $identifier : false;
}
}

View File

@ -2,7 +2,7 @@
declare(strict_types=1); declare(strict_types=1);
namespace mirzaev\pechatalka\models; namespace svoboda\pechatalka\models;
// Framework for PHP // Framework for PHP
use mirzaev\minimal\model, use mirzaev\minimal\model,
@ -14,13 +14,12 @@ use exception;
/** /**
* Models core * Models core
* *
* @package mirzaev\pechatalka\models * @package svoboda\pechatalka\models
* *
* @method void __construct() Constructor * @method void __construct() Constructor
* *
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License * @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy> * @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
* @author mirzaev <mail@domain.zone>
*/ */
class core extends model class core extends model
{ {
@ -38,8 +37,5 @@ class core extends model
* *
* @return void * @return void
*/ */
public function __construct() public function __construct() {}
{
}
} }

View File

@ -2,14 +2,14 @@
declare(strict_types=1); declare(strict_types=1);
namespace mirzaev\pechatalka\models\enumerations; namespace svoboda\pechatalka\models\enumerations;
/** /**
* Language * Language
* *
* Types of languages by ISO 639-1 standart * Types of languages by ISO 639-1 standart
* *
* @package mirzaev\pechatalka\models\enumerations * @package svoboda\pechatalka\models\enumerations
* *
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License * @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy> * @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>

View File

@ -0,0 +1,33 @@
<?php
declare(strict_types=1);
namespace svoboda\pechatalka\models\telegram;
// Files of the project
use svoboda\pechatalka\models\core,
svoboda\pechatalka\models\account as model,
svoboda\pechatalka\models\enumerations\language;
// Framework for Telegram
use Zanzara\Context as context,
Zanzara\Telegram\Type\Message as message;
// Baza database
use mirzaev\baza\record;
// Built-in libraries
use Error as error;
/**
* Telegram account
*
* @package svoboda\pechatalka\models\telegram
*
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
*/
final class account extends core
{
}

View File

@ -0,0 +1,533 @@
<?php
declare(strict_types=1);
namespace svoboda\pechatalka\models\telegram;
// Files of the project
use svoboda\pechatalka\models\core,
svoboda\pechatalka\models\account,
svoboda\pechatalka\models\distribution,
svoboda\pechatalka\models\member,
svoboda\pechatalka\models\telegram\processes\language\select as process_language_select,
svoboda\pechatalka\models\enumerations\language;
// Framework for Telegram
use Zanzara\Context as context,
Zanzara\Telegram\Type\Message as message,
Zanzara\Telegram\Type\Input\InputFile as file_input;
// Baza database
use mirzaev\baza\record;
/**
* Telegram commands
*
* @package svoboda\pechatalka\models\telegram
*
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
*/
final class commands extends core
{
/**
* Menu
*
* Responce for the commands: "/start", '/menu'
*
* @param context $context Request data from Telegram
*
* @return void
*/
public static function menu(context $context): void
{
// Initializing the account
$account = $context->get('account');
if ($account instanceof record) {
// Initialized the account
// Initializing localization
$localization = $context->get('localization');
if ($localization) {
// Initialized localization
// Initializing the title
$title = '📋 *' . $localization['menu_title'] . '*';
// Initializing accounts
$accounts = '*' . $localization['menu_accounts'] . ':* ' . ((new account)->database->count() ?? 0);
// Initializing the data syncronization for the message
$syncronization = '⛓️‍💥 ' . $localization['menu_not_syncronized'];
// Sending the message
$context->sendMessage(
<<<TXT
$title
$accounts
$syncronization
TXT,
[
'reply_markup' => [
'inline_keyboard' => [
[
[
'text' => '🔘 ' . $localization['menu_button_constructor_pin'],
'web_app' => [
'url' => 'https://pechatalka.svoboda.works/constructor/pin'
]
]
]
],
'disable_notification' => true,
'remove_keyboard' => true
],
]
)
->then(function (message $message) use ($context) {
// Sended the message
});
} else {
// Not initialized localization
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $message) use ($context) {
// Ending the conversation process
$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $message) use ($context) {
// Ending the conversation process
$context->endConversation();
});
}
}
/**
* Account
*
* Responce for the command: "/account"
*
* Sends information about account with menu
*
* @param context $context Request data from Telegram
*
* @return void
*/
public static function account(context $context): void
{
// Initializing the account
$account = $context->get('account');
if ($account instanceof record) {
// Initialized the account
// Initializing localization
$localization = $context->get('localization');
if ($localization) {
// Initialized localization
// Initializing title for the message
$title = '🫵 ' . $localization['account_title'];
// Declaring buufer of rows about authorizations
$authorizations = '';
// Initializing rows about authorization
foreach ($account->values() as $key => $value) {
// Iterating over account parameters
if (str_starts_with($key, 'authorized_')) {
// Iterating over account authorizations
// Skipping system authorizations
if (str_starts_with($key, 'authorized_system_')) continue;
// Writing into buffer of rows about authorizations
$authorizations .= ($value ? '✅' : '❎') . ' *' . ($localization["account_$key"] ?? $key) . ':* ' . ($value ? $localization['yes'] : $localization['no']) . "\n";
}
}
// Trimming the last line break character
$authorizations = trim($authorizations, "\n");
// Initializing the data export for the message
$export = '📤 ' . $localization['account_export'];
// Initializing the data security for the message
$data = $localization['account_data'];
// Initializing the data security repository for the message
$security = '📁 [' . $localization['account_security_repository'] . '](https://git.svoboda.works/mirzaev/security) \([' . $localization['account_security_repository_mirror_github'] . '](https://github.com/mature-woman/security)\)';
// Sending the message
$context->sendMessage(
<<<TXT
$title
$authorizations
$export
$data
$security
TXT,
[
'reply_markup' => [
'remove_keyboard' => true,
'disable_notification' => true
],
'link_preview_options' => [
'is_disabled' => true
]
]
);
} else {
// Not initialized localization
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $message) use ($context) {
// Ending the conversation process
$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $message) use ($context) {
// Ending the conversation process
$context->endConversation();
});
}
}
/**
* Language
*
* Responce for the command: "/language"
*
* Send the language selection menu
*
* @param context $context Request data from Telegram
*
* @return void
*/
public static function language(context $context): void
{
// Initializing the account
$account = $context->get('account');
if ($account instanceof record) {
// Initialized the account
// Initializing language
$language = $context->get('language');
if ($language instanceof language) {
// Initialized language
// Initializing localization
$localization = $context->get('localization');
if ($localization) {
// Initialized localization
// Sending the language selection
process_language_select::menu(
context: $context,
prefix: 'settings_language_',
title: '🌏 *' . $localization['settings_select_language_title'] . '*',
description: '🌏 *' . $localization['settings_select_language_description'] . '*'
);
} else {
// Not initialized localization
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
}
} else {
// Not initialized language
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize language*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
}
}
/**
* Repository
*
* Responce for the command: "/repository"
*
* Sends information about project and menu
*
* @param context $context Request data from Telegram
*
* @return void
*/
public static function repository(context $context): void
{
// Initializing the account
$account = $context->get('account');
if ($account instanceof record) {
// Initialized the account
// Initializing localization
$localization = $context->get('localization');
if ($localization) {
// Initialized localization
// Initializing title of the message
$title = '🏛️ ' . $localization['repository_title'];
// Sending the message
$context->sendMessage($title . "\n\n" . $localization['repository_text'], [
'reply_markup' => [
'inline_keyboard' => [
[
[
'text' => '🏛️ ' . $localization['repository_button_code'],
'url' => 'https://git.mirzaev.sexy/svoboda/pechatalka'
]
],
[
[
'text' => '⚠️ ' . $localization['repository_button_issues'],
'url' => 'https://git.mirzaev.sexy/svoboda/pechatalka/issues'
],
[
'text' => '🌱 ' . $localization['repository_button_suggestions'],
'url' => 'https://git.mirzaev.sexy/svoboda/pechatalka/issues'
]
]
],
'remove_keyboard' => true,
'disable_notification' => true
],
'link_preview_options' => [
'is_disabled' => true
]
]);
} else {
// Not initialized localization
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $message) use ($context) {
// Ending the conversation process
$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $message) use ($context) {
// Ending the conversation process
$context->endConversation();
});
}
}
/**
* Author
*
* Responce for the command: "/author"
*
* Sends
*
* @param context $context Request data from Telegram
*
* @return void
*/
public static function author(context $context): void
{
// Initializing the account
$account = $context->get('account');
if ($account instanceof record) {
// Initialized the account
// Initializing localization
$localization = $context->get('localization');
if ($localization) {
// Initialized localization
// Initializing title of the message
$title = '👽 ' . $localization['author_title'];
// Sending the message
$context->sendMessage($title . "\n\n" . $localization['author_text'], [
'reply_markup' => [
'inline_keyboard' => [
[
[
'text' => '📚 ' . $localization['author_button_neurojournal'],
'url' => 'https://mirzaev.sexy'
],
[
'text' => '🤟 ' . $localization['author_button_projects'],
'url' => 'https://git.svoboda.works/mirzaev?tab=activity'
]
],
[
[
'text' => '📣 ' . $localization['author_button_telegram'],
'url' => 'https://t.me/blog_mirzaev_sexy'
],
[
'text' => '✖️ ' . $localization['author_button_twitter'],
'url' => 'https://x.com/mirzaev_sexy'
],
[
'text' => '🦋 ' . $localization['author_button_bluesky'],
'url' => 'https://bsky.app/profile/mirzaev.bsky.social'
],
[
'text' => '⛓️ ' . $localization['author_button_bastyon'],
'url' => 'https://bsky.app/profile/mirzaev.bsky.social'
]
],
[
[
'text' => '🇺🇸 ' . $localization['author_button_youtube_english'],
'url' => 'https://www.youtube.com/@MIRZAEV'
],
[
'text' => '🇷🇺 ' . $localization['author_button_youtube_russian'],
'url' => 'https://www.youtube.com/@MIRZAEV'
]
],
[
[
'text' => '✉️ ' . $localization['author_button_message'],
'url' => 'https://t.me/mirzaev_sexy'
]
]
],
'remove_keyboard' => true,
'disable_notification' => true
],
'link_preview_options' => [
'is_disabled' => true
]
]);
} else {
// Not initialized localization
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $message) use ($context) {
// Ending the conversation process
$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $message) use ($context) {
// Ending the conversation process
$context->endConversation();
});
}
}
/**
* Society
*
* Responce for the command: "/society"
*
* Sends the "mushroom" image and the localized text "why so shroomious"
*
* @param context $context Request data from Telegram
*
* @return void
*/
public static function society(context $context): void
{
// Initializing the account
$account = $context->get('account');
if ($account instanceof record) {
// Initialized the account
// Initializing localization
$localization = $context->get('localization');
if ($localization) {
// Initialized localization
// Sending the message
$context->sendPhoto(
new file_input(STORAGE . DIRECTORY_SEPARATOR . 'images' . DIRECTORY_SEPARATOR . 'mushroom.jpg'),
[
'caption' => $localization['why_so_shroomious'],
'disable_notification' => true
]
);
} else {
// Not initialized localization
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $message) use ($context) {
// Ending the conversation process
$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $message) use ($context) {
// Ending the conversation process
$context->endConversation();
});
}
}
}

View File

@ -0,0 +1,372 @@
<?php
declare(strict_types=1);
namespace svoboda\pechatalka\models\telegram;
// Files of the project
use svoboda\pechatalka\models\core,
svoboda\pechatalka\models\account,
svoboda\pechatalka\models\enumerations\language;
// Framework for Telegram
use Zanzara\Context as context,
Zanzara\Telegram\Type\Message as message,
Zanzara\Middleware\MiddlewareNode as node;
// Baza database
use mirzaev\baza\record;
// Built-in libraries
use Error as error;
/**
* Telegram middlewares
*
* @package svoboda\pechatalka\models\telegram
*
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
*/
final class middlewares extends core
{
/**
* Account (middleware)
*
* Initialize or registrate the account and write it to the `account` variable inside the `$context`
*
* @param context $context
* @param node $next
*
* @return void
*/
public static function account(context $context, node $next): void
{
// Is the process stopped?
if ($context->get('stop')) return;
// Initializing the telegram account
$telegram = $context->getEffectiveUser();
// Initializing the account
/* $account = new account()->initialize($telegram); */
$account = (new account())->initialize($telegram);
if ($account instanceof record) {
// Initialized the account
// Writing the account into the context variable
$context->set('account', $account);
// Continuation of the process
$next($context);
} else {
// Not initialized the account
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
}
}
/**
* Language (middleware)
*
* Implement the account language
*
* @param context $context
* @param node $next
*
* @return void
*/
public static function language(context $context, node $next): void
{
// Is the process stopped?
if ($context->get('stop')) return;
// Initializing the account
$account = $context->get('account');
if ($account instanceof record) {
// Initialized the account
if ($account->language) {
// Initialized the language parameter
try {
// Writing the account language into the context variable
$context->set('language', language::{$account->language});
} catch (error $error) {
// Not initialized the language
// Writing the default language into the context variable
$context->set('language', language::en);
}
} else {
// Not initialized the language parameter
// Writing the default language into the context variable
$context->set('language', language::en);
}
// Continuation of the process
$next($context);
} else {
// Not initialized the account
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
}
}
/**
* Localization (middleware)
*
* Implement the account language and initialize the localization file
*
* @param context $context
* @param node $next
*
* @return void
*/
public static function localization(context $context, node $next): void
{
// Is the process stopped?
if ($context->get('stop')) return;
// Initializing the account
$account = $context->get('account');
if ($account instanceof record) {
// Initialized the account
// Initializing the language
$language = $context->get('language');
if ($language instanceof language) {
// Initialized the language
// Initializing path to the localization file
$file = LOCALIZATIONS . DIRECTORY_SEPARATOR . strtolower($language->label()) . '.php';
if (file_exists($file) && is_readable($file)) {
// Found the localization file
// Initializing localization
$localization = require($file);
if (is_array($localization)) {
// Initialized localization
// Writing localization into the context variable
$context->set('localization', $localization);
// Continuation of the process
$next($context);
} else {
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
}
} else {
// Not found the localization file
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize the localization file*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
}
} else {
// Not initialized language
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize language*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
}
}
/**
* Settings (middleware)
*
* Check the account for access to the settings
*
* @param context $context
* @param node $next
*
* @return void
*/
public static function settings(context $context, node $next): void
{
// Is the process stopped?
if ($context->get('stop')) return;
// Initializing the account
$account = $context->get('account');
if ($account instanceof record) {
// Initialized the account
if ($account->authorized_settings) {
// Authorized the account to the settings
// Continuation of the process
$next($context);
} else {
// Not authorized the account to the settings
// Initializing localization
$localization = $context->get('localization');
if ($localization) {
// Initialized localization
// Sending the message
$context->sendMessage('⛔ *' . $localization['not_authorized_settings'] . '*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
// Stopping the process
$context->set('stop', true);
} else {
// Not initialized localization
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
}
}
} else {
// Not initialized the account
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
}
}
/**
* System settings (middleware)
*
* Check the account for access to the system settings
*
* @param context $context
* @param node $next
*
* @return void
*/
public static function system_settings(context $context, node $next): void
{
// Is the process stopped?
if ($context->get('stop')) return;
// Initializing the account
$account = $context->get('account');
if ($account instanceof record) {
// Initialized the account
if ($account->authorized_system_settings) {
// Authorized the account to the system settings
// Continuation of the process
$next($context);
} else {
// Not authorized the account to the system settings
// Initializing localization
$localization = $context->get('localization');
if ($localization) {
// Initialized localization
// Sending the message
$context->sendMessage('⛔ *' . $localization['not_authorized_system_settings'] . '*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
// Stopping the process
$context->set('stop', true);
} else {
// Not initialized localization
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
}
}
} else {
// Not initialized the account
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
}
}
}

View File

@ -0,0 +1,150 @@
<?php
declare(strict_types=1);
namespace svoboda\pechatalka\models\telegram\processes\language;
// Files of the project
use svoboda\pechatalka\models\core,
svoboda\pechatalka\models\enumerations\language;
// Framework for Telegram
use Zanzara\Context as context,
Zanzara\Telegram\Type\Message as message;
// Baza database
use mirzaev\baza\record;
/**
* Telegram language select
*
* @package svoboda\pechatalka\models\telegram\processes\language
*
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
*/
final class select extends core
{
/**
* Language
*
* Send the language choose menu
*
* @param context $context Request data from Telegram
* @param string $prefix Prefix for 'callback_data' (`$prefix . $language->name`)
* @param string $title Title of the message
* @param string $description Description of the message
* @param array $exclude Languages that will be excluded ['ru', 'en'...]
*
* @return void
*/
public static function menu(context $context, string $prefix, string $title, string $description, array $exclude = []): void
{
// Initializing the account
$account = $context->get('account');
if ($account instanceof record) {
// Initialized the account
// Initializing language
$language = $context->get('language');
if ($language) {
// Initialized language
// Initializing localization
$localization = $context->get('localization');
if ($localization) {
// Initialized localization
// Declaring the buffer of generated keyboard with languages
$keyboard = [];
// Initializing the iterator of rows
$row = 0;
// Initializing buffer of languages
$languages = language::cases();
// Deleting the actual language from buffer of languages
unset($languages[array_search($language, $languages, strict: true)]);
// Sorting buffer of languages by the actual language
$languages = [$language, ...$languages];
foreach ($languages as $language) {
// Iterating over languages
// Skipping excluded languages
if (array_search($language->name, $exclude, strict: true) !== false) continue;
// Initializing the row
$keyboard[$row] ??= [];
// Writing the language choose button into the buffer of generated keyboard with languages
$keyboard[$row][] = [
'text' => ($language->flag() ? $language->flag() . ' ' : '') . $language->label($language),
'callback_data' => $prefix . $language->name
];
// When reaching 4 buttons in a row, move to the next row
if (count($keyboard[$row]) === 4) ++$row;
}
// Writing the button for helping lozalizing
$keyboard[$row === 0 && empty($keyboard[0]) ? 0 : ++$row] = [
[
'text' => '🗂 ' . $localization['select_language_button_add'],
'url' => 'https://git.svoboda.works/svoboda/pechatalka/src/branch/stable/svoboda/pechatalka/system/localizations'
]
];
// Sending the message
$context->sendMessage(
$title ?? '🌏 *' . $localization['select_language_title'] . "*\n" . ($description ?? $localization['select_language_description']),
[
'reply_markup' => [
'inline_keyboard' => $keyboard,
'disable_notification' => true
],
]
);
} else {
// Not initialized localization
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
}
} else {
// Not initialized language
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize language*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $message) use ($context) {
// Sended the message
// Ending the conversation process
$context->endConversation();
});
}
}
}

View File

@ -0,0 +1,157 @@
<?php
declare(strict_types=1);
namespace svoboda\pechatalka\models\telegram;
// Files of the project
use svoboda\pechatalka\models\core,
svoboda\pechatalka\models\account,
svoboda\pechatalka\models\enumerations\language,
svoboda\pechatalka\models\telegram\middlewares;
// Framework for Telegram
use Zanzara\Zanzara,
Zanzara\Context as context,
Zanzara\Telegram\Type\Message as message,
Zanzara\Middleware\MiddlewareNode as node;
// Baza database
use mirzaev\baza\record;
// Built-in libraries
use Error as error;
/**
* Telegram settings
*
* @package svoboda\pechatalka\models\telegram
*
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>
*/
final class settings extends core
{
/**
* Language
*
* Write language into the account record
*
* @param context $context Request data from Telegram
* @param language $language The language
*
* @return void
*/
public static function language(context $context, language $language): void
{
// Initializing the account
$account = $context->get('account');
if ($account instanceof record) {
// Initialized the account
// Initializing localization
$localization = $context->get('localization');
if ($localization) {
// Initialized localization
// Initializing the account model
$model = new account();
// Updating the account in the database
$updated = $model->database->read(
filter: fn(record $record) => $record->identifier === $account->identifier,
update: function (record &$record) use ($language) {
// Writing new language value into the record
$record->language = $language->name;
},
amount: 1
)[0] ?? null;
if ($updated instanceof record) {
// Updated the account in the database
// Writing the updated account into the context variable
$context->set('account', $updated);
middlewares::language($context, new node(function (context $context) use ($account, $updated) {
// Updated language
middlewares::localization($context, new node(function (context $context) use ($account, $updated) {
// Updated localization
// Initializing localization
$localization = $context->get('localization');
if ($localization) {
// Initialized localization
try {
// Initializing the old language
$old = language::{$account->language};
// Initializing the new language
$new = language::{$updated->language};
// Sending the message
$context->sendMessage('✅ *' . $localization['settings_language_update_success'] . '* ' . ($old->flag() ? $old->flag() . ' ' : '') . $old->label($new) . ' → *' . ($new->flag() ? $new->flag() . ' ' : '') . $new->label($new) . '*')
->then(function (message $message) use ($context) {
// Ending the conversation process
$context->endConversation();
});
} catch (error $error) {
// Failed to send the message about language update
// Sending the message
$context->sendMessage('❎ *' . $localization['settings_language_update_fail'])
->then(function (message $message) use ($context) {
// Ending the conversation process
$context->endConversation();
});
}
} else {
// Not initialized localization
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $message) use ($context) {
// Ending the conversation process
$context->endConversation();
});
}
}));
}));
} else {
// Not updated the account in the database
// Sending the message
$context->sendMessage('❎ *' . $localization['settings_language_update_fail'])
->then(function (message $message) use ($context) {
// Ending the conversation process
$context->endConversation();
});
}
} else {
// Not initialized localization
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize localization*')
->then(function (message $message) use ($context) {
// Ending the conversation process
$context->endConversation();
});
}
} else {
// Not initialized the account
// Sending the message
$context->sendMessage('⚠️ *Failed to initialize your Telegram account*')
->then(function (message $message) use ($context) {
// Ending the conversation process
$context->endConversation();
});
}
}
}

View File

@ -2,7 +2,7 @@
declare(strict_types=1); declare(strict_types=1);
namespace mirzaev\pechatalka\models\traits; namespace svoboda\pechatalka\models\traits;
// Built-in libraries // Built-in libraries
use exception; use exception;
@ -14,7 +14,7 @@ use exception;
* *
* @method static void delete(string $directory, array &$errors) * @method static void delete(string $directory, array &$errors)
* *
* @package mirzaev\pechatalka\models\traits * @package svoboda\pechatalka\models\traits
* *
* @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License * @license http://www.wtfpl.net/ Do What The Fuck You Want To Public License
* @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy> * @author Arsen Mirzaev Tatyano-Muradovich <arsen@mirzaev.sexy>

View File

@ -0,0 +1,8 @@
@charset "UTF-8";
@font-face {
font-family: 'Cascadia Code';
src: url("/fonts/geologica/CascadiaCode-Regular.woff2");
font-weight: 400;
font-style: normal;
}

View File

@ -0,0 +1,7 @@
@charset "UTF-8";
@font-face {
font-family: 'Commissioner';
src: url('/fonts/commissioner/commissioner.ttf');
font-weight: 400;
}

View File

@ -0,0 +1,65 @@
@charset "UTF-8";
@font-face {
font-family: 'Geologica';
src: url("/fonts/geologica/Geologica-Thin.ttf");
font-weight: 100;
font-style: normal;
}
@font-face {
font-family: 'Geologica';
src: url("/fonts/geologica/Geologica_Auto-ExtraLight.ttf");
font-weight: 200;
font-style: normal;
}
@font-face {
font-family: 'Geologica';
src: url("/fonts/geologica/Geologica-Light.ttf");
font-weight: 300;
font-style: normal;
}
@font-face {
font-family: 'Geologica';
src: url("/fonts/geologica/Geologica-Regular.ttf");
font-weight: 400;
font-style: normal;
}
@font-face {
font-family: 'Geologica';
src: url("/fonts/geologica/Geologica-Medium.ttf");
font-weight: 500;
font-style: normal;
}
@font-face {
font-family: 'Geologica';
src: url("/fonts/geologica/Geologica-SemiBold.ttf");
font-weight: 600;
font-style: normal;
}
@font-face {
font-family: 'Geologica';
src: url("/fonts/geologica/Geologica-Bold.ttf");
font-weight: 700;
font-style: normal;
}
@font-face {
font-family: 'Geologica';
src: url("/fonts/geologica/Geologica-ExtraBold.ttf");
font-weight: 800;
font-style: normal;
}
@font-face {
font-family: 'Geologica';
src: url("/fonts/geologica/Geologica-Black.ttf");
font-weight: 900;
font-style: normal;
}

Some files were not shown because too many files have changed in this diff Show More