在 SwiftUI 小部件中添加观察者的位置

vis*_*kar 3 notifications ios swift widgetkit swiftui

我正在尝试构建一个更好的小部件,我希望每次电池状态或电量发生变化时都会刷新该小部件。我知道两者都需要添加通知。所以我的逻辑是,当有关电池状态的通知发生变化时,我想刷新时间线。

但是我无法理解将观察者放在 TimelineProvider 或 Widget 视图下的位置?

struct timeWidgetEntryView : View {
    var entry: Provider.Entry
    let NC = NotificationCenter.default
    
    init(entry:Provider.Entry) {
        self.entry = entry

    }

    func batteryLevelDidChange(_ notification: Notification) {
        UIDevice.current.isBatteryMonitoringEnabled = true
        var level = UIDevice.current.batteryLevel
        level = level * 100
        WidgetCenter.shared.reloadAllTimelines()
        NSLog("Level changed / state changed")
    }
    
    var body: some View {
        VStack {
            Text(entry.date, style: .time).foregroundColor(.black).frame(minWidth: /*@START_MENU_TOKEN@*/0/*@END_MENU_TOKEN@*/, idealWidth: 100, maxWidth: 100, minHeight: /*@START_MENU_TOKEN@*/0/*@END_MENU_TOKEN@*/, idealHeight: /*@START_MENU_TOKEN@*/100/*@END_MENU_TOKEN@*/, maxHeight: 50, alignment: /*@START_MENU_TOKEN@*/.center/*@END_MENU_TOKEN@*/)
            Text("Battery : \(entry.batteryState)").foregroundColor(.black).frame(minWidth: /*@START_MENU_TOKEN@*/0/*@END_MENU_TOKEN@*/, idealWidth: /*@START_MENU_TOKEN@*/100/*@END_MENU_TOKEN@*/, maxWidth: /*@START_MENU_TOKEN@*/.infinity/*@END_MENU_TOKEN@*/, minHeight: /*@START_MENU_TOKEN@*/0/*@END_MENU_TOKEN@*/, idealHeight: /*@START_MENU_TOKEN@*/100/*@END_MENU_TOKEN@*/, maxHeight: 50, alignment: /*@START_MENU_TOKEN@*/.center/*@END_MENU_TOKEN@*/)
        }
        .cornerRadius(10)
        .frame(width:.infinity, height:.infinity, alignment: .center)
        .onReceive(NotificationCenter.default.publisher(for: UIDevice.batteryStateDidChangeNotification))
               { obj in
                  // Change key as per your "userInfo"
                  if let userInfo = obj.userInfo, let info = userInfo["info"] {
                     print(info)
                  }
            
            WidgetCenter.shared.reloadAllTimelines()
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

但它没有按预期工作,所以我的问题是我应该将观察者移至时间线提供者吗?如果是,在哪里?

paw*_*222 5

您需要收听主应用程序中的通知 - 正如 Apple 教程中所述:

然后,当您收到 时UIDevice.batteryStateDidChangeNotification,只需调用:

WidgetCenter.shared.reloadAllTimelines()
Run Code Online (Sandbox Code Playgroud)

要在您的应用程序不可见时继续收到通知,您可能需要启用后台通知。请参阅此线程: