Android Widget 开发踩坑

关于Android widget 小部件开发的文章,搜到的都比较老旧,并且很多已经不适用于高版本的android系统了。本文收集了一些笔者在widget使用过程中踩过的坑,以供参考(本文写于2022.07.22)。

1.系统级应用和第三方应用widget的UI区别

先看图,这里以小米手机为例


WechatIMG190.jpeg

图中红色框内是系统应用的widget,绿色框则是我demo的widget,可以看到,系统widget底部有文本“笔记”,并布局上是对齐的,是类iOS风格,小米/华为等国产手机的系统widget都是这种风格。而绿色框内,下方并没有文本,导致布局高度上显得很长。

那我们自己的应用能不能实现这种UI,答案是不行。原因如下:这个“笔记”文本非android api原生设置,就决定了我们无法准确知道文本的间距,字号,颜色等,也无法跟随系统皮肤/壁纸/深色模式自动切换。
贴个图感受一下:

截屏2022-07-22 12.28.43.png

2.Widget如何在后台定时刷新

定时刷新是widget最核心的问题,很多文章说的启动后台服务的方式已经过时了,Android8之后,后台服务限制越来越严格,在主App被杀死的情况下,已经连偷偷启动后台服务都做不到了,什么守护线程,什么广播唤起,都不管用了,反正我实现不了,有大佬能做到的望分享下。
我实践过,行得通的方案只有三种:

1.使用widget自带的刷新机制

配置updatePeriodMillis属性,能实现定时刷新。经过测试,哪怕主程序没启动过或已经被杀死,系统都能间隔一段时间后调用该方法,但时间不一定准确,比如说设置间隔是30分钟,但可能30多分钟才会回调,估计受系统运行状态/电量等影响。这种方式刷新稳定但也有明显限制:
1.刷新间隔有限,最快只能30分钟回调一次。
2.刷新时回调AppWidgetProvider.onUpdate()函数,由于AppWidgetProvider本身是个广播接收器,而广播接收器的生命周期很短,像网络请求这些异步耗时操作无法在onUpdate里执行, 所以还得另想办法完成耗时操作.

注意:在AppWidgetProvider内开启后台服务执行耗时/异步操作已经行不通,在高版本Android上,主App没启动的情况下,不允许启动后台服务,只能启动前台服务。

2.使用前台服务,设置定时器,自己维护刷新

使用前台服务,需要自己维护前台服务的保活,当然由于是前台服务,就几乎不会被杀死,即使被杀死,根据onStartCommand()的返回值设置,服务仍然可以在资源充足的条件下立即重启。这个方案并不是完美方案,难度在于前台服务的保活。比如说在前台服务被杀死时,重新启动自己;主App启动/运行时,检查前台服务;在widget上提供刷新按钮,让用户可以主动刷新。前台服务的优缺点如下:

优点
1.定时任务较稳定,大部分情况下能正常运行。
2.刷新间隔想设多少就多少,适用于对刷新十分频繁的应用,如时钟天气类应用。

缺点
1.会增大应用的耗电量
2.会在通知栏里显示服务且无法移除该通知

3.使用WorkManager

WorkManager不了解的同学请自行百度一下。WorkManager在主App被杀死的情况,还能正常执行任务。Google推荐的widget异步请求刷新方式也是使用WorkManager。优缺点如下:

优点
1.定时任务稳定,App被杀死也能正常执行任务
2.实现简单,解决了widget在App不存活时的数据刷新问题,是后台服务的替代者

缺点
1.刷新间隔有限,最快只能15分钟执行一次。

这块例子比较少,下面实现一个Widget + WorkManager 的Demo,供参考
先上效果图:

截屏2022-07-22 16.15.05.png

右侧就是示例Widget,文本显示的是时间,点击右上角可以刷新时间。可以点击刷新,或者每15分钟定时刷新。

直接上关键代码,源码点击这里

TestWidgetProvider.java

public class TestWidgetProvider extends AppWidgetProvider {

    //系统更新广播
    public static final String APPWIDGET_UPDATE = "android.appwidget.action.APPWIDGET_UPDATE";
    //自定义的刷新广播
    private static final String REFRESH_ACTION = "android.appwidget.action.REFRESH";
    //定期任务的name
    private static final String WORKER_NAME = "TestWorker";

    @Override
    public void onReceive(Context context, Intent intent) {
        super.onReceive(context, intent);
        //接收主动点击刷新广播/系统刷新广播
        if (TextUtils.equals(intent.getAction(), REFRESH_ACTION)
                || TextUtils.equals(intent.getAction(), APPWIDGET_UPDATE)) {
            //执行一次任务
            WorkManager.getInstance(context).enqueue(OneTimeWorkRequest.from(TestWorker.class));
        }
    }

