Android桌面組件Widget初探
Android桌面組件widget初探是本文要介紹的內(nèi)容,主要是來了解Android Widget 的應(yīng)用,本來打算晚上繼續(xù)ApiDemos系列的,不過今天下午的時(shí)候無聊去玩了一下桌面組件AppWidget覺得挺不錯(cuò)的一個(gè)東西,對(duì)它很是感興趣,玩了一下碰到很多問題,一直在解決問題到了晚上10點(diǎn)。只能怪自己理解不深刻,不過最后還是解決了,把一些領(lǐng)悟?qū)懗鰜硪菜憬o自己一個(gè)交代。下面是本篇的大綱:
1、AppWidget框架類
2、在Android如何使用Widget
3、AppWidget框架的主要類介紹
4、DEMO講解
1、AppWidget框架類
1、AppWidgetProvider:繼承自BroadcastRecevier,在AppWidget應(yīng)用update、enable、disable和delete時(shí)接收通知。其中,onUpdate、onReceive是最常用到的方法,它們接收更新通知。
2、AppWidgetProvderInfo:描述AppWidget的大小、更新頻率和初始界面等信息,以XML文件形式存在于應(yīng)用的res/xml/目錄下。
3、AppWidgetManger:負(fù)責(zé)管理AppWidget,向AppwidgetProvider發(fā)送通知。
4、RemoteViews:一個(gè)可以在其他應(yīng)用進(jìn)程中運(yùn)行的類,向AppWidgetProvider發(fā)送通知。
2、在Android如何使用Widget
1、長按主界面

2、之后彈出一個(gè)對(duì)話框,里面就有android內(nèi)置的一些桌面組件

