1

我正在开发我的第一个 iOS 14 小部件。我知道小部件大多是静态的,不能显示动态内容,如动画或。但是是否可以更改小部件中视图的布局/数量?

由于我是 SwiftUI 的新手,因此我看不到任何更改视图内容的方法。

语境:

假设一个 ToDo 应用程序的小部件应显示最多 5 个条目的按钮/链接。但是,如果应用程序中当前只有 3 个条目,那么小部件当然应该只显示 3 个链接。

该应用程序已经提供了 Today Widget。在这里,我通过简单地将固定数量的条目(例如 5 个)的控件(按钮)添加到小部件视图来解决了这个问题。如果应显示较少的条目,则当小部件更新其视图时,未使用的控件将被隐藏。

在 SwiftUI 中创建具有固定数量的条目(链接)的小部件视图是没有问题的。但是如何隐藏/删除未使用的视图?

在 SwiftUI 中,视图是一个some View变量,我看不到动态更改其内容的方法:

struct MyEntryView: View {
    var body: some View {
        Hstrack {
            // Item 1

            // Item 2

            ...

            // Item n
        }
    }
}
4

1 回答 1

1

小部件视图是静态的,但只要将新条目传递给它们,它们就会重新绘制。

无法创建会在集合更改时自行更新的 Widget 视图。

您可以:

  • 提前创建条目
  • 或在您检测到收藏更改时强制刷新时间线

在这两种情况下,您都需要创建一个条目:

struct SimpleEntry: TimelineEntry {
    let date: Date
    let items: [String]
}

并在视图中显示其项目:

struct SimpleWidgetEntryView: View {
    var entry: SimpleProvider.Entry

    var body: some View {
        VStack {
            ForEach(entry.items, id: \.self) {
                Text($0)
            }
        }
    }
}

唯一的区别是您如何创建条目:

你可以:

  • 提前创建条目(大小items可能会有所不同):
func getTimeline(in context: Context, completion: @escaping (Timeline<SimpleEntry>) -> Void) {
    var entries = [SimpleEntry]()

    let currentDate = Date()

    for offset in 0 ..< 5 {
        let entryDate = Calendar.current.date(byAdding: .minute, value: offset, to: currentDate)!
        entries.append(SimpleEntry(date: entryDate, items: Array(repeating: "Test", count: offset + 1)))
    }

    let timeline = Timeline(entries: entries, policy: .atEnd)
    completion(timeline)
}
  • 或使用当前数据创建单个条目,并且每当数据更改时,通过调用刷新小部件:
WidgetCenter.shared.reloadAllTimelines()

(重新加载时间线的频率可能存在限制)。

于 2020-10-01T16:53:14.240 回答