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 38761256..5a556f3c 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/AbstractDependentCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/AbstractDependentCell.kt @@ -4,9 +4,9 @@ import world.phantasmal.core.unsafe.unsafeCast import world.phantasmal.observable.ChangeEvent import world.phantasmal.observable.Dependent -abstract class AbstractDependentCell : AbstractCell(), Dependent { +abstract class AbstractDependentCell> : AbstractCell(), Dependent { - private var _value: T? = null + protected var _value: T? = null final override val value: T get() { computeValueAndEvent() @@ -15,17 +15,12 @@ abstract class AbstractDependentCell : AbstractCell(), Dependent { return unsafeCast(_value) } - final override var changeEvent: ChangeEvent? = null + final override var changeEvent: Event? = null get() { computeValueAndEvent() return field } - private set + protected set protected abstract fun computeValueAndEvent() - - protected fun setValueAndEvent(value: T, changeEvent: ChangeEvent?) { - _value = value - this.changeEvent = changeEvent - } } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/AbstractFlatteningDependentCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/AbstractFlatteningDependentCell.kt new file mode 100644 index 00000000..d8bf7232 --- /dev/null +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/AbstractFlatteningDependentCell.kt @@ -0,0 +1,100 @@ +package world.phantasmal.observable.cell + +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.Observable + +abstract class AbstractFlatteningDependentCell, Event : ChangeEvent>( + private val dependencies: Array>, + private val compute: () -> ComputedCell, +) : AbstractDependentCell() { + + private var computedCell: ComputedCell? = null + private var computedInDeps = false + private var shouldRecomputeCell = true + private var valid = false + + override fun computeValueAndEvent() { + if (!valid) { + val oldValue = _value + val hasDependents = dependents.isNotEmpty() + + val computedCell: ComputedCell + + if (shouldRecomputeCell) { + this.computedCell?.removeDependent(this) + computedCell = compute() + + if (hasDependents) { + // Only hold onto and depend on the computed cell if we have dependents + // ourselves. + computedCell.addDependent(this) + this.computedCell = computedCell + computedInDeps = dependencies.any { it === computedCell } + shouldRecomputeCell = false + } else { + // Set field to null to allow the cell to be garbage collected. + this.computedCell = null + } + } else { + computedCell = unsafeAssertNotNull(this.computedCell) + } + + val newValue = computedCell.value + _value = newValue + changeEvent = createEvent(oldValue, newValue) + // We stay invalid if we have no dependents to ensure our value is always recomputed. + valid = hasDependents + } + } + + protected abstract fun createEvent(oldValue: T?, newValue: T): Event + + override fun addDependent(dependent: Dependent) { + super.addDependent(dependent) + + if (dependents.size == 1) { + for (dependency in dependencies) { + dependency.addDependent(this) + } + + // Called to ensure that we depend on the computed cell. This could be optimized by + // avoiding the value and changeEvent calculation. + computeValueAndEvent() + } + } + + override fun removeDependent(dependent: Dependent) { + super.removeDependent(dependent) + + if (dependents.isEmpty()) { + valid = false + computedCell?.removeDependent(this) + // Set field to null to allow the cell to be garbage collected. + computedCell = null + shouldRecomputeCell = true + + for (dependency in dependencies) { + dependency.removeDependent(this) + } + } + } + + override fun dependencyInvalidated(dependency: Dependency<*>) { + valid = false + + // We should recompute the computed cell when any dependency except the computed cell is + // invalidated. When the computed cell is in our dependency array (i.e. the computed cell + // itself takes part in determining what the computed cell is) we should also recompute. + if (dependency !== computedCell || computedInDeps) { + // We're not allowed to change the dependency graph at this point, so we just set this + // field to true and remove ourselves as dependency from the computed cell right before + // we recompute it. + shouldRecomputeCell = true + } + + emitDependencyInvalidated() + } +} 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 c8d969f1..e525bd4d 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/DependentCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/DependentCell.kt @@ -11,7 +11,7 @@ import world.phantasmal.observable.Observable class DependentCell( private vararg val dependencies: Observable<*>, private val compute: () -> T, -) : AbstractDependentCell() { +) : AbstractDependentCell>() { private var valid = false @@ -21,7 +21,8 @@ class DependentCell( // when they change. if (!valid) { val newValue = compute() - setValueAndEvent(newValue, ChangeEvent(newValue)) + _value = newValue + changeEvent = ChangeEvent(newValue) valid = dependents.isNotEmpty() } } 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 26a589e1..37a3cd15 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/FlatteningDependentCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/FlatteningDependentCell.kt @@ -1,100 +1,16 @@ package world.phantasmal.observable.cell -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.Observable /** * Similar to [DependentCell], except that this cell's [compute] returns a cell. */ -// TODO: Shares 99% of its code with FlatteningDependentListCell, should use common super class. class FlatteningDependentCell( - private vararg val dependencies: Observable<*>, - private val compute: () -> Cell, -) : AbstractDependentCell() { + vararg dependencies: Observable<*>, + compute: () -> Cell, +) : AbstractFlatteningDependentCell, ChangeEvent>(dependencies, compute) { - private var computedCell: Cell? = null - private var computedInDeps = false - private var shouldRecomputeCell = true - private var valid = false - - override fun computeValueAndEvent() { - if (!valid) { - val hasDependents = dependents.isNotEmpty() - - val computedCell: Cell - - if (shouldRecomputeCell) { - this.computedCell?.removeDependent(this) - computedCell = compute() - - if (hasDependents) { - // Only hold onto and depend on the computed cell if we have dependents - // ourselves. - computedCell.addDependent(this) - this.computedCell = computedCell - computedInDeps = dependencies.any { it === computedCell } - shouldRecomputeCell = false - } else { - // Set field to null to allow the cell to be garbage collected. - this.computedCell = null - } - } else { - computedCell = unsafeAssertNotNull(this.computedCell) - } - - val newValue = computedCell.value - setValueAndEvent(newValue, ChangeEvent(newValue)) - // We stay invalid if we have no dependents to ensure our value is always recomputed. - valid = hasDependents - } - } - - override fun addDependent(dependent: Dependent) { - super.addDependent(dependent) - - if (dependents.size == 1) { - for (dependency in dependencies) { - dependency.addDependent(this) - } - - // Called to ensure that we depend on the computed cell. This could be optimized by - // avoiding the value and changeEvent calculation. - computeValueAndEvent() - } - } - - override fun removeDependent(dependent: Dependent) { - super.removeDependent(dependent) - - if (dependents.isEmpty()) { - valid = false - computedCell?.removeDependent(this) - // Set field to null to allow the cell to be garbage collected. - computedCell = null - shouldRecomputeCell = true - - for (dependency in dependencies) { - dependency.removeDependent(this) - } - } - } - - override fun dependencyInvalidated(dependency: Dependency<*>) { - valid = false - - // We should recompute the computed cell when any dependency except the computed cell is - // invalidated. When the computed cell is in our dependency array (i.e. the computed cell - // itself takes part in determining what the computed cell is) we should also recompute. - if (dependency !== computedCell || computedInDeps) { - // We're not allowed to change the dependency graph at this point, so we just set this - // field to true and remove ourselves as dependency from the computed cell right before - // we recompute it. - shouldRecomputeCell = true - } - - emitDependencyInvalidated() - } + override fun createEvent(oldValue: T?, newValue: T): ChangeEvent = + ChangeEvent(newValue) } 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 177df4f3..ccd59a24 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 @@ -1,123 +1,75 @@ package world.phantasmal.observable.cell.list +import world.phantasmal.core.disposable.Disposable import world.phantasmal.core.unsafe.unsafeAssertNotNull -import world.phantasmal.observable.Dependency -import world.phantasmal.observable.Dependent +import world.phantasmal.observable.CallbackChangeObserver +import world.phantasmal.observable.ChangeObserver import world.phantasmal.observable.Observable +import world.phantasmal.observable.cell.AbstractFlatteningDependentCell +import world.phantasmal.observable.cell.Cell +import world.phantasmal.observable.cell.DependentCell /** - * Similar to [DependentListCell], except that this cell's [computeElements] returns a [ListCell]. + * Similar to [DependentListCell], except that this cell's computeElements returns a [ListCell]. */ -// TODO: Shares 99% of its code with FlatteningDependentCell, should use common super class. class FlatteningDependentListCell( - private vararg val dependencies: Observable<*>, - private val computeElements: () -> ListCell, -) : AbstractListCell(), Dependent { + vararg dependencies: Observable<*>, + computeElements: () -> ListCell, +) : + AbstractFlatteningDependentCell, ListCell, ListChangeEvent>( + dependencies, + computeElements + ), + ListCell { - private var computedCell: ListCell? = null - private var computedInDeps = false - private var shouldRecomputeCell = true - private var valid = false - - private var _value:List = emptyList() - override val value: List + private var _size: Cell? = null + override val size: Cell get() { - computeValueAndEvent() - return _value + if (_size == null) { + _size = DependentCell(this) { value.size } + } + + return unsafeAssertNotNull(_size) } - override var changeEvent: ListChangeEvent? = null + private var _empty: Cell? = null + override val empty: Cell get() { - computeValueAndEvent() - return field - } - private set - - private fun computeValueAndEvent() { - if (!valid) { - val oldElements = _value - val hasDependents = dependents.isNotEmpty() - - val computedCell: ListCell - - if (shouldRecomputeCell) { - this.computedCell?.removeDependent(this) - computedCell = computeElements() - - if (hasDependents) { - // Only hold onto and depend on the computed cell if we have dependents - // ourselves. - computedCell.addDependent(this) - this.computedCell = computedCell - computedInDeps = dependencies.any { it === computedCell } - shouldRecomputeCell = false - } else { - // Set field to null to allow the cell to be garbage collected. - this.computedCell = null - } - } else { - computedCell = unsafeAssertNotNull(this.computedCell) + if (_empty == null) { + _empty = DependentCell(this) { value.isEmpty() } } - val newElements = computedCell.value - _value = newElements - changeEvent = ListChangeEvent( - newElements, - listOf(ListChange( - index = 0, - prevSize = oldElements.size, - removed = oldElements, - inserted = newElements, - )), - ) - // We stay invalid if we have no dependents to ensure our value is always recomputed. - valid = hasDependents + return unsafeAssertNotNull(_empty) } - } - override fun addDependent(dependent: Dependent) { - super.addDependent(dependent) - - if (dependents.size == 1) { - for (dependency in dependencies) { - dependency.addDependent(this) + private var _notEmpty: Cell? = null + override val notEmpty: Cell + get() { + if (_notEmpty == null) { + _notEmpty = DependentCell(this) { value.isNotEmpty() } } - // Called to ensure that we depend on the computed cell. This could be optimized by - // avoiding the value and changeEvent calculation. - computeValueAndEvent() - } - } - - override fun removeDependent(dependent: Dependent) { - super.removeDependent(dependent) - - if (dependents.isEmpty()) { - valid = false - computedCell?.removeDependent(this) - // Set field to null to allow the cell to be garbage collected. - computedCell = null - shouldRecomputeCell = true - - for (dependency in dependencies) { - dependency.removeDependent(this) - } - } - } - - override fun dependencyInvalidated(dependency: Dependency<*>) { - valid = false - - // We should recompute the computed cell when any dependency except the computed cell is - // invalidated. When the computed cell is in our dependency array (i.e. the computed cell - // itself takes part in determining what the computed cell is) we should also recompute. - if (dependency !== computedCell || computedInDeps) { - // We're not allowed to change the dependency graph at this point, so we just set this - // field to true and remove ourselves as dependency from the computed cell right before - // we recompute it. - shouldRecomputeCell = true + return unsafeAssertNotNull(_notEmpty) } - emitDependencyInvalidated() + override fun observeChange(observer: ChangeObserver>): Disposable = + observeListChange(observer) + + override fun observeListChange(observer: ListChangeObserver): Disposable = + CallbackChangeObserver(this, observer) + + override fun toString(): String = listCellToString(this) + + override fun createEvent(oldValue: List?, newValue: List): ListChangeEvent { + val old = oldValue ?: emptyList() + return ListChangeEvent( + newValue, + listOf(ListChange( + index = 0, + prevSize = old.size, + removed = old, + inserted = newValue, + )), + ) } }