Andriod开发——widget组件开发

1.引言

1.1 Andriod widget的概念

Android widget 是一种特殊的应用组件,是自定义主屏幕的一个重要的方面。它允许用户将应用的特定功能或信息直接显示在主屏幕上。Widget 可以提供快速访问应用功能、显示实时数据或提供交互式控件,而无需用户打开完整的应用界面,用户可以在其主屏幕面板上移动 widget,并且如果支持,还可以调整 widget 的大小,以根据自己的偏好调整 widget 中的信息量。

1.2 Widget在应用中的作用和优势

  • 快速访问:Widget 允许用户快速访问应用的核心功能或信息,而无需打开应用本身。
  • 实时信息展示:Widget 可以显示实时数据,如天气预报、日历日程等,用户能够即时获取重要信息。
    天气应用组件
  • 交互性:用户可以通过与 Widget 交互来执行特定任务,例如音乐播放微件,ToDo list微件。
    音乐播放组件
    控件组件
  • 个性化体验:Widget 支持自定义配置,用户可以根据自己的喜好和需求调整 Widget 的内容和外观。
  • 节省空间:Widget 通常比完整的应用界面更紧凑,能够在主屏幕上占用较少的空间。

2. Widget的基本结构

创建一个widget,主要需要以下基本组件:

  • AppWidgetProviderInfo对象
  • AppWidgetProvider 类
  • 视图布局

2.1 AppWidgetProviderInfo对象

描述微件的元数据,用于定义Widget的属性和行为。例如微件的布局、更新频率和 AppWidgetProvider 类。AppWidgetProviderInfo 是在 XML 中定义的。这个文件通常位于res/xml目录下。

2.2 AppWidgetProvider 类

这是一个广播接收器类,用于定义允许以编程方式与微件连接的基本方法,即用于定义widget的行为。通过它,您会在更新、启用、停用或删除微件时收到广播。
需要在清单中声明 AppWidgetProvider,然后实现它。
AppWidgetProvider 类扩展了 BroadcastReceiver 作为一个辅助类来处理微件广播,实现onUpdate()、onEnabled()、onDisabled()等,以处理Widget的生命周期事件。

2.3 视图布局

在 XML 中为 widget 定义初始布局,并将其保存在项目的 res/layout/ 目录中。
widget的布局基于RemoteViews,在Widget中创建复杂的布局和更新内容,而不会影响主线程的性能。
widget的处理流程

3.创建一个简单的widget

接下来创建一个TODO LIST应用的Widget进行展示。

3.1 创建一个新的App Widget

Android Studio 可以自动创建一组 AppWidgetProviderInfo、AppWidgetProvider 和 View 布局文件。
依次选择New -> Widget > AppWidget
新建widget

3.2 创建Todo数据模型和SharedPreferences工具类

ToDo.java:

package com.example.appwidget;

public class Todo {
    private String id;
    private String content;
    private boolean completed;

    public Todo(String id, String content, boolean completed) {
        this.id = id;
        this.content = content;
        this.completed = completed;
    }

    public String getId() { return id; }
    public String getContent() { return content; }
    public boolean isCompleted() { return completed; }
    public void setCompleted(boolean completed) { this.completed = completed; }
} 

使用SharedPreferences实现数据持久化。
TodoPreferences.java:

package com.example.appwidget;

import android.content.Context;
import android.content.SharedPreferences;
import com.google.gson.Gson;
import com.google.gson.reflect.TypeToken;
import java.util.ArrayList;
import java.util.List;
import java.util.UUID;

public class TodoPreferences {
    private static final String PREF_NAME = "todo_preferences";
    private static final String KEY_TODOS = "todos";
    private final SharedPreferences preferences;
    private final Gson gson;

    public TodoPreferences(Context context) {
        preferences = context.getSharedPreferences(PREF_NAME, Context.MODE_PRIVATE);
        gson = new Gson();
    }

    public List<Todo> getTodos() {
        String json = preferences.getString(KEY_TODOS, "[]");
        return gson.fromJson(json, new TypeToken<List<Todo>>(){}.getType());
    }

