发布或设置MutableLiveData - 观察者onChanged未调用

时间:2017-12-30 10:38:14

标签: android android-fragments rx-android android-livedata android-viewmodel

我发现了新的Android架构组件,我想通过一个小型测试应用程序测试一对ViewModel / LiveData。后者有两个片段(在ViewPager中),第一个创建/更新卡片列表(通过EditText),第二个片段显示所有卡片。

我的ViewModel:


public class CardsScanListViewModel extends AndroidViewModel {

    private MutableLiveData> cardsLiveData = new MutableLiveData();
    private HashMap cardsMap = new HashMap();

    public CardsScanListViewModel(@NonNull Application application) {
        super(application);
    }

    public MutableLiveData> getCardsLiveData() {
        return this.cardsLiveData;
    }

    public void saveOrUpdateCard(String id) {
        if(!cardsMap.containsKey(id)) {
            cardsMap.put(id, new Card(id, new AtomicInteger(0)));
        }
        cardsMap.get(id).getCount().incrementAndGet();
        this.cardsLiveData.postValue(cardsMap);
    }
}

我的第二个片段:

public class CardsListFragment extends Fragment {

    CardsAdapter cardsAdapter;

    RecyclerView recyclerCardsList;

    public CardsListFragment() {}

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        final CardsScanListViewModel viewModel =
                ViewModelProviders.of(this).get(CardsScanListViewModel.class);

        observeViewModel(viewModel);
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        View v = inflater.inflate(R.layout.fragment_cards_list, container, false);
        recyclerCardsList = v.findViewById(R.id.recyclerCardsList);
        recyclerCardsList.setLayoutManager(new LinearLayoutManager(getActivity()));
        cardsAdapter = new CardsAdapter(getActivity());
        recyclerCardsList.setAdapter(cardsAdapter);

        return v;
    }

    private void observeViewModel(CardsScanListViewModel viewModel) {
        viewModel.getCardsLiveData().observe(this, new Observer  > () {
            @Override
            public void onChanged(@Nullable HashMap  cards) {
                if (cards != null) {
                    cardsAdapter.setCardsList(cards.values());
                }
            }
        });
    }
}

HashMap与我的MutableLiveData一样,更新得很好但我的第二个片段没有通过观察者收到信息。

1 个答案:

答案 0 :(得分:1)

观察 ViewModel新实例,而不是观察第一个使用的ViewModel片段。

final CardsScanListViewModel viewModel =
            ViewModelProviders.of(this).get(CardsScanListViewModel.class);

上面的代码为您的第二个片段CardsScanListViewModel初始化CardsListFragment的新实例,因为您将this作为上下文传递。 如果您更新此片段中的任何数据,则会在ViewModel实例更新

它适用于您的第一个片段,因为它更新数据并观察来自ViewModel

的同一实例的数据

通过在两个片段中传递活动上下文而不是片段上下文来保持ViewModel之间的数据通用启动视图模型。

final CardsScanListViewModel viewModel =
            ViewModelProviders.of(getActivity()).get(CardsScanListViewModel.class);

这将创建CardsScanListViewModel单个实例,并且当片段从{{单个实例 LiveData观察ViewModel时,将在片段之间共享数据1}}。

要进行确认,如果您尚未在适配器中完成此操作,则需要在更新列表后添加notifyDataSetChanged()

private void observeViewModel(CardsScanListViewModel viewModel) {
    viewModel.getCardsLiveData().observe(this, new Observer  > () {
        @Override
        public void onChanged(@Nullable HashMap  cards) {
            if (cards != null) {
                cardsAdapter.setCardsList(cards.values());
                cardsAdapter.notifyDataSetChanged();
            }
        }
    });
}