Android 4.1:アプリケーションの通知を無効にする方法を教えてください。


98

Android 4.1では、特定のアプリケーションの通知を無効にするためのチェックボックスがユーザーに提供されています。

ただし、開発者として、通知の呼び出しが有効であったかどうかを知る方法はありません。

現在のアプリケーションで通知が無効になっているかどうかを確認する必要がありますが、APIでその設定を見つけることができません。

コードでこの設定を確認する方法はありますか?


1
あなたは本当にそれに気をつけるべきではありません。通知が成功したと想定してください。ユーザーが通知を明示的に無効にした場合は、おそらく無効にする理由があり、アプリケーションは通知が表示されたかどうかを気にする必要はありません。
ケビンコポック

最初のアンサーのコメントで理由を説明しました。
ギヨームペロット

1
これがstar / track code.google.com/p/android/issues/detail?id=38482の問題です。これが本当に必要です...
brandall

回答:


147

あなたは100%できないことはできません。

このGoogle I / O 2012ビデオで尋ねられ、新しい通知のプロジェクトリーダーは、それができないことを宣言します。


編集する

2016年の更新:このGoogle I / O 2016ビデオで述べたように、これを確認できます。

NotificationManagerCompat.areNotificationsEnabled()サポートライブラリのを使用して、API 19以降で通知がブロックされているかどうかを確認します。API 19以下のバージョンはtrueを返します(通知は有効です)。

ここに画像の説明を入力してください


2
この設定を読み取ることができないことを示すビデオには何もありません。ただ明確にするために、チェックボックスの現在の状態を変更するのではなく、読み取れるようにしたいだけです。私の質問が理解できなかったと思います。
ギヨームペロット

2
一度に1つの通知を表示するため、これが必要です(アプリ内またはシステムバーに表示できます)。システム通知が表示される場合、アプリ内バナーは表示されません。逆も同様です。通知が表示されるかどうかがわからない場合、そのライフサイクルを管理できなくなります。通知の管理方法を完全に変更する必要があると思います...
Guillaume Perrot

9
参考までに、質問はビデオ(質問と回答の間)の48:05に1つの短い単語で質問(および回答)されます...いいえ。youtube.com/...
Devunwired

2
@Stavros_Sがフルリンクで回答を更新しました。NotificationManagerCompat.areNotificationsEnabled()
スーフィアン

16
@Stavros_S使用する必要がありますNotificationManagerCompat.from(ctx).areNotificationsEnabled()
AlexAndro

38

実際、これは非常に簡単です。

/**
 * Created by desgraci on 5/7/15.
*/
public class NotificationsUtils {

    private static final String CHECK_OP_NO_THROW = "checkOpNoThrow";
    private static final String OP_POST_NOTIFICATION = "OP_POST_NOTIFICATION";

    public static boolean isNotificationEnabled(Context context) {

        AppOpsManager mAppOps = (AppOpsManager) context.getSystemService(Context.APP_OPS_SERVICE);

        ApplicationInfo appInfo = context.getApplicationInfo();

        String pkg = context.getApplicationContext().getPackageName();

        int uid = appInfo.uid;

        Class appOpsClass = null; /* Context.APP_OPS_MANAGER */

        try {

            appOpsClass = Class.forName(AppOpsManager.class.getName());

            Method checkOpNoThrowMethod = appOpsClass.getMethod(CHECK_OP_NO_THROW, Integer.TYPE, Integer.TYPE, String.class);

            Field opPostNotificationValue = appOpsClass.getDeclaredField(OP_POST_NOTIFICATION);
            int value = (int)opPostNotificationValue.get(Integer.class);

            return ((int)checkOpNoThrowMethod.invoke(mAppOps,value, uid, pkg) == AppOpsManager.MODE_ALLOWED);

        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
        return false;
    }
}

3
質問が投稿された時点では、Android 4.1が最新でした。これはAndroid 4.4以降のみを対象としており、リフレクションを使用しているようで、ドキュメントではシステム以外のアプリでの使用は推奨されていません。
Guillaume Perrot

@GuillaumePerrot実際には、あなたは反射について正しいですが、繰り返しになりますが、androidのドキュメントと公式声明は、それを行うことはできないと言っています。バージョンの問題でごめんなさい、私はあなたを助けることができません。クライアント/ソリューションで必要な場合は、SDKがその時点でユーザーを制限しているため、必要なバージョンを少し上げることを検討してください。別の方法を見つけた場合はお知らせください。
desgraci

1
十分に公平ですが、情報を取得できない場合はtrueを返すと想定する必要があります。少なくとも私のユースケースではそれはもっと理にかなっています。または、静的関数のパラメーターとしてデフォルト値を設定して、再利用しやすくします。
Guillaume Perrot、2015

1
@Rahul Matte、GlobalContextは、私がContextへの参照を保持するために使用するユーティリティクラスにすぎません。その構造を使用または使用しない場合は、メソッドを介してContextを渡すことができます。お役に立てれば!
desgraci

1
:pではなく、Google XDからのものであることを願っています。これはリフレクションを使用しているため、OP_POST_NOTIFICATIONが同じ問題に苦しんでいるのを見つけた後、grepのコードを読み通していました。
desgraci

37

@blundellからの回答は正しいですが、新しいバージョンでは小さな変更があります。

NotificationManagerCompat.from(context).areNotificationsEnabled()

5

Xamarinを使用していて、この答えが必要な場合は、次のコードを使用できます。

//return true if this option is not supported.
public class NotificationsUtils 
{
    private const String CHECK_OP_NO_THROW = "checkOpNoThrow";
    private const String OP_POST_NOTIFICATION = "OP_POST_NOTIFICATION";

