From ecb95b5f5eeda27b7f66431a8a3e502e25fe9da5 Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Fri, 3 Dec 2021 15:35:29 +0100 Subject: [PATCH] Turned most cell convenience methods into extension methods to avoid circular dependencies between different cell implementations. Added simple implementation of ListCell.sortedWith and ListCell.filtered with predicate cell. --- .../world/phantasmal/observable/cell/Cell.kt | 29 ------ .../observable/cell/CellCreation.kt | 93 +++++++++++++++++++ .../observable/cell/CellExtensions.kt | 64 ------------- .../observable/cell/list/AbstractListCell.kt | 2 +- .../observable/cell/list/ImmutableListCell.kt | 16 +--- .../observable/cell/list/ListCell.kt | 24 ----- .../observable/cell/list/ListCellCreation.kt | 37 ++++++++ .../phantasmal/web/core/undo/UndoManager.kt | 1 + .../MethodsForEpisodeController.kt | 66 ++++++------- .../OptimizationResultController.kt | 1 + .../controllers/WantedItemsController.kt | 7 +- .../controllers/AsmEditorController.kt | 5 +- .../controllers/EntityInfoController.kt | 7 +- .../controllers/EventsController.kt | 4 +- .../controllers/NpcCountsController.kt | 3 + .../controllers/QuestInfoController.kt | 4 +- .../rendering/QuestEditorMeshManager.kt | 2 + .../rendering/input/state/StateContext.kt | 1 + .../questEditor/stores/QuestEditorStore.kt | 1 + .../web/questEditor/undo/TextModelUndo.kt | 5 +- .../web/questEditor/widgets/EventWidget.kt | 1 + .../CharacterClassOptionsController.kt | 2 + .../controllers/ViewerToolbarController.kt | 2 + .../web/viewer/stores/ViewerStore.kt | 1 + .../phantasmal/webui/widgets/ResultDialog.kt | 4 +- 25 files changed, 189 insertions(+), 193 deletions(-) delete mode 100644 observable/src/commonMain/kotlin/world/phantasmal/observable/cell/CellExtensions.kt diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/Cell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/Cell.kt index 4fd280c1..d36e5dd3 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/Cell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/Cell.kt @@ -19,33 +19,4 @@ interface Cell : Observable { * @param callNow Call [observer] immediately with the current [value]. */ fun observe(callNow: Boolean = false, observer: Observer): Disposable - - /** - * Map a transformation function over this cell. - * - * @param transform called whenever this cell changes - */ - fun map(transform: (T) -> R): Cell = - DependentCell(this) { transform(value) } - - fun mapToList(transform: (T) -> List): ListCell = - DependentListCell(this) { transform(value) } - - /** - * Map a transformation function that returns a cell over this cell. The resulting cell will - * change when this cell changes and when the cell returned by [transform] changes. - * - * @param transform called whenever this cell changes - */ - fun flatMap(transform: (T) -> Cell): Cell = - FlatteningDependentCell(this) { transform(value) } - - fun flatMapNull(transform: (T) -> Cell?): Cell = - FlatteningDependentCell(this) { transform(value) ?: nullCell() } - - fun isNull(): Cell = - map { it == null } - - fun isNotNull(): Cell = - map { it != null } } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/CellCreation.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/CellCreation.kt index a4e8cf96..ff036e1d 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/CellCreation.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/CellCreation.kt @@ -36,6 +36,16 @@ fun mutableCell(value: T): MutableCell = SimpleCell(value) fun mutableCell(getter: () -> T, setter: (T) -> Unit): MutableCell = DelegatingCell(getter, setter) +/** + * Map a transformation function over this cell. + * + * @param transform called whenever this cell changes + */ +fun Cell.map( + transform: (T) -> R, +): Cell = + DependentCell(this) { transform(value) } + /** * Map a transformation function over 2 cells. * @@ -61,6 +71,20 @@ fun map( ): Cell = DependentCell(c1, c2, c3) { transform(c1.value, c2.value, c3.value) } +fun Cell>.flatten(): Cell = + FlatteningDependentCell(this) { this.value } + +/** + * Map a transformation function that returns a cell over this cell. The resulting cell will + * change when this cell changes and when the cell returned by [transform] changes. + * + * @param transform called whenever this cell changes + */ +fun Cell.flatMap( + transform: (T) -> Cell, +): Cell = + FlatteningDependentCell(this) { transform(value) } + /** * Map a transformation function that returns a cell over 2 cells. The resulting cell will change * when either cell changes and also when the cell returned by [transform] changes. @@ -74,5 +98,74 @@ fun flatMap( ): Cell = FlatteningDependentCell(c1, c2) { transform(c1.value, c2.value) } +fun Cell.flatMapNull(transform: (T) -> Cell?): Cell = + FlatteningDependentCell(this) { transform(value) ?: nullCell() } + +fun Cell<*>.isNull(): Cell = + map { it == null } + +fun Cell<*>.isNotNull(): Cell = + map { it != null } + +infix fun Cell.eq(value: T): Cell = + map { it == value } + +infix fun Cell.eq(other: Cell): Cell = + map(this, other) { a, b -> a == b } + +infix fun Cell.ne(value: T): Cell = + map { it != value } + +infix fun Cell.ne(other: Cell): Cell = + map(this, other) { a, b -> a != b } + +fun Cell.orElse(defaultValue: () -> T): Cell = + map { it ?: defaultValue() } + +infix fun > Cell.gt(value: T): Cell = + map { it > value } + +infix fun > Cell.gt(other: Cell): Cell = + map(this, other) { a, b -> a > b } + +infix fun > Cell.lt(value: T): Cell = + map { it < value } + +infix fun > Cell.lt(other: Cell): Cell = + map(this, other) { a, b -> a < b } + fun and(vararg cells: Cell): Cell = DependentCell(*cells) { cells.all { it.value } } + +infix fun Cell.and(other: Cell): Cell = + map(this, other) { a, b -> a && b } + +infix fun Cell.and(other: Boolean): Cell = + if (other) this else falseCell() + +infix fun Cell.or(other: Cell): Cell = + map(this, other) { a, b -> a || b } + +infix fun Cell.xor(other: Cell): Cell = + // Use != because of https://youtrack.jetbrains.com/issue/KT-31277. + map(this, other) { a, b -> a != b } + +operator fun Cell.not(): Cell = map { !it } + +operator fun Cell.plus(value: Int): Cell = + map { it + value } + +operator fun Cell.minus(value: Int): Cell = + map { it - value } + +fun Cell.isEmpty(): Cell = + map { it.isEmpty() } + +fun Cell.isNotEmpty(): Cell = + map { it.isNotEmpty() } + +fun Cell.isBlank(): Cell = + map { it.isBlank() } + +fun Cell.isNotBlank(): Cell = + map { it.isNotBlank() } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/CellExtensions.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/CellExtensions.kt deleted file mode 100644 index 03dc87da..00000000 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/CellExtensions.kt +++ /dev/null @@ -1,64 +0,0 @@ -package world.phantasmal.observable.cell - -infix fun Cell.eq(value: T): Cell = - map { it == value } - -infix fun Cell.eq(other: Cell): Cell = - map(this, other) { a, b -> a == b } - -infix fun Cell.ne(value: T): Cell = - map { it != value } - -infix fun Cell.ne(other: Cell): Cell = - map(this, other) { a, b -> a != b } - -fun Cell.orElse(defaultValue: () -> T): Cell = - map { it ?: defaultValue() } - -infix fun > Cell.gt(value: T): Cell = - map { it > value } - -infix fun > Cell.gt(other: Cell): Cell = - map(this, other) { a, b -> a > b } - -infix fun > Cell.lt(value: T): Cell = - map { it < value } - -infix fun > Cell.lt(other: Cell): Cell = - map(this, other) { a, b -> a < b } - -infix fun Cell.and(other: Cell): Cell = - map(this, other) { a, b -> a && b } - -infix fun Cell.and(other: Boolean): Cell = - if (other) this else falseCell() - -infix fun Cell.or(other: Cell): Cell = - map(this, other) { a, b -> a || b } - -infix fun Cell.xor(other: Cell): Cell = - // Use != because of https://youtrack.jetbrains.com/issue/KT-31277. - map(this, other) { a, b -> a != b } - -operator fun Cell.not(): Cell = map { !it } - -operator fun Cell.plus(value: Int): Cell = - map { it + value } - -operator fun Cell.minus(value: Int): Cell = - map { it - value } - -fun Cell.isEmpty(): Cell = - map { it.isEmpty() } - -fun Cell.isNotEmpty(): Cell = - map { it.isNotEmpty() } - -fun Cell.isBlank(): Cell = - map { it.isBlank() } - -fun Cell.isNotBlank(): Cell = - map { it.isNotBlank() } - -fun Cell>.flatten(): Cell = - FlatteningDependentCell(this) { this.value } 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 0fc91615..275c614e 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 @@ -35,7 +35,7 @@ abstract class AbstractListCell : AbstractCell>(), ListCell { @Suppress("LeakingThis") final override val size: Cell = DependentCell(this) { value.size } - final override val empty: Cell = size.map { it == 0 } + final override val empty: Cell = DependentCell(size) { size.value == 0 } final override val notEmpty: Cell = !empty diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ImmutableListCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ImmutableListCell.kt index 7e8d930b..b1883d76 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ImmutableListCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ImmutableListCell.kt @@ -2,15 +2,15 @@ package world.phantasmal.observable.cell.list import world.phantasmal.core.disposable.Disposable import world.phantasmal.core.disposable.nopDisposable -import world.phantasmal.core.unsafe.unsafeAssertNotNull import world.phantasmal.observable.AbstractDependency import world.phantasmal.observable.ChangeEvent import world.phantasmal.observable.Observer -import world.phantasmal.observable.cell.* +import world.phantasmal.observable.cell.Cell +import world.phantasmal.observable.cell.cell +import world.phantasmal.observable.cell.falseCell +import world.phantasmal.observable.cell.trueCell class ImmutableListCell(private val elements: List) : AbstractDependency(), ListCell { - private var firstOrNull: Cell? = null - override val size: Cell = cell(elements.size) override val empty: Cell = if (elements.isEmpty()) trueCell() else falseCell() override val notEmpty: Cell = if (elements.isNotEmpty()) trueCell() else falseCell() @@ -38,14 +38,6 @@ class ImmutableListCell(private val elements: List) : AbstractDependency() return nopDisposable() } - override fun firstOrNull(): Cell { - if (firstOrNull == null) { - firstOrNull = ImmutableCell(elements.firstOrNull()) - } - - return unsafeAssertNotNull(firstOrNull) - } - override fun emitDependencyChanged() { error("ImmutableListCell can't change.") } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListCell.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListCell.kt index a78b9788..fb90f73f 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListCell.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListCell.kt @@ -2,13 +2,8 @@ package world.phantasmal.observable.cell.list import world.phantasmal.core.disposable.Disposable import world.phantasmal.observable.cell.Cell -import world.phantasmal.observable.cell.DependentCell interface ListCell : Cell> { - /** - * Do not keep long-lived references to a [ListCell]'s [value], it may or may not be mutated - * when the [ListCell] is mutated. - */ override val value: List val size: Cell @@ -21,24 +16,5 @@ interface ListCell : Cell> { fun observeList(callNow: Boolean = false, observer: ListObserver): Disposable - // TODO: Optimize this. - fun listMap(transform: (E) -> R): ListCell = - DependentListCell(this) { value.map(transform) } - - fun fold(initialValue: R, operation: (R, E) -> R): Cell = - DependentCell(this) { value.fold(initialValue, operation) } - - fun all(predicate: (E) -> Boolean): Cell = - DependentCell(this) { value.all(predicate) } - - fun sumOf(selector: (E) -> Int): Cell = - DependentCell(this) { value.sumOf(selector) } - - fun filtered(predicate: (E) -> Boolean): ListCell = - FilteredListCell(this, predicate) - - fun firstOrNull(): Cell = - DependentCell(this) { value.firstOrNull() } - operator fun contains(element: @UnsafeVariance E): Boolean = element in value } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListCellCreation.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListCellCreation.kt index 5f47e0ac..31d5b792 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListCellCreation.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/cell/list/ListCellCreation.kt @@ -1,6 +1,7 @@ package world.phantasmal.observable.cell.list import world.phantasmal.observable.cell.Cell +import world.phantasmal.observable.cell.DependentCell private val EMPTY_LIST_CELL = ImmutableListCell(emptyList()) @@ -17,6 +18,42 @@ fun mutableListCell( ): MutableListCell = SimpleListCell(mutableListOf(*elements), extractDependencies) +fun ListCell.listMap(transform: (E) -> R): ListCell = + DependentListCell(this) { value.map(transform) } + +fun ListCell.fold(initialValue: R, operation: (R, E) -> R): Cell = + DependentCell(this) { value.fold(initialValue, operation) } + +fun ListCell.all(predicate: (E) -> Boolean): Cell = + DependentCell(this) { value.all(predicate) } + +fun ListCell.sumOf(selector: (E) -> Int): Cell = + DependentCell(this) { value.sumOf(selector) } + +fun ListCell.filtered(predicate: (E) -> Boolean): ListCell = + FilteredListCell(this, predicate) + +fun ListCell.filtered(predicate: Cell<(E) -> Boolean>): ListCell = + DependentListCell(this, predicate) { value.filter(predicate.value) } + +fun ListCell.sortedWith(comparator: Cell>): ListCell = + DependentListCell(this, comparator) { value.sortedWith(comparator.value) } + +fun ListCell.firstOrNull(): Cell = + DependentCell(this) { value.firstOrNull() } + +fun Cell.mapToList( + transform: (T) -> List, +): ListCell = + DependentListCell(this) { transform(value) } + +fun mapToList( + c1: Cell, + c2: Cell, + transform: (T1, T2) -> List, +): ListCell = + DependentListCell(c1, c2) { transform(c1.value, c2.value) } + fun Cell.flatMapToList( transform: (T) -> ListCell, ): ListCell = diff --git a/web/src/main/kotlin/world/phantasmal/web/core/undo/UndoManager.kt b/web/src/main/kotlin/world/phantasmal/web/core/undo/UndoManager.kt index c30df861..cc4719af 100644 --- a/web/src/main/kotlin/world/phantasmal/web/core/undo/UndoManager.kt +++ b/web/src/main/kotlin/world/phantasmal/web/core/undo/UndoManager.kt @@ -1,6 +1,7 @@ package world.phantasmal.web.core.undo import world.phantasmal.observable.cell.* +import world.phantasmal.observable.cell.list.fold import world.phantasmal.observable.cell.list.mutableListCell import world.phantasmal.web.core.actions.Action diff --git a/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/controllers/MethodsForEpisodeController.kt b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/controllers/MethodsForEpisodeController.kt index c2fb6472..a749b4af 100644 --- a/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/controllers/MethodsForEpisodeController.kt +++ b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/controllers/MethodsForEpisodeController.kt @@ -1,8 +1,10 @@ package world.phantasmal.web.huntOptimizer.controllers import world.phantasmal.observable.cell.list.ListCell +import world.phantasmal.observable.cell.list.filtered import world.phantasmal.observable.cell.list.listCell -import world.phantasmal.observable.cell.list.mutableListCell +import world.phantasmal.observable.cell.list.sortedWith +import world.phantasmal.observable.cell.mutableCell import world.phantasmal.psolib.Episode import world.phantasmal.psolib.fileFormats.quest.NpcType import world.phantasmal.web.huntOptimizer.models.HuntMethodModel @@ -18,47 +20,16 @@ class MethodsForEpisodeController( private val huntMethodStore: HuntMethodStore, episode: Episode, ) : TableController() { - private val methods = mutableListCell() private val enemies: List = NpcType.VALUES.filter { it.enemy && it.episode == episode } - private var sortColumns: List> = emptyList() - - private val comparator: Comparator = - Comparator { a, b -> - for (sortColumn in sortColumns) { - val cmp = when (sortColumn.column.key) { - METHOD_COL_KEY -> - a.name.asDynamic().localeCompare(b.name).unsafeCast() - - TIME_COL_KEY -> a.time.value.compareTo(b.time.value) - - else -> { - val type = NpcType.valueOf(sortColumn.column.key) - (a.enemyCounts[type] ?: 0) - (b.enemyCounts[type] ?: 0) - } - } - - if (cmp != 0) { - return@Comparator if (sortColumn.direction == SortDirection.Asc) cmp else -cmp - } - } - - 0 - } + private val comparator = mutableCell(Comparator { _, _ -> 0 }) override val fixedColumns = 2 override val values: ListCell by lazy { - // TODO: Use ListCell.sortedWith when this is available. - observe(huntMethodStore.methods) { allMethods -> - methods.value = allMethods - .asSequence() - .filter { it.episode == episode } - .sortedWith(comparator) - .toList() - } - - methods + huntMethodStore.methods + .filtered { it.episode == episode } + .sortedWith(comparator) } override val loadingStatus: LoadingStatusCell = huntMethodStore.methodsStatus @@ -97,8 +68,27 @@ class MethodsForEpisodeController( ) override fun sort(sortColumns: List>) { - this.sortColumns = sortColumns - methods.sortWith(comparator) + comparator.value = Comparator { a, b -> + for (sortColumn in sortColumns) { + val cmp = when (sortColumn.column.key) { + METHOD_COL_KEY -> + a.name.asDynamic().localeCompare(b.name).unsafeCast() + + TIME_COL_KEY -> a.time.value.compareTo(b.time.value) + + else -> { + val type = NpcType.valueOf(sortColumn.column.key) + (a.enemyCounts[type] ?: 0) - (b.enemyCounts[type] ?: 0) + } + } + + if (cmp != 0) { + return@Comparator if (sortColumn.direction == SortDirection.Asc) cmp else -cmp + } + } + + 0 + } } suspend fun setMethodTime(method: HuntMethodModel, time: Duration) { diff --git a/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/controllers/OptimizationResultController.kt b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/controllers/OptimizationResultController.kt index 11bd0b13..1d0b395b 100644 --- a/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/controllers/OptimizationResultController.kt +++ b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/controllers/OptimizationResultController.kt @@ -2,6 +2,7 @@ package world.phantasmal.web.huntOptimizer.controllers import world.phantasmal.observable.cell.cell import world.phantasmal.observable.cell.list.ListCell +import world.phantasmal.observable.cell.list.mapToList import world.phantasmal.web.huntOptimizer.models.OptimalMethodModel import world.phantasmal.web.huntOptimizer.stores.HuntOptimizerStore import world.phantasmal.webui.controllers.Column diff --git a/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/controllers/WantedItemsController.kt b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/controllers/WantedItemsController.kt index e1b2ead4..e81544ab 100644 --- a/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/controllers/WantedItemsController.kt +++ b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/controllers/WantedItemsController.kt @@ -3,6 +3,7 @@ package world.phantasmal.web.huntOptimizer.controllers import world.phantasmal.observable.cell.Cell import world.phantasmal.observable.cell.MutableCell import world.phantasmal.observable.cell.list.ListCell +import world.phantasmal.observable.cell.list.filtered import world.phantasmal.observable.cell.mutableCell import world.phantasmal.web.huntOptimizer.models.WantedItemModel import world.phantasmal.web.huntOptimizer.stores.HuntOptimizerStore @@ -14,10 +15,8 @@ class WantedItemsController( ) : Controller() { private val selectableItemsFilter: MutableCell<(ItemType) -> Boolean> = mutableCell { true } - // TODO: Use ListCell.filtered with a Cell when this is supported. - val selectableItems: Cell> = selectableItemsFilter.flatMap { filter -> - huntOptimizerStore.huntableItems.filtered(filter) - } + val selectableItems: Cell> = + huntOptimizerStore.huntableItems.filtered(selectableItemsFilter) val wantedItems: ListCell = huntOptimizerStore.wantedItems diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/AsmEditorController.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/AsmEditorController.kt index 1473c258..b6c4992d 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/AsmEditorController.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/AsmEditorController.kt @@ -1,10 +1,7 @@ package world.phantasmal.web.questEditor.controllers import world.phantasmal.observable.Observable -import world.phantasmal.observable.cell.Cell -import world.phantasmal.observable.cell.not -import world.phantasmal.observable.cell.or -import world.phantasmal.observable.cell.orElse +import world.phantasmal.observable.cell.* import world.phantasmal.web.externals.monacoEditor.ITextModel import world.phantasmal.web.externals.monacoEditor.createModel import world.phantasmal.web.questEditor.stores.AsmStore diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/EntityInfoController.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/EntityInfoController.kt index b89dcc7d..ced7321b 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/EntityInfoController.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/EntityInfoController.kt @@ -2,13 +2,12 @@ package world.phantasmal.web.questEditor.controllers import world.phantasmal.core.math.degToRad import world.phantasmal.core.math.radToDeg -import world.phantasmal.psolib.fileFormats.quest.EntityPropType -import world.phantasmal.observable.cell.Cell -import world.phantasmal.observable.cell.cell +import world.phantasmal.observable.cell.* import world.phantasmal.observable.cell.list.ListCell import world.phantasmal.observable.cell.list.emptyListCell import world.phantasmal.observable.cell.list.flatMapToList -import world.phantasmal.observable.cell.zeroIntCell +import world.phantasmal.observable.cell.list.listMap +import world.phantasmal.psolib.fileFormats.quest.EntityPropType import world.phantasmal.web.core.euler import world.phantasmal.web.externals.three.Euler import world.phantasmal.web.externals.three.Vector3 diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/EventsController.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/EventsController.kt index ea4d81db..d9b4234d 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/EventsController.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/EventsController.kt @@ -1,8 +1,6 @@ package world.phantasmal.web.questEditor.controllers -import world.phantasmal.observable.cell.Cell -import world.phantasmal.observable.cell.and -import world.phantasmal.observable.cell.eq +import world.phantasmal.observable.cell.* import world.phantasmal.observable.cell.list.ListCell import world.phantasmal.observable.cell.list.listCell import world.phantasmal.web.questEditor.actions.* diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/NpcCountsController.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/NpcCountsController.kt index 81ef14f3..328f3b33 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/NpcCountsController.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/NpcCountsController.kt @@ -2,7 +2,10 @@ package world.phantasmal.web.questEditor.controllers import world.phantasmal.psolib.fileFormats.quest.NpcType import world.phantasmal.observable.cell.Cell +import world.phantasmal.observable.cell.flatMap +import world.phantasmal.observable.cell.isNull import world.phantasmal.observable.cell.list.emptyListCell +import world.phantasmal.observable.cell.map import world.phantasmal.web.questEditor.models.QuestNpcModel import world.phantasmal.web.questEditor.stores.QuestEditorStore import world.phantasmal.webui.controllers.Controller diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/QuestInfoController.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/QuestInfoController.kt index 1342f4d0..b290bd52 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/QuestInfoController.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/QuestInfoController.kt @@ -1,8 +1,6 @@ package world.phantasmal.web.questEditor.controllers -import world.phantasmal.observable.cell.Cell -import world.phantasmal.observable.cell.cell -import world.phantasmal.observable.cell.emptyStringCell +import world.phantasmal.observable.cell.* import world.phantasmal.web.questEditor.actions.EditPropertyAction import world.phantasmal.web.questEditor.stores.QuestEditorStore import world.phantasmal.webui.controllers.Controller diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestEditorMeshManager.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestEditorMeshManager.kt index eb7df822..6ea360d6 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestEditorMeshManager.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/QuestEditorMeshManager.kt @@ -1,6 +1,8 @@ package world.phantasmal.web.questEditor.rendering +import world.phantasmal.observable.cell.flatMapNull import world.phantasmal.observable.cell.list.emptyListCell +import world.phantasmal.observable.cell.list.filtered import world.phantasmal.web.questEditor.loading.AreaAssetLoader import world.phantasmal.web.questEditor.loading.EntityAssetLoader import world.phantasmal.web.questEditor.stores.QuestEditorStore diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/input/state/StateContext.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/input/state/StateContext.kt index f8a74265..07d89dec 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/input/state/StateContext.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/input/state/StateContext.kt @@ -3,6 +3,7 @@ package world.phantasmal.web.questEditor.rendering.input.state import mu.KotlinLogging import world.phantasmal.core.asJsArray import world.phantasmal.observable.cell.Cell +import world.phantasmal.observable.cell.flatMapNull import world.phantasmal.psolib.fileFormats.ninja.XjObject import world.phantasmal.web.core.dot import world.phantasmal.web.core.minusAssign diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/QuestEditorStore.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/QuestEditorStore.kt index be585087..55c1f1f5 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/QuestEditorStore.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/QuestEditorStore.kt @@ -6,6 +6,7 @@ import world.phantasmal.psolib.Episode import world.phantasmal.observable.cell.* import world.phantasmal.observable.cell.list.ListCell import world.phantasmal.observable.cell.list.emptyListCell +import world.phantasmal.observable.cell.list.filtered import world.phantasmal.observable.cell.list.flatMapToList import world.phantasmal.web.core.PwToolType import world.phantasmal.web.core.actions.Action diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/undo/TextModelUndo.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/undo/TextModelUndo.kt index bafea35a..f05e354e 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/undo/TextModelUndo.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/undo/TextModelUndo.kt @@ -4,10 +4,7 @@ import world.phantasmal.core.disposable.Disposable import world.phantasmal.core.disposable.TrackedDisposable import world.phantasmal.observable.ChangeEvent import world.phantasmal.observable.Observable -import world.phantasmal.observable.cell.Cell -import world.phantasmal.observable.cell.MutableCell -import world.phantasmal.observable.cell.eq -import world.phantasmal.observable.cell.mutableCell +import world.phantasmal.observable.cell.* import world.phantasmal.observable.emitter import world.phantasmal.web.core.actions.Action import world.phantasmal.web.core.undo.Undo diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/EventWidget.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/EventWidget.kt index 6e0af648..a410de82 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/EventWidget.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/EventWidget.kt @@ -1,6 +1,7 @@ package world.phantasmal.web.questEditor.widgets import org.w3c.dom.* +import world.phantasmal.observable.cell.map import world.phantasmal.web.questEditor.controllers.EventsController import world.phantasmal.web.questEditor.models.QuestEventModel import world.phantasmal.webui.dom.* diff --git a/web/src/main/kotlin/world/phantasmal/web/viewer/controllers/CharacterClassOptionsController.kt b/web/src/main/kotlin/world/phantasmal/web/viewer/controllers/CharacterClassOptionsController.kt index 3f2696aa..4a34305b 100644 --- a/web/src/main/kotlin/world/phantasmal/web/viewer/controllers/CharacterClassOptionsController.kt +++ b/web/src/main/kotlin/world/phantasmal/web/viewer/controllers/CharacterClassOptionsController.kt @@ -1,6 +1,8 @@ package world.phantasmal.web.viewer.controllers import world.phantasmal.observable.cell.Cell +import world.phantasmal.observable.cell.isNotNull +import world.phantasmal.observable.cell.map import world.phantasmal.observable.cell.plus import world.phantasmal.web.shared.dto.SectionId import world.phantasmal.web.viewer.stores.ViewerStore diff --git a/web/src/main/kotlin/world/phantasmal/web/viewer/controllers/ViewerToolbarController.kt b/web/src/main/kotlin/world/phantasmal/web/viewer/controllers/ViewerToolbarController.kt index 134de2ef..c3e862f8 100644 --- a/web/src/main/kotlin/world/phantasmal/web/viewer/controllers/ViewerToolbarController.kt +++ b/web/src/main/kotlin/world/phantasmal/web/viewer/controllers/ViewerToolbarController.kt @@ -14,6 +14,8 @@ import world.phantasmal.psolib.fileFormats.parseAfs import world.phantasmal.psolib.fileFormats.parseAreaCollisionGeometry import world.phantasmal.psolib.fileFormats.parseAreaRenderGeometry import world.phantasmal.observable.cell.Cell +import world.phantasmal.observable.cell.isNotNull +import world.phantasmal.observable.cell.map import world.phantasmal.observable.cell.mutableCell import world.phantasmal.observable.change import world.phantasmal.web.core.files.cursor diff --git a/web/src/main/kotlin/world/phantasmal/web/viewer/stores/ViewerStore.kt b/web/src/main/kotlin/world/phantasmal/web/viewer/stores/ViewerStore.kt index c23aa392..70683a0a 100644 --- a/web/src/main/kotlin/world/phantasmal/web/viewer/stores/ViewerStore.kt +++ b/web/src/main/kotlin/world/phantasmal/web/viewer/stores/ViewerStore.kt @@ -13,6 +13,7 @@ import world.phantasmal.observable.cell.Cell import world.phantasmal.observable.cell.and import world.phantasmal.observable.cell.list.ListCell import world.phantasmal.observable.cell.list.mutableListCell +import world.phantasmal.observable.cell.map import world.phantasmal.observable.cell.mutableCell import world.phantasmal.observable.change import world.phantasmal.web.core.PwToolType diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/ResultDialog.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/ResultDialog.kt index 7f94212a..0114e018 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/ResultDialog.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/ResultDialog.kt @@ -3,9 +3,7 @@ package world.phantasmal.webui.widgets import org.w3c.dom.Node import world.phantasmal.core.Failure import world.phantasmal.core.PwResult -import world.phantasmal.observable.cell.Cell -import world.phantasmal.observable.cell.emptyStringCell -import world.phantasmal.observable.cell.trueCell +import world.phantasmal.observable.cell.* import world.phantasmal.webui.dom.div import world.phantasmal.webui.dom.li import world.phantasmal.webui.dom.ul