mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-04 06:28:28 +08:00
Moved code shared between FlatteningDependentCell and FlatteningDependentListCell to supper class AbstractFlatteningDependentCell.
This commit is contained in:
parent
9cc6c51b9c
commit
b389cb9521
@ -4,9 +4,9 @@ import world.phantasmal.core.unsafe.unsafeCast
|
|||||||
import world.phantasmal.observable.ChangeEvent
|
import world.phantasmal.observable.ChangeEvent
|
||||||
import world.phantasmal.observable.Dependent
|
import world.phantasmal.observable.Dependent
|
||||||
|
|
||||||
abstract class AbstractDependentCell<T> : AbstractCell<T>(), Dependent {
|
abstract class AbstractDependentCell<T, Event : ChangeEvent<T>> : AbstractCell<T>(), Dependent {
|
||||||
|
|
||||||
private var _value: T? = null
|
protected var _value: T? = null
|
||||||
final override val value: T
|
final override val value: T
|
||||||
get() {
|
get() {
|
||||||
computeValueAndEvent()
|
computeValueAndEvent()
|
||||||
@ -15,17 +15,12 @@ abstract class AbstractDependentCell<T> : AbstractCell<T>(), Dependent {
|
|||||||
return unsafeCast(_value)
|
return unsafeCast(_value)
|
||||||
}
|
}
|
||||||
|
|
||||||
final override var changeEvent: ChangeEvent<T>? = null
|
final override var changeEvent: Event? = null
|
||||||
get() {
|
get() {
|
||||||
computeValueAndEvent()
|
computeValueAndEvent()
|
||||||
return field
|
return field
|
||||||
}
|
}
|
||||||
private set
|
protected set
|
||||||
|
|
||||||
protected abstract fun computeValueAndEvent()
|
protected abstract fun computeValueAndEvent()
|
||||||
|
|
||||||
protected fun setValueAndEvent(value: T, changeEvent: ChangeEvent<T>?) {
|
|
||||||
_value = value
|
|
||||||
this.changeEvent = changeEvent
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -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<T, ComputedCell : Cell<T>, Event : ChangeEvent<T>>(
|
||||||
|
private val dependencies: Array<out Observable<*>>,
|
||||||
|
private val compute: () -> ComputedCell,
|
||||||
|
) : AbstractDependentCell<T, Event>() {
|
||||||
|
|
||||||
|
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()
|
||||||
|
}
|
||||||
|
}
|
@ -11,7 +11,7 @@ import world.phantasmal.observable.Observable
|
|||||||
class DependentCell<T>(
|
class DependentCell<T>(
|
||||||
private vararg val dependencies: Observable<*>,
|
private vararg val dependencies: Observable<*>,
|
||||||
private val compute: () -> T,
|
private val compute: () -> T,
|
||||||
) : AbstractDependentCell<T>() {
|
) : AbstractDependentCell<T, ChangeEvent<T>>() {
|
||||||
|
|
||||||
private var valid = false
|
private var valid = false
|
||||||
|
|
||||||
@ -21,7 +21,8 @@ class DependentCell<T>(
|
|||||||
// when they change.
|
// when they change.
|
||||||
if (!valid) {
|
if (!valid) {
|
||||||
val newValue = compute()
|
val newValue = compute()
|
||||||
setValueAndEvent(newValue, ChangeEvent(newValue))
|
_value = newValue
|
||||||
|
changeEvent = ChangeEvent(newValue)
|
||||||
valid = dependents.isNotEmpty()
|
valid = dependents.isNotEmpty()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,100 +1,16 @@
|
|||||||
package world.phantasmal.observable.cell
|
package world.phantasmal.observable.cell
|
||||||
|
|
||||||
import world.phantasmal.core.unsafe.unsafeAssertNotNull
|
|
||||||
import world.phantasmal.observable.ChangeEvent
|
import world.phantasmal.observable.ChangeEvent
|
||||||
import world.phantasmal.observable.Dependency
|
|
||||||
import world.phantasmal.observable.Dependent
|
|
||||||
import world.phantasmal.observable.Observable
|
import world.phantasmal.observable.Observable
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Similar to [DependentCell], except that this cell's [compute] returns a cell.
|
* 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<T>(
|
class FlatteningDependentCell<T>(
|
||||||
private vararg val dependencies: Observable<*>,
|
vararg dependencies: Observable<*>,
|
||||||
private val compute: () -> Cell<T>,
|
compute: () -> Cell<T>,
|
||||||
) : AbstractDependentCell<T>() {
|
) : AbstractFlatteningDependentCell<T, Cell<T>, ChangeEvent<T>>(dependencies, compute) {
|
||||||
|
|
||||||
private var computedCell: Cell<T>? = null
|
override fun createEvent(oldValue: T?, newValue: T): ChangeEvent<T> =
|
||||||
private var computedInDeps = false
|
ChangeEvent(newValue)
|
||||||
private var shouldRecomputeCell = true
|
|
||||||
private var valid = false
|
|
||||||
|
|
||||||
override fun computeValueAndEvent() {
|
|
||||||
if (!valid) {
|
|
||||||
val hasDependents = dependents.isNotEmpty()
|
|
||||||
|
|
||||||
val computedCell: Cell<T>
|
|
||||||
|
|
||||||
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()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -1,123 +1,75 @@
|
|||||||
package world.phantasmal.observable.cell.list
|
package world.phantasmal.observable.cell.list
|
||||||
|
|
||||||
|
import world.phantasmal.core.disposable.Disposable
|
||||||
import world.phantasmal.core.unsafe.unsafeAssertNotNull
|
import world.phantasmal.core.unsafe.unsafeAssertNotNull
|
||||||
import world.phantasmal.observable.Dependency
|
import world.phantasmal.observable.CallbackChangeObserver
|
||||||
import world.phantasmal.observable.Dependent
|
import world.phantasmal.observable.ChangeObserver
|
||||||
import world.phantasmal.observable.Observable
|
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<E>(
|
class FlatteningDependentListCell<E>(
|
||||||
private vararg val dependencies: Observable<*>,
|
vararg dependencies: Observable<*>,
|
||||||
private val computeElements: () -> ListCell<E>,
|
computeElements: () -> ListCell<E>,
|
||||||
) : AbstractListCell<E>(), Dependent {
|
) :
|
||||||
|
AbstractFlatteningDependentCell<List<E>, ListCell<E>, ListChangeEvent<E>>(
|
||||||
|
dependencies,
|
||||||
|
computeElements
|
||||||
|
),
|
||||||
|
ListCell<E> {
|
||||||
|
|
||||||
private var computedCell: ListCell<E>? = null
|
private var _size: Cell<Int>? = null
|
||||||
private var computedInDeps = false
|
override val size: Cell<Int>
|
||||||
private var shouldRecomputeCell = true
|
|
||||||
private var valid = false
|
|
||||||
|
|
||||||
private var _value:List<E> = emptyList()
|
|
||||||
override val value: List<E>
|
|
||||||
get() {
|
get() {
|
||||||
computeValueAndEvent()
|
if (_size == null) {
|
||||||
return _value
|
_size = DependentCell(this) { value.size }
|
||||||
|
}
|
||||||
|
|
||||||
|
return unsafeAssertNotNull(_size)
|
||||||
}
|
}
|
||||||
|
|
||||||
override var changeEvent: ListChangeEvent<E>? = null
|
private var _empty: Cell<Boolean>? = null
|
||||||
|
override val empty: Cell<Boolean>
|
||||||
get() {
|
get() {
|
||||||
computeValueAndEvent()
|
if (_empty == null) {
|
||||||
return field
|
_empty = DependentCell(this) { value.isEmpty() }
|
||||||
}
|
|
||||||
private set
|
|
||||||
|
|
||||||
private fun computeValueAndEvent() {
|
|
||||||
if (!valid) {
|
|
||||||
val oldElements = _value
|
|
||||||
val hasDependents = dependents.isNotEmpty()
|
|
||||||
|
|
||||||
val computedCell: ListCell<E>
|
|
||||||
|
|
||||||
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)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
val newElements = computedCell.value
|
return unsafeAssertNotNull(_empty)
|
||||||
_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
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
override fun addDependent(dependent: Dependent) {
|
private var _notEmpty: Cell<Boolean>? = null
|
||||||
super.addDependent(dependent)
|
override val notEmpty: Cell<Boolean>
|
||||||
|
get() {
|
||||||
if (dependents.size == 1) {
|
if (_notEmpty == null) {
|
||||||
for (dependency in dependencies) {
|
_notEmpty = DependentCell(this) { value.isNotEmpty() }
|
||||||
dependency.addDependent(this)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Called to ensure that we depend on the computed cell. This could be optimized by
|
return unsafeAssertNotNull(_notEmpty)
|
||||||
// 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 observeChange(observer: ChangeObserver<List<E>>): Disposable =
|
||||||
|
observeListChange(observer)
|
||||||
|
|
||||||
|
override fun observeListChange(observer: ListChangeObserver<E>): Disposable =
|
||||||
|
CallbackChangeObserver(this, observer)
|
||||||
|
|
||||||
|
override fun toString(): String = listCellToString(this)
|
||||||
|
|
||||||
|
override fun createEvent(oldValue: List<E>?, newValue: List<E>): ListChangeEvent<E> {
|
||||||
|
val old = oldValue ?: emptyList()
|
||||||
|
return ListChangeEvent(
|
||||||
|
newValue,
|
||||||
|
listOf(ListChange(
|
||||||
|
index = 0,
|
||||||
|
prevSize = old.size,
|
||||||
|
removed = old,
|
||||||
|
inserted = newValue,
|
||||||
|
)),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user