102

Android 4.1 为用户提供了一个复选框来禁用特定应用程序的通知。

但是,作为开发人员,我们无法知道通知调用是否有效。

我真的需要检查当前应用程序是否禁用了通知,但我在 API 中找不到任何设置。

有没有办法在代码中检查此设置?

4

6 回答 6

153

你不能100%不能。

这个 Google I/O 2012 视频中被问到,新通知的项目负责人声明你不能。


编辑

2016 年更新:现在您可以查看它,如Google I/O 2016 视频中所述。

使用NotificationManagerCompat.areNotificationsEnabled()支持库中的 来检查通知是否在 API 19+ 上被阻止。API 19 以下的版本将返回 true(启用通知)。

在此处输入图像描述

于 2012-07-25T12:25:15.523 回答
46

@blundell 的回答是正确的,但在较新的版本中有一个小的变化。

NotificationManagerCompat.from(context).areNotificationsEnabled()
于 2017-01-02T16:58:35.077 回答
39

其实这很容易做到:

/**
 * 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;
    }
}
于 2015-05-07T17:31:32.987 回答
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;
    }
}
于 2016-02-02T06:11:56.907 回答
4

似乎没有办法查询通知状态。

我推荐这个:

  • 使用通知设计您的应用程序。
  • 让用户禁用应用程序设置中的通知。
  • 检查通知是否被点击。如果用户单击通知,请将其保存到首选项。
  • 在您的应用中,如果通知设置已打开,并且用户是 Android 4.1+ (API 16),但如果用户在几天/几周内没有点击通知,则假定用户禁用了通知。

不是100%正确。但这给出了一个意见。
例如,如果用户在 10-15 天内没有点击任何应用通知,则可能是他禁用了它

于 2014-05-08T12:36:21.080 回答
3

我使用这种方法来检查通知是否启用,上述方法将用于检查通知是否启用。但是从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;
    }

于 2020-04-18T12:05:02.173 回答