Подтвердить что ты не робот

Новый способ управления диалоговыми окнами USSD в android

Я читал темы о диалогах USSD здесь и других форумах в течение нескольких дней. (по USSD я имею в виду уведомления оператора со сведениями о стоимости вызовов в нем).

Я видел много решений, которые, по-видимому, работали для более низких уровней API в android, но после многих тестов выяснили, что они больше не работают или, по крайней мере, я не мог заставить их работать.

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

но все, что я мог получить из журналов, было каким-то образом связано с моим приложением, хотя я мог видеть их в Eclipse LogCat, но в моем приложении не было записей, связанных с MMI. возможно, он работал в более низких версиях Android, чем мой (4.2.2).

тогда я решил использовать "IExtendedNetworkService", как он использовался в следующих ссылках и многих других:

https://github.com/alaasalman/ussdinterceptor

Использование IExtendedNetworkService для получения ответа USSD на Android

Как читать сообщения USSD в android?

Внедрение функций USSD. Связывание службы с PhoneUtils без перезапуска телефона при каждом обновлении

но он также был бесполезен для Android 4.2.2 и выше.

то я нашел эту ссылку: Как отклонить системный диалог на Android?

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

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

Кто-нибудь знает, работает ли какой-либо из этих методов или других лиц для управления или даже обнаружения сообщений USSD в android 4.2.2 и выше?

Я очень ценю любую помощь, спасибо заранее.

4b9b3361

Ответ 1

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

public class USSDService extends AccessibilityService {

public static String TAG = USSDService.class.getSimpleName();

@Override
public void onAccessibilityEvent(AccessibilityEvent event) {
    Log.d(TAG, "onAccessibilityEvent");

    AccessibilityNodeInfo source = event.getSource();
    /* if (event.getEventType() == AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED && !event.getClassName().equals("android.app.AlertDialog")) { // android.app.AlertDialog is the standard but not for all phones  */
    if (event.getEventType() == AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED && !String.valueOf(event.getClassName()).contains("AlertDialog")) {
        return;
    }
    if(event.getEventType() == AccessibilityEvent.TYPE_WINDOW_CONTENT_CHANGED && (source == null || !source.getClassName().equals("android.widget.TextView"))) {
        return;
    }
    if(event.getEventType() == AccessibilityEvent.TYPE_WINDOW_CONTENT_CHANGED && TextUtils.isEmpty(source.getText())) {
        return;
    }

    List<CharSequence> eventText;

    if(event.getEventType() == AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED) {
        eventText = event.getText();
    } else {
        eventText = Collections.singletonList(source.getText());
    }

    String text = processUSSDText(eventText);

    if( TextUtils.isEmpty(text) ) return;

    // Close dialog
    performGlobalAction(GLOBAL_ACTION_BACK); // This works on 4.1+ only

    Log.d(TAG, text);
    // Handle USSD response here

}

private String processUSSDText(List<CharSequence> eventText) {
    for (CharSequence s : eventText) {
        String text = String.valueOf(s);
        // Return text if text is the expected ussd response
        if( true ) {
            return text;
        }
    }
    return null;
}

@Override
public void onInterrupt() {
}

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

Объявить его в Android-манифеста

<service android:name=".USSDService"
android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE">
<intent-filter>
    <action android:name="android.accessibilityservice.AccessibilityService" />
</intent-filter>
<meta-data android:name="android.accessibilityservice"
    android:resource="@xml/ussd_service" />

Создайте XML файл, который описывает службу доступности, называемую ussd_service

<?xml version="1.0" encoding="utf-8"?>
<accessibility-service xmlns:android="http://schemas.android.com/apk/res/android"
android:accessibilityEventTypes="typeWindowStateChanged|typeWindowContentChanged"
android:accessibilityFeedbackType="feedbackGeneric"
android:accessibilityFlags="flagDefault"
android:canRetrieveWindowContent="true"
android:description="@string/accessibility_service_description"
android:notificationTimeout="0"
android:packageNames="com.android.phone" />

Что это. После того, как приложение установлено, вам необходимо включить службу в настройках доступности (настройка- > Параметры специальных возможностей → имя_файла).

Решение описано здесь и здесь (русский).

Ответ 2

К сожалению, нет способа.

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

В результате мы смогли достичь соглашения с операторами мобильной связи в нашей стране (у нас только 3 оператора мобильной связи), и они предоставили нам API для получения баланса номера телефона, но этот метод работает только на местном уровне.