如何创建发出单个事件的LiveData并仅通知最后订阅的观察者?

问题描述 投票:3回答:1

我创建了实时数据,它发出了一个单独的事件,就像在这个example中一样。

我的问题是:当LiveData中的值发生变化时,如何仅通知最后订阅的观察者?

我想到的是将观察者存储在SingleLiveData类的链表中,然后仅当传递的观察者与列表的最后一个元素相同时才调用super.observe

我不确定这是不是最好的方法。

我想使用这种机制将FAB点击事件从活动传播到ViewPager内部显示的片段。片段被动态添加到查看寻呼机适配器,所以让我们说我们知道片段的顺序。

android android-architecture-components android-jetpack android-livedata
1个回答
0
投票

最后,我找到了解决此问题的方法。我不得不放弃发出单个事件的实时数据,因为它不能像我需要的那样表现。

而不是这个,我使用了简单的可变实时数据,它发出了一个事件对象,它包含了像JoseAlcérreca这个article的最后一段中的数据。

我在视图寻呼机中显示片段,所以我当时只有一个可见片段。

所以我的视图模型看起来像这样:

class ActionViewModel : ViewModel() {
  private val onCreateLiveData: MutableLiveData<Event<String>> = MutableLiveData()

  fun observeOnCreateEvent(): LiveData<Event<String>> = onCreateLiveData

  fun onCreateCollectionClick(message: String) {
    this.onCreateLiveData.value = Event(message)
  }
}

事件包装器类实现如下所示:

/*Used as a wrapper for data that is exposed via a LiveData that represents an 
 event.*/

open class Event<out T>(private val content: T) {

  var hasBeenHandled = false
    private set // Allow external read but not write

  /**
   * Returns the content and prevents its use again.
  */
  fun getContentIfNotHandled(): T? {
    return if (hasBeenHandled) {
      null
    } else {
      hasBeenHandled = true
      content
    }
  }

  /**
    * Returns the content, even if it's already been handled.
  */
  fun peekContent(): T = content
}

现在我们可以在片段中观察这样的事件:

override fun onActivityCreated(savedInstanceState: Bundle?) {
   super.onActivityCreated(savedInstanceState)

   actionViewModel = ViewModelProviders.of(requireActivity()).get(ActionViewModel::class.java)
   actionViewModel.observeOnCreateEvent()
       .observe(this, Observer {
         it?.takeIf { userVisibleHint }?.getContentIfNotHandled()?.let {
           //DO what ever is needed
         }
       })
}

如果片段当前对用户可见,则片段userVisibleHint属性将返回true。因为我们现在只显示一个片段,这对我们有效。这意味着片段只有在可见时才会访问事件数据。

此外,Event包装器的实现只允许读取一次该值,因此Observer每次下次获取此事件时,其值将为null,我们将忽略它。

结论:这样我们就可以模拟单个事件实时数据,它只通知最后订阅的观察者。

© www.soinside.com 2019 - 2024. All rights reserved.