<?xml version="1.0" encoding="utf-8"?> 
<rss version="2.0"
  xmlns:itunes="http://www.itunes.com/dtds/podcast-1.0.dtd"
  xmlns:atom="http://www.w3.org/2005/Atom">

<channel>

<title>В наушниках по жизни: заметки с тегом Android</title>
<link>https://alexbirukov.ru/?go=tags/android/</link>
<description>Здесь я пишу свои идеи, мысли, описываю события своей жизни и просто делаю личные и рабочие заметки. «А Вы что, собираетесь жить вечно?»</description>
<author></author>
<language>ru</language>
<generator>Aegea 11.3 (v4134)</generator>

<itunes:subtitle>Здесь я пишу свои идеи, мысли, описываю события своей жизни и просто делаю личные и рабочие заметки. «А Вы что, собираетесь жить вечно?»</itunes:subtitle>
<itunes:image href="" />
<itunes:explicit></itunes:explicit>

<item>
<title>PHP сервер для рассылки Push на Android и iOS</title>
<guid isPermaLink="false">326</guid>
<link>https://alexbirukov.ru/?go=all/php-server-dlya-rassylki-push-na-android-i-ios/</link>
<pubDate>Fri, 02 Oct 2015 14:13:19 +0500</pubDate>
<author></author>
<comments>https://alexbirukov.ru/?go=all/php-server-dlya-rassylki-push-na-android-i-ios/</comments>
<description>
&lt;p&gt;Долгое время искал в интернете нормальное решение для массовой рассылки Push сообщений на Android и iOS, но ничего подходящего и нормально работающего не попадалось.&lt;/p&gt;
&lt;p&gt;В Delphi долгое время приходилось использовать компоненты Kinvey (или Pasre). При этом возникали новые проблемы зависимости от платформы, ключей и пр. С любым серьёзным изменением приходилось изменять, как сам проект, так и настройки проекта. В некоторых случаях сторонние компоненты отказывались работать на той или иной платформе по не понятной причине. А так же, огромным минусом Kinvey, было то, что при получении Push сообщения на iOS не воспроизводилось звуковое уведомление. В конце концов, зависимость от третьей стороны всегда является определённым минусом любого проекта.&lt;/p&gt;
&lt;p&gt;И вот после очередных поисков я объединил все имеющиеся знания и сделал серверную часть для регистрации устройств и рассылки Push сообщений.&lt;/p&gt;
&lt;p&gt;Принцип работы скрипта достаточно прост. В нём имеются функции для регистрации (удаления) токенов устройств, ну и собственно сами функции рассылки. Большая часть мануалов? которые мне удалось найти были написаны либо неправильно, либо попросту были неполными.&lt;/p&gt;
&lt;h2&gt;Введение&lt;/h2&gt;
&lt;p&gt;Немного теории. Для отправки Push сообщений используются сервисы &lt;a href="https://developer.apple.com/library/ios/documentation/NetworkingInternet/Conceptual/RemoteNotificationsPG/Chapters/ApplePushService.html"&gt;APNS&lt;/a&gt; у Apple и &lt;a href="http://developer.android.com/google/gcm/index.html"&gt;GCM&lt;/a&gt; у Google. Наш скрипт может выступать в качестве самостоятельного сервера рассылки Push через сервера Apple и Google, а так же принимать токены устройств регистрируемых в базе данных.&lt;/p&gt;
&lt;h2&gt;Apple, iOS и ApnsPHP&lt;/h2&gt;
&lt;p&gt;Начнём с более сложной части.&lt;br /&gt;
Я не стану изобретать велосипед и воспользуюсь уже имеющимися в сети, бесплатными, библиотеками написанными на PHP. Набор библиотек ApnsPHP представляет собой набор классов для работы с Push сообщениями APNS. Скачать их с примерами можно на GitHub по &lt;a href="https://github.com/immobiliare/ApnsPHP/tree/master/ApnsPHP"&gt;ссылке&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;Но одних библиотек нам мало. Самое сложное, это генерация и конвертирование сертификатов для работы с Push на iOS. Я опишу полный алгоритм экспорта и конвертации сертификатов, а затем их дальнейшее применение в скриптах.&lt;/p&gt;
&lt;h3&gt;Генерация сертификата для отправки Push сообщений&lt;/h3&gt;
&lt;p&gt;Будем предполагать, что вы уже создали и настроили сертификаты в личном кабинете разработчика и загрузили их в «Связку ключей» на ваш Mac.&lt;/p&gt;
&lt;ol start="1"&gt;
&lt;li&gt;Запускаем утилиту &lt;b&gt;«Связка ключей»&lt;/b&gt; и в левой части выбираем раздел &lt;b&gt;«Мои сертификаты»&lt;/b&gt;.&lt;/li&gt;
&lt;li&gt;Раскрываем сертификат интересующего нас проекта. &lt;b&gt;БУДЬТЕ ВНИМАТЕЛЬНЫ!&lt;/b&gt; Есть две версии сертификатов, &lt;b&gt;Developer&lt;/b&gt; (используется при отладке и разработке) и &lt;b&gt;Production&lt;/b&gt; (используется для рабочей версии). Выбираем сертификат и закрытый ключ данного сертификата.&lt;/li&gt;
&lt;li&gt;Нажимаем правой кнопкой и выбираем пункт «Экспортировать объектов: 2». Сохраняем полученные сертификаты. Я сохранил с именами &lt;i&gt;&lt;b&gt;server_certificates_bundle_sandbox.p12&lt;/b&gt;&lt;/i&gt; (для сертификата разработки) и &lt;i&gt;&lt;b&gt;server_certificates_bundle_prod.p12&lt;/b&gt;&lt;/i&gt; (для сертификата публикации). &lt;b&gt;Внимание!&lt;/b&gt; При экспорте поле пароля оставляем пустым. В дальнейшем вы сможете усложнить систему защиты и выгрузить сертификаты с указанием пароля.&lt;/li&gt;
&lt;/ol&gt;
&lt;p&gt;Для подтверждения валидности сертификатов пользователя которые мы только что получили, необходим корневой сертификат. Для получения корневого сертификата необходимо выполнить следующие действия:&lt;/p&gt;
&lt;ol start="1"&gt;
&lt;li&gt;Переходим на &lt;a href="https://www.entrust.net/downloads/root_index.cfm"&gt;сайт загрузки корневых сертификатов&lt;/a&gt;.&lt;/li&gt;
&lt;li&gt;Выбираем &lt;b&gt;«Personal Use and Secure Server Installation»&lt;/b&gt; и нажимаем кнопку &lt;b&gt;«Download Certificates»&lt;/b&gt;.&lt;/li&gt;
&lt;li&gt;Нажимаем на &lt;b&gt;«Root Certificates»&lt;/b&gt;.&lt;/li&gt;
&lt;li&gt;Нас интересуют два сертификата &lt;b&gt;«entrust_ssl_ca.cer»&lt;/b&gt; и &lt;b&gt;«entrust_2048_ca.cer»&lt;/b&gt;. Скачиваем их. Вообще, в изначальном описании написано, что сертификат «entrust_ssl_ca.cer» используется в &lt;b&gt;Prodution&lt;/b&gt; режиме, а «entrust_2048_ca.cer» в режиме &lt;b&gt;Development&lt;/b&gt;. Но есть пометка, что после 22 декабря 2010 в качестве Production сертификата так же можно использовать «entrust_2048_ca.cer». &lt;b&gt;&lt;i&gt;Т. е. данный сертификат можно использовать в обоих режимах работы, меняя лишь сертификаты пользователя&lt;/i&gt;&lt;/b&gt;.&lt;/li&gt;
&lt;/ol&gt;
&lt;div class="e2-text-picture"&gt;
&lt;div class="fotorama" data-width="1280" data-ratio="1.25"&gt;
&lt;img src="https://alexbirukov.ru/pictures/keychain.png" width="1280" height="1024" alt="Открываем "Связку ключей"" /&gt;
&lt;img src="https://alexbirukov.ru/pictures/keychain-2.png" width="1280" height="1024" alt="Экспортируем сертификаты" /&gt;
&lt;/div&gt;
&lt;/div&gt;
&lt;p&gt;Далее, нам необходимо открыть скачанные сертификаты на Mac’е, т. е. импортировать их в «Связку ключей». Для этого просто делаем двойной клик на каждом сертификате.&lt;/p&gt;
&lt;ol start="1"&gt;
&lt;li&gt;Открываем в левой части &lt;b&gt;«Связки ключей»&lt;/b&gt; раздел &lt;b&gt;«Вход»&lt;/b&gt; («Сертификаты»).&lt;/li&gt;
&lt;li&gt;Находим наш сертификат и экспортируем его &lt;b&gt;«entrust_root_certification_authority_2048.pem»&lt;/b&gt;.&lt;/li&gt;
&lt;/ol&gt;
&lt;div class="e2-text-picture"&gt;
&lt;div class="fotorama" data-width="1280" data-ratio="1.25"&gt;
&lt;img src="https://alexbirukov.ru/pictures/keychain3.png" width="1280" height="1024" alt="Открываем корневой сертификат." /&gt;
&lt;img src="https://alexbirukov.ru/pictures/keychain4.png" width="1280" height="1024" alt="Экспортируем сертификат." /&gt;
&lt;/div&gt;
&lt;/div&gt;
&lt;p&gt;На данный момент, мы получили 3 основных файла для работы с сервисом APNS:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;&lt;b&gt;«entrust_root_certification_authority_2048.pem»&lt;/b&gt; — корневой сертификат.&lt;/li&gt;
&lt;li&gt;&lt;b&gt;«server_certificates_bundle_sandbox.p12»&lt;/b&gt; — сертификат пользователя для режима Development.&lt;/li&gt;
&lt;li&gt;&lt;b&gt;«server_certificates_bundle_prod.p12»&lt;/b&gt; — сертификат пользователя для режима Production.&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;Как можно заметить, сертификаты пользователя имеют расширение *.p12, но для работы нам необходимы файлы формата *.pem.&lt;/p&gt;
&lt;p&gt;Для конвертации я воспользовался бесплатным набором утилит &lt;a href="https://slproweb.com/products/Win32OpenSSL.html"&gt;OpenSSL для Windows&lt;/a&gt;.&lt;br /&gt;
После установки пакета, для простоты выполнения команд я использовал TotalCommander.&lt;/p&gt;
&lt;ol start="1"&gt;
&lt;li&gt;Открываем в левой части каталог установленного OpenSSL и переносим туда наши сертификаты.&lt;/li&gt;
&lt;li&gt;В нижней части TotalCommander в командной строке выполняем следующие команды:&lt;/li&gt;
&lt;/ol&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;openssl pkcs12 -in server_certificates_bundle_sandbox.p12 -out server_certificates_bundle_sandbox.pem -nodes -clcerts
openssl pkcs12 -in server_certificates_bundle_prod.p12 -out server_certificates_bundle_prod.pem -nodes -clcerts&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;На запрос ввода пароля, просто нажимаем Enter.&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;div class="fotorama" data-width="1280" data-ratio="1.25"&gt;
&lt;img src="https://alexbirukov.ru/pictures/convert.png" width="1280" height="1024" alt="Подготавливаем сертификаты и программу." /&gt;
&lt;img src="https://alexbirukov.ru/pictures/convert2.png" width="1280" height="1024" alt="Сертификаты." /&gt;
&lt;img src="https://alexbirukov.ru/pictures/convert4.png" width="1280" height="1024" alt="Выполнение команд конвертации сертификатов." /&gt;
&lt;img src="https://alexbirukov.ru/pictures/convert5.png" width="1280" height="1024" alt="Просто нажимаем Enter." /&gt;
&lt;img src="https://alexbirukov.ru/pictures/convert3.png" width="1280" height="1024" alt="Готовые сертификаты." /&gt;
&lt;/div&gt;
&lt;/div&gt;
&lt;p&gt;И так, теперь у нас есть набор необходимых сертификатов для работы с Push сообщениями на iOS.&lt;/p&gt;
&lt;h2&gt;Google и GCM&lt;/h2&gt;
&lt;p&gt;Для работы с Google и GCM мы будем использовать библиотеку &lt;a href="https://github.com/CodeMonkeysRu/GCMMessage"&gt;GCM PHP Server Library&lt;/a&gt;. Скачанный архив библиотек и пример работы из коробки у меня по какой-то причине отказался работать, поэтому пришлось немного модифицировать код.&lt;/p&gt;
&lt;p&gt;Для работы с GCM нам потребуется API ключ проекта. Выполняем следующие действия:&lt;/p&gt;
&lt;ol start="1"&gt;
&lt;li&gt;Открываем &lt;a href="https://console.developers.google.com/project"&gt;консоль разработчика Google&lt;/a&gt;.&lt;/li&gt;
&lt;li&gt;Выбираем интересующий нас проект.&lt;/li&gt;
&lt;li&gt;Открываем раздел &lt;b&gt;«APIs &amp; auth»&lt;/b&gt; — &lt;b&gt;«Credentials»&lt;/b&gt; и копируем ключ Server Key.&lt;/li&gt;
&lt;li&gt;Если вы ещё не создавали ключи, то вам потребуется включить API «Google Cloud Messaging for Android».&lt;/li&gt;
&lt;/ol&gt;
&lt;div class="e2-text-picture"&gt;
&lt;img src="https://alexbirukov.ru/pictures/apikey.png" width="1280" height="864" alt="API ключ." /&gt;
&lt;/div&gt;
&lt;p&gt;Теперь у нас имеются все необходимые данные для организации PHP сервера рассылки Push сообщений.&lt;/p&gt;
&lt;h2&gt;База данных&lt;/h2&gt;
&lt;p&gt;Для рассылки Push сообщений на устройства пользователей используются токены устройств, которые выдаётся серверами Google и Apple, т. е. для получения Push сообщения, токен устройства должен быть зарегистрирован в системе. Наш сервер так же отправки Push сообщений будет использовать токены устройств. Для хранения токено устройств, ID устройств и платформы я использовал базу данных.&lt;/p&gt;
&lt;p&gt;Текст скрипта для создания базы данных выглядит так.&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;CREATE TABLE devices (
  id int(10) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT 'Уникальный идентификатор  записи в таблице',
  deviceID varchar(255) NOT NULL COMMENT 'Идентификатор устройства',
  deviceToken varchar(255) NOT NULL COMMENT 'Токен устройства',
  devicePlatform varchar(255) NOT NULL COMMENT 'Платформа устройства',
  PRIMARY KEY (id),
  UNIQUE INDEX deviceID (deviceID)
)
ENGINE = MYISAM
AUTO_INCREMENT = 20
AVG_ROW_LENGTH = 166
CHARACTER SET utf8
COLLATE utf8_general_ci
COMMENT = 'Зарегистрированные устройства для рассылки PUSH сообщений';&lt;/code&gt;&lt;/pre&gt;&lt;div class="e2-text-picture"&gt;
&lt;img src="https://alexbirukov.ru/pictures/db-table.png" width="263" height="179" alt="Таблица для хранения токенов." /&gt;
&lt;/div&gt;
&lt;p&gt;Для добавления устройств в базу данных используется отдельная функция скрипта сервера, но об этом чуть позже.&lt;/p&gt;
&lt;h2&gt;Сервер&lt;/h2&gt;
&lt;p&gt;Переходим к самому серверу и его настройке. Все файлы я выложу отдельным архивом.&lt;br /&gt;
Суть скрипта состоит в следующем: подключаем библиотеки для работы с Push, настраиваем сертификаты и API ключ и с помощью имеющихся функций выполняем необходимые действия, будь то рассылка сообщений, регистрация устройств и пр.&lt;/p&gt;
&lt;p&gt;В состав сервера входит файл конфигурации. Приведу отдельно его текст.&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;&amp;lt;?php

