亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > Android > 正文

Android 桌面Widget開發要點解析(時間日期Widget)

2020-04-11 12:04:13
字體:
來源:轉載
供稿:網友

最近需要編寫一個日期時間的桌面Widget用來關聯日歷程序,以前很少寫桌面Widget。對這方面技術不是很熟悉,今天花時間重新整理了一下,順便把編寫一個簡單時間日期程序過程記錄下來。

桌面Widget其實就是一個顯示一些信息的工具(現在也有人開發了一些有實際操作功能的widget。例如相機widget,可以直接桌面拍照)。不過總的來說,widget主要功能就是顯示一些信息。我們今天編寫一個很簡單的作為widget,顯示時間、日期、星期幾等信息。需要顯示時間信息,那就需要實時更新,一秒或者一分鐘更新一次。

這個時間Widget我是參考(Android應用開發揭秘)書里面的一個demo例子做的,只是把功能和界面完善了一下。下面是這次的效果圖:



1、繼承AppWidgetProvider
我們編寫的桌面Widget需要提供數據更新,這里就需用用到AppWidgetProvider,它里面有一些系統回調函數。提供更新數據的操作。AppWidgetProvider是BrocastReceiver的之類,也就是說它其實本質是一個廣播接收器。下面我們看看AppWidgetProvider的幾個重要的回調方法:

復制代碼 代碼如下:

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方法:
復制代碼 代碼如下:

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配置
復制代碼 代碼如下:

  <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配置:
復制代碼 代碼如下:

<?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服務

復制代碼 代碼如下:

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界面,需要自定義控件。這部分后面有時間再說~

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲欧美日韩网| 日韩日本欧美亚洲| 国产精品美女www爽爽爽视频| 精品欧美aⅴ在线网站| 亚洲午夜激情免费视频| 国内成人精品一区| 日韩免费av一区二区| 日韩在线视频线视频免费网站| 色播久久人人爽人人爽人人片视av| 亚洲夜晚福利在线观看| 久久久极品av| 欧美激情精品久久久久久变态| 日韩av电影在线播放| 91人成网站www| 91美女高潮出水| 亚洲免费电影一区| 欧美日韩国产丝袜另类| 91福利视频网| 日韩欧美在线观看视频| 亚洲国产日韩欧美在线图片| 国产91在线视频| 精品久久在线播放| 亚洲国产精品一区二区三区| 91亚洲精华国产精华| 一区二区欧美在线| 欧美激情三级免费| 国产视频在线观看一区二区| 久久久99久久精品女同性| 影音先锋欧美在线资源| 亚洲社区在线观看| 欧美精品激情在线观看| 亚洲精品自拍第一页| 日本午夜在线亚洲.国产| 亚洲福利小视频| 欧美第一黄网免费网站| 欧美日韩国产成人在线观看| 亚洲欧美精品在线| 久久久精品一区二区| 欧美激情视频一区二区| 日韩精品视频在线观看网址| 国内精品模特av私拍在线观看| 亚洲天堂av高清| 亚洲国产毛片完整版| 2019中文字幕在线免费观看| 国产成人高清激情视频在线观看| 欧美成人午夜激情| 韩日精品中文字幕| 欧美激情aaaa| 久久综合国产精品台湾中文娱乐网| 欧美日韩高清区| 国产亚洲人成网站在线观看| 久久精品美女视频网站| 精品精品国产国产自在线| 国产日韩欧美综合| 欧美日韩中文字幕在线| 国产精品私拍pans大尺度在线| 亚洲黄在线观看| 国产午夜精品一区理论片飘花| 久久av中文字幕| 另类图片亚洲另类| 91久久久久久久久久久| 97久久伊人激情网| 欧美性猛交丰臀xxxxx网站| 久久久久免费精品国产| 91色视频在线导航| 国产精品自拍偷拍视频| 在线观看日韩av| 国内精品一区二区三区| 丝袜情趣国产精品| 久久久av网站| 91av在线看| 亚洲美女中文字幕| 成人免费直播live| 亚洲成人a**站| 91超碰中文字幕久久精品| 青青草原成人在线视频| 最近2019好看的中文字幕免费| 国产精品永久免费观看| 夜夜躁日日躁狠狠久久88av| 91精品国产777在线观看| 成人黄色午夜影院| 久久久久久久色| 国产精品美女久久| 欧美多人乱p欧美4p久久| 91精品啪在线观看麻豆免费| 亚洲高清福利视频| 国产精品综合久久久| 欧美精品999| 57pao国产精品一区| 国产精品自产拍在线观看| 国外成人免费在线播放| 最好看的2019的中文字幕视频| 久久免费视频观看| 久久精品久久久久| 黑人巨大精品欧美一区二区三区| 国产精品视频不卡| 97精品国产aⅴ7777| 中文字幕少妇一区二区三区| 亚洲一区999| 欧美日韩ab片| 国产一区二区三区毛片| 欧美日韩中文字幕日韩欧美| 亚洲精品视频中文字幕| 亚洲欧洲国产一区| 欧洲成人免费aa| 欧美激情视频在线免费观看 欧美视频免费一| 国产精品视频免费在线观看| 久久精品视频99| 色妞在线综合亚洲欧美| 欧美午夜性色大片在线观看| 国产精品午夜国产小视频| 6080yy精品一区二区三区| 欧美日韩在线第一页| 91免费在线视频网站| 在线视频免费一区二区| 2020久久国产精品| 成人福利网站在线观看| 欧美寡妇偷汉性猛交| 中文字幕无线精品亚洲乱码一区| 中文字幕日韩视频| 欧美性猛交xxxx免费看漫画| 国产成人在线一区二区| 中文字幕亚洲一区在线观看| 成人网页在线免费观看| 亚洲国产日韩欧美综合久久| 欧美在线亚洲在线| 国产极品精品在线观看| 日韩精品在线免费观看| 2019中文字幕在线观看| 欧美老妇交乱视频| 亚洲一级一级97网| 日本久久久a级免费| 操人视频在线观看欧美| 成人黄色午夜影院| 中文字幕日本欧美| 国产精品三级在线| 亚洲国产日韩欧美综合久久| 国产suv精品一区二区三区88区| 久久久精品国产| 亚洲tv在线观看| 久久婷婷国产麻豆91天堂| 日韩精品久久久久| 永久免费毛片在线播放不卡| 欧美午夜精品伦理| 2019日本中文字幕| 91情侣偷在线精品国产| 国产精品 欧美在线| 欧美精品精品精品精品免费| 精品人伦一区二区三区蜜桃免费| 中文综合在线观看| 久久国产精品久久国产精品| 日本一区二区三区在线播放| 日本久久久久亚洲中字幕| 欧美性20hd另类| 色偷偷91综合久久噜噜| 蜜臀久久99精品久久久无需会员| 亚洲欧洲日产国产网站| 中文字幕亚洲在线| 国产激情999| 91精品国产91久久| 国产精品欧美亚洲777777| 国产精品视频播放| 日韩av电影在线网| 欧美在线一级va免费观看|