前面一节已经实现了一个简单的App Widget,这里将通过一个实例继续深入学习App Widget。
      首先继续了解下App Widget框架的主要的类:
      AppWidgetProvider:继承自BroadcastReceiver,在App Widget应用update,enable,disable和deleted时接受通知。其中onUpdate,onReceive是最常用到的方法。
     AppWidgetProviderInfo:描述AppWidget的大小,更新频率和初始界面等信息,以xml文件的形式存在于应用中的res/xml目录下。
    AppWidgetManager:负责管理AppWidget,向AppWidgetProvider发送通知。
    RemoteViews:一个可以在其他应用进程中运行的类,是构造AppWidget的核心。


    下面开始代码的编写,首先在res/xml下建立myappwidetprovider.xml、

[html]view plaincopyprint?

  1. <?xmlversion="1.0"encoding="utf-8"?>
  2. <appwidget-providerxmlns:android="http://schemas.android.com/apk/res/android"
[*]android:minWidth="100dp"

[*]android:minHeight="72dp"
[*]android:updatePeriodMillis="86400000"
[*]android:initialLayout="@layout/myappwidget"
[*]>
[*]</appwidget-provider>
<?xml version="1.0" encoding="utf-8"?><appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"    android:minWidth="100dp"    android:minHeight="72dp"    android:updatePeriodMillis="86400000"    android:initialLayout="@layout/myappwidget"    ></appwidget-provider>  上面分别是 定义widget的宽度,高度,更新周期,以及layout的widget布局。

 下面是我们的布局文件:

[html]view plaincopyprint?

  1. <?xmlversion="1.0"encoding="utf-8"?>
  2. <LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"
[*]android:layout_width="match_parent"

[*]android:layout_height="match_parent"
[*]android:background="@drawable/widget_bg1"
[*]android:gravity="center"
[*]android:id="@+id/layout"
[*]android:orientation="vertical">
[*]
[*]<TextView
[*]android:id="@+id/txtMonth"
[*]android:layout_width="wrap_content"
[*]android:layout_height="wrap_content"
[*]android:textColor="#000000"
[*]android:layout_margin="2dp"
[*]android:text=""/>
[*]<TextView
[*]android:id="@+id/txtDay"
[*]android:layout_width="wrap_content"
[*]android:layout_height="wrap_content"
[*]android:textColor="#990033"
[*]android:textSize="25dp"
[*]android:text=""/>
[*]<TextView
[*]android:id="@+id/txtWeekDay"
[*]android:layout_width="wrap_content"
[*]android:layout_height="wrap_content"
[*]android:layout_margin="2dp"
[*]android:textColor="#000000"
[*]android:text=""/>
[*]</LinearLayout>
<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"    android:layout_width="match_parent"    android:layout_height="match_parent"    android:background="@drawable/widget_bg1"    android:gravity="center"    android:id="@+id/layout"    android:orientation="vertical" >    <TextView        android:id="@+id/txtMonth"        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:textColor="#000000"        android:layout_margin="2dp"        android:text="" /><TextView        android:id="@+id/txtDay"        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:textColor="#990033"        android:textSize="25dp"        android:text="" /><TextView        android:id="@+id/txtWeekDay"        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:layout_margin="2dp"        android:textColor="#000000"        android:text="" /></LinearLayout>

  对应布局widget要求比较高,大家自行设计,更加美观的界面。
 接下来是我们的核心代码ExampleAppWidgetProvider类了:

[html]view plaincopyprint?

  1. import android.app.PendingIntent;  
  2. import android.appwidget.AppWidgetManager;  
  3. import android.appwidget.AppWidgetProvider;  
  4. import android.content.Context;  
  5. import android.content.Intent;  
  6. import android.text.format.Time;  
  7. import android.widget.RemoteViews;  
  8. import android.widget.Toast;  

  9. public class ExampleAppWidgetProvider extends AppWidgetProvider{  
  10.    private String[] months={"一月","二月","三月","四月","五月","六月","七月","八月","九月","十月","十一月","十二月"};  
[*]    private String[] days={"星期日","星期一","星期二","星期三","星期四","星期五","星期六"};  

[*]    @Override  
[*]    public void onUpdate(Context context, AppWidgetManager appWidgetManager,  
[*]            int[] appWidgetIds) {  
[*]        // TODO Auto-generated method stub  
[*]
[*]        RemoteViews remoteViews=new RemoteViews(context.getPackageName(), R.layout.myappwidget);  
[*]        Time time=new Time();  
[*]        time.setToNow();  
[*]        String month=time.year+" "+months[time.month];  
[*]        remoteViews.setTextViewText(R.id.txtDay, new Integer(time.monthDay).toString());  
[*]        remoteViews.setTextViewText(R.id.txtMonth, month);  
[*]        remoteViews.setTextViewText(R.id.txtWeekDay, days[time.weekDay]);  
[*]        Intent intent=new Intent("cn.com.karl.widget.click");  
[*]        PendingIntent pendingIntent=PendingIntent.getBroadcast(context, 0, intent, 0);  
[*]        remoteViews.setOnClickPendingIntent(R.id.layout, pendingIntent);  
[*]        appWidgetManager.updateAppWidget(appWidgetIds, remoteViews);  
[*]
[*]        super.onUpdate(context, appWidgetManager, appWidgetIds);  
[*]    }  
[*]
[*]    @Override  
[*]    public void onReceive(Context context, Intent intent) {  
[*]        // TODO Auto-generated method stub  
[*]        super.onReceive(context, intent);  
[*]        if(intent.getAction().equals("cn.com.karl.widget.click")){  
[*]            Toast.makeText(context, "点击了widget日历", 1).show();  
[*]        }  
[*]    }  
[*]}  
import android.app.PendingIntent;import android.appwidget.AppWidgetManager;import android.appwidget.AppWidgetProvider;import android.content.Context;import android.content.Intent;import android.text.format.Time;import android.widget.RemoteViews;import android.widget.Toast;public class ExampleAppWidgetProvider extends AppWidgetProvider{    private String[] months={"一月","二月","三月","四月","五月","六月","七月","八月","九月","十月","十一月","十二月"};    private String[] days={"星期日","星期一","星期二","星期三","星期四","星期五","星期六"};    @Override    public void onUpdate(Context context, AppWidgetManager appWidgetManager,            int[] appWidgetIds) {        // TODO Auto-generated method stub                RemoteViews remoteViews=new RemoteViews(context.getPackageName(), R.layout.myappwidget);        Time time=new Time();        time.setToNow();        String month=time.year+" "+months[time.month];        remoteViews.setTextViewText(R.id.txtDay, new Integer(time.monthDay).toString());        remoteViews.setTextViewText(R.id.txtMonth, month);        remoteViews.setTextViewText(R.id.txtWeekDay, days[time.weekDay]);        Intent intent=new Intent("cn.com.karl.widget.click");        PendingIntent pendingIntent=PendingIntent.getBroadcast(context, 0, intent, 0);        remoteViews.setOnClickPendingIntent(R.id.layout, pendingIntent);        appWidgetManager.updateAppWidget(appWidgetIds, remoteViews);                super.onUpdate(context, appWidgetManager, appWidgetIds);    }        @Override    public void onReceive(Context context, Intent intent) {        // TODO Auto-generated method stub        super.onReceive(context, intent);        if(intent.getAction().equals("cn.com.karl.widget.click")){            Toast.makeText(context, "点击了widget日历", 1).show();        }    }}

  上面代码忘记做注释了,在这类分别解释下,使用remoteViews类分别加载上来布局文件的相应ID设置好值,然后PendingIntent 这就没什么好解释的了。
 最后在manifest中加入:

[html]view plaincopyprint?

  1. <receiverandroid:name="ExampleAppWidgetProvider">
  2. <intent-filter>
[*]<actionandroid:name="android.appwidget.action.APPWIDGET_UPDATE"/>

[*]<actionandroid:name="cn.com.karl.widget.click">
[*]</action>
[*]</intent-filter>
[*]
[*]<meta-data
[*]android:name="android.appwidget.provider"
[*]android:resource="@xml/myappwidetprovider"/>
[*]</receiver>
<receiver android:name="ExampleAppWidgetProvider" >            <intent-filter >                <action android:name="android.appwidget.action.APPWIDGET_UPDATE" />                <action android:name="cn.com.karl.widget.click" >                </action>            </intent-filter>            <meta-data                android:name="android.appwidget.provider"                android:resource="@xml/myappwidetprovider" />        </receiver>
  这样就完成了,运行项目看一下载手机上运行的效果吧:


  上面就是我们自己定义的AppWidget显示效果,点击它:

  这里为了表示点击了它,使用了Toast打印信息,当然我们也可以点击它之后启动相应的Activity。
http://blog.csdn.net/wangkuifeng0118/article/details/7362660