android提供向桌面上放置一个远程布局的功能。最近学了一下,在这里记录一下学习笔记。
public class MyWidget extends AppWidgetProvider{ public static String TAG = "MyWidget"; @Override public void onReceive(Context context, Intent intent) { Log.i(TAG, "onReceive: "); super.onReceive(context, intent); } @Override public void onEnabled(Context context) { Log.i(TAG, "onEnabled: "); super.onEnabled(context); } @Override public void onDeleted(Context context, int[] appWidgetIds) { Log.i(TAG, "onDeleted: "); super.onDeleted(context, appWidgetIds); } @Override public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) { Log.i(TAG, "onUpdate: "); super.onUpdate(context, appWidgetManager, appWidgetIds); } @Override public void onRestored(Context context, int[] oldWidgetIds, int[] newWidgetIds) { Log.i(TAG, "onRestored: "); super.onRestored(context, oldWidgetIds, newWidgetIds); } @Override public void onAppWidgetOptionsChanged(Context context, AppWidgetManager appWidgetManager, int appWidgetId, Bundle newOptions) { Log.i(TAG, "onAppWidgetOptionsChanged: "); super.onAppWidgetOptionsChanged(context, appWidgetManager, appWidgetId, newOptions); } @Override public void onDisabled(Context context) { Log.i(TAG, "onDisabled: "); super.onDisabled(context); } }
1,首先要建一个类MyWidget继承AppWidgetProvider,通过查看他的源码可以知道它是一个BroadcastReceiver。所以要在清单文件当中注册。
它的action为:android.appwidget.action.APPWIDGET_UPDATE。
<receiver android:name=".MyWidget" android:icon="@mipmap/desk_play" android:label="音乐"> <intent-filter> <action android:name="android.appwidget.action.APPWIDGET_UPDATE" /> </intent-filter> <meta-data android:name="android.appwidget.provider" android:resource="@xml/appwidget_info" /> </receiver>
<meta-data>元素中定义了Widget的AppWidgetInfo的信息。appwidget_info.xml文件在res/xml文件夹下。内容是:
然后就可以在布局文件当中设置widget的显示布局。在Widget的布局要依赖RemoteViews ,RemoteViews对指定的控件进行支持<?xml version="1.0" encoding="utf-8"?> <appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android" android:minWidth="40dp" android:minHeight="40dp" android:updatePeriodMillis="86400000" android:previewImage="@drawable/play_ic" android:initialLayout="@layout/my_widget" android:resizeMode="horizontal|vertical" android:widgetCategory="home_screen|keyguard"> </appwidget-provider> <!-- <appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android" android:minWidth="40dp" android:minHeight="40dp" android:updatePeriodMillis="86400000" android:previewImage="@drawable/preview" android:initialLayout="@layout/example_appwidget" android:configure="com.example.android.ExampleAppWidgetConfigure" android:resizeMode="horizontal|vertical" android:widgetCategory="home_screen|keyguard" android:initialKeyguardLayout="@layout/example_keyguard"> </appwidget-provider> minWidth和minHeight是widget在桌面上的的最小的宽高 updatePeriodMillis是Widget的刷新时间,单位是:ms。如果要是需要频繁的更新。可以用AlarmManager,并且设置Type为ELAPSED_REALTIME或者RTC。 initialLayout是一个布局就是Widget的布局。 configure的是一个Activity。当向桌面上添加Widget的时候就会启动,用来提示这个Widget的作用 previewImage是当选择该Widget添加到桌面的时候的的显示图片。拖动widget向桌面的图片 resizeMode是widget在桌面上调整大小的时候的模式,水平方向,垂直方向。 widgetCategory是widget显示的位置,桌面或锁屏的界面 initialKeyguardLayout是锁屏的时候的显示布局 -->
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="match_parent" android:layout_height="match_parent" android:background="@drawable/play_ic"> </LinearLayout> <!-- Widget的布局中只支持一下的布局和控件。不支持这些控件的子类是不支持的: FrameLayout LinearLayout RelativeLayout GridLayout AnalogClock Button Chronometer ImageButton ImageView ProgressBar TextView ViewFlipper ListView GridView StackView AdapterViewFlipper -->
做完这些就可以在桌面上添加Widget了。
2,Widget的生命周期。
第一次创建一个Widget: 04-03 10:26:13.416 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onReceive: 04-03 10:26:13.416 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onEnabled: 04-03 10:26:13.416 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onReceive: 04-03 10:26:13.420 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onUpdate: 第二次创建一个Widget: 04-03 10:26:39.515 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onReceive: 04-03 10:26:39.515 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onUpdate: 第三次创建一个Widget 04-03 10:29:07.485 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onReceive: 04-03 10:29:07.485 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onUpdate: 修改Widget在桌面上的大小的时候 04-03 10:30:09.057 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onReceive: 04-03 10:30:09.057 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onAppWidgetOptionsChanged: 移除一个Widget 04-03 10:30:34.692 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onReceive: 04-03 10:30:34.692 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onDeleted: 再移除一个: 04-03 10:30:34.692 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onReceive: 04-03 10:30:34.692 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onDeleted: 当最后一个被移除的时候: 04-03 10:31:44.211 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onReceive: 04-03 10:31:44.211 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onDeleted: 04-03 10:31:44.219 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onReceive: 04-03 10:31:44.219 1300-1300/com.example.yyw.appwidgetdemo I/MyWidget: onDisabled:从上面的log可以看出:
onReceive在每一次的创建和移除都会调用。
onEnabled只有在第一的时候才会调用。
onUpdate在每次创建都会调用的。而且在android:updatePeriodMillis的时间到的时候也会调用。
onDeleted在每次移除一个Widget的时候会被调用。
onDisabled在全部移除Widget的时候会调用到。
onAppWidgetOptionsChanged会在改变Widget大小的时候调用。
3,添加桌面view的点击事件。
@Override public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) { for (int id : appWidgetIds ) { Log.i(TAG, "onUpdate: " + id); AppWidgetManager am = AppWidgetManager.getInstance(context); //Widget是在其他应用上面显示的所以要用RemoteViews来管理 RemoteViews remoteViews = new RemoteViews(context.getPackageName(),R.layout.my_widget); Intent intent = new Intent(context,MainActivity.class); PendingIntent pendingIntent = PendingIntent.getActivity(context,2016,intent,0); //设置点击图片的点击事件,打开一个Activity remoteViews.setOnClickPendingIntent(R.id.iv,pendingIntent); //把Widget更新 am.updateAppWidget(id,remoteViews); } // super.onUpdate(context, appWidgetManager, appWidgetIds); }