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 f554da7b..e2a8c70e 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 @@ -7,6 +7,7 @@ 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.list.listCell +import world.phantasmal.observable.cell.map import world.phantasmal.web.questEditor.actions.* import world.phantasmal.web.questEditor.models.QuestEventActionModel import world.phantasmal.web.questEditor.models.QuestEventModel @@ -165,6 +166,17 @@ class EventsController(private val store: QuestEditorStore) : Controller() { store.executeAction(DeleteEventActionAction(::selectEvent, event, index, action)) } + fun canGoToEvent(eventId: Cell): Cell = + map(enabled, events, eventId) { en, evts, id -> + en && evts.any { it.id.value == id } + } + + fun goToEvent(eventId: Int) { + events.value.find { it.id.value == eventId }?.let { event -> + store.setSelectedEvent(event) + } + } + fun setActionSectionId( event: QuestEventModel, action: QuestEventActionModel.SpawnNpcs, diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestModel.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestModel.kt index 9641fc5a..5dd8a40a 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestModel.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestModel.kt @@ -38,7 +38,7 @@ class QuestModel( private val _mapDesignations = mutableCell(mapDesignations) private val _npcs = SimpleListCell(npcs) { arrayOf(it.sectionInitialized, it.wave) } private val _objects = SimpleListCell(objects) { arrayOf(it.sectionInitialized) } - private val _events = SimpleListCell(events) + private val _events = SimpleListCell(events) { arrayOf(it.id) } val id: Cell = _id val language: Cell = _language diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/EventActionWidget.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/EventActionWidget.kt new file mode 100644 index 00000000..1fec8f52 --- /dev/null +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/EventActionWidget.kt @@ -0,0 +1,123 @@ +package world.phantasmal.web.questEditor.widgets + +import org.w3c.dom.Node +import world.phantasmal.observable.cell.cell +import world.phantasmal.web.questEditor.controllers.EventsController +import world.phantasmal.web.questEditor.models.QuestEventActionModel +import world.phantasmal.web.questEditor.models.QuestEventModel +import world.phantasmal.webui.dom.Icon +import world.phantasmal.webui.dom.td +import world.phantasmal.webui.dom.th +import world.phantasmal.webui.dom.tr +import world.phantasmal.webui.widgets.Button +import world.phantasmal.webui.widgets.IntInput +import world.phantasmal.webui.widgets.Widget + +class EventActionWidget( + private val ctrl: EventsController, + private val event: QuestEventModel, + private val action: QuestEventActionModel, +) : Widget() { + override fun Node.createElement() = + tr { + className = "pw-quest-editor-event-action" + + th { textContent = "${action.shortName}:" } + + when (action) { + is QuestEventActionModel.SpawnNpcs -> { + td { + addChild( + IntInput( + enabled = ctrl.enabled, + tooltip = cell("Section"), + value = action.sectionId, + onChange = { ctrl.setActionSectionId(event, action, it) }, + min = 0, + step = 1, + ) + ) + addChild( + IntInput( + enabled = ctrl.enabled, + tooltip = cell("Appear flag"), + value = action.appearFlag, + onChange = { ctrl.setActionAppearFlag(event, action, it) }, + min = 0, + step = 1, + ) + ) + } + } + is QuestEventActionModel.Door -> { + td { + addChild( + IntInput( + enabled = ctrl.enabled, + tooltip = cell("Door"), + value = action.doorId, + onChange = { ctrl.setActionDoorId(event, action, it) }, + min = 0, + step = 1, + ) + ) + } + } + is QuestEventActionModel.TriggerEvent -> { + td { + addChild( + IntInput( + enabled = ctrl.enabled, + value = action.eventId, + onChange = { ctrl.setActionEventId(event, action, it) }, + min = 0, + step = 1, + ) + ) + } + } + } + + td { + className = "pw-quest-editor-event-action-buttons" + + addChild( + Button( + enabled = ctrl.enabled, + tooltip = cell("Remove this action from the event"), + iconLeft = Icon.Remove, + onClick = { ctrl.removeAction(event, action) } + ) + ) + + if (action is QuestEventActionModel.TriggerEvent) { + addChild( + Button( + enabled = ctrl.canGoToEvent(action.eventId), + tooltip = cell("Go to event"), + iconLeft = Icon.ArrowRight, + onClick = { e -> + e.stopPropagation() + ctrl.goToEvent(action.eventId.value) + } + ) + ) + } + } + } + + companion object { + init { + @Suppress("CssUnusedSymbol", "CssUnresolvedCustomProperty") + // language=css + style( + """ + .pw-quest-editor-event-action-buttons { + display: flex; + flex-direction: row; + } + """.trimIndent() + ) + } + } +} 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 665660ae..6e0af648 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,15 +1,10 @@ package world.phantasmal.web.questEditor.widgets import org.w3c.dom.* -import world.phantasmal.core.disposable.Disposable -import world.phantasmal.core.disposable.Disposer -import world.phantasmal.observable.cell.cell import world.phantasmal.web.questEditor.controllers.EventsController -import world.phantasmal.web.questEditor.models.QuestEventActionModel import world.phantasmal.web.questEditor.models.QuestEventModel import world.phantasmal.webui.dom.* import world.phantasmal.webui.obj -import world.phantasmal.webui.widgets.Button import world.phantasmal.webui.widgets.Dropdown import world.phantasmal.webui.widgets.IntInput import world.phantasmal.webui.widgets.Widget @@ -118,8 +113,8 @@ class EventWidget( } } tbody { - bindDisposableChildrenTo(event.actions) { action, _ -> - createActionElement(action) + bindChildWidgetsTo(event.actions) { action, _ -> + EventActionWidget(ctrl, event, action) } } tfoot { @@ -139,91 +134,12 @@ class EventWidget( } } - private fun Node.createActionElement(action: QuestEventActionModel): Pair { - val disposer = Disposer() - - val node = tr { - th { textContent = "${action.shortName}:" } - - when (action) { - is QuestEventActionModel.SpawnNpcs -> { - td { - addWidget( - disposer.add(IntInput( - enabled = ctrl.enabled, - tooltip = cell("Section"), - value = action.sectionId, - onChange = { ctrl.setActionSectionId(event, action, it) }, - min = 0, - step = 1, - )), - addToDisposer = false, - ) - addWidget( - disposer.add(IntInput( - enabled = ctrl.enabled, - tooltip = cell("Appear flag"), - value = action.appearFlag, - onChange = { ctrl.setActionAppearFlag(event, action, it) }, - min = 0, - step = 1, - )), - addToDisposer = false, - ) - } - } - is QuestEventActionModel.Door -> { - td { - addWidget( - disposer.add(IntInput( - enabled = ctrl.enabled, - tooltip = cell("Door"), - value = action.doorId, - onChange = { ctrl.setActionDoorId(event, action, it) }, - min = 0, - step = 1, - )), - addToDisposer = false, - ) - } - } - is QuestEventActionModel.TriggerEvent -> { - td { - addWidget( - disposer.add(IntInput( - enabled = ctrl.enabled, - value = action.eventId, - onChange = { ctrl.setActionEventId(event, action, it) }, - min = 0, - step = 1, - )), - addToDisposer = false, - ) - } - } - } - - td { - addWidget( - disposer.add(Button( - enabled = ctrl.enabled, - tooltip = cell("Remove this action from the event"), - iconLeft = Icon.Remove, - onClick = { ctrl.removeAction(event, action) } - )), - addToDisposer = false, - ) - } - } - - return Pair(node, disposer) - } - companion object { init { @Suppress("CssUnusedSymbol", "CssUnresolvedCustomProperty") // language=css - style(""" + style( + """ .pw-quest-editor-event { display: flex; flex-wrap: wrap; @@ -250,11 +166,11 @@ class EventWidget( } .pw-quest-editor-event-props { - width: 120px; + width: 115px; } .pw-quest-editor-event-actions { - width: 150px; + width: 165px; } .pw-quest-editor-event > div > table { @@ -265,7 +181,8 @@ class EventWidget( .pw-quest-editor-event th { text-align: left; } - """.trimIndent()) + """.trimIndent() + ) } } } diff --git a/web/src/test/kotlin/world/phantasmal/web/questEditor/controllers/EventsControllerTests.kt b/web/src/test/kotlin/world/phantasmal/web/questEditor/controllers/EventsControllerTests.kt index a0025565..d804fd7c 100644 --- a/web/src/test/kotlin/world/phantasmal/web/questEditor/controllers/EventsControllerTests.kt +++ b/web/src/test/kotlin/world/phantasmal/web/questEditor/controllers/EventsControllerTests.kt @@ -1,12 +1,10 @@ package world.phantasmal.web.questEditor.controllers import world.phantasmal.web.questEditor.models.QuestEventActionModel +import world.phantasmal.web.questEditor.models.QuestEventModel import world.phantasmal.web.test.WebTestSuite import world.phantasmal.web.test.createQuestModel -import kotlin.test.Test -import kotlin.test.assertEquals -import kotlin.test.assertFalse -import kotlin.test.assertTrue +import kotlin.test.* class EventsControllerTests : WebTestSuite { @Test @@ -74,4 +72,76 @@ class EventsControllerTests : WebTestSuite { assertEquals(1, event.actions.value.size) } + + @Test + fun canGoToEvent() = testAsync { + // Setup. + val store = components.questEditorStore + // Quest with two events, the first event triggers the second event. + val quest = createQuestModel( + mapDesignations = mapOf(1 to 0), + events = listOf( + QuestEventModel( + id = 100, + areaId = 1, + sectionId = 11, + waveId = 1, + delay = 50, + unknown = 0, + actions = mutableListOf(QuestEventActionModel.TriggerEvent(101)), + ), + QuestEventModel( + id = 101, + areaId = 1, + sectionId = 11, + waveId = 2, + delay = 50, + unknown = 0, + actions = mutableListOf(QuestEventActionModel.Door.Unlock(7)), + ), + ), + ) + store.setCurrentQuest(quest) + store.setCurrentArea(quest.areaVariants.value.first().area) + + val ctrl = disposer.add(EventsController(store)) + + val canGoToEvent = ctrl.canGoToEvent( + (ctrl.events[0].actions[0] as QuestEventActionModel.TriggerEvent).eventId + ) + + // We test the observed value instead of the cell's value property. + var canGoToEventValue: Boolean? = null + + disposer.add(canGoToEvent.observe(callNow = true) { + assertNull(canGoToEventValue) + canGoToEventValue = it.value + }) + + assertEquals(true, canGoToEventValue) + + // Let event 100 point to nonexistent event 102. + canGoToEventValue = null + ctrl.setActionEventId( + ctrl.events[0], + ctrl.events[0].actions[0] as QuestEventActionModel.TriggerEvent, + 102, + ) + + assertEquals(false, canGoToEventValue) + + // Add event 102. + canGoToEventValue = null + ctrl.selectEvent(null) // Deselect so the next event will be added at the end of the list. + ctrl.addEvent() + ctrl.setId(ctrl.events.value.last(), 102) + + assertEquals(true, canGoToEventValue) + + // Remove event 102. + canGoToEventValue = null + ctrl.removeEvent(ctrl.events.value.last()) + + assertEquals(false, canGoToEventValue) + } } diff --git a/webui/src/main/kotlin/world/phantasmal/webui/dom/Dom.kt b/webui/src/main/kotlin/world/phantasmal/webui/dom/Dom.kt index cb2862d3..eaf3216e 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/dom/Dom.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/dom/Dom.kt @@ -100,6 +100,7 @@ fun getRoot(): HTMLElement = document.getElementById("pw-root") as HTMLElement enum class Icon { ArrowDown, + ArrowRight, Eye, File, GitHub, @@ -122,6 +123,7 @@ enum class Icon { fun Node.icon(icon: Icon): HTMLElement { val iconStr = when (icon) { Icon.ArrowDown -> "fas fa-arrow-down" + Icon.ArrowRight -> "fas fa-arrow-right" Icon.Eye -> "far fa-eye" Icon.File -> "fas fa-file" Icon.GitHub -> "fab fa-github"