29 January 2015

Pebble: пример использования Android-компаньона

Programming
Tutorial
Благодаря официальному мобильному приложению Pebble отлично справляются с информированием о состоянии вашего смартфона — показывают входящие сообщения, информацию о вызовах и прочие уведомления. Но что делать, если такая нужная «мелочь», как например состояние батареи смартфона, количество непрочитанных SMS и e-mail, недоступна для использования в своих приложениях на часах? Вариант, реализовать это самому.

И так, о том как использовать PebbleKit Android для интеграции Pebble и Android-приложения на примере уведомления о количестве пропущенных вызовов: немного кода, перевод выдержек из документации [1] и совсем мало картинок.


Тезисно, как работает взаимодействие мобильного приложения и watchappp:
  • приложение-компаньон распознает, соединяется и обменивается информацией с Pebble через библиотеку PebbleKit;
  • watchapp (watchface) и мобильное приложение обмениваются сообщениями в двустороннем режиме, идентификация канала с обоих сторон осуществляется по UUID приложения Pebble;
  • обмен между Pebble-приложением и мобильным приложением осуществляется объектами Dictionary [2].

Для решения поставленной задачи, вывода уведомления на часах о количестве пропущенных вызовов, необходимо:
  • Pebble-приложение, которое умеет получать, обрабатывать и отображать информацию из приложения-компаньона;
  • Android-приложение, которое отслеживает вызовы и отсылает количество неотвеченных в наше приложение на часах.


Watchapp



В Pebble-приложении, точно так же как и в случае с PebbleKit JS, используется механизм AppMessage API [3].
Так как наше приложении само не инициирует сеанс связи, а только ждет сообщение со стороны смартфона, я буду использовать Synchronizing App UI [4] — вспомогательный слой поверх AppMessage, который упрощает синхронизацию значений между watchapp и мобильным устройством. Используется только два обратных вызова, один из которых в случае когда изменяется предопределенное значение, и другой, если происходит ошибка.

Определяем ключ, AppSync и буфер для синхронизации кортежа:
#define KEY_CALLS_COUNT 41

static AppSync s_sync;
static uint8_t s_sync_buffer[32];

Инициализацию механизама синхронизации и начального значения кортежа вынесем в отдельную функцию:
/* ... */
static void start_sync() {
  app_message_open(app_message_inbox_size_maximum(),
                   app_message_outbox_size_maximum());

  // Начальная инициализация кортежа (0 неотвеченных)
  Tuplet initial_values[] = {
    TupletInteger(KEY_CALLS_COUNT, 0),
  };

  app_sync_init(&s_sync, s_sync_buffer, sizeof(s_sync_buffer),
                initial_values, ARRAY_LENGTH(initial_values),
                sync_changed_handler, sync_error_handler, NULL);
}

static void init(void) {
/* ... */

  start_sync();
}

Определяем обратные вызовы:
/* ... */
static char s_count_buffer[4];

/* ... */
static void sync_changed_handler(const uint32_t key, const Tuple *new_tuple,
                                 const Tuple *old_tuple, void *context) {

  // Преобразовываем полученное значение в строку
  snprintf(s_count_buffer, sizeof(s_count_buffer),
           "%d", (int)new_tuple->value->int32);

  // Помечаем слой с индикатором для перерисовки
  layer_mark_dirty(layer);
}

static void sync_error_handler(DictionaryResult dict_error,
                               AppMessageResult app_message_error,
                               void *context) {
  APP_LOG(APP_LOG_LEVEL_ERROR, "sync error!");
}
/* ... */

N.B. В данном случае, ключ используемый для обмена значением не требует регистрации в appinfo.json.

Если сейчас запустить наше простое приложение, то на экране будет одинокий индикатор (0 пропущенных вызовов):


Android-приложение



Получить количество пропущенных вызовов в Android можно разными способами, в данном примере я буду брать её из «шторки» с уведомлениями — Notification Area.
Само приложение состоит из двух классов: MainActivity для ввода UUID watchapp'а которое будет принимать данные и сервиса NotificationListener, который отслеживает уведомления.