# Переменные для работы  с базой данных
$config['db']['host'] = &amp;quot;localhost&amp;quot;;
$config['db']['name'] = &amp;quot;Имя БД&amp;quot;;
$config['db']['user'] = &amp;quot;Пользователь БД&amp;quot;;
$config['db']['pass'] = &amp;quot;Пароль БД&amp;quot;;

# Управление рассылкой. True - рассылка производится, false - рассылка отключена.
$config['apn']['send'] = true;
$config['gcm']['send'] = true;

# Ключи для доступа к сервисам
# Ключ для доступа Google Server API
$config['gcm']['apikey'] = &amp;quot;ВАШ API КЛЮЧ&amp;quot;;
# Путь к сертификатам APN для подключения к серверу рассылки
$config['apn']['sert'] = 'Certificates/server_certificates_bundle_sandbox.pem';
$config['apn']['sert_prod'] = 'Certificates/server_certificates_bundle_prod.pem';
$config['apn']['RootCertificat'] = 'Certificates/entrust_root_certification_authority_2048.pem';
$config['apn']['production'] = false;

?&amp;gt;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Думаю, что раздел конфигурации базы данных не требует объяснения, тут надо указать настройки для доступа к базе данных, в которой хранится таблица &lt;b&gt;«Devices»&lt;/b&gt;.&lt;/p&gt;
&lt;p&gt;Параметры:&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;$config['apn']['send'] = true;
$config['gcm']['send'] = true;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Отвечают за включение или отключение рассылки Push сообщений, если значение false, то рассылка на указанную платформу не производится.&lt;/p&gt;
&lt;p&gt;Дальше идут настройки API ключа и указание каталогов сертификатов, что в общем-то тоже я думаю понятно всем.&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;# Ключи для доступа к сервисам
# Ключ для доступа Google Server API
$config['gcm']['apikey'] = &amp;quot;ВАШ API КЛЮЧ&amp;quot;;
# Путь к сертификатам APN для подключения к серверу рассылки
$config['apn']['sert'] = 'Certificates/server_certificates_bundle_sandbox.pem';
$config['apn']['sert_prod'] = 'Certificates/server_certificates_bundle_prod.pem';
$config['apn']['RootCertificat'] = 'Certificates/entrust_root_certification_authority_2048.pem';&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Отдельно рассмотрим ключ:&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;$config['apn']['production'] = false;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Данный ключ переключает режим работы сервера. Development или Production режимы рассылки.&lt;/p&gt;
&lt;p&gt;Как можно заметить, все сертификаты необходимо положить в каталог «Certificates» рядом со скрипом.&lt;/p&gt;
&lt;p&gt;Если при генерации сертификатов Apn использовался пароль, то в настройках, необходимо указать пароль:&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;$config['apn']['sertPass'] = 'пароль сертификата';&lt;/code&gt;&lt;/pre&gt;&lt;h2&gt;Работа со скриптом&lt;/h2&gt;
&lt;p&gt;После заполнения файла конфигурации скрипт готов к работе, осталось зарегистрировать устройства.&lt;br /&gt;
Скрипт сервера push.php принимает 2 значения параметра &lt;b&gt;action&lt;/b&gt;:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;&lt;b&gt;register-device&lt;/b&gt;&lt;/li&gt;
&lt;li&gt;&lt;b&gt;send-push&lt;/b&gt;&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;В каждом действии требуются дополнительные параметры. Для регистрации устройства дополнительными параметрами являются:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;&lt;b&gt;did&lt;/b&gt; — Идентификатор устройства (не используется, но на всякий случай).&lt;/li&gt;
&lt;li&gt;&lt;b&gt;token&lt;/b&gt; — токен устройства, используется для рассылки Push сообщений.&lt;/li&gt;
&lt;li&gt;&lt;b&gt;platform&lt;/b&gt; — платформа устройства, используется функциями рассылки. Может принимать 2 значения: &lt;b&gt;ios&lt;/b&gt; или &lt;b&gt;andoid&lt;/b&gt;.&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;Для рассылки Push сообщений используется действие send-push и принимается всего 1 дополнительный параметр: &lt;b&gt;text&lt;/b&gt; — текст Push сообщения.&lt;/p&gt;
&lt;p&gt;Примеры запросов:&lt;/p&gt;
&lt;p&gt;Регистрация устройства iOS&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;http://example.ru/push.php?action=register-device&amp;amp;did=DC92EFED-9271&amp;amp;token=feaab9d122f53a4fdca&amp;amp;platform=ios&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Регистрация устройства Android&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;http://example.ru/push.php?action=register-device&amp;amp;did=DC92EFED-9271&amp;amp;token=feaab9d122f53a4fdca&amp;amp;platform=andoid&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Отправка Push сообщений&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;http://example.ru/push.php?action=send-push&amp;amp;text=Добрый день, мой господин!&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Первые два запроса регистрируют устройства в базе данных и в момент рассылки скрипты выбирает токены устройств по полю platform и производит отправку Push сообщений.&lt;/p&gt;
&lt;p&gt;Второй запрос производит рассылку Push сообщения все зарегистрированным устройствам.&lt;/p&gt;
&lt;p&gt;В общем-то, всё достаточно просто, каждый желающий может в дальнейшем модифицировать скрипт по своему желанию, вводя защиту от несанкционированного доступа и пр. Прошу меня не пинать, за полное отсутствие защиты в скрипте, скрипт призван показать логику и рабочий пример работы с Push сообщениями. Плюс данного скрипта состоят в том, что данные мы можем отправлять с любой платформы, будь то веб форма, Windiws приложения или мобильные приложения.&lt;/p&gt;
&lt;h2&gt;Delphi&lt;/h2&gt;
&lt;p&gt;Настало время рассмотреть небольшой код написанный для Delphi, который позволяет регистрировать токены устройств на нашем сервере. Каждое мобильное устройство после запуска приложения получает свой DeviceID и токен, после чего отправляет полученные данные на наш сервер.&lt;/p&gt;
&lt;p&gt;В разделе public нашего юнита добавляем следующие строки:&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;public
    { Public declarations }
    APushService : TPushService;
    AServiceConnection : TPushServiceConnection;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;А в раздел type добавляем описание 2-х функций:&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;procedure OnReceiveNotificationEvent(Sender: TObject; const ANotification : TPushServiceNotification);
    procedure OnServiceConnectionChange(Sender: TObject; AChange : TPushService.TChanges);&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;В момент отображения формы добавляем код получения DID и токена. Будьте внимательны, значение &lt;b&gt;PROJECTNUM&lt;/b&gt; необходимо заменить на свой номер проекта из личного кабинета Google разработчика.&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;/// На данном этапе происходит регистрация устройства для получения Push.
