Android Widget 开发详解(二) +支持listView滑动的widget
转载请标明出处:http://blog.csdn.net/sk719887916/article/details/47027263
不少开发项目中都会有widget功能,别小瞧了它,他也是android的七大组件之一,对widget陌生的朋友可以阅读下我的上篇文章< Android Widget工作原理详解(一)>关于内部的介绍,还没掌握的同学不要担心,开发AppWidget套路很简单,今天我们就实现一个可以加入listView滑动的widget,熟悉下一个普通widget的开发步骤。
一 创建AppWidgetProvider
remoteViews)来加载或更新widget布局,也可以通过onReceive()
@Override
public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) { // 获取Widget的组件名 ComponentName thisWidget = new ComponentName(context, MyAppListWidgetProvider.class); // 创建一个RemoteView RemoteViews remoteViews = new RemoteViews(context.getPackageName(), R.layout.my_widget_layout); // 把这个Widget绑定到RemoteViewsService Intent intent = new Intent(context, MyRemoteViewsService.class); intent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_ID, appWidgetIds[0]); // 设置适配器 remoteViews.setRemoteAdapter(R.id.widget_list, intent); // 设置当显示的widget_list为空显示的View remoteViews.setEmptyView(R.id.widget_list, R.layout.none_data); // 点击列表触发事件 Intent clickIntent = new Intent(context, MyAppListWidgetProvider.class); // 设置Action,方便在onReceive中区别点击事件 clickIntent.setAction(clickAction); clickIntent.setData(Uri.parse(clickIntent.toUri(Intent.URI_INTENT_SCHEME))); PendingIntent pendingIntentTemplate = PendingIntent.getBroadcast( context, 0, clickIntent, PendingIntent.FLAG_UPDATE_CURRENT); remoteViews.setPendingIntentTemplate(R.id.widget_list, pendingIntentTemplate); // 刷新按钮 final Intent refreshIntent = new Intent(context, MyAppListWidgetProvider.class); refreshIntent.setAction("refresh"); final PendingIntent refreshPendingIntent = PendingIntent.getBroadcast( context, 0, refreshIntent, PendingIntent.FLAG_UPDATE_CURRENT); remoteViews.setOnClickPendingIntent(R.id.button_refresh, refreshPendingIntent); // 更新Wdiget appWidgetManager.updateAppWidget(thisWidget, remoteViews); }
此方功类似广播的onReceive()用发,用开接收和处理广播,如果我们在manifest.xml注册了MyAppListWidgetProvider为一个appwidget,那么不必须为此广播加上widget标示,添加一action:<action android:name="android.appwidget.action.APPWIDGET_UPDATE" />,下面的 <meta-data>标签用来定义widget的属性,指定一个widget描述信息,具体释义请阅读 上篇widget原理详解文章,
<!-- Widget必须添加到manifest文件中,和Broadcaset Receiver一样使用“receiver”标签 --> <receiver android:name=".MyAppListWidgetProvider" > <!-- 此处设置Wdiget更新动作 --> <intent-filter> <action android:name="android.appwidget.action.APPWIDGET_UPDATE" /> </intent-filter> <!-- 此处设置Widget的描述资源res/xml/my_widget.xml --> <meta-data android:name="android.appwidget.provider" android:resource="@xml/widget_info" > </meta-data> </receiver>
onReceive里处理代码逻辑,比如我这里用来接收widget的用来更新我们在onUpdate()给刷新按钮定义的点击事件,处理刷新界面需求,
/** * 接收Intent */ @Override public void onReceive(Context context, Intent intent) { super.onReceive(context, intent); String action = intent.getAction(); if (action.equals("refresh")) { // 刷新Widget final AppWidgetManager mgr = AppWidgetManager.getInstance(context); final ComponentName cn = new ComponentName(context, MyAppListWidgetProvider.class); MyRemoteViewsFactory.mList.add("音乐"+i); // 这句话会调用RemoteViewSerivce中RemoteViewsFactory的onDataSetChanged()方法。 mgr.notifyAppWidgetViewDataChanged(mgr.getAppWidgetIds(cn), R.id.widget_list); } else if (action.equals(clickAction)) { // 单击Wdiget中ListView的某一项会显示一个Toast提示。 Toast.makeText(context, intent.getStringExtra("content"), Toast.LENGTH_SHORT).show(); } i=i+1; }
@Override public void onEnabled(Context context) { // TODO Auto-generated method stub super.onEnabled(context); Toast.makeText(context, "用户将widget添加桌面了", Toast.LENGTH_SHORT).show(); }
@Override public void onDeleted(Context context, int[] appWidgetIds) { // TODO Auto-generated method stub。 Toast.makeText(context, "用户将widget从桌面移除了", Toast.LENGTH_SHORT).show(); super.onDeleted(context, appWidgetIds); }
二 创建RemoteViewsFactory
public class MyRemoteViewsFactory implements RemoteViewsFactory { private final Context mContext; public static List<String> mList = new ArrayList<String>(); /* * 构造函数 */ public MyRemoteViewsFactory(Context context, Intent intent) { mContext = context; } /* * MyRemoteViewsFactory调用时执行,这个方法执行时间超过20秒回报错。 * 如果耗时长的任务应该在onDataSetChanged或者getViewAt中处理 */ @Override public void onCreate() { // 需要显示的数据 mList.add(""); for (int i = 0; i < 5; i++) { mList.add("item"+ i); } } /* * 当调用notifyAppWidgetViewDataChanged方法时,触发这个方法 * 例如:MyRemoteViewsFactory.notifyAppWidgetViewDataChanged(); */ @Override public void onDataSetChanged() { } /* * 这个方法不用多说了把,这里写清理资源,释放内存的操作 */ @Override public void onDestroy() { mList.clear(); } /* * 返回集合数量 */ @Override public int getCount() { return mList.size(); } /* * 创建并且填充,在指定索引位置显示的View,这个和BaseAdapter的getView类似 */ @Override public RemoteViews getViewAt(int position) { if (position < 0 || position >= mList.size()) return null; String content = mList.get(position); // 创建在当前索引位置要显示的View final RemoteViews rv = new RemoteViews(mContext.getPackageName(), R.layout.my_widget_layout_item); // 设置要显示的内容 rv.setTextViewText(R.id.widget_list_item_tv, content); // 填充Intent,填充在AppWdigetProvider中创建的PendingIntent Intent intent = new Intent(); // 传入点击行的数据 intent.putExtra("content", content); rv.setOnClickFillInIntent(R.id.widget_list_item_tv, intent); return rv; } /* * 显示一个"加载"View。返回null的时候将使用默认的View */ @Override public RemoteViews getLoadingView() { return null; } /* * 不同View定义的数量。默认为1(本人一直在使用默认值) */ @Override public int getViewTypeCount() { return 1; } /* * 返回当前索引的。 */ @Override public long getItemId(int position) { return position; } /* * 如果每个项提供的ID是稳定的,即她们不会在运行时改变,就返回true(没用过。。。) */ @Override public boolean hasStableIds() { return true; }
三 RemoteViewsService
具体地说,其子类RemoteViewsService是一个远程的服务适配器 可以请求RemoteViews,管理RemoteViews的服务。我们继承RemoteViewsService来获得一个视图工厂,
@TargetApi(Build.VERSION_CODES.HONEYCOMB) public class MyRemoteViewsService extends RemoteViewsService { @Override public RemoteViewsFactory onGetViewFactory(Intent intent) { return new MyRemoteViewsFactory(this.getApplicationContext(), intent); } }
四 增加widet基础属性配置
<?xml version="1.0" encoding="utf-8"?> <appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android" android:initialLayout="@layout/my_widget_layout" android:minHeight="120dp" android:minWidth="280dp" android:previewImage="@drawable/ic_launcher" android:resizeMode="horizontal|vertical" android:updatePeriodMillis="0" > <!-- sdk1.5之后updatePeriodMillis已失效,置为0,循环执行自行在代码中实现。 至于其他属性可以查一下。在其他随笔中我也给出了 --> </appwidget-provider>
2 新建widget资源文件xml
属性来指定。而widget描述信息我们在manifest.xml中 用<meta-data>标签用来指定。
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="200dp" android:background="@android:color/white" android:orientation="vertical" > <Button android:id="@+id/button_refresh" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_gravity="center_horizontal" android:background="@drawable/lite_widget_item_choosed_background_icon" android:textColor="@android:color/white" android:layout_marginTop="2dp" android:text="刷新" /> <ListView android:id="@+id/widget_list" android:layout_width="match_parent" android:layout_height="wrap_content" android:cacheColorHint="#00000000" android:scrollbars="none" /> <!-- 此处的ListView 可以换成StackView或者GridView --> </LinearLayout>
Android Widget 开发详解(二) +支持listView滑动的widget的更多相关文章
- Android WebView 开发详解(二)
转载请注明出处 http://blog.csdn.net/typename/article/details/39495409 powered by miechal zhao 概览: Androi ...
- Android USB 开发详解
Android USB 开发详解 先附上 Android USB 官方文档 Android通过两种模式支持各种 USB 外设和 Android USB 附件(实现Android附件协议的硬件):USB ...
- JMessage Android 端开发详解
目前越来越多的应用会需要集成即时通讯功能,这里就为大家详细讲一下如何通过集成 JMessage 来为你的 App 增加即时通讯功能. 首先,一个最基础的 IM 应用会需要有哪些功能? 用户注册 / 登 ...
- Android WebView 开发详解
Android WebView 开发详解 参见 http://blog.csdn.net/typename/article/details/39030091
- 《Android游戏开发详解》一1.7 控制流程第1部分——if和else语句
本节书摘来异步社区<Android游戏开发详解>一书中的第1章,第1.7节,译者: 李强 责编: 陈冀康,更多章节内容可以访问云栖社区"异步社区"公众号查看. 1.7 ...
- Android Studio 插件开发详解二:工具类
转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/78112856 本文出自[赵彦军的博客] 在插件开发过程中,我们按照开发一个正式的项 ...
- Android 时间日期Widget 开发详解
桌面Widget其实就是一个显示一些信息的工具(现在也有人开发了一些有实际操作功能的widget.例如相机widget,可以直接桌面拍照).不过总的来说,widget主要功能就是显示一些信息.我们今天 ...
- Android WebView 开发详解(一)
转载请注明出处 http://blog.csdn.net/typename/article/details/39030091 powered by meichal zhao 概览: Android ...
- Android WebView 开发详解(三)
转载请注明出处 http://blog.csdn.net/typename/article/details/40302351 powered by miechal zhao 概览 Android ...
随机推荐
- Go 语言多维数组
Go 语言支持多维数组,以下为常用的多维数组声明方式: var variable_name [SIZE1][SIZE2]...[SIZEN] variable_type 以下实例声明了三维的整型数组: ...
- Docker rancher 部署
Docker-rancher #环境 centos7.4 , Docker version 17.12.0-ce #下载docker镜像 docker pull mysql:5.7 docker pu ...
- 【机器学习】从SVM到SVR
注:最近在工作中,高频率的接触到了SVM模型,而且还有使用SVM模型做回归的情况,即SVR.另外考虑到自己从第一次知道这个模型到现在也差不多两年时间了,从最开始的腾云驾雾到现在有了一点直观的认识,花费 ...
- Rails报找不到sanitize和raw方法的解决
以下一段代码作用是对html字符串做过滤作用: sanitize(raw(content.split.map{ |s| wrap_long_string(s) }.join(' '))) 不过实际会报 ...
- Redis之(一)初识Redis
1.Redis概述 我们知道,内存是电脑主板上的存储部件,用于存储当前正在使用的数据和程序,CPU可以与内存直接沟通,所以访问速速非常高:而外存数据必须加载到内存以后程序才能使用.如果把CPU当做一个 ...
- linux下内存大小、起始地址的解析与修改
在实际的工作中,由于产品型号的不同,经常需要调整linux所管理的内存的大小,而内核在启动阶段,会两次去解析从uboot传递过来的关于内存的信息,具体如下: 一.解析从uboot传递过来的tag(在p ...
- Mac小技巧:快速查看指定应用程序的所有窗口
我们知道在Mac中快速在系统所有程序中切换得快捷键为: cmd + tab 不过有时我们需要快速查看某一个程序的所有窗口,那又该如何呢? 以下方法在MacOS 10.12中测试成功! Mac默认该功能 ...
- python 3 黑色魔法元类初探
最近读django源码,发现必须了解元类才能理解一些很神奇的行为. 发现元类实际上是控制class的创建过程.比如类B继承某个看似平淡无奇的类A之后,你在类B中定义的属性或方法可能会遭到彻底改变. 假 ...
- [Mysql]由Data truncated for column联想到的sql_mode配置
系统日志中出现了 ata truncated for column 'agent' at row 1 mysql出现这个问题的原因,无非就是字符集设置 或者是 字段过长导致的. mysql在初始化的时 ...
- Java安全套接字扩展——JSSE
上节已经介绍了SSL/TLS协议的通信模式,而对于这些底层协议,如果要每个开发者都自己去实现显然会带来不必要的麻烦,正是为了解决这个问题Java为广大开发者提供了Java安全套接字扩展--JSSE,它 ...