先简单说说Widget的原理。Widget是在桌面上的一块显示信息的东西,也通过单击Widget跳转到一个程序里面。而系统自带的程序,典型的Widget是music,这个Android内置的音乐播放小程序。这个是典型的Widget+app应用。就是一个程序既可以通过Widget启动,也可以通过App启动。Widget就是一个AppWidgetProvider+一个UI界面显示(预先绑定了好多Intent),界面上的信息可以通过程序控制而改变,单击Widget,上的控件只能激发发送一个Intent,或发出一个Service的启动通知。而AppWidgetProvider可以拦截这个Intent,而进行相应的处理(比如显示新的信息)。Android开发里要大量的通过手动的方式配置好多xml文件。这对于.net开发来说不得不说是个梦靥呀。这也许也是一般的java程序员比.net程序的工资高的一个原因吧。毕竟做Java开发,特别是Android开发,确实很累。你可以看下源码对照着源码进行讲解。我们先开发一个比较简单的Widget应用,实现的主要功能是可以通过的不断变化,而不断的显示当前时间。首先,要自己手动建一个名为xml的文件夹。建一个xml文件,加入如下代码:
<appwidget-providerxmlns:android="http://schemas.android.com/apk/res/android" android:minHeight="72px" android:minWidth="72px" android:updatePeriodMillis="3800000"android:initialLayout="@layout/main"> </appwidget-provider>
public class TestAppWidget extends AppWidgetProvider { private static final String TAG="TestAppWidget"; private static final String FRESH="com.sinxiao.app.fresh"; private Context mContext ; private boolean run = true ; BroadcastReceiver mBroadcast =newBroadcastReceiver() { public void onReceive(Contextcontext, Intent intent) { String action =intent.getAction(); if(action.equals(Intent.ACTION_TIME_TICK)) { mContext.sendBroadcast(newIntent(FRESH)); } } }; /** * 通知Widget每个1秒刷新一次 */ Thread myThread = new Thread(){ public void run() { while (run) { try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } mContext.sendBroadcast(newIntent(FRESH));//通知刷新Widget的Intent } }; }; @Override public void onUpdate(Contextcontext, AppWidgetManager appWidgetManager, int[] appWidgetIds) { // 用来给Widget刷新界面显示 Log.d(TAG,"onUpdate"); super.onUpdate(context,appWidgetManager, appWidgetIds); mContext = context; RemoteViews views = newRemoteViews(context.getPackageName(),R.layout.main); Calendar cal=Calendar.getInstance(); System.out.println(cal.getTime().toLocaleString()); views.setTextViewText(R.id.txttim,cal.getTime().toLocaleString()); appWidgetManager.updateAppWidget(appWidgetIds,views); myThread.start(); /** * 本类作为一个bracastReveiver能自己再,注册个监听器 (可以取消注释,看报什么错误) */ // context.registerReceiver(mBroadcast,new IntentFilter(Intent.ACTION_TIME_TICK)); } @Override public void onReceive(Contextcontext, Intent intent) { Log.d(tag,"onReceive"); String action =intent.getAction(); Log.d(tag, "theaction is "+action); if (FRESH.equals(action)){ showTime(context); }elseif(Intent.ACTION_TIME_TICK.equals(action)){ showTime(context); } super.onReceive(context,intent); } private void showTime(Contextcontext) { RemoteViews views = newRemoteViews(context.getPackageName(),R.layout.main); Calendar cal=Calendar.getInstance(); views.setTextViewText(R.id.txttim,cal.getTime().toLocaleString()); ComponentName thisWidget =new ComponentName(context,TestAppWidget.class); AppWidgetManager.getInstance(context).updateAppWidget(thisWidget,views); } public void onDisabled(Contextcontext) { Log.d(tag,"onDisabled"); super.onDisabled(context); run = false ; } }
private static final StringFRESH="com.sinxiao.app.fresh"; public void onReceive(Contextcontext, Intent intent) { String action=intent.getAction(); Log.d(tag, "theaction is "+action); if (FRESH.equals(action)){ showTime(context); }elseif(Inent.ACTION_TIME_TICK.equals(action)){ showTime(context); } super.onReceive(context,intent); }
<?xml version="1.0"encoding="utf-8"?> <manifestxmlns:android="http://schemas.android.com/apk/res/android" package="com.sinxiao.widgetapp" android:versionCode="1"android:versionName="1.0.0"> <applicationandroid:icon="@drawable/icon"android:label="@string/app_name"> <receiver android:name=".setting.TestAppWidget"> <intent-filter> <actionandroid:name="android.appwidget.action.APPWIDGET_UPDATE"/> <!-- 这个Intent不支持在配置文件里的注册 所以这个是没用的 --> <!-- <actionandroid:name ="android.intent.action.TIME_TICK"/> --> </intent-filter> <intent-filter> <!-- 将IntentAction 手动配置在 mainset文件上 --> <action android:name="com.sinxiao.app.fresh"/> </intent-filter> <meta-data android:name="android.appwidget.provider" android:resource="@xml/testwidget_setting" /> </receiver> </application> </manifest>
RemoteViews views = newRemoteViews(context.getPackageName(),R.layout.main); Calendar cal=Calendar.getInstance(); System.out.println(cal.getTime().toLocaleString()); views.setTextViewText(R.id.txttim,cal.getTime().toLocaleString()); appWidgetManager.updateAppWidget(appWidgetIds,views);
ComponentName thisWidget = new ComponentName(context,TestAppWidget.class); AppWidgetManager.getInstance(context).updateAppWidget(thisWidget,views);