procedure TMainForm.FormShow(Sender: TObject);
var
  ADeviceID, ADeviceToken : string;
begin

  // Получение и отправка токена устройства
  {$ifdef ANDROID}
    // Для Android
    APushService := TPushServiceManager.Instance.GetServiceByName(TPushService.TServiceNames.GCM);
    APushService.AppProps[ TPushService.TAppPropNames.GCMAppID ] := 'PROJECTNUM';
  {$else}
    // Для iOS
    APushService := TPushServiceManager.Instance.GetServiceByName(TPushService.TServiceNames.APS);
  {$endif}
  // Создаём подключение к серверу
  AServiceConnection := TPushServiceConnection.Create( APushService );
  // Активируем подключение
  AServiceConnection.Active   := True;
  // Подключаем делегаты
  AServiceConnection.OnChange := OnServiceConnectionChange;
  AServiceConnection.OnReceiveNotification := OnReceiveNotificationEvent;

  ADeviceID    := APushService.DeviceIDValue[ TPushService.TDeviceIDNames.DeviceID ];
  ADeviceToken := APushService.DeviceTokenValue[ TPushService.TDeviceTokenNames.DeviceToken ];

  if (ADeviceID &amp;lt;&amp;gt; '') AND (ADeviceToken &amp;lt;&amp;gt; '') then
  begin

    // Регистрируем устройство на сервере для отправки push сообщений
    RegisterDevice(ADeviceID, ADeviceToken);

  end;

