`
java-mans
  • 浏览: 11730415 次
文章分类
社区版块
存档分类
最新评论

Android Launcher开发之AppWidget(桌面小部件)全面解析

 
阅读更多
导读:先简单说说Widget的原理。Widget是在桌面上的一块显示信息的东西,也通过单击Widget跳转到一个程序里面。而系统自带的程序,典型的Widget是music,这个Android内置的音乐播放小程序。先简单说说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的文件夹。建一个xml文件,加入如下代码:
 <appwidget-providerxmlns: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就是从程序里分离的字符串,在实现国际化的时候可能会用到。 看看layout里的main.xml ,只有一个空间就是TextView,这个是用来显示时间用的。 建一个类TestAppWidget继承于AppWidgetProvider,而AppWidgetProvider继承与android.content.BroadcastReceiver,所以TestAppWidget就是一个拦截处理Intent的BroadcastReceiver,这些Intent只能在Androidmainfest里设置来拦截处理。


public class TestAppWidget extends AppWidgetProvider {
  private static final 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刷新界面显示
    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();
   views.setTextViewText(R.id.txttim,cal.getTime().toLocaleString());
    ComponentName thisWidget =new ComponentName(context,TestAppWidget.class);
    AppWidgetManager.getInstance(context).updateAppWidget(thisWidget,views);
    }
    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";
	public void onReceive(Contextcontext, Intent intent) {
	String action=intent.getAction();
	Log.d(tag, "theaction is "+action);
	if (FRESH.equals(action)){
	showTime(context);
	}elseif(Inent.ACTION_TIME_TICK.equals(action)){
	showTime(context);
	}
	super.onReceive(context,intent);
	}



这个FRESH就显示的就是处理我们,刚才下面绑定的这个FRESH Intent。
  <?xml version="1.0"encoding="utf-8"?>
    <manifestxmlns:android="http://schemas.android.com/apk/res/android"
    package="com.sinxiao.widgetapp"
    android:versionCode="1"android:versionName="1.0.0">
   <applicationandroid:icon="@drawable/icon"android:label="@string/app_name">
    <receiver android:name=".setting.TestAppWidget">
   <intent-filter>
   <actionandroid: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);


分享到:
评论

相关推荐

    Android中实现Launcher功能之二 ----- 添加窗口小部件以及AppWidget的创建详解.pdf

    在Android系统中,窗口小部件(AppWidget)是一种能够让用户在Home屏幕上获取应用信息和快捷操作的小部件,它也是实现Launcher功能的一部分。创建AppWidget主要涉及以下几个关键组件和概念: #### 1. ...

    Androidlauncher开发.pdf

    Android_launcher 是 Android 系统中的一個重要组件,负责管理用户的桌面环境。作为一个 GUI,它不仅需要提供对所有应用程序的映射,还需要具备良好的交互性和美观的界面设计。在 launcher 的开发中,我们需要了解...

    android 桌面控件appwidget

    Android 桌面控件(AppWidget)是Android操作系统提供的一种独特功能,允许开发者创建可以在用户主屏幕上显示的小型应用程序组件。这些控件通常提供快速访问应用的核心功能或展示实时信息,如天气预报、日历事件或...

    android安卓app开发之widget界面控件使用教程.zip

    了解了基本概念后,开发者可以通过阅读《Android 之桌面组件 App Widget 案例》、《Android Launcher开发之桌面小部件AppWidget详解》以及相关的博客文章来深入学习。这些资料提供了丰富的实例和代码解析,帮助理解...

    android Launcher添加widget源码

    在Android系统中,Launcher是用户界面的一个重要组成部分,它提供了应用程序的启动图标和小部件(Widgets)等元素,使得用户可以方便地访问和管理手机上的应用。本篇将深入探讨如何在自定义的Launcher中添加Widget,...

    Android5.0 launcher 小部件预览图片

    修改Android5.0 launcher 小部件预览图片大小

    android手把手教你开发launcher.pdf

    Android Launcher 是 Android 操作系统中的桌面应用程序,负责管理桌面图标、widget 和其他桌面元素。下面我们将一步步指导您如何开发一个自己的 Launcher,並使其替代系统的默认 Launcher。 了解 Android ...

    androidlauncher应用开发完整清晰版

    ### Android Launcher ...通过以上对《androidlauncher应用开发完整清晰版》一书的内容总结,可以看出该书旨在全面深入地介绍Launcher开发的核心技术和实践经验,适合希望深入了解Android桌面开发的技术人员阅读学习。

    android手把手教你开发launcher(AndroidStudio版)

    在Android系统中,桌面启动器(Launcher)是用户与Android设备交互的起点,负责展示应用图标、列表和各种小部件。Android允许开发者创建自己的Launcher应用,可以替换掉系统默认的桌面应用。 知识点二:...

    Android项目源码-安卓Launcher原生桌面源码

    3. **Widget管理**:Android小部件在Launcher上是如何添加、更新和删除的。这涉及到AppWidgetHost和AppWidgetManager的使用,以及BroadcastReceiver来监听小部件的更新请求。 4. **应用抽屉**:源码展示了如何实现...

    Android应用源码之Android Launcher 桌面分页滑动代码.zip

    在Android系统中,Launcher是用户接触的首个界面,它提供了应用程序快捷方式、小部件和桌面分页等功能。这个"Android应用源码之Android Launcher 桌面分页滑动代码"压缩包,显然是为了帮助开发者深入理解Android ...

    安卓Android源码——安卓Android Launcher 桌面分页滑动代码.rar

    在安卓系统中,Launcher是用户接触最频繁的组件之一,它是设备主屏幕,包含了应用快捷方式、小部件等元素,并支持横向滑动切换分页。本压缩包中的"TestPagedView"文件很可能是实现这一功能的一个示例或实验性的源...

    android手把手教你开发launcher

    Android Launcher 是 Android 系统的桌面应用程序,负责显示应用程序列表、 Widget、快捷方式等内容。在本文中,我们将手把手教您开发一个简单的 Launcher 应用程序。 Launcher 的基本概念 Launcher 是 Android ...

    Android 8.0 Launcher开发指南

    在Android操作系统中,Launcher是用户与系统交互的前端界面,它负责展示应用图标、桌面小部件以及管理手机主屏幕。Android 8.0(代号Oreo)对Launcher进行了诸多改进,以提升用户体验和开发者效率。本指南将深入探讨...

    android launcher3源码 循环翻页

    Android Launcher3是Android系统的核心组件之一,它是用户与系统交互的门户,承担着应用程序启动、桌面布局管理等关键任务。本篇文章将深入探讨Launcher3的源码,特别是其循环翻页的实现原理,帮助开发者更好地理解...

    安卓Android源码——Android Launcher 源码修改可编译.zip

    Launcher是用户与Android系统交互的主要界面,它管理应用程序快捷方式、小部件和其他桌面元素。"源码修改可编译"意味着这份资料包含了已经做过调整,能够成功编译的Launcher源代码,适合开发者学习和定制自己的启动...

    Android Launcher3源码

    Android系统的桌面启动器(Launcher)是用户与设备交互的核心界面,它负责展示应用快捷方式、小部件以及主屏幕布局。其中,`Launcher3`是Android开源项目AOSP(Android Open Source Project)中的默认启动器实现。...

    Android launcher 开发博客集

    launcher 开发 在这里 我们继承了launcher 开发前人经验 一同学习

    Android_Launcher应用开发_示例代码.zip

    - **启动器的结构**:一个Android Launcher通常由主屏幕、应用抽屉(App Drawer)、小部件区域等部分组成,每个部分都需要定制布局和逻辑。 - **Activity和Intent过滤器**:Launcher作为一个Activity,需要配置...

    android Launcher源码详解

    Android Launcher 是 Android 系统中一个非常重要的组件,负责显示桌面程序和管理应用程序图标。了解 Launcher 的源码可以帮助我们更好地理解 Android 系统的设计思想和实现机制。本文将对 Launcher 的源码进行深入...

Global site tag (gtag.js) - Google Analytics