SwiftUI:具有计算属性的ObservableObject

时间:2020-07-05 12:40:14

标签: ios swift swiftui computed-properties observableobject

我有一个带有自定义月份选择器的简单视图。每次您向左或向右点击雪佛龙,

在DateView内部,我有两个私有变量: startDateOfMonth2:日期 endDateOfMonth2:日期。但是它们仅在DateView中可用。

问题

如何使这两个变量在其他视图中可用?

我试图将它们添加为@Published vars,但出现错误:

属性包装器不能应用于计算出的属性

我只找到了几个类似的问题,但是我无法在我的代码中使用它们的答案。

import SwiftUI

class SelectedDate: ObservableObject {
    @Published var selectedMonth: Date = Date()
    
    @Published var startDateOfMonth2: Date {
        let components = Calendar.current.dateComponents([.year, .month], from: selectedMonth)
        let startOfMonth = Calendar.current.date(from: components)!
        return startOfMonth
    }

    @Published var endDateOfMonth2: Date {
        var components = Calendar.current.dateComponents([.year, .month], from: selectedMonth)
        components.month = (components.month ?? 0) + 1
        let endOfMonth = Calendar.current.date(from: components)!
        return endOfMonth
    }
}

struct DateView: View {
    @EnvironmentObject var selectedDate: SelectedDate
    
    static let dateFormat: DateFormatter = {
        let formatter = DateFormatter()
        formatter.setLocalizedDateFormatFromTemplate("yyyy MMMM")
        return formatter
    }()
    
    var body: some View {
        
        HStack {
            
            Image(systemName: "chevron.left")
                .frame(width: 50, height: 50)
                .contentShape(Rectangle())
                .onTapGesture {
                    self.changeMonthBy(-1)
            }
            
            Spacer()

            Text("\(selectedDate.selectedMonth, formatter: Self.dateFormat)")
            
            Spacer()
            
            Image(systemName: "chevron.right")
                .frame(width: 50, height: 50)
                .contentShape(Rectangle())
                .onTapGesture {
                    self.changeMonthBy(1)
            }
            
        }
        .padding(EdgeInsets(top: 5, leading: 5, bottom: 5, trailing: 5))
        .background(Color.yellow)
    }
    
    func changeMonthBy(_ months: Int) {
        if let selectedMonth = Calendar.current.date(byAdding: .month, value: months, to: selectedDate.selectedMonth) {
            self.selectedDate.selectedMonth = selectedMonth
        }
    }
}

2 个答案:

答案 0 :(得分:2)

无需将您的计算值声明为“已发布”,因为它们取决于“已发布”值。当发布的值更改时,它们将被重新计算。

class SelectedDate: ObservableObject {
    @Published var selectedMonth: Date = Date()
    
    var startDateOfMonth2: Date {
        let components = Calendar.current.dateComponents([.year, .month], from: self.selectedMonth)
        let startOfMonth = Calendar.current.date(from: components)!
        
        print(startOfMonth)
        return startOfMonth
    }

    var endDateOfMonth2: Date {
        var components = Calendar.current.dateComponents([.year, .month], from: self.selectedMonth)
        components.month = (components.month ?? 0) + 1
        let endOfMonth = Calendar.current.date(from: components)!

        print(endOfMonth)
        return endOfMonth
    }
}

在点击时打印endDateOfMonth2时,将设置它正在更改。

答案 1 :(得分:-1)

计算属性实质上是函数。 它不能存储值,只能从getter中的其他变量或属性计算值,并将值更新回setter中计算的值。 仔细考虑您财产的性质。 大多数情况下,您可以直接读写计算属性。 只记得它是功能核心和属性皮肤