Filtered list cells now support changing multiple times in a single mutation.

This commit is contained in:
Daan Vanden Bosch 2022-10-31 20:59:13 +01:00
parent c3be91a05e
commit f5b1008c48
11 changed files with 285 additions and 155 deletions

View File

@ -3,7 +3,10 @@ package world.phantasmal.cell
typealias ChangeObserver<T> = (ChangeEvent<T>) -> Unit typealias ChangeObserver<T> = (ChangeEvent<T>) -> Unit
open class ChangeEvent<out T>( open class ChangeEvent<out T>(
/** The cell's new value. */ /**
* The cell's new value. Don't keep long-lived references to this object, it may change after
* change observers have been called.
*/
val value: T, val value: T,
) { ) {
operator fun component1() = value operator fun component1() = value

View File

@ -4,18 +4,27 @@ import world.phantasmal.core.assert
import kotlin.contracts.InvocationKind.EXACTLY_ONCE import kotlin.contracts.InvocationKind.EXACTLY_ONCE
import kotlin.contracts.contract import kotlin.contracts.contract
// TODO: Throw exception by default when triggering early recomputation during change set. Allow to
// to turn this check off, because partial, early recomputation might be useful in rare cases.
// Dependents will need to partially apply ListChangeEvents etc. and remember which part of
// the event they've already applied (i.e. an index into the changes list).
object MutationManager { object MutationManager {
private val invalidatedLeaves = HashSet<LeafDependent>() private val invalidatedLeaves = HashSet<LeafDependent>()
/** Non-zero when a mutation is active. */ /** Non-zero when a mutation is active. */
private var mutationNestingLevel = 0 private var mutationNestingLevel: Int = 0
/**
* ID of the current outermost mutation. Meaningless when not in a mutation. Nested mutations
* don't have IDs at the moment.
*/
var currentMutationId: Long = -1
private set
/**
* Set to true when a mutation was automatically started by changing a cell without first
* manually starting a mutation.
*/
private var artificialMutation = false
/** Whether a dependency's value is changing at the moment. */
private var dependencyChanging = false private var dependencyChanging = false
private var pulling = false
private val deferredMutations: MutableList<() -> Unit> = mutableListOf() private val deferredMutations: MutableList<() -> Unit> = mutableListOf()
private var applyingDeferredMutations = false private var applyingDeferredMutations = false
@ -35,7 +44,7 @@ object MutationManager {
} }
fun mutateDeferred(block: () -> Unit) { fun mutateDeferred(block: () -> Unit) {
if (dependencyChanging || mutationNestingLevel > 0) { if (mutationNestingLevel > 0) {
deferredMutations.add(block) deferredMutations.add(block)
} else { } else {
block() block()
@ -43,23 +52,38 @@ object MutationManager {
} }
fun mutationStart() { fun mutationStart() {
assert(!dependencyChanging) { "Can't start a mutation while a dependency is changing." }
assert(!pulling) { "Can't start a mutation while pulling." }
if (mutationNestingLevel == 0) {
currentMutationId++
}
mutationNestingLevel++ mutationNestingLevel++
} }
fun mutationEnd() { fun mutationEnd() {
assert(mutationNestingLevel > 0) { "No mutation was started." } assert(mutationNestingLevel > 0) { "No mutation was started." }
mutationNestingLevel-- if (mutationNestingLevel == 1) {
assert(!pulling) { "Already pulling." }
if (mutationNestingLevel == 0) {
try { try {
pulling = true
for (dependent in invalidatedLeaves) { for (dependent in invalidatedLeaves) {
dependent.pull() dependent.pull()
} }
} finally { } finally {
dependencyChanging = false
pulling = false
mutationNestingLevel--
invalidatedLeaves.clear() invalidatedLeaves.clear()
applyDeferredMutations() applyDeferredMutations()
} }
} else {
dependencyChanging = false
mutationNestingLevel--
} }
} }
@ -79,23 +103,20 @@ object MutationManager {
fun dependencyChangeStart() { fun dependencyChangeStart() {
check(!dependencyChanging) { "A cell is already changing." } check(!dependencyChanging) { "A cell is already changing." }
assert(!pulling) { "Can't change a cell while pulling." }
if (mutationNestingLevel == 0) {
mutationStart()
artificialMutation = true
}
dependencyChanging = true dependencyChanging = true
} }
fun dependencyChangeEnd() { fun dependencyChangeEnd() {
assert(dependencyChanging) { "No cell was changing." } if (artificialMutation) {
artificialMutation = false
if (mutationNestingLevel == 0) { mutationEnd()
try {
for (dependent in invalidatedLeaves) {
dependent.pull()
}
} finally {
dependencyChanging = false
invalidatedLeaves.clear()
applyDeferredMutations()
}
} else { } else {
dependencyChanging = false dependencyChanging = false
} }
@ -114,9 +135,7 @@ object MutationManager {
var idx = 0 var idx = 0
while (idx < deferredMutations.size) { while (idx < deferredMutations.size) {
mutate { mutate(deferredMutations[idx])
deferredMutations[idx]()
}
idx++ idx++
} }

View File

@ -2,6 +2,8 @@ package world.phantasmal.cell.list
import world.phantasmal.cell.Dependency import world.phantasmal.cell.Dependency
import world.phantasmal.cell.Dependent import world.phantasmal.cell.Dependent
import world.phantasmal.cell.MutationManager
import world.phantasmal.core.unsafe.unsafeCast
abstract class AbstractFilteredListCell<E>( abstract class AbstractFilteredListCell<E>(
protected val list: ListCell<E>, protected val list: ListCell<E>,
@ -10,6 +12,10 @@ abstract class AbstractFilteredListCell<E>(
/** Set during a change wave when [list] changes. */ /** Set during a change wave when [list] changes. */
private var listInvalidated = false private var listInvalidated = false
/** Mutation ID during which the current list of changes was created. */
private var changesMutationId: Long = -1
private var listChangeIndex = 0
/** Set during a change wave when [predicateDependency] changes. */ /** Set during a change wave when [predicateDependency] changes. */
private var predicateInvalidated = false private var predicateInvalidated = false
@ -25,12 +31,12 @@ abstract class AbstractFilteredListCell<E>(
return elementsWrapper return elementsWrapper
} }
final override var changeEvent: ListChangeEvent<E>? = null private var _changeEvent: ListChangeEvent<E>? = null
final override val changeEvent: ListChangeEvent<E>?
get() { get() {
computeValueAndEvent() computeValueAndEvent()
return field return _changeEvent
} }
private set
private fun computeValueAndEvent() { private fun computeValueAndEvent() {
if (!valid) { if (!valid) {
@ -43,22 +49,34 @@ abstract class AbstractFilteredListCell<E>(
ignoreOtherChanges() ignoreOtherChanges()
recompute() recompute()
changeEvent = ListChangeEvent( _changeEvent = ListChangeEvent(
elementsWrapper, elementsWrapper,
listOf(ListChange(0, removed.size, removed, elementsWrapper)), listOf(ListChange(0, removed.size, removed, elementsWrapper)),
) )
} else { } else {
// TODO: Conditionally copyAndResetWrapper? // TODO: Conditionally copyAndResetWrapper?
copyAndResetWrapper() copyAndResetWrapper()
val filteredChanges = mutableListOf<ListChange<E>>()
if (listInvalidated) { // Reuse the same list of changes during a mutation.
list.changeEvent?.let { listChangeEvent -> val event = _changeEvent
for (change in listChangeEvent.changes) { val filteredChanges: MutableList<ListChange<E>> =
if (event == null || changesMutationId != MutationManager.currentMutationId) {
changesMutationId = MutationManager.currentMutationId
listChangeIndex = 0
mutableListOf()
} else {
// This cast is safe because we know we always instantiate our change event with a mutable list.
unsafeCast(event.changes)
}
val listChangeEvent = list.changeEvent
if (listInvalidated && listChangeEvent != null) {
for (change in listChangeEvent.changes.listIterator(listChangeIndex)) {
val prevSize = elements.size val prevSize = elements.size
// Map the incoming change index to an index into our own elements list. // Map the incoming change index to an index into our own elements list.
// TODO: Avoid this loop by storing the index where an element "would" // TODO: Avoid this loop by storing the index where an element "would"
// be if it passed the predicate. // be if it passed the predicate?
var eventIndex = prevSize var eventIndex = prevSize
for (index in change.index..maxDepIndex()) { for (index in change.index..maxDepIndex()) {
@ -124,12 +142,13 @@ abstract class AbstractFilteredListCell<E>(
) )
} }
} }
}
listChangeIndex = listChangeEvent.changes.size
} }
processOtherChanges(filteredChanges) processOtherChanges(filteredChanges)
changeEvent = _changeEvent =
if (filteredChanges.isEmpty()) { if (filteredChanges.isEmpty()) {
null null
} else { } else {

View File

@ -36,8 +36,8 @@ class ListElementsDependentCell<E>(
private fun updateElementDependenciesAndEvent() { private fun updateElementDependenciesAndEvent() {
if (!valid) { if (!valid) {
if (listInvalidated) { if (listInvalidated) {
// At this point we can remove this dependent from the removed elements' dependencies // At this point we can remove this dependent from the removed elements'
// and add it to the newly inserted elements' dependencies. // dependencies and add it to the newly inserted elements' dependencies.
list.changeEvent?.let { listChangeEvent -> list.changeEvent?.let { listChangeEvent ->
for (change in listChangeEvent.changes) { for (change in listChangeEvent.changes) {
for (i in change.index until (change.index + change.removed.size)) { for (i in change.index until (change.index + change.removed.size)) {

View File

@ -1,11 +1,12 @@
package world.phantasmal.cell.list package world.phantasmal.cell.list
import world.phantasmal.cell.MutationManager
import world.phantasmal.core.replaceAll import world.phantasmal.core.replaceAll
import world.phantasmal.core.unsafe.unsafeCast
/** /**
* @param elements The backing list for this [ListCell]. * @param elements The backing list for this [ListCell].
*/ */
// TODO: Support change sets by sometimes appending to changeEvent instead of always overwriting it.
class SimpleListCell<E>( class SimpleListCell<E>(
override val elements: MutableList<E>, override val elements: MutableList<E>,
) : AbstractElementsWrappingListCell<E>(), MutableListCell<E> { ) : AbstractElementsWrappingListCell<E>(), MutableListCell<E> {
@ -19,6 +20,9 @@ class SimpleListCell<E>(
override var changeEvent: ListChangeEvent<E>? = null override var changeEvent: ListChangeEvent<E>? = null
private set private set
/** Mutation ID during which the current list of changes was created. */
private var changesMutationId: Long = -1
override operator fun get(index: Int): E = override operator fun get(index: Int): E =
elements[index] elements[index]
@ -135,6 +139,10 @@ class SimpleListCell<E>(
} }
override fun clear() { override fun clear() {
if (elements.isEmpty()) {
return
}
applyChange { applyChange {
val prevSize = elements.size val prevSize = elements.size
val removed = elementsWrapper val removed = elementsWrapper
@ -185,9 +193,19 @@ class SimpleListCell<E>(
removed: List<E>, removed: List<E>,
inserted: List<E>, inserted: List<E>,
) { ) {
changeEvent = ListChangeEvent( val event = changeEvent
elementsWrapper,
listOf(ListChange(index, prevSize, removed, inserted)), // Reuse the same list of changes during a mutation.
) val changes: MutableList<ListChange<E>> =
if (event == null || changesMutationId != MutationManager.currentMutationId) {
changesMutationId = MutationManager.currentMutationId
mutableListOf()
} else {
// This cast is safe because we know we always instantiate our change event with a mutable list.
unsafeCast(event.changes)
}
changes.add(ListChange(index, prevSize, removed, inserted))
changeEvent = ListChangeEvent(elementsWrapper, changes)
} }
} }

View File

@ -1,8 +1,6 @@
package world.phantasmal.cell package world.phantasmal.cell
import world.phantasmal.cell.test.CellTestSuite import world.phantasmal.cell.test.CellTestSuite
import world.phantasmal.cell.test.Snapshot
import world.phantasmal.cell.test.snapshot
import world.phantasmal.core.disposable.use import world.phantasmal.core.disposable.use
import kotlin.test.Test import kotlin.test.Test
import kotlin.test.assertEquals import kotlin.test.assertEquals
@ -245,11 +243,15 @@ interface CellTests : CellTestSuite {
} }
) )
// Repeat 3 times to check if temporary state is always reset.
repeat(3) {
changes = 0
mutate { mutate {
repeat(5) { repeat(5) {
p.emit() p.emit()
// Change should be deferred until this lambda returns. // Change should be deferred until this mutation finishes.
assertEquals(0, changes) assertEquals(0, changes)
} }
} }
@ -257,21 +259,26 @@ interface CellTests : CellTestSuite {
// All changes to the same cell should be collapsed to a single change. // All changes to the same cell should be collapsed to a single change.
assertEquals(1, changes) assertEquals(1, changes)
} }
}
@Test @Test
fun value_can_be_accessed_during_a_mutation() = test { fun value_can_be_accessed_during_a_mutation() = test {
val p = createProvider() val p = createProvider()
// Change will be observed exactly once.
var observedValue: Snapshot? = null var observedValue: Snapshot? = null
disposer.add( disposer.add(
p.cell.observeChange { p.cell.observeChange {
// Change will be observed exactly once every loop iteration.
assertNull(observedValue) assertNull(observedValue)
observedValue = it.value.snapshot() observedValue = it.value.snapshot()
} }
) )
// Repeat 3 times to check that temporary state is always reset.
repeat(3) {
observedValue = null
val v1 = p.cell.value.snapshot() val v1 = p.cell.value.snapshot()
var v3: Snapshot? = null var v3: Snapshot? = null
@ -286,6 +293,8 @@ interface CellTests : CellTestSuite {
assertNotEquals(v2, v3) assertNotEquals(v2, v3)
p.emit() p.emit()
assertNull(observedValue)
} }
val v4 = p.cell.value.snapshot() val v4 = p.cell.value.snapshot()
@ -294,15 +303,16 @@ interface CellTests : CellTestSuite {
assertNotEquals(v3, v4) assertNotEquals(v3, v4)
assertEquals(v4, observedValue) assertEquals(v4, observedValue)
} }
}
@Test @Test
fun mutations_can_be_nested() = test { fun mutations_can_be_nested() = test {
// 3 Cells. // 3 Cells.
val ps = Array(3) { createProvider() } val ps = Array(3) { createProvider() }
val observedChanges = IntArray(3) val observedChanges = IntArray(ps.size)
// Observe each cell. // Observe each cell.
repeat(3) { idx -> for (idx in ps.indices) {
disposer.add( disposer.add(
ps[idx].cell.observeChange { ps[idx].cell.observeChange {
assertEquals(0, observedChanges[idx]) assertEquals(0, observedChanges[idx])
@ -342,3 +352,16 @@ interface CellTests : CellTestSuite {
fun emit() fun emit()
} }
} }
/** See [snapshot]. */
typealias Snapshot = String
/**
* We use toString to create "snapshots" of values throughout the tests. Most of the time cells will
* actually have a new value after emitting a change event, but this is not always the case with
* more complex cells or cells that point to complex values. So instead of keeping references to
* values and comparing them with == (or using e.g. assertEquals), we compare snapshots.
*
* This of course assumes that all values have sensible toString implementations.
*/
fun Any?.snapshot(): Snapshot = toString()

View File

@ -4,12 +4,16 @@ import world.phantasmal.cell.Cell
import world.phantasmal.cell.ImmutableCell import world.phantasmal.cell.ImmutableCell
import world.phantasmal.cell.SimpleCell import world.phantasmal.cell.SimpleCell
import world.phantasmal.cell.test.CellTestSuite import world.phantasmal.cell.test.CellTestSuite
import kotlin.test.* import kotlin.test.Test
import kotlin.test.assertEquals
import kotlin.test.assertNotNull
import kotlin.test.assertNull
import kotlin.test.assertTrue
/** /**
* Tests that apply to all filtered list implementations. * Tests that apply to all filtered list implementations.
*/ */
interface SuperFilteredListCellTests : CellTestSuite { interface AbstractFilteredListCellTests : CellTestSuite {
fun <E> createFilteredListCell(list: ListCell<E>, predicate: Cell<(E) -> Boolean>): ListCell<E> fun <E> createFilteredListCell(list: ListCell<E>, predicate: Cell<(E) -> Boolean>): ListCell<E>
@Test @Test

View File

@ -8,7 +8,7 @@ import world.phantasmal.cell.map
// TODO: A test suite that tests FilteredListCell while the predicate results are changing. // TODO: A test suite that tests FilteredListCell while the predicate results are changing.
// TODO: A test suite that tests FilteredListCell while all 3 types of dependencies are changing. // TODO: A test suite that tests FilteredListCell while all 3 types of dependencies are changing.
@Suppress("unused") @Suppress("unused")
class FilteredListCellTests : SuperFilteredListCellTests { class FilteredListCellTests : AbstractFilteredListCellTests {
override fun <E> createFilteredListCell(list: ListCell<E>, predicate: Cell<(E) -> Boolean>) = override fun <E> createFilteredListCell(list: ListCell<E>, predicate: Cell<(E) -> Boolean>) =
FilteredListCell(list, predicate.map { p -> { cell(p(it)) } }) FilteredListCell(list, predicate.map { p -> { cell(p(it)) } })
} }

View File

@ -1,7 +1,14 @@
package world.phantasmal.cell.list package world.phantasmal.cell.list
import world.phantasmal.cell.CellTests import world.phantasmal.cell.CellTests
import kotlin.test.* import world.phantasmal.cell.mutate
import kotlin.test.Test
import kotlin.test.assertEquals
import kotlin.test.assertFailsWith
import kotlin.test.assertNotEquals
import kotlin.test.assertNotNull
import kotlin.test.assertNull
import kotlin.test.assertTrue
/** /**
* Test suite for all [ListCell] implementations. There is a subclass of this suite for every * Test suite for all [ListCell] implementations. There is a subclass of this suite for every
@ -216,6 +223,56 @@ interface ListCellTests : CellTests {
} }
} }
/**
* The same as [value_can_be_accessed_during_a_mutation], except that it also checks expected
* list sizes.
*/
@Test
fun list_cell_value_can_be_accessed_during_a_mutation() = test {
val p = createProvider()
var observedValue: List<Any>? = null
disposer.add(
p.cell.observeChange {
// Change will be observed exactly once every loop iteration.
assertNull(observedValue)
observedValue = it.value.toList()
}
)
// Repeat 3 times to check that temporary state is always reset.
repeat(3) {
observedValue = null
val v1 = p.cell.value.toList()
var v3: List<Any>? = null
mutate {
val v2 = p.cell.value.toList()
assertEquals(v1, v2)
p.addElement()
v3 = p.cell.value.toList()
assertNotEquals(v2, v3)
assertEquals(v2.size + 1, v3!!.size)
p.addElement()
assertNull(observedValue)
}
val v4 = p.cell.value.toList()
assertNotNull(v3)
assertNotEquals(v3, v4)
assertEquals(v3!!.size + 1, v4.size)
assertEquals(v4, observedValue)
}
}
interface Provider : CellTests.Provider { interface Provider : CellTests.Provider {
override val cell: ListCell<Any> override val cell: ListCell<Any>

View File

@ -9,7 +9,7 @@ import world.phantasmal.cell.Cell
* [SimpleFilteredListCellPredicateDependencyEmitsTests]. * [SimpleFilteredListCellPredicateDependencyEmitsTests].
*/ */
@Suppress("unused") @Suppress("unused")
class SimpleFilteredListCellTests : SuperFilteredListCellTests { class SimpleFilteredListCellTests : AbstractFilteredListCellTests {
override fun <E> createFilteredListCell(list: ListCell<E>, predicate: Cell<(E) -> Boolean>) = override fun <E> createFilteredListCell(list: ListCell<E>, predicate: Cell<(E) -> Boolean>) =
SimpleFilteredListCell(list, predicate) SimpleFilteredListCell(list, predicate)
} }

View File

@ -7,16 +7,3 @@ fun <E> assertListCellEquals(expected: List<E>, actual: ListCell<E>) {
assertEquals(expected.size, actual.size.value) assertEquals(expected.size, actual.size.value)
assertEquals(expected, actual.value) assertEquals(expected, actual.value)
} }
/** See [snapshot]. */
typealias Snapshot = String
/**
* We use toString to create "snapshots" of values throughout the tests. Most of the time cells will
* actually have a new value after emitting a change event, but this is not always the case with
* more complex cells or cells that point to complex values. So instead of keeping references to
* values and comparing them with == (or using e.g. assertEquals), we compare snapshots.
*
* This of course assumes that all values have sensible toString implementations.
*/
fun Any?.snapshot(): Snapshot = toString()