一个Android开发学习开发ios桌面小组件

1 创建widget extension

2 创建完成之后 理论上就可以运行了

关键代码包括 TimelineProvider TimelineEntry

正常代码是下面这样的

js 复制代码
//
//  home_widget_study.swift
//  home_widget_study
//
//  Created by gy on 2025/3/7.
//

import WidgetKit
import SwiftUI

struct Provider: TimelineProvider {
    func placeholder(in context: Context) -> SimpleEntry {
        SimpleEntry(date: Date(), emoji: "😀")
    }

    func getSnapshot(in context: Context, completion: @escaping (SimpleEntry) -> ()) {
        let entry = SimpleEntry(date: Date(), emoji: "😀")
        completion(entry)
    }

    func getTimeline(in context: Context, completion: @escaping (Timeline<Entry>) -> ()) {
        var entries: [SimpleEntry] = []

        // Generate a timeline consisting of five entries an hour apart, starting from the current date.
        let currentDate = Date()
        for hourOffset in 0 ..< 5 {
            let entryDate = Calendar.current.date(byAdding: .hour, value: hourOffset, to: currentDate)!
            let entry = SimpleEntry(date: entryDate, emoji: "😀")
            entries.append(entry)
        }

         //entries 定义了刷新的时间 这里数组长度为5 每个元素的时间相差一小时 意思就是从小组件添加之后一小时刷新一次 好像这个时间不能设置的太短 最短好像是5分钟
        let timeline = Timeline(entries: entries, policy: .atEnd)
        completion(timeline)
    }

//    func relevances() async -> WidgetRelevances<Void> {
//        // Generate a list containing the contexts this widget is relevant in.
//    }
}


//里面的属性可以根据自己的需求 自定义
//但是这个date 好像是必须的 意思是下次刷新的时间 因为小组件不会自己刷新
struct SimpleEntry: TimelineEntry {
    let date: Date
    let emoji: String
}

struct home_widget_studyEntryView : View {
    var entry: Provider.Entry

    var body: some View {
        VStack {
            Text("Time:")
            Text(entry.date, style: .time)

            Text("Emoji:")
            Text(entry.emoji)
        }
    }
}

struct home_widget_study: Widget {
    let kind: String = "home_widget_study"

    var body: some WidgetConfiguration {
        StaticConfiguration(kind: kind, provider: Provider()) { entry in
            if #available(iOS 17.0, *) {
                home_widget_studyEntryView(entry: entry)
                    .containerBackground(.fill.tertiary, for: .widget)
            } else {
                home_widget_studyEntryView(entry: entry)
                    .padding()
                    .background()
            }
        }
        .configurationDisplayName("My Widget")
        .description("This is an example widget.")
    }
}

#Preview(as: .systemSmall) {
    home_widget_study()
} timeline: {
    SimpleEntry(date: .now, emoji: "😀")
    SimpleEntry(date: .now, emoji: "🤩")
}

如果你的TimerLineProvider初始代码如下,运行会报错。你可以参考上面的代码修改调整一下。下面的代码好像是高版本的写法。

js 复制代码
func placeholder(in context: Context) -> SimpleEntry {
        SimpleEntry(date: Date(),model: 0)
    }

    func snapshot(for configuration: ConfigurationAppIntent, in context: Context) async -> SimpleEntry {
        SimpleEntry(date: Date(), configuration: configuration,model: "456")
    }
    
    func timeline(for configuration: ConfigurationAppIntent, in context: Context) async -> Timeline<SimpleEntry> {
        var entries: [SimpleEntry] = []

        print("timelinetimelinetimelinetimelinetimeline===================")
        // Generate a timeline consisting of five entries an hour apart, starting from the current date.
        let currentDate = Date()
        let userdefaults = UserDefaults(suiteName: "group.com.aiintelligent.iossoftware.homewidget")
        let model = userdefaults?.string(forKey: "model") ?? "model"
        let a = print("==========model===",model)
        for hourOffset in 0 ..< 5 {
            let entryDate = Calendar.current.date(byAdding: .minute, value: hourOffset, to: currentDate)!
            let entry = SimpleEntry(date: entryDate,model: model)
            entries.append(entry)
        }

        return Timeline(entries: entries, policy: .atEnd)
    }

如果你的widget body 是如下代码 也参照上面的修改 这个好像也是高版本代码

js 复制代码
var body: some WidgetConfiguration {
        AppIntentConfiguration(kind: kind, intent: ConfigurationAppIntent.self, provider: Provider()) { entry in
            1EntryView(entry: entry)
                .containerBackground(.fill.tertiary, for: .widget)
        }
    }

3 主动刷新小组件

js 复制代码
WidgetCenter.shared.reloadTimelines(ofKind: "home_widget_study")

ofKind 在这里

4 主程序与小组件之间通信 传递数据

group_name 需要在开发者后台配置 在小组件 跟 主程序使用的group_name一定要一样,切必须是开发者后台配置的值。

js 复制代码
 var userdefaults = UserDefaults(suiteName: "group_name")
相关推荐
JMchen1231 小时前
Android后台服务与网络保活:WorkManager的实战应用
android·java·网络·kotlin·php·android-studio
crmscs2 小时前
剪映永久解锁版/电脑版永久会员VIP/安卓SVIP手机永久版下载
android·智能手机·电脑
localbob2 小时前
杀戮尖塔 v6 MOD整合版(Slay the Spire)安卓+PC端免安装中文版分享 卡牌肉鸽神作!杀戮尖塔中文版,电脑和手机都能玩!杀戮尖塔.exe 杀戮尖塔.apk
android·杀戮尖塔apk·杀戮尖塔exe·游戏分享
机建狂魔2 小时前
手机秒变电影机:Blackmagic Camera + LUT滤镜包的专业级视频解决方案
android·拍照·摄影·lut滤镜·拍摄·摄像·录像
hudawei9962 小时前
flutter和Android动画的对比
android·flutter·动画
lxysbly4 小时前
md模拟器安卓版带金手指2026
android
儿歌八万首5 小时前
硬核春节:用 Compose 打造“赛博鞭炮”
android·kotlin·compose·春节
消失的旧时光-19437 小时前
从 Kotlin 到 Dart:为什么 sealed 是处理「多种返回结果」的最佳方式?
android·开发语言·flutter·架构·kotlin·sealed
Jinkxs7 小时前
Gradle - 与Groovy/Kotlin DSL对比 构建脚本语言选择指南
android·开发语言·kotlin
&有梦想的咸鱼&7 小时前
Kotlin委托机制的底层实现深度解析(74)
android·开发语言·kotlin