ReactiveSwift Обновить данные

Я новичок в ReactiveSwift. Это получение кода в моей модели представления:

private let viewDidLoadProperty = MutableProperty<Void?>(nil)
    public func viewDidLoad() {

        disposables += self.weatherFetcher.fetchCurrentWeather().startWithResult { (result) in
            switch result {
            case .success(let value):
                _ = value?.list?
                    .map { [weak self] weatherData in
                        if let weather = weatherData.weather?.first {
                            self?.weatherFetcher.fetchWeatherImage(icon: weather.icon).startWithResult { (result) in
                                switch result {
                                case .success(let iconData):
                                    self?.cellViewModels.append(WeatherCellViewModel(with: weatherData, iconData: iconData))
                                case .failure(let error):
                                    print("something went wrong - \(error)")
                                }
                            }
                        } else {
                            self?.cellViewModels.append(WeatherCellViewModel(with: weatherData, iconData: nil))
                        }
                }
            case .failure(let error):
                print(error)
            }
        }

        self.viewDidLoadProperty.value = ()
    }

Когда viewDidLoad вызывается в ViewController, тогда модель представления начинает выборку данных. Как сообщить VC, что выборка завершена и можно вызвать refreshData? Есть ли возможность поймать конец функции viewDidLoad, я имею в виду после выборки.

initCode:

init(weatherFetcher: WeatherFetcher) {
    self.weatherFetcher = weatherFetcher
    didStartLoadingWeather = self.viewDidLoadProperty.signal.skipNil()
}

person user8938042    schedule 11.09.2018    source источник


Ответы (2)


Прежде всего, я бы посоветовал вам использовать ViewModel, который отвечал бы за выполнение этих операций от имени UIViewController.

Отвечаю прямо на ваш вопрос. Вам придется использовать какой-то механизм для хранения данных. Это может быть Property или MutableProperty. Мой совет для первых. Вам также понадобится триггер, чтобы вы могли сообщить об этом, когда viewDidLoad произойдет. Предполагая, что у вас есть ViewModel:

import ReactiveSwift
import enum Result.NoError

public enum ViewState<T> {
    case loading
    case loaded([T])
    case failure(YourError)
}

class ViewModel {

    private let (signal, observer) = Signal<Void, NoError>.pipe()
    let state: Property<ViewState<WeatherCellViewModel>>

    init() {
        let fetch = signal.flatMap(.latest, transform: fetcher)
        self.state = Property.init(initial: .loading then: fetch)
    }

    func fetch() {
        observer.send(value: ())
    }
}

Я оставлю вам доработку fetch. Но:

  1. Этот подход позволяет вам сохранять состояние (через свойство) и разрешать внешний триггер.
  2. Теперь вы должны прочитать значения из state.
  3. fetcher создается во время инициализации и запускается только после вызова функции fetch.
person Rui Peres    schedule 11.09.2018

Хорошая практика для загрузки представлений таблиц с помощью ReactiveSwift (если это ваш случай) заключается в том, чтобы просто привязать данные представления таблицы к MutableProperty ‹[ваши данные]>

вы просто получаете свои данные, а когда значение получено, перезагружаете табличное представление. После этого вид таблицы волшебным образом обновится.


в вашей модели просмотра:

struct WeatherInfo {
    var temp: Int
    var icon: String
}

var data: MutableProperty<[WeatherInfo]>([])

func fetch() -> SignalProducer<Bool, SomeError> {
    return weatherFetcher.fetchCurrentWeather().on(value: { val in
         let mapped = myCustomMapFunc(val) //transforms 'val' to [WeatherInfo]
         self.data.swap(mapped)
    })
}

в вашем контроллере представления:

let viewModel = MyViewModel()

func viewDidLoad() {
   viewModel.fetch().startWithCompleted {
        self.tableView.reloadData()
   }
}

public override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
     return viewModel.data.value.count
}

public override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {

     let cell = self.tableView.dequeueReusableCell(withIdentifier: "myCellId", for:indexPath) as! MyCustomCell
     let info = viewModel.data.value[indexPath.row]
     cell.txtTemp.text = info.temp
     cell.icon = info.icon

     return cell
}

Если это не табличный вид, просто сохраните идею и делайте это как хотите. Например, если это простая ячейка, загрузите в нее новые данные:

в вашей модели просмотра:

// var data: MutableProperty<WeatherInfo>([]) //you don't need it anymore

    func fetch() -> SignalProducer<WethearInfo, SomeError> {
        return weatherFetcher.fetchCurrentWeather().map({
             return myCustomMapFunc($0) //transforms the input into WeatherInfo
        })
    }

в вашем контроллере представления:

    let viewModel = MyViewModel()

    func viewDidLoad() {
       viewModel.fetch().startWithValues { val in
            self.reloadMyCell(info: val)
       }
    }

    func reloadMyCell(info: WeatherInfo) {
       mycell.temp = info.temp
       mycell.icon = info.icon
    }
person Al___    schedule 12.09.2018