3、AppWidget框架的主要類介紹
1)AppWidgetManger類
- bindAppWidgetId(intappWidgetId,ComponentNameprovider)
通過給定的ComponentName綁定appWidgetId
- getAppWidgetIds(ComponentNameprovider)
通過給定的ComponentName獲取AppWidgetId
- getAppWidgetInfo(intappWidgetId)
通過AppWidgetId獲取AppWidget信息
- getInstalledProviders()
返回一個(gè)List的信息
- getInstance(Contextcontext)
獲取AppWidgetManger實(shí)例使用的上下文對(duì)象
- updateAppWidget(int[]appWidgetIds,RemoteViewsviews)
通過appWidgetId對(duì)傳進(jìn)來的RemoteView進(jìn)行修改,并重新刷新AppWidget組件
- updateAppWidget(ComponentNameprovider,RemoteViewsviews)
通過ComponentName對(duì)傳進(jìn)來的RemoeteView進(jìn)行修改,并重新刷新AppWidget組件
- updateAppWidget(intappWidgetId,RemoteViewsviews)
通過appWidgetId對(duì)傳進(jìn)來的RemoteView進(jìn)行修改,并重新刷新AppWidget組件
2)繼承自AppWidgetProvider可實(shí)現(xiàn)的方法為如下:
1、
- onDeleted(Contextcontext,int[]appWidgetIds)
2、
- onDisabled(Contextcontext)
3、
- onEnabled(Contextcontext)
4、
- onReceive(Contextcontext,Intentintent)
Tip:因?yàn)锳ppWidgetProvider是繼承自BroadcastReceiver所以可以重寫onRecevie方法,當(dāng)然必須在后臺(tái)注冊(cè)Receiver
5、
- onUpdate(Contextcontext,AppWidgetManagerappWidgetManager,int[]appWidgetIds)
4、Demo講解
下面是我今天做的一個(gè)實(shí)例,提供給大家練習(xí)時(shí)做參考,效果如下:在布局中放一個(gè)TextView做桌面組件,然后設(shè)置TextView的Clickable="true"使其有點(diǎn)擊的功能,然后我們點(diǎn)擊它時(shí)改變它的字體,再點(diǎn)擊時(shí)變回來,詳細(xì)操作如下流程:
1、新建AppWidgetProvderInfo
2、寫一個(gè)類繼承自AppWidgetProvider
3、后臺(tái)注冊(cè)Receiver
4、使AppWidget組件支持點(diǎn)擊事件
5、如何使TextView在兩種文本間來回跳轉(zhuǎn)
問題拋出來了,那么一起解決它吧。
1、新建AppWidgetProvderInfo
代碼如下:
- android:minWidth="60dp"
- android:minHeight="30dp"
- android:updatePeriodMillis="86400000"
- android:initialLayout="@layout/main">
Tip:上文說過AppWidgetProvderInfo是在res/xml的文件形式存在的,看參數(shù)不難理解,比較重要的是這里android:initialLayout="@layout/main"此句為指定桌面組件的布局文件。
2、寫一個(gè)類繼承自AppWidgetProvider
主要代碼如下:
- public class widgetProvider extends AppWidgetProvider
并重寫兩個(gè)方法
- @Override
- public void onUpdate(Context context, AppWidgetManager appWidgetManager,
- int[] appWidgetIds) {}
- @Override
- public void onReceive(Context context, Intent intent) {}
Tip:onUpdate為組件在桌面上生成時(shí)調(diào)用,并更新組件UI,onReceiver為接收廣播時(shí)調(diào)用更新UI,一般這兩個(gè)方法是比較常用的。
3、后臺(tái)注冊(cè)Receiver
后臺(tái)配置文件代碼如下:
- <receiver android:name=".widgetProvider">
- <meta-data android:name="android.appwidget.provider"
- android:resource="@xml/appwidget_provider"></meta-data>
- <intent-filter>
- <action android:name="com.terry.action.widget.click"></action>
- <action android:name="android.appwidget.action.APPWIDGET_UPDATE" />
- </intent-filter>
- </receiver>
Tip:因?yàn)槭亲烂娼M件,所以暫時(shí)不考慮使用Activity界面,當(dāng)然你在實(shí)現(xiàn)做項(xiàng)目時(shí)可能會(huì)需要點(diǎn)擊時(shí)跳轉(zhuǎn)到Activity應(yīng)用程序上做操作,典型的案例為Android提供的音樂播放器。上面代碼中比較重要的是這一句大意為指定桌面應(yīng)用程序的AppWidgetProvderInfo文件,使其可作其管理文件。
4、使AppWidget組件支持點(diǎn)擊事件
先看代碼:
- public static void updateAppWidget(Context context,
- AppWidgetManager appWidgeManger, int appWidgetId) {
- rv = new RemoteViews(context.getPackageName(), R.layout.main);
- Intent intentClick = new Intent(CLICK_NAME_ACTION);
- PendingIntent pendingIntent = PendingIntent.getBroadcast(context, 0,
- intentClick, 0);
- rv.setOnClickPendingIntent(R.id.TextView01, pendingIntent);
- appWidgeManger.updateAppWidget(appWidgetId, rv);
- }
此方法為創(chuàng)建組件時(shí)onUpdate調(diào)用的更新UI的方法,代碼中使用RemoteView找到組件的布局文件,同時(shí)為其設(shè)置廣播接收器CLICK_NAME_ACTION并且通過RemoteView的setOnClickPendingIntent方法找到我想觸發(fā)事件的TextView為其設(shè)置廣播。接著
- @Override
- public void onReceive(Context context, Intent intent) {
- // TODO Auto-generated method stub
- super.onReceive(context, intent);
- if (rv == null) {
- rv = new RemoteViews(context.getPackageName(), R.layout.main);
- }
- if (intent.getAction().equals(CLICK_NAME_ACTION)) {
- if (uitil.isChange) {
- rv.setTextViewText(R.id.TextView01, context.getResources()
- .getString(R.string.load));
- } else {
- rv.setTextViewText(R.id.TextView01, context.getResources()
- .getString(R.string.change));
- }
- Toast.makeText(context, Boolean.toString(uitil.isChange),
- Toast.LENGTH_LONG).show();
- uitil.isChange = !uitil.isChange;
- }
- AppWidgetManager appWidgetManger = AppWidgetManager
- .getInstance(context);
- int[] appIds = appWidgetManger.getAppWidgetIds(new ComponentName(
- context, widgetProvider.class));
- appWidgetManger.updateAppWidget(appIds, rv);
- }
在onReceiver中通過判斷傳進(jìn)來的廣播來觸發(fā)動(dòng)作。
5、如何使TextView在兩種文本間來回跳轉(zhuǎn)
如何TextView在來兩種狀態(tài)中來回呢?這也是我比較調(diào)試最久的一個(gè)難點(diǎn),問題出在對(duì)AppWidget的理解不夠深入。如果我的設(shè)想沒錯(cuò)的話AppWidget的生命周期應(yīng)該在每接收一次廣播執(zhí)行一次為一個(gè)生命周期結(jié)束,也就是說你在重寫的AppWidgetProvider類里面聲明全局變量做狀態(tài)判斷,每次狀態(tài)改變AppWidgetProvider再接收第二次廣播時(shí)即為你重新初始化也就是說桌件為你重新實(shí)例化了一次AppWidgetProvider。今天我因?yàn)樵诶锩娣帕艘粋€(gè)boolean值初始化為true,觀察調(diào)試看到每次進(jìn)入都為TRUE故你在設(shè)置桌面組件時(shí),全局變量把它聲明在另外一個(gè)實(shí)體類用來判斷是沒問題的,切忌放在本類。代碼參考o(jì)nReceiver方法。
效果圖如下:

