SwiftUI - 在视图中包装 Button,以创建自定义按钮

Posted

技术标签:

【中文标题】SwiftUI - 在视图中包装 Button,以创建自定义按钮【英文标题】:SwiftUI - wrapping Button in a view, to create a custom button 【发布时间】:2020-04-28 12:47:46 【问题描述】:

我正在尝试创建自己的 Button 版本,方法是将其包装在视图中,从而打开添加更多功能/隐藏样式修饰符的能力。我知道这不会带来好处,而且 ButtonStyles 很强大。但是为了超级干净的代码,我对它是如何实现的很感兴趣。

在它最精简的形式中,我想写一些类似的东西(基于 Button 自己的签名):

struct MyCustomButton: View 
    let action : () -> Void
    let contents : () -> PrimitiveButtonStyleConfiguration.Label

    var body : some View 
        Button(action: self.action) 
            self.contents()
        
    

然而当我尝试使用它时......

struct MyView : View 
    var body : some View 
        MyCustomButton(action:  doSomething() ) 
            Text("My custom button")
        
    

...我收到以下编译错误:无法将类型“文本”的值转换为闭包结果类型“PrimitiveButtonStyleConfiguration.Label”

【问题讨论】:

【参考方案1】:

想通了:

struct NewButton<Content: View> : View 
    let content : ()-> Content
    let action: () -> Void

    init(@ViewBuilder content: @escaping () -> Content, action: @escaping () -> Void) 
        self.content = content
        self.action = action
    

    var body: some View 
        Button(action: self.action) 
            content()
        
    

【讨论】:

【参考方案2】:

您无需更改Button,根据 SwiftUI 设计意图,只需提供自定义ButtonStyle,如下例所示。

struct ScaleButtonStyle: ButtonStyle 
    let bgColor: Color
    func makeBody(configuration: Self.Configuration) -> some View 
        configuration.label
            .background(bgColor)
            .scaleEffect(configuration.isPressed ? 5 : 1)
    


struct DemoScaleButtonStyle: View 
    var body: some View 
        Button(action:  ) 
            Text("Button")
                .foregroundColor(.white)
        .buttonStyle(ScaleButtonStyle(bgColor: Color.red))
    


【讨论】:

是的 - 我知道可以使用 ButtonStyles。然而,我想更进一步——基于我在 React 中所做的一些工作,可以将视图包装在另一个视图中,并添加功能,而不仅仅是样式。我也知道有 ViewModifiers,但它们的语法感觉不那么自然。如果我要考虑通用问题,如何编写自己的“HStack”,并保持语法风格,而不是我们今天拥有的尾随 .modifier(HStack())。

以上是关于SwiftUI - 在视图中包装 Button,以创建自定义按钮的主要内容,如果未能解决你的问题,请参考以下文章

在 SwiftUI 中使用 @State 属性包装器未更新视图

如何在 SwiftUI 视图中包装 UIBarButtonItem?

如何在 SwiftUI 视图中使用自定义 UI 控件,例如 Button?

SwiftUI 的 List/ForEach 视图中是不是可以有一个 Button(action: -)

如何以编程方式在 SwiftUI 中滚动?

iOS 15.3+ SwiftUI的Form视图中嵌入List及Button若干诡异行为的解决