Android中经常用到APP Widget,如时钟,天气预报等。
长按屏幕,在弹出的对话框中选择“窗口小部件”,然后就列出了可选择的小部件,这些小部件就是APP Widget。
本文开发一个APP Widget,在屏幕上显示当前的时间,并且每秒更新一次。
开发APP Widget需要以下三个xml文件。
(1)AndroidManifest.xml,这个是所有APP都有的文件,APP Widget的AndroidManifest.xml和其他的AndroidManifest.xml有所不同。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
|
<?xml version= "1.0"
encoding= "utf-8" ?>
<manifest xmlns:android= "http://schemas.android.com/apk/res/android"
package = "com.hzhi.time_widget"
android:versionCode= "1"
android:versionName= "1.0"
>
<uses-sdk
android:minSdkVersion= "8"
android:targetSdkVersion= "17"
/>
<application
android:allowBackup= "true"
android:icon= "@drawable/ic_launcher"
android:label= "@string/app_name"
android:theme= "@style/AppTheme"
>
<receiver android:name= "MyTime"
android:label= "MyTime" >
<intent-filter>
<action android:name= "android.appwidget.action.APPWIDGET_UPDATE"
/>
</intent-filter>
<meta-data android:name= "android.appwidget.provider"
android:resource= "@xml/my_time"
/>
</receiver>
<service android:name= "MyTime$MyService"
/>
</application>
</manifest> |
其中,一个Receiver就代表一个APP Widget,如果想在一个工程里面开发多个APP Widget,多写几个Receiver就可以。<receiver android:name="MyTime" android:label="MyTime">表明该APP Widget的名称和标签。
<action android:name="android.appwidget.action.APPWIDGET_UPDATE" />表明接收APPWIDGET_UPDATE,这是必须有的,否则APP Widget无法更新。
<meta-data android:name="android.appwidget.provider" android:resource="@xml/my_time" />指明了APP Widget的信息文件为@xml/my_time,这也是下面将介绍的。
(2)APP Widget的信息文件,该文件是APP Widget所特有的。在res文件夹下面新建一个xml文件夹,在里面新建一个my_time.xml文件,即是APP Widget的信息文件。
1
2
3
4
5
6
7
8
|
<?xml version= "1.0"
encoding= "utf-8" ?>
<appwidget-provider android:minWidth= "220dip"
android:minHeight= "146dip"
android:updatePeriodMillis= "0"
android:initialLayout= "@layout/my_time"
/> |
该文件设置了APP Widget的长和宽(长和宽的值应为74*n-2),以及更新间隔时间(android:updatePeriodMillis,本例使用Timer更新,所以设置为0),最后的android:initialLayout="@layout/my_time" 指明了APP Widget的布局文件,也就是显示在桌面上的布局。
(3)APP Widget的布局文件,本例为my_time.xml,基本和其他APP的布局文件一样,区别就在于整个布局文件只有一个TextView。
1
2
3
4
5
6
7
8
9
|
<?xml version= "1.0"
encoding= "utf-8" ?>
<TextView xmlns:android= "http://schemas.android.com/apk/res/android"
android:id= "@+id/TextView01"
android:layout_width= "fill_parent"
android:layout_height= "fill_parent"
android:gravity= "center"
android:text= "Starting..."
android:textColor= "#FFFF00"
android:textSize= "20pt"
/>
|
最后是java文件MyTime.java:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
|
package
com.hzhi.time_widget;
import
java.util.Date;
import
java.util.Timer;
import
java.util.TimerTask;
import
android.os.Bundle;
import
android.os.Handler;
import
android.os.IBinder;
import
android.os.Message;
import
android.app.Activity;
import
android.app.Service;
import
android.appwidget.AppWidgetManager;
import
android.appwidget.AppWidgetProvider;
import
android.content.ComponentName;
import
android.content.Context;
import
android.content.Intent;
import
android.util.Log;
import
android.view.Menu;
import
android.widget.RemoteViews;
import
android.widget.Toast;
public
class
MyTime extends
AppWidgetProvider {
Timer timer;
Context context;
//onUpdate
@Override
public
void
onUpdate(Context con, AppWidgetManager appWidgetManager, int [] appWidgetIds) {
context = con;
Intent intent = new
Intent(context, MyService. class );
timer = new
Timer();
timer.schedule(timertask, 0 , 1000 );
}
//MyService服务程序
public
static
class
MyService extends
Service {
@Override
public
void
onStart(Intent intent, int
startId) {
RemoteViews remoteViews = new
RemoteViews(getPackageName(), R.layout.my_time);
remoteViews.setTextViewText(R.id.TextView01, new
Date().toLocaleString());
ComponentName thisWidget = new
ComponentName( this , MyTime. class );
AppWidgetManager manager = AppWidgetManager.getInstance( this );
manager.updateAppWidget(thisWidget, remoteViews);
}
@Override
public
IBinder onBind(Intent intent) {
return
null ;
}
};
// Handler
private
Handler handler = new
Handler(){
public
void
handleMessage(Message msg){
Intent intent = new
Intent(context, MyService. class );
context.startService(intent);
}
};
private
TimerTask timertask = new
TimerTask(){
public
void
run(){
Message message = new
Message();
handler.sendMessage(message);
}
};
} |
该Java文件使用Timer,每秒钟开始一次服务MyService。
MyService获得当前的时间,并且以AppWidgetManager.updateAppWidget()方法更新APP Widget所显示的时间。
运行APP,并添加到桌面上。