end;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Так же, надо добавить функцию, которая срабатывает при изменении состояния получения DID и токена.&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;/// Устройства не всегда успевают получить токен,
/// поэтому при изменении состояния опять проверяем токен
procedure TMainForm.OnServiceConnectionChange(Sender: TObject;
  AChange : TPushService.TChanges);
var
  ADeviceID, ADeviceToken : string;
begin
    // При измении состояния компонента
    ADeviceID    := APushService.DeviceIDValue[ TPushService.TDeviceIDNames.DeviceID ];
    ADeviceToken := APushService.DeviceTokenValue[ TPushService.TDeviceTokenNames.DeviceToken ];

    if (ADeviceID &amp;lt;&amp;gt; '') AND (ADeviceToken &amp;lt;&amp;gt; '') then
    begin

      // Регистрируем устройство на сервере для отправки push сообщений
      RegisterDevice(ADeviceID, ADeviceToken);

    end;
end;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Для вывода Push сообщения добавляем функцию:&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;/// Процедура вывода сообщение при получении Push уведомления от сервера
procedure TMainForm.OnReceiveNotificationEvent(Sender: TObject;
  const ANotification : TPushServiceNotification);
var
  MessageText : string;
begin
  // Получаем текст сообщения в зависимости ль платформы
  {$ifdef ANDROID}
    MessageText := ANotification.DataObject.GetValue('message').Value;
  {$else}
    MessageText := ANotification.DataObject.GetValue('alert').Value;
  {$endif};

  // Выводим сообщение
  ShowNotification(MessageText, 0);
