mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-14 03:20:49 +00:00
Merged via /review-pr -> /prepare-pr -> /merge-pr.
Prepared head SHA: 7751f9c531
Co-authored-by: mbelinky <132747814+mbelinky@users.noreply.github.com>
Co-authored-by: mbelinky <132747814+mbelinky@users.noreply.github.com>
Reviewed-by: @mbelinky
111 lines
4.0 KiB
Swift
111 lines
4.0 KiB
Swift
import SwiftUI
|
|
import Foundation
|
|
import os
|
|
import UIKit
|
|
|
|
final class OpenClawAppDelegate: NSObject, UIApplicationDelegate {
|
|
private let logger = Logger(subsystem: "ai.openclaw.ios", category: "Push")
|
|
private var pendingAPNsDeviceToken: Data?
|
|
weak var appModel: NodeAppModel? {
|
|
didSet {
|
|
guard let model = self.appModel, let token = self.pendingAPNsDeviceToken else { return }
|
|
self.pendingAPNsDeviceToken = nil
|
|
Task { @MainActor in
|
|
model.updateAPNsDeviceToken(token)
|
|
}
|
|
}
|
|
}
|
|
|
|
func application(
|
|
_ application: UIApplication,
|
|
didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]? = nil
|
|
) -> Bool
|
|
{
|
|
application.registerForRemoteNotifications()
|
|
return true
|
|
}
|
|
|
|
func application(_ application: UIApplication, didRegisterForRemoteNotificationsWithDeviceToken deviceToken: Data) {
|
|
if let appModel = self.appModel {
|
|
Task { @MainActor in
|
|
appModel.updateAPNsDeviceToken(deviceToken)
|
|
}
|
|
return
|
|
}
|
|
|
|
self.pendingAPNsDeviceToken = deviceToken
|
|
}
|
|
|
|
func application(_ application: UIApplication, didFailToRegisterForRemoteNotificationsWithError error: any Error) {
|
|
self.logger.error("APNs registration failed: \(error.localizedDescription, privacy: .public)")
|
|
}
|
|
|
|
func application(
|
|
_ application: UIApplication,
|
|
didReceiveRemoteNotification userInfo: [AnyHashable: Any],
|
|
fetchCompletionHandler completionHandler: @escaping (UIBackgroundFetchResult) -> Void)
|
|
{
|
|
self.logger.info("APNs remote notification received keys=\(userInfo.keys.count, privacy: .public)")
|
|
Task { @MainActor in
|
|
guard let appModel = self.appModel else {
|
|
self.logger.info("APNs wake skipped: appModel unavailable")
|
|
completionHandler(.noData)
|
|
return
|
|
}
|
|
let handled = await appModel.handleSilentPushWake(userInfo)
|
|
self.logger.info("APNs wake handled=\(handled, privacy: .public)")
|
|
completionHandler(handled ? .newData : .noData)
|
|
}
|
|
}
|
|
}
|
|
|
|
@main
|
|
struct OpenClawApp: App {
|
|
@State private var appModel: NodeAppModel
|
|
@State private var gatewayController: GatewayConnectionController
|
|
@UIApplicationDelegateAdaptor(OpenClawAppDelegate.self) private var appDelegate
|
|
@Environment(\.scenePhase) private var scenePhase
|
|
|
|
init() {
|
|
Self.installUncaughtExceptionLogger()
|
|
GatewaySettingsStore.bootstrapPersistence()
|
|
let appModel = NodeAppModel()
|
|
_appModel = State(initialValue: appModel)
|
|
_gatewayController = State(initialValue: GatewayConnectionController(appModel: appModel))
|
|
}
|
|
|
|
var body: some Scene {
|
|
WindowGroup {
|
|
RootCanvas()
|
|
.environment(self.appModel)
|
|
.environment(self.appModel.voiceWake)
|
|
.environment(self.gatewayController)
|
|
.task {
|
|
self.appDelegate.appModel = self.appModel
|
|
}
|
|
.onOpenURL { url in
|
|
Task { await self.appModel.handleDeepLink(url: url) }
|
|
}
|
|
.onChange(of: self.scenePhase) { _, newValue in
|
|
self.appModel.setScenePhase(newValue)
|
|
self.gatewayController.setScenePhase(newValue)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
extension OpenClawApp {
|
|
private static func installUncaughtExceptionLogger() {
|
|
NSLog("OpenClaw: installing uncaught exception handler")
|
|
NSSetUncaughtExceptionHandler { exception in
|
|
// Useful when the app hits NSExceptions from SwiftUI/WebKit internals; these do not
|
|
// produce a normal Swift error backtrace.
|
|
let reason = exception.reason ?? "(no reason)"
|
|
NSLog("UNCAUGHT EXCEPTION: %@ %@", exception.name.rawValue, reason)
|
|
for line in exception.callStackSymbols {
|
|
NSLog(" %@", line)
|
|
}
|
|
}
|
|
}
|
|
}
|