RX-Android + ViewModel + Retrofit не вызывает OnComplete () - PullRequest
0 голосов
/ 03 марта 2019

Я не могу получить метод OnComplete () для вызова после обработки всех элементов.Мне нужно сделать это для того, чтобы (как минимум, скрыть представление загрузки).Я немного новичок в JavaRX, поэтому я не знаю, где именно проблема.Можете ли вы помочь мне вызвать OnComplete () при вызове всех элементов?

Код выполняет следующие действия:

  1. Показывает представление загрузки и получает список элементов (простоссылки).
  2. Проверьте, являются ли они локальными или удаленными элементами.
  3. Если они локальные, найдите их и добавьте в список.
  4. Если они являются удаленными, загрузитеи добавьте их в список.
  5. После создания списка нарисуйте данные в пользовательском интерфейсе.
  6. Окончательная обработка и скрытие представления загрузки.

Код следующий:

private void loadDataRX(final long fromTime, final long toTime) {
    mLoadingPb.setVisibility(View.VISIBLE);
    iCompositeDisposable.clear();
    iCompositeDisposable.add(mViewModel.getItems(fromTime, toTime)
            .subscribeOn(Schedulers.io())
            .flatMap(items -> {
                Activity context = ItemFragment.this.getActivity();
                if (context == null) {
                    Log.e(TAG, "Cannot present results: context is null");
                    return Flowable.empty();
                } else {
                    context.runOnUiThread(() -> {
                        mItems.clear();
                        mCustomView.reset();
                    });
                    if (items != null && items.size() > 0) {
                        return Flowable.just(items);
                    } else {
                        Log.i(TAG, "No items.");
                        return Flowable.just(Collections.singletonList(new Item(-1))); // This is my current way of solving a similar problem so as to know if I don't have any items
                    }
                }
            })
            .concatMapIterable(items -> items)
            .concatMap(item -> {
                if (item.getUid() == -1) {
                    return Flowable.just(item);
                }
                String file = item.getFileName();
                boolean uploaded = item.isUploaded();
                if (uploaded) { // Remote file
                    if (item.getUid() > 0) {
                        return iRetrofit.create(RestApi.class).getItem(item.getUid());
                    } else {
                        return Flowable.empty();
                    }
                } else { // Local file
                    return Flowable.just(item);
                }
            })
            .observeOn(AndroidSchedulers.mainThread())
            .subscribe(item -> {
                Log.i(TAG, "Loaded items RX");
                if (item instanceof Item) {
                    //Do stuff with the item and the files
                } else if (item instanceof ResponseBody) {
                    //This is dirty but I didn't find another way. So here I basically extract the items and the files from the server's response. At least, it works.
                } else {
                    Log.i(TAG, "No results for the given dates");
                }
            }, throwable -> {
                mLoadingPb.setVisibility(View.GONE);
                Log.e(TAG, "Error: " + throwable.getMessage());
            }, () -> {
                mLoadingPb.setVisibility(View.GONE);
                Log.i(TAG, "Loading results completed"); // Can't get this to be called
            })
    );
}

Заранее спасибо.

1 Ответ

0 голосов
/ 03 марта 2019

Полагаю, что mViewModel.getItems возвращает Flowable.Для завершения потока нам нужно явным образом утилизировать его.

Чтобы решить, что вы можете заставить mViewModel.getItems вернуть Single<List<ItemType>>, а затем преобразовать поток, используя .flatMapObservable { Observable.fromIterable(it) } для обработки каждого элемента.

...