    public void saveTodos(List<Todo> todos) {
        String json = gson.toJson(todos);
        preferences.edit().putString(KEY_TODOS, json).apply();
    }

    public void addTodo(String content) {
        List<Todo> todos = getTodos();
        todos.add(new Todo(UUID.randomUUID().toString(), content, false));
        saveTodos(todos);
    }

    public void toggleTodo(String id) {
        List<Todo> todos = getTodos();
        for (Todo todo : todos) {
            if (todo.getId().equals(id)) {
                todo.setCompleted(!todo.isCompleted());
                break;
            }
        }
        saveTodos(todos);
    }

    public void deleteTodo(String id) {
        List<Todo> todos = getTodos();
        todos.removeIf(todo -> todo.getId().equals(id));
        saveTodos(todos);
    }
} 

3.3 创建RemoteViewsFactory来提供列表数据

TodoRemoteViewsFactory.java:

package com.example.appwidget;

import android.content.Context;
import android.content.Intent;
import android.widget.RemoteViews;
import android.widget.RemoteViewsService;
import java.util.List;

public class TodoRemoteViewsFactory implements RemoteViewsService.RemoteViewsFactory {
    private Context context;
    private List<Todo> todos;
    private TodoPreferences preferences;

    public TodoRemoteViewsFactory(Context context) {
        this.context = context;
        this.preferences = new TodoPreferences(context);
    }

    @Override
    public void onCreate() {
        // 初始化时不需要加载数据
    }

    @Override
    public void onDataSetChanged() {
        // 当数据更新时重新加载
        todos = preferences.getTodos();
    }

    @Override
    public void onDestroy() {
        todos.clear();
    }

    @Override
    public int getCount() {
        return todos.size();
    }

    @Override
    public RemoteViews getViewAt(int position) {
        if (position < 0 || position >= todos.size()) {
            return null;
        }

        Todo todo = todos.get(position);
        RemoteViews rv = new RemoteViews(context.getPackageName(), R.layout.todo_item);
        rv.setTextViewText(R.id.todo_text, todo.getContent());
        rv.setImageViewResource(R.id.todo_checkbox, 
            todo.isCompleted() ? R.drawable.ic_checkbox_checked : R.drawable.ic_checkbox_unchecked);

        // 设置点击事件的填充Intent
        Intent fillIntent = new Intent();
        fillIntent.putExtra("todo_id", todo.getId());
        rv.setOnClickFillInIntent(R.id.todo_item_container, fillIntent);

        return rv;
    }

    @Override
    public RemoteViews getLoadingView() {
        return null;
    }

    @Override
    public int getViewTypeCount() {
        return 1;
    }

    @Override
    public long getItemId(int position) {
        return position;
    }

    @Override
    public boolean hasStableIds() {
        return true;
    }
} 

3.4 创建RemoteViewsService

TodoRemoteViewsService.java:

package com.example.appwidget;

import android.content.Intent;
import android.widget.RemoteViewsService;

public class TodoRemoteViewsService extends RemoteViewsService {
    @Override
    public RemoteViewsFactory onGetViewFactory(Intent intent) {
        return new TodoRemoteViewsFactory(this.getApplicationContext());
    }
} 

3.5 修改widget布局

todo_list_widget.xml:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    style="@style/Widget.AppWidget.AppWidget.Container"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:theme="@style/Theme.AppWidget.AppWidgetContainer">

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="vertical">

        <TextView
            android:id="@+id/widget_title"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:background="@color/light_blue_600"
            android:padding="8dp"
            android:text="@string/todo_list"
            android:textColor="@color/white"
            android:textSize="18sp"
            android:textStyle="bold" />

        <ListView
            android:id="@+id/todo_list"
            android:layout_width="match_parent"
            android:layout_height="0dp"
            android:layout_weight="1"
            android:divider="@android:color/darker_gray"
            android:dividerHeight="1dp" />

        <TextView
            android:id="@+id/empty_view"
            android:layout_width="match_parent"
            android:layout_height="0dp"
            android:layout_weight="1"
            android:gravity="center"
            android:text="@string/empty_list"
            android:textSize="16sp"
            android:visibility="gone" />

        <ImageButton
            android:id="@+id/add_button"
            android:layout_width="48dp"
            android:layout_height="48dp"
            android:layout_gravity="end|bottom"
            android:layout_margin="8dp"
            android:background="@drawable/circular_button"
            android:src="@android:drawable/ic_input_add"
            android:tint="@color/white" />
    </LinearLayout>

