最近需要编写一个日期时间的桌面Widget用来关联日历程序,以前很少写桌面Widget。对这方面技术不是很熟悉,今天花时间重新整理了一下,顺便把编写一个简单时间日期程序过程记录下来。
桌面Widget其实就是一个显示一些信息的工具(现在也有人开发了一些有实际操作功能的widget。例如相机widget,可以直接桌面拍照)。不过总的来说,widget主要功能就是显示一些信息。我们今天编写一个很简单的作为widget,显示时间、日期、星期几等信息。需要显示时间信息,那就需要实时更新,一秒或者一分钟更新一次。
这个时间Widget我是参考(Android应用开发揭秘)书里面的一个demo例子做的,只是把功能和界面完善了一下。下面是这次的效果图:
1、继承AppWidgetProvider
我们编写的桌面Widget需要提供数据更新,这里就需用用到AppWidgetProvider,它里面有一些系统回调函数。提供更新数据的操作。AppWidgetProvider是BrocastReceiver的之类,也就是说它其实本质是一个广播接收器。下面我们看看AppWidgetProvider的几个重要的回调方法:
//Edited by mythou
//http://www.cnblogs.com/mythou/
public class WidgetProvider extends AppWidgetProvider { private static final String TAG="mythou_Widget_Tag"; // 没接收一次广播消息就调用一次,使用频繁 public void onReceive(Context context, Intent intent) { Log.d(TAG, "mythou--------->onReceive"); super.onReceive(context, intent); } // 每次更新都调用一次该方法,使用频繁 public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) { Log.d(TAG, "mythou--------->onUpdate"); super.onUpdate(context, appWidgetManager, appWidgetIds); } // 没删除一个就调用一次 public void onDeleted(Context context, int[] appWidgetIds) { Log.d(TAG, "mythou--------->onDeleted"); super.onDeleted(context, appWidgetIds); } // 当该Widget第一次添加到桌面是调用该方法,可添加多次但只第一次调用 public void onEnabled(Context context) { Log.d(TAG, "mythou--------->onEnabled"); super.onEnabled(context); } // 当最后一个该Widget删除是调用该方法,注意是最后一个 public void onDisabled(Context context) { Log.d(TAG, "mythou--------->onDisabled"); super.onDisabled(context); } }
其中我们比较常用的是onUpdate和onDelete方法。我这里刷新时间使用了一个Service,因为要定时刷新服务,还需要一个Alarm定时器服务。下面给出我的onUpdate方法:
//Edited by mythou
//http://www.cnblogs.com/mythou/
@Override public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) { super.onUpdate(context, appWidgetManager, appWidgetIds); Time time = new Time(); time.setToNow(); //使用Service更新时间 Intent intent = new Intent(context, UpdateService.class); PendingIntent pendingIntent = PendingIntent.getService(context, 0, intent, 0); //使用Alarm定时更新界面数据 AlarmManager alarm = (AlarmManager)context.getSystemService(Context.ALARM_SERVICE); alarm.setRepeating(AlarmManager.RTC, time.toMillis(true), 60*1000, pendingIntent); }
2、AndroidManifest.xml配置
//Edited by mythou
//http://www.cnblogs.com/mythou/
<application android:icon="@drawable/icon" android:label="@string/app_name"> <!-- AppWidgetProvider的注册 mythou--> <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> <!-- 更新时间的后台服务 mythou--> <service android:name="com.owl.mythou.UpdateService"></service> </application>
AndroidManifest主要是配置一个receiver,因为AppWidgetProvider就是一个广播接收器。另外需要注意的是,里面需要提供一个action,这个是系统的更新widget的action。还有meta-data里面需要指定widget的配置文件。这个配置文件,需要放到res\xml目录下面,下面我们看看time_widget_config.xml的配置
3、appWidget配置:
//Edited by mythou
//http://www.cnblogs.com/mythou/
<?xml version="1.0" encoding="utf-8"?> <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一样。(对这个不了解可以看我Launcher分析文章)
- android:updatePerioMillis 使用系统定时更新服务,单位毫秒。
这里需要说明android:updatePerioMillis的问题,系统为了省电,默认是30分钟更新一次,如果你设置的值比30分钟小,系统也是30分钟才会更新一次。对于我们做时间Widget来说,显然不靠谱。所以只能自己编写一个Alarm定时服务更新。
4、更新Widget的Service服务
//Edited by mythou
//http://www.cnblogs.com/mythou/
public class UpdateService extends Service { @Override public void onStart(Intent intent, int startId) { super.onStart(intent, startId); UpdateWidget(this); } private void UpdateWidget(Context context) { //不用Calendar,Time对cpu负荷较小 Time time = new Time(); time.setToNow(); int hour = time.hour; int min = time.minute; int second = time.second; int year = time.year; int month = time.month+1; int day = time.monthDay; String strTime = String.format("%02d:%02d:%02d %04d-%02d-%02d", hour, min, second,year,month,day); RemoteViews updateView = new RemoteViews(context.getPackageName(), R.layout.time_widget_layout); //时间图像更新 String packageString="org.owl.mythou"; String timePic="time"; int hourHbit = hour/10; updateView.setImageViewResource(R.id.hourHPic, getResources().getIdentifier(timePic+hourHbit, "drawable", packageString)); int hourLbit = hour%10; updateView.setImageViewResource(R.id.hourLPic, getResources().getIdentifier(timePic+hourLbit, "drawable", packageString)); int minHbit = min/10; updateView.setImageViewResource(R.id.MinuteHPic, getResources().getIdentifier(timePic+minHbit, "drawable", packageString)); int minLbit = min%10; updateView.setImageViewResource(R.id.MinuteLPic, getResources().getIdentifier(timePic+minLbit, "drawable", packageString)); //星期几 updateView.setTextViewText(R.id.weekInfo, getWeekString(time.weekDay+1)); //日期更新,根据日期,计算使用的图片 String datePic="date"; int year1bit = year/1000; updateView.setImageViewResource(R.id.Year1BitPic, getResources().getIdentifier(datePic+year1bit, "drawable", packageString)); int year2bit = (year%1000)/100; updateView.setImageViewResource(R.id.Year2BitPic, getResources().getIdentifier(datePic+year2bit, "drawable", packageString)); int year3bit = (year%100)/10; updateView.setImageViewResource(R.id.Year3BitPic, getResources().getIdentifier(datePic+year3bit, "drawable", packageString)); int year4bit = year%10; updateView.setImageViewResource(R.id.Year4BitPic, getResources().getIdentifier(datePic+year4bit, "drawable", packageString)); //月 int mouth1bit = month/10; updateView.setImageViewResource(R.id.mouth1BitPic, getResources().getIdentifier(datePic+mouth1bit, "drawable", packageString)); int mouth2bit = month%10; updateView.setImageViewResource(R.id.mouth2BitPic, getResources().getIdentifier(datePic+mouth2bit, "drawable", packageString)); //日 int day1bit = day/10; updateView.setImageViewResource(R.id.day1BitPic, getResources().getIdentifier(datePic+day1bit, "drawable", packageString)); int day2bit = day%10; updateView.setImageViewResource(R.id.day2BitPic, getResources().getIdentifier(datePic+day2bit, "drawable", packageString)); //点击widget,启动日历 Intent launchIntent = new Intent(); launchIntent.setComponent(new ComponentName("com.mythou.mycalendar", "com.mythou.mycalendar.calendarMainActivity")); launchIntent.setAction(Intent.ACTION_MAIN); launchIntent.addCategory(Intent.CATEGORY_LAUNCHER); launchIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED); PendingIntent intentAction = PendingIntent.getActivity(context, 0, launchIntent, 0); updateView.setOnClickPendingIntent(R.id.SmallBase, intentAction); AppWidgetManager awg = AppWidgetManager.getInstance(context); awg.updateAppWidget(new ComponentName(context, TimeWidgetSmall.class), updateView); } }
上面就是我的Service,因为我的界面时间和日期都是使用图片做的(纯属为了好看点)。所以多了很多根据时间日期计算使用的图片名字的代码,这些就是个人实际处理,这里不多说。
有一点需要说明的是RemoteViews
RemoteViews updateView = new RemoteViews(context.getPackageName(), R.layout.time_widget_layout);
从我们的界面配置文件生成一个远程Views更新的对象,这个可以在不同进程中操作别的进程的View。因为Widget是运行在Launcher的进程里面的,而不是一个独立的进程。这也是一种远程访问机制。最后就是加了一个点击桌面Widget启动一个程序的功能,也是使用了PendingIntent的方法。
编写一个桌面Widget主要就是这些步骤,最后补充一点,桌面Widget的界面布局只支持一部分android的标准控件,如果需要做复杂widget界面,需要自定义控件。这部分后面有时间再说~
Edited by mythou
原创博文,转载请标明出处:http://www.cnblogs.com/mythou/p/3177250.html
相关推荐
# Android Widget开发代码示例详细说明 ## 一、概览 ...通过以上对`SynWidgetProvider`的详细分析,我们可以了解到Widget开发的基本流程和技术要点,这对于理解Android Widget的工作机制是非常有帮助的。
在Android应用开发中,Widget是桌面小部件,它允许用户在主屏幕上与应用程序进行交互,无需实际打开应用程序。本项目“Android应用源码Widget炫酷特效 (宏基扇子型效果)”显然着重于展示一种独特且吸引人的Widget...
总的来说,"LoveNote"是一个实用的Android便签Widget示例,它涵盖了BroadcastReceiver、Widget布局、数据存储、用户交互和后台更新等多个核心概念,对初学者理解Android Widget开发具有很高的参考价值。通过学习这个...
本篇文章将深入探讨Android桌面时钟的源代码,揭示其背后的开发原理和技术要点。 一、Android Widget基础 在Android系统中,Widget是用户界面的一种特殊形式,允许开发者在主屏幕上创建可交互的小部件。Android...
【Android系统原理与开发要点详解】/底层 应用 框架 Android核心分析28篇,强烈推荐android初学者,android进阶者看看这个系列教程 Android应用开发者指南:性能优化 android开发教程合集(推荐新手看下这一季教程)...
### Android桌面组件(AppWidget)开发详解 #### 一、引言 随着移动设备的普及,Android平台成为开发者们关注的重点之一。在众多Android应用类型中,AppWidget因其独特的特性和功能,成为用户与应用程序交互的重要...
- **[Android桌面Widget开发要点解析(时间日期Widget)](https://example.com/article4)** - **[Android时间选择器、日期选择器实现代码](https://example.com/article5)** 通过上述步骤和注意事项,你应该能够成功...
综上所述,这款Android MP3播放器不仅具备了基本的音乐播放功能,还加入了一些高级特性,如桌面Widget、同步歌词显示等,这些都大大提升了用户体验。对于初学者来说,通过研究此项目的源码和功能实现,可以有效地...
这有助于实现快速重启已关闭但未彻底结束的进程,例如桌面Widget的即时响应能力。然而,这种机制也带来了内存管理上的挑战。 内存回收程序在Android中扮演关键角色,它自动检测并释放无用内存,确保系统的流畅运行...
总的来说,理解和掌握悬浮窗与桌面的实现技术是Android开发中的重要技能。悬浮窗的实现涉及到系统级别的窗口管理,而桌面封装则需要深入理解Android的启动器机制。同时,开发者应当谨慎使用这些技术,避免侵犯用户...
Flutter是Google推出的一种开源UI工具包,用于构建高性能、高保真、跨平台的应用程序,支持iOS、Android、Web以及即将支持的桌面平台。在这个"flutter开发app,项目源代码"中,我们可以深入理解Flutter的开发流程和...
AppWidget是Android桌面环境中的轻量级组件,它们由一系列XML布局文件定义,可以包含各种UI元素,如按钮、文本视图等。开发者需要提供一个`AppWidgetProvider`,这是一个继承自`BroadcastReceiver`的类,负责处理与...
【Android天气预报源代码解析】 ...以上是基于"android天气预报源代码"这一主题的常见技术要点,通过深入研究这个源代码,开发者不仅可以学习到Android应用开发的基础知识,还能掌握实际项目中的最佳实践。
开发者只需编写一次代码,即可将其编译为适用于 iOS 和 Android 的应用,这不仅节省了大量的时间和精力,还降低了维护成本。 ##### 2.2 高性能表现 为了确保应用能够达到接近原生应用的性能水平,Flutter 使用了...