来自:http://blog.csdn.net/silenceburn/article/details/6093074

桌面便签软件是android上常用软件的一种,比如比较早的Sticky Note,就曾非常流行,

Sticky Note的介绍可以参见 http://www.tompda.com/c/article/11778/

而实际上使用android平台对widget开发的支持,桌面便签类软件是非常易于开发的。

本文通过逐步实现一个简单的桌面便签软件,和大家分享进行widget开发的过程和方法。


1.MyNote的最终实现效果

为了提起大家阅读本文的兴趣,先介绍一下最终实现的效果。

首先可以通过桌面增加我们的MyNote小部件,如下图所示:

图中的“我的便签”就是我们之后将要开发的便签程序。

点击后启动添加日志界面,如下图所示:

输入便签内容后,可以点击下面所列的四种图标之一作为便签图标。

比如点击第一个后,桌面上就会添加一个便签:

点击桌面上的便签,可以再次对便签内容进行修改,并更换图标。

桌面上可以同时存在多个便签,并可以分别进行修改。

如下图所示,我们将刚才创建的便签的图标修改一下,并新增了一个便签:

每个便签的内容都是分别独立保存的,可以随时点击桌面图标修改。


2.开发方式

开发的目的和追求的效果已经十分清楚了,首先我们确定一下开发方式。

在本文中,将采取一种渐进式的开发,也就是说不会一口气从头做到尾。

而是分为好几个阶段。每个阶段都完成一定的目标,然后下个阶段增加更多的功能,

每个阶段都离最终目标更进一步,OK,你可以说这是一次敏捷开发 :)

第一个阶段,首先我们会搭建一个widget原型程序,

它是完全可以运行的,可以创建桌面widget。

第二个阶段,我们改进 widget 配置Activity 部分的实现

使其具备创建便签的功能

第三个阶段,我们改进 widget 点击响应部分的实现,

使其具备修改便签的功能


3.搭建widget原型程序

本节我们会做一个最简单的widget程序原型,但是它是可以运行的。

一般来说 widget 程序由以下部分组成:

a. AppWidgetProvider 的实现

b. widget外观布局定义文件

c. 新增widget时的配置Activity的实现(可选)

d. widget 参数配置文件

以下分别讲解

a. AppWidgetProvider 的实现

首先我们新建一个android工程起名为MyNote,然后修改 MyNote.java 的代码,

使MyNote继承自 AppWidgetProvider ,并重写 onUpdate 和 onDeleted 方法。

其中onUpdate 会在widget创建及被更新时调用, onDeleted 会在widget被删除时调用。

目前我们不需要在这里实现任何功能,只是简单的记录日志以便我们观察其运行,编写好的代码如下:

[java] view plaincopy
  1. package com.silenceburn;
  2. import android.appwidget.AppWidgetManager;
  3. import android.appwidget.AppWidgetProvider;
  4. import android.content.Context;
  5. import android.util.Log;
  6. public class MyNote extends AppWidgetProvider {
  7. /** Called when the activity is first created. */
  8. final String mPerfName = "com.silenceburn.MyColorNoteConf";
  9. @Override
  10. public void onUpdate(Context context, AppWidgetManager appWidgetManager,
  11. int[] appWidgetIds) {
  12. // TODO Auto-generated method stub
  13. final int N = appWidgetIds.length;
  14. for (int i = 0; i < N; i++) {
  15. int appWidgetId = appWidgetIds[i];
  16. Log.i("myLog", "this is [" + appWidgetId + "] onUpdate!");
  17. }
  18. }
  19. @Override
  20. public void onDeleted(Context context, int[] appWidgetIds) {
  21. // TODO Auto-generated method stub
  22. final int N = appWidgetIds.length;
  23. for (int i = 0; i < N; i++) {
  24. int appWidgetId = appWidgetIds[i];
  25. Log.i("myLog", "this is [" + appWidgetId + "] onDelete!");
  26. }
  27. }
  28. }

b. widget外观布局定义文件

我们需要为widget编写一个外观布局文件,在本示例中,布局非常简单,只需要一个imageView即可

编写好的 my_note_widget.xml 文件如下:

[xhtml] view plaincopy
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <ImageView xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:id="@+id/my_widget_img"
  4. android:layout_width="wrap_content"
  5. android:layout_height="wrap_content"
  6. android:src="@drawable/sketchy_paper_008"
  7. android:clickable="true"/>

这里用到了一个外部图片 sketchy_paper_008.png,来源于网络,感谢图片原作者。

可以到 http://dryicons.com/free-icons/preview/sketchy-paper-icons/ 打包下载。

(  注意下载下来的包中的文件名可能和我写的程序中的命名有差异,请注意自行调整。)

c. 新增widget时的配置Activity的实现(可选)

android平台为widget提供一个配置界面的功能,我们可以自定义一个Activity,

在widget参数配置文件中配置好相关参数后,此Activity会在用户新增widget时自动调用。

