如何在不同子视图中使用 TextFields 更改 SwiftUI 应用程序的 FocusState,而不需要刷新导致弹跳效果的视图?

Mic*_*lis 5 focus textfield ios swift swiftui

我的问题:我希望用户能够从 Textfield 转到 TextField,而视图不会弹起,如下面的 gif 所示。

我的用例:我在多个子视图中有多个文本字段和文本编辑器。这些 TextField 是动态生成的,因此我希望 FocusState 成为一个单独的问题。

我在下面制作了一个 gif 示例和代码示例。

在此输入图像描述

请检查一下,任何建议表示赞赏。

正如评论中所建议的,我做了一些更改,但对弹跳没有影响:

 - Using Identfiable does not change the bounce
 - A single observed object or multiple and a view model does not change the bounce
Run Code Online (Sandbox Code Playgroud)

我认为这是来自状态更改刷新。如果不是刷新导致跳出(正如用户在评论中建议的那样),那么是什么?使用 FocusState 时有没有办法阻止这种反弹?

重现:创建一个新的 iOS 应用程序 xcode 项目,并将内容视图替换为下面的代码正文。当用户从一个文本字段转到下一个文本字段时,它似乎会刷新视图,导致整个屏幕跳动。

代码示例

import SwiftUI

struct MyObject: Identifiable, Equatable {
    var id: String
    public var value: String
    init(name: String, value: String) {
        self.id = name
        self.value = value
    }
}

struct ContentView: View {

    @State var myObjects: [MyObject] = [
        MyObject(name: "aa", value: "1"),
        MyObject(name: "bb", value: "2"),
        MyObject(name: "cc", value: "3"),
        MyObject(name: "dd", value: "4")
    ]
    @State var focus: MyObject?

    var body: some View {
        VStack {
            Text("Header")
            ForEach(self.myObjects) { obj in
                Divider()
                FocusField(displayObject: obj, focus: $focus, nextFocus: {
                    guard let index = self.myObjects.firstIndex(of: $0) else {
                        return
                    }
                    self.focus = myObjects.indices.contains(index + 1) ? myObjects[index + 1] : nil
                })
            }
            Divider()
            Text("Footer")
        }
    }
}

struct FocusField: View {

    @State var displayObject: MyObject
    @FocusState var isFocused: Bool
    @Binding var focus: MyObject?
    var nextFocus: (MyObject) -> Void

    var body: some View {
        TextField("Test", text: $displayObject.value)
            .onChange(of: focus, perform: { newValue in
                self.isFocused = newValue == displayObject
            })
            .focused(self.$isFocused)
            .submitLabel(.next)
            .onSubmit {
                self.nextFocus(displayObject)
            }
    }
}
Run Code Online (Sandbox Code Playgroud)

Yrb*_*Yrb 4

在经历了很多次之后,我突然意识到,在使用时,FocusState你真的应该处于ScrollViewForm其他类型的贪婪视图中。即使是一个GeometryReader也会起作用。其中任何一个都会消除反弹。

struct MyObject: Identifiable, Equatable {
    public let id = UUID()
    public var name: String
    public var value: String
}

class MyObjViewModel: ObservableObject {

    @Published var myObjects: [MyObject]
    
    init(_ objects: [MyObject]) {
        myObjects = objects
    }
}


struct ContentView: View {
    @StateObject var viewModel = MyObjViewModel([
        MyObject(name: "aa", value: "1"),
        MyObject(name: "bb", value: "2"),
        MyObject(name: "cc", value: "3"),
        MyObject(name: "dd", value: "4")
    ])

    @State var focus: UUID?
    
    var body: some View {
        VStack {
            Form {
                Text("Header")
                ForEach($viewModel.myObjects) { $obj in
                    FocusField(object: $obj, focus: $focus, nextFocus: {
                        guard let index = viewModel.myObjects.map( { $0.id }).firstIndex(of: obj.id) else {
                            return
                        }
                        focus = viewModel.myObjects.indices.contains(index + 1) ? viewModel.myObjects[index + 1].id : viewModel.myObjects[0].id
                    })
                }
                Text("Footer")
            }
        }
    }
}

struct FocusField: View {
    
    @Binding var object: MyObject
    @Binding var focus: UUID?
    var nextFocus: () -> Void
    
    @FocusState var isFocused: UUID?

    var body: some View {
        TextField("Test", text: $object.value)
            .onChange(of: focus, perform: { newValue in
                self.isFocused = newValue
            })
            .focused(self.$isFocused, equals: object.id)
            .onSubmit {
                self.nextFocus()
            }
    }
}
Run Code Online (Sandbox Code Playgroud)

编辑:

id另外,按照您的方式设置结构是一个非常糟糕的主意。id 应该是唯一的。它在这里有效,但最佳实践是UUID.

第二次编辑:收紧代码。

  • 我对发布的代码不太满意,所以我加强了它并重新发布了所有内容。您会看到我使用了“id”而不是对象,并将“var nextFocus: (MyObject) -> Void”更改为“var nextFocus: () -> Void”,因为不需要将任何内容发送到闭包中。(顺便说一句,我喜欢你的解决方案)。基本上所有修改“TextField”的内容都已被收紧。 (2认同)