当前位置:   article > 正文

Android 桌面小组件使用

Android 桌面小组件使用

基本步骤

1.创建小组件布局

这里需要注意的事,小组件布局里不能使用自定义View,只能使用原生的组件,比如说LinearLayout,TextView,连约束布局都不能使用

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. xmlns:app="http://schemas.android.com/apk/res-auto"
  4. android:layout_width="match_parent"
  5. android:layout_height="wrap_content"
  6. android:background="@color/white"
  7. android:orientation="vertical"
  8. android:padding="16dp">
  9. <LinearLayout
  10. android:layout_width="match_parent"
  11. android:layout_height="wrap_content"
  12. android:orientation="horizontal">
  13. <TextView
  14. android:id="@+id/tvDate"
  15. style="@style/textStyle14"
  16. android:textColor="#313131"
  17. android:layout_width="wrap_content"
  18. android:layout_height="wrap_content"
  19. android:text="2023-12-10" />
  20. <ImageView
  21. android:layout_width="0dp"
  22. android:layout_height="wrap_content"
  23. android:layout_weight="1" />
  24. <TextView
  25. android:id="@+id/tvTime"
  26. android:textColor="#313131"
  27. style="@style/textStyle14"
  28. android:layout_width="wrap_content"
  29. android:layout_height="wrap_content"
  30. android:text="12:10" />
  31. </LinearLayout>
  32. <LinearLayout
  33. android:layout_marginTop="16dp"
  34. android:layout_width="match_parent"
  35. android:layout_height="wrap_content"
  36. android:orientation="horizontal">
  37. <ImageView
  38. android:layout_width="wrap_content"
  39. android:layout_height="wrap_content"
  40. android:src="@drawable/result_clean"/>
  41. <LinearLayout
  42. android:orientation="vertical"
  43. android:layout_width="0dp"
  44. android:layout_marginStart="9dp"
  45. android:gravity="center_vertical"
  46. android:layout_height="match_parent"
  47. android:layout_weight="1" >
  48. <TextView
  49. style="@style/textStyle14"
  50. android:textColor="#313131"
  51. android:textStyle="bold"
  52. android:layout_width="wrap_content"
  53. android:layout_height="wrap_content"
  54. android:text="125.4MB"/>
  55. <TextView
  56. style="@style/textStyle14"
  57. android:textColor="#313131"
  58. android:textStyle="bold"
  59. android:layout_width="wrap_content"
  60. android:layout_height="wrap_content"
  61. android:text="Junk"/>
  62. </LinearLayout>
  63. <TextView
  64. android:layout_gravity="center_vertical"
  65. android:id="@+id/tvClean"
  66. android:textColor="#313131"
  67. style="@style/textStyle14"
  68. android:layout_width="wrap_content"
  69. android:layout_height="wrap_content"
  70. android:text="Clean" />
  71. </LinearLayout>
  72. </LinearLayout>

2.创建provider

  1. import android.appwidget.AppWidgetManager
  2. import android.appwidget.AppWidgetProvider
  3. import android.content.Context
  4. import android.widget.RemoteViews
  5. import android.widget.RemoteViews.RemoteView
  6. import ten.jou.recover.R
  7. class CleaningWidget : AppWidgetProvider() {
  8. override fun onUpdate(
  9. context: Context,
  10. appWidgetManager: AppWidgetManager,
  11. appWidgetIds: IntArray
  12. ) {
  13. appWidgetIds.forEach {
  14. //如果小组件布局中使用不支持的组件,这里创建RemoteViews时候,IDE会报红提示!
  15. val remoteView = RemoteViews(context.packageName, R.layout.widget_layout)
  16. //绑定数据
  17. remoteView.setTextViewText(R.id.tv1,"hello world")
  18. appWidgetManager.updateAppWidget(it, remoteView)
  19. }
  20. }
  21. }

AppWidgetProvider本质就是一个广播接收器,所以在清单文件需要声明(见步骤4)

这里先补充下,RemoteViews对于TextView,ImageView等View,有设置文本,字体颜色,图片等相关方法,但并不是所有方法都支持,绑定数据的时候需要注意下小组件是否支持!

3.创建xml属性声明

