利用NotificationListenerService获取微信通知消息的头像和内容

    技术2022-07-31  73

    强调下:利用Android原生功能获取第三方APP通知消息是流氓且不道德的(可能沾上官司),本文只讲原理

    强调下:利用Android原生功能获取第三方APP通知消息是流氓且不道德的(可能沾上官司),本文只讲原理

    强调下:利用Android原生功能获取第三方APP通知消息是流氓且不道德的(可能沾上官司),本文只讲原理

    提供一篇灵感文章,写的很好推荐看看https://blog.csdn.net/Vanswells/article/details/81033280

    获取Android 微信通知有两条思路,一条是走辅助功能(AccessibilityService),一条是走NotificationListenerService,这里讲第二条路NotificationListenerService。

    在App中加入一个类继承NotificationListenerService,用法类似于AccessibilityService,都是需要开通一个权限,但是开通的权限不同

    需要在应用启动的时候加一个权限判断,没有开启跳转到对应的Settings界面

    private void toggleNotificationListenerService(Context context) {         PackageManager pm = context.getPackageManager();         pm.setComponentEnabledSetting(new ComponentName(context, MyNotificationListenerService .class),                 PackageManager.COMPONENT_ENABLED_STATE_DISABLED, PackageManager.DONT_KILL_APP);           pm.setComponentEnabledSetting(new ComponentName(context, MyNotificationListenerService .class),                 PackageManager.COMPONENT_ENABLED_STATE_ENABLED, PackageManager.DONT_KILL_APP);     }       protected boolean gotoNotificationAccessSetting() {         try {             Intent intent = new Intent("android.settings.ACTION_NOTIFICATION_LISTENER_SETTINGS");             intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);             startActivity(intent);             return true;         } catch (ActivityNotFoundException e) {//普通情况下找不到的时候需要再特殊处理找一次             try {                 Intent intent = new Intent();                 intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);                 ComponentName cn = new ComponentName("com.android.settings", "com.android.settings.Settings$NotificationAccessSettingsActivity");                 intent.setComponent(cn);                 intent.putExtra(":settings:show_fragment", "NotificationAccessSettings");                 startActivity(intent);                 return true;             } catch (Exception e1) {                 e1.printStackTrace();             }             Toast.makeText(this, "对不起,您的手机暂不支持", Toast.LENGTH_SHORT).show();             System.out.println("-------------对不起,您的手机暂不支持------------->>");             e.printStackTrace();             return false;         }     }       public boolean isNotificationListenersEnabled() {         String pkgName = getPackageName();         final String flat = Settings.Secure.getString(getContentResolver(), "enabled_notification_listeners");         System.out.println("-----flat-------->" + flat);         if (!TextUtils.isEmpty(flat)) {             final String[] names = flat.split(":");             for (int i = 0; i < names.length; i++) {                 final ComponentName cn = ComponentName.unflattenFromString(names[i]);                 if (cn != null) {                     if (TextUtils.equals(pkgName, cn.getPackageName())) {                         return true;                     }                 }             }         }         return false;     } 权限开通以后,在sevices中就可以获取到我们想要的数据了

    import android.app.Notification; import android.app.PendingIntent; import android.graphics.Bitmap; import android.os.Bundle; import android.service.notification.NotificationListenerService; import android.service.notification.StatusBarNotification;   import org.greenrobot.eventbus.EventBus;     public class MyNotificationListenerService extends NotificationListenerService {       @Override     public void onListenerConnected() {         //当连接成功时调用,一般在开启监听后会回调一次该方法         System.out.println("----------------onListenerConnected--------->");     }       @Override     public void onNotificationPosted(StatusBarNotification sbn) {         //当收到一条消息时回调,sbn里面带有这条消息的具体信息         System.out.println("----------------onNotificationPosted--------->");         System.out.println("----------------onNotificationPosted----1----->" + sbn.getPackageName());         Bundle extras = sbn.getNotification().extras;         String title = extras.getString(Notification.EXTRA_TITLE); //通知title         String content = extras.getString(Notification.EXTRA_TEXT); //通知内容         int smallIconId = extras.getInt(Notification.EXTRA_SMALL_ICON); //通知小图标id         Bitmap largeIcon =  extras.getParcelable(Notification.EXTRA_LARGE_ICON); //通知的大图标,注意和获取小图标的区别         PendingIntent pendingIntent = sbn.getNotification().contentIntent; //获取通知的PendingIntent         System.out.println("----------------onNotificationPosted----1--1--->" + extras);         System.out.println("----------------onNotificationPosted----2----->" + title);         System.out.println("----------------onNotificationPosted----3----->" + content);         System.out.println("----------------onNotificationPosted----4----->" + smallIconId);         System.out.println("----------------onNotificationPosted----5----->" + largeIcon);         System.out.println("----------------onNotificationPosted----6----->" + pendingIntent);           NoticesBean mBean = new NoticesBean();         mBean.setLargeIcon(largeIcon);         mBean.setSmallIconId(smallIconId);         if (title == null || largeIcon == null) {             return;         }         EventBus.getDefault().post(mBean);     }       @Override     public void onNotificationRemoved(StatusBarNotification sbn) {         //当移除一条消息的时候回调,sbn是被移除的消息         System.out.println("----------------onNotificationRemoved--------->");     } } log就看的很明显了

    title可以用来区分是群聊还是个人消息,如果是群聊,title就是群聊,如果是个人消息,这里就显示消息的发送人

    content 是通知消息内容,结构是这样的:[第几条]发件人: 消息内容

    largeIcon:获取头像直接用就可以了(个人聊天是可以拿到一个bitmap对象,群聊获取的值是null或者是bitmap对象,是null还是bitmap主要取决于title值的内容,如果title值是“群聊”,那就是null,如果不是群聊就会有头像的bitmap对象)

    smallIcon:值衡为0

    我加的非空判断是因为有时候会有收到一条消息,但是上报两次,其中第二次是任何数据都没的,为了过滤掉无效数据,这里加个判断

    由于数据是在service中获取的,我们肯定是要在Activity或者是fragment中修改view,用EventBus传递数据就可以,很方便

     

    在AndroidManifest.xml配置service

            <service             android:name=".MyNotificationListenerService"             android:label="@string/app_name"             android:permission="android.permission.BIND_NOTIFICATION_LISTENER_SERVICE">             <intent-filter>                 <action android:name="android.service.notification.NotificationListenerService" />             </intent-filter>         </service>  

    Processed: 0.009, SQL: 9