欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

AppWidget初步入门

程序员文章站 2022-06-08 22:24:26
...

AppWidget 就是HomeScreen上显示的小部件,提供直观的交互操作。通过在HomeScreen中长按,在弹出的对话框中选择Widget部件来进行创建,长 按部件后并拖动到垃圾箱里进行删除。同一个Widget部件可以同时创建多个。

AppWidget的实现主要涉及到以下类:
1AppWidgetProvider :继承自 BroadcastRecevier , 在AppWidget 应用 updateenabledisable delete 时接收通知。其中,onUpdateonReceive 是最常用到的方法,它们接收更新通知。


2AppWidgetProvderInfo:描述 AppWidget 的大小、更新频率和初始界面等信息,以XML 文件形式存在于应用的 res/xml/目录下。


3AppWidgetManger :负责管理 AppWidget ,向 AppwidgetProvider 发送通知。更新AppWidget的相关状态,获取已安装AppWidget的状态和其它相关AppWidget的状态

4RemoteViews :一个可以在其他应用进程中运行的类,向 AppWidgetProvider 发送通知。一个描述了可以在其他进程中显示的view层,它inflate于layout资源文件,并且这个类对提供一些基本操作修改view内容

一、首先需要提供一个定义了Widget界面布局的XML文件(位于res/layout/..),需要注意的是使用的组件必须是RemoteViews所支 持的,目前原生API中支持的组件如下:


FrameLayout
LinearLayout
RelativeLayout

AnalogClock
Button
Chronmeter
ImageButton
ImageView
ProgressBar
TextView
ViewFlipper
ListView
GridView
StackView
AdapterViewFlipper

*如果使用了除此之外的组件,则在Widget创建时会导致android.view.InflateExceptionn异常。


二、然后需要提供一个xml文件来定义Widget的基本属性,放置到res/xml/..目录下。
如果使用的是Eclipse可按如下操作:
1) res/目录下创建xml/目录
2)创建xml文件(名字可任意),选择类型为AppWidgetProvider
3)在弹出的便捷界面进行参数设置

主要设置的参数如下:
minWidth: 定义Wdiget组件的宽度
minHeight: 定义Wdiget组件的高度
updatePeriodMillis: 更新的时间周期
initialLayout: Widget的布局文件
configure: 如果需要在启动前先启动一个Activity进行设置,在这里给出Activity的完整类名(后面会说到,与一般Activity的实现有些许差别)

*Widget大小的计算 :(单元格数*74)-2API上说是为了防止像素计算时的整数舍入导致错所以-2...不是很明白

上代码,先上效果图:

AppWidget初步入门AppWidget初步入门

11、先在res\layout中新建一个 main.xml 的文件里面内容如下:这个XML文件是AppWidget的布局文件

AppWidget中只有一个布局文件就是一个Button

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical" >

    <Button 
        android:id="@+id/btn"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="飞鱼"
        />

</LinearLayout>

12、在res目录下新建一个xml目录,这个xml目录存放AppWidget的相关信息,在xml目录下新建一个 appwidgetinfo.xml 的xml文件

<?xml version="1.0" encoding="utf-8"?>
<appwidget-provider
  xmlns:android="http://schemas.android.com/apk/res/android"
  android:minWidth = "294dp"
  android:minHeight = "72dp"
  android:updatePeriodMillis = "86400000"
  android:initialLayout = "@layout/main"
  >
</appwidget-provider>

minWidth:AppWidget的宽度

minHeight:AppWidget的高度

updatePeriodMillis:更新频率。主要用于执行AppWidgetProvider中的Update()函数

initialLayout = "@layout/main":指明AppWidget的布局文件

13、新建一个MyWidget的类继承自AppWidgetProvider

public class MyWidget extends AppWidgetProvider {

//	private static final String TAG = "MyWidget";
	private static final String broadcastString = "com.fy.MyWidget";

	// 每当一个Widget被删除的时候调用此函数
	@Override
	public void onDeleted(Context context, int[] appWidgetIds) {
		// TODO Auto-generated method stub
		super.onDeleted(context, appWidgetIds);
	}

	// 当最后一个Widget被删除的时候调用此函数
	// 在此方法中一般情况下执行一些AppWidget的结束性操作,如删除一个临时数据库等操作。
	@Override
	public void onDisabled(Context context) {
		// TODO Auto-generated method stub
		super.onDisabled(context);
	}

	// 当第一个Widget被创建的时候调用此函数
	// 在此方法中一般情况下执行一些AppWidget的初始化的工作,如新建一个数据库或者一些操作等
	@Override
	public void onEnabled(Context context) {
		// TODO Auto-generated method stub
		super.onEnabled(context);
	}

	// 调用其它函数时先调用此函数
	// onReceive相当一个经理,当外界来任务了,将任务分发给其下的员工(onEnable,onDeleted,onDisabled,onUpdate)处理 
	@Override
	public void onReceive(Context context, Intent intent) {
		
		if(intent.getAction().equals(broadcastString)) {
			RemoteViews remotesViews = new RemoteViews(context.getPackageName(), R.layout.main);
			remotesViews.setTextViewText(R.id.btn, "FY");
			
			// 获取AppWidgetManager更新appWidget的实例
			AppWidgetManager appWidgetManager = AppWidgetManager.getInstance(context);
			
			// 组件名称
			ComponentName componentName = new ComponentName(context, MyWidget.class);
			appWidgetManager.updateAppWidget(componentName, remotesViews);
		}
		
		super.onReceive(context, intent);
	}

	// 在这里onUpdate执行的条件有两个:1、到了我们所设定的更新周期了;2、添加一个新的AppWidget。 
    // 在onUpdate中一般为控件定义事件处理或者开启一个临时的Service。 
    // 这里是为我们的button控件添加点击事件处理 
    // context表示这个接收器正在运行的环境 
    // appWManager表示AppWidget的管理器可通过调用updateAppWidget(ComponentName, RemoteViews)更改指定AppWidget的控件属性 
    // appWidgetIds表示Appwidget的编号,系统自动生成 
	@Override
	public void onUpdate(Context context, AppWidgetManager appWidgetManager,
			int[] appWidgetIds) {
		// 初始化Intent
		Intent intent = new Intent();
		intent.setAction(broadcastString);
		PendingIntent pendingIntent = PendingIntent.getBroadcast(context, 0, intent, 0);
		
		// 获取appWidget上的相关控件
		RemoteViews remoteViews = new RemoteViews(context.getPackageName(), R.layout.main);
		remoteViews.setOnClickPendingIntent(R.id.btn, pendingIntent);
		
		// 更新所有的appWidget
		appWidgetManager.updateAppWidget(appWidgetIds, remoteViews);
		
	}

14、在AndroidManifest.xml中注册处理这个广播的类

<application
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name" >
        <receiver android:name=".MyWidget">
            <intent-filter >
<!--                 系统发出的广播 -->
                <action android:name="android.appwidget.action.APPWIDGET_UPDATE" />
<!--                 程序发出的广播 -->
				<action android:name="com.fy.MyWidget"/>
            </intent-filter>
            <meta-data
                android:name="android.appwidget.provider"
                android:resource="@xml/appwidgetinfo"
                />
                
        </receiver>
    </application>

这样一个简单的Widget就完成了。其实想人人网的Widget还有腾讯微博的Widget都很复杂。这只是一个简单的入门。

我现在在分析android自带音乐播放器的MediaAppWidgetProvider.java,如果今天晚上能完成的话就发表出来。