質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.50%
Swift

Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

Q&A

解決済

2回答

3357閲覧

RxSwift 双方向 Bind

tosi_1211

総合スコア138

Swift

Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

0グッド

0クリップ

投稿2017/03/27 08:14

##質問
現在やってる方法より良いものがアドバイスを頂きたいです。

##やりたいこと

  • viewModel側でデフォルト値を代入
  • textFieldにデフォルト値を反映
  • textFieldのユーザーの入力をviewModelのtextにbind

##どうやっているか
RxSwiftを使って以下のように実装しています。

import RxSwift class ViewModel { let text: Variable<String> = Variable<String>("") init() { setDefaultValue() } private func setDefaultValue() { text.value = "hogehoge" } }
class ViewController { @IBOutlet weak var textField: NoCaretTextField! private let disposeBag = DisposeBag() override func viewDidLoad() { super.viewDidLoad() viewModel.text.asObservable() .bindTo(textField.rx.text) .disposed(by: disposeBag) textField.rx.text.orEmpty .bindTo(viewModel.text) .disposed(by: disposeBag) } }

よろしくお願いします!

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答2

0

ベストアンサー

RxSwiftの公式サンプルに<->というOperatorがありますが、
そちらの箇所をスッキリと表現したいのでしたら<->を利用すれば
textField.rx.text <-> viewModel.text
といったような表記が可能になります。

実行可能なコードを添付致しますので、ご確認頂ければ。

Swift

1import UIKit 2import RxSwift 3import RxCocoa 4 5class ViewModel { 6 let text = Variable<String?>("初期値") 7} 8 9class ViewController: UIViewController { 10 11 @IBOutlet weak var textField: UITextField! 12 private let bag = DisposeBag() 13 private let viewModel = ViewModel() 14 15 override func viewDidLoad() { 16 super.viewDidLoad() 17 (textField.rx.text <-> viewModel.text).addDisposableTo(bag) 18 } 19} 20 21infix operator <-> : DefaultPrecedence 22 23func <-> <T>(property: ControlProperty<T>, variable: Variable<T>) -> Disposable { 24 let bindToUIDisposable = variable.asObservable() 25 .bindTo(property) 26 let bindToVariable = property 27 .subscribe(onNext: { n in 28 variable.value = n 29 }, onCompleted: { 30 bindToUIDisposable.dispose() 31 }) 32 33 return Disposables.create(bindToUIDisposable, bindToVariable) 34}

詳細はこちら:RxExample/Operators.swift

投稿2017/03/29 06:53

編集2017/03/29 07:02
oden

総合スコア76

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

0

初期値はinitではなく宣言時に入れて良いと思います。
またnilが代入される可能性のあるOptionalの値を利用するのであれば、orEmptyを使ってしまうとnilが代入されても通知されないので、ViewModel側へ明示的にnilが代入される可能性があることを宣言しておく方が良いと思います。

Swift

1import UIKit 2import RxSwift 3import RxCocoa 4 5class ViewModel { 6 let text = Variable<String?>("hogehoge") 7} 8 9class ViewController: UIViewController { 10 11 @IBOutlet weak var textField: UITextField! 12 private let viewModel = ViewModel() 13 private let bag = DisposeBag() 14 15 override func viewDidLoad() { 16 super.viewDidLoad() 17 18 viewModel.text.asObservable() 19 .bindTo(textField.rx.text) 20 .disposed(by: bag) 21 textField.rx.text 22 .bindTo(viewModel.text) 23 .disposed(by: bag) 24 } 25}

投稿2017/03/28 09:46

oden

総合スコア76

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

tosi_1211

2017/03/28 18:01

アドバイスありがとうございます! すごく参考になりました。 質問の内容としては、 ``` viewModel.text.asObservable() .bindTo(textField.rx.text) .disposed(by: disposeBag) textField.rx.text.orEmpty .bindTo(viewModel.text) .disposed(by: disposeBag) ``` ここの部分もっとスッキリ書きたいという質問でした
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.50%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問