Android 4.1:如何检查已禁用应用程序的通知?


98

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

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

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

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


1
您真的不应该为此担心。假设您的通知成功。如果用户明确禁用了您的通知,则他/她可能有充分的理由这样做,并且您的应用程序不应该关心是否显示了通知。
凯文·科波克

我在第一批评论者的评论中解释了原因。
Guillaume Perrot

1
这里是问题明星/跟踪code.google.com/p/android/issues/detail?id=38482真的需要这个....
brandall

Answers:


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提问(并回答),并用一个简短的单词...否。youtube.com/...
Devunwired

2
@Stavros_S使用完整链接更新了答案。NotificationManagerCompat.areNotificationsEnabled()
苏菲安

16
@Stavros_S您需要使用NotificationManagerCompat.from(ctx).areNotificationsEnabled()
AlexAndro '16

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+,并且似乎使用了反射,并且文档不建议将其用于非系统应用程序。
纪尧姆·佩罗

@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()中的溶液在24 API添加 developer.android.com/reference/android/app/...
淑娜Kjærgård

您是正确的,以前必须误读它。我删除了原始评论,以免混淆任何人。
亚当·佩德利

5

似乎无法查询通知状态。

我建议这样做:

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

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


1
这是一种非常广泛和抽象的方法。
IgorGanapolsky'3

这是最好的方法!我们正在应用程序中执行此操作,您可以准确地说出是否禁用了通知。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;
    }
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.