diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/Observer.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/Observer.kt index 12b3fa8c..22cf76a4 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/Observer.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/Observer.kt @@ -2,7 +2,7 @@ package world.phantasmal.observable open class ChangeEvent( /** - * The observable's new value + * The observable's new value. */ val value: T, ) { 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 9dff2b33..0fc91615 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 @@ -1,6 +1,7 @@ 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.cell.AbstractCell @@ -9,6 +10,28 @@ import world.phantasmal.observable.cell.DependentCell import world.phantasmal.observable.cell.not abstract class AbstractListCell : AbstractCell>(), ListCell { + /** + * When [value] is accessed and this property is null, a new wrapper is created that points to + * [elements]. Before changes to [elements] are made, if there's a wrapper, the current + * wrapper's backing list is set to a copy of [elements] and this property is set to null. This + * way, accessing [value] acts like accessing a snapshot without making an actual copy + * everytime. This is necessary because the contract is that a cell's new value is always != to + * 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. + private var _elementsWrapper: DelegatingList? = null + protected val elementsWrapper: DelegatingList + get() { + if (_elementsWrapper == null) { + _elementsWrapper = DelegatingList(elements) + } + + return unsafeAssertNotNull(_elementsWrapper) + } + + protected abstract val elements: List + @Suppress("LeakingThis") final override val size: Cell = DependentCell(this) { value.size } @@ -35,4 +58,9 @@ abstract class AbstractListCell : AbstractCell>(), ListCell { return observingCell } + + protected fun copyAndResetWrapper() { + _elementsWrapper?.backingList = elements.toList() + _elementsWrapper = null + } } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/DelegatingList.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/DelegatingList.kt new file mode 100644 index 00000000..2f99899f --- /dev/null +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/DelegatingList.kt @@ -0,0 +1,36 @@ +package world.phantasmal.observable.cell.list + +/** + * Simply delegates all methods to [backingList], even [equals], [hashCode] and [toString]. + */ +class DelegatingList(var backingList: List) : List { + override val size: Int = backingList.size + + override fun contains(element: E): Boolean = backingList.contains(element) + + override fun containsAll(elements: Collection): Boolean = backingList.containsAll(elements) + + override fun get(index: Int): E = backingList[index] + + override fun indexOf(element: E): Int = backingList.indexOf(element) + + override fun isEmpty(): Boolean = backingList.isEmpty() + + override fun iterator(): Iterator = backingList.iterator() + + override fun lastIndexOf(element: E): Int = backingList.lastIndexOf(element) + + override fun listIterator(): ListIterator = backingList.listIterator() + + override fun listIterator(index: Int): ListIterator = backingList.listIterator(index) + + override fun subList(fromIndex: Int, toIndex: Int): List = + backingList.subList(fromIndex, toIndex) + + @Suppress("SuspiciousEqualsCombination") + override fun equals(other: Any?): Boolean = this === other || other == backingList + + override fun hashCode(): Int = backingList.hashCode() + + override fun toString(): String = backingList.toString() +} 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 10a9ef26..baeb6828 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 @@ -1,5 +1,6 @@ package world.phantasmal.observable.cell.list +import world.phantasmal.core.unsafe.unsafeAssertNotNull import world.phantasmal.observable.ChangeEvent import world.phantasmal.observable.Dependency import world.phantasmal.observable.Dependent @@ -14,7 +15,7 @@ class FilteredListCell( */ private val indexMap = mutableListOf() - private val elements = mutableListOf() + override val elements = mutableListOf() override val value: List get() { @@ -22,7 +23,7 @@ class FilteredListCell( recompute() } - return elements + return elementsWrapper } override fun addDependent(dependent: Dependent) { @@ -125,6 +126,7 @@ class FilteredListCell( } } + copyAndResetWrapper() elements.add(insertIndex, change.updated) indexMap[change.index] = insertIndex @@ -151,6 +153,7 @@ class FilteredListCell( if (index != -1) { // If the element now doesn't pass the test and it previously did // pass, remove it and emit a structural change. + copyAndResetWrapper() elements.removeAt(index) indexMap[change.index] = -1 @@ -181,7 +184,7 @@ class FilteredListCell( if (filteredChanges.isEmpty()) { emitDependencyChanged(null) } else { - emitDependencyChanged(ListChangeEvent(elements, filteredChanges)) + emitDependencyChanged(ListChangeEvent(elementsWrapper, filteredChanges)) } } else { emitDependencyChanged(null) @@ -195,6 +198,7 @@ class FilteredListCell( } private fun recompute() { + copyAndResetWrapper() elements.clear() indexMap.clear() diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/FlatteningDependentListCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/FlatteningDependentListCell.kt index 54758186..74df3ba9 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/FlatteningDependentListCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/FlatteningDependentListCell.kt @@ -29,7 +29,7 @@ class FlatteningDependentListCell( computedCell = computeElements.invoke().also { computedCell -> computedCell.addDependent(this) computedInDeps = dependencies.any { it === computedCell } - elements = computedCell.value.toList() + elements = computedCell.value } } @@ -70,6 +70,6 @@ class FlatteningDependentListCell( shouldRecompute = false } - elements = unsafeAssertNotNull(computedCell).value.toList() + elements = unsafeAssertNotNull(computedCell).value } } 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 d61622e4..f7697e96 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 @@ -17,7 +17,7 @@ typealias DependenciesExtractor = (element: E) -> Array * event. */ class SimpleListCell( - private val elements: MutableList, + override val elements: MutableList, private val extractDependencies: DependenciesExtractor? = null, ) : AbstractListCell(), MutableListCell { @@ -30,7 +30,7 @@ class SimpleListCell( private var changes = mutableListOf>() override var value: List - get() = elements + get() = elementsWrapper set(value) { replaceAll(value) } @@ -42,6 +42,7 @@ class SimpleListCell( checkIndex(index, elements.lastIndex) emitMightChange() + copyAndResetWrapper() val removed = elements.set(index, element) if (dependents.isNotEmpty() && extractDependencies != null) { @@ -59,6 +60,7 @@ class SimpleListCell( emitMightChange() val index = elements.size + copyAndResetWrapper() elements.add(element) finalizeStructuralChange(index, emptyList(), listOf(element)) @@ -68,6 +70,7 @@ class SimpleListCell( checkIndex(index, elements.size) emitMightChange() + copyAndResetWrapper() elements.add(index, element) finalizeStructuralChange(index, emptyList(), listOf(element)) @@ -88,6 +91,7 @@ class SimpleListCell( checkIndex(index, elements.lastIndex) emitMightChange() + copyAndResetWrapper() val removed = elements.removeAt(index) finalizeStructuralChange(index, listOf(removed), emptyList()) @@ -97,19 +101,21 @@ class SimpleListCell( override fun replaceAll(elements: Iterable) { emitMightChange() - val removed = ArrayList(this.elements) + val removed = elementsWrapper + copyAndResetWrapper() this.elements.replaceAll(elements) - finalizeStructuralChange(0, removed, ArrayList(this.elements)) + finalizeStructuralChange(0, removed, elementsWrapper) } override fun replaceAll(elements: Sequence) { emitMightChange() - val removed = ArrayList(this.elements) + val removed = elementsWrapper + copyAndResetWrapper() this.elements.replaceAll(elements) - finalizeStructuralChange(0, removed, ArrayList(this.elements)) + finalizeStructuralChange(0, removed, elementsWrapper) } override fun splice(fromIndex: Int, removeCount: Int, newElement: E) { @@ -121,6 +127,7 @@ class SimpleListCell( emitMightChange() + copyAndResetWrapper() repeat(removeCount) { elements.removeAt(fromIndex) } elements.add(fromIndex, newElement) @@ -130,7 +137,8 @@ class SimpleListCell( override fun clear() { emitMightChange() - val removed = ArrayList(this.elements) + val removed = elementsWrapper + copyAndResetWrapper() elements.clear() finalizeStructuralChange(0, removed, emptyList()) @@ -139,7 +147,8 @@ class SimpleListCell( override fun sortWith(comparator: Comparator) { emitMightChange() - val removed = ArrayList(elements) + val removed = elementsWrapper + copyAndResetWrapper() var throwable: Throwable? = null try { @@ -148,7 +157,7 @@ class SimpleListCell( throwable = e } - finalizeStructuralChange(0, removed, ArrayList(elements)) + finalizeStructuralChange(0, removed, elementsWrapper) if (throwable != null) { throw throwable @@ -180,7 +189,7 @@ class SimpleListCell( override fun emitDependencyChanged() { val currentChanges = changes changes = mutableListOf() - emitDependencyChanged(ListChangeEvent(elements, currentChanges)) + emitDependencyChanged(ListChangeEvent(elementsWrapper, currentChanges)) } private fun checkIndex(index: Int, maxIndex: Int) { 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 eebf87f8..2d9825fa 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/CellTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/CellTests.kt @@ -122,6 +122,32 @@ interface CellTests : ObservableTests { } } + /** + * [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 + * queried. + */ + @Test + fun reflects_changes_without_observers() = test { + val p = createProvider() + + var old: Any? + + repeat(5) { + // Value should change after emit. + old = p.observable.value + + p.emit() + + val new = p.observable.value + + assertNotEquals(old, new) + + // Value should not change when emit hasn't been called since the last access. + assertEquals(new, p.observable.value) + } + } + interface Provider : ObservableTests.Provider { override val observable: Cell } diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/DependentCellWithSimpleListCellTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/DependentCellWithSimpleListCellTests.kt new file mode 100644 index 00000000..f6b285d0 --- /dev/null +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/DependentCellWithSimpleListCellTests.kt @@ -0,0 +1,17 @@ +package world.phantasmal.observable.cell + +import world.phantasmal.observable.cell.list.SimpleListCell + +class DependentCellWithSimpleListCellTests : CellTests { + override fun createProvider() = Provider() + + class Provider : CellTests.Provider { + private val dependencyCell = SimpleListCell(mutableListOf("a", "b", "c")) + + override val observable = DependentCell(dependencyCell) { dependencyCell.value } + + override fun emit() { + dependencyCell.add("x") + } + } +} diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/FlatteningDependentCellWithSimpleListCellTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/FlatteningDependentCellWithSimpleListCellTests.kt new file mode 100644 index 00000000..42aff945 --- /dev/null +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/FlatteningDependentCellWithSimpleListCellTests.kt @@ -0,0 +1,17 @@ +package world.phantasmal.observable.cell + +import world.phantasmal.observable.cell.list.SimpleListCell + +class FlatteningDependentCellWithSimpleListCellTests : CellTests { + override fun createProvider() = Provider() + + class Provider : CellTests.Provider { + private val dependencyCell = SimpleListCell(mutableListOf("a", "b", "c")) + + override val observable = FlatteningDependentCell(dependencyCell) { dependencyCell } + + override fun emit() { + dependencyCell.add("x") + } + } +} 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 72174e49..6ac72ee7 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/MutableCellTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/MutableCellTests.kt @@ -1,9 +1,6 @@ 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 diff --git a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/RegularCellTests.kt b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/RegularCellTests.kt index 68fce6cb..c77a9c3c 100644 --- a/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/RegularCellTests.kt +++ b/observable/src/commonTest/kotlin/world/phantasmal/observable/cell/RegularCellTests.kt @@ -9,32 +9,6 @@ import kotlin.test.* interface RegularCellTests : CellTests { fun createWithValue(value: T): Cell - /** - * [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 - * queried. - */ - @Test - fun reflects_changes_without_observers() = test { - val p = createProvider() - - var old: Any? - - repeat(5) { - // Value should change after emit. - old = p.observable.value - - p.emit() - - val new = p.observable.value - - assertNotEquals(old, new) - - // Value should not change when emit hasn't been called since the last access. - assertEquals(new, p.observable.value) - } - } - @Test fun convenience_methods() = test { listOf(Any(), null).forEach { any ->