Created
April 26, 2018 10:25
-
-
Save JoseAlcerreca/5b661f1800e1e654f07cc54fe87441af to your computer and use it in GitHub Desktop.
An event wrapper for data that is exposed via a LiveData that represents an event.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/** | |
* 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 | |
} | |
That looks nearly like my latest solution. However, I don't use a special EventObserver, since a Kotlin extension function can do the job:
class Event<out T>(private val content: T) {
private val consumedScopes = HashSet<String>()
fun isConsumed(scope: String = "") = consumedScopes.contains(scope)
@MainThread
fun consume(scope: String = ""): T? {
return if (isConsumed(scope)) {
null
} else {
consumedScopes.add(scope)
content
}
}
fun peek(): T = content
}
fun <T> LiveData<Event<T>>.observeEvent(lifecycleOwner: LifecycleOwner, scope: String = "", observer: Observer<T>) {
observe(lifecycleOwner) { event ->
event?.consume(scope)?.let { observer.onChanged(it) }
}
}
// How to use it
myObservable.observeEvent { ... }
myObservable.observeEvent("specialScope") { ... }
This is so concise.
In Kotlin, passing events to a single observer is also possible with Channel.receiveAsFlow()
and lifecycle-aware collector. Advantages:
- Can queue multiple events while observer is inactive (configuration change, app in background, fragment in back stack), with customizable buffer size and
onBufferOverflow
strategy - Optionally supports
null
values - No need to wrap data in
Event
That looks nearly like my latest solution. However, I don't use a special EventObserver, since a Kotlin extension function can do the job:
class Event<out T>(private val content: T) { private val consumedScopes = HashSet<String>()
Adding more syntactic sugar to choirwire's contribution
class Event<out T>(private val content: T) {
private val consumedScopes by lazy { HashSet<String>() }
fun isConsumed(scope: String = "") = scope in consumedScopes
@MainThread
fun consume(scope: String = ""): T? {
return content.takeIf { !isConsumed(scope) }?.also { consumedScopes.add(scope) }
}
fun peek(): T = content
}
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
I modified the EventObserver class as well as the Event class. In the EventObserver class I added a secondary constructor that takes a string value to be used by the event class to check whether the event has been consumed by this specific observer (represented by that string value). This string value is passed on to the Event class that uses its presence (or absence) to determine whether to operate as a SingleObservationEvent or as a MultipleObservationEvent.
And to use it
Vanilla (Default SingleFire Observable)
MultipleObserver
or the lambda version