    @Override
    public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) {
        super.onUpdate(context, appWidgetManager, appWidgetIds);
        //到达指定的更新时间或者当用户向桌面添加AppWidget时被调用,或更新widget时

        //点击事件
        Intent intent = new Intent();
        intent.setClass(context, TestWidgetProvider.class);
        intent.setAction(REFRESH_ACTION);

        //设置pendingIntent
        PendingIntent pendingIntent;
        if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.S) {
            pendingIntent = PendingIntent.getBroadcast(context, 0, intent, PendingIntent.FLAG_MUTABLE);
        } else {
            pendingIntent = PendingIntent.getBroadcast(context, 0, intent, PendingIntent.FLAG_ONE_SHOT);
        }
        //Retrieve a PendingIntent that will perform a broadcast
        RemoteViews remoteViews = new RemoteViews(context.getPackageName(), R.layout.widget_layout);


        //为刷新按钮绑定一个事件便于发送广播
        remoteViews.setOnClickPendingIntent(R.id.iv_refresh, pendingIntent);
        appWidgetManager.updateAppWidget(appWidgetIds, remoteViews);
    }

    @Override
    public void onDeleted(Context context, int[] appWidgetIds) {
        super.onDeleted(context, appWidgetIds);
        //删除一个AppWidget时调用
    }

    @Override
    public void onEnabled(Context context) {
        //AppWidget的实例第一次被创建时调用
        super.onEnabled(context);
        //开始定时工作,间隔15分钟刷新一次
        PeriodicWorkRequest workRequest = new PeriodicWorkRequest.Builder(TestWorker.class,
                PeriodicWorkRequest.MIN_PERIODIC_INTERVAL_MILLIS, TimeUnit.MILLISECONDS)
                .setConstraints(new Constraints.Builder()
                        .setRequiresCharging(true)
                        .build())
                .build();
        WorkManager.getInstance(context)
                .enqueueUniquePeriodicWork(WORKER_NAME, ExistingPeriodicWorkPolicy.KEEP, workRequest);
    }

    @Override
    public void onDisabled(Context context) {
        //删除一个AppWidget时调用
        super.onDisabled(context);
        //停止任务
        WorkManager.getInstance(context).cancelUniqueWork(WORKER_NAME);
    }

TestWorker.java

public class TestWorker extends Worker {

    public TestWorker(@NonNull Context context, @NonNull WorkerParameters workerParams) {
        super(context, workerParams);
    }

    @NonNull
    @Override
    public Result doWork() {
        //模拟耗时/网络请求操作
        try {
            Thread.sleep(500);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        //刷新widget
        updateWidget(getApplicationContext());

        return Result.success();
    }

    /**
     * 刷新widget
     */
    private void updateWidget(Context context) {
        String data = TimeUtil.long2String(System.currentTimeMillis(), TimeUtil.HOUR_MM_SS);
        //只能通过远程对象来设置appwidget中的控件状态
        RemoteViews remoteViews = new RemoteViews(context.getPackageName(), R.layout.widget_layout);
        //通过远程对象修改textview
        remoteViews.setTextViewText(R.id.tv_text, data);

        //获得appwidget管理实例,用于管理appwidget以便进行更新操作
        AppWidgetManager appWidgetManager = AppWidgetManager.getInstance(context);
        //获得所有本程序创建的appwidget
        ComponentName componentName = new ComponentName(context, TestWidgetProvider.class);
        //更新appwidget
        appWidgetManager.updateAppWidget(componentName, remoteViews);
    }
}

简单说下整个过程:
1.在onEnabled里启动定时任务,在onDisabled里移除定时任务
2.在onUpdate设置点击事件,在onReceive接收事件广播,执行刷新
3.在TestWorker的doWork内执行耗时操作,并更新UI

注意:Demo中使用的 targetSdk 是32,对应的work版本是2.7.1。如果你的项目targetSdk低于31,可以先升级到32,或者将work版本降低为2.3.3.

对于Widget刷新,一般情况下,推荐使用Workmanager的方式,除非是对刷新频率要求很高的应用,才使用前台服务。

3.如何加载网络图片

这里主要介绍如何通过Glide加载图片.
方式一:

        //设置icon
        AppWidgetTarget target = new AppWidgetTarget(context, R.id.iv_icon, remoteViews, componentName);
        String iconUrl = "https://gw.alicdn.com/tps/TB1W_X6OXXXXXcZXVXXXXXXXXXX-400-400.png";
        //与Glide3不同,Glide4的asBitmap()方法必须在load方法前面
        Glide.with(context)
                .asBitmap()
                .load(iconUrl)
                .apply(new RequestOptions().placeholder(R.mipmap.ic_launcher_round).circleCrop())
                .into(target);    //into(target) 必须在主线程内调用

        //更新appwidget
        appWidgetManager.updateAppWidget(componentName, remoteViews);

方式二:

String iconUrl = "https://gw.alicdn.com/tps/TB1W_X6OXXXXXcZXVXXXXXXXXXX-400-400.png";
        try {
            //同步获取bitmap
            Bitmap bitmap = Glide.with(context)
                    .asBitmap()
                    .load(iconUrl)
                    .apply(new RequestOptions().placeholder(R.drawable.ic_token_logo).circleCrop())
                    .submit().get();
            remoteViews.setImageViewBitmap(R.id.iv_icon, bitmap);
        } catch (ExecutionException | InterruptedException e) {
            e.printStackTrace();
        }

        //更新appwidget
        appWidgetManager.updateAppWidget(componentName, remoteViews);
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351

推荐阅读更多精彩内容