end;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;В результате мы получаем почти готовую программу. Кто был более внимательным, заметил, что в коде есть несколько функций, не объявленных ранее, таких как &lt;b&gt;ShowNotification&lt;/b&gt; или &lt;b&gt;RegisterDevice&lt;/b&gt;. Для удобства, я вынес их в отдельный файл global.pas. Достаточно добавить данный файл в проект, чтобы не копипастить функции. В данном файле вам потребуется заменить константу DOMAIN для указания пути к файлу скрипта на сервере, и если вы положили скрипт не в корне сервера, а в случайный каталог, то потребуется внести дополнительные изменения для указания точной адресации.&lt;/p&gt;
&lt;p&gt;Так же имеется функция &lt;b&gt;SendPush&lt;/b&gt;, которая позволяет отправлять Push сообщения прямо с устройства.&lt;/p&gt;
&lt;p&gt;Все функции в файле global.pas имеют подробное описание и думаю их использование не вызовет никаких вопросов.&lt;/p&gt;
&lt;p&gt;Таким образом мы получаем устройства, которые регистрируются в системе, отправляя свои DID и токены скрипту, а скрипт записывает их в базу данных для дальнейшего использования при отправке Push сообщений. Если в базе имеются токены устройств, на которых программа не установлен (удалена), то такие токены будут удалены из базы данных после попытки отправки Push сообщений.&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;img src="https://alexbirukov.ru/pictures/db.png" width="751" height="128" alt="Зарегистрированные устройства." /&gt;
&lt;/div&gt;
&lt;h2&gt;Используемые файлы.&lt;/h2&gt;
&lt;p&gt;&lt;a href="http://alexbirukov.ru/files/push-server/global.pas"&gt;Файл global.pas&lt;/a&gt;&lt;br /&gt;
&lt;a href="http://alexbirukov.ru/files/push-server/Push_Server.7z"&gt;PHP сервер Push сообщений&lt;/a&gt;&lt;/p&gt;
&lt;h2&gt;Заключение&lt;/h2&gt;
&lt;p&gt;В данной статье я попытался максимально подробно изложить принципы и алгоритмы работы с Push сообщениями с использованием дополнительных библиотек GCMMessage и ApnsPHP в php скрипте и взаимодействие с программным обеспечением (в моём случае Windows приложения и мобильные платформы).&lt;/p&gt;
&lt;p&gt;Данная статья не претендует на оригинальность и истину последней инстанции. Имеются открытые места в безопасности, такие, как сертификаты без паролей, открытый доступ к функциям скрипта без идентификации пользователей и пр.&lt;/p&gt;
&lt;p&gt;Если у кого-нибудь будет желание доработать данный скрипт в плане безопасности или внести поправки я буду только рад. Ваши изменения и замечания вы можете писать в комментариях к статье.&lt;/p&gt;
&lt;p&gt;&lt;b&gt;UPDATE&lt;/b&gt;&lt;/p&gt;
&lt;p&gt;Для удобства, загрузил проект на &lt;a href="https://github.com/alexbirukov/PHP-Push-Sender"&gt;GitHub&lt;/a&gt;.&lt;/p&gt;
&lt;p&gt;&lt;b&gt;UPDATE 2&lt;/b&gt; из комментария читателя&lt;/p&gt;
&lt;blockquote&gt;
&lt;p&gt;Если у кого не работает на Rad Studio 10.2, то вместо «AServiceConnection.Active := True;» сделайте так:&lt;/p&gt;
&lt;/blockquote&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;// Активируем подключение
TTask.Run(
procedure
begin
AServiceConnection.Active := True;
end
);&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;В uses должно быть System.Threading.&lt;/p&gt;
</description>
</item>

