Как стать автором
Обновить

Простой USSD-запрос в Android 4.0+

Время на прочтение3 мин
Количество просмотров38K
В Android до сих пор нет API для USSD-запросов. Баг висит уже 6 лет!
Я находил разные способы создания и получения информации из USSD запросов, но в итоге ни один не устроил.
Затем я нашел упоминания о том, что с помощью обновленных в Android 4.0 служб спец. возможностей можно легко получать содержимое окон и так получить текст из окна и результатом USSD запроса. Попробовал — получается отлично! Без перезагрузок и надежно.

Отправка запроса

Для начала, надо отослать сам USSD-запрос. Это достаточно просто:

String encodedHash = Uri.encode("#");
String ussd = "*100" + encodedHash;
startActivityForResult(new Intent("android.intent.action.CALL",
Uri.parse("tel:" + ussd)), 1);

Для получения разрешение на работу с телефоном, надо получить разрешение, для этого прописываем в Android.Manifest.xml:

<uses-permission android:name="android.permission.CALL_PHONE" />

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

Accessibility service

Для работы со спец. возможностями надо экспортировать наш сервис. Для этого добавляем в Android.Manifest.xml:

<service 
      android:name=".USSDService"
      android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE" >
	<intent-filter>
   	<action android:name="android.accessibilityservice.AccessibilityService" />
	</intent-filter>
</service>

Вместо USSDService напишите название вашего класса.

Сам класс я брал почти без изменений отсюда.

Для того, чтобы не получать лишнего, меняем метод onServiceConnected:

protected void onServiceConnected() {
    super.onServiceConnected();
    Log.v(TAG, "onServiceConnected");
    AccessibilityServiceInfo info = new AccessibilityServiceInfo();
    info.flags = AccessibilityServiceInfo.DEFAULT;
    info.packageNames = new String[]
            {"com.android.phone"};
    info.eventTypes = AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED;
    info.feedbackType = AccessibilityServiceInfo.FEEDBACK_GENERIC;
    setServiceInfo(info);
}

Еще нужен будет дополнительный фильтр в методе onAccessibilityEvent:

public void onAccessibilityEvent(AccessibilityEvent event) {
    String text = getEventText(event);
    Log.v(TAG, String.format(
            "onAccessibilityEvent: [type] %s [class] %s [package] %s [time] %s [text] %s",
            getEventType(event), event.getClassName(), event.getPackageName(),
            event.getEventTime(), getEventText(event)));
    if (event.getClassName().equals("android.app.AlertDialog")) {
		performGlobalAction(GLOBAL_ACTION_BACK);
    		Log.i(TAG, text);
		Intent intent = new Intent("REFRESH");
		intent.putExtra("message", text);
		sendBroadcast(intent);
    }
}

Метод performGlobalAction(GLOBAL_ACTION_BACK) требует Android 4.1+, если его не использовать, можно уложиться в 4.0. Он закрывает окошко сразу же после появления AlertDialog, поэтому окно не останется висеть.
Для простоты я уже добавил метод sendBroadcast, чтобы отослать полученное сообщение дальше.
Чтобы получить сообщение, добавляем в нужный класс такие методы:

private BroadcastReceiver mMessageReceiver = new BroadcastReceiver() {
  @Override
  public void onReceive(Context context, Intent intent) {
    String message = intent.getStringExtra("message");
    Log.i("receiver", "Got message: " + message);
    showText(message);
  }
};

showText — ваша процедура, которая что-то делает с полученным текcтом.

Для работы BroadcastReceiver'а надо добавить немного кода в onCreate() или подобный метод вашего класса:

IntentFilter mFilter = new IntentFilter("REFRESH");
mContext.registerReceiver(mMessageReceiver, mFilter);
isRegistered = true;

И это — в onPause() или подобный:

try
{
    if (isRegistered) {
        mContext.unregisterReceiver(mMessageReceiver);
        isRegistered = false;
    }
}
catch (Exception e) {
    e.printStackTrace();
}

Вот и все! Как видите, просто и надежно.

Подводные камни

Могут быть проблемы с работой в фоне: например, если экран заблокирован, то AccessibilityService вообще не будет работать — надо будить устройство. В разблокированном же состоянии запрос всегда будет вылазить на первый план, что то же не удобно.

AccessibilityService будет слушать постоянно, даже когда пользователь сам набирает USSD код или, еще хуже, если com.android.phone кидает AlertDialog. Так что надо либо усиливать фильтры (например парсить только если в сообщении есть определенная последовательность), либо использовать флаг, чтобы обрабатывать события только если ваше приложение сделало USSD-запрос.

И не забудьте в настройках спец. возможностей активировать ваше приложение, кстати для этого тоже неплохо было бы добавить проверку %)

Альтернативы:
habrahabr.ru/post/130717 — берет ответ из логов. Неудобно тем, что для подавления окна надо прибегать к хакам, а сделать это через AccessibilityService чище и проще.
github.com/alaasalman/ussdinterceptor — еще старый способ. Использует недокументированный класс. По отзывам часто отваливается и требуется перезагрузка.
Теги:
Хабы:
Всего голосов 61: ↑54 и ↓7+47
Комментарии11

Публикации

Истории

Работа

Java разработчик
239 вакансий

Ближайшие события

27 января
Deckhouse Conf 2025
Москва
25 – 26 апреля
IT-конференция Merge Tatarstan 2025
Казань