2017-09-16 1 views
0

Je suis assez nouveau avec ReactiveSwift et ReactiveCocoa et j'ai l'impression d'avoir rencontré un blocage concernant la manière correcte d'initialiser une propriété qui a des dépendances. Par exemple dans le code suivant, j'essaye d'initialiser une propriété mais j'obtiens une erreur de compilateur qui est attendue. Ma question est de savoir comment/quelle est la "bonne" manière de le faire.Initialisation d'une propriété RACtiveSwift RAC possédant plusieurs dépendances?

class SomeViewModel { 
// illustration purposes, in reality the property (dependency) we will observe will change over time 
let dependency = Property(value: true) 
let dependency2 = Property(value: "dependency2") 
let dependency3 = Property(value: 12345) 
let weightLabel: Property<String> 

// private(set) var weightLabel: Property<String>! 
// using private(set) var weightLabel: Property<String>! works, 
// however this changes the meaning behind using let, because we could 
// reinitalize weightLabel again which is not similar to using a let so not a good alternative 

// let weightLabel: Property<String> = Property(value: "") 
// another solution that will work but will result in a wrong value 
// upon initalization then, changed into the "correct value" thus, i 
// am discrading this as well 

init() { 
    weightLabel = dependency.map { 
     // compiler error, 'self' captured by closure before all members were initalized. 
     // My question is if there is a way to handle this scenario properly 
     if $0 && self.dependency2.value == "dependency2" && self.dependency3.value == 12345 { 
      return "" 
     } 
     return "" 
    } 
} 
} 

Comme vous avez sans doute remarqué plus haut dans les commentaires que je me demande s'il y a un moyen de gérer ce scénario avec ReactiveSwift autres que ceux mentionnés ci-dessus i qui ne sont pas vraiment des solutions idéales.

Répondre

3

L'instrument qui correspond au scénario est combineLatest, qui fournit une version combinée de toutes ces propriétés (flux) chaque fois que l'une d'elles a été mise à jour.

weightLabel = Property.combineLatest(dependency, dependency2, dependency3) 
    .map { d1, d2, d3 in 
     return "Hello World! \(d1) \(d2) \(d3)" 
    } 

En ce qui concerne l'erreur du compilateur, le problème est que vous capturez/référence à self dans une fermeture avant chaque propriété stockée a été initialisées. Selon l'intention, vous pouvez utiliser une liste de capture pour capturer directement les valeurs et les objets qui vous intéressent sans self.

let title: String 
let action:() -> Void 

init() { 
    title = "Hello World!" 

    // `action` has not been initialised when `self` is 
    // being captured. 
    action = { print(self.title) } 

    // ✅ Capture `title` directly. Now the compiler is happy. 
    action = { [title] in print(title) } 
} 
+0

Sweet! Merci pour cette explication détaillée! :) –