</RelativeLayout>

3.6 创建列表项布局

todo_item.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/todo_item_container"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="horizontal"
    android:padding="8dp">

    <ImageView
        android:id="@+id/todo_checkbox"
        android:layout_width="24dp"
        android:layout_height="24dp"
        android:layout_gravity="center_vertical"
        android:src="@drawable/ic_checkbox_unchecked" />

    <TextView
        android:id="@+id/todo_text"
        android:layout_width="0dp"
        android:layout_height="wrap_content"
        android:layout_gravity="center_vertical"
        android:layout_marginStart="8dp"
        android:layout_weight="1"
        android:textSize="16sp" />

</LinearLayout> 

3.7 修改Widget提供者类

TodoListWidget.java

package com.example.appwidget;

import android.app.PendingIntent;
import android.appwidget.AppWidgetManager;
import android.appwidget.AppWidgetProvider;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.net.Uri;
import android.widget.RemoteViews;

/**
 * Implementation of App Widget functionality.
 */
public class TodoListWidget extends AppWidgetProvider {

    public static final String ACTION_REFRESH = "com.example.appwidget.ACTION_REFRESH";
    public static final String ACTION_TOGGLE_TODO = "com.example.appwidget.ACTION_TOGGLE_TODO";
    public static final String EXTRA_TODO_ID = "todo_id";

    @Override
    public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) {
        for (int appWidgetId : appWidgetIds) {
            updateAppWidget(context, appWidgetManager, appWidgetId);
        }
    }

    @Override
    public void onReceive(Context context, Intent intent) {
        super.onReceive(context, intent);
        
        if (ACTION_REFRESH.equals(intent.getAction())) {
            AppWidgetManager appWidgetManager = AppWidgetManager.getInstance(context);
            ComponentName thisWidget = new ComponentName(context, TodoListWidget.class);
            int[] appWidgetIds = appWidgetManager.getAppWidgetIds(thisWidget);
            appWidgetManager.notifyAppWidgetViewDataChanged(appWidgetIds, R.id.todo_list);
        } else if (ACTION_TOGGLE_TODO.equals(intent.getAction())) {
            String todoId = intent.getStringExtra(EXTRA_TODO_ID);
            if (todoId != null) {
                TodoPreferences preferences = new TodoPreferences(context);
                preferences.toggleTodo(todoId);
                // 刷新widget
                Intent refreshIntent = new Intent(context, TodoListWidget.class);
                refreshIntent.setAction(ACTION_REFRESH);
                context.sendBroadcast(refreshIntent);
            }
        }
    }

    static void updateAppWidget(Context context, AppWidgetManager appWidgetManager, int appWidgetId) {
        RemoteViews views = new RemoteViews(context.getPackageName(), R.layout.todo_list_widget);

        // 设置ListView的适配器
        Intent serviceIntent = new Intent(context, TodoRemoteViewsService.class);
        serviceIntent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_ID, appWidgetId);
        serviceIntent.setData(Uri.parse(serviceIntent.toUri(Intent.URI_INTENT_SCHEME)));
        views.setRemoteAdapter(R.id.todo_list, serviceIntent);

        // 设置空视图
        views.setEmptyView(R.id.todo_list, R.id.empty_view);

        // 设置列表项点击事件
        Intent toggleIntent = new Intent(context, TodoListWidget.class);
        toggleIntent.setAction(ACTION_TOGGLE_TODO);
        PendingIntent togglePendingIntent = PendingIntent.getBroadcast(
            context, 0, toggleIntent, PendingIntent.FLAG_UPDATE_CURRENT | PendingIntent.FLAG_MUTABLE);
        views.setPendingIntentTemplate(R.id.todo_list, togglePendingIntent);

        // 设置添加按钮点击事件
        Intent mainIntent = new Intent(context, MainActivity.class);
        PendingIntent mainPendingIntent = PendingIntent.getActivity(
            context, 0, mainIntent, PendingIntent.FLAG_UPDATE_CURRENT | PendingIntent.FLAG_MUTABLE);
        views.setOnClickPendingIntent(R.id.add_button, mainPendingIntent);

        appWidgetManager.updateAppWidget(appWidgetId, views);
    }

    @Override
    public void onEnabled(Context context) {
        // Enter relevant functionality for when the first widget is created
    }

    @Override
    public void onDisabled(Context context) {
        // Enter relevant functionality for when the last widget is disabled
    }
}

