AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / coding / 问题 / 79223915
Accepted
duckSern1108
duckSern1108
Asked: 2024-11-26 00:45:10 +0800 CST2024-11-26 00:45:10 +0800 CST 2024-11-26 00:45:10 +0800 CST

在闭包中捕获@State[重复]

  • 772
此问题这里已有答案:
SwiftUI:了解使用常量与 @Binding 初始化器时的 .sheet / .fullScreenCover 生命周期 (2 个答案)
14 小时前关闭。

我有一个简单的注册表格,当用户按下时Continue Button,如果他们没有输入任何字段,我会显示一个弹出窗口。

struct ContentView: View {
    @State var firstName = ""
    @State var lastName = ""
    
    @State private var validationMessage = ""
    @State private var showValidationMessage = false
    
    private var isFormValid: Bool {
        !firstName.isEmpty &&
        !lastName.isEmpty
    }
    
    private func checkGeneralFormCompletion() {
        if isFormValid {
            showValidationMessage = false
        } else {
            let errorText = {
                if firstName.isEmpty {
                    return "First name empty"
                } else if lastName.isEmpty {
                    return "Last name empty"
                } else {
                    return ""
                }
            }()
            validationMessage = errorText
            showValidationMessage = !errorText.isEmpty
        }
    }
    
    var body: some View {
        
        VStack(spacing: 12) {
            let _ = Self._printChanges()
            
            TextField("First Name", text: $firstName)
                .textFieldStyle(RoundedBorderTextFieldStyle())
                .padding(.bottom, 10)
            
            TextField("Last Name", text: $lastName)
                .textFieldStyle(RoundedBorderTextFieldStyle())
                .padding(.bottom, 10)
            
            Button(action: {
                checkGeneralFormCompletion()
            }) {
                Text("Continue")
                    .foregroundColor(.white)
                    .padding(.horizontal, 45)
                    .padding([.top, .bottom], 10)
                    .background( Color.red)
                    .cornerRadius(5)
            }
            
            .popover(isPresented: self.$showValidationMessage,
                     attachmentAnchor: .point(.top),
                     arrowEdge: .top,
                     content: { [validationMessage] in
                let _ = print("validationMessage :: popover :: ", validationMessage)
                VStack {
                    Text(validationMessage)
                }
                .multilineTextAlignment(.center)
                .lineLimit(0)
                .foregroundStyle(.black)
                .font(.system(size: 18, weight: .semibold, design: .rounded))
                .padding()
                .presentationCompactAdaptation(.none)
                .fixedSize(horizontal: false, vertical: true)
                .frame(minWidth: 200)
            })
            .padding(.top, 70)
            
        }
        .padding()
//        .onChange(of: validationMessage) { oldValue, newValue in
//            print("CHANGE text VALIDATE MSG")
//        }
//        .onChange(of: showValidationMessage) { oldValue, newValue in
//            print("CHANGE VALIDATE MSG")
//        }
    }
}

如果我注释掉强捕获[validationMessage] in,当用户第一次按下按钮时,弹出窗口将显示空文本并且body不会重新渲染(基于调用)。但是如果我为of_printChanges提供强捕获或添加修饰符,则主体将重新渲染并且弹出窗口将显示正确的文本。closure.popover.onChangeVStack

我的问题是,为什么当我提供 的强捕获时@State,它会导致body重新渲染?任何见解或建议都将不胜感激!

  • 2 2 个回答
  • 48 Views