<item>
<title>Аудио-плеер на Delphi для Android с применением библиотеки BASS.</title>
<guid isPermaLink="false">316</guid>
<link>https://alexbirukov.ru/?go=all/udio-pleer-na-delphi-dlya-android-code-gear-rad-studio-xe7/</link>
<pubDate>Tue, 24 Mar 2015 09:48:04 +0500</pubDate>
<author></author>
<comments>https://alexbirukov.ru/?go=all/udio-pleer-na-delphi-dlya-android-code-gear-rad-studio-xe7/</comments>
<description>
&lt;p&gt;В середине прошлого года появилась необходимость написания простого приложения для мобильных платформ Android и iOS. Первым рабочим проектом стал &lt;a href="http://alexbirukov.ru/?go=tags/kteam/"&gt;KTeam&lt;/a&gt;. После его успешного запуска, появилась необходимость написания аудио-плеера, способного воспроизводить потоковое вещание интернет радиостанций.&lt;/p&gt;
&lt;p&gt;И так, сегодня я подробно опишу, как создать простой плеер потокового вещания.&lt;/p&gt;
&lt;p&gt;Для работы нам понадобятся среда программирования Code Gear RAD Studio XE7, работать я рекомендую именно в данной версии программы, т. к. в используемых библиотеках, имеются некоторые не стыковки в работе с предыдущими версиями.&lt;/p&gt;
&lt;p&gt;По причине того, что встроенных средств проигрывания интернет потока не имеется, мы воспользуемся замечательной библиотекой BASS, для написания приложения для Android (для iOS следует использовать другие компоненты).&lt;/p&gt;
&lt;p&gt;Для корректной работы библиотеки BASS я рекомендую так же установить компоненты &lt;a href="http://sourceforge.net/projects/dpfdelphiandroid/"&gt;DPF Android Native Components&lt;/a&gt;, я в своём примере буду использовать версию 2.8.6.&lt;/p&gt;
&lt;h2&gt;Начало&lt;/h2&gt;
&lt;p&gt;Ну я думаю процесс установки Code Gear RAD Studio XE7 описывать не стоит, поэтому сразу переходим к установке дополнительных компонентов и библиотек, потому что именно эта часть вызывает наибольшие проблемы, и именно поэтому данный этап я опишу наиболее подробно.&lt;/p&gt;
&lt;h2&gt;BASS&lt;/h2&gt;
&lt;p&gt;Идём на &lt;a href="http://www.un4seen.com"&gt;официальный сайт un4seen.com&lt;/a&gt; библиотеки, нажимаем в левом меню на ссылку &lt;a href="http://www.un4seen.com/bass.html"&gt;BASS&lt;/a&gt; и скачиваем там последнюю версию для &lt;a href="http://www.un4seen.com/forum/?topic=13225"&gt;Android&lt;/a&gt;. На момент написания данной статьи на сайте опубликована версия 2.4 от 13 марта 2015 года. &lt;a href="http://www.un4seen.com/stuff/bass24-android.zip"&gt;Прямая ссылка&lt;/a&gt; на скачивание библиотеки с официального сайта. &lt;a href="http://alexbirukov.ru/files/radio-andr/bass24-android.zip"&gt;Ссылка на скачивание с моего сайта&lt;/a&gt;.&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;div class="fotorama" data-width="1276" data-ratio="1.3632478632479"&gt;
&lt;img src="https://alexbirukov.ru/pictures/-2015-03-24-09.25.46.png" width="1276" height="936" alt="Сайт библиотеки BASS" /&gt;
&lt;img src="https://alexbirukov.ru/pictures/-2015-03-24-09.29.03.png" width="1274" height="959" alt="Ссылки на скачивание библиотек" /&gt;
&lt;/div&gt;
&lt;/div&gt;
&lt;p&gt;&lt;i&gt;&lt;b&gt;Примечание:&lt;/b&gt; я рекомендую использовать именно данную версию библиотеки, потому что предыдущие версии не работают с XE7, и при инициализации библиотеки появляется ошибка.&lt;/i&gt;&lt;/p&gt;
&lt;h2&gt;BASSFunctions&lt;/h2&gt;
&lt;p&gt;Для корректной работы с библиотекой, нам потребуется ещё один дополнительный файл, это &lt;a href="http://alexbirukov.ru/files/radio-andr/BASSFunctions.pas"&gt;BASSFunctions.pas&lt;/a&gt;. Данный файл опубликовал на форуме пользователь с ником 3delite.&lt;a href="http://www.3delite.hu/Object%20Pascal%20Developer%20Resources/Download/BASSFunctions.pas"&gt;Ссылка&lt;/a&gt; на оригинальный файл.&lt;/p&gt;
&lt;h2&gt;DPF Android Native Components&lt;/h2&gt;
&lt;p&gt;Для корректной работы библиотек BASS, часто требуется установка компонентов DPF Android Native Components, т. к. в их составе имеется компонент для работы с аудио, без которого компилятор будет просто выдавать ошибку.&lt;/p&gt;
&lt;p&gt;Идём на &lt;a href="http://sourceforge.net/projects/dpfdelphiandroid/"&gt;официальный сайт&lt;/a&gt; и скачиваем последнюю версию компонентов. Или вы можете скачать версию &lt;a href="http://alexbirukov.ru/files/radio-andr/DPF.Android.Native.Components.v2.8.6.zip"&gt;2.8.6&lt;/a&gt; с моего сайта.&lt;/p&gt;
&lt;h2&gt;Поехали&lt;/h2&gt;
&lt;p&gt;После того, как мы скачали все необходимые дополнительные файлы, приступаем к установке компонентов.&lt;/p&gt;
&lt;p&gt;Начнём с DPF Android Native Components. Я рекомендую (во избежании дополнительных проблем с подключением) распаковать файлы в папку DPF.Android.Native.Components.v2.8.6 в корень диска C.&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-09-55-10.png" width="1280" height="1024" alt="Распаковываем файлы" /&gt;
&lt;/div&gt;
&lt;p&gt;В каталоге с файлами, имеется файл Install.txt. У кого хорошо с английским, могут сделать всё по инструкции описанной в файле, для тех кто в школе учил немецкий, я опишу процесс установки компонентов подробно.&lt;/p&gt;
&lt;p&gt;Запускаем Codegear RAD Studio. Далее в верхнем меню открываем Tools-&gt;Options-&gt;Delphi Options-&gt;Library.&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;div class="fotorama" data-width="1280" data-ratio="1.25"&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-09-58-33.png" width="1280" height="1024" alt="" /&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-10-00-13.png" width="1280" height="1024" alt="" /&gt;
&lt;/div&gt;
&lt;/div&gt;
&lt;p&gt;В выпадающем списке справа выбираем платформу «32-bit Windows» и добавляем в раздел «Library path» путь к нашим компонентам.&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-10-04-06.png" width="1280" height="1024" alt="" /&gt;
&lt;/div&gt;
&lt;p&gt;Теперь то же самое проделываем для платформы «Android».&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-10-04-55.png" width="1280" height="1024" alt="" /&gt;
&lt;/div&gt;
&lt;p&gt;Теперь нам потребуется скомпилировать и установить компоненты. Для этого нажимаем File -&gt; Open Project. Открываем проект из папки компонентов, соответствующий вашей среде программирования.&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;div class="fotorama" data-width="1280" data-ratio="1.25"&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-10-06-17.png" width="1280" height="1024" alt="Открываем проект" /&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-10-06-31.png" width="1280" height="1024" alt="DPFAndroidPackagesXE7.dproj" /&gt;
&lt;/div&gt;
&lt;/div&gt;
&lt;p&gt;Далее нам надо скомпилировать проект и установить полученные компоненты.&lt;/p&gt;
&lt;p&gt;Нажимаем правой кнопкой на проекте в инспекторе проекта и выбираем пункт &lt;b&gt;«Build»&lt;/b&gt;, начнётся построение проекта.&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-10-13-23.png" width="1280" height="1024" alt="Выполняем Build" /&gt;
&lt;/div&gt;
&lt;p&gt;После завершение построение, опять нажимаем правой кнопкой на проекте в инспекторе и выбираем пункт &lt;b&gt;«Install»&lt;/b&gt;.&lt;/p&gt;
&lt;p&gt;Компоненты готовы к работе. Приступим к созданию проекта.&lt;/p&gt;
&lt;p&gt;Я набросал уже небольшой проект с парой кнопок для запуска и остановки проигрывателя и картинкой для привлечения внимания :)&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-10-24-28.png" width="1280" height="1024" alt="Пробный проект." /&gt;
&lt;/div&gt;
&lt;p&gt;Для работы в библиотекой BASS, нам потребуется в каталог с проектом добавить файл BASSFunctions.pas и указать его в разделе &lt;b&gt;uses&lt;/b&gt;.&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;div class="fotorama" data-width="1280" data-ratio="1.25"&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-10-21-50.png" width="1280" height="1024" alt="Добавляем файл BASSFunctions.pas." /&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-10-22-52.png" width="1280" height="1024" alt="Добавляем в раздел uses." /&gt;
&lt;/div&gt;
&lt;/div&gt;
&lt;p&gt;Теперь собственно сама библиотека &lt;b&gt;libbass.so&lt;/b&gt;. Нам необходимо добавить данную библиотек в наш проект, иначе ничего не заработает.&lt;/p&gt;
&lt;p&gt;Открываем архив bass24-android.zip и видим... Множество папок, нас интересует на данный момент только одна &lt;b&gt;libs&lt;/b&gt;. Распаковываем её в каталог нашего проекта.&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;img src="https://alexbirukov.ru/pictures/-2015-03-24-10.29.21.png" width="1273" height="993" alt="" /&gt;
&lt;/div&gt;
&lt;p&gt;В каталоге libs, есть три подкаталога, нас интересуют 2 из них:&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;armeabi-v7a&lt;/li&gt;
&lt;li&gt;armeabi&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;В них находятся два разных файла libbass.so.&lt;/p&gt;
&lt;p&gt;После того, как мы скопировали данный каталог в каталог проекта, снова возвращаемся к среде программирования и открываем раздел &lt;b&gt;Deployment&lt;/b&gt;, для добавления наших библиотек в программу.&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;div class="fotorama" data-width="1280" data-ratio="1.25"&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-10-36-12.png" width="1280" height="1024" alt="Deployment" /&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-10-37-49.png" width="1280" height="1024" alt="Добавляем библиотеки" /&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-10-38-56.png" width="1280" height="1024" alt="" /&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-10-39-15.png" width="1280" height="1024" alt="" /&gt;
&lt;/div&gt;
&lt;/div&gt;
&lt;p&gt;После того, как мы добавили библиотеки в наш проект, надо поменять параметр «Remote Path» для каждой библиотеки соответственно на «&lt;b&gt;library\lib\armeabi&lt;/b&gt;» и на «&lt;b&gt;library\lib\armeabi-v7a&lt;/b&gt;». Данную процедуру надо проделать для обеих конфигураций «Debug» и «Release».&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;div class="fotorama" data-width="1280" data-ratio="1.25"&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-10-43-47.png" width="1280" height="1024" alt="Библиотеки" /&gt;
&lt;img src="https://alexbirukov.ru/pictures/Windows-7-x64-Delphi-XE7-2015-03-24-10-42-46.png" width="1280" height="1024" alt="Конфигурации "Debug" и "Release"" /&gt;
&lt;/div&gt;
&lt;/div&gt;
&lt;p&gt;Ну что, вот всё и готово к началу написания кода, а код на самом деле достаточно простой.&lt;/p&gt;
&lt;p&gt;Объявляем глобальную переменную.&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;var
  uStream:HSTREAM;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;В момент создания формы инициализируем библиотеку.&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;procedure TmainForm.FormCreate(Sender: TObject);
