Android 4.1 : 애플리케이션에 대한 알림이 비활성화되었는지 확인하는 방법은 무엇입니까?


98

Android 4.1은 사용자에게 특정 애플리케이션에 대한 알림을 비활성화하는 확인란을 제공합니다.

그러나 개발자로서 알림 호출이 효과적인지 여부를 알 수있는 방법이 없습니다.

현재 애플리케이션에 대한 알림이 비활성화되어 있는지 확인해야하지만 API에서 해당 설정을 찾을 수 없습니다.

코드에서이 설정을 확인하는 방법이 있습니까?


1
당신은 그것에 대해 정말로 걱정해서는 안됩니다. 알림이 성공했다고 가정하십시오. 사용자가 귀하의 알림을 명시 적으로 비활성화했다면 그 / 그녀는 그럴만 한 이유가있을 수 있으며, 애플리케이션은 알림이 표시되는지 여부를 신경 쓰지 않아야합니다.
Kevin Coppock 2012-07-25

나는 첫 번째 anwser의 의견에서 그 이유를 설명했습니다.
Guillaume Perrot 2012

1
여기에 스타에 문제가있다 / 추적 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
이 설정을 읽을 수 없음을 보여주는 비디오가 없습니다. 명확하게 말하면 확인란의 현재 상태를 변경하지 않고 읽을 수 있기를 원합니다. 제 질문을 이해하지 못하신 것 같습니다.
Guillaume Perrot 2012

2
한 번에 하나의 알림을 표시하기 때문에 필요합니다 (앱 내 또는 시스템 표시 줄에있을 수 있음). 시스템 알림이 표시되면 인앱 배너가 표시되지 않으며 그 반대의 경우도 마찬가지입니다. 알림이 표시되는지 여부를 알 수 없으면 더 이상 수명주기를 관리 할 수 ​​없습니다. 나는 우리가 완전히 우리가 지금 알림을 관리하는 방법 ... 변경해야 할 것 같아요
기욤 PERROT

9
참고로, 비디오 (Q & A 중)에서 48:05에 짧은 단어 한 개로 질문 (답변)됩니다. youtube.com/…
Devunwired 2011

2
@Stavros_S는 전체 링크로 답변을 업데이트했습니다. NotificationManagerCompat.areNotificationsEnabled () .
Sufian

16
당신은 사용할 필요가 @Stavros_SNotificationManagerCompat.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 사실, 당신은 반성에 대해 옳지 만, 안드로이드의 문서와 공식 성명은 당신이 할 수 없다고 말하고 있습니다. 버전 문제로 죄송합니다. 도와 드릴 수 없습니다. 클라이언트 / 솔루션에 필요한 경우 SDK가 해당 시점에서 제한되기 때문에 필요한 버전을 약간 올리는 것을 고려할 수 있습니다. 다른 방법을 찾으면 알려주세요.
desgraci

1
충분히 공평하지만 정보를 얻을 수없는 경우에는 사실로 간주해야합니다. 적어도 내 사용 사례에서는 더 의미가 있습니다. 또는보다 재사용 가능하도록 정적 함수의 매개 변수로 기본값을 사용하십시오.
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/...
Sune Kjærgård을

정확합니다. 이전에 잘못 읽었을 것입니다. 다른 사람을 혼동하지 않도록 원래 댓글을 삭제했습니다.
Adam Pedley

5

알림 상태를 쿼리 할 방법이없는 것 같습니다.

나는 이것을 추천한다 :

  • 알림으로 애플리케이션을 디자인하세요.
  • 사용자가 애플리케이션 설정에서 알림을 비활성화 할 수 있도록합니다.
  • 알림이 클릭되었는지 확인하십시오. 사용자가 알림을 클릭하면이를 기본 설정에 저장합니다.
  • 앱에서 알림 설정이 켜져 있고 사용자가 Android 4.1 이상 (API 16) 인 경우 사용자가 며칠 / 주 동안 알림을 클릭하지 않으면 사용자가 알림을 비활성화했다고 가정합니다.

100 % 정확하지 않습니다. 그러나 이것은 의견을 제공합니다.
예를 들어 사용자가 10 ~ 15 일 동안 앱 알림을 클릭하지 않으면 비활성화했을 수 있습니다.


1
이것은 매우 광범위하고 추상적 인 접근 방식입니다.
IgorGanapolsky

이것이 최선의 방법입니다! 우리는 응용 프로그램에서이 작업을 수행하고 있으며 알림이 비활성화되어 있는지 정확하게 말할 수 있습니다. 모든 작업에 대해 PendingIndent를 사용하고 기본 설정에 저장합니다. 스마트 폰이 다시 시작되면 재설정하는 것을 잊지 마십시오.
JacksOnF1re 2015-06-11

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.