SwiftUI 和三指撤消手势

SwiftUI and the three-finger undo gesture

我正在尝试在 iOS 的 SwiftUI 应用程序中实现撤消,但我无法使撤消手势起作用。这是一个演示问题的示例:

class Model: ObservableObject {
    @Published var active = false

    func registerUndo(_ newValue: Bool, in undoManager: UndoManager?) {
        let oldValue = active
        undoManager?.registerUndo(withTarget: self) { target in
            target.active = oldValue
        }
        active = newValue
    }
}

struct UndoTest: View {
    @ObservedObject private var model = Model()
    @Environment(\.undoManager) var undoManager

    var body: some View {
        VStack {
            Toggle(isOn: Binding<Bool>(
                get: { self.model.active },
                set: { self.model.registerUndo([=10=], in: self.undoManager) }
            )) {
                Text("Toggle")
            }
            .frame(width: 120)
            Button(action: {
                self.undoManager?.undo()
            }, label: {
                Text("Undo")
                    .foregroundColor(.white)
                    .padding()
                    .background(self.undoManager?.canUndo == true ? Color.blue : Color.gray)
            })
        }
    }
}

切换开关然后点击撤消按钮可以正常工作。使用三指撤消手势或摇动撤消不会执行任何操作。怎么配合系统手势?

似乎编辑手势需要 window 有第一响应者,并且 SwiftUI 没有设置任何 UIWindow 默认情况下想要选择作为第一响应者的东西。

如果您继承 UIHostingController,并且在您的子类中,您将 canBecomeFirstResponder 覆盖为 return true,那么 UIWindow 会将您的控制器设置为默认为第一响应者,这似乎足以启用编辑手势。

我在 iPad Pro 运行 iPadOS 13.1 beta 2 (17A5831c) 上测试了以下代码。它主要有效。我相信有一个 iOS 错误,可能已在较新的测试版中修复:当撤消堆栈为空时,手势有时不起作用(即使可以重做操作)。切换到主屏幕然后返回测试应用程序(不终止测试应用程序)似乎使编辑手势再次起作用。

import UIKit
import SwiftUI

class MyHostingController<Content: View>: UIHostingController<Content> {
    override var canBecomeFirstResponder: Bool { true }
}

class Model: ObservableObject {
    init(undoManager: UndoManager) {
        self.undoManager = undoManager
    }

    let undoManager: UndoManager

    @Published var active = false {
        willSet {
            let oldValue = active
            undoManager.registerUndo(withTarget: self) { me in
                me.active = oldValue
            }
        }
    }
}

struct ContentView: View {
    @ObservedObject var model: Model
    @Environment(\.undoManager) var undoManager

    var body: some View {
        VStack {
            Toggle("Active", isOn: $model.active)
                .frame(width: 120)
            HStack {
                Button("Undo") {
                    withAnimation {
                        self.undoManager?.undo()
                    }
                }.disabled(!(undoManager?.canUndo ?? false))
                Button("Redo") {
                    withAnimation {
                        self.undoManager?.redo()
                    }
                }.disabled(!(undoManager?.canRedo ?? false))
            }
        }
    }
}

class SceneDelegate: UIResponder, UIWindowSceneDelegate {

    var window: UIWindow?

    func scene(_ scene: UIScene, willConnectTo session: UISceneSession, options connectionOptions: UIScene.ConnectionOptions) {
        guard let scene = scene as? UIWindowScene else { return }

        let window = UIWindow(windowScene: scene)
        let model = Model(undoManager: window.undoManager!)
        let contentView = ContentView(model: model)

        window.rootViewController = MyHostingController(rootView: contentView)
        self.window = window
        window.makeKeyAndVisible()
    }
}