首先我们要先了解AppWidgetProvider这个类,其中主要的回调函数的方法有:
onUpdate,onDeleted,onDisabled, onEnabled,onReceive, 其中个人认为最重要的是onUpdate方法,如果只是简单的widget小组件只重写onUpdate组建就可以了
我们涉及的类还有
import android.app.PendingIntent;
import android.app.Service;
import android.appwidget.AppWidgetManager;
import android.appwidget.AppWidgetProvider;
import android.content.ComponentName;
import android.os.IBinder;
import android.widget.RemoteViews;
下面我们就以一个小组件为例分享widget的做法,这个小组件主要实现了显示系统的device信息,包括:Build.ID,Build.MODEL,时间等,这个主要是可以再布局文件中添加,或改变。
首先书写AndroidManifest.xml文件:
<application android:icon="@drawable/icon" android:label="@string/app_name"> <receiver android:name=".BuildWidget" android:label="@string/widget_name"> <intent-filter> <action android:name="android.appwidget.action.APPWIDGET_UPDATE" /> </intent-filter> <meta-data android:name="android.appwidget.provider" android:resource="@xml/widget_build" /> </receiver> <service android:name=".BuildWidget$UpdateService" /> </application>
其中BuildWidget是AppWidgetProvider的子类,并且通过<meta-data android:name="android.appwidget.provider" android:resource="@xml/widget_build" />去指出widget的布局文件
我们再看widget_build.xml文件:
<appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android" android:minWidth="72dip" android:minHeight="72dip" android:updatePeriodMillis="0" android:initialLayout="@layout/widget" />
其中android:initialLayout="@layout/widget"指出了布局文件为widget.xml.
这个widget.xml文件比较简单,主要是为了测试功能加上了几个textview
<TextView android:id="@+id/build_info" android:layout_width="wrap_content" android:layout_height="wrap_content" android:gravity="center" style="@style/Text.BuildInfo.Fancy" android:textSize="18dip" /> <TextView android:id="@+id/build_date" android:layout_width="wrap_content" android:layout_height="wrap_content" android:gravity="center" style="@style/Text.BuildInfo.Fancy" android:textSize="14dip" /> <TextView android:id="@+id/build_extra" android:layout_width="wrap_content" android:layout_height="wrap_content" android:gravity="center" style="@style/Text.BuildInfo.Fancy" />
res中的布局文件差不多已经完成了,然后我们就可以再onUpdate中去写相应的逻辑了。
一般的非UI逻辑我们不建议在UI主线程中去完成,因为这会导致ANR timeout产生的可能性。所以我们建立一个service响应点击widget的事件。
在BuildWidget中建立内部子类UpdateService继承自Service。
在onUpdate方法中启动service。
在service onStart方法中添加处理逻辑,添加widget中的UI元素,这里我们用到了RemoteView来创建UI
废话少说,大家来参详代码吧。
public class BuildWidget extends AppWidgetProvider { @Override public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) { // To prevent any ANR timeouts, we perform the update in a service context.startService(new Intent(context, UpdateService.class)); } public static class UpdateService extends Service { @Override public void onStart(Intent intent, int startId) { // Build the widget update RemoteViews updateViews = buildUpdate(this); // Push update for this widget to the home screen ComponentName thisWidget = new ComponentName(this, BuildWidget.class); AppWidgetManager manager = AppWidgetManager.getInstance(this); manager.updateAppWidget(thisWidget, updateViews); } public RemoteViews buildUpdate(Context context) { // Pick out month names from resources Resources res = context.getResources(); RemoteViews updateViews = new RemoteViews( context.getPackageName(), R.layout.widget); PendingIntent pendingIntent = PendingIntent.getActivity(context, 0 /* no requestCode */, new Intent("android.settings.DEVICE_INFO_SETTINGS"), 0 /* no flags */); updateViews.setOnClickPendingIntent(R.id.widget, pendingIntent); updateViews.setTextViewText(R.id.build_info, android.os.Build.ID); updateViews.setTextViewText(R.id.build_date, DateUtils.formatDateTime(context, android.os.Build.TIME, DateUtils.FORMAT_NUMERIC_DATE)); updateViews.setTextViewText(R.id.build_extra, android.os.Build.MODEL); return updateViews; } @Override public IBinder onBind(Intent intent) { // We don't need to bind to this service return null; } } }