begin
  // Загружаем и инициализируем библиотеку
  Loadbass;
  BASS_Init(-1,44100,0,Handle,nil);
end;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;На кнопку «Играть» вещаем обработчик:&lt;/p&gt;
&lt;pre class="e2-text-code"&gt;&lt;code class=""&gt;procedure TmainForm.mainToolbarBarItems1Click(Sender: TObject);
var
  otStream : TStringStream;
  Result : integer;
begin
  try

    // Включаем индикатор загрузки контента
    ind1.Enabled := True;
    ind1.Visible := True;

    // Код проигрывателя
    Result := 0;

    // Закрываем старый поток
    BASS_StreamFree(uStream);

    // Создаём новый поток
    uStream :=
      BASS_StreamCreateURL(PChar('http://motochelyabinsk.ru:8000/play'),0,BASS_STREAM_STATUS or BASS_UNICODE,nil,nil);

    // Включаем проигрывание
    BASS_ChannelPlay(uStream, True);

    // Получаем и выводим название текущего трека из скрипта на сайте
    // Создаём поток UTF8
    otStream := TStringStream.Create('', CP_UTF8);
    // Делаем запрос к скрипту сайта
    IdHTTP1.Get('motochelyabinsk.ru/current-track.php', otStream);
    IdHTTP1.Disconnect;
    // Записываем название трека в Label
    Label1.Text := otStream.DataString;
    // Очищаем переменную потока
    otStream.Free;

    // Запустить таймер отслеживания изменений названия трека
    Timer1.Enabled := True;

    // Включаем и отключаем кнопки
    btn1.Enabled := False;
    btn2.Enabled := True;

  except on E: Exception do
    begin
      ShowMessage('Ошибка доступа к сети!');
    end;
  end;

