SwiftUI - возможно ли заставить didSet срабатывать при изменении структуры @Published? - PullRequest
0 голосов
/ 30 марта 2020

Я только что обновился до XCode 11.4, и часть моего кода перестала работать. У меня есть @Published переменные структуры в ObservableObject. Ранее, когда я обновлял свойства для структуры, метод didSet вызывал опубликованное свойство, но это уже не так. Возможно ли, что это поведение было изменено в соответствии с дизайном в последнем обновлении Swift?

Вот тривиальный пример:


import SwiftUI

struct PaddingRect {
  var left: CGFloat = 20
  var right: CGFloat = 20
}

final class SomeStore : ObservableObject {
  @Published var someOtherValue: String = "Waiting for didSet"

  @Published var paddingRect:PaddingRect = PaddingRect() {
    didSet {
      someOtherValue = "didSet fired"
    }
  }
}

struct ObserverIssue: View {
  @ObservedObject var store = SomeStore()

  var body: some View {
    VStack {
      Spacer()

      Rectangle()
        .fill(Color.yellow)
        .padding(.leading, store.paddingRect.left)
        .padding(.trailing, store.paddingRect.right)
        .frame(height: 100)

      Text(store.someOtherValue)

      HStack {
        Button(action: {
          // This doesn't call didSet
          self.store.paddingRect.left += 20

          // This does call didSet, ie. setting the whole thing
//          self.store.paddingRect = PaddingRect(
//            left: self.store.paddingRect.left + 20,
//            right: self.store.paddingRect.right
//          )

        }) {
          Text("Padding left +20")
        }

        Button(action: {
          self.store.paddingRect.right += 20
        }) {
          Text("Padding right +20")
        }
      }

      Spacer()
    }
  }
}

struct ObserverIssue_Previews: PreviewProvider {
    static var previews: some View {
        ObserverIssue()
    }
}

Свойство обновляется, но didSet не срабатывает.

Можно ли получить вложенные свойства структуры для запуска метода didSet издателя?

1 Ответ

0 голосов
/ 30 марта 2020

Наблюдатель за недвижимостью наблюдает за недвижимостью. Проблема заключается в новом синтаксисе Swift, связанном с оболочками свойств. В вашем случае вы пытаетесь наблюдать, изменилось ли значение Опубликовано (которое представляет собой структуру, определяющую оболочку специализированного свойства), а не значение обернутого свойства.

Если вам нужно отслеживать левые или правые значения в PaddingRect просто наблюдайте за этими значениями напрямую.

import SwiftUI


struct PaddingRect {
    var left: CGFloat = 20 {
        didSet {
            print("left padding change from:", oldValue, "to:", left)
        }
    }
    var right: CGFloat = 20 {
        didSet {
            print("right padding change from:", oldValue, "to:", right)
        }
    }
}

final class SomeStore : ObservableObject {
    @Published var someOtherValue: String = "Waiting for didSet"
    @Published var paddingRect:PaddingRect = PaddingRect()
}

struct ContentView: View {
    @ObservedObject var store = SomeStore()

    var body: some View {
        VStack {
            Spacer()

            Rectangle()
                .fill(Color.yellow)
                .padding(.leading, store.paddingRect.left)
                .padding(.trailing, store.paddingRect.right)
                .frame(height: 100)

            Text(store.someOtherValue)

            HStack {
                Button(action: {
                    // This doesn't call didSet
                    self.store.paddingRect.left += 20

                    // This does call didSet, ie. setting the whole thing
                    self.store.paddingRect = PaddingRect(
                        left: self.store.paddingRect.left + 20,
                        right: self.store.paddingRect.right
                    )

                }) {
                    Text("Padding left +20")
                }

                Button(action: {
                    self.store.paddingRect.right += 20
                }) {
                    Text("Padding right +20")
                }
            }

            Spacer()
        }
    }
}

struct ContentView_Preview: PreviewProvider {
    static var previews: some View {
        ContentView()
    }
}

enter image description here

Или воспользуйтесь тем преимуществом, что Опубликованное прогнозируемое значение равно Publisher , и примените следующее модификатор любого вида

.onReceive(store.$paddingRect) { (p) in
            print(p)
        }
...