3.8 更新AndroidManifest.xml

修改todo_list_widget_info.xml:

<?xml version="1.0" encoding="utf-8"?>
<appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"
    android:description="@string/app_widget_description"
    android:initialKeyguardLayout="@layout/todo_list_widget"
    android:initialLayout="@layout/todo_list_widget"
    android:minWidth="180dp"
    android:minHeight="110dp"
    android:previewImage="@drawable/example_appwidget_preview"
    android:previewLayout="@layout/todo_list_widget"
    android:resizeMode="horizontal|vertical"
    android:targetCellWidth="3"
    android:targetCellHeight="2"
    android:updatePeriodMillis="1800000"
    android:widgetCategory="home_screen" />

在AndroidManifest.xml中注册新的Service:

<service
            android:name=".TodoRemoteViewsService"
            android:permission="android.permission.BIND_REMOTEVIEWS" />

3.9 完善主应用以实现TODO List功能

  • 创建一个布局文件用于主界面
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <com.google.android.material.appbar.AppBarLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content">

        <androidx.appcompat.widget.Toolbar
            android:id="@+id/toolbar"
            android:layout_width="match_parent"
            android:layout_height="?attr/actionBarSize"
            android:background="?attr/colorPrimary"
            android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar" />

    </com.google.android.material.appbar.AppBarLayout>

    <androidx.recyclerview.widget.RecyclerView
        android:id="@+id/todo_list"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="1" />

    <com.google.android.material.floatingactionbutton.FloatingActionButton
        android:id="@+id/fab_add"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="end|bottom"
        android:layout_margin="16dp"
        android:src="@android:drawable/ic_input_add" />

</LinearLayout>
  • 创建一个列表项布局
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="horizontal"
    android:padding="16dp">

    <CheckBox
        android:id="@+id/checkbox"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center_vertical" />

    <TextView
        android:id="@+id/text_todo"
        android:layout_width="0dp"
        android:layout_height="wrap_content"
        android:layout_gravity="center_vertical"
        android:layout_marginStart="16dp"
        android:layout_weight="1"
        android:textSize="16sp" />

    <ImageButton
        android:id="@+id/button_delete"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center_vertical"
        android:background="?attr/selectableItemBackgroundBorderless"
        android:padding="8dp"
        android:src="@android:drawable/ic_menu_delete" />

</LinearLayout> 
  • 创建一个对话框布局用于添加新任务
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="vertical"
    android:padding="16dp">

    <EditText
        android:id="@+id/edit_todo"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:hint="@string/hint_add_todo"
        android:inputType="text"
        android:maxLines="1" />

</LinearLayout> 
  • 创建RecyclerView的适配器
package com.example.appwidget;

import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.CheckBox;
import android.widget.ImageButton;
import android.widget.TextView;
import androidx.recyclerview.widget.RecyclerView;
import java.util.List;

public class TodoAdapter extends RecyclerView.Adapter<TodoAdapter.ViewHolder> {
    private List<Todo> todos;
    private TodoPreferences preferences;

    public TodoAdapter(List<Todo> todos, TodoPreferences preferences) {
        this.todos = todos;
        this.preferences = preferences;
    }

