web-dev-qa-db-ja.com

androidでUSSDダイアログを管理する新しい方法

私はここや他のフォーラムでUSSDダイアログに関するスレッドを数日間読んでいます。 (USSDとは、通話料金の詳細を含むオペレーターの通知を意味します)。

Android)で明らかに低いAPIレベルで機能する多くのソリューションを見てきましたが、多くのテストを行った結果、それらが機能しないか、少なくとも機能させることができませんでした。

まず、ユーザーにussdダイアログが表示されたかどうかを検出する方法があるかどうかを知りたいと思いました。だから私はこれを試しました: SSDダイアログを防ぎ、USSD応答を読みますか?

ログから取得できるすべては、何らかの形で私のアプリに関連していましたが、EclipseのLogCatでそれらを表示できましたが、アプリでMMI関連のログはキャプチャされませんでした。多分それは私よりも低いAndroidバージョンで動作していました(4.2.2)。

次に、次のリンクや他の多くのリンクで使用されているように、「IExtendedNetworkService」を使用することにしました。

https://github.com/alaasalman/ussdinterceptor

AndroidでISD応答を取得するためにIExtendedNetworkServiceを使用

AndroidでUSSDメッセージを読み取る方法

SSD機能の実装。更新のたびに電話を再起動せずにサービスをPhoneUtilsにバインドする

しかし、Android 4.2.2以降では役に立たなかった。

それから私はこのリンクを見つけました: Androidのシステムダイアログを閉じる方法?

それは有望に思えたが、たくさんのテストをした後でさえ、それを機能させることができなかった。多分私は何か悪いことをしているのか、それともより低いAPIのためでもあると思いました。

その後、ホームキーとバックキーをシミュレートしたり、プログラムでタッチして通知を非表示にしたりするなど、他にも多くのことを試しましたが、どれも自分の活動で動作していたため、どれも動作しませんでした。

Android 4.2.2以上でUSSDメッセージを管理または検出するためにこれらの方法または他の方法が機能するかどうかを誰かが知っていますか?

前もって感謝します。

31
Lord Sepid

アクセシビリティサービスを利用して可能です。最初にサービスクラスを作成します。

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 manifestで宣言します

<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" />

Ussd_serviceと呼ばれるアクセシビリティサービスを記述するxmlファイルを作成します。

<?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" />

それでおしまい。アプリをインストールしたら、ユーザー補助の設定(設定->ユーザー補助の設定-> YourAppName)でサービスを有効にする必要があります。

ソリューションの説明 here および here(ロシア語)

8
HenBoy331

残念ながらできません。

このコマンドの検索もこの問題の解決策でしたが、すべてのデバイスで100%の動作を保証するものではない、電話でのログの解析を通じてのみ情報を取得することが判明した場合。

その結果、我が国の携帯電話事業者(3社のみ)と合意に達し、電話番号の残高を取得するためのAPIを提供してくれましたが、この方法はローカルでしか機能しません。

0