Android插件框架
来源:互联网 发布:云计算平台技术 编辑:程序博客网 时间:2024/06/14 10:57
相信大家如果在项目中使用过插件框架或是对插件框架有一些了解,会知道插件框架有一个很无奈的问题,就是无法让插件自主发送通知,特别是那些自定义RemoteViews的通知.
就连大名鼎鼎的DroidPlugin也在实现中只让插件发送那些使用系统自带布局的通知,可查看具体的实现代码, –蹭个关键字 哈哈
为什么通知不好实现
1. small icon
发送通知需要一个的drawable id,系统在接收到发送通知请求后,会根据包名和resource id检查该icon是否存在,不存在就crash. 而插件的包名是没有存在系统已安装应用列表内的,所以直接crash有人会说,我hook后使用宿主的包名不就行了,这就又有一个问题,宿主里面不一定存在这个id的drawable,即使存在,也不太容易保证是你想要的那个id,毕竟id的生成工作是ide来协助完成的当然我知道有项目真的在自己来控制生成resource id,但这个不是我们今天讨论的话题,就不过于深入了
2. 通知的显示由系统负责
这部分主要包含两个意思:
- 显示的进程是系统进程
- View的构造是由系统来负责
也带来了以下两个问题:
- 无法通过插件框架核心的hook技术来拦截系统实现,在低版本的rom中,甚至在通知显示时,宿主程序都没有启动
- 插件自定义布局系统找不到,也就无从构造
3. 插件通知的点击事件一般直接指向插件,系统在处理这些事件时,因为找不到插件也会引发crash
主要问题是因为点击事件触发后是由系统来处理的,系统处理时,插件甚至宿主都不一定启动,就算启动了,根据插件提供的信息系统也找不到相应的应用的
解决该问题的核心思路
1. 截图
提取插件即将要发送的通知的RemoteViews,转化成View,然后根据通知类型(是否包含bigContentView)截图成一个图片,然后宿主直接发送纯图片通知
2. 事件封包转发,统一处理
对插件通知的点击事件进行二次封包,当用户点击首先接收到消息的是插件框架(宿主),然后插件框架再解包,找出原始的真正意图,再通过插件框架将其跳转指定页面/启动service/发送通知等等
3. 使用占位的方式解决事件点击问题
宿主在发送插件通知的时候(实际是一张图片),如果该通知布局内包含按钮(开始/暂停,跳转之类的),那在该图片的下方or上方指定位置显示一个透明的按钮,参照思路2进行点击事件封装,然后供用户点击
具体的解决步骤
1.Hook系统的NotificationManager
实际是通过getService提取其INotificationManager,然后hook INotificationManager的以下几个方法:
- enqueueNotification
- enqueueNotificationWithTag
- cancelNotification
- cancelNotificationWithTag
- cancelAllNotifications
当然,开始的时候按需hook就成,后期再去完善
2.将插件的RemoteViews转成View
核心方法是RemoteViews的apply方法,但需要关注一个细节:
在使用系统布局进行显示通知的时候,布局中会涉及到large icon和small icon.
然而在apply的时候,如果指定的是resource id,还是会根据包名和resource id去构造图片,依然会遇见找不到的问题,这样你构造出来的view,是缺少large icon和small icon的,在一些rom中会显示一个很扎眼的灰色块
所以,在转成view之前,你需要将RemoteViews里面包含large icon/small icon转成bitmap
如果large icon本身已经是bitmap的无需再转化,但如果是android M引进来的Icon对象,则依然需要将其转成bitmap,因为Icon最终在构造view时,依然是通过包名+resource id进行查找的
3.将View转成Bitmap
View转成Bitmap需要注意以下几个问题:
- view的高度需要指定,默认情况下普通通知的高度是64dp,包含bigContentView的是255dp,按照插件发送的通知类型指定高度
- 将view转化成bitmap需要有一个前提是view已经被layout的过,不然截出来的bitmap是黑色的
- 经过测试发现,在将RemoteViews转成View的过程中,部分机型会将插件没有设置背景的layout设置成黑色,于是在截图的时候发现bitmap部分区域是黑色的,这个时候,需要将View中黑色layout替换成透明色
这部分因为和业务无关,相对无耦合,直接贴出代码吧
public static Bitmap printView(View v, int width, int height) { v.layout(0, 0, width, height); int measuredWidth = View.MeasureSpec.makeMeasureSpec(width, View.MeasureSpec.EXACTLY); int measuredHeight = View.MeasureSpec.makeMeasureSpec(height, View.MeasureSpec.EXACTLY); v.measure(measuredWidth, measuredHeight); v.layout(0, 0, v.getMeasuredWidth(), v.getMeasuredHeight()); int viewWidth = v.getMeasuredWidth(); int viewHeight = v.getMeasuredHeight(); if (viewWidth > 0 && viewHeight > 0) { opration(v); Bitmap bitmap = Bitmap.createBitmap(viewWidth, viewHeight, Bitmap.Config.ARGB_8888); Canvas canvas = new Canvas(bitmap); v.draw(canvas); return bitmap; } return null; } private static void opration(View view) { try { Drawable background = view.getBackground(); if (background instanceof ColorDrawable) { ColorDrawable colorDrawable = (ColorDrawable) background; if (colorDrawable.getColor() == Color.BLACK) { colorDrawable.setColor(Color.TRANSPARENT); } } if (view instanceof ViewGroup) { ViewGroup vg = (ViewGroup) view; int childCount = vg.getChildCount(); for (int i = 0; i < childCount; i++) { opration(vg.getChildAt(i)); } } } catch (Exception e) { LogUtils.e(e); } }
4.找出RemoteViews中的所有点击事件
用户是通过setOnClickPendingIntent来设置点击事件的,而实现的原理是生成一个内部类SetOnClickPendingIntent实例,然后插入到一个mActions数组中.所以具体的实现方法如下:
- 通过反射构造SetOnClickPendingIntent的Class
Class<?> onClickHandlerClass = Class.forName("android.widget.RemoteViews$SetOnClickPendingIntent");
- 继续通过反射找出mActions数组 – 方法略
- 遍历mActions数组,找出所有的SetOnClickPendingIntent
5.从SetOnClickPendingIntent中找出响应该点击事件的view,找出其原始坐标
具体方法如下:
- 反射找出viewId
- 根据viewId,从构造出来的通知View中找出目标子View
- 找出子View相对根View的坐标,具体的方法如下:
public static float getDescendantCoordRelativeToSelf(View descendant, View rootView, int[] coord) { float scale = 1.0f; float[] pt = {coord[0], coord[1]}; //坐标值进行当前窗口的矩阵映射,比如View进行了旋转之类,它的坐标系会发生改变。map之后,会把点转换为改变之前的坐标。 descendant.getMatrix().mapPoints(pt); //转换为直接父窗口的坐标 scale *= descendant.getScaleX(); pt[0] += descendant.getLeft(); pt[1] += descendant.getTop(); ViewParent viewParent = descendant.getParent(); //循环获得父窗口的父窗口,并且依次计算在每个父窗口中的坐标 while (viewParent instanceof View && viewParent != rootView) { final View view = (View) viewParent; view.getMatrix().mapPoints(pt); scale *= view.getScaleX();//这个是计算X的缩放值。此处可以不管 //转换为相当于可视区左上角的坐标,scrollX,scollY是去掉滚动的影响 pt[0] += view.getLeft() - view.getScrollX(); pt[1] += view.getTop() - view.getScrollY(); viewParent = view.getParent(); } coord[0] = Math.round(pt[0]); coord[1] = Math.round(pt[1]); return scale; }
6.将原始的点击事件PendingIntent进行封包,统一处理插件通知的所有点击事件
再次解释一下为什么这么做的原因: 点击事件由用户触发,但真正执行是由系统来执行,所以如果使用插件通知原有的点击事件,一般是导向插件本身的,系统会因为找不到插件而出现crash亦或没有反应
具体的解决思路如下:
- 提取插件通知所有的PendingInteng(包括contentPendingIntent以及通知View内部的子view的PendingIntent)
- 在宿主中定义一个透明的ClickHandleActivity,将上一步提取的PendingIntent通过参数的方式设置进去,构造出一个启动ClickHandleActivity的点击事件
- ClickHandleActivity接到事件后,提取原始的PendingInteng,然后再根据插件框架的原理进行后续操作(启动Activity/启动Service/发送Broadcast),同时记得要把这个ClickHandleActivity给finish掉
其中有一个需要注意的点是,我们在获取PendingIntent的时候,已经没办法知道这个事件到底是要去干什么,是要启动Activity还是service,亦或是发送广播的
我这边的解决思路是在插件发送广播时,hook IActivityManager的getIntentSender方法,然后将类型给保存下来,然后在封包插件通知点击事件的时候,再将类型给传过去
这个思路虽然已经验证可行,但总觉得有点low,如果有哪位找到更合适的方法,麻烦同步一下给我,非常感谢
7.使用子View的相对坐标以及封包后的点击事件,设置到新构造出来的RemoteViews中去
解决该问题的思路是:
- 在新的RemoteViews的layout中提前预设多个透明子view
- 根据之前获取的子view相对坐标,调整预设的透明子view的位置
- 将封包后的点击事件设置到预设的透明子view中
这里面根据经验发现有一个坑需要填:在RemoteViews没有提供设置View margin的方法,只能设置padding
所以,没办法,我在预设透明子view的结构如下:
<RelativeLayout android:id="@+id/notify_layout1" android:layout_width="match_parent" android:layout_height="match_parent" android:visibility="gone"> <TextView android:id="@+id/notify_btn1" android:layout_width="wrap_content" android:layout_height="wrap_content" /> </RelativeLayout> <RelativeLayout android:id="@+id/notify_layout2" android:layout_width="match_parent" android:layout_height="match_parent"> <TextView android:id="@+id/notify_btn2" android:layout_width="0dp" android:layout_height="0dp" /> </RelativeLayout> ...
实际响应事件的是notify_btn*,但需要通过notify_layout*来调整位置
8.将截图的bitmap设置到新的RemoteViews中
9.copy原插件Notification的属性,构造新的Notification对象,设置small icon为宿主的small icon,然后发送即可
到此,整个实现过程就结束了,对于绝大部分场景应该都能完美解决了,希望能帮到大家
- Android插件开发框架
- android 插件框架机制
- Android插件框架VirtualAPK
- Android插件框架
- Android插件化框架介绍
- Android插件式换肤框架搭建
- Android插件式换肤框架搭建
- Android插件式换肤框架搭建
- Android插件化框架SpeedTools
- Android 插件化框架DroidPlugin
- Android插件化框架总结
- Android-DLPlugin插件化框架
- Android插件化开发框架
- android插件化框架-VirtualApk
- android插件化框架-Replugin
- PhoneGap框架android插件的实现
- Android 插件框架 xCombine 开发思路简介
- PhoneGap框架android插件的实现
- BlockingQueue使用说明
- ACM 数论
- LASSO与redge回归区别 L1 L2范数之间的区别
- CSS实现tip框三角形
- 简单的获取当地时间
- Android插件框架
- bzoj1531[POI2005]Bank notes 多重背包
- 简单的计算器,包含UI
- 关于飞思卡尔电磁组舵机反偏(乱打角)问题的总结
- glog的使用教程
- Wow6432Node
- [金融]衍生品定价,债券,期权,期货
- 简单的聊天室,包含UI
- AtCoder Grand Contest 014C: Closed Rooms 题解