当前位置:   article > 正文

AppWidget实现机制分析--什么是桌面插件_android widgetprovider原理

android widgetprovider原理
    离开android开发一年半时间,终于又回到久违的android开发,与2011比,android系统更是搞得风生水起,市面上充斥着各种android系统的产品,从手机到电视,从医疗设备到汽车电子,魔多之眼在召唤,我又回来了,离开那与世隔绝小镇(无外网访问的一家it公司),来到中土世界,终于又可以在csdn写博客了。

    废话不多说,进入正题,最近两周查一个桌面插件(appwidget)问题,搞两周下来感觉对桌面插件的认识有了一个深入,全面的认识,有必要把它给总结一下,就从一个简单的textview的桌面插件为例开始讲起吧,这个例很简单,android应用开发解密和android sdk中带的demo都有,就是不断更新桌面插件textview的文本,很容易获取这样一个例子,你可以先下一个安装下,对桌面插件有一个直观的认识,本文不是告诉你怎么写一个桌面插件应用,而是阐述其内部实现机制。

   首先,我想我们得了解一个问题,那就是桌面插件是什么?先看AndroidManifest.xml中对一个桌面插件的定义

  1. <receiver android:name=".ExampleAppWidgetProvider">
  2.             <meta-data android:name="android.appwidget.provider"
  3.                     android:resource="@xml/appwidget_provider" />
  4.             <intent-filter>
  5.                 <action android:name="android.appwidget.action.APPWIDGET_UPDATE" />
  6.                 <action android:name="android.appwidget.action.APPWIDGET_DELETED" />
  7.                 <action android:name="android.appwidget.action.APPWIDGET_DISABLED" />
  8.                 <action android:name="android.appwidget.action.APPWIDGET_ENABLED" />
  9.                 <action android:name="android.appwidget.action.APPWIDGET_ENABLED" />
  10.             </intent-filter>
  11.         </receiver>

ExampleAppWidgetProvider是一个桌面插件provider类,它继承AppWidgetProvider,从AndroidManifest.xml中对一个桌面插件的定义中我们可以看出ExampleAppWidgetProvider实质上是一个BroadcastReceiver(广播接受器),从下面的类继承关系图我们也可以看出这一点:

但肯定它是有区别与一个普通的广播接收器的,试想,当我们添加桌面插件到桌面时有一个桌面插件选择列表,列表是有包管理器(packagemanager)加载出来的,packagemanager必须要知道它是桌面插件而非只是一个广播接收器,否则它是没有办法得出这个列表的,而决定一个BroadcastReceiver是否是一个AppWidgetProvider的玄机就是receiver中的meta-data定义,因为AppWidgetService服务启动就会加载所有桌面插件,所以我们就可以从AppWidgetService服务启动去剖析它

AppWidgetService->systemReady->loadAppWidgetList:

  1. /**通过PackageManager加载所有系统插件应用*/
  2. void loadAppWidgetList() {
  3. PackageManager pm = mPackageManager;
  4. Intent intent = new Intent(AppWidgetManager.ACTION_APPWIDGET_UPDATE);
  5. //查询所有的broadcastReceivers
  6. List<ResolveInfo> broadcastReceivers = pm.queryBroadcastReceivers(intent,
  7. PackageManager.GET_META_DATA);
  8. final int N = broadcastReceivers == null ? 0 : broadcastReceivers.size();
  9. //遍历所有的broadcastReceivers,添加provider
  10. for (int i=0; i<N; i++) {
  11. ResolveInfo ri = broadcastReceivers.get(i);
  12. //添加一个Provider若这个broadcastReceivers包含
  13. //name="android.appwidget.provider"的meta-data
  14. addProviderLocked(ri);
  15. }
  16. }
关键函数是这个loadAppWidgetList->addProviderLocked->parseProviderInfoXml:

  1. private Provider parseProviderInfoXml(ComponentName component, ResolveInfo ri) {
  2. Provider p = null;
  3. ActivityInfo activityInfo = ri.activityInfo;
  4. XmlResourceParser parser = null;
  5. try {
  6. //获取一个xml文件解析器,对应的xml解析文件是META_DATA_APPWIDGET_PROVIDER
  7. //这个meta_data指定的xml文件
  8. parser = activityInfo.loadXmlMetaData(mPackageManager,
  9. AppWidgetManager.META_DATA_APPWIDGET_PROVIDER);
  10. if (parser == null) {
  11. Slog.w(TAG, "No " + AppWidgetManager.META_DATA_APPWIDGET_PROVIDER + " meta-data for "
  12. + "AppWidget provider '" + component + '\'');
  13. return null;
  14. }
上面这一段代码清楚地说明了我前面提到的问题 :决定一个BroadcastReceiver是否是一个AppWidgetProvider的玄机就是receiver中的meta-data定义,你可以看一下AppWidgetManager.META_DATA_APPWIDGET_PROVIDER="android.appwidget.provider",它是一个meta-data name通过它获取到一个xml文件资源,得到一个xml文件解析器,若没有定义这个meta-data显然parser==null,那就不能生成返回一Provider对象,这就是为什么普通BroadcastReceiver不会当作桌面插件来处理的原因

    继续往下,AppWidgetService会根据meta-data指定的xml文件生成一个Provider对象,先看一下这个xml文件的定义:

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:minWidth="100dp"
  4. android:minHeight="50dp"
  5. android:updatePeriodMillis="50"
  6. android:initialLayout="@layout/appwidget_provider"
  7. android:configure="com.yarin.android.Examples_09_07.Activity01"
  8. >
  9. </appwidget-provider>
这个xml文件定义了一个桌面插件的配置信息,我这里对一些基本属性作了一个总结:

--android:minWidth:指定桌面组件的最小宽度,与AppWidget -ProviderInfo. minHeight成员对应

--android:minHeight: 指定桌面组件的最小高度,与AppWidget -ProviderInfo. minWidth成员对应

--android:updatePeriodMillis:指定桌面组件的更新周期,与AppWidget –ProviderInfo. updatePeriodMillis成员对应,这个注意AppWidgetService中作了一个限制,更新周期最少30分钟

--android:initialLayout:指定桌面组件初始化布局文件,与AppWidget –ProviderInfo. initialLayout成员对应,后面可为RemoteViews对象重新设置布局文件

--android:configure:指定桌面组件创建时候的一个初始配置activity,与AppWidgetProviderInfo. configured对应,可选。

遍历系统所有的BroadcastReceiver后,会在AppWidgetService中生成一个桌面插件应用列表mInstalledProviders,我这里把这个数据结构给描述一下,以便有一个直观认识:

前面xml中桌面插件配置信息都放在了AppWidgetProviderInfo中,当需要获取系统所有安装的桌面插件信息时,就可以调用getInstalledProviders获取,实质上就是获取mInstalledProviders列表。

   讲到这里我们再会过头前面我提出的问题,什么是桌面插件,是不是已经解答了?不然,我说到这里其实只是讲了桌面插件应用,说到桌面插件应用与桌面插件的关系你可以理解为类与对象的关系,桌面插件应用是静态的类型,桌面插件是动态生成的实例,为了便于理解我后面统一称为桌面插件实例,那我们继续解答什么是桌面插件这一问题。既然是桌面插件,那我们就从桌面入手,分析桌面插件是如何生成和添加的。

   欲知后事如何,请听下回分解。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/爱喝兽奶帝天荒/article/detail/861093
推荐阅读
相关标签
  

闽ICP备14008679号