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

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

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

RxSwiftは、Reactive ExtensionsのSwift向けの実装です。iOS開発に用いられ、リアクティブプログラミングを可能にします。

iOS

iOSとは、Apple製のスマートフォンであるiPhoneやタブレット端末のiPadに搭載しているオペレーションシステム(OS)です。その他にもiPod touch・Apple TVにも搭載されています。

Swift

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

Q&A

解決済

1回答

2187閲覧

RxswiftでUIButton.currentImageの値を渡したい

ttah

総合スコア35

RxSwift

RxSwiftは、Reactive ExtensionsのSwift向けの実装です。iOS開発に用いられ、リアクティブプログラミングを可能にします。

iOS

iOSとは、Apple製のスマートフォンであるiPhoneやタブレット端末のiPadに搭載しているオペレーションシステム(OS)です。その他にもiPod touch・Apple TVにも搭載されています。

Swift

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

0グッド

0クリップ

投稿2020/05/22 02:22

現在のボタンの画像をviewModelへ渡したいのですがどうすれば渡せるでしょか?
何時間か調べたのですがRxswif初心者で解決する事ができませんでした

試した事
textfieldのtextを渡す様に試してみましたが
エラー(Value of type 'Reactive<UIImage>' has no member 'bind')となってしましました

swift

1 button.currentImage.rx.bind(to:viewmodel.image)

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

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

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

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

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

guest

回答1

0

ベストアンサー

View側からimageの変更イベントをViewModel側に伝えたいシチュエーションがいまいち想像できません。ViewModel側にUIButtonに表示するためのUIImageを伝えるObservableを持たせて、ViewModel内で利用するようにすれば解決しませんか?

  • ViewModel側

swift

1let buttonImage: Driver<UIImage> = ... 2 3// imageを使って何か処理をしたい場合 4self.buttonImage 5 .drive(onNext: { image in 6 // 何かする... 7 }) 8 .disposed(by: ...)
  • View側

swift

1viewModel.buttonImage.drive(button.rx.image())

追記

コードを載せていただいていないので推測になりますが、ご質問の状況はMVVMの設計思想から外れた実装をしているために起きているものだと思います。基本的に、MVVMではViewを更新する際にはViewModelを介して行います。ViewとViewModelのやり取りは

  • View => ViewModel - UIに関するイベントをViewModelに伝える
  • ViewModel => View - UIの状態を表すデータを更新し、Viewに伝える

という2通りのものであるべきです。

おそらく、質問者さんのコードだとユーザーが画像を選択した際にUIButtonにそのまま画像を設定しているのだと思いますが、それだとViewがViewModelを介さずに勝手にUIを更新していることになります。これではMVVMの原則に反してしまうので、次のようにViewModelを介するように変更してみてください。これでViewModelとViewの状態は同じに保たれるため、UIButtonのimageをViewModelに伝える方法を考える必要は無くなります。

  1. View: 画像を選択したイベントをViewModelに伝える
  2. ViewModel: UIButtonの画像を表すデータを更新し、Viewに伝える

以下、実装イメージを載せておきます。

  • View側

swift

1viewModel.buttonImage.drive(button.rx.image()) 2registerButton.rx.tap.bind(to: viewModel.onRegisterButtonClick) 3 4// カメラロール等で画像を選択した時のコールバック(適当に読み替えてください) 5func imageDidSelect(image: UIImage) { 6 viewModel.onImageSelected.accept(image) 7}
  • ViewModel側

swift

1// Output: ViewModel => View 2let buttonImage: Driver<UIImage> 3 4// Input: View => ViewModel 5let onImageSelected = PublishRelay<UIImage>() 6let onRegisterButtonClick = PublishRelay<Void>() 7 8init(...) { 9 self.buttonImage = self.onImageSelected.asDriver(...) 10 11 self.onRegisterButtonClick 12 .withLatestFrom(self.buttonImage) 13 .subscribe(onNext: { selectedImage in 14 // 画像をアップロードする処理... 15 }) 16 .disposed(by: ...) 17}

投稿2020/05/23 02:11

編集2020/05/24 04:09
kakajika

総合スコア3131

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

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

ttah

2020/05/23 08:17 編集

kakajika様、回答有難うございます シチュエーションはtwitterのアイコン登録の様なケースです ユーザーがアイコン登録ボタンをタップし写真を選択すると ボタンの画像が選択した写真になります その後ボタンの画像をviewmodelに送りサーバーにUPする事を想定しています 現在はアイコン登録ボタンとは別の、登録ボタン的なのを押した時に viewmodel.buttonImage.accept(self?.button.currentImage) で送ってますが、 アイコンの画像が切り替わったタイミングでviewmodelに値を送る方法が分からない状態です (またそもそもアイコン画像変わるたびに値を送る必要はないのか?普通はどうしているのかな?と疑問を持っている状態です)
kakajika

2020/05/24 03:22

なるほど、なんとなく状況はわかりました。すでに自己解決の回答をされていますが、おそらくMVVMの設計思想についてあまり理解されずにコーディングされていると思いますので、今後も同じような場面で悩むことになると思います。その辺り含めて回答に追記しますね。
ttah

2020/05/26 10:08

追記での解説有難うございます。 mvvm初心者で色々なサイトで勉強してましたがkakajika様の追加が大変分かりやすく 理解が進みました。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問