在xml文件夹里新建widget_info.xml文件:

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"
  3. xmlns:tools="http://schemas.android.com/tools"
  4. android:targetCellWidth="4"
  5. android:targetCellHeight="2"
  6. android:minWidth="250dp"
  7. android:minHeight="110dp"
  8. android:updatePeriodMillis="0"
  9. android:initialLayout="@layout/widget_layout"
  10. tools:targetApi="s">
  11. </appwidget-provider>

Android12版本以上新增的2个属性,声明组件是4*2大小

  • targetCellWidth
  • targetCellHeight

4.清单文件声明

  1. <receiver
  2. android:name=".view.CleaningWidget"
  3. android:enabled="true"
  4. android:exported="true">
  5. <intent-filter>
  6. <action android:name="android.appwidget.action.APPWIDGET_UPDATE" />
  7. </intent-filter>
  8. <meta-data
  9. android:name="android.appwidget.provider"
  10. android:resource="@xml/widget_info" />
  11. </receiver>

5.代码添加小组件

官方说Android12不允许直接通过代码添加小组件,只能让用户手动去桌面拖动添加,但是我手头的三星系统却是支持的(也是Android12),具体还没有细究...

而官方文档上的写的例子如下:

  1. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
  2. val context = this@DesktopWidgetActivity
  3. val appWidgetManager: AppWidgetManager =
  4. context.getSystemService(AppWidgetManager::class.java)
  5. val myProvider = ComponentName(context, CleaningWidget::class.java)
  6. //判断启动器是否支持小组件pin
  7. val successCallback = if (appWidgetManager.isRequestPinAppWidgetSupported) {
  8. // Create the PendingIntent object only if your app needs to be notified
  9. // that the user allowed the widget to be pinned. Note that, if the pinning
  10. // operation fails, your app isn't notified.
  11. Intent(context, CleaningWidget::class.java).let { intent ->
  12. // Configure the intent so that your app's broadcast receiver gets
  13. // the callback successfully. This callback receives the ID of the
  14. // newly-pinned widget (EXTRA_APPWIDGET_ID).
  15. //适配android12的
  16. val flags = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.S) {
  17. PendingIntent.FLAG_MUTABLE
  18. } else {
  19. PendingIntent.FLAG_UPDATE_CURRENT
  20. }
  21. PendingIntent.getBroadcast(
  22. context,
  23. 0,
  24. intent,
  25. flags
  26. )
  27. }
  28. } else {
  29. null
  30. }
  31. appWidgetManager.requestPinAppWidget(myProvider, null, successCallback)
  32. }

这里提下,上面的设置flags方法

  1. val flags = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.S) {
  2. PendingIntent.FLAG_MUTABLE
  3. } else {
  4. PendingIntent.FLAG_UPDATE_CURRENT
  5. }

有个新项目的targetSdk为34(即Android14),如果使用上面的代码会出现下面崩溃错误提示

Targeting U+ (version 34 and above) disallows creating or retrieving a PendingIntent with FLAG_MUTABLE, an implicit Intent within and without FLAG_NO_CREATE and FLAG_ALLOW_UNSAFE_IMPLICIT_INTENT for security reasons. To retrieve an already existing PendingIntent, use FLAG_NO_CREATE, however, to create a new PendingIntent with an implicit Intent use FLAG_IMMUTABLE.

实际上提示已经告诉我们怎么去改代码了,我这里把PendingIntent.FLAG_MUTABLE改为FLAG_IMMUTABLE就不会出现了上述的崩溃问题

应该是Android14添加的限制:

  • 如果Intent不传数据,必须使用PendingIntent.FLAG_IMMUTABLE
  • 如果是需要传递数据,则还是需要使用PendingIntent.FLAG_MUTABLE

定时刷新小组件UI

首先,我们得知道,如何主动去更新数据:

  1. val context = it.context
  2. val appWidgetManager: AppWidgetManager = context.getSystemService(AppWidgetManager::class.java)
  3. val myProvider = ComponentName(context, CleaningWidget::class.java)
  4. val remoview = CleaningWidget.getRemoteViewTest(context)
  5. //更新某类组件
  6. appWidgetManager.updateAppWidget(myProvider,remoview)
  7. //更新具体某个组件id
  8. appWidgetManager.updateAppWidget(widgetId,remoview)

getRemoteViewTest方法就是创建一个remoteview,然后调用remoteview相关方法设置文本之类的进行数据填充,代码就略过不写了,详见上述基本步骤2

上面的方法我们注意到updateAppWidget可以传不同的参数,一般我们用的第二个方法,指定更新某个组件

但这里又是需要我们传一个组件id,所以就是在步骤2的时候,我们根据需要需要存储下widgetId比较好,一般存入数据库,或者使用SharePreference也可

然后,就是对于定时的情况和对应方案:

  1. 如果是间隔多长更新一次,可以使用开一个服务,在服务中开启协程进行
  2. 如果是单纯的时间文本更新,可以使用TextClock组件,比如说 12:21这种
  3. 小组件的xml中默认可以设置定时更新时长,不过最短只能需要15分钟
  4. 可以使用闹钟服务AlarmManager来实现定时,不过此用法需要结合pendingintent和广播接收器使用,最终要在广播接收器里调用更新数据方法
  5. JobScheduler来实现定时更新,似乎受系统省电策略影响,适用于不太精确的定时事件(官方文档上推荐这个)
  6. WorkManager来实现定时更新(实际上算是JobScheduler升级版),似乎受系统省电策略影响,适用于不太精确的定时事件

应该是除了第一种方法,其他都是可以在应用被杀死的情况进行更新小组件UI

小组件播放动画

progressbar实现

帧动画不手动调用anim.start()方法是不会播放的,然后在网上看到一篇文章,使用了progressbar来实现,步骤如下:

在drawable文件夹准备帧动画文件

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <animation-list xmlns:android="http://schemas.android.com/apk/res/android" android:oneshot="false" android:visible="true">
  3. <item android:drawable="@drawable/cat_1" android:duration="100" />
  4. <item android:drawable="@drawable/cat_2" android:duration="100" />
  5. <item android:drawable="@drawable/cat_3" android:duration="100" />
  6. <item android:drawable="@drawable/cat_4" android:duration="100" />
  7. </animation-list>
  1. <ProgressBar
  2. android:indeterminateDrawable="@drawable/cat_animdrawable"
  3. android:layout_width="wrap_content"
  4. android:layout_height="wrap_content"/>

indeterminateDrawable设置为上面的帧动画文件即可

layoutanim实现

主要是利用viewgroup的初次显示的时候,会展示当前view的添加动画效果,从而实现比较简单的动画效果,如平移,缩放等

可以看实现的敲木鱼一文Android-桌面小组件RemoteViews播放木鱼动画 - 掘金

使用ViewFlipper

ViewFlipper主要是轮播使用的

里面可放几个元素,之后通过设置autoStart为true,则保证自动轮播

flipInterval属性则是每个元素的间隔时间(帧动画的时间),单位为ms

不过在remoteview中使用的话,缺点就是里面的元素数目只能固定死

否则只能通过定义不同layout文件(如3个元素则是某个layout,4个元素则是某个layout,然后根据选择来创建remoteview)

  1. <ViewFlipper
  2. android:id="@+id/viewFlipper"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. android:layout_gravity="center"
  6. android:layout_margin="4dp"
  7. android:autoStart="true"
  8. android:flipInterval="800">
  9. <ImageView
  10. android:id="@+id/vf_img_1"
  11. android:layout_width="match_parent"
  12. android:layout_height="match_parent"
  13. android:scaleType="fitXY"
  14. android:src="@drawable/peace_talisman_1" />
  15. <ImageView
  16. android:id="@+id/vf_img_2"
  17. android:layout_width="match_parent"
  18. android:layout_height="match_parent"
  19. android:scaleType="fitXY"
  20. android:src="@drawable/peace_talisman_2" />
  21. </ViewFlipper>

补充

获取当前桌面的组件id列表

  1. //获得当前桌面已添加的组件的id列表(可能用户添加了多个)
  2. val context = it.context
  3. val appWidgetManager: AppWidgetManager = context.getSystemService(AppWidgetManager::class.java)
  4. val myProvider = ComponentName(context, CleaningWidget::class.java)
  5. val info =appWidgetManager.getAppWidgetIds(myProvider)
  6. toast(info.size.toString())

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/运维做开发/article/detail/841838
推荐阅读
相关标签
  

闽ICP备14008679号