N.B. Доступ к Notification Area через NotificationListenerService появился в Android 4.3. Так как используются дополнительные метаданные Notification.extras, пример будет гарантировано работать только на Android 4.4+.

Для Android Studio добавление PebbleKit осуществляется через gradle-файл.
Добавляем в app/build.gradle:
dependencies {
    compile 'com.getpebble:pebblekit:2.6.0'
}

repositories {
    mavenCentral()
    maven { url "https://oss.sonatype.org/content/groups/public/" }
}

После синхронизации PebbleKit можно использовать в своем проекте.

Для доступа к уведомлениям использую класс NotificationListener, унаследованный от NotificationListenerService с переопределенными методами onNotificationPosted() и onNotificationRemoved():
public class NotificationListener extends
        NotificationListenerService {

    @Override
    public void onNotificationPosted(
            StatusBarNotification sbn) {
        this.getMissedCalls();
    }

    @Override
    public void onNotificationRemoved(
            StatusBarNotification sbn) {
        this.getMissedCalls();
    }
}

При добавлении или удалении уведомления, просматриваются все активные и если присутствует информация о пропущенных вызовах, количество их посылается в приложение на часах:
public class NotificationListener extends
        NotificationListenerService {


    // Счетчик пропущенных вызовов
    int missedCallsCount = 0;

    /*...*/

    void getMissedCalls() {
        int tCount = 0;

        for (StatusBarNotification notif :
                this.getActiveNotifications()) {

            if (notif.getPackageName().equals("com.android.phone")) {
                String extras_text = notif.getNotification().extras.getString(Notification.EXTRA_TEXT);
                if (extras_text.indexOf("Пропущенных вызовов:") != -1) {
                    tCount = Integer.parseInt(extras_text.split(":")[1].trim());
                }
            }

        }

        if (tCount != missedCallsCount) {
            missedCallsCount = tCount;
            this.sendMissedCalls(missedCallsCount);
        }
    }
}

Для отправки данных на часы определяем ключ, UUID, формируем словарь и используем метод sendDataToPebble:
public class NotificationListener extends
        NotificationListenerService {

    UUID APPS_UUID;
    private static final int CALLS_KEY = 41;

 /*...*/
    public void sendMissedCalls(int missedCalls) {
        APPS_UUID = UUID.fromString(this.getUUID());

        PebbleDictionary data = new PebbleDictionary();
        data.addUint32(CALLS_KEY, missedCalls);

        PebbleKit.sendDataToPebble(getApplicationContext(), APPS_UUID, data);
    }
 /*...*/
}


Для предоставления приложению прав доступа к уведомлениям необходимо добавить в манифест:
        <service android:name=".NotificationListener"
            android:label="@string/app_name"
            android:permission=
                "android.permission.BIND_NOTIFICATION_LISTENER_SERVICE">
            <intent-filter>
                <action android:name=
                    "android.service.notification.NotificationListenerService" />
            </intent-filter>
        </service>


Замечание, после установки приложения необходимо разрешить сервис, без авторизаци работать не будет. Настройка находится по пути «Settings» -> «Security» -> «Notification access». В русской локали у меня «Настройки» -> «Конфиденциальность» -> «Доступ к уведомлениям».


Запускаем android-приложение, указываем в нем UUID Pebble-приложения, звоним и не отвечаем, смотрим как меняется уведомление на часах:


Исходники:
notice watchapppbw
PebbleNotify (проект Android Studio)apk

1. Pebble Developers // Mobile App Developer Guide
2. Pebble Developers // Dictionary
3. Pebble Developers // App Communication
4. Pebble Developers // Synchronizing App UI
Tags:pebblepebble watchtutorialandroid appsandroid
Hubs: Programming
+4
4.2k 24
Leave a comment
Popular right now
Android
from 150,000 ₽NatsONМоскваRemote job
Android-разработчик
from 80,000 ₽FlowwowRemote job
Android-разработчик
to 80,000 ₽AmigowebМагнитогорскRemote job
Android-разработчик
from 170,000 to 230,000 ₽ENJOY PROСанкт-ПетербургRemote job
Android Developer
from 150,000 to 200,000 ₽/difway.studioRemote job