Android开发的应用除了程序应用,还有是Widget应用。好多人会开发程序应用而不会开发Widget应用。本帖子就是帮助大家学习如何开发Widget应用的。
先简单说说Widget的原理。Widget是在桌面上的一块显示信息的东西,也通过单击Widget跳转到一个程序里面。而系统自带的程序,典型的Widget是music,这个Android内置的音乐播放小程序。这个是典型的Widget+app应用。就是一个程序既可以通过Widget启动,也可以通过App启动。Widget就是一个AppWidgetProvider+一个UI界面显示(预先绑定了好多Intent),界面上的信息可以通过程序控制而改变,单击Widget,上的控件只能激发发送一个Intent,或发出一个Service的启动通知。而AppWidgetProvider可以拦截这个Intent,而进行相应的处理(比如显示新的信息)。
Android开发里要大量的通过手动的方式配置好多xml文件。这对于.net开发来说不得不说是个梦靥呀。这也许也是一般的java程序员比.net程序的工资高的一个原因吧。毕竟做Java开发,特别是Android开发,确实很累。你可以看下源码对照着源码进行讲解。
我们先开发一个比较简单的Widget应用,实现的主要功能是可以通过的不断变化,而不断的显示当前时间。首先,要自己手动建一个名为xml的文件夹。
[attach]91110[/attach]
建一个xml文件,放进去一下东西:
<appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"
android:minHeight="72px"
android:minWidth="72px"
android:updatePeriodMillis="3800000" android:initialLayout="@layout/main">
</appwidget-provider>
这个是Widget的显示设置,是对Widget属性的一个配置文件这个android:minHeight是Widget的高,这个android:minWidth
是Widget的宽。这个android:updatePeriodMillis属性是设置Widget页面的
更新页面的时间的频率。而这个android:initialLayout属性是表示的是初始化页面的布局,Android里画UI的地方都是通过xml文件,也可以通过代码程序来画,不过这样画的太麻烦了。
看下以下的文件系统,res文件夹是系统存放资源文件的目录。以drawable开头的文件夹是存放图片资源的文件夹。而后面的hdpi和ldpi等,都是平常在不同的状态如(横屏与竖屏时)系统调用不同的图片资源。Layout就是存放的一般都是xml,UI设计就是在这个layout文件夹里。Value里放的strings.xml就是从程序里分离的字符串,在实现国际化的时候可能会用到。
[attach]91109[/attach]
看看layout里的main.xml ,只有一个空间就是TextView,这个是用来显示时间用的。
建一个类TestAppWidget继承于AppWidgetProvider,而AppWidgetProvider继承与android.content.BroadcastReceiver,所以TestAppWidget就是一个拦截处理Intent的BroadcastReceiver,这些Intent只能在Androidmainfest里设置来拦截处理。
package com.sinxiao.widgetapp.setting;
import java.util.Calendar;
import com.sinxiao.widgetapp.R;
import android.appwidget.AppWidgetManager;
import android.appwidget.AppWidgetProvider;
import android.content.BroadcastReceiver;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.util.Log;
import android.widget.RemoteViews;
public class TestAppWidget extends AppWidgetProvider {
private String tag="--------TestAppWidget";
private static
final String FRESH="com.sinxiao.app.fresh";
private Context mContext ;
private boolean run = true ;
/**
* 获得系统每秒
*/
BroadcastReceiver mBroadcast =newBroadcastReceiver() {
public void onReceive(Contextcontext, Intent intent) {
String action =intent.getAction();
if(action.equals(Intent.ACTION_TIME_TICK)) {
mContext.sendBroadcast(newIntent(FRESH));
}
}
};
/**
* 通知Widget每个1秒刷新一次
*/
Thread myThread = new
Thread(){
public void run() {
while (run) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
mContext.sendBroadcast(newIntent(FRESH));//通知刷新Widget的Intent
}
};
};
@Override
public void onUpdate(Contextcontext, AppWidgetManager appWidgetManager,
int[] appWidgetIds) {//这个是每次Widget更新时调用的函数
用来给Widget刷新界面显示
Log.d(tag,"onUpdate");
super.onUpdate(context,appWidgetManager, appWidgetIds);
mContext = context;
RemoteViews views = newRemoteViews(context.getPackageName(),R.layout.main);
Calendar
cal=Calendar.getInstance();
System.out.println(cal.getTime().toLocaleString());
views.setTextViewText(R.id.txttim,cal.getTime().toLocaleString());
appWidgetManager.updateAppWidget(appWidgetIds,views);
myThread.start();
/**
* 本类作为一个bracastReveiver能自己再,注册个监听器
(可以取消注释,看报什么错误)
*/
//
context.registerReceiver(mBroadcast,new IntentFilter(Intent.ACTION_TIME_TICK));
}
@Override
public void onReceive(Contextcontext, Intent intent) {
Log.d(tag,"onReceive");
String action =intent.getAction();
Log.d(tag, "theaction is "+action);
if (FRESH.equals(action)){
showTime(context);
}elseif(Intent.ACTION_TIME_TICK.equals(action)){
showTime(context);
}
super.onReceive(context,intent);
}
private void showTime(Contextcontext) {
RemoteViews views = newRemoteViews(context.getPackageName(),R.layout.main);
Calendar
cal=Calendar.getInstance();
System.out.println(cal.getTime().toLocaleString());
views.setTextViewText(R.id.txttim,cal.getTime().toLocaleString());
ComponentName thisWidget =new ComponentName(context,TestAppWidget.class);
AppWidgetManager.getInstance(context).updateAppWidget(thisWidget,views);
}
@Override
public void onEnabled(Contextcontext) {
Log.d(tag, "onEnabled");
super.onEnabled(context);
run = true ;
}
@Override
public void onDeleted(Contextcontext, int[] appWidgetIds) {
Log.d(tag,"onDeleted");
super.onDeleted(context,appWidgetIds);
}
@Override
public void onDisabled(Contextcontext) {
Log.d(tag,"onDisabled");
super.onDisabled(context);
run = false ;
}
}
以上代码就是用来改变显示时间和处理刷新FRESHIntent的主程序。
看AndroidMainifest里是如何将FRESH Intent绑定到这个TestAppWidget的,看TestAppWidget这个reciever 里有个
<action android:name="com.sinxiao.app.fresh" />
这就是要拦截的Intent的一个标示。主程序里的重写了父类里的OnReceive()方法在。
private static
final StringFRESH="com.sinxiao.app.fresh";
@Override
public void onReceive(Contextcontext, Intent intent) {
Log.d(tag,"onReceive");
String action=intent.getAction();
Log.d(tag, "theaction is "+action);
if (FRESH.equals(action)){
showTime(context);
}elseif(Intent.ACTION_TIME_TICK.equals(action)){
showTime(context);
}
super.onReceive(context,intent);
}
这个FRESH就显示的就是处理我们,刚才下面绑定的这个FRESH Intent。
<?xml version="1.0"encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.sinxiao.widgetapp"
android:versionCode="1"android:versionName="1.0.0">
<application android:icon="@drawable/icon"android:label="@string/app_name">
<receiver android:name=".setting.TestAppWidget">
<intent-filter>
<action android:name="android.appwidget.action.APPWIDGET_UPDATE"/>
<!-- 这个Intent不支持在配置文件里的
注册
所以这个是没用的
-->
<!-- <actionandroid:name ="android.intent.action.TIME_TICK"/> -->
</intent-filter>
<intent-filter>
<!-- 将IntentAction 手动配置在 mainset文件上 -->
<action android:name="com.sinxiao.app.fresh"/>
</intent-filter>
<meta-data android:name="android.appwidget.provider"
android:resource="@xml/testwidget_setting" />
</receiver>
</application>
</manifest>
这样就是Widget的简单的实现思路。需要特别说明的是Android 的Widget里有好多潜规则呀。一不小心,就可能中招。这就说明了实践是检验真理的唯一标准呀。
关于对Widget上控件如何赋值?如下所示:
RemoteViews views = newRemoteViews(context.getPackageName(),R.layout.main);
Calendar
cal=Calendar.getInstance();
System.out.println(cal.getTime().toLocaleString());
views.setTextViewText(R.id.txttim,cal.getTime().toLocaleString());
appWidgetManager.updateAppWidget(appWidgetIds,views);
为什么要通过RemoteViews来向Widget设置界面呢?我也不知道,那位高人可以告诉我呢?这就是潜规则呀。在一般app应用里,直接用
setContentView(R.layout.main);
来实现。这个RemoteViews更特殊,设置Text时,必须通过
views.setTextViewText(R.id.txttim,cal.getTime().toLocaleString());
这种变态的方式才可以赋值,实在令人琢磨不透呀。
通过appWidgetManager来,更新了页面,也就刷新了Widget界面。在获得ACTION 为 com.sinxiao.app.fresh的Intent时,也是这样更新页面。在showTime();的方法里,
ComponentName thisWidget = new ComponentName(context,TestAppWidget.class);
AppWidgetManager.getInstance(context).updateAppWidget(thisWidget,views);
不过变成以上这种方式来更新了。这样的好处,是只要有个context 就可以实现了界面的刷新。
说的半天,你要是不会启动Widget就杯具了,因为这事你要骂我,我就餐具了。按menu键
,然后单击Add,出现以下页面,或者长按屏幕也可以出现。
,单击Widget就出现了我们的Widget小程序了。
,单击选择我们的程序就成功了。
分享到:
相关推荐
总结,`android AppWidget ListView`的使用涉及到AppWidget框架、RemoteViews、ListView与Adapter的结合以及事件处理等多个核心概念。通过源码学习,可以深入理解这些组件的协同工作方式,提升Android开发技能。
虽然较旧,但它仍能提供基本的了解和启发,可以作为学习AppWidget开发的起点。 总结来说,Android AppWidget为开发者提供了一种在用户主屏幕上展现应用功能的方式。通过理解其基本概念、工作原理及创建步骤,开发者...
<action android:name="android.appwidget.action.APPWIDGET_UPDATE" /> android:name="android.appwidget.provider" android:resource="@xml/app_widget_info" /> ``` 这里,`YourWidgetProvider`是你的...
<action android:name="android.appwidget.action.APPWIDGET_UPDATE" /> android:name="android.appwidget.provider" android:resource="@xml/app_widget_info" /> ``` 其中,`<meta-data>` 标签引用了一个 ...
在本入门教程中,我们将逐步学习如何创建一个基本的AppWidget,并实现与客户端应用程序的交互。 **一、创建简单的AppWidget** 1. **定义AppWidget配置文件** 首先,我们需要在`res/xml`目录下创建一个XML文件,...
《TvWidget:构建电视端Android AppWidget的深度解析》 在Android开发领域,AppWidget是扩展桌面功能的一种重要方式,而TvWidget则是专为电视设备设计的这一特性。本篇文章将深入探讨TvWidget的核心概念、设计原则...
一个Android Widget通常由XML布局文件定义其外观,如`app_widget_info.xml`,其中包含Widget的大小、更新频率等信息。在本项目中,这个文件可能已经设置好用来显示电池信息的布局元素,如电量百分比、电池图标等。 ...
为了使Widget具有交互性,我们需要在AppWidgetProvider的onReceive()方法中处理ACTION_APPWIDGET_UPDATE和ACTION_APPWIDGET_CLICK广播。当用户点击Widget时,ACTION_APPWIDGET_CLICK会被触发,此时我们可以根据用户...
【Launcher 2源码修改学习总结】 在Android系统中,Launcher是用户与设备交互的核心界面,它是系统的桌面,负责展示应用快捷方式、小部件以及管理屏幕布局。Launcher 2是Android 2.3(Gingerbread)至Android 4.0.3...
手机小部件(Widgets)是移动操作系统中的一种功能,允许用户在主屏幕上快速访问或操作应用程序的部分...了解和学习这些小部件的设计原理和实现方法,对于开发自己的移动应用或增强现有应用的功能具有重要的参考价值。
杨洋同学是大数据与计算机学院A23 计算机科学与技术 3 班的学生,在张淑荣老师的指导下完成了本次学习总结。通过该课程的学习,杨洋同学不仅掌握了安卓应用开发的基础知识,还深入理解了不同布局类型的特点及其应用...
<action android:name="android.appwidget.action.APPWIDGET_UPDATE" /> <!-- 添加这一行以处理点击事件 --> <action android:name="android.appwidget.action.APPWIDGET_CLICKED" /> android:name=...
【AppWidgetDemo】是一个Android应用示例,主要展示了如何创建一个动态更新的桌面小部件(AppWidget)。在Android系统中,AppWidgets是轻量级的应用组件,用户可以在主屏幕上添加,以便快速查看或操作应用程序的功能...
在这个示例中,我们学习了如何使用 JSON 解析来解析服务器的响应,如何使用 AppWidget 来显示信息,如何使用 HTTP 通讯来获取响应,如何使用正则表达式来匹配字符串模式,如何使用 SimpleWikiHelper 类来发送请求和...
这需要在`AppWidgetProviderInfo`中定义`configActivity`属性,并在`onReceive()`方法中处理`ACTION_APPWIDGET_CONFIGURE`事件,启动配置Activity。 在`TimeWidget`的布局文件(如`res/layout/time_widget.xml`)中...
描述中提到的“昨天无意发现自己电脑里面还有Android工程”,暗示这些资源可能来自于一个实际的Android项目,可能是开发者在进行Android应用开发时积累的经验总结或者是学习资料。 标签“Android资源”进一步确认了...
【标题解析】 ...总结,这个压缩包可能是为了帮助开发者了解和实现Android TV上的Widget小组件,其中包括了源代码、资源文件等,通过学习和调整,可以提升开发者在Android TV应用开发中的Widget设计能力。
总结来说,"Android_Widget开发实例-电量监控"是一个教学示例,它演示了如何利用Android的AppWidget、Service和RemoteViews来创建一个实时更新的电量监控Widget。通过这个例子,开发者可以学习到如何在Android平台上...
《eoeAndroid特刊》第四期中的“App Widget 基础编程实战三部曲”系列文章提供了深入浅出的指导,帮助读者从零开始学习如何开发Widget。 1. **基础篇**:介绍了Widget的基础概念,包括如何创建一个简单的Widget、...
intent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_ID, appWidgetId); PendingIntent pendingIntent = PendingIntent.getService(context, 0, intent, 0); RemoteViews views = new CustomWidgetView(context, ...