    @Override
    public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        View view = LayoutInflater.from(parent.getContext())
                .inflate(R.layout.item_todo, parent, false);
        return new ViewHolder(view);
    }

    @Override
    public void onBindViewHolder(ViewHolder holder, int position) {
        Todo todo = todos.get(position);
        holder.textTodo.setText(todo.getContent());
        holder.checkbox.setChecked(todo.isCompleted());
        
        holder.checkbox.setOnClickListener(v -> {
            preferences.toggleTodo(todo.getId());
            refreshData();
        });
        
        holder.buttonDelete.setOnClickListener(v -> {
            preferences.deleteTodo(todo.getId());
            refreshData();
        });
    }

    @Override
    public int getItemCount() {
        return todos.size();
    }

    public void refreshData() {
        todos = preferences.getTodos();
        notifyDataSetChanged();
    }

    static class ViewHolder extends RecyclerView.ViewHolder {
        CheckBox checkbox;
        TextView textTodo;
        ImageButton buttonDelete;

        ViewHolder(View view) {
            super(view);
            checkbox = view.findViewById(R.id.checkbox);
            textTodo = view.findViewById(R.id.text_todo);
            buttonDelete = view.findViewById(R.id.button_delete);
        }
    }
} 
  • 修改MainActivity
package com.example.appwidget;

import android.app.AlertDialog;
import android.appwidget.AppWidgetManager;
import android.content.ComponentName;
import android.content.Intent;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.widget.EditText;
import androidx.appcompat.app.AppCompatActivity;
import androidx.appcompat.widget.Toolbar;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;
import com.google.android.material.floatingactionbutton.FloatingActionButton;

public class MainActivity extends AppCompatActivity {
    private TodoPreferences preferences;
    private TodoAdapter adapter;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Toolbar toolbar = findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);
        
        preferences = new TodoPreferences(this);
        
        RecyclerView recyclerView = findViewById(R.id.todo_list);
        recyclerView.setLayoutManager(new LinearLayoutManager(this));
        adapter = new TodoAdapter(preferences.getTodos(), preferences);
        recyclerView.setAdapter(adapter);

        FloatingActionButton fab = findViewById(R.id.fab_add);
        fab.setOnClickListener(v -> showAddTodoDialog());
    }

    private void showAddTodoDialog() {
        View dialogView = LayoutInflater.from(this)
                .inflate(R.layout.dialog_add_todo, null);
        EditText editTodo = dialogView.findViewById(R.id.edit_todo);

        new AlertDialog.Builder(this)
                .setTitle(R.string.add_todo)
                .setView(dialogView)
                .setPositiveButton(android.R.string.ok, (dialog, which) -> {
                    String content = editTodo.getText().toString().trim();
                    if (!content.isEmpty()) {
                        preferences.addTodo(content);
                        adapter.refreshData();
                        refreshWidget();
                    }
                })
                .setNegativeButton(android.R.string.cancel, null)
                .show();
    }

    private void refreshWidget() {
        Intent intent = new Intent(this, TodoListWidget.class);
        intent.setAction(TodoListWidget.ACTION_REFRESH);
        sendBroadcast(intent);
    }
}

3.10 运行效果

  • 通过长按应用图标添加widget,在主屏幕显示TODO列表
    运行效果0
    运行效果1
  • 支持调整大小以及切换任务完成状态
    运行效果2
  • 添加按钮打开主应用,支持添加、删除任务,实现了刷新机制
    运行效果3

4.Widget开发的新纪元——Jetpack Glance

4.1 Glance简介

Jetpack Glance是Google推出的新一代Widget开发框架,它使用Kotlin和类似Jetpack Compose的声明式UI语法,极大地简化了Android Widget的开发过程。

主要特点

  • 声明式UI设计
  • 简化的状态管理
  • 更好的开发体验
  • 与Jetpack Compose兼容
  • 更少的模板代码

4.2 基础实现

  • 添加依赖
dependencies {
    // ... existing dependencies ...
    implementation("androidx.glance:glance-appwidget:1.0.0")
    implementation("androidx.glance:glance-material3:1.0.0")
}
  • 创建基础Widget
