From e5c1c81be3ef74f177d66a3921e1d5bced830cb2 Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Sun, 30 May 2021 15:16:58 +0200 Subject: [PATCH] Propagation of changes to observables can now be deferred until the end of a code block. --- .../world/phantasmal/observable/Change.kt | 10 ++ .../phantasmal/observable/ChangeManager.kt | 45 +++++++ .../world/phantasmal/observable/Dependency.kt | 5 + .../world/phantasmal/observable/Observer.kt | 7 +- .../phantasmal/observable/SimpleEmitter.kt | 20 ++- .../observable/cell/AbstractCell.kt | 10 +- .../observable/cell/AbstractDependentCell.kt | 10 +- .../observable/cell/DelegatingCell.kt | 7 +- .../observable/cell/DependentCell.kt | 4 +- .../cell/FlatteningDependentCell.kt | 4 +- .../phantasmal/observable/cell/SimpleCell.kt | 7 +- .../phantasmal/observable/cell/StaticCell.kt | 4 + .../cell/list/AbstractDependentListCell.kt | 2 +- .../observable/cell/list/FilteredListCell.kt | 12 +- .../observable/cell/list/SimpleListCell.kt | 39 +++--- .../observable/cell/list/StaticListCell.kt | 4 + .../cell/CellWithDependenciesTests.kt | 5 + .../phantasmal/observable/cell/ChangeTests.kt | 37 +++++ .../observable/cell/MutableCellTests.kt | 115 +++++++++++++++- .../questEditor/actions/CreateEntityAction.kt | 7 +- .../questEditor/actions/CreateEventAction.kt | 13 +- .../actions/CreateEventActionAction.kt | 13 +- .../questEditor/actions/DeleteEntityAction.kt | 9 +- .../questEditor/actions/DeleteEventAction.kt | 13 +- .../actions/DeleteEventActionAction.kt | 13 +- .../actions/EditEntityPropAction.kt | 13 +- .../actions/EditEventPropertyAction.kt | 13 +- .../questEditor/actions/RotateEntityAction.kt | 25 ++-- .../actions/TranslateEntityAction.kt | 17 ++- .../questEditor/models/QuestEntityModel.kt | 127 ++++++++++-------- .../controllers/ViewerToolbarController.kt | 27 ++-- .../web/viewer/stores/ViewerStore.kt | 60 +++++---- .../webui/dom/HTMLElementSizeCell.kt | 6 +- 33 files changed, 521 insertions(+), 182 deletions(-) create mode 100644 observable/src/commonMain/kotlin/world/phantasmal/observable/Change.kt create mode 100644 observable/src/commonMain/kotlin/world/phantasmal/observable/ChangeManager.kt create mode 100644 observable/src/commonTest/kotlin/world/phantasmal/observable/cell/ChangeTests.kt diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/Change.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/Change.kt new file mode 100644 index 00000000..277a6bbe --- /dev/null +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/Change.kt @@ -0,0 +1,10 @@ +package world.phantasmal.observable + +/** + * Defer propagation of changes to observables until the end of a code block. All changes to + * observables in a single change set won't be propagated to their dependencies until the change set + * is completed. + */ +fun change(block: () -> Unit) { + ChangeManager.inChangeSet(block) +} diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/ChangeManager.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/ChangeManager.kt new file mode 100644 index 00000000..11ef2952 --- /dev/null +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/ChangeManager.kt @@ -0,0 +1,45 @@ +package world.phantasmal.observable + +object ChangeManager { + private var currentChangeSet: ChangeSet? = null + + fun inChangeSet(block: () -> Unit) { + val existingChangeSet = currentChangeSet + val changeSet = existingChangeSet ?: ChangeSet().also { + currentChangeSet = it + } + + try { + block() + } finally { + if (existingChangeSet == null) { + currentChangeSet = null + changeSet.complete() + } + } + } + + fun changed(dependency: Dependency) { + val changeSet = currentChangeSet + + if (changeSet == null) { + dependency.emitDependencyChanged() + } else { + changeSet.changed(dependency) + } + } +} + +private class ChangeSet { + private val changedDependencies: MutableList = mutableListOf() + + fun changed(dependency: Dependency) { + changedDependencies.add(dependency) + } + + fun complete() { + for (dependency in changedDependencies) { + dependency.emitDependencyChanged() + } + } +} diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/Dependency.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/Dependency.kt index 06220dd1..3c6e74df 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/Dependency.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/Dependency.kt @@ -11,4 +11,9 @@ interface Dependency { * This method is not meant to be called from typical application code. */ fun removeDependent(dependent: Dependent) + + /** + * This method is not meant to be called from typical application code. + */ + fun emitDependencyChanged() } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/Observer.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/Observer.kt index ac24aef3..12b3fa8c 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/Observer.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/Observer.kt @@ -1,6 +1,11 @@ package world.phantasmal.observable -open class ChangeEvent(val value: T) { +open class ChangeEvent( + /** + * The observable's new value + */ + val value: T, +) { operator fun component1() = value } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/SimpleEmitter.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/SimpleEmitter.kt index dd1e750c..e9a9bf32 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/SimpleEmitter.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/SimpleEmitter.kt @@ -3,16 +3,30 @@ package world.phantasmal.observable import world.phantasmal.core.disposable.Disposable class SimpleEmitter : AbstractDependency(), Emitter { + private var event: ChangeEvent? = null + override fun emit(event: ChangeEvent) { for (dependent in dependents) { dependent.dependencyMightChange() } - for (dependent in dependents) { - dependent.dependencyChanged(this, event) - } + this.event = event + + ChangeManager.changed(this) } override fun observe(observer: Observer): Disposable = CallbackObserver(this, observer) + + override fun emitDependencyChanged() { + if (event != null) { + try { + for (dependent in dependents) { + dependent.dependencyChanged(this, event) + } + } finally { + 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 6ba8db07..37b7564b 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/AbstractCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/AbstractCell.kt @@ -32,11 +32,13 @@ abstract class AbstractCell : AbstractDependency(), Cell { } } - protected fun emitChanged(event: ChangeEvent?) { - mightChangeEmitted = false + protected fun emitDependencyChanged(event: ChangeEvent<*>?) { + if (mightChangeEmitted) { + mightChangeEmitted = false - for (dependent in dependents) { - dependent.dependencyChanged(this, event) + for (dependent in dependents) { + dependent.dependencyChanged(this, event) + } } } } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/AbstractDependentCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/AbstractDependentCell.kt index f928f75e..6299392f 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/AbstractDependentCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/AbstractDependentCell.kt @@ -24,10 +24,16 @@ abstract class AbstractDependentCell : AbstractCell(), Dependent { dependenciesChanged() } else { - emitChanged(null) + emitDependencyChanged(null) } } } - abstract fun dependenciesChanged() + override fun emitDependencyChanged() { + // Nothing to do because dependent cells emit dependencyChanged immediately. They don't + // defer this operation because they only change when there is no transaction or the current + // transaction is being committed. + } + + protected abstract fun dependenciesChanged() } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/DelegatingCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/DelegatingCell.kt index 36fa4bb5..88a6966a 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/DelegatingCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/DelegatingCell.kt @@ -1,6 +1,7 @@ package world.phantasmal.observable.cell import world.phantasmal.observable.ChangeEvent +import world.phantasmal.observable.ChangeManager class DelegatingCell( private val getter: () -> T, @@ -16,7 +17,11 @@ class DelegatingCell( setter(value) - emitChanged(ChangeEvent(value)) + ChangeManager.changed(this) } } + + override fun emitDependencyChanged() { + emitDependencyChanged(ChangeEvent(value)) + } } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/DependentCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/DependentCell.kt index 4c6bb395..d5a2b07a 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/DependentCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/DependentCell.kt @@ -50,9 +50,9 @@ class DependentCell( if (newValue != _value) { _value = newValue - emitChanged(ChangeEvent(newValue)) + emitDependencyChanged(ChangeEvent(newValue)) } else { - emitChanged(null) + emitDependencyChanged(null) } } } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/FlatteningDependentCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/FlatteningDependentCell.kt index 960e395e..3c618cc4 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/FlatteningDependentCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/FlatteningDependentCell.kt @@ -82,9 +82,9 @@ class FlatteningDependentCell( if (newValue != _value) { _value = newValue - emitChanged(ChangeEvent(newValue)) + emitDependencyChanged(ChangeEvent(newValue)) } else { - emitChanged(null) + emitDependencyChanged(null) } } } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/SimpleCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/SimpleCell.kt index 4d71655d..20ba550e 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/SimpleCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/SimpleCell.kt @@ -1,6 +1,7 @@ package world.phantasmal.observable.cell import world.phantasmal.observable.ChangeEvent +import world.phantasmal.observable.ChangeManager class SimpleCell(value: T) : AbstractCell(), MutableCell { override var value: T = value @@ -10,7 +11,11 @@ class SimpleCell(value: T) : AbstractCell(), MutableCell { field = value - emitChanged(ChangeEvent(value)) + ChangeManager.changed(this) } } + + override fun emitDependencyChanged() { + emitDependencyChanged(ChangeEvent(value)) + } } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/StaticCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/StaticCell.kt index 8b64a27e..e07c8881 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/StaticCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/StaticCell.kt @@ -16,4 +16,8 @@ class StaticCell(override val value: T) : AbstractDependency(), Cell { } override fun observe(observer: Observer): Disposable = nopDisposable() + + override fun emitDependencyChanged() { + error("StaticCell 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 7f66f7de..be93e940 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 @@ -57,7 +57,7 @@ abstract class AbstractDependentListCell : computeElements() - emitChanged( + emitDependencyChanged( ListChangeEvent(elements, listOf(ListChange.Structural(0, oldElements, elements))) ) } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/FilteredListCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/FilteredListCell.kt index ab637839..878d55cb 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/FilteredListCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/FilteredListCell.kt @@ -179,15 +179,21 @@ class FilteredListCell( } if (filteredChanges.isEmpty()) { - emitChanged(null) + emitDependencyChanged(null) } else { - emitChanged(ListChangeEvent(elements, filteredChanges)) + emitDependencyChanged(ListChangeEvent(elements, filteredChanges)) } } else { - emitChanged(null) + emitDependencyChanged(null) } } + override fun emitDependencyChanged() { + // Nothing to do because FilteredListCell emits dependencyChanged immediately. We don't + // defer this operation because FilteredListCell only changes when there is no transaction + // or the current transaction is being committed. + } + private fun recompute() { val newElements = mutableListOf() indexMap.clear() diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/SimpleListCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/SimpleListCell.kt index 3afadc9e..86e94a1a 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/SimpleListCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/SimpleListCell.kt @@ -5,6 +5,7 @@ import world.phantasmal.core.unsafe.unsafeAssertNotNull import world.phantasmal.observable.ChangeEvent import world.phantasmal.observable.Dependency import world.phantasmal.observable.Dependent +import world.phantasmal.observable.ChangeManager typealias DependenciesExtractor = (element: E) -> Array @@ -27,7 +28,7 @@ class SimpleListCell( */ private val elementDependents = mutableListOf() private var changingElements = 0 - private var elementListChanges = mutableListOf>() + private var changes = mutableListOf>() override var value: List get() = elements @@ -50,12 +51,8 @@ class SimpleListCell( elementDependents[index] = ElementDependent(index, element) } - emitChanged( - ListChangeEvent( - elements, - listOf(ListChange.Structural(index, listOf(removed), listOf(element))), - ), - ) + changes.add(ListChange.Structural(index, listOf(removed), listOf(element))) + ChangeManager.changed(this) return removed } @@ -180,6 +177,14 @@ class SimpleListCell( } } + override fun emitDependencyChanged() { + try { + emitDependencyChanged(ListChangeEvent(elements, changes)) + } finally { + changes = mutableListOf() + } + } + private fun checkIndex(index: Int, maxIndex: Int) { if (index !in 0..maxIndex) { throw IndexOutOfBoundsException( @@ -206,12 +211,8 @@ class SimpleListCell( } } - emitChanged( - ListChangeEvent( - elements, - listOf(ListChange.Structural(index, removed, inserted)), - ), - ) + changes.add(ListChange.Structural(index, removed, inserted)) + ChangeManager.changed(this) } private inner class ElementDependent( @@ -249,19 +250,11 @@ class SimpleListCell( if (--changingDependencies == 0) { if (dependenciesActuallyChanged) { dependenciesActuallyChanged = false - elementListChanges.add(ListChange.Element(index, element)) + changes.add(ListChange.Element(index, element)) } if (--changingElements == 0) { - try { - if (elementListChanges.isNotEmpty()) { - emitChanged(ListChangeEvent(value, elementListChanges)) - } else { - emitChanged(null) - } - } finally { - elementListChanges = mutableListOf() - } + ChangeManager.changed(this@SimpleListCell) } } } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/StaticListCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/StaticListCell.kt index fd7d9259..f89865de 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/StaticListCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/StaticListCell.kt @@ -45,4 +45,8 @@ class StaticListCell(private val elements: List) : AbstractDependency(), L return unsafeAssertNotNull(firstOrNull) } + + override fun emitDependencyChanged() { + error("StaticListCell can't change.") + } } 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 9a1760d7..17fecd95 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/CellWithDependenciesTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/CellWithDependenciesTests.kt @@ -36,6 +36,11 @@ interface CellWithDependenciesTests : CellTests { val publicDependents: List = dependents override val value: Int = 5 + + override fun emitDependencyChanged() { + // Not going to change. + throw NotImplementedError() + } } val cell = p.createWithDependencies(dependency) diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/ChangeTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/ChangeTests.kt new file mode 100644 index 00000000..d3a18b02 --- /dev/null +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/ChangeTests.kt @@ -0,0 +1,37 @@ +package world.phantasmal.observable.cell + +import world.phantasmal.observable.change +import world.phantasmal.observable.test.ObservableTestSuite +import kotlin.test.Test +import kotlin.test.assertEquals +import kotlin.test.assertFails + +class ChangeTests : ObservableTestSuite { + @Test + fun exceptions_during_a_change_set_are_allowed() = test { + val dependency = mutableCell(7) + val dependent = dependency.map { 2 * it } + + var dependentObservedValue: Int? = null + disposer.add(dependent.observe { dependentObservedValue = it.value }) + + assertFails { + change { + dependency.value = 11 + throw Exception() + } + } + + // The change to dependency is still propagated because it happened before the exception. + assertEquals(22, dependentObservedValue) + assertEquals(22, dependent.value) + + // The machinery behind change is still in a valid state. + change { + dependency.value = 13 + } + + assertEquals(26, dependentObservedValue) + assertEquals(26, dependent.value) + } +} 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 266b4396..56f61f2b 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/MutableCellTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/MutableCellTests.kt @@ -1,5 +1,9 @@ package world.phantasmal.observable.cell +import world.phantasmal.observable.ChangeEvent +import world.phantasmal.observable.Dependency +import world.phantasmal.observable.Dependent +import world.phantasmal.observable.change import kotlin.test.Test import kotlin.test.assertEquals import kotlin.test.assertNull @@ -25,12 +29,121 @@ interface MutableCellTests : CellTests { assertEquals(newValue, observedValue) } + /** + * Modifying mutable cells in a change set doesn't result in calls to + * [Dependent.dependencyChanged] of their dependents until the change set is completed. + */ + @Test + fun cell_changes_in_change_set_dont_immediately_produce_dependencyChanged_calls() = test { + val dependencies = (1..5).map { createProvider() } + + var dependencyMightChangeCount = 0 + var dependencyChangedCount = 0 + + val dependent = object : Dependent { + override fun dependencyMightChange() { + dependencyMightChangeCount++ + } + + override fun dependencyChanged(dependency: Dependency, event: ChangeEvent<*>?) { + dependencyChangedCount++ + } + } + + for (dependency in dependencies) { + dependency.observable.addDependent(dependent) + } + + change { + for (dependency in dependencies) { + dependency.observable.value = dependency.createValue() + } + + // Calls to dependencyMightChange happen immediately. + assertEquals(dependencies.size, dependencyMightChangeCount) + // Calls to dependencyChanged happen later. + assertEquals(0, dependencyChangedCount) + } + + assertEquals(dependencies.size, dependencyMightChangeCount) + assertEquals(dependencies.size, dependencyChangedCount) + } + + /** + * Modifying a mutable cell multiple times in one change set results in a single call to + * [Dependent.dependencyMightChange] and [Dependent.dependencyChanged]. + */ + @Test + fun multiple_changes_to_one_cell_in_change_set() = test { + val dependency = createProvider() + + var dependencyMightChangeCount = 0 + var dependencyChangedCount = 0 + + val dependent = object : Dependent { + override fun dependencyMightChange() { + dependencyMightChangeCount++ + } + + override fun dependencyChanged(dependency: Dependency, event: ChangeEvent<*>?) { + dependencyChangedCount++ + } + } + + dependency.observable.addDependent(dependent) + + // Change the dependency multiple times in a transaction. + change { + repeat(5) { + dependency.observable.value = dependency.createValue() + } + + // Calls to dependencyMightChange happen immediately. + assertEquals(1, dependencyMightChangeCount) + // Calls to dependencyChanged happen later. + assertEquals(0, dependencyChangedCount) + } + + assertEquals(1, dependencyMightChangeCount) + assertEquals(1, dependencyChangedCount) + } + + /** + * Modifying two mutable cells in a change set results in a single recomputation of their + * dependent. + */ + @Test + fun modifying_two_cells_together_results_in_one_recomputation() = test { + val dependency1 = createProvider() + val dependency2 = createProvider() + + var computeCount = 0 + + val dependent = DependentCell(dependency1.observable, dependency2.observable) { + computeCount++ + Unit + } + + // Observe dependent to ensure it gets recomputed when its dependencies change. + disposer.add(dependent.observe {}) + + // DependentCell's compute function is called once when we start observing. + assertEquals(1, computeCount) + + change { + dependency1.observable.value = dependency1.createValue() + dependency2.observable.value = dependency2.createValue() + } + + assertEquals(2, computeCount) + } + interface Provider : CellTests.Provider { override val observable: MutableCell /** * Returns a value that can be assigned to [observable] and that's different from - * [observable]'s current value. + * [observable]'s current and all previous values. */ fun createValue(): T } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/CreateEntityAction.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/CreateEntityAction.kt index 71bff2eb..4f2546c2 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/CreateEntityAction.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/CreateEntityAction.kt @@ -1,5 +1,6 @@ package world.phantasmal.web.questEditor.actions +import world.phantasmal.observable.change import world.phantasmal.web.core.actions.Action import world.phantasmal.web.questEditor.models.QuestEntityModel import world.phantasmal.web.questEditor.models.QuestModel @@ -12,8 +13,10 @@ class CreateEntityAction( override val description: String = "Add ${entity.type.name}" override fun execute() { - quest.addEntity(entity) - setSelectedEntity(entity) + change { + quest.addEntity(entity) + setSelectedEntity(entity) + } } override fun undo() { diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/CreateEventAction.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/CreateEventAction.kt index 743a47fe..4e8cea2a 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/CreateEventAction.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/CreateEventAction.kt @@ -1,5 +1,6 @@ package world.phantasmal.web.questEditor.actions +import world.phantasmal.observable.change import world.phantasmal.web.core.actions.Action import world.phantasmal.web.questEditor.models.QuestEventModel import world.phantasmal.web.questEditor.models.QuestModel @@ -13,12 +14,16 @@ class CreateEventAction( override val description: String = "Add event ${event.id.value}" override fun execute() { - quest.addEvent(index, event) - setSelectedEvent(event) + change { + quest.addEvent(index, event) + setSelectedEvent(event) + } } override fun undo() { - setSelectedEvent(null) - quest.removeEvent(event) + change { + setSelectedEvent(null) + quest.removeEvent(event) + } } } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/CreateEventActionAction.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/CreateEventActionAction.kt index d86ea594..9d9e12bc 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/CreateEventActionAction.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/CreateEventActionAction.kt @@ -1,5 +1,6 @@ package world.phantasmal.web.questEditor.actions +import world.phantasmal.observable.change import world.phantasmal.web.core.actions.Action import world.phantasmal.web.questEditor.models.QuestEventActionModel import world.phantasmal.web.questEditor.models.QuestEventModel @@ -16,12 +17,16 @@ class CreateEventActionAction( "Add ${action.shortName} action to event ${event.id.value}" override fun execute() { - event.addAction(action) - setSelectedEvent(event) + change { + event.addAction(action) + setSelectedEvent(event) + } } override fun undo() { - event.removeAction(action) - setSelectedEvent(event) + change { + event.removeAction(action) + setSelectedEvent(event) + } } } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/DeleteEntityAction.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/DeleteEntityAction.kt index 6cb63532..8a81d066 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/DeleteEntityAction.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/DeleteEntityAction.kt @@ -1,5 +1,6 @@ package world.phantasmal.web.questEditor.actions +import world.phantasmal.observable.change import world.phantasmal.web.core.actions.Action import world.phantasmal.web.questEditor.models.QuestEntityModel import world.phantasmal.web.questEditor.models.QuestModel @@ -8,7 +9,7 @@ class DeleteEntityAction( private val setSelectedEntity: (QuestEntityModel<*, *>) -> Unit, private val quest: QuestModel, private val entity: QuestEntityModel<*, *>, -) :Action{ +) : Action { override val description: String = "Delete ${entity.type.name}" override fun execute() { @@ -16,7 +17,9 @@ class DeleteEntityAction( } override fun undo() { - quest.addEntity(entity) - setSelectedEntity(entity) + change { + quest.addEntity(entity) + setSelectedEntity(entity) + } } } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/DeleteEventAction.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/DeleteEventAction.kt index c8732cc3..61bcdd7a 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/DeleteEventAction.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/DeleteEventAction.kt @@ -1,5 +1,6 @@ package world.phantasmal.web.questEditor.actions +import world.phantasmal.observable.change import world.phantasmal.web.core.actions.Action import world.phantasmal.web.questEditor.models.QuestEventModel import world.phantasmal.web.questEditor.models.QuestModel @@ -13,12 +14,16 @@ class DeleteEventAction( override val description: String = "Delete event ${event.id.value}" override fun execute() { - setSelectedEvent(null) - quest.removeEvent(event) + change { + setSelectedEvent(null) + quest.removeEvent(event) + } } override fun undo() { - quest.addEvent(index, event) - setSelectedEvent(event) + change { + quest.addEvent(index, event) + setSelectedEvent(event) + } } } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/DeleteEventActionAction.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/DeleteEventActionAction.kt index 085b6c03..658c774a 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/DeleteEventActionAction.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/DeleteEventActionAction.kt @@ -1,5 +1,6 @@ package world.phantasmal.web.questEditor.actions +import world.phantasmal.observable.change import world.phantasmal.web.core.actions.Action import world.phantasmal.web.questEditor.models.QuestEventActionModel import world.phantasmal.web.questEditor.models.QuestEventModel @@ -17,12 +18,16 @@ class DeleteEventActionAction( "Remove ${action.shortName} action from event ${event.id.value}" override fun execute() { - setSelectedEvent(event) - event.removeAction(action) + change { + setSelectedEvent(event) + event.removeAction(action) + } } override fun undo() { - setSelectedEvent(event) - event.addAction(index, action) + change { + setSelectedEvent(event) + event.addAction(index, action) + } } } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/EditEntityPropAction.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/EditEntityPropAction.kt index b965e20c..d143fcee 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/EditEntityPropAction.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/EditEntityPropAction.kt @@ -1,5 +1,6 @@ package world.phantasmal.web.questEditor.actions +import world.phantasmal.observable.change import world.phantasmal.web.core.actions.Action import world.phantasmal.web.questEditor.models.QuestEntityModel import world.phantasmal.web.questEditor.models.QuestEntityPropModel @@ -14,12 +15,16 @@ class EditEntityPropAction( override val description: String = "Edit ${entity.type.simpleName} ${prop.name}" override fun execute() { - setSelectedEntity(entity) - prop.setValue(newValue) + change { + setSelectedEntity(entity) + prop.setValue(newValue) + } } override fun undo() { - setSelectedEntity(entity) - prop.setValue(oldValue) + change { + setSelectedEntity(entity) + prop.setValue(oldValue) + } } } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/EditEventPropertyAction.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/EditEventPropertyAction.kt index 193810bf..628a46e7 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/EditEventPropertyAction.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/EditEventPropertyAction.kt @@ -1,5 +1,6 @@ package world.phantasmal.web.questEditor.actions +import world.phantasmal.observable.change import world.phantasmal.web.core.actions.Action import world.phantasmal.web.questEditor.models.QuestEventModel @@ -12,12 +13,16 @@ class EditEventPropertyAction( private val oldValue: T, ) : Action { override fun execute() { - setSelectedEvent(event) - setter(newValue) + change { + setSelectedEvent(event) + setter(newValue) + } } override fun undo() { - setSelectedEvent(event) - setter(oldValue) + change { + setSelectedEvent(event) + setter(oldValue) + } } } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/RotateEntityAction.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/RotateEntityAction.kt index 1b2c5596..5b20827f 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/RotateEntityAction.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/RotateEntityAction.kt @@ -1,5 +1,6 @@ package world.phantasmal.web.questEditor.actions +import world.phantasmal.observable.change import world.phantasmal.web.core.actions.Action import world.phantasmal.web.externals.three.Euler import world.phantasmal.web.questEditor.models.QuestEntityModel @@ -14,22 +15,26 @@ class RotateEntityAction( override val description: String = "Rotate ${entity.type.simpleName}" override fun execute() { - setSelectedEntity(entity) + change { + setSelectedEntity(entity) - if (world) { - entity.setWorldRotation(newRotation) - } else { - entity.setRotation(newRotation) + if (world) { + entity.setWorldRotation(newRotation) + } else { + entity.setRotation(newRotation) + } } } override fun undo() { - setSelectedEntity(entity) + change { + setSelectedEntity(entity) - if (world) { - entity.setWorldRotation(oldRotation) - } else { - entity.setRotation(oldRotation) + if (world) { + entity.setWorldRotation(oldRotation) + } else { + entity.setRotation(oldRotation) + } } } } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/TranslateEntityAction.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/TranslateEntityAction.kt index cf45cfb2..edf790cc 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/TranslateEntityAction.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/TranslateEntityAction.kt @@ -1,5 +1,6 @@ package world.phantasmal.web.questEditor.actions +import world.phantasmal.observable.change import world.phantasmal.web.core.actions.Action import world.phantasmal.web.externals.three.Vector3 import world.phantasmal.web.questEditor.models.QuestEntityModel @@ -16,18 +17,22 @@ class TranslateEntityAction( override val description: String = "Move ${entity.type.simpleName}" override fun execute() { - setSelectedEntity(entity) + change { + setSelectedEntity(entity) - newSection?.let(setEntitySection) + newSection?.let(setEntitySection) - entity.setPosition(newPosition) + entity.setPosition(newPosition) + } } override fun undo() { - setSelectedEntity(entity) + change { + setSelectedEntity(entity) - oldSection?.let(setEntitySection) + oldSection?.let(setEntitySection) - entity.setPosition(oldPosition) + entity.setPosition(oldPosition) + } } } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModel.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModel.kt index 559063a0..dd684e4c 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModel.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModel.kt @@ -7,6 +7,7 @@ import world.phantasmal.observable.cell.Cell import world.phantasmal.observable.cell.list.ListCell import world.phantasmal.observable.cell.list.listCell import world.phantasmal.observable.cell.mutableCell +import world.phantasmal.observable.change import world.phantasmal.web.core.minus import world.phantasmal.web.core.rendering.conversion.vec3ToEuler import world.phantasmal.web.core.rendering.conversion.vec3ToThree @@ -60,11 +61,13 @@ abstract class QuestEntityModel>( }) open fun setSectionId(sectionId: Int) { - entity.sectionId = sectionId.toShort() - _sectionId.value = sectionId + change { + entity.sectionId = sectionId.toShort() + _sectionId.value = sectionId - if (sectionId != _section.value?.id) { - _section.value = null + if (sectionId != _section.value?.id) { + _section.value = null + } } } @@ -81,87 +84,97 @@ abstract class QuestEntityModel>( "Quest entities can't be moved across areas." } - entity.sectionId = section.id.toShort() - _sectionId.value = section.id + change { + entity.sectionId = section.id.toShort() + _sectionId.value = section.id - _section.value = section + _section.value = section - if (keepRelativeTransform) { - // Update world position and rotation by calling setPosition and setRotation with the - // current position and rotation. - setPosition(position.value) - setRotation(rotation.value) - } else { - // Update relative position and rotation by calling setWorldPosition and - // setWorldRotation with the current world position and rotation. - setWorldPosition(worldPosition.value) - setWorldRotation(worldRotation.value) + if (keepRelativeTransform) { + // Update world position and rotation by calling setPosition and setRotation with the + // current position and rotation. + setPosition(position.value) + setRotation(rotation.value) + } else { + // Update relative position and rotation by calling setWorldPosition and + // setWorldRotation with the current world position and rotation. + setWorldPosition(worldPosition.value) + setWorldRotation(worldRotation.value) + } + + setSectionInitialized() } - - setSectionInitialized() } fun setPosition(pos: Vector3) { - entity.setPosition(pos.x.toFloat(), pos.y.toFloat(), pos.z.toFloat()) + change { + entity.setPosition(pos.x.toFloat(), pos.y.toFloat(), pos.z.toFloat()) - _position.value = pos + _position.value = pos - val section = section.value + val section = section.value - _worldPosition.value = - if (section == null) pos - else pos.clone().applyEuler(section.rotation).add(section.position) + _worldPosition.value = + if (section == null) pos + else pos.clone().applyEuler(section.rotation).add(section.position) + } } fun setWorldPosition(pos: Vector3) { - val section = section.value + change { + val section = section.value - val relPos = - if (section == null) pos - else (pos - section.position).applyEuler(section.inverseRotation) + val relPos = + if (section == null) pos + else (pos - section.position).applyEuler(section.inverseRotation) - entity.setPosition(relPos.x.toFloat(), relPos.y.toFloat(), relPos.z.toFloat()) + entity.setPosition(relPos.x.toFloat(), relPos.y.toFloat(), relPos.z.toFloat()) - _worldPosition.value = pos - _position.value = relPos + _worldPosition.value = pos + _position.value = relPos + } } fun setRotation(rot: Euler) { - floorModEuler(rot) + change { + floorModEuler(rot) - entity.setRotation(rot.x.toFloat(), rot.y.toFloat(), rot.z.toFloat()) - _rotation.value = rot + entity.setRotation(rot.x.toFloat(), rot.y.toFloat(), rot.z.toFloat()) + _rotation.value = rot - val section = section.value + val section = section.value - if (section == null) { - _worldRotation.value = rot - } else { - q1.setFromEuler(rot) - q2.setFromEuler(section.rotation) - q1 *= q2 - _worldRotation.value = floorModEuler(q1.toEuler()) + if (section == null) { + _worldRotation.value = rot + } else { + q1.setFromEuler(rot) + q2.setFromEuler(section.rotation) + q1 *= q2 + _worldRotation.value = floorModEuler(q1.toEuler()) + } } } fun setWorldRotation(rot: Euler) { - floorModEuler(rot) + change { + floorModEuler(rot) - val section = section.value + val section = section.value - val relRot = if (section == null) { - rot - } else { - q1.setFromEuler(rot) - q2.setFromEuler(section.rotation) - q2.invert() - q1 *= q2 - floorModEuler(q1.toEuler()) + val relRot = if (section == null) { + rot + } else { + q1.setFromEuler(rot) + q2.setFromEuler(section.rotation) + q2.invert() + q1 *= q2 + floorModEuler(q1.toEuler()) + } + + entity.setRotation(relRot.x.toFloat(), relRot.y.toFloat(), relRot.z.toFloat()) + _worldRotation.value = rot + _rotation.value = relRot } - - entity.setRotation(relRot.x.toFloat(), relRot.y.toFloat(), relRot.z.toFloat()) - _worldRotation.value = rot - _rotation.value = relRot } companion object { diff --git a/web/src/main/kotlin/world/phantasmal/web/viewer/controllers/ViewerToolbarController.kt b/web/src/main/kotlin/world/phantasmal/web/viewer/controllers/ViewerToolbarController.kt index da7321dd..cd3f2cdb 100644 --- a/web/src/main/kotlin/world/phantasmal/web/viewer/controllers/ViewerToolbarController.kt +++ b/web/src/main/kotlin/world/phantasmal/web/viewer/controllers/ViewerToolbarController.kt @@ -15,6 +15,7 @@ import world.phantasmal.lib.fileFormats.parseAreaCollisionGeometry import world.phantasmal.lib.fileFormats.parseAreaRenderGeometry import world.phantasmal.observable.cell.Cell import world.phantasmal.observable.cell.mutableCell +import world.phantasmal.observable.change import world.phantasmal.web.core.files.cursor import world.phantasmal.web.viewer.stores.NinjaGeometry import world.phantasmal.web.viewer.stores.ViewerStore @@ -75,11 +76,11 @@ class ViewerToolbarController(private val store: ViewerStore) : Controller() { val result = PwResult.build(logger) var success = false - try { - var ninjaGeometry: NinjaGeometry? = null - var textures: List? = null - var ninjaMotion: NjMotion? = null + var ninjaGeometry: NinjaGeometry? = null + var textures: List? = null + var ninjaMotion: NjMotion? = null + try { for (file in files) { val extension = file.extension()?.toLowerCase() @@ -92,7 +93,8 @@ class ViewerToolbarController(private val store: ViewerStore) : Controller() { fileResult = njResult if (njResult is Success) { - ninjaGeometry = njResult.value.firstOrNull()?.let(NinjaGeometry::Object) + ninjaGeometry = + njResult.value.firstOrNull()?.let(NinjaGeometry::Object) } } @@ -101,7 +103,8 @@ class ViewerToolbarController(private val store: ViewerStore) : Controller() { fileResult = xjResult if (xjResult is Success) { - ninjaGeometry = xjResult.value.firstOrNull()?.let(NinjaGeometry::Object) + ninjaGeometry = + xjResult.value.firstOrNull()?.let(NinjaGeometry::Object) } } @@ -157,15 +160,17 @@ class ViewerToolbarController(private val store: ViewerStore) : Controller() { success = true } } - - ninjaGeometry?.let(store::setCurrentNinjaGeometry) - textures?.let(store::setCurrentTextures) - ninjaMotion?.let(store::setCurrentNinjaMotion) } catch (e: Exception) { result.addProblem(Severity.Error, "Couldn't parse files.", cause = e) } - setResult(if (success) result.success(Unit) else result.failure()) + change { + ninjaGeometry?.let(store::setCurrentNinjaGeometry) + textures?.let(store::setCurrentTextures) + ninjaMotion?.let(store::setCurrentNinjaMotion) + + setResult(if (success) result.success(Unit) else result.failure()) + } } fun dismissResultDialog() { diff --git a/web/src/main/kotlin/world/phantasmal/web/viewer/stores/ViewerStore.kt b/web/src/main/kotlin/world/phantasmal/web/viewer/stores/ViewerStore.kt index 4713c9c8..67b68c72 100644 --- a/web/src/main/kotlin/world/phantasmal/web/viewer/stores/ViewerStore.kt +++ b/web/src/main/kotlin/world/phantasmal/web/viewer/stores/ViewerStore.kt @@ -14,6 +14,7 @@ import world.phantasmal.observable.cell.and import world.phantasmal.observable.cell.list.ListCell import world.phantasmal.observable.cell.list.mutableListCell import world.phantasmal.observable.cell.mutableCell +import world.phantasmal.observable.change import world.phantasmal.web.core.PwToolType import world.phantasmal.web.core.rendering.conversion.PSO_FRAME_RATE import world.phantasmal.web.core.stores.UiStore @@ -163,14 +164,16 @@ class ViewerStore( } fun setCurrentNinjaGeometry(geometry: NinjaGeometry?) { - if (_currentCharacterClass.value != null) { - _currentCharacterClass.value = null - _currentTextures.clear() - } + change { + if (_currentCharacterClass.value != null) { + _currentCharacterClass.value = null + _currentTextures.clear() + } - _currentAnimation.value = null - _currentNinjaMotion.value = null - _currentNinjaGeometry.value = geometry + _currentAnimation.value = null + _currentNinjaMotion.value = null + _currentNinjaGeometry.value = geometry + } } fun setCurrentTextures(textures: List) { @@ -200,8 +203,10 @@ class ViewerStore( } fun setCurrentNinjaMotion(njm: NjMotion) { - _currentNinjaMotion.value = njm - _animationPlaying.value = true + change { + _currentNinjaMotion.value = njm + _animationPlaying.value = true + } } suspend fun setCurrentAnimation(animation: AnimationModel?) { @@ -244,34 +249,41 @@ class ViewerStore( val char = currentCharacterClass.value ?: return - val sectionId = currentSectionId.value - val body = currentBody.value - try { + val sectionId = currentSectionId.value + val body = currentBody.value val ninjaObject = characterClassAssetLoader.loadNinjaObject(char) val textures = characterClassAssetLoader.loadXvrTextures(char, sectionId, body) - if (clearAnimation) { - _currentAnimation.value = null - _currentNinjaMotion.value = null - } + change { + if (clearAnimation) { + _currentAnimation.value = null + _currentNinjaMotion.value = null + } - _currentNinjaGeometry.value = NinjaGeometry.Object(ninjaObject) - _currentTextures.replaceAll(textures) + _currentNinjaGeometry.value = NinjaGeometry.Object(ninjaObject) + _currentTextures.replaceAll(textures) + } } catch (e: Exception) { logger.error(e) { "Couldn't load Ninja model for $char." } - _currentAnimation.value = null - _currentNinjaMotion.value = null - _currentNinjaGeometry.value = null - _currentTextures.clear() + change { + _currentAnimation.value = null + _currentNinjaMotion.value = null + _currentNinjaGeometry.value = null + _currentTextures.clear() + } } } private suspend fun loadAnimation(animation: AnimationModel) { try { - _currentNinjaMotion.value = animationAssetLoader.loadAnimation(animation.filePath) - _animationPlaying.value = true + val ninjaMotion = animationAssetLoader.loadAnimation(animation.filePath) + + change { + _currentNinjaMotion.value = ninjaMotion + _animationPlaying.value = true + } } catch (e: Exception) { logger.error(e) { "Couldn't load Ninja motion for ${animation.name} (path: ${animation.filePath})." diff --git a/webui/src/main/kotlin/world/phantasmal/webui/dom/HTMLElementSizeCell.kt b/webui/src/main/kotlin/world/phantasmal/webui/dom/HTMLElementSizeCell.kt index ed67240b..5f67ae5b 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/dom/HTMLElementSizeCell.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/dom/HTMLElementSizeCell.kt @@ -63,6 +63,10 @@ class HTMLElementSizeCell(element: HTMLElement? = null) : AbstractCell() { } } + override fun emitDependencyChanged() { + error("HTMLElementSizeCell emits dependencyChanged immediately.") + } + private fun getSize(): Size = element ?.let { Size(it.offsetWidth.toDouble(), it.offsetHeight.toDouble()) } @@ -78,7 +82,7 @@ class HTMLElementSizeCell(element: HTMLElement? = null) : AbstractCell() { if (newValue != _value) { emitMightChange() _value = newValue - emitChanged(ChangeEvent(newValue)) + emitDependencyChanged(ChangeEvent(newValue)) } } }