一般来说,这个配置界面的作用是用户新建widget时,让用户配置widget的一些属性,比如颜色、大小等等。

但是在我们的这个示例程序中,我们用它来当做创建便签的地方!

不过本节只是先实现一个原型程序,所以暂时不做处理,我们只是新建一个Activity即可。

新建名为MyNoteConf的Activity,重写onCreate方法,在OnCreate方法中,

由于这个Activity是由系统在新增widget时自动调用的,

所以我们可以用getIntent获取到传入的widgetId。可以判断其是否是一个有效的widgetId,

最后我们必须返回一个RESULT_OK的Intent,并结束当前Activity,系统才会认为配置成功,在桌面上放置这个widget。

如果返回RESULT_CANCELED,系统会认为配置失败,终止widget的创建过程。

编写好的MyNoteConf的代码如下:

[java] view plaincopy
  1. package com.silenceburn;
  2. import android.app.Activity;
  3. import android.appwidget.AppWidgetManager;
  4. import android.content.Intent;
  5. import android.os.Bundle;
  6. import android.util.Log;
  7. public class MyNoteConf extends Activity {
  8. int mAppWidgetId;
  9. @Override
  10. protected void onCreate(Bundle savedInstanceState) {
  11. // TODO Auto-generated method stub
  12. super.onCreate(savedInstanceState);
  13. Log.i("myLog"," on WidgetConf ... ");
  14. setResult(RESULT_CANCELED);
  15. // Find the widget id from the intent.
  16. Intent intent = getIntent();
  17. Bundle extras = intent.getExtras();
  18. if (extras != null) {
  19. mAppWidgetId = extras.getInt(AppWidgetManager.EXTRA_APPWIDGET_ID,
  20. AppWidgetManager.INVALID_APPWIDGET_ID);
  21. }
  22. // If they gave us an intent without the widget id, just bail.
  23. if (mAppWidgetId == AppWidgetManager.INVALID_APPWIDGET_ID) {
  24. finish();
  25. }
  26. // return OK
  27. Intent resultValue = new Intent();
  28. resultValue.putExtra(AppWidgetManager.EXTRA_APPWIDGET_ID,
  29. mAppWidgetId);
  30. setResult(RESULT_OK, resultValue);
  31. finish();
  32. }
  33. }

d. widget 参数配置文件

最后我们需要编写一个widget参数配置文件,将布局文件、配置Activity关联起来。

我们在res下新建目录xml,在xml目录下新增文件 my_note_widget.xml ,编写如下:

[xhtml] view plaincopy
  1. <appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"
  2. android:minWidth="72dp" android:minHeight="72dp"
  3. android:updatePeriodMillis="86400000" android:initialLayout="@layout/my_note_widget"
  4. android:configure="com.silenceburn.MyNoteConf">
  5. </appwidget-provider>

其中 minWidth minHeight 用来指定widget的大小,如果我们只占用一个格子,也就是俗称的1X1,

那么72dp的长宽是android平台推荐的一个最佳实践值。

然后用 initialLayout 参数关联了我们编写好的 layout 文件,

用 configure 参数关联了我们编写好的配置用Activity:MyNoteConf,

此外还有一个参数 updatePeriodMills 指定widget的刷新周期,

从省电角度考虑,一般都把此值设置的比较大,如果一定要对widget做周期性的事情,可以使用AlarmManager。

至此所有widget的要素都已经准备好,我们运行一下来看看吧。


4.运行widget原型程序

为了运行widget,我们还需要修改一下 AndroidManifest.xml 来声明我们的widget。

声明一个receiver,过滤 android.appwidget.action.APPWIDGET_UPDATE ,

并且用metadata关联到我们自己编写的 appWidgetProvider 实现。

声明一个activity关联到我们的配置类 MyNoteConf,过滤 android.appwidget.action.APPWIDGET_CONFIGURE。

最后修改一下应用图标,此图标会出现在系统的新增widget列表中。

编写好的AndroidManifest.xml 如下:

[xhtml] view plaincopy
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"
  3. package="com.silenceburn" android:versionCode="1" android:versionName="1.0">
  4. <application android:icon="@drawable/sketchy_paper_008"
  5. android:label="@string/app_name">
  6. <receiver android:name=".MyNote">
  7. <intent-filter>
  8. <action android:name="android.appwidget.action.APPWIDGET_UPDATE" />
  9. </intent-filter>
  10. <meta-data android:name="android.appwidget.provider"
  11. android:resource="@xml/my_note_widget" />
  12. </receiver>
  13. <activity android:name=".MyNoteConf">
  14. <intent-filter>
  15. <action android:name="android.appwidget.action.APPWIDGET_CONFIGURE" />
  16. </intent-filter>
  17. </activity>
  18. </application>
  19. </manifest>

至此原型程序全部开发完成,运行一下看看效果吧!

在桌面上长点,可以选择我们刚刚写的原型widget“MyNote”了,