end;&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;Ну вот в общем самый простой и работающий пример создания проигрывателя интернет радиостанций.&lt;br /&gt;
P.S. Чуть позже, выложу готовый пример данного приложения.&lt;/p&gt;
</description>
</item>

<item>
<title>Мотосообщество и новые проекты</title>
<guid isPermaLink="false">310</guid>
<link>https://alexbirukov.ru/?go=all/motosoobschestvo-i-novye-proekty/</link>
<pubDate>Sun, 22 Jun 2014 20:20:29 +0500</pubDate>
<author></author>
<comments>https://alexbirukov.ru/?go=all/motosoobschestvo-i-novye-proekty/</comments>
<description>
&lt;p&gt;Как-то давно я не писал ничего в блог, видимо потому, что стало значительно меньше свободного времени и меньше времени для размышлений. Но тем не менее жизнь идёт своим чередом и за период времени с середины прошедшей зимы мы с группой единомышленников создали проект взамипомомощи среди мотоциклистов и назвали проект KTeam (сокращение от &lt;a href="http://vk.com/motochel74"&gt;Кировка Тим&lt;/a&gt;).&lt;/p&gt;
&lt;p&gt;Сначала мы занялись проектированием платформы и &lt;a href="http://motochelyabinsk.ru/"&gt;сайтом&lt;/a&gt; для работы всей системы. Для начала мы сделали простенький сайт, целью которого на начальном этапе было — отображение заявок. Так же параллельно сайту появилась платформа для работы с поступающими заявками от различных мобильных платформ и СМС операторов. Изначально мы расчитывали на входящие СМС сообщения, но затем решили расширить проект и написать приложения для платформ iOS и Android. И хочется заметить, у нас всё получилось.&lt;/p&gt;
&lt;p&gt;И так 17 мая 2014 на открытии мотосезона состоялась презентация нашего небольшого, но крайне полезного проекта.&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;img src="https://alexbirukov.ru/pictures/4ld350kahse.jpg" width="573" height="807" alt="Приложение KTeam" /&gt;
&lt;/div&gt;
&lt;p&gt;Суть приложения и всей платформы проста — создать автоматизированную систему оповещения о происходящих происшествиях с участием мотоциклистов. Приложение примечательно тем, что может автоматически определять координаты местоположения аварии и создавать на их основании заявку. После чего все пользователи зарегистрированные в системе получают СМС уведомления с типом и адресом места происшествия.&lt;/p&gt;
&lt;p&gt;С момента запуска проекта прошло чуть больше месяца, но система оповещение уже успела зарекомендовать себя, как очень действенный инструмент моментального оповещения участников мотодвижения в городе Челябинске.&lt;/p&gt;
&lt;p&gt;Так же, с приходом тепла мы стали заниматься не только разработкой приложения и сайта, но и продумывать и организовывать различные социальные мероприятия. Недавно состоялась поездка в детский приют в пос. Солнечный. Хотелось бы сказать огромное СПАСИБО всем тем, кто откликнулся и помог финансово, морально и физически организовать столь замечательное мероприятие.&lt;/p&gt;
&lt;div class="e2-text-picture"&gt;
&lt;img src="https://alexbirukov.ru/pictures/logo_KT.png" width="869" height="870" alt="KTeam. Кировка. Челябинск. Мотосообщество." /&gt;
&lt;/div&gt;
&lt;p&gt;P.S. Ждите новых мероприятий и принимайте активное участие!&lt;/p&gt;
</description>
</item>


</channel>
</rss>