mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
Filtered list cells now support changing multiple times in a single mutation.
This commit is contained in:
parent
c3be91a05e
commit
f5b1008c48
@ -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
|
||||||
|
@ -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++
|
||||||
}
|
}
|
||||||
|
@ -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 {
|
||||||
|
@ -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)) {
|
||||||
|
@ -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)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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()
|
||||||
|
@ -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
|
@ -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)) } })
|
||||||
}
|
}
|
||||||
|
@ -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>
|
||||||
|
|
||||||
|
@ -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)
|
||||||
}
|
}
|
||||||
|
@ -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()
|
|
||||||
|
Loading…
Reference in New Issue
Block a user