mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-04 22:58:29 +08:00
Event actions now have a "Go to event" button.
This commit is contained in:
parent
d7f0e536ec
commit
403e03b0ee
@ -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<Int>): Cell<Boolean> =
|
||||
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,
|
||||
|
@ -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<Int> = _id
|
||||
val language: Cell<Int> = _language
|
||||
|
@ -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()
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
@ -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<Node, Disposable> {
|
||||
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()
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -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)
|
||||
}
|
||||
}
|
||||
|
@ -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"
|
||||
|
Loading…
Reference in New Issue
Block a user