From 89ea739c65fda32cda1caaf159cad022469e2663 Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Sat, 4 Dec 2021 20:55:12 +0100 Subject: [PATCH] Improved observable API and slightly simplified implementation of some observables. --- .../observable/CallbackChangeObserver.kt | 33 ++++ .../phantasmal/observable/CallbackObserver.kt | 34 ++++- .../{Observer.kt => ChangeObserver.kt} | 4 +- .../world/phantasmal/observable/Dependency.kt | 2 +- .../world/phantasmal/observable/Observable.kt | 5 +- .../observable/ObservableCreation.kt | 3 - .../phantasmal/observable/ObservableUtils.kt | 8 + .../phantasmal/observable/SimpleEmitter.kt | 4 +- .../observable/cell/AbstractCell.kt | 18 +-- .../world/phantasmal/observable/cell/Cell.kt | 9 -- .../cell/{CellCreation.kt => CellUtils.kt} | 59 ++++++++ .../observable/cell/ImmutableCell.kt | 15 +- .../cell/list/AbstractDependentListCell.kt | 29 +--- .../observable/cell/list/AbstractListCell.kt | 31 +--- .../observable/cell/list/ImmutableListCell.kt | 29 +--- .../observable/cell/list/ListCell.kt | 6 +- .../{ListCellCreation.kt => ListCellUtils.kt} | 0 ...{ListObserver.kt => ListChangeObserver.kt} | 2 +- .../phantasmal/observable/ObservableTests.kt | 4 +- .../phantasmal/observable/cell/CellTests.kt | 142 +++++++++--------- .../cell/CellWithDependenciesTests.kt | 4 +- .../phantasmal/observable/cell/ChangeTests.kt | 2 +- .../observable/cell/ImmutableCellTests.kt | 13 +- .../observable/cell/MutableCellTests.kt | 2 +- .../cell/list/FilteredListCellTests.kt | 8 +- .../cell/list/ImmutableListCellTests.kt | 26 +--- .../observable/cell/list/ListCellTests.kt | 38 +++-- .../cell/list/MutableListCellTests.kt | 2 +- .../cell/list/SimpleListCellTests.kt | 2 +- .../web/application/widgets/PwToolButton.kt | 6 +- .../PathAwareTabContainerController.kt | 2 +- .../phantasmal/web/core/stores/UiStore.kt | 6 +- .../phantasmal/web/core/widgets/DockWidget.kt | 2 +- .../web/core/widgets/RendererWidget.kt | 4 +- .../huntOptimizer/stores/HuntMethodStore.kt | 2 +- .../stores/HuntOptimizerStore.kt | 8 +- .../rendering/DestinationInstance.kt | 4 +- .../questEditor/rendering/EntityInstance.kt | 6 +- .../rendering/EntityMeshManager.kt | 17 ++- .../rendering/QuestEditorMeshManager.kt | 8 +- .../questEditor/rendering/QuestMeshManager.kt | 10 +- .../questEditor/rendering/QuestRenderer.kt | 4 +- .../rendering/input/QuestInputManager.kt | 4 +- .../web/questEditor/stores/AsmStore.kt | 6 +- .../questEditor/stores/QuestEditorStore.kt | 6 +- .../web/questEditor/undo/TextModelUndo.kt | 2 +- .../questEditor/widgets/AsmEditorWidget.kt | 10 +- .../questEditor/widgets/EntityInfoWidget.kt | 2 +- .../web/questEditor/widgets/EventWidget.kt | 2 +- .../web/viewer/rendering/MeshRenderer.kt | 16 +- .../web/viewer/rendering/TextureRenderer.kt | 2 +- .../controllers/EventsControllerTests.kt | 5 +- .../phantasmal/webui/DisposableContainer.kt | 109 +++++--------- .../kotlin/world/phantasmal/webui/dom/Dom.kt | 31 +++- .../world/phantasmal/webui/widgets/Button.kt | 2 +- .../phantasmal/webui/widgets/Checkbox.kt | 2 +- .../phantasmal/webui/widgets/ComboBox.kt | 4 +- .../world/phantasmal/webui/widgets/Dialog.kt | 2 +- .../world/phantasmal/webui/widgets/Input.kt | 4 +- .../phantasmal/webui/widgets/LazyLoader.kt | 2 +- .../world/phantasmal/webui/widgets/Menu.kt | 6 +- .../world/phantasmal/webui/widgets/Select.kt | 2 +- .../phantasmal/webui/widgets/TabContainer.kt | 4 +- .../world/phantasmal/webui/widgets/Table.kt | 7 +- .../phantasmal/webui/widgets/TextArea.kt | 4 +- .../world/phantasmal/webui/widgets/Widget.kt | 19 ++- 66 files changed, 453 insertions(+), 413 deletions(-) create mode 100644 observable/src/commonMain/kotlin/world/phantasmal/observable/CallbackChangeObserver.kt rename observable/src/commonMain/kotlin/world/phantasmal/observable/{Observer.kt => ChangeObserver.kt} (76%) delete mode 100644 observable/src/commonMain/kotlin/world/phantasmal/observable/ObservableCreation.kt create mode 100644 observable/src/commonMain/kotlin/world/phantasmal/observable/ObservableUtils.kt rename observable/src/commonMain/kotlin/world/phantasmal/observable/cell/{CellCreation.kt => CellUtils.kt} (79%) rename observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/{ListCellCreation.kt => ListCellUtils.kt} (100%) rename observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/{ListObserver.kt => ListChangeObserver.kt} (95%) diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/CallbackChangeObserver.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/CallbackChangeObserver.kt new file mode 100644 index 00000000..2de704dd --- /dev/null +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/CallbackChangeObserver.kt @@ -0,0 +1,33 @@ +package world.phantasmal.observable + +import world.phantasmal.core.disposable.TrackedDisposable +import world.phantasmal.core.unsafe.unsafeCast + +/** + * Calls [callback] when [dependency] changes. + */ +class CallbackChangeObserver>( + private val dependency: Dependency, + // We don't use Observer because of type system limitations. It would break e.g. + // AbstractListCell.observeListChange. + private val callback: (E) -> Unit, +) : TrackedDisposable(), Dependent { + init { + dependency.addDependent(this) + } + + override fun dispose() { + dependency.removeDependent(this) + super.dispose() + } + + override fun dependencyMightChange() { + // Do nothing. + } + + override fun dependencyChanged(dependency: Dependency, event: ChangeEvent<*>?) { + if (event != null) { + callback(unsafeCast(event)) + } + } +} diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/CallbackObserver.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/CallbackObserver.kt index 63f20f27..c1708e57 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/CallbackObserver.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/CallbackObserver.kt @@ -1,28 +1,46 @@ package world.phantasmal.observable import world.phantasmal.core.disposable.TrackedDisposable -import world.phantasmal.core.unsafe.unsafeCast -class CallbackObserver>( - private val dependency: Dependency, - private val callback: (E) -> Unit, +/** + * Calls [callback] when one or more dependency in [dependencies] changes. + */ +class CallbackObserver( + private vararg val dependencies: Dependency, + private val callback: () -> Unit, ) : TrackedDisposable(), Dependent { + private var changingDependencies = 0 + private var dependenciesActuallyChanged = false + init { - dependency.addDependent(this) + for (dependency in dependencies) { + dependency.addDependent(this) + } } override fun dispose() { - dependency.removeDependent(this) + for (dependency in dependencies) { + dependency.removeDependent(this) + } + super.dispose() } override fun dependencyMightChange() { - // Do nothing. + changingDependencies++ } override fun dependencyChanged(dependency: Dependency, event: ChangeEvent<*>?) { if (event != null) { - callback(unsafeCast(event)) + dependenciesActuallyChanged = true + } + + changingDependencies-- + + if (changingDependencies == 0 && dependenciesActuallyChanged) { + dependenciesActuallyChanged = false + + callback() } } } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/Observer.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/ChangeObserver.kt similarity index 76% rename from observable/src/commonMain/kotlin/world/phantasmal/observable/Observer.kt rename to observable/src/commonMain/kotlin/world/phantasmal/observable/ChangeObserver.kt index 22cf76a4..58c06706 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/Observer.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/ChangeObserver.kt @@ -1,5 +1,7 @@ package world.phantasmal.observable +typealias ChangeObserver = (ChangeEvent) -> Unit + open class ChangeEvent( /** * The observable's new value. @@ -8,5 +10,3 @@ open class ChangeEvent( ) { operator fun component1() = value } - -typealias Observer = (ChangeEvent) -> Unit diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/Dependency.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/Dependency.kt index 3c6e74df..33d5334e 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/Dependency.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/Dependency.kt @@ -3,7 +3,7 @@ package world.phantasmal.observable interface Dependency { /** * This method is not meant to be called from typical application code. Usually you'll want to - * use [Observable.observe]. + * use [Observable.observeChange]. */ fun addDependent(dependent: Dependent) diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/Observable.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/Observable.kt index 3568aee5..ec52ce27 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/Observable.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/Observable.kt @@ -3,5 +3,8 @@ package world.phantasmal.observable import world.phantasmal.core.disposable.Disposable interface Observable : Dependency { - fun observe(observer: Observer): Disposable + /** + * [observer] will be called whenever this observable changes. + */ + fun observeChange(observer: ChangeObserver): Disposable } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/ObservableCreation.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/ObservableCreation.kt deleted file mode 100644 index c9519a2a..00000000 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/ObservableCreation.kt +++ /dev/null @@ -1,3 +0,0 @@ -package world.phantasmal.observable - -fun emitter(): Emitter = SimpleEmitter() diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/ObservableUtils.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/ObservableUtils.kt new file mode 100644 index 00000000..46258bac --- /dev/null +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/ObservableUtils.kt @@ -0,0 +1,8 @@ +package world.phantasmal.observable + +import world.phantasmal.core.disposable.Disposable + +fun emitter(): Emitter = SimpleEmitter() + +fun Observable.observe(observer: (T) -> Unit): Disposable = + observeChange { observer(it.value) } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/SimpleEmitter.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/SimpleEmitter.kt index e9a9bf32..866e49bb 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/SimpleEmitter.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/SimpleEmitter.kt @@ -15,8 +15,8 @@ class SimpleEmitter : AbstractDependency(), Emitter { ChangeManager.changed(this) } - override fun observe(observer: Observer): Disposable = - CallbackObserver(this, observer) + override fun observeChange(observer: ChangeObserver): Disposable = + CallbackChangeObserver(this, observer) override fun emitDependencyChanged() { if (event != null) { diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/AbstractCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/AbstractCell.kt index 37b7564b..132aeff4 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/AbstractCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/AbstractCell.kt @@ -2,25 +2,15 @@ package world.phantasmal.observable.cell import world.phantasmal.core.disposable.Disposable import world.phantasmal.observable.AbstractDependency -import world.phantasmal.observable.CallbackObserver +import world.phantasmal.observable.CallbackChangeObserver import world.phantasmal.observable.ChangeEvent -import world.phantasmal.observable.Observer +import world.phantasmal.observable.ChangeObserver abstract class AbstractCell : AbstractDependency(), Cell { private var mightChangeEmitted = false - final override fun observe(observer: Observer): Disposable = - observe(callNow = false, observer) - - override fun observe(callNow: Boolean, observer: Observer): Disposable { - val observingCell = CallbackObserver(this, observer) - - if (callNow) { - observer(ChangeEvent(value)) - } - - return observingCell - } + override fun observeChange(observer: ChangeObserver): Disposable = + CallbackChangeObserver(this, observer) protected fun emitMightChange() { if (!mightChangeEmitted) { diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/Cell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/Cell.kt index d36e5dd3..77aad022 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/Cell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/Cell.kt @@ -1,10 +1,6 @@ package world.phantasmal.observable.cell -import world.phantasmal.core.disposable.Disposable import world.phantasmal.observable.Observable -import world.phantasmal.observable.Observer -import world.phantasmal.observable.cell.list.DependentListCell -import world.phantasmal.observable.cell.list.ListCell import kotlin.reflect.KProperty /** @@ -14,9 +10,4 @@ interface Cell : Observable { val value: T operator fun getValue(thisRef: Any?, property: KProperty<*>): T = value - - /** - * @param callNow Call [observer] immediately with the current [value]. - */ - fun observe(callNow: Boolean = false, observer: Observer): Disposable } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/CellCreation.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/CellUtils.kt similarity index 79% rename from observable/src/commonMain/kotlin/world/phantasmal/observable/cell/CellCreation.kt rename to observable/src/commonMain/kotlin/world/phantasmal/observable/cell/CellUtils.kt index ff036e1d..2668fc33 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/CellCreation.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/CellUtils.kt @@ -1,5 +1,8 @@ package world.phantasmal.observable.cell +import world.phantasmal.core.disposable.Disposable +import world.phantasmal.observable.CallbackObserver + private val TRUE_CELL: Cell = ImmutableCell(true) private val FALSE_CELL: Cell = ImmutableCell(false) private val NULL_CELL: Cell = ImmutableCell(null) @@ -36,6 +39,62 @@ fun mutableCell(value: T): MutableCell = SimpleCell(value) fun mutableCell(getter: () -> T, setter: (T) -> Unit): MutableCell = DelegatingCell(getter, setter) +fun Cell.observeNow( + observer: (T) -> Unit, +): Disposable { + observer(value) + + return observeChange { observer(it.value) } +} + +fun observeNow( + c1: Cell, + c2: Cell, + observer: (T1, T2) -> Unit, +): Disposable { + observer(c1.value, c2.value) + + return CallbackObserver(c1, c2) { observer(c1.value, c2.value) } +} + +fun observeNow( + c1: Cell, + c2: Cell, + c3: Cell, + observer: (T1, T2, T3) -> Unit, +): Disposable { + observer(c1.value, c2.value, c3.value) + + return CallbackObserver(c1, c2, c3) { observer(c1.value, c2.value, c3.value) } +} + +fun observeNow( + c1: Cell, + c2: Cell, + c3: Cell, + c4: Cell, + observer: (T1, T2, T3, T4) -> Unit, +): Disposable { + observer(c1.value, c2.value, c3.value, c4.value) + + return CallbackObserver(c1, c2, c3, c4) { observer(c1.value, c2.value, c3.value, c4.value) } +} + +fun observeNow( + c1: Cell, + c2: Cell, + c3: Cell, + c4: Cell, + c5: Cell, + observer: (T1, T2, T3, T4, T5) -> Unit, +): Disposable { + observer(c1.value, c2.value, c3.value, c4.value, c5.value) + + return CallbackObserver(c1, c2, c3, c4, c5) { + observer(c1.value, c2.value, c3.value, c4.value, c5.value) + } +} + /** * Map a transformation function over this cell. * diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/ImmutableCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/ImmutableCell.kt index 8a0c27ca..5045621c 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/ImmutableCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/ImmutableCell.kt @@ -3,21 +3,12 @@ package world.phantasmal.observable.cell import world.phantasmal.core.disposable.Disposable import world.phantasmal.core.disposable.nopDisposable import world.phantasmal.observable.AbstractDependency -import world.phantasmal.observable.ChangeEvent -import world.phantasmal.observable.Observer +import world.phantasmal.observable.ChangeObserver class ImmutableCell(override val value: T) : AbstractDependency(), Cell { - override fun observe(callNow: Boolean, observer: Observer): Disposable { - if (callNow) { - observer(ChangeEvent(value)) - } - - return nopDisposable() - } - - override fun observe(observer: Observer): Disposable = nopDisposable() + override fun observeChange(observer: ChangeObserver): Disposable = nopDisposable() override fun emitDependencyChanged() { - error("StaticCell can't change.") + error("ImmutableCell can't change.") } } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/AbstractDependentListCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/AbstractDependentListCell.kt index f2483204..00fcc0dd 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/AbstractDependentListCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/AbstractDependentListCell.kt @@ -2,9 +2,9 @@ package world.phantasmal.observable.cell.list import world.phantasmal.core.disposable.Disposable import world.phantasmal.core.unsafe.unsafeAssertNotNull -import world.phantasmal.observable.CallbackObserver +import world.phantasmal.observable.CallbackChangeObserver +import world.phantasmal.observable.ChangeObserver import world.phantasmal.observable.Dependent -import world.phantasmal.observable.Observer import world.phantasmal.observable.cell.AbstractDependentCell import world.phantasmal.observable.cell.Cell import world.phantasmal.observable.cell.DependentCell @@ -55,28 +55,11 @@ abstract class AbstractDependentListCell : return unsafeAssertNotNull(_notEmpty) } - final override fun observe(callNow: Boolean, observer: Observer>): Disposable = - observeList(callNow, observer as ListObserver) + final override fun observeChange(observer: ChangeObserver>): Disposable = + observeListChange(observer) - override fun observeList(callNow: Boolean, observer: ListObserver): Disposable { - val observingCell = CallbackObserver(this, observer) - - if (callNow) { - observer( - ListChangeEvent( - value, - listOf(ListChange.Structural( - index = 0, - prevSize = 0, - removed = emptyList(), - inserted = value, - )), - ) - ) - } - - return observingCell - } + override fun observeListChange(observer: ListChangeObserver): Disposable = + CallbackChangeObserver(this, observer) final override fun dependenciesChanged() { val oldElements = value diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/AbstractListCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/AbstractListCell.kt index dcf5b337..0be00770 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/AbstractListCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/AbstractListCell.kt @@ -2,8 +2,8 @@ package world.phantasmal.observable.cell.list import world.phantasmal.core.disposable.Disposable import world.phantasmal.core.unsafe.unsafeAssertNotNull -import world.phantasmal.observable.CallbackObserver -import world.phantasmal.observable.Observer +import world.phantasmal.observable.CallbackChangeObserver +import world.phantasmal.observable.ChangeObserver import world.phantasmal.observable.cell.AbstractCell import world.phantasmal.observable.cell.Cell import world.phantasmal.observable.cell.DependentCell @@ -19,7 +19,7 @@ abstract class AbstractListCell : AbstractCell>(), ListCell { * its old value whenever a change event was emitted. */ // TODO: Optimize this by using a weak reference to avoid copying when nothing references the - // wrapper. + // wrapper. private var _elementsWrapper: DelegatingList? = null protected val elementsWrapper: DelegatingList get() { @@ -39,28 +39,11 @@ abstract class AbstractListCell : AbstractCell>(), ListCell { final override val notEmpty: Cell = !empty - final override fun observe(callNow: Boolean, observer: Observer>): Disposable = - observeList(callNow, observer as ListObserver) + final override fun observeChange(observer: ChangeObserver>): Disposable = + observeListChange(observer) - override fun observeList(callNow: Boolean, observer: ListObserver): Disposable { - val observingCell = CallbackObserver(this, observer) - - if (callNow) { - observer( - ListChangeEvent( - value, - listOf(ListChange.Structural( - index = 0, - prevSize = 0, - removed = emptyList(), - inserted = value - )), - ) - ) - } - - return observingCell - } + override fun observeListChange(observer: ListChangeObserver): Disposable = + CallbackChangeObserver(this, observer) protected fun copyAndResetWrapper() { _elementsWrapper?.backingList = elements.toList() diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ImmutableListCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ImmutableListCell.kt index 14c9ff2a..b26400d2 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ImmutableListCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ImmutableListCell.kt @@ -3,8 +3,7 @@ package world.phantasmal.observable.cell.list import world.phantasmal.core.disposable.Disposable import world.phantasmal.core.disposable.nopDisposable import world.phantasmal.observable.AbstractDependency -import world.phantasmal.observable.ChangeEvent -import world.phantasmal.observable.Observer +import world.phantasmal.observable.ChangeObserver import world.phantasmal.observable.cell.Cell import world.phantasmal.observable.cell.cell import world.phantasmal.observable.cell.falseCell @@ -20,31 +19,9 @@ class ImmutableListCell(private val elements: List) : AbstractDependency() override fun get(index: Int): E = elements[index] - override fun observe(callNow: Boolean, observer: Observer>): Disposable { - if (callNow) { - observer(ChangeEvent(value)) - } + override fun observeChange(observer: ChangeObserver>): Disposable = nopDisposable() - return nopDisposable() - } - - override fun observe(observer: Observer>): Disposable = nopDisposable() - - override fun observeList(callNow: Boolean, observer: ListObserver): Disposable { - if (callNow) { - observer(ListChangeEvent( - value, - listOf(ListChange.Structural( - index = 0, - prevSize = 0, - removed = emptyList(), - inserted = value, - )), - )) - } - - return nopDisposable() - } + override fun observeListChange(observer: ListChangeObserver): Disposable = nopDisposable() override fun emitDependencyChanged() { error("ImmutableListCell can't change.") diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListCell.kt index fb90f73f..3ff6ffae 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListCell.kt @@ -14,7 +14,11 @@ interface ListCell : Cell> { operator fun get(index: Int): E = value[index] - fun observeList(callNow: Boolean = false, observer: ListObserver): Disposable + /** + * List variant of [Cell.observeChange]. + */ + // Exists solely because function parameters are invariant. + fun observeListChange(observer: ListChangeObserver): Disposable operator fun contains(element: @UnsafeVariance E): Boolean = element in value } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListCellCreation.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListCellUtils.kt similarity index 100% rename from observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListCellCreation.kt rename to observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListCellUtils.kt diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListObserver.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListChangeObserver.kt similarity index 95% rename from observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListObserver.kt rename to observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListChangeObserver.kt index 20a06b3f..83985b2b 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListObserver.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListChangeObserver.kt @@ -42,4 +42,4 @@ sealed class ListChange { ) : ListChange() } -typealias ListObserver = (ListChangeEvent) -> Unit +typealias ListChangeObserver = (ListChangeEvent) -> Unit diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/ObservableTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/ObservableTests.kt index 0a4f2fd3..0d58391d 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/ObservableTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/ObservableTests.kt @@ -16,7 +16,7 @@ interface ObservableTests : DependencyTests { var changes = 0 disposer.add( - p.observable.observe { + p.observable.observeChange { changes++ } ) @@ -37,7 +37,7 @@ interface ObservableTests : DependencyTests { val p = createProvider() var changes = 0 - val observer = p.observable.observe { + val observer = p.observable.observeChange { changes++ } diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/CellTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/CellTests.kt index 6de5c9d5..3d8dc3b8 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/CellTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/CellTests.kt @@ -1,6 +1,7 @@ package world.phantasmal.observable.cell import world.phantasmal.core.disposable.use +import world.phantasmal.observable.ChangeEvent import world.phantasmal.observable.ObservableTests import kotlin.test.* @@ -15,6 +16,7 @@ interface CellTests : ObservableTests { fun value_is_accessible_without_observers() = test { val p = createProvider() + // We literally just test that accessing the value property doesn't throw or return null. assertNotNull(p.observable.value) } @@ -22,106 +24,55 @@ interface CellTests : ObservableTests { fun value_is_accessible_with_observers() = test { val p = createProvider() - var observedValue: Any? = null + disposer.add(p.observable.observeChange {}) - disposer.add(p.observable.observe(callNow = true) { - observedValue = it.value - }) - - assertNotNull(observedValue) + // We literally just test that accessing the value property doesn't throw or return null. assertNotNull(p.observable.value) } @Test - fun propagates_changes_to_mapped_cell() = test { + fun emits_no_change_event_until_changed() = test { val p = createProvider() - val mapped = p.observable.map { it.hashCode() } - val initialValue = mapped.value - var observedValue: Int? = null + var observedEvent: ChangeEvent? = null - disposer.add(mapped.observe { - assertNull(observedValue) - observedValue = it.value + disposer.add(p.observable.observeChange { changeEvent -> + observedEvent = changeEvent }) + assertNull(observedEvent) + p.emit() - assertNotEquals(initialValue, mapped.value) - assertEquals(mapped.value, observedValue) - } - - @Test - fun propagates_changes_to_flat_mapped_cell() = test { - val p = createProvider() - - val mapped = p.observable.flatMap { ImmutableCell(it.hashCode()) } - val initialValue = mapped.value - - var observedValue: Int? = null - - disposer.add(mapped.observe { - assertNull(observedValue) - observedValue = it.value - }) - - p.emit() - - assertNotEquals(initialValue, mapped.value) - assertEquals(mapped.value, observedValue) + assertNotNull(observedEvent) } @Test fun emits_correct_value_in_change_events() = test { val p = createProvider() + var prevValue: Any? var observedValue: Any? = null - disposer.add(p.observable.observe { + disposer.add(p.observable.observeChange { changeEvent -> assertNull(observedValue) - observedValue = it.value + observedValue = changeEvent.value }) repeat(3) { + prevValue = observedValue observedValue = null p.emit() + // We should have observed a value, it should be different from the previous value, and + // it should be equal to the cell's current value. assertNotNull(observedValue) + assertNotEquals(prevValue, observedValue) assertEquals(p.observable.value, observedValue) } } - /** - * When [Cell.observe] is called with callNow = true, it should call the observer immediately. - * Otherwise it should only call the observer when it changes. - */ - @Test - fun respects_call_now_argument() = test { - val p = createProvider() - var changes = 0 - - // Test callNow = false - p.observable.observe(callNow = false) { - changes++ - }.use { - p.emit() - - assertEquals(1, changes) - } - - // Test callNow = true - changes = 0 - - p.observable.observe(callNow = true) { - changes++ - }.use { - p.emit() - - assertEquals(2, changes) - } - } - /** * [Cell.value] should correctly reflect changes even when the [Cell] has no observers. * Typically this means that the cell's value is not updated in real time, only when it is @@ -148,6 +99,63 @@ interface CellTests : ObservableTests { } } + // + // CellUtils Tests + // + + @Test + fun propagates_changes_to_observeNow_observers() = test { + val p = createProvider() + var changes = 0 + + p.observable.observeNow { + changes++ + }.use { + p.emit() + + assertEquals(2, changes) + } + } + + @Test + fun propagates_changes_to_mapped_cell() = test { + val p = createProvider() + val mapped = p.observable.map { it.hashCode() } + val initialValue = mapped.value + + var observedValue: Int? = null + + disposer.add(mapped.observeChange { changeEvent -> + assertNull(observedValue) + observedValue = changeEvent.value + }) + + p.emit() + + assertNotEquals(initialValue, mapped.value) + assertEquals(mapped.value, observedValue) + } + + @Test + fun propagates_changes_to_flat_mapped_cell() = test { + val p = createProvider() + + val mapped = p.observable.flatMap { ImmutableCell(it.hashCode()) } + val initialValue = mapped.value + + var observedValue: Int? = null + + disposer.add(mapped.observeChange { + assertNull(observedValue) + observedValue = it.value + }) + + p.emit() + + assertNotEquals(initialValue, mapped.value) + assertEquals(mapped.value, observedValue) + } + interface Provider : ObservableTests.Provider { override val observable: Cell } diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/CellWithDependenciesTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/CellWithDependenciesTests.kt index 17fecd95..8efa6df4 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/CellWithDependenciesTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/CellWithDependenciesTests.kt @@ -19,7 +19,7 @@ interface CellWithDependenciesTests : CellTests { var observedChanges = 0 - disposer.add(leaf.observe { observedChanges++ }) + disposer.add(leaf.observeChange { observedChanges++ }) // Change root, which results in both branches changing and thus two dependencies of leaf // changing. @@ -47,7 +47,7 @@ interface CellWithDependenciesTests : CellTests { assertTrue(dependency.publicDependents.isEmpty()) - disposer.add(cell.observe { }) + disposer.add(cell.observeChange { }) assertEquals(1, dependency.publicDependents.size) } diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/ChangeTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/ChangeTests.kt index d3a18b02..1f6cb8e4 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/ChangeTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/ChangeTests.kt @@ -13,7 +13,7 @@ class ChangeTests : ObservableTestSuite { val dependent = dependency.map { 2 * it } var dependentObservedValue: Int? = null - disposer.add(dependent.observe { dependentObservedValue = it.value }) + disposer.add(dependent.observeChange { dependentObservedValue = it.value }) assertFails { change { diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/ImmutableCellTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/ImmutableCellTests.kt index 9c409846..55264188 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/ImmutableCellTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/ImmutableCellTests.kt @@ -7,24 +7,21 @@ import kotlin.test.assertEquals class ImmutableCellTests : ObservableTestSuite { @Test - fun observing_it_should_never_create_leaks() = test { + fun observing_it_never_creates_leaks() = test { val cell = ImmutableCell("test value") TrackedDisposable.checkNoLeaks { - // We never call dispose on the returned disposables. - cell.observe {} - cell.observe(callNow = false) {} - cell.observe(callNow = true) {} + // We never call dispose on the returned disposable. + cell.observeChange {} } } @Test - fun observe_respects_callNow() = test { + fun observeNow_calls_the_observer_once() = test { val cell = ImmutableCell("test value") var calls = 0 - cell.observe(callNow = false) { calls++ } - cell.observe(callNow = true) { calls++ } + cell.observeNow { calls++ } assertEquals(1, calls) } diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/MutableCellTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/MutableCellTests.kt index 6ac72ee7..dd1ac316 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/MutableCellTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/MutableCellTests.kt @@ -14,7 +14,7 @@ interface MutableCellTests : CellTests { var observedValue: Any? = null - disposer.add(p.observable.observe { + disposer.add(p.observable.observeChange { assertNull(observedValue) observedValue = it.value }) diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/FilteredListCellTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/FilteredListCellTests.kt index c1b2db68..6d2405e6 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/FilteredListCellTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/FilteredListCellTests.kt @@ -46,10 +46,10 @@ class FilteredListCellTests : ListCellTests { var changes = 0 var listChanges = 0 - disposer.add(list.observe { + disposer.add(list.observeChange { changes++ }) - disposer.add(list.observeList { + disposer.add(list.observeListChange { listChanges++ }) @@ -74,7 +74,7 @@ class FilteredListCellTests : ListCellTests { val list = FilteredListCell(dep, predicate = { it % 2 == 0 }) var event: ListChangeEvent? = null - disposer.add(list.observeList { + disposer.add(list.observeListChange { assertNull(event) event = it }) @@ -128,7 +128,7 @@ class FilteredListCellTests : ListCellTests { val list = FilteredListCell(dep, predicate = { it.value % 2 == 0 }) var event: ListChangeEvent>? = null - disposer.add(list.observeList { + disposer.add(list.observeListChange { assertNull(event) event = it }) diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/ImmutableListCellTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/ImmutableListCellTests.kt index 4008fa58..3640f0a8 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/ImmutableListCellTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/ImmutableListCellTests.kt @@ -1,43 +1,29 @@ package world.phantasmal.observable.cell.list import world.phantasmal.core.disposable.TrackedDisposable +import world.phantasmal.observable.cell.observeNow import world.phantasmal.observable.test.ObservableTestSuite import kotlin.test.Test import kotlin.test.assertEquals class ImmutableListCellTests : ObservableTestSuite { @Test - fun observing_it_should_never_create_leaks() = test { + fun observing_it_never_creates_leaks() = test { val listCell = ImmutableListCell(listOf(1, 2, 3)) TrackedDisposable.checkNoLeaks { // We never call dispose on the returned disposables. - listCell.observe {} - listCell.observe(callNow = false) {} - listCell.observe(callNow = true) {} - listCell.observeList(callNow = false) {} - listCell.observeList(callNow = true) {} + listCell.observeChange {} + listCell.observeListChange {} } } @Test - fun observe_respects_callNow() = test { + fun observeNow_calls_the_observer_once() = test { val listCell = ImmutableListCell(listOf(1, 2, 3)) var calls = 0 - listCell.observe(callNow = false) { calls++ } - listCell.observe(callNow = true) { calls++ } - - assertEquals(1, calls) - } - - @Test - fun observeList_respects_callNow() = test { - val listCell = ImmutableListCell(listOf(1, 2, 3)) - var calls = 0 - - listCell.observeList(callNow = false) { calls++ } - listCell.observeList(callNow = true) { calls++ } + listCell.observeNow { calls++ } assertEquals(1, calls) } diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/ListCellTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/ListCellTests.kt index 03092688..243b1560 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/ListCellTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/ListCellTests.kt @@ -16,6 +16,8 @@ interface ListCellTests : CellTests { fun list_value_is_accessible_without_observers() = test { val p = createListProvider(empty = false) + // We literally just test that accessing the value property doesn't throw or return the + // wrong list. assertTrue(p.observable.value.isNotEmpty()) } @@ -23,14 +25,28 @@ interface ListCellTests : CellTests { fun list_value_is_accessible_with_observers() = test { val p = createListProvider(empty = false) - var observedValue: List<*>? = null + disposer.add(p.observable.observeListChange {}) - disposer.add(p.observable.observe(callNow = true) { - observedValue = it.value + // We literally just test that accessing the value property doesn't throw or return the + // wrong list. + assertTrue(p.observable.value.isNotEmpty()) + } + + @Test + fun emits_no_list_change_event_until_changed() = test { + val p = createListProvider(empty = false) + + var observedEvent: ListChangeEvent? = null + + disposer.add(p.observable.observeListChange { listChangeEvent -> + observedEvent = listChangeEvent }) - assertTrue(observedValue!!.isNotEmpty()) - assertTrue(p.observable.value.isNotEmpty()) + assertNull(observedEvent) + + p.emit() + + assertNotNull(observedEvent) } @Test @@ -40,7 +56,7 @@ interface ListCellTests : CellTests { var event: ListChangeEvent<*>? = null disposer.add( - p.observable.observeList { + p.observable.observeListChange { assertNull(event) event = it } @@ -64,7 +80,7 @@ interface ListCellTests : CellTests { var observedSize: Int? = null disposer.add( - p.observable.size.observe { + p.observable.size.observeChange { assertNull(observedSize) observedSize = it.value } @@ -102,7 +118,7 @@ interface ListCellTests : CellTests { var observedValue: Int? = null - disposer.add(fold.observe { + disposer.add(fold.observeChange { assertNull(observedValue) observedValue = it.value }) @@ -127,7 +143,7 @@ interface ListCellTests : CellTests { var observedValue: Int? = null - disposer.add(sum.observe { + disposer.add(sum.observeChange { assertNull(observedValue) observedValue = it.value }) @@ -152,7 +168,7 @@ interface ListCellTests : CellTests { var event: ListChangeEvent<*>? = null - disposer.add(filtered.observeList { + disposer.add(filtered.observeListChange { assertNull(event) event = it }) @@ -177,7 +193,7 @@ interface ListCellTests : CellTests { var observedValue: Any? = null - disposer.add(firstOrNull.observe { + disposer.add(firstOrNull.observeChange { assertNull(observedValue) observedValue = it.value }) diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/MutableListCellTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/MutableListCellTests.kt index 3e80e739..5a0aab62 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/MutableListCellTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/MutableListCellTests.kt @@ -16,7 +16,7 @@ interface MutableListCellTests : ListCellTests, MutableCellTests? = null - disposer.add(p.observable.observeList { + disposer.add(p.observable.observeListChange { assertNull(changeEvent) changeEvent = it }) diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/SimpleListCellTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/SimpleListCellTests.kt index 94f9d733..e4ddfb87 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/SimpleListCellTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/list/SimpleListCellTests.kt @@ -245,7 +245,7 @@ class SimpleListCellTests : MutableListCellTests { var event: ListChangeEvent>? = null - disposer.add(list.observeList { + disposer.add(list.observeListChange { assertNull(event) event = it }) diff --git a/web/src/main/kotlin/world/phantasmal/web/application/widgets/PwToolButton.kt b/web/src/main/kotlin/world/phantasmal/web/application/widgets/PwToolButton.kt index 9d9b1bc8..9e8a52c1 100644 --- a/web/src/main/kotlin/world/phantasmal/web/application/widgets/PwToolButton.kt +++ b/web/src/main/kotlin/world/phantasmal/web/application/widgets/PwToolButton.kt @@ -1,7 +1,7 @@ package world.phantasmal.web.application.widgets import org.w3c.dom.Node -import world.phantasmal.observable.Observable +import world.phantasmal.observable.cell.Cell import world.phantasmal.observable.cell.nullCell import world.phantasmal.observable.cell.trueCell import world.phantasmal.web.core.PwToolType @@ -12,7 +12,7 @@ import world.phantasmal.webui.widgets.Control class PwToolButton( private val tool: PwToolType, - private val toggled: Observable, + private val toggled: Cell, private val onMouseDown: () -> Unit, ) : Control(visible = trueCell(), enabled = trueCell(), tooltip = nullCell()) { private val inputId = "pw-application-pw-tool-button-${tool.name.lowercase()}" @@ -25,7 +25,7 @@ class PwToolButton( type = "radio" id = inputId name = "pw-application-pw-tool-button" - observe(toggled) { checked = it } + observeNow(toggled) { checked = it } } label { htmlFor = inputId diff --git a/web/src/main/kotlin/world/phantasmal/web/core/controllers/PathAwareTabContainerController.kt b/web/src/main/kotlin/world/phantasmal/web/core/controllers/PathAwareTabContainerController.kt index 080ce606..7b72093d 100644 --- a/web/src/main/kotlin/world/phantasmal/web/core/controllers/PathAwareTabContainerController.kt +++ b/web/src/main/kotlin/world/phantasmal/web/core/controllers/PathAwareTabContainerController.kt @@ -15,7 +15,7 @@ open class PathAwareTabContainerController( tabs: List, ) : TabContainerController(tabs) { init { - observe(uiStore.path) { path -> + observeNow(uiStore.path) { path -> if (uiStore.currentTool.value == tool) { tabs.find { path.startsWith(it.path) }?.let { setActiveTab(it, replaceUrl = true) diff --git a/web/src/main/kotlin/world/phantasmal/web/core/stores/UiStore.kt b/web/src/main/kotlin/world/phantasmal/web/core/stores/UiStore.kt index a72cc3fb..ad8a0e38 100644 --- a/web/src/main/kotlin/world/phantasmal/web/core/stores/UiStore.kt +++ b/web/src/main/kotlin/world/phantasmal/web/core/stores/UiStore.kt @@ -82,7 +82,7 @@ class UiStore(private val applicationUrl: ApplicationUrl) : Store() { window.disposableListener("keydown", ::dispatchGlobalKeyDown), ) - observe(applicationUrl.url) { setDataFromUrl(it) } + observeNow(applicationUrl.url) { setDataFromUrl(it) } } fun setCurrentTool(tool: PwToolType) { @@ -126,12 +126,12 @@ class UiStore(private val applicationUrl: ApplicationUrl) : Store() { } return Disposer( - value.observe { + value.observeChange { if (it.value != param.value) { setParameter(tool, path, param, it.value, replaceUrl = false) } }, - param.observe { onChange(it.value) }, + param.observeChange { onChange(it.value) }, ) } diff --git a/web/src/main/kotlin/world/phantasmal/web/core/widgets/DockWidget.kt b/web/src/main/kotlin/world/phantasmal/web/core/widgets/DockWidget.kt index 17a63a3d..33a60a6d 100644 --- a/web/src/main/kotlin/world/phantasmal/web/core/widgets/DockWidget.kt +++ b/web/src/main/kotlin/world/phantasmal/web/core/widgets/DockWidget.kt @@ -64,7 +64,7 @@ class DockWidget( style.width = "" style.height = "" - addDisposable(size.observe { (size) -> + addDisposable(size.observeChange { (size) -> goldenLayout.updateSize(size.width, size.height) }) } diff --git a/web/src/main/kotlin/world/phantasmal/web/core/widgets/RendererWidget.kt b/web/src/main/kotlin/world/phantasmal/web/core/widgets/RendererWidget.kt index 3bb75265..5e6c00b4 100644 --- a/web/src/main/kotlin/world/phantasmal/web/core/widgets/RendererWidget.kt +++ b/web/src/main/kotlin/world/phantasmal/web/core/widgets/RendererWidget.kt @@ -12,7 +12,7 @@ class RendererWidget( div { className = "pw-core-renderer" - observe(selfOrAncestorVisible) { visible -> + observeNow(selfOrAncestorVisible) { visible -> if (visible) { renderer.startRendering() } else { @@ -20,7 +20,7 @@ class RendererWidget( } } - addDisposable(size.observe { (size) -> + addDisposable(size.observeChange { (size) -> renderer.setSize(size.width.toInt(), size.height.toInt()) }) diff --git a/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/stores/HuntMethodStore.kt b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/stores/HuntMethodStore.kt index ad549917..89401d9a 100644 --- a/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/stores/HuntMethodStore.kt +++ b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/stores/HuntMethodStore.kt @@ -31,7 +31,7 @@ class HuntMethodStore( /** Hunting methods supported by the current server. */ val methods: ListCell by lazy { - observe(uiStore.server) { _methodsStatus.load() } + observeNow(uiStore.server) { _methodsStatus.load() } _methods } diff --git a/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/stores/HuntOptimizerStore.kt b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/stores/HuntOptimizerStore.kt index c05fe1b1..42ab0af6 100644 --- a/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/stores/HuntOptimizerStore.kt +++ b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/stores/HuntOptimizerStore.kt @@ -55,7 +55,7 @@ class HuntOptimizerStore( private var wantedItemsPersistenceObserver: Disposable? = null val huntableItems: ListCell by lazy { - observe(uiStore.server) { server -> + observeNow(uiStore.server) { server -> _huntableItems.clear() scope.launch { @@ -71,12 +71,12 @@ class HuntOptimizerStore( } val wantedItems: ListCell by lazy { - observe(uiStore.server) { loadWantedItems(it) } + observeNow(uiStore.server) { loadWantedItems(it) } _wantedItems } val optimizationResult: Cell by lazy { - observe(wantedItems, huntMethodStore.methods) { wantedItems, huntMethods -> + observeNow(wantedItems, huntMethodStore.methods) { wantedItems, huntMethods -> scope.launch(Dispatchers.Default) { _optimizationResult.value = optimize(wantedItems, huntMethods) } @@ -114,7 +114,7 @@ class HuntOptimizerStore( _wantedItems.replaceAll(wantedItems) // Wanted items are loaded, start observing them and persist whenever they change. - wantedItemsPersistenceObserver = _wantedItems.observe { + wantedItemsPersistenceObserver = _wantedItems.observeChange { val items = it.value scope.launch(Dispatchers.Main) { diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/DestinationInstance.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/DestinationInstance.kt index c80a49aa..dfe3b452 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/DestinationInstance.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/DestinationInstance.kt @@ -11,8 +11,8 @@ class DestinationInstance( ) : Instance(entity, mesh, instanceIndex) { init { addDisposables( - entity.destinationPosition.observe { updateMatrix() }, - entity.destinationRotationY.observe { updateMatrix() }, + entity.destinationPosition.observeChange { updateMatrix() }, + entity.destinationRotationY.observeChange { updateMatrix() }, ) } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/EntityInstance.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/EntityInstance.kt index 72074a69..00cda40f 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/EntityInstance.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/EntityInstance.kt @@ -13,14 +13,14 @@ class EntityInstance( ) : Instance>(entity, mesh, instanceIndex) { init { if (entity is QuestObjectModel) { - addDisposable(entity.model.observe(callNow = false) { + addDisposable(entity.model.observeChange { modelChanged(this.instanceIndex) }) } addDisposables( - entity.worldPosition.observe { updateMatrix() }, - entity.worldRotation.observe { updateMatrix() }, + entity.worldPosition.observeChange { updateMatrix() }, + entity.worldRotation.observeChange { updateMatrix() }, ) } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/EntityMeshManager.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/EntityMeshManager.kt index b9775243..38c6810c 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/EntityMeshManager.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/EntityMeshManager.kt @@ -5,6 +5,7 @@ import mu.KotlinLogging import org.khronos.webgl.Float32Array import world.phantasmal.core.disposable.DisposableSupervisedScope import world.phantasmal.core.disposable.Disposer +import world.phantasmal.observable.cell.observeNow import world.phantasmal.psolib.fileFormats.quest.EntityType import world.phantasmal.web.core.rendering.disposeObject3DResources import world.phantasmal.web.externals.three.* @@ -92,13 +93,13 @@ class EntityMeshManager( } init { - observe(questEditorStore.highlightedEntity) { entity -> + observeNow(questEditorStore.highlightedEntity) { entity -> // getEntityInstance can return null at this point because the entity mesh might not be // loaded yet. markHighlighted(entity?.let(::getEntityInstance)) } - observe(questEditorStore.selectedEntity) { entity -> + observeNow(questEditorStore.selectedEntity) { entity -> // getEntityInstance can return null at this point because the entity mesh might not be // loaded yet. markSelected(entity?.let(::getEntityInstance)) @@ -135,7 +136,9 @@ class EntityMeshManager( destinationInstanceContainer.addInstance(entity) } } catch (e: CancellationException) { - // Do nothing. + logger.trace(e) { + "Mesh loading for entity of type ${entity.type} cancelled." + } } catch (e: Throwable) { logger.error(e) { "Couldn't load mesh for entity of type ${entity.type}." @@ -228,12 +231,12 @@ class EntityMeshManager( newInstance.entity is QuestObjectModel && newInstance.entity.hasDestination ) { - warpLineDisposer.add(newInstance.entity.worldPosition.observe(callNow = true) { - warpLineBufferAttribute.setXYZ(0, it.value.x, it.value.y, it.value.z) + warpLineDisposer.add(newInstance.entity.worldPosition.observeNow { + warpLineBufferAttribute.setXYZ(0, it.x, it.y, it.z) warpLineBufferAttribute.needsUpdate = true }) - warpLineDisposer.add(newInstance.entity.destinationPosition.observe(callNow = true) { - warpLineBufferAttribute.setXYZ(1, it.value.x, it.value.y, it.value.z) + warpLineDisposer.add(newInstance.entity.destinationPosition.observeNow { + warpLineBufferAttribute.setXYZ(1, it.x, it.y, it.z) warpLineBufferAttribute.needsUpdate = true }) warpLines.visible = true diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestEditorMeshManager.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestEditorMeshManager.kt index 6ea360d6..d0cf1d3c 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestEditorMeshManager.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestEditorMeshManager.kt @@ -14,14 +14,14 @@ class QuestEditorMeshManager( renderContext: QuestRenderContext, ) : QuestMeshManager(areaAssetLoader, entityAssetLoader, questEditorStore, renderContext) { init { - observe( + observeNow( questEditorStore.currentQuest, questEditorStore.currentAreaVariant, ) { quest, areaVariant -> loadAreaMeshes(quest?.episode, areaVariant) } - observe( + observeNow( questEditorStore.currentQuest, questEditorStore.currentArea, questEditorStore.selectedEvent.flatMapNull { it?.wave }, @@ -39,7 +39,7 @@ class QuestEditorMeshManager( ) } - observe( + observeNow( questEditorStore.currentQuest, questEditorStore.currentArea, ) { quest, area -> @@ -54,7 +54,7 @@ class QuestEditorMeshManager( ) } - observe(questEditorStore.showCollisionGeometry) { + observeNow(questEditorStore.showCollisionGeometry) { renderContext.collisionGeometryVisible = it renderContext.renderGeometryVisible = !it } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestMeshManager.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestMeshManager.kt index 28e43918..632acac4 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestMeshManager.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestMeshManager.kt @@ -5,10 +5,10 @@ import kotlinx.coroutines.Job import kotlinx.coroutines.launch import world.phantasmal.core.disposable.Disposable import world.phantasmal.core.disposable.DisposableSupervisedScope -import world.phantasmal.psolib.Episode import world.phantasmal.observable.cell.list.ListCell import world.phantasmal.observable.cell.list.ListChange import world.phantasmal.observable.cell.list.ListChangeEvent +import world.phantasmal.psolib.Episode import world.phantasmal.web.questEditor.loading.AreaAssetLoader import world.phantasmal.web.questEditor.loading.EntityAssetLoader import world.phantasmal.web.questEditor.models.AreaVariantModel @@ -55,7 +55,9 @@ abstract class QuestMeshManager protected constructor( npcObserver?.dispose() npcMeshManager.removeAll() - npcObserver = npcs.observeList(callNow = true, ::npcsChanged) + npcs.value.forEach(npcMeshManager::add) + + npcObserver = npcs.observeListChange(::npcsChanged) } } @@ -65,7 +67,9 @@ abstract class QuestMeshManager protected constructor( objectObserver?.dispose() objectMeshManager.removeAll() - objectObserver = objects.observeList(callNow = true, ::objectsChanged) + objects.value.forEach(objectMeshManager::add) + + objectObserver = objects.observeListChange(::objectsChanged) } } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestRenderer.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestRenderer.kt index ff7cb48f..99a1b06d 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestRenderer.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestRenderer.kt @@ -39,7 +39,7 @@ class QuestRenderer( ), ) - observe(questEditorStore.currentQuest) { inputManager.resetCamera() } - observe(questEditorStore.currentAreaVariant) { inputManager.resetCamera() } + observeNow(questEditorStore.currentQuest) { inputManager.resetCamera() } + observeNow(questEditorStore.currentAreaVariant) { inputManager.resetCamera() } } } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/input/QuestInputManager.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/input/QuestInputManager.kt index e0d7c762..90164fac 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/input/QuestInputManager.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/input/QuestInputManager.kt @@ -84,8 +84,8 @@ class QuestInputManager( stateContext = StateContext(questEditorStore, renderContext, cameraInputManager) state = IdleState(stateContext, entityManipulationEnabled) - observe(questEditorStore.selectedEntity) { returnToIdleState() } - observe(questEditorStore.questEditingEnabled) { entityManipulationEnabled = it } + observeNow(questEditorStore.selectedEntity) { returnToIdleState() } + observeNow(questEditorStore.questEditingEnabled) { entityManipulationEnabled = it } pointerTrap.className = "pw-quest-editor-input-manager-pointer-trap" pointerTrap.hidden = true diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/AsmStore.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/AsmStore.kt index 9d379bb9..94d1ef7f 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/AsmStore.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/AsmStore.kt @@ -55,11 +55,11 @@ class AsmStore( val problems: ListCell = asmAnalyser.problems init { - observe(questEditorStore.currentQuest) { quest -> + observeNow(questEditorStore.currentQuest) { quest -> setTextModel(quest, inlineStackArgs.value) } - observe(inlineStackArgs) { inlineStackArgs -> + observeNow(inlineStackArgs) { inlineStackArgs -> // Ensure we have the most up-to-date bytecode before we disassemble it again. if (setBytecodeIrTimeout != null) { setBytecodeIr() @@ -72,7 +72,7 @@ class AsmStore( scope.launch { questEditorStore.setMapDesignations(it) } } - observe(problems) { problems -> + observeNow(problems) { problems -> textModel.value?.let { model -> val markers = Array(problems.size) { val problem = problems[it] diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/QuestEditorStore.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/QuestEditorStore.kt index 55c1f1f5..9a4d26c1 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/QuestEditorStore.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/QuestEditorStore.kt @@ -94,13 +94,13 @@ class QuestEditorStore( }, ) - observe(uiStore.currentTool) { tool -> + observeNow(uiStore.currentTool) { tool -> if (tool == PwToolType.QuestEditor) { makeMainUndoCurrent() } } - observe(currentQuest.flatMap { it?.npcs ?: emptyListCell() }) { npcs -> + observeNow(currentQuest.flatMap { it?.npcs ?: emptyListCell() }) { npcs -> val selected = selectedEntity.value if (selected is QuestNpcModel && selected !in npcs) { @@ -108,7 +108,7 @@ class QuestEditorStore( } } - observe(currentQuest.flatMap { it?.objects ?: emptyListCell() }) { objects -> + observeNow(currentQuest.flatMap { it?.objects ?: emptyListCell() }) { objects -> val selected = selectedEntity.value if (selected is QuestObjectModel && selected !in objects) { diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/undo/TextModelUndo.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/undo/TextModelUndo.kt index f05e354e..8988f763 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/undo/TextModelUndo.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/undo/TextModelUndo.kt @@ -53,7 +53,7 @@ class TextModelUndo( init { undoManager.addUndo(this) - modelObserver = model.observe(callNow = true) { onModelChange(it.value) } + modelObserver = model.observeNow(::onModelChange) } override fun dispose() { diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/AsmEditorWidget.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/AsmEditorWidget.kt index 0e1a063b..aa08ec76 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/AsmEditorWidget.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/AsmEditorWidget.kt @@ -32,11 +32,11 @@ class AsmEditorWidget(private val ctrl: AsmEditorController) : Widget() { addDisposable(disposable { editor.dispose() }) - observe(ctrl.textModel) { editor.setModel(it) } + observeNow(ctrl.textModel) { editor.setModel(it) } - observe(ctrl.readOnly) { editor.updateOptions(obj { readOnly = it }) } + observeNow(ctrl.readOnly) { editor.updateOptions(obj { readOnly = it }) } - addDisposable(size.observe { (size) -> + addDisposable(size.observeChange { (size) -> if (size.width > .0 && size.height > .0) { editor.layout(obj { width = size.width @@ -65,7 +65,7 @@ class AsmEditorWidget(private val ctrl: AsmEditorController) : Widget() { editor.trigger( source = AsmEditorWidget::class.simpleName, handlerId = "undo", - payload = undefined + payload = undefined, ) } @@ -74,7 +74,7 @@ class AsmEditorWidget(private val ctrl: AsmEditorController) : Widget() { editor.trigger( source = AsmEditorWidget::class.simpleName, handlerId = "redo", - payload = undefined + payload = undefined, ) } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/EntityInfoWidget.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/EntityInfoWidget.kt index e9eb1b46..3a73ca1a 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/EntityInfoWidget.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/EntityInfoWidget.kt @@ -100,7 +100,7 @@ class EntityInfoWidget(private val ctrl: EntityInfoController) : Widget(enabled val inputValue = mutableCell(value.value) var timeout = -1 - observe(value) { + observeNow(value) { if (timeout == -1) { timeout = window.setTimeout({ inputValue.value = value.value diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/EventWidget.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/EventWidget.kt index a410de82..2280d527 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/EventWidget.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/EventWidget.kt @@ -36,7 +36,7 @@ class EventWidget( } } - observe(isSelected) { + observeNow(isSelected) { if (it) { scrollIntoView(obj { behavior = ScrollBehavior.SMOOTH diff --git a/web/src/main/kotlin/world/phantasmal/web/viewer/rendering/MeshRenderer.kt b/web/src/main/kotlin/world/phantasmal/web/viewer/rendering/MeshRenderer.kt index 059008d7..bbd8278a 100644 --- a/web/src/main/kotlin/world/phantasmal/web/viewer/rendering/MeshRenderer.kt +++ b/web/src/main/kotlin/world/phantasmal/web/viewer/rendering/MeshRenderer.kt @@ -56,14 +56,14 @@ class MeshRenderer( ) init { - observe(viewerStore.currentNinjaGeometry) { rebuildMesh(resetCamera = true) } - observe(viewerStore.currentTextures) { rebuildMesh(resetCamera = true) } - observe(viewerStore.applyTextures) { rebuildMesh(resetCamera = false) } - observe(viewerStore.currentNinjaMotion, ::ninjaMotionChanged) - observe(viewerStore.showSkeleton) { skeletonHelper?.visible = it } - observe(viewerStore.animationPlaying, ::animationPlayingChanged) - observe(viewerStore.frameRate, ::frameRateChanged) - observe(viewerStore.frame, ::frameChanged) + observeNow(viewerStore.currentNinjaGeometry) { rebuildMesh(resetCamera = true) } + observeNow(viewerStore.currentTextures) { rebuildMesh(resetCamera = true) } + observeNow(viewerStore.applyTextures) { rebuildMesh(resetCamera = false) } + observeNow(viewerStore.currentNinjaMotion, ::ninjaMotionChanged) + observeNow(viewerStore.showSkeleton) { skeletonHelper?.visible = it } + observeNow(viewerStore.animationPlaying, ::animationPlayingChanged) + observeNow(viewerStore.frameRate, ::frameRateChanged) + observeNow(viewerStore.frame, ::frameChanged) } override fun dispose() { diff --git a/web/src/main/kotlin/world/phantasmal/web/viewer/rendering/TextureRenderer.kt b/web/src/main/kotlin/world/phantasmal/web/viewer/rendering/TextureRenderer.kt index 81987620..56e01f8f 100644 --- a/web/src/main/kotlin/world/phantasmal/web/viewer/rendering/TextureRenderer.kt +++ b/web/src/main/kotlin/world/phantasmal/web/viewer/rendering/TextureRenderer.kt @@ -46,7 +46,7 @@ class TextureRenderer( )) init { - observe(store.currentTextures) { + observeNow(store.currentTextures) { texturesChanged(it.filterNotNull()) } } diff --git a/web/src/test/kotlin/world/phantasmal/web/questEditor/controllers/EventsControllerTests.kt b/web/src/test/kotlin/world/phantasmal/web/questEditor/controllers/EventsControllerTests.kt index ad13dc2a..51051944 100644 --- a/web/src/test/kotlin/world/phantasmal/web/questEditor/controllers/EventsControllerTests.kt +++ b/web/src/test/kotlin/world/phantasmal/web/questEditor/controllers/EventsControllerTests.kt @@ -1,5 +1,6 @@ package world.phantasmal.web.questEditor.controllers +import world.phantasmal.observable.cell.observeNow import world.phantasmal.web.questEditor.models.QuestEventActionModel import world.phantasmal.web.questEditor.models.QuestEventModel import world.phantasmal.web.test.WebTestSuite @@ -115,9 +116,9 @@ class EventsControllerTests : WebTestSuite { // We test the observed value instead of the cell's value property. var canGoToEventValue: Boolean? = null - disposer.add(canGoToEvent.observe(callNow = true) { + disposer.add(canGoToEvent.observeNow { assertNull(canGoToEventValue) - canGoToEventValue = it.value + canGoToEventValue = it }) assertEquals(true, canGoToEventValue) diff --git a/webui/src/main/kotlin/world/phantasmal/webui/DisposableContainer.kt b/webui/src/main/kotlin/world/phantasmal/webui/DisposableContainer.kt index 8a6cbaad..d2abe958 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/DisposableContainer.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/DisposableContainer.kt @@ -4,8 +4,9 @@ import world.phantasmal.core.disposable.Disposable import world.phantasmal.core.disposable.Disposer import world.phantasmal.core.disposable.TrackedDisposable import world.phantasmal.observable.Observable -import world.phantasmal.observable.Observer import world.phantasmal.observable.cell.Cell +import world.phantasmal.observable.cell.observeNow +import world.phantasmal.observable.observe abstract class DisposableContainer : TrackedDisposable() { private val disposer = Disposer() @@ -29,85 +30,55 @@ abstract class DisposableContainer : TrackedDisposable() { disposer.remove(disposable, dispose) } - protected fun observe(observable: Observable, operation: (V1) -> Unit) { - addDisposable( - if (observable is Cell) { - observable.observe(callNow = true) { operation(it.value) } - } else { - observable.observe { operation(it.value) } - } - ) + protected fun observe( + o1: Observable, + observer: (T) -> Unit, + ) { + addDisposable(o1.observe(observer)) } - protected fun observe( - v1: Cell, - v2: Cell, - operation: (V1, V2) -> Unit, + protected fun observeNow( + c1: Cell, + observer: (T) -> Unit, ) { - val observer: Observer<*> = { - operation(v1.value, v2.value) - } - addDisposables( - v1.observe(observer), - v2.observe(observer), - ) - operation(v1.value, v2.value) + addDisposable(c1.observeNow(observer)) } - protected fun observe( - v1: Cell, - v2: Cell, - v3: Cell, - operation: (V1, V2, V3) -> Unit, + protected fun observeNow( + c1: Cell, + c2: Cell, + observer: (T1, T2) -> Unit, ) { - val observer: Observer<*> = { - operation(v1.value, v2.value, v3.value) - } - addDisposables( - v1.observe(observer), - v2.observe(observer), - v3.observe(observer), - ) - operation(v1.value, v2.value, v3.value) + addDisposable(world.phantasmal.observable.cell.observeNow(c1, c2, observer)) } - protected fun observe( - v1: Cell, - v2: Cell, - v3: Cell, - v4: Cell, - operation: (V1, V2, V3, V4) -> Unit, + protected fun observeNow( + c1: Cell, + c2: Cell, + c3: Cell, + observer: (T1, T2, T3) -> Unit, ) { - val observer: Observer<*> = { - operation(v1.value, v2.value, v3.value, v4.value) - } - addDisposables( - v1.observe(observer), - v2.observe(observer), - v3.observe(observer), - v4.observe(observer), - ) - operation(v1.value, v2.value, v3.value, v4.value) + addDisposable(world.phantasmal.observable.cell.observeNow(c1, c2, c3, observer)) } - protected fun observe( - v1: Cell, - v2: Cell, - v3: Cell, - v4: Cell, - v5: Cell, - operation: (V1, V2, V3, V4, V5) -> Unit, + protected fun observeNow( + c1: Cell, + c2: Cell, + c3: Cell, + c4: Cell, + observer: (T1, T2, T3, T4) -> Unit, ) { - val observer: Observer<*> = { - operation(v1.value, v2.value, v3.value, v4.value, v5.value) - } - addDisposables( - v1.observe(observer), - v2.observe(observer), - v3.observe(observer), - v4.observe(observer), - v5.observe(observer), - ) - operation(v1.value, v2.value, v3.value, v4.value, v5.value) + addDisposable(world.phantasmal.observable.cell.observeNow(c1, c2, c3, c4, observer)) + } + + protected fun observeNow( + c1: Cell, + c2: Cell, + c3: Cell, + c4: Cell, + c5: Cell, + observer: (T1, T2, T3, T4, T5) -> Unit, + ) { + addDisposable(world.phantasmal.observable.cell.observeNow(c1, c2, c3, c4, c5, observer)) } } diff --git a/webui/src/main/kotlin/world/phantasmal/webui/dom/Dom.kt b/webui/src/main/kotlin/world/phantasmal/webui/dom/Dom.kt index 95e75f56..4866ebc7 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/dom/Dom.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/dom/Dom.kt @@ -249,11 +249,8 @@ private fun bindChildrenTo( list: Cell>, createChild: Node.(T, index: Int) -> Node, childrenRemoved: () -> Unit, -): Disposable = - list.observe(callNow = true) { (items) -> - parent.innerHTML = "" - childrenRemoved() - +): Disposable { + inline fun createChildNodes(items: List) { val frag = document.createDocumentFragment() items.forEachIndexed { i, item -> @@ -263,14 +260,33 @@ private fun bindChildrenTo( parent.appendChild(frag) } + parent.innerHTML = "" + createChildNodes(list.value) + + return list.observeChange { (items) -> + parent.innerHTML = "" + childrenRemoved() + createChildNodes(items) + } +} + private fun bindChildrenTo( parent: Element, list: ListCell, createChild: Node.(T, index: Int) -> Node, childrenRemoved: (index: Int, count: Int) -> Unit, after: (ListChangeEvent) -> Unit, -): Disposable = - list.observeList(callNow = true) { event: ListChangeEvent -> +): Disposable { + parent.innerHTML = "" + val initialFrag = document.createDocumentFragment() + + list.value.forEachIndexed { i, value -> + initialFrag.appendChild(initialFrag.createChild(value, i)) + } + + parent.appendChild(initialFrag) + + return list.observeListChange { event: ListChangeEvent -> for (change in event.changes) { if (change is ListChange.Structural) { if (change.allRemoved) { @@ -299,3 +315,4 @@ private fun bindChildrenTo( after(event) } +} diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Button.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Button.kt index b6fa3ad8..b8d4bc17 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Button.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Button.kt @@ -54,7 +54,7 @@ open class Button( className = "pw-button-center" if (textCell != null) { - observe(textCell) { + observeNow(textCell) { textContent = it hidden = it.isEmpty() } diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Checkbox.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Checkbox.kt index e1dbc26b..c676626a 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Checkbox.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Checkbox.kt @@ -23,7 +23,7 @@ class Checkbox( type = "checkbox" if (this@Checkbox.checked != null) { - observe(this@Checkbox.checked) { + observeNow(this@Checkbox.checked) { checked = it } } diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/ComboBox.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/ComboBox.kt index 388de9c1..13abb07d 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/ComboBox.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/ComboBox.kt @@ -42,8 +42,8 @@ class ComboBox( id = labelId placeholderText?.let { placeholder = it } hidden(!visible) - observe(enabled) { disabled = !it } - observe(selected) { value = it?.let(itemToString) ?: "" } + observeNow(enabled) { disabled = !it } + observeNow(selected) { value = it?.let(itemToString) ?: "" } onmousedown = ::onInputMouseDown onkeydown = ::onInputKeyDown diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Dialog.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Dialog.kt index 0bd0a8f2..5a536303 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Dialog.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Dialog.kt @@ -73,7 +73,7 @@ open class Dialog( } init { - observe(visible) { + observeNow(visible) { if (it) { setPosition( (window.innerWidth - WIDTH) / 2, diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Input.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Input.kt index 90136798..c5edf5f0 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Input.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Input.kt @@ -34,13 +34,13 @@ abstract class Input( id = labelId classList.add("pw-input-inner") - observe(this@Input.enabled) { disabled = !it } + observeNow(this@Input.enabled) { disabled = !it } onchange = { callOnChange(this) } interceptInputElement(this) - observe(this@Input.value) { + observeNow(this@Input.value) { setInputValue(this, it) } } diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/LazyLoader.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/LazyLoader.kt index 31b9211c..5a175888 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/LazyLoader.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/LazyLoader.kt @@ -16,7 +16,7 @@ class LazyLoader( div { className = "pw-lazy-loader" - observe(this@LazyLoader.visible) { v -> + observeNow(this@LazyLoader.visible) { v -> if (v && !initialized) { initialized = true addChild(createWidget()) diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Menu.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Menu.kt index e817dd2d..b6b419f1 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Menu.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Menu.kt @@ -53,7 +53,7 @@ class Menu( } } - observe(this@Menu.visible) { + observeNow(this@Menu.visible) { if (it) { onDocumentMouseDownListener = document.disposableListener("mousedown", ::onDocumentMouseDown) @@ -66,13 +66,13 @@ class Menu( } } - observe(enabled) { + observeNow(enabled) { if (!it) { clearHighlightItem() } } - observe(items) { + observeNow(items) { clearHighlightItem() } diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Select.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Select.kt index 84c5bf89..ceee5665 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Select.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Select.kt @@ -37,7 +37,7 @@ class Select( this@Select.className?.let { classList.add(it) } // Default to a single space so the inner text part won't be hidden. - observe(selected) { buttonText.value = it?.let(itemToString) ?: " " } + observeNow(selected) { buttonText.value = it?.let(itemToString) ?: " " } addWidget(Button( enabled = enabled, diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/TabContainer.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/TabContainer.kt index 2ba503cd..5b6c5fbd 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/TabContainer.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/TabContainer.kt @@ -28,7 +28,7 @@ class TabContainer( title = tab.title textContent = tab.title - observe(ctrl.activeTab) { + observeNow(ctrl.activeTab) { if (it == tab) { classList.add(ACTIVE_CLASS) } else { @@ -55,7 +55,7 @@ class TabContainer( } init { - observe(selfOrAncestorVisible, ctrl::visibleChanged) + observeNow(selfOrAncestorVisible, ctrl::visibleChanged) } companion object { diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Table.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Table.kt index 362024e7..00040393 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Table.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Table.kt @@ -4,6 +4,7 @@ import org.w3c.dom.* import world.phantasmal.core.disposable.Disposable import world.phantasmal.core.disposable.Disposer import world.phantasmal.observable.cell.Cell +import world.phantasmal.observable.cell.observeNow import world.phantasmal.observable.cell.trueCell import world.phantasmal.webui.LoadingStatus import world.phantasmal.webui.controllers.Column @@ -30,7 +31,7 @@ class Table( div { className = "pw-table-notification" - observe(loadingStatus) { status -> + observeNow(loadingStatus) { status -> when (status) { LoadingStatus.Uninitialized, LoadingStatus.InitialLoad, @@ -167,8 +168,8 @@ class Table( style.width = "${column.width}px" column.textAlign?.let { style.textAlign = it } - disposer.add(column.footer.observe(callNow = true) { textContent = it.value ?: "" }) - disposer.add(column.footerTooltip.observe(callNow = true) { title = it.value ?: "" }) + disposer.add(column.footer.observeNow { textContent = it ?: "" }) + disposer.add(column.footerTooltip.observeNow { title = it ?: "" }) } return Pair(cell, disposer) diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/TextArea.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/TextArea.kt index d75bcb6b..b15239cf 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/TextArea.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/TextArea.kt @@ -37,13 +37,13 @@ class TextArea( id = labelId className = "pw-text-area-inner" - observe(this@TextArea.enabled) { disabled = !it } + observeNow(this@TextArea.enabled) { disabled = !it } if (onChange != null) { onchange = { onChange.invoke(value) } } - observe(this@TextArea.value) { value = it } + observeNow(this@TextArea.value) { value = it } this@TextArea.maxLength?.let { maxLength = it } fontFamily?.let { style.fontFamily = it } diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Widget.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Widget.kt index f82b319a..3285cd2e 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Widget.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Widget.kt @@ -12,7 +12,6 @@ import org.w3c.dom.pointerevents.PointerEvent import world.phantasmal.core.disposable.Disposable import world.phantasmal.core.disposable.DisposableSupervisedScope import world.phantasmal.core.disposable.disposable -import world.phantasmal.observable.Observable import world.phantasmal.observable.cell.* import world.phantasmal.webui.DisposableContainer import world.phantasmal.webui.dom.* @@ -36,12 +35,12 @@ abstract class Widget( private val elementDelegate = lazy { val el = documentFragment().createElement() - observe(visible) { visible -> + observeNow(visible) { visible -> el.hidden = !visible children.forEach { setAncestorVisible(it, visible && ancestorVisible.value) } } - observe(enabled) { enabled -> + observeNow(enabled) { enabled -> if (enabled) { el.removeAttribute("disabled") el.classList.remove("pw-disabled") @@ -51,7 +50,7 @@ abstract class Widget( } } - observe(tooltip) { tooltip -> + observeNow(tooltip) { tooltip -> if (tooltip == null) { el.removeAttribute("title") } else { @@ -111,16 +110,16 @@ abstract class Widget( super.dispose() } - protected fun Node.text(observable: Observable) { - observe(observable) { textContent = it } + protected fun Node.text(cell: Cell) { + observeNow(cell) { textContent = it } } - protected fun HTMLElement.hidden(observable: Observable) { - observe(observable) { hidden = it } + protected fun HTMLElement.hidden(cell: Cell) { + observeNow(cell) { hidden = it } } - protected fun HTMLElement.toggleClass(className: String, observable: Observable) { - observe(observable) { + protected fun HTMLElement.toggleClass(className: String, cell: Cell) { + observeNow(cell) { if (it) classList.add(className) else classList.remove(className) }