选择后出现我们定义的配置界面MyNoteConf,

但是由于我们在onCreate中finish了,所以是一闪而过的。

之后MyNote就出现在桌面上了。

我们可以随便拖动它,或者把它丢进垃圾箱,观察一下日志输出。


上半部分总结

上半部分主要完成了一个widget的原型,它没有任何业务功能,

但是已经是一个可以运行的骨架了。

在下半部分中我们为它添加血和肉,让它真正具备业务功能。

桌面便签程序的实现详解和源码 (上)相关推荐

  1. 桌面便签程序的实现详解和源码 (下)

    来自:http://blog.csdn.net/silenceburn/article/details/6094705 在上半部分中,已经实现了一个可以运行的widget,但是没有任何业务功能, 因此 ...

  2. 并发编程五:java并发线程池底层原理详解和源码分析

    文章目录 java并发线程池底层原理详解和源码分析 线程和线程池性能对比 Executors创建的三种线程池分析 自定义线程池分析 线程池源码分析 继承关系 ThreadPoolExecutor源码分 ...

  3. 计算机桌面打标签,在电脑桌面上添加便签的方法步骤详解(2)

    电脑的桌面便签软件推荐 1.CintaNotes 电脑的桌面便签软件推荐图1 CintaNotes是一款非常轻巧实用的笔记软件,可看作EverNote轻量级替代品.CintaNotes只需1个exe, ...

  4. python列表两两相减_用Python做打字软件,让你的打字快如闪电!附超详解和源码链接

    版本正在更新--,目前1.3.2 项目结果 GIF制作十分粗劣,自己看着都揪心,不过基本看清楚了 详细解释 下面我会详细解释一下,如果不看,就不看吧,直接向下看源码,传送门 基本结构+标题标签+提示+ ...

  5. 让你的打字快如闪电!用Python做打字软件,附超详解和源码链接

  6. python2.7+PyQt5 制作桌面便签小程序

    初学PyQt5,尝试制作了如图所示桌面便签小程序,功能比较单一. 现按以下顺序记录: 一.PyQt5的安装 二.程序 三.打包 #################################### ...

  7. python制作桌面小程序_python2.7+PyQt5 制作桌面便签小程序

    初学PyQt5,尝试制作了如图所示桌面便签小程序,功能比较单一. 现按以下顺序记录: 一.PyQt5的安装 二.程序 三.打包 #################################### ...

  8. 免装版_一款好用的便签工具 桌面便签免安装版

    点击上方蓝字关注我们 如您喜欢我们的公众号,不妨推荐给身边的朋友 资源介绍: 资源来源于网络,工作中我们很多时候都要建立便签/备忘录来梳理我们的工作,今天小编给大家安利一款好用的便签工具 - 桌面便签 ...

  9. python软件桌面图标_python制作一个桌面便签软件

    # 2014.10.15 更新了memo.zip, 网盘的exe:修复:1.隔日启动不能正常加载json,加入:1.隐藏任务栏图标,2.通过垃圾桶进行窗口移动. # 2014.10.8 10.36更新 ...

最新文章

  1. TCP/IP详解--第五章
  2. webApp之meta标签
  3. css 背景效果_css基础篇06--背景样式
  4. How to determine what causes a particular wait type
  5. python找不到自带的argparse_python argparse用法总结
  6. php 按位左移,PHP位运算符
  7. slz-servlet的引入
  8. Android视频编码的坑
  9. 【SW】利用3D打印机打印 PCB 钢网的方法
  10. ArcGIS地形图地形标注详解(附练习数据下载)
  11. 提问的智慧/ 如何优雅的提问
  12. pyecharts对于经纬度_echarts 根据经纬度坐标在地图上描点
  13. 五笔难拆字拆分方法汇总及详解
  14. c# Npoi导出Excel并合并行列
  15. 微分法MATLAB语言程序,matlab微积分运算命令与例题pdf
  16. echarts折线图设置圆点_echarts 设置折线图单个数据小圆圈样式
  17. Yield Guild Games 和 MOBLAND 达成合作
  18. 算法 6:集成学习与随机森林
  19. ENVI app store报错:打不开和显示无法联网
  20. ecshop二次开发-- 添加手机短信接口实例

热门文章

  1. python计算数学题_「每日一练」巧用python做小学的数学题
  2. Redshift渲染器为什么这么受欢迎?
  3. Centos换源方法
  4. mess组网 中继_Mesh组网技术登场,中继电力猫都弱爆
  5. matlab htk tools,基于HTK调用MATLAB的语音识别的研究
  6. python计算系统的阶跃响应和脉冲响应
  7. 机器学习算法---微积分与矩阵
  8. 爱联模组接入华为hilink总结——调试问题汇总(二)
  9. unescape解密 php,[PHP]php中escape函数加密与JS中unescape解密
  10. 人人人人人人人人人人人人人人人人人人人