商道如狼道 发表于 2024-10-10 07:22:53

Android 桌面小组件使用

根本步调

1.创建小组件布局

这里须要留意的事,小组件布局里不能使用自定义View,只能使用原生的组件,比如说LinearLayout,TextView,连约束布局都不能使用
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:background="@color/white"
    android:orientation="vertical"
    android:padding="16dp">

    <LinearLayout
      android:layout_width="match_parent"
      android:layout_height="wrap_content"
      android:orientation="horizontal">

      <TextView
            android:id="@+id/tvDate"
            style="@style/textStyle14"
            android:textColor="#313131"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="2023-12-10" />

      <ImageView
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="1" />

      <TextView
            android:id="@+id/tvTime"
            android:textColor="#313131"
            style="@style/textStyle14"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="12:10" />

    </LinearLayout>

    <LinearLayout
      android:layout_marginTop="16dp"
      android:layout_width="match_parent"
      android:layout_height="wrap_content"
      android:orientation="horizontal">
      <ImageView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:src="@drawable/result_clean"/>

      <LinearLayout
            android:orientation="vertical"
            android:layout_width="0dp"
            android:layout_marginStart="9dp"
            android:gravity="center_vertical"
            android:layout_height="match_parent"
            android:layout_weight="1" >
            <TextView
                style="@style/textStyle14"
                android:textColor="#313131"
                android:textStyle="bold"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="125.4MB"/>
            <TextView
                style="@style/textStyle14"
                android:textColor="#313131"
                android:textStyle="bold"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="Junk"/>
      </LinearLayout>

      <TextView
            android:layout_gravity="center_vertical"
            android:id="@+id/tvClean"
            android:textColor="#313131"
            style="@style/textStyle14"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="Clean" />

    </LinearLayout>

</LinearLayout> 2.创建provider

import android.appwidget.AppWidgetManager
import android.appwidget.AppWidgetProvider
import android.content.Context
import android.widget.RemoteViews
import android.widget.RemoteViews.RemoteView
import ten.jou.recover.R

class CleaningWidget : AppWidgetProvider() {
    override fun onUpdate(
      context: Context,
      appWidgetManager: AppWidgetManager,
      appWidgetIds: IntArray
    ) {
      appWidgetIds.forEach {
                        //如果小组件布局中使用不支持的组件,这里创建RemoteViews时候,IDE会报红提示!
            val remoteView = RemoteViews(context.packageName, R.layout.widget_layout)
                        //绑定数据
                        remoteView.setTextViewText(R.id.tv1,"hello world")
            appWidgetManager.updateAppWidget(it, remoteView)
      }

    }
} AppWidgetProvider本质就是一个广播吸收器,以是在清单文件须要声明(见步调4)
这里先补充下,RemoteViews对于TextView,ImageView等View,有设置文本,字体颜色,图片等相关方法,但并不是全部方法都支持,绑定数据的时间须要留意下小组件是否支持!
3.创建xml属性声明

在xml文件夹里新建widget_info.xml文件:
<?xml version="1.0" encoding="utf-8"?>
<appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:targetCellWidth="4"
    android:targetCellHeight="2"
    android:minWidth="250dp"
    android:minHeight="110dp"
    android:updatePeriodMillis="0"
    android:initialLayout="@layout/widget_layout"
    tools:targetApi="s">
</appwidget-provider> Android12版本以上新增的2个属性,声明组件是4*2大小


[*]targetCellWidth
[*]targetCellHeight
4.清单文件声明

<receiver
        android:name=".view.CleaningWidget"
        android:enabled="true"
        android:exported="true">
        <intent-filter>
                <action android:name="android.appwidget.action.APPWIDGET_UPDATE" />
        </intent-filter>
        <meta-data
                android:name="android.appwidget.provider"
                android:resource="@xml/widget_info" />
</receiver> 5.代码添加小组件

官方说Android12不允许直接通过代码添加小组件,只能让用户手动去桌面拖动添加,但是我手头的三星体系却是支持的(也是Android12),具体还没有细究...
而官方文档上的写的例子如下:
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
        val context = this@DesktopWidgetActivity

        val appWidgetManager: AppWidgetManager =
                context.getSystemService(AppWidgetManager::class.java)
        val myProvider = ComponentName(context, CleaningWidget::class.java)

        //判断启动器是否支持小组件pin
        val successCallback = if (appWidgetManager.isRequestPinAppWidgetSupported) {
                // Create the PendingIntent object only if your app needs to be notified
                // that the user allowed the widget to be pinned. Note that, if the pinning
                // operation fails, your app isn't notified.
                Intent(context, CleaningWidget::class.java).let { intent ->
                        // Configure the intent so that your app's broadcast receiver gets
                        // the callback successfully. This callback receives the ID of the
                        // newly-pinned widget (EXTRA_APPWIDGET_ID).
                        //适配android12的
                        val flags = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.S) {
                                PendingIntent.FLAG_MUTABLE
                        } else {
                                PendingIntent.FLAG_UPDATE_CURRENT
                        }
                        PendingIntent.getBroadcast(
                                context,
                                0,
                                intent,
                                flags
                        )
                }
        } else {
                null
        }

        appWidgetManager.requestPinAppWidget(myProvider, null, successCallback)
} 这里提下,上面的设置flags方法
val flags = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.S) {
        PendingIntent.FLAG_MUTABLE
} else {
        PendingIntent.FLAG_UPDATE_CURRENT
} 有个新项目的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