package com.example.appwidget3

import androidx.compose.runtime.Composable
import androidx.glance.GlanceModifier
import androidx.glance.appwidget.GlanceAppWidget
import androidx.glance.appwidget.GlanceAppWidgetReceiver
import androidx.glance.layout.Column
import androidx.glance.layout.fillMaxSize
import androidx.glance.text.Text

class SimpleAppWidget : GlanceAppWidget() {
    @Composable
    override fun Content() {
        Column(
            modifier = GlanceModifier.fillMaxSize()
        ) {
            Text(text = "Hello from Glance Widget!")
        }
    }
}dgetReceiver : GlanceAppWidgetReceiver() {
   override val glanceAppWidget: GlanceAppWidget = SimpleAppWidget()

  • 创建布局文件 glance_default_loading_layout.xml
<?xml version="1.0" encoding="utf-8"?>
FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
   android:layout_width="match_parent"
   android:layout_height="match_parent"
   android:padding="8dp">
    <ProgressBar
       android:layout_width="24dp"
       android:layout_height="24dp"
       android:layout_gravity="center" />
</FrameLayout>
  • 在AndroidManifest.xml中注册Widget
<manifest>
    <application>
        <!-- ... existing content ... -->
        <receiver
            android:name=".SimpleAppWidgetReceiver"
            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/simple_app_widget_info" />
        </receiver>
    </application>
</manifest>
  • 创建Widget配置文件 simple_app_widget_info.xml
<?xml version="1.0" encoding="utf-8"?>
<appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"
    android:minWidth="180dp"
    android:minHeight="50dp"
    android:updatePeriodMillis="1800000"
    android:initialLayout="@layout/glance_default_loading_layout"
    android:resizeMode="horizontal|vertical"
    android:widgetCategory="home_screen">
</appwidget-provider>

4.3 状态管理

  • 定义状态
data class WeatherWidgetState(
    val temperature: Int = 0,
    val condition: String = "",
    val lastUpdated: Long = System.currentTimeMillis()
)
  • 实现状态更新
class WeatherWidget : GlanceAppWidget() {
    val weatherState = GlanceStateDefinition<WeatherWidgetState>()
    
    @Composable
    override fun Content() {
        val state = currentState<WeatherWidgetState>()
        
        Column {
            Text("温度: ${state.temperature}°C")
            Text("天气: ${state.condition}")
            Text("更新时间: ${formatTime(state.lastUpdated)}")
        }
    }
}

4.4 交互处理

  • 点击事件
@Composable
fun InteractiveWidgetContent() {
    Column {
        Button(
            text = "刷新",
            onClick = actionStartActivity<MainActivity>()
        )
        
        Button(
            text = "设置",
            onClick = actionStartActivity<SettingsActivity>()
        )
    }
}
  • 定期更新
class WidgetUpdateWorker(
    context: Context,
    params: WorkerParameters
) : CoroutineWorker(context, params) {
    
    override suspend fun doWork(): Result {
        // 更新widget数据
        val newState = fetchWeatherData()
        WeatherWidget().updateState(context) { 
            newState 
        }
        return Result.success()
    }
}

4.5 高级功能

  • 响应式布局
@Composable
fun ResponsiveWidgetContent() {
    val sizeMode = LocalSize.current
    
    when (sizeMode) {
        GlanceSize.Small -> SmallWidgetLayout()
        GlanceSize.Medium -> MediumWidgetLayout()
        GlanceSize.Large -> LargeWidgetLayout()
    }
}
  • 主题适配
@Composable
fun ThemedWidgetContent() {
    val colorScheme = GlanceTheme.colors
    
    Column(
        modifier = GlanceModifier
            .background(colorScheme.background)
    ) {
        Text(
            text = "主题适配示例",
            style = TextStyle(
                color = colorScheme.onBackground
            )
        )
    }
}

参考链接:Android Developer–应用widget简介

作者:骆婧颖
原文链接:https://blog.csdn.net/weixin_74107597/article/details/144560360?spm=1001.2014.3001.5501

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值