2 个回答

  • Voted
  1. Best Answer
    malhal
    2024-11-26T03:01:38+08:002024-11-26T03:01:38+08:00

    SwiftUI 具有依赖项跟踪功能。如果之前在 中调用了它的 getter,则只有在设置a 时才会body再次调用。这样做是伪造对设置依赖项跟踪的 getter 的调用,这就是为什么在它更改时调用 的原因。通常这意味着您的事实来源是错误的。似乎在您的情况下,您使用布尔变量作为事实来源,但实际上应该在有消息要显示时使用。您可以从消息中创建一个计算变量以与弹出窗口一起使用,或者改用新的。@Statebody[validationMessage] inbodyBinding<Bool>popover(item:)

    • 2
  2. Vitaliy
    2024-11-26T03:15:31+08:002024-11-26T03:15:31+08:00

    核心问题在于如何showValidationMessage将 用作 的触发器popover,而实际有意义的状态(validationMessage)并没有一致地集成到 SwiftUI 的依赖系统中。

    当前方法(强制强捕获)是一种变通方法,而不是一种干净的设计。相反,请考虑以下解决方案:

    1. 使用popover(item:)

    该popover(item:)API 旨在根据可选的可识别项显示内容。您可以制作validationMessage一个可选的String:

    @State private var validationMessage: String? = nil
    
    private func checkGeneralFormCompletion() {
        if isFormValid {
            validationMessage = nil
        } else {
            validationMessage = firstName.isEmpty ? "First name empty" : "Last name empty"
        }
    }
    
    .popover(item: $validationMessage) { message in
        VStack {
            Text(message)
        }
        .padding()
        .frame(minWidth: 200)
    }
    
    • 这会将 validationMessage 直接绑定到弹出窗口,并且对其进行的更改会自动处理弹出窗口的显示/隐藏,而无需单独的布尔标志。
    1. 使用计算绑定showValidationMessage

    无需明确管理showValidationMessage,直接从中派生validationMessage:

    private var showValidationMessage: Bool {
        !validationMessage.isEmpty
    }
    
    • 更新checkGeneralFormCompletion以确保validationMessage是唯一事实来源:
    private func checkGeneralFormCompletion() {
        validationMessage = firstName.isEmpty ? "First name empty" : lastName.isEmpty ? "Last name empty" : ""
    }
    
    • 更新popover依赖showValidationMessage:
    .popover(isPresented: Binding<Bool>(
        get: { !validationMessage.isEmpty },
        set: { if !$0 { validationMessage = "" } }
    )) {
        VStack {
            Text(validationMessage)
        }
        .padding()
        .frame(minWidth: 200)
    }
    
    • 1

相关问题

  • 将复制活动的序列号添加到 Blob

  • Packer 动态源重复工件

  • 选择每组连续 1 的行

  • 图形 API 调用列表 subscribedSkus 状态权限不足,但已授予权限

  • 根据列值创建单独的 DF 的函数

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    Vue 3:创建时出错“预期标识符但发现‘导入’”[重复]

    • 1 个回答
  • Marko Smith

    为什么这个简单而小的 Java 代码在所有 Graal JVM 上的运行速度都快 30 倍,但在任何 Oracle JVM 上却不行?

    • 1 个回答
  • Marko Smith

    具有指定基础类型但没有枚举器的“枚举类”的用途是什么?

    • 1 个回答
  • Marko Smith

    如何修复未手动导入的模块的 MODULE_NOT_FOUND 错误?

    • 6 个回答
  • Marko Smith

    `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它?

    • 3 个回答
  • Marko Smith

    何时应使用 std::inplace_vector 而不是 std::vector?

    • 3 个回答
  • Marko Smith

    在 C++ 中,一个不执行任何操作的空程序需要 204KB 的堆,但在 C 中则不需要

    • 1 个回答
  • Marko Smith

    PowerBI 目前与 BigQuery 不兼容:Simba 驱动程序与 Windows 更新有关

    • 2 个回答
  • Marko Smith

    AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String”

    • 1 个回答
  • Marko Smith

    我正在尝试仅使用海龟随机和数学模块来制作吃豆人游戏

    • 1 个回答
  • Martin Hope
    Aleksandr Dubinsky 为什么 InetAddress 上的 switch 模式匹配会失败,并出现“未涵盖所有可能的输入值”? 2024-12-23 06:56:21 +0800 CST
  • Martin Hope
    Phillip Borge 为什么这个简单而小的 Java 代码在所有 Graal JVM 上的运行速度都快 30 倍,但在任何 Oracle JVM 上却不行? 2024-12-12 20:46:46 +0800 CST
  • Martin Hope
    Oodini 具有指定基础类型但没有枚举器的“枚举类”的用途是什么? 2024-12-12 06:27:11 +0800 CST
  • Martin Hope
    sleeptightAnsiC `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它? 2024-11-09 07:18:53 +0800 CST
  • Martin Hope
    The Mad Gamer 何时应使用 std::inplace_vector 而不是 std::vector? 2024-10-29 23:01:00 +0800 CST
  • Martin Hope
    Chad Feller 在 5.2 版中,bash 条件语句中的 [[ .. ]] 中的分号现在是可选的吗? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench 为什么双破折号 (--) 会导致此 MariaDB 子句评估为 true? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng 为什么 `dict(id=1, **{'id': 2})` 有时会引发 `KeyError: 'id'` 而不是 TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String” 2024-03-20 03:12:31 +0800 CST
  • Martin Hope
    MarkB 为什么 GCC 生成有条件执行 SIMD 实现的代码? 2024-02-17 06:17:14 +0800 CST

热门标签

python javascript c++ c# java typescript sql reactjs html

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve