当前位置: 网学 > 编程文档 > Android > 正文

widget

来源:Http://myeducs.cn 联系QQ:点击这里给我发消息 作者: myeducs.cn 发布时间: 13/03/17

【网学网提醒】:文章导读:在新的一年中,各位网友都进入紧张的学习或是工作阶段。网学会员整理了widget的相关内容供大家参考,祝大家在新的一年里工作和学习顺利!


    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文件,放进去一下东西:
        android:minHeight="72px"
    android:minWidth="72px"
    android:updatePeriodMillis="3800000"android:initialLayout="@layout/main">
    
    这个是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的BroadcastRecei
    ver,这些Intent只能在Androidmainfest里设置来拦截处理。
    packagecom.sinxiao.widgetapp.setting;
    importjava.util.Calendar;
    importcom.sinxiao.widgetapp.R;
    importandroid.appwidget.AppWidgetManager;
    importandroid.appwidget.AppWidgetProvider;
    importandroid.content.BroadcastReceiver;
    importandroid.content.ComponentName;
    importandroid.content.Context;
    importandroid.content.Intent;
    importandroid.util.Log;
    importandroid.widget.RemoteViews;
    publicclassTestAppWidgetextendsAppWidgetProvider{
    privateStringtag="--------TestAppWidget";
    privatestatic
    finalStringFRESH="com.sinxiao.app.fresh";
    privateContextmContext;
    privatebooleanrun=true;
    /**
    *获得系统每秒
    */
    BroadcastReceivermBroadcast=newBroadcastReceiver(){
    publicvoidonReceive(Contextcontext,Intentintent){
    Stringaction=intent.getAction();
    if(action.equals(Intent.ACTION_TIME_TICK)){
    mContext.sendBroadcast(newIntent(FRESH));
    }
    }
    };
    /**
    *通知Widget每个1秒刷新一次
    */
    ThreadmyThread=new
    Thread(){
    publicvoidrun(){
    while(run){
    try{
    Thread.sleep(1000);
    }catch(InterruptedExceptione){
    e.printStackTrace();
    }
    mContext.sendBroadcast(newIntent(FRESH));//通知刷新Widget的Intent
    }
    };
    };
    @Override
    publicvoidonUpdate(Contextcontext,AppWidgetManagerappWidgetManager,
    int[]appWidgetIds){//这个是每次Widget更新时调用的函数
    用来给Widget刷新界面显示
    Log.d(tag,"onUpdate");
    super.onUpdate(context,appWidgetManager,appWidgetIds);
    mContext=context;
    RemoteViewsviews=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,newIntentFilter(Intent.ACTION_TIME_TICK));
    }
    @Override
    publicvoidonReceive(Contextcontext,Intentintent){
    Log.d(tag,"onReceive");
    Stringaction=intent.getAction();
    Log.d(tag,"theactionis"+action);
    if(FRESH.equals(action)){
    showTime(context);
    }elseif(Intent.ACTION_TIME_TICK.equals(action)){
    showTime(context);
    }
    super.onReceive(context,intent);
    }
    privatevoidshowTime(Contextcontext){
    RemoteViewsviews=newRemoteViews(context.getPackageName(),R.layout.main);
    Calendar
    cal=Calendar.getInstance();
    System.out.println(cal.getTime().toLocaleString());
    views.setTextViewText(R.id.txttim,cal.getTime().toLocaleString());
    ComponentNamethisWidget=newComponentName(context,TestAppWidget.class);
    AppWidgetManager.getIn
    stance(context).updateAppWidget(thisWidget,views);
    }
    @Override
    publicvoidonEnabled(Contextcontext){
    Log.d(tag,"onEnabled");
    super.onEnabled(context);
    run=true;
    }
    @Override
    publicvoidonDeleted(Contextcontext,int[]appWidgetIds){
    Log.d(tag,"onDeleted");
    super.onDeleted(context,appWidgetIds);
    }
    @Override
    publicvoidonDisabled(Contextcontext){
    Log.d(tag,"onDisabled");
    super.onDisabled(context);
    run=false;
    }
    }
    以上代码就是用来改变显示时间和处理刷新FRESHIntent的主程序
    看AndroidMainifest里是如何将FRESHIntent绑定到这个TestAppWidget的,看TestAppWidget这个reciever里有个
    
    这就是要拦截的Intent的一个标示。主程序里的重写了父类里的OnReceive()方法在。
    privatestatic
    finalStringFRESH="com.sinxiao.app.fresh";
    @Override
    publicvoidonReceive(Contextcontext,Intentintent){
    Log.d(tag,"onReceive");
    Stringaction=intent.getAction();
    Log.d(tag,"theactionis"+action);
    if(FRESH.equals(action)){
    showTime(context);
    }elseif(Intent.ACTION_TIME_TICK.equals(action)){
    showTime(context);
    }
    super.onReceive(context,intent);
    }
    这个FRESH就显示的就是处理我们,刚才下面绑定的这个FRESHIntent。
    
        package="com.sinxiao.widgetapp"
    android:versionCode="1"android:versionName="1.0.0">
    
    
    
    
    
    
    

    
    
    
    

        android:resource="@xml/testwidget_setting"/>
    
    
    
    这样就是Widget的简单的实现思路。需要特别说明的是Android的Widget里有好多潜规则呀。一不小心,就可能中招。这就说明了实践是检验真理的唯一标准呀。
    关于对Widget上控件如何赋值?如下所示:
    RemoteViewsviews=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();的方法里,
    ComponentNamethisWidget=newComponentName(context,TestAppWidget.class);
    AppWidgetManager.getInstance(context).updateAppWidget(thisWidget,views);
    不过变成以上这种方式来更新了。这样的好处,是只要有个context就可以实现了界面的刷新。
    说的半天,你要是不会启动Widget就杯具了,因为这事你要骂我,我就餐具了。按menu键
    ,
    然后单击Add,出现以下页面,或者长按屏幕也可以出现。
    ,
    单击Widget就出现了我们的Widget小程序了。
    ,
    单击选择我们的程序就成功了。
    帖子有点乱,看了好好缕缕,有什么不懂的可以问我啊。如果被我带到沟里,可不要骂我啊。看我的原帖与下载源码,请看:androidin.net/bbs/viewthread.php?tid=89115&;rpid=1419288&;fav=yes&;ordertype=0&;page=1#pid1419288
    
  • 下一篇资讯: VPN学习
  • 网学推荐

    免费论文

    原创论文

    浏览:
    设为首页 | 加入收藏 | 论文首页 | 论文专题 | 设计下载 | 网学软件 | 论文模板 | 论文资源 | 程序设计 | 关于网学 | 站内搜索 | 网学留言 | 友情链接 | 资料中心
    版权所有 QQ:3710167 邮箱:3710167@qq.com 网学网 [Myeducs.cn] 您电脑的分辨率是 像素
    Copyright 2008-2015 myeducs.Cn www.myeducs.Cn All Rights Reserved
    湘ICP备09003080号