From be99dd457b8fbb0a123687e7a27d9fb51ab1fc79 Mon Sep 17 00:00:00 2001 From: andreykovalev Date: Mon, 20 Mar 2023 12:27:27 +0000 Subject: [PATCH] Make children immutable to avoid concurrent modification exception and check if Node has been destroyed before calling lifecycle events --- .../src/main/java/com/badoo/ribs/core/Node.kt | 64 +++++++++++++++++-- 1 file changed, 59 insertions(+), 5 deletions(-) diff --git a/libraries/rib-base/src/main/java/com/badoo/ribs/core/Node.kt b/libraries/rib-base/src/main/java/com/badoo/ribs/core/Node.kt index c4c7b1609..936bd7201 100644 --- a/libraries/rib-base/src/main/java/com/badoo/ribs/core/Node.kt +++ b/libraries/rib-base/src/main/java/com/badoo/ribs/core/Node.kt @@ -99,6 +99,8 @@ open class Node @VisibleForTesting internal constructor( is AncestryInfo.Child -> ancestryInfo.anchor } + private var isDestroyed: Boolean = false + val plugins: List = buildContext.defaultPlugins(this) + plugins + if (this is Plugin) listOf(this) else emptyList() @@ -107,9 +109,8 @@ open class Node @VisibleForTesting internal constructor( internal val externalLifecycleRegistry = LifecycleRegistry(this) - @VisibleForTesting - internal val _children: MutableList> = mutableListOf() - val children: List> get() = _children + var children: List> = listOf() + internal set internal open val lifecycleManager = LifecycleManager(this) @@ -145,6 +146,14 @@ open class Node @VisibleForTesting internal constructor( @CallSuper open fun onCreate() { + if (isDestroyed) { + RIBs.errorHandler.handleNonFatalError( + "Calling onCreate when Node has been destroyed. $this", + RuntimeException("Calling onCreate when Node has been destroyed. $this") + ) + return + } + plugins .filterIsInstance() .forEach { it.onCreate(lifecycleManager.ribLifecycle.lifecycle) } @@ -209,6 +218,16 @@ open class Node @VisibleForTesting internal constructor( } open fun onDestroy(isRecreating: Boolean) { + if (isDestroyed) { + RIBs.errorHandler.handleNonFatalError( + "Calling onDestroy when Node has been destroyed. $this", + RuntimeException("Calling onDestroy when Node has been destroyed. $this") + ) + return + } + + isDestroyed = true + if (view != null) { RIBs.errorHandler.handleNonFatalError( "View was not detached before node detach!", @@ -237,7 +256,8 @@ open class Node @VisibleForTesting internal constructor( @MainThread fun attachChildNode(child: Node<*>) { verifyNotRoot(child) - _children.add(child) + val newChildren = children.toMutableList().apply { add(child) } + children = newChildren lifecycleManager.onAttachChild(child) child.onCreate() onAttachChildNode(child) @@ -309,7 +329,9 @@ open class Node @VisibleForTesting internal constructor( @MainThread fun detachChildNode(child: Node<*>, isRecreating: Boolean) { plugins.filterIsInstance().forEach { it.onChildDetached(child) } - _children.remove(child) + + val newChildren = children.toMutableList().apply { remove(child) } + children = newChildren child.onDestroy(isRecreating) } @@ -325,6 +347,14 @@ open class Node @VisibleForTesting internal constructor( * To be called from the hosting environment (Activity, Fragment, etc.) */ fun onStart() { + if (isDestroyed) { + RIBs.errorHandler.handleNonFatalError( + "Calling onStart when Node has been destroyed. $this", + RuntimeException("Calling onStart when Node has been destroyed. $this") + ) + return + } + lifecycleManager.onStartExternal() plugins.filterIsInstance().forEach { it.onStart() } } @@ -333,6 +363,14 @@ open class Node @VisibleForTesting internal constructor( * To be called from the hosting environment (Activity, Fragment, etc.) */ fun onStop() { + if (isDestroyed) { + RIBs.errorHandler.handleNonFatalError( + "Calling onStop when Node has been destroyed. $this", + RuntimeException("Calling onStop when Node has been destroyed. $this") + ) + return + } + lifecycleManager.onStopExternal() plugins.filterIsInstance().forEach { it.onStop() } } @@ -341,6 +379,14 @@ open class Node @VisibleForTesting internal constructor( * To be called from the hosting environment (Activity, Fragment, etc.) */ fun onResume() { + if (isDestroyed) { + RIBs.errorHandler.handleNonFatalError( + "Calling onResume when Node has been destroyed. $this", + RuntimeException("Calling onResume when Node has been destroyed. $this") + ) + return + } + lifecycleManager.onResumeExternal() plugins.filterIsInstance().forEach { it.onResume() } } @@ -349,6 +395,14 @@ open class Node @VisibleForTesting internal constructor( * To be called from the hosting environment (Activity, Fragment, etc.) */ fun onPause() { + if (isDestroyed) { + RIBs.errorHandler.handleNonFatalError( + "Calling onPause when Node has been destroyed. $this", + RuntimeException("Calling onPause when Node has been destroyed. $this") + ) + return + } + lifecycleManager.onPauseExternal() plugins.filterIsInstance().forEach { it.onPause() } }