我们这一片是接着上一篇来继续进行讲解开发插件,那么我们现在就来继续的讲解App Widget布局,那就不多说了,我们来看看内容吧:



       

创建App Widget布局



  你必须在XML中为你的App Widget定义一个初始布局并保存到项目的res/layout/ 目录下。你可以使用如下所列的视图对象来设计你的App Widget,但是在此之前,请先阅读并理解App Widget Design Guidelines.



  如果你熟悉在XML中声明布局,那么创建这个App Widget布局是很简单的。但是,你必须意识到那个App Widget布局是基于RemoteViews, 这并不支持所有类型的布局或视图小部件。



  一个RemoteViews对象(以及,相应的,一个App Widget)可以支持下面这个布局类:



Java代码:



[java]  view plain copy


1. FrameLayout  
2. LinearLayout  
3. RelativeLayout  
4. //以及下面的小部件类:  
5. AnalogClock  
6. Button  
7. Chronometer  
8. ImageButton  
9. ImageView  
10. ProgressBar  
11. TextView



      不支持这些类的派生。

  使用AppWidgetProvider类

  你必须通过在清单文件中使用< receiver>元素来声明你的AppWidgetProvider 类实现为一个广播接收器(参见上面的Declaring an App Widget in the Manifest)。

  AppWidgetProvider 类扩展BroadcastReceiver 为一个简便类来处理App Widget广播。AppWidgetProvider只接收和这个App Widget相关的事件广播,比如这个App Widget被更新,删除,启用,以及禁用。当这些广播事件发生时,AppWidgetProvider 将接收到下面的方法调用:

Java代码:


[java]  view plain copy


1. onUpdate(Context, AppWidgetManager, int[])


       这个方法调用来间隔性的更新App Widget,间隔时间用AppWidgetProviderInfo 里的updatePeriodMillis属性定义。这个方法也会在用户添加App Widget时被调用,因此它应该执行基础的设置,比如为视图定义事件处理器并启动一个临时的服务Service,如果需要的话。但是,如果你已经声明了一个配置活动,这个方法在用户添加App Widget时将不会被调用,而只在后续更新时被调用。配置活动应该在配置完成时负责执行第一次更新。



       

onDeleted(Context, int[])


  当App Widget从宿主中删除时被调用。


  

onEnabled(Context)


  当一个App Widget实例第一次创建时被调用。比如,如果用户添加两个你的App Widget实例,只在第一次被调用。如果你需要打开一个新的数据库或者执行其他对于所有的App Widget实例只需要发生一次的设置,那么这里是完成这个工作的好地方。


  

onDisabled(Context)


  当你的App Widget的最后一个实例被从宿主中删除时被调用。你应该在onEnabled(Context)中做一些清理工作,比如删除一个临时的数据库。


  

onReceive(Context, Intent)


  这个接收到每个广播时都会被调用,而且在上面的回调函数之前。你通常不需要实现这个方法,因为缺省的AppWidgetProvider 实现过滤所有AppWidget 广播并恰当的调用上述方法。




  

注意:  在Android 1.5中, 有一个已知问题,onDeleted()方法在该调用时不被调用。为了规避这个问题,你可以像Group post中描述的那样实现onReceive() 来接收这个onDeleted()回调。



        最重要的AppWidgetProvider 回调函数是onUpdated(), 因为它是在每个App Widget添加进宿主时被调用的(除非你使用一个配置活动)。如果你的App Widget 要接受任何用户交互事件,那么你需要在这个回调函数中注册事件处理器。如果你的App Widget不创建临时文件或数据库,或者执行其它需要清理的工作,那么onUpdated() 可能是你需要定义的唯一的回调函数。比如,如果你想要一个带一个按钮的App Widget,当点击时启动一个活动,你可以使用下面的AppWidgetProvider实现:



Java代码:



[java]  view plain copy


1. public class ExampleAppWidgetProvider extends AppWidgetProvider {  
2. public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) {  
3.   
4. final int N = appWidgetIds.length;  
5. for (int i=0; i< N; i++) {  
6. int appWidgetId = appWidgetIds[i];  
7. Intent intent = new Intent(context, ExampleActivity.class);  
8.   
9. PendingIntent pendingIntent = PendingIntent.getActivity(context, 0, intent, 0);  
10. RemoteViews views = new RemoteViews(context.getPackageName(),R.layout.appwidget_provider_layout);  
11.   
12. views.setOnClickPendingIntent(R.id.button, pendingIntent);  
13. appWidgetManager.updateAppWidget(appWidgetId, views);  
14. }  
15.   
16. }  
17.   
18. }



        这个AppWidgetProvider 仅定义了onUpdated() 方法,为了定义一个PendingIntent,来启动一个活动并使用setOnClickPendingIntent(int, PendingIntent)方法把它附着到这个App Widget的按钮上。注意它包含了一个遍历appWidgetIds中所有项的循环,这是一个IDs数组,每个ID用来标识由这个Provider创建的一个App Widget。这样,如果用户创建多于一个这个App Widget的实例,那么它们将被同步更新。不过,对于所有的App Widget实例,只有一个updatePeriodMillis 时间表被管理。比如,如果这个更新时间表被定义为每隔两个小时,而且App Widget的第二个实例是在第一个后面一小时添加的,那么它们将按照第一个所定义的周期来更新而第二个被忽略。