如何在自定义视图中创建更改属性的修饰符

Posted

技术标签:

【中文标题】如何在自定义视图中创建更改属性的修饰符【英文标题】:How can create modifiers that are changing properties in a Custom View 【发布时间】:2021-02-04 19:42:39 【问题描述】:

我有这个用 SwiftUI 编写的 CircularProgressView。我想创建一种方法来添加一些自定义修饰符,这样我就可以从父视图中更改一些属性。下面的代码不起作用。

我说的是 func lineWidth(width:CGFloat) -> some View , func progressBarColor(color:Binding) -> some View , func tintColor(color:Color) -> 一些视图

struct CircularProgressView: View 


@Binding var progress: CGFloat

@State private var lineWidth:CGFloat = 15
@State private var progressBarColor:Color = Color.red
@State private var tintColor:Color = Color.gray

 func lineWidth(width:CGFloat) -> some View 
    self.lineWidth = width
    return self


func progressBarColor(color:Color) -> some View 
    self.progressBarColor = color
    return self


func tintColor(color:Color) -> some View 
    self.tintColor = color
    return self


var body: some View 
    GeometryReader  geometry in
        ZStack 
            Circle()
                .stroke(tintColor, lineWidth: lineWidth)
                .aspectRatio(1, contentMode: .fit)
            Circle()
                .trim(from: 0.0, to: progress)
                .stroke(progressBarColor, lineWidth: lineWidth)
                .rotationEffect(Angle(degrees: -90))
                .aspectRatio(1, contentMode: .fit)
        
    

例如。我想像这样使用这个 CircularProgressView:

    struct ContentView: View 
    var body: some View 
        CircularProgressView(progress: .constant(0.3)).lineWidth(width: 20)
    

如何使用 ViewModifiers 实现这一点?我认为这一切都错了吗?

【问题讨论】:

什么不行,你想达到什么,不清楚? 我做了一些改变。对此感到抱歉 【参考方案1】:

你只是不需要 @State 包装所有这些属性,所以使用 as

struct CircularProgressView: View 

  @Binding var progress: CGFloat

  private var lineWidth:CGFloat = 15
  private var progressBarColor:Color = Color.red
  private var tintColor:Color = Color.gray

// ... other code

  func lineWidth(width:CGFloat) -> some View        // << here !!
    var newView = self
    newView.lineWidth = width
    return newView
  

@Binding 应该保持原样,因为它就像对外部事实来源的引用(某些动态属性持有真实值),绑定本身不做任何事情 - 它就像一个代理。

所以下面的没用

func progressBarColor(color:Binding<Color>) -> some View 
    self.progressBarColor = color.wrappedValue
    return self

你应该像这样使用它

struct ContentView: View 
   @State private var progress = 0.3
    var body: some View 
        CircularProgressView(progress: $progress).lineWidth(width: 20)
        // ... modify progress somewhere later
    

*** 但是我在提供的场景中看不到为什么您需要绑定...如果您将其从 CircularProgressView 中删除,那么以下内容也可以使用

struct ContentView: View 
   @State private var progress = 0.3
    var body: some View 
        CircularProgressView(progress: progress).lineWidth(width: 20)
        // ... modify progress somewhere later
    

【讨论】:

我不是在谈论进步。我说的是 .lineWidth(width: 20)。它不工作 查看修复的修改器以进行演示。

以上是关于如何在自定义视图中创建更改属性的修饰符的主要内容,如果未能解决你的问题,请参考以下文章

表单修饰符自定义指令计算属性侦听器过滤器生命周期

表单修饰符自定义指令计算属性侦听器过滤器生命周期

在自定义表格单元中更改图像视图

ios) 如何在自定义键盘上方创建建议视图

在自定义视图单元中创建可绑定事件处理程序

在自定义 tableviewcell 中更改 UIImage