首先,我们得知道,怎样主动去更新数据:
val context = it.context
val appWidgetManager: AppWidgetManager = context.getSystemService(AppWidgetManager::class.java)
val myProvider = ComponentName(context, CleaningWidget::class.java)
val remoview = CleaningWidget.getRemoteViewTest(context)

//更新某类组件
appWidgetManager.updateAppWidget(myProvider,remoview)
//更新具体某个组件id
appWidgetManager.updateAppWidget(widgetId,remoview) getRemoteViewTest方法就是创建一个remoteview,然后调用remoteview相关方法设置文本之类的举行数据填充,代码就略过不写了,详见上述根本步调2
上面的方法我们留意到updateAppWidget可以传差别的参数,一样寻常我们用的第二个方法,指定更新某个组件
但这里又是须要我们传一个组件id,以是就是在步调2的时间,我们根据须要须要存储下widgetId比力好,一样寻常存入数据库,或者使用SharePreference也可
然后,就是对于定时的情况和对应方案:

[*]假如是间隔多长更新一次,可以使用开一个服务,在服务中开启协程举行
[*]假如是单纯的时间文本更新,可以使用TextClock组件,比如说 12:21这种
[*]小组件的xml中默认可以设置定时更新时长,不过最短只能须要15分钟
[*]可以使用闹钟服务AlarmManager来实现定时,不过此用法须要联合pendingintent和广播吸收器使用,终极要在广播吸收器里调用更新数据方法
[*]JobScheduler来实现定时更新,好像受体系省电计谋影响,适用于不太精确的定时事件(官方文档上推荐这个)
[*]WorkManager来实现定时更新(实际上算是JobScheduler升级版),好像受体系省电计谋影响,适用于不太精确的定时事件
应该是除了第一种方法,其他都是可以在应用被杀死的情况举行更新小组件UI
小组件播放动画

progressbar实现

帧动画不手动调用anim.start()方法是不会播放的,然后在网上看到一篇文章,使用了progressbar来实现,步调如下:
在drawable文件夹准备帧动画文件
<?xml version="1.0" encoding="utf-8"?>
<animation-list xmlns:android="http://schemas.android.com/apk/res/android" android:oneshot="false" android:visible="true">
    <item android:drawable="@drawable/cat_1" android:duration="100" />
    <item android:drawable="@drawable/cat_2" android:duration="100" />
    <item android:drawable="@drawable/cat_3" android:duration="100" />
    <item android:drawable="@drawable/cat_4" android:duration="100" />
</animation-list> <ProgressBar
        android:indeterminateDrawable="@drawable/cat_animdrawable"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"/> indeterminateDrawable设置为上面的帧动画文件即可
layoutanim实现

主要是利用viewgroup的初次显示的时间,会展示当前view的添加动画效果,从而实现比力简单的动画效果,如平移,缩放等
可以看实现的敲木鱼一文Android-桌面小组件RemoteViews播放木鱼动画 - 掘金
使用ViewFlipper

ViewFlipper主要是轮播使用的
里面可放几个元素,之后通过设置autoStart为true,则包管主动轮播
flipInterval属性则是每个元素的间隔时间(帧动画的时间),单位为ms
不过在remoteview中使用的话,缺点就是里面的元素数量只能固定死
否则只能通过定义差别layout文件(如3个元素则是某个layout,4个元素则是某个layout,然后根据选择来创建remoteview)
<ViewFlipper
      android:id="@+id/viewFlipper"
      android:layout_width="match_parent"
      android:layout_height="match_parent"
      android:layout_gravity="center"
      android:layout_margin="4dp"
      android:autoStart="true"
      android:flipInterval="800">

      <ImageView
            android:id="@+id/vf_img_1"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:scaleType="fitXY"
            android:src="@drawable/peace_talisman_1" />

      <ImageView
            android:id="@+id/vf_img_2"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:scaleType="fitXY"
            android:src="@drawable/peace_talisman_2" />
    </ViewFlipper> 补充

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

//获得当前桌面已添加的组件的id列表(可能用户添加了多个)
val context = it.context
val appWidgetManager: AppWidgetManager = context.getSystemService(AppWidgetManager::class.java)
val myProvider = ComponentName(context, CleaningWidget::class.java)

val info =appWidgetManager.getAppWidgetIds(myProvider)
toast(info.size.toString())

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
页: [1]
查看完整版本: Android 桌面小组件使用