SwiftUI Widgets 之02 TimelineProvider 基础概念

TimelineProvider

建议WidgetKit何时更新窗口小部件显示的类型。

protocol TimelineProvider

使用教程

在不同的时间,WidgetKit向提供者请求时间轴。 时间线是符合TimelineEntry的对象数组。 每个时间轴条目都有一个日期,您可以指定其他属性来显示小部件。
例如,考虑显示游戏角色健康等级的小部件。 在游戏中,当角色的健康水平低于100%时,其恢复速度为每小时25%。 如果角色的健康等级为25%,则提供商会创建一个包含四个条目的时间表:
当前时间,健康水平25%
从现在开始1小时,健康水平50%
从现在开始2个小时,健康水平达到75%
从现在开始3个小时,健康水平100%
以下代码显示了封装此信息的结构。

struct CharacterDetailEntry: TimelineEntry {
    var date: Date
    var healthLevel: Double
}

WidgetKit通过以下两种方式之一请求时间轴条目:

  • snapshot 单个即时快照,代表小部件的当前状态。
  • entries 一系列条目,包括当前时刻以及小部件状态将改变的任何将来的日期(如果知道)。

在瞬态情况下(例如,当用户添加窗口小部件时)显示窗口小部件时,WidgetKit会发出快照请求。 WidgetKit提供了一个上下文参数,其中包含有关如何使用该条目的详细信息,包括它是小部件库的预览,还是要显示的小部件的系列或大小。如果context.isPreview为true,则该小部件将出现在小部件库中,并且需要提供者的快速响应。如果生成快照所需的信息不可用,或者需要额外的时间来加载,请改用示例数据。例如,如果确定角色的健康等级需要从服务器获取数据,则小部件可以显示75%的健康等级。以下代码显示了游戏小部件如何实现其快照方法。

struct CharacterDetailProvider: TimelineProvider {
    func snapshot(with context: Context, completion: @escaping (Entry) -> ()) {
        let date = Date()
        let entry: CharacterDetailEntry

        if context.isPreview && !hasHealthLevel {
            entry = CharacterDetailEntry(date: date, healthLevel: 0.75)
        } else {
            entry = CharacterDetailEntry(date: date, healthLevel: currentHealthLevel)
        }
        completion(entry)
    }
}

用户从窗口小部件库中添加窗口小部件后,WidgetKit发出时间轴请求。 由于您的窗口小部件扩展程序并不总是运行,因此WidgetKit需要知道何时激活它来更新窗口小部件。 您的提供商生成的时间线会在您希望更新窗口小部件时通知WidgetKit。 以下示例显示了角色的健康水平为25%时生成的时间轴。

struct CharacterDetailProvider: TimelineProvider {
    func timeline(with context: Context, completion: @escaping (Timeline<CharacterDetailEntry>) -> ()) {
        var date = Date()
        var healthLevel = 0.25
        var entries: [CharacterDetailEntry] = []

        while healthLevel <= 1 {
            // Add the current health level for a given date.
            entries.append(CharacterDetailEntry(date: date, healthLevel: healthLevel))

            // Health recovers at 25 percent per hour, with a maximum of 100 percent.
            healthLevel = min(1, healthLevel + 0.25)

            // Move the date forward by 1 hour.
            date = Calendar.current.date(byAdding: .hour, value: 1, to: date)!
        }

        // Create the timeline and call the completion handler.
        let timeline = Timeline(entries: entries, policy: .atEnd)
        completion(timeline)
    }
}

如果您的提供者需要执行异步工作以生成时间线(例如,从服务器获取数据),请存储对完成处理程序的引用,并在完成异步工作后调用它。

确定刷新策略

创建时间轴时,提供程序会指定一个刷新策略,该策略控制WidgetKit请求新时间轴的时间。 默认行为是使用.atEnd在时间轴中的条目指定的最后日期之后请求新的时间轴。 但是,如果WidgetKit请求新时间轴的日期不同,则可以将刷新策略指定为.after(date :)。 例如,一条龙将在2.5小时后出现,并可能与游戏角色进行战斗。 由于战斗的结果可能会改变角色的健康状况,因此提供商可以告诉WidgetKit在战斗后请求新的时间表。

// Request a timeline refresh after 2.5 hours.
let date = Calendar.current.date(byAdding: .minute, value: 150, to: Date())
let timeline = Timeline(entries: entries, policy: .after(date))
completion(timeline)

使用不同日期的其他例子包括:

在显示股市详细信息的小部件中,您可以指定下一个市场开市或收市日期,因为信息通常不会在一夜之间或在周末期间发生变化。
飞行着陆后,飞行跟踪小部件可能会继续显示“飞行着陆”指示。在这种情况下,您可以指定一个日期,而不是在航班降落之后指定日期,以便其状态在清除之前的一段时间内保持可见状态。
或者,如果将来的事件不可预测,则可以通过为策略指定.never来告诉WidgetKit根本不请求新的时间轴。在这种情况下,当有新的时间轴可用时,您的应用程序将调用WidgetCenter函数reloadTimelines(ofKind :)。使用.never有意义的一些示例包括:
当用户拥有配置为显示角色健康状况的小部件,但该角色不再积极参与战斗并且其健康水平不会改变时。
窗口小部件的内容取决于用户是否已登录帐户而当前尚未登录。
在这两个示例中,当您的应用确定状态已更改时,它将调用WidgetCenter函数reloadTimelines(ofKind :),并且WidgetKit请求新的时间轴。

推荐

基础文章推荐

经典教程推荐

技术源码推荐

推荐文章

CoreData篇

Combine篇

TextField篇

JSON文件篇


一篇文章系列

技术交流

QQ:3365059189
SwiftUI技术交流QQ群:518696470

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