widget机制

android中的widget是非常有意思的一个功能,widget不是运行在自己的进程中,通常是运行在桌面、锁屏等应用上,如何理解跨进程通信并更新widget界面,是理解widget的关键。

widget核心是一个广播接收器,AppWidgetProvider,它有几个核心的回调函数。

// 没接收一次广播消息就调用一次,使用频繁
public void onReceive(Context context, Intent intent) {
    super.onReceive(context, intent);
}

// 每次更新都调用一次该方法,使用频繁
public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) {
    super.onUpdate(context, appWidgetManager, appWidgetIds);
}

// 没删除一个就调用一次
public void onDeleted(Context context, int[] appWidgetIds) {
    super.onDeleted(context, appWidgetIds);
}

// 当该Widget第一次添加到桌面是调用该方法,可添加多次但只第一次调用
public void onEnabled(Context context) {
    super.onEnabled(context);
}

// 当最后一个该Widget删除是调用该方法,注意是最后一个
public void onDisabled(Context context) {
    super.onDisabled(context);
}

根据实际需要,监听对应事件,实现相应的方法,在相应接口中更新widget界面。

widget必须在androidmenifest.xml文件中配置插件信息,以便包管理服务识别。

    <receiver 
        android:label="@string/app_name_timewidget" 
        android:name="com.owl.mythou.TimeWidget">
        <intent-filter>
        <action android:name="android.appwidget.action.APPWIDGET_UPDATE"></action>
        </intent-filter>
        <meta-data 
          android:name="android.appwidget.provider" 
          android:resource="@xml/time_widget_config">
        </meta-data>
    </receiver>

widget的初始配置文件如下,需要指定插件的最小宽高以及其它可选项,刷新频率,默认效果图片等等。

    <appwidget-provider
        xmlns:android="http://schemas.android.com/apk/res/android"
        android:initialLayout="@layout/time_widget_layout"
        android:minWidth="286dip" 
        android:minHeight="142dip" 
        android:updatePeriodMillis="0">
    </appwidget-provider>
  • android:initialLayout 指定界面布局的Layout文件,和activity的Layout一样
  • android:minWidth 你的widget的最小宽度。根据Layout的单元格计算(72*格子数-2)
  • android:minHeigh 你的widget的最小高度。计算方式和minwidth一样。
  • android:updatePerioMillis 使用系统定时更新服务,单位毫秒。

widget的具体开发如上,但到底是如何影响到桌面显示,它们的包名不一样,进程也不一样。它们是如何进行跨进程通信。在理解widget之前,必须要理解android中的remoteviews是如何工作的。

remoteviews,顾名思义是远程view,但它不是一个真正的view,因为真正的view是无法跨进程通信的,remoteviews实现了parcelable接口,可以实现跨进程通信,可以将remoteviews理解为一个view的描述文件,其内部维护了一个action列表,每个action记录了用户调用的方法名,参数,最终实现反射调用,实现用户设置的ui效果。

widget应用维护着一个remoteviews,当widget需要更新时,必须通过widget服务刷新,以便通知桌面或锁屏实现界面更新。

widget更新时序图.png

逻辑调用时序图如上所示。AppWidgetHost有一个实现了IAppWidgetHost的callback内部对象,这样能就通过aidl方式实现跨进程通信了。具体调用如上,widget服务跨进程调用到widgethost当中,完成widgethostview的更新,这样桌面就实现了界面刷新。因为桌面在初始化的时候会将widgethostview添加到界面上。

widget机制,欢迎交流补充。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,916评论 25 708
  • 一、鸡汤 appwidget是android中小组件,我们经常说的widget其实是指的那些button、text...
    欢乐斗佛阅读 2,287评论 1 8
  • 概述 RemoteViews顾名思义就是远程View,它表示的是一个View结构,它可以在其他进程中显示,为了跨进...
    shenhuniurou阅读 24,346评论 1 20
  • 1. 引言 为什么会抽象出设计模式?设计模式是在众多人开发大型项目遇到问题,为了解决特定问题而抽象出来的解决方案。...
    LeeLeCoder阅读 358评论 0 0
  • 有些时候,你不得不承认,你对你身上某些方面越自信,你可能就越缺那些东西。所以,要明白,过去,现在和以后所经历的最终...
    皆然阅读 618评论 0 0