代碼:
代碼
- package com.terry;
- import android.app.PendingIntent;
- import android.appwidget.AppWidgetManager;
- import android.appwidget.AppWidgetProvider;
- import android.content.ComponentName;
- import android.content.Context;
- import android.content.Intent;
- import android.widget.RemoteViews;
- import android.widget.Toast;
- public class widgetProvider extends AppWidgetProvider {
- private static final String CLICK_NAME_ACTION = "com.terry.action.widget.click";
- private static RemoteViews rv;
- @Override
- public void onUpdate(Context context, AppWidgetManager appWidgetManager,
- int[] appWidgetIds) {
- // TODO Auto-generated method stub
- final int N = appWidgetIds.length;
- for (int i = 0; i < N; i++) {
- int appWidgetId = appWidgetIds[i];
- updateAppWidget(context, appWidgetManager, appWidgetId);
- }
- }
- @Override
- public void onReceive(Context context, Intent intent) {
- // TODO Auto-generated method stub
- super.onReceive(context, intent);
- if (rv == null) {
- rv = new RemoteViews(context.getPackageName(), R.layout.main);
- }
- if (intent.getAction().equals(CLICK_NAME_ACTION)) {
- if (uitil.isChange) {
- rv.setTextViewText(R.id.TextView01, context.getResources()
- .getString(R.string.load));
- } else {
- rv.setTextViewText(R.id.TextView01, context.getResources()
- .getString(R.string.change));
- }
- Toast.makeText(context, Boolean.toString(uitil.isChange),
- Toast.LENGTH_LONG).show();
- uitil.isChange = !uitil.isChange;
- }
- AppWidgetManager appWidgetManger = AppWidgetManager
- .getInstance(context);
- int[] appIds = appWidgetManger.getAppWidgetIds(new ComponentName(
- context, widgetProvider.class));
- appWidgetManger.updateAppWidget(appIds, rv);
- }
- public static void updateAppWidget(Context context,
- AppWidgetManager appWidgeManger, int appWidgetId) {
- rv = new RemoteViews(context.getPackageName(), R.layout.main);
- Intent intentClick = new Intent(CLICK_NAME_ACTION);
- PendingIntent pendingIntent = PendingIntent.getBroadcast(context, 0,
- intentClick, 0);
- rv.setOnClickPendingIntent(R.id.TextView01, pendingIntent);
- appWidgeManger.updateAppWidget(appWidgetId, rv);
- }
- }
小結(jié):Android桌面組件widget初探的內(nèi)容介紹完了,希望通過Android Widget 的應(yīng)用內(nèi)容的學(xué)習(xí)能對(duì)你有所幫助!