更新widget,我所知道的大概有两种,一种是利用服务,一种是利用广播.
效果是,一个textview每秒显示一个当前时间.
点击button会打10086这个电话
下面贴上利用服务更新的步骤以及代码
1.首先要写一个service.用于产生时间,并传递到widget中
package com.example.widget;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;
import android.app.PendingIntent;
import android.app.Service;
import android.content.Intent;
import android.net.Uri;
import android.os.IBinder;
import android.view.View;
import android.widget.RemoteViews;
import android.widget.TextView;
public class widgetservice extends Service
{
private Timer timer;
private TimerTask timerTask=new TimerTask() {
public void run() {
/** 获取并格式化当前时间 */
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd-HH-mm-ss");
String time = sdf.format(new Date());
/*
* widgetProvider是一个继承与AppWidgetProvider的类.
* updateAppwidget用于获取从service中生成的时间
*/
widgetProvider.updateAppwidget(getApplicationContext(),time);
}
};
public void onCreate()
{
timer=new Timer();
timer.schedule(timerTask, 0,1000);
};
public void onDestroy() {
super.onDestroy();
timer.cancel();
timer = null;
}
public IBinder onBind(Intent intent) {
return null;
}
}
2.在AndroidMainfest.xml中声明此service
<!-- 声明service -->
<service android:name="com.example.widget.widgetservice"/>
3.在写一个类widgetProvider继承于AppWidgetProvider,这个也是widget的主类
package com.example.widget;
import android.app.PendingIntent;
import android.appwidget.AppWidgetManager;
import android.appwidget.AppWidgetProvider;
import android.content.Context;
import android.content.Intent;
import android.net.Uri;
import android.widget.RemoteViews;
public class widgetProvider extends AppWidgetProvider {
private static int widgetIds[];//用于保存widget的ID值,应为假如你在多个桌面添加widget.那么就会有多个widget,并一一对应多个widgetID
/**
* 用于从服务中调用此方法更新widget
* @param context
* @param time
*/
public static void updateAppwidget(Context context,String time)
{
//获取widget的布局
RemoteViews views=new RemoteViews(context.getPackageName(), R.layout.widget_layout);
views.setTextViewText(R.id.textviews, time);
/** 调用拨号程序 */
Intent intent = new Intent(Intent.ACTION_VIEW,Uri.parse("tel:10086"));
PendingIntent pendingIntent = PendingIntent.getActivity(context, 0, intent, 0);
//为按钮设置事件处理器;
views.setOnClickPendingIntent(R.id.button1, pendingIntent);
//获取widget管理
AppWidgetManager aWidgetManager=AppWidgetManager.getInstance(context);
//更新AppWidget
aWidgetManager.updateAppWidget(widgetIds, views);
}
//onReceiver 为接收广播时调用更新UI,一般这两个方法是比较常用的。
public void onReceive(Context context, Intent intent) {
// TODO Auto-generated method stub
super.onReceive(context, intent);
}
//onUpdate 为组件在桌面上生成时调用,并更新组件UI
public void onUpdate(Context context, AppWidgetManager appWidgetManager,
int[] appWidgetIds) {
widgetIds=appWidgetIds;//保存appwidgetid
context.startService(new Intent(context, widgetservice.class));//启动更新widget的service
super.onUpdate(context, appWidgetManager, appWidgetIds);
}
/**
* 第一次往桌面添加Widgets的时候才会被调用,
* 往后往桌面添加同类型的widgets时候不会被调用
*/
public void onEnabled(Context context) {
context.startService(new Intent(context, widgetservice.class));//启动更新widget的service
}
/** 当用户从桌面上删除widgets的时候被调用 */
public void onDeleted(Context context, int[] appWidgetIds) {
context.stopService(new Intent(context, widgetservice.class));//停止更新widget的service
}
}
4.在AndroidMainfes.xml中声明此widget
<receiver android:name="com.example.widget.widgetProvider" android:label="桌面时钟小插件">
<meta-data android:name="android.appwidget.provider" android:resource="@layout/widgets"/>
<intent-filter>
<action android:name="android.appwidget.action.APPWIDGET_UPDATE"/>
</intent-filter>
</receiver>
5.涉及到的布局文件
widget_layout.xml
<?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:orientation="vertical" >
<TextView
android:id="@+id/textviews"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="时间测试"
android:textSize="24sp"/>
<Button
android:id="@+id/button1"
style="?android:attr/buttonStyleSmall"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Button" />
</LinearLayout>
widgets.xml
<?xml version="1.0" encoding="UTF-8"?>
<appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"
android:minWidth="150dp"
android:minHeight="50dp"
android:initialLayout="@layout/widget_layout"
android:previewImage="@drawable/ic_launcher">
</appwidget-provider>
demo地址:http://yunpan.cn/QGWu5MC4zFkF2
另外一种更新widget就是利用广播:
1, 在AndroidManifest文件中注册新的intent-filter;
<intent-filter>
<actionandroid:name="jalen.android.APP_WIDGET_UPDATE"></action>
</intent-filter>
2, 使用getBroadcast()方法创建一个PendingIntent;
//创建一个Intent对象;
Intent intent=new Intent();
//为Intent对象设置指定的行为;
intent.setAction(UPDATE_ACTION);
//创建一个PendindIntent对象;
PendingIntent pendingIntent=PendingIntent.getBroadcast(context, 0, intent,0);
3, 为AppWidget中的控件注册处理器;
RemoteViews views = new RemoteViews(context.getPackageName(), R.layout.example_provider);
//为按钮设置事件处理器; views.setOnClickPendingIntent(R.id.testButton,pendingIntent);
//更新AppWidget
appWidgetManager.updateAppWidget(appWidgetIds, //用于指定被更新的Appwidget的ID;
views);
1. 在onReceive方法中接受广播消息;
if(intent.getAction().equals(UPDATE_ACTION))
{
//得到AppWidgetMangaer实例;
AppWidgetManager awm=AppWidgetManager.getInstance(context);
RemoteViews views = new RemoteViews(context.getPackageName(), R.layout.example_provider);
//更新AppWidget中控件的状态-------修改TextView的值;
views.setTextViewText(R.id.testView, "修改后的值");
//创建一个标识;
ComponentName componentName=new ComponentName(context, MyWidgetProvider.class);
//更新AppWidget
awm.updateAppWidget(componentName, views);
}
else{
super.onReceive(context, intent);
}