废话不多说,进入正题,最近两周查一个桌面插件(appwidget)问题,搞两周下来感觉对桌面插件的认识有了一个深入,全面的认识,有必要把它给总结一下,就从一个简单的textview的桌面插件为例开始讲起吧,这个例很简单,android应用开发解密和android sdk中带的demo都有,就是不断更新桌面插件textview的文本,很容易获取这样一个例子,你可以先下一个安装下,对桌面插件有一个直观的认识,本文不是告诉你怎么写一个桌面插件应用,而是阐述其内部实现机制。
首先,我想我们得了解一个问题,那就是桌面插件是什么?先看AndroidManifest.xml中对一个桌面插件的定义
<receiver android:name=".ExampleAppWidgetProvider">
<meta-data android:name="android.appwidget.provider"
android:resource="@xml/appwidget_provider" />
<intent-filter>
<action android:name="android.appwidget.action.APPWIDGET_UPDATE" />
<action android:name="android.appwidget.action.APPWIDGET_DELETED" />
<action android:name="android.appwidget.action.APPWIDGET_DISABLED" />
<action android:name="android.appwidget.action.APPWIDGET_ENABLED" />
<action android:name="android.appwidget.action.APPWIDGET_ENABLED" />
</intent-filter>
</receiver>
ExampleAppWidgetProvider是一个桌面插件provider类,它继承AppWidgetProvider,从AndroidManifest.xml中对一个桌面插件的定义中我们可以看出ExampleAppWidgetProvider实质上是一个BroadcastReceiver(广播接受器),从下面的类继承关系图我们也可以看出这一点:
但肯定它是有区别与一个普通的广播接收器的,试想,当我们添加桌面插件到桌面时有一个桌面插件选择列表,列表是有包管理器(packagemanager)加载出来的,packagemanager必须要知道它是桌面插件而非只是一个广播接收器,否则它是没有办法得出这个列表的,而决定一个BroadcastReceiver是否是一个AppWidgetProvider的玄机就是receiver中的meta-data定义,因为AppWidgetService服务启动就会加载所有桌面插件,所以我们就可以从AppWidgetService服务启动去剖析它
AppWidgetService->systemReady->loadAppWidgetList:
/**通过PackageManager加载所有系统插件应用*/
void loadAppWidgetList() {
PackageManager pm = mPackageManager;
Intent intent = new Intent(AppWidgetManager.ACTION_APPWIDGET_UPDATE);
//查询所有的broadcastReceivers
List<ResolveInfo> broadcastReceivers = pm.queryBroadcastReceivers(intent,
PackageManager.GET_META_DATA);
final int N = broadcastReceivers == null ? 0 : broadcastReceivers.size();
//遍历所有的broadcastReceivers,添加provider
for (int i=0; i<N; i++) {
ResolveInfo ri = broadcastReceivers.get(i);
//添加一个Provider若这个broadcastReceivers包含
//name="android.appwidget.provider"的meta-data
addProviderLocked(ri);
}
}
关键函数是这个loadAppWidgetList->addProviderLocked->parseProviderInfoXml:
private Provider parseProviderInfoXml(ComponentName component, ResolveInfo ri) {
Provider p = null;
ActivityInfo activityInfo = ri.activityInfo;
XmlResourceParser parser = null;
try {
//获取一个xml文件解析器,对应的xml解析文件是META_DATA_APPWIDGET_PROVIDER
//这个meta_data指定的xml文件
parser = activityInfo.loadXmlMetaData(mPackageManager,
AppWidgetManager.META_DATA_APPWIDGET_PROVIDER);
if (parser == null) {
Slog.w(TAG, "No " + AppWidgetManager.META_DATA_APPWIDGET_PROVIDER + " meta-data for "
+ "AppWidget provider '" + component + '\'');
return null;
}
上面这一段代码清楚地说明了我前面提到的问题
:决定一个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文件的定义:
<?xml version="1.0" encoding="utf-8"?>
<appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"
android:minWidth="100dp"
android:minHeight="50dp"
android:updatePeriodMillis="50"
android:initialLayout="@layout/appwidget_provider"
android:configure="com.yarin.android.Examples_09_07.Activity01"
>
</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列表。
讲到这里我们再会过头前面我提出的问题,什么是桌面插件,是不是已经解答了?不然,我说到这里其实只是讲了桌面插件应用,说到桌面插件应用与桌面插件的关系你可以理解为类与对象的关系,桌面插件应用是静态的类型,桌面插件是动态生成的实例,为了便于理解我后面统一称为桌面插件实例,那我们继续解答什么是桌面插件这一问题。既然是桌面插件,那我们就从桌面入手,分析桌面插件是如何生成和添加的。
欲知后事如何,请听下回分解。