    public static bool IsNotificationEnabled(global::Android.Content.Context context) {

        AppOpsManager mAppOps = (AppOpsManager) context.GetSystemService(global::Android.Content.Context.AppOpsService);

        ApplicationInfo appInfo = context.ApplicationInfo;

        String pkg = context.ApplicationContext.PackageName;

        int uid = appInfo.Uid;

        try {

            var appOpsClass = Java.Lang.Class.ForName("android.app.AppOpsManager");
            var checkOpNoThrowMethod = appOpsClass.GetMethod(CHECK_OP_NO_THROW,Java.Lang.Integer.Type,Java.Lang.Integer.Type,new Java.Lang.String().Class);//need to add String.Type

            var opPostNotificationValue = appOpsClass.GetDeclaredField (OP_POST_NOTIFICATION);
            var value = (int)opPostNotificationValue.GetInt(Java.Lang.Integer.Type);
            var mode = (int)checkOpNoThrowMethod.Invoke(mAppOps,value, uid, pkg);
            return (mode == (int)AppOpsManagerMode.Allowed);

        } catch (Exception) 
        {
            System.Diagnostics.Debug.WriteLine  ("Notification services is off or not supported");
        } 
        return true;
    }
}

@AdamPedley AreNotificationsEnabled()API 24に加え developer.android.com/reference/android/app/...
SuneKjærgård

あなたは正しいです、以前にそれを誤解している必要があります。誰も混乱させないように、元のコメントは削除しました。
Adam Pedley 2017

5

通知状態を照会する方法がないようです。

私はこれをお勧めします:

  • 通知を使用してアプリケーションを設計します。
  • ユーザーがアプリケーションの設定からの通知を無効にできるようにします。
  • 通知がクリックされているかどうかを確認します。ユーザーが通知をクリックした場合、これを設定に保存します。
  • アプリで、通知設定がオンで、ユーザーがAndroid 4.1以降(API 16)の場合、ユーザーが数日/数週間通知をクリックしない場合は、ユーザーが通知を無効にしたと想定します。

100%正解ではありません。しかし、これは意見を与えます。
たとえば、ユーザーが10〜15日間アプリの通知をクリックしない場合、おそらくそれを無効にします


1
これは非常に広く抽象的なアプローチです。
IgorGanapolsky 2015年

これが最善の方法です。これはアプリケーションで行っており、通知が無効になっている場合は、正確に言うことができます。すべてのアクションのPendingIndentと設定に保存します。スマートフォンが再起動した場合は、忘れずにリセットしてください。
JacksOnF1re 2015年

1

このメソッドを使用して、通知が有効かどうかを確認します。上記の方法は、通知が有効かどうかを確認するために機能します。ただし、Android 8以降では、通知を作成するために最初チャネルを作成する必要があるため、Oreoから、通知チャネルが有効になっているかどうかを確認する必要があります

    /**
     * Checking Whether notifications are enabled or not
     * @return true if notifications are enabled otherwise false
     */
    public static final String CHANNEL_ID = your_channel_id";

    private boolean isNotificationChannelEnabled(){
        if(NotificationManagerCompat.from(this).areNotificationsEnabled()) {
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                NotificationManager manager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
                NotificationChannel channel = manager.getNotificationChannel(CHANNEL_ID);
                if (channel == null)
                    return true; //channel is not yet created so return boolean
                // by only checking whether notifications enabled or not
                return channel.getImportance() != NotificationManager.IMPORTANCE_NONE;
            }
            return true;
        }
        return false;
    }
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.