diff --git a/FEATURES.md b/FEATURES.md index 7bdc46fe..3cdad155 100644 --- a/FEATURES.md +++ b/FEATURES.md @@ -161,8 +161,6 @@ Features that are in ***bold italics*** are planned but not yet implemented. - When a modal dialog is open, global keybindings should be disabled - The ASM editor is slow with big scripts, e.g. Seat of the Heart (#27) - Improve the default camera target for Crater Interior -- The 3D doesn't update when a different variant is configured with a map designate instruction for - the current area - Entities with rendering issues: - Caves 4 Button door - Pofuilly Slime diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/TranslateEntityAction.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/TranslateEntityAction.kt index 33556ba2..cf45cfb2 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/TranslateEntityAction.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/actions/TranslateEntityAction.kt @@ -3,40 +3,31 @@ package world.phantasmal.web.questEditor.actions import world.phantasmal.web.core.actions.Action import world.phantasmal.web.externals.three.Vector3 import world.phantasmal.web.questEditor.models.QuestEntityModel -import world.phantasmal.web.questEditor.models.SectionModel class TranslateEntityAction( private val setSelectedEntity: (QuestEntityModel<*, *>) -> Unit, + private val setEntitySection: (Int) -> Unit, private val entity: QuestEntityModel<*, *>, - private val newSection: SectionModel?, - private val oldSection: SectionModel?, + private val newSection: Int?, + private val oldSection: Int?, private val newPosition: Vector3, private val oldPosition: Vector3, - private val world: Boolean, ) : Action { override val description: String = "Move ${entity.type.simpleName}" override fun execute() { setSelectedEntity(entity) - newSection?.let(entity::setSection) + newSection?.let(setEntitySection) - if (world) { - entity.setWorldPosition(newPosition) - } else { - entity.setPosition(newPosition) - } + entity.setPosition(newPosition) } override fun undo() { setSelectedEntity(entity) - oldSection?.let(entity::setSection) + oldSection?.let(setEntitySection) - if (world) { - entity.setWorldPosition(oldPosition) - } else { - entity.setPosition(oldPosition) - } + entity.setPosition(oldPosition) } } 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 4442616e..3767d804 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 @@ -123,12 +123,12 @@ class EntityInfoController( questEditorStore.executeAction(TranslateEntityAction( setSelectedEntity = questEditorStore::setSelectedEntity, + setEntitySection = { /* Won't be called. */ }, entity, - entity.section.value, - entity.section.value, - Vector3(x, y, z), - entity.position.value, - false, + newSection = null, + oldSection = null, + newPosition = Vector3(x, y, z), + oldPosition = entity.position.value, )) } @@ -165,7 +165,7 @@ class EntityInfoController( )) } - fun setPropValue(prop:QuestEntityPropModel, value:Any) { + fun setPropValue(prop: QuestEntityPropModel, value: Any) { questEditorStore.selectedEntity.value?.let { entity -> questEditorStore.executeAction(EditEntityPropAction( setSelectedEntity = questEditorStore::setSelectedEntity, diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/loading/AreaAssetLoader.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/loading/AreaAssetLoader.kt index e96a7956..3b1ab23d 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/loading/AreaAssetLoader.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/loading/AreaAssetLoader.kt @@ -76,6 +76,9 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine ): Object3D = cache.get(EpisodeAndAreaVariant(episode, areaVariant)).collisionGeometry + fun getCachedSections(episode: Episode, areaVariant: AreaVariantModel): List? = + cache.getIfPresentNow(EpisodeAndAreaVariant(episode, areaVariant))?.sections + private suspend fun getAreaAsset( episode: Episode, areaVariant: AreaVariantModel, @@ -506,6 +509,24 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine "s_n_0_5i_1_iqqrft", "s_n_0_g_1_iipv9r", "s_n_0_c_1_ihboen", + "s_n_0_3l_2_iljrhl", + "s_n_0_5t_2_ill0ej", + "s_n_0_4e_2_iobj4y", // Deletes useful walls. + "s_n_0_6y_2_ipln11", // Deletes useful walls. + "s_n_0_43_1_iqbzr4", + "s_n_0_o_1_ikqpac", + "s_n_0_c_1_ihrvdk", + "s_n_0_c_1_ih2ob6", + "s_n_0_c_1_ihwsxo", + "s_n_0_c_1_igrh47", + "s_n_0_j9_4_iqqrft", // Deletes useful walls. + "s_n_0_p_2_ihe7ca", + "s_n_0_l_2_igkyx3", + "s_n_0_n_2_igubtb", + "s_n_0_l_2_ihuczl", + "s_n_0_o_1_ijn9y2", + "s_n_0_f_1_ijpzol", + "s_n_0_2n_1_ilgim5", ), ), // Cave 3 diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModel.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModel.kt index a59ac8e9..fbc2c6ce 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModel.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModel.kt @@ -62,26 +62,10 @@ abstract class QuestEntityModel>( open fun setSectionId(sectionId: Int) { entity.sectionId = sectionId.toShort() _sectionId.value = sectionId - } - fun initializeSection(section: SectionModel) { - require(!sectionInitialized.value) { - "Section is already initialized." + if (sectionId != _section.value?.id) { + _section.value = null } - require(section.areaVariant.area.id == areaId) { - "Section should lie within the entity's area." - } - - setSectionId(section.id) - - _section.value = section - - // Update world position and rotation by calling setPosition and setRotation with the - // current position and rotation. - setPosition(position.value) - setRotation(rotation.value) - - setSectionInitialized() } fun setSectionInitialized() { @@ -89,21 +73,32 @@ abstract class QuestEntityModel>( } /** - * Will update the entity's relative transformation but keep its world transformation constant. + * @param keepRelativeTransform If true, keep the entity's relative transform and update its + * world transform. Otherwise keep its world transform and update its relative transform. */ - fun setSection(section: SectionModel) { + fun setSection(section: SectionModel, keepRelativeTransform: Boolean = false) { require(section.areaVariant.area.id == areaId) { "Quest entities can't be moved across areas." } - setSectionId(section.id) + entity.sectionId = section.id.toShort() + _sectionId.value = section.id _section.value = section - // Update relative position and rotation by calling setWorldPosition and setWorldRotation - // with the current world position and rotation. - setWorldPosition(worldPosition.value) - setWorldRotation(worldRotation.value) + if (keepRelativeTransform) { + // Update world position and rotation by calling setPosition and setRotation with the + // current position and rotation. + setPosition(position.value) + setRotation(rotation.value) + } else { + // Update relative position and rotation by calling setWorldPosition and + // setWorldRotation with the current world position and rotation. + setWorldPosition(worldPosition.value) + setWorldRotation(worldRotation.value) + } + + setSectionInitialized() } fun setPosition(pos: Vector3) { 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 72379835..6b3bfb6c 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 @@ -6,6 +6,8 @@ import world.phantasmal.lib.fileFormats.quest.DatUnknown import world.phantasmal.observable.value.Val import world.phantasmal.observable.value.list.ListVal import world.phantasmal.observable.value.list.SimpleListVal +import world.phantasmal.observable.value.list.flatMapToList +import world.phantasmal.observable.value.list.listVal import world.phantasmal.observable.value.map import world.phantasmal.observable.value.mutableVal @@ -57,7 +59,7 @@ class QuestModel( /** * One variant per area. */ - val areaVariants: Val> + val areaVariants: ListVal val npcs: ListVal = _npcs val objects: ListVal = _objects @@ -88,23 +90,24 @@ class QuestModel( map } - areaVariants = map(entitiesPerArea, this.mapDesignations) { entitiesPerArea, mds -> - val variants = mutableMapOf() + areaVariants = + flatMapToList(entitiesPerArea, this.mapDesignations) { entitiesPerArea, mds -> + val variants = mutableMapOf() - for (areaId in entitiesPerArea.values) { - getVariant(episode, areaId, 0)?.let { - variants[areaId] = it + for (areaId in entitiesPerArea.keys) { + getVariant(episode, areaId, 0)?.let { + variants[areaId] = it + } } - } - for ((areaId, variantId) in mds) { - getVariant(episode, areaId, variantId)?.let { - variants[areaId] = it + for ((areaId, variantId) in mds) { + getVariant(episode, areaId, variantId)?.let { + variants[areaId] = it + } } - } - variants.values.toList() - } + listVal(*variants.values.toTypedArray()) + } } fun setId(id: Int): QuestModel { 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 5fbffec4..5d7f30ab 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 @@ -12,10 +12,14 @@ class QuestEditorMeshManager( renderContext: QuestRenderContext, ) : QuestMeshManager(areaAssetLoader, entityAssetLoader, questEditorStore, renderContext) { init { - observe(questEditorStore.currentQuest, questEditorStore.currentArea) { quest, area -> - val areaVariant = quest?.let { + observe( + questEditorStore.currentQuest, + questEditorStore.currentQuest.flatMapNull { it?.areaVariants }, + questEditorStore.currentArea, + ) { quest, questAreaVariants, area -> + val areaVariant = questAreaVariants?.let { area?.let { - quest.areaVariants.value.find { it.area.id == area.id } + questAreaVariants.find { it.area.id == area.id } ?: area.areaVariants.first() } } 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 803fa52f..79bddc56 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 @@ -130,12 +130,12 @@ class StateContext( ) { questEditorStore.executeAction(TranslateEntityAction( ::setSelectedEntity, + { questEditorStore.setEntitySection(entity, it) }, entity, - newSection, - oldSection, + newSection?.id, + oldSection?.id, newPosition, oldPosition, - world = true, )) } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/input/state/TranslationState.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/input/state/TranslationState.kt index d528253d..59f7b670 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/input/state/TranslationState.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/rendering/input/state/TranslationState.kt @@ -15,7 +15,7 @@ class TranslationState( private val grabOffset: Vector3, ) : State() { private val initialSection: SectionModel? = entity.section.value - private val initialPosition: Vector3 = entity.worldPosition.value + private val initialPosition: Vector3 = entity.position.value private val pointerDevicePosition = Vector2() private var shouldTranslate = false private var shouldTranslateVertically = false @@ -46,7 +46,7 @@ class TranslationState( entity, entity.section.value, initialSection, - entity.worldPosition.value, + entity.position.value, initialPosition, ) } @@ -87,6 +87,6 @@ class TranslationState( entity.setSection(initialSection) } - entity.setWorldPosition(initialPosition) + entity.setPosition(initialPosition) } } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/AreaStore.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/AreaStore.kt index fc06ec2e..298aa24b 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/AreaStore.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/AreaStore.kt @@ -9,8 +9,8 @@ import world.phantasmal.webui.stores.Store import world.phantasmal.lib.fileFormats.quest.getAreasForEpisode as getAreasForEpisodeLib class AreaStore(private val areaAssetLoader: AreaAssetLoader) : Store() { - private val areas: Map> = Episode.values() - .map { episode -> + private val areas: Map> = + Episode.values().associate { episode -> episode to getAreasForEpisodeLib(episode).map { area -> val variants = mutableListOf() val areaModel = AreaModel(area.id, area.name, area.order, variants) @@ -22,7 +22,6 @@ class AreaStore(private val areaAssetLoader: AreaAssetLoader) : Store() { areaModel } } - .toMap() fun getAreasForEpisode(episode: Episode): List = areas.getValue(episode) @@ -42,4 +41,7 @@ class AreaStore(private val areaAssetLoader: AreaAssetLoader) : Store() { suspend fun getSections(episode: Episode, variant: AreaVariantModel): List = areaAssetLoader.loadSections(episode, variant) + + fun getLoadedSections(episode: Episode, variant: AreaVariantModel): List? = + areaAssetLoader.getCachedSections(episode, variant) } diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/AsmStore.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/AsmStore.kt index 4f1f26d9..a0e53ebe 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/AsmStore.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/stores/AsmStore.kt @@ -1,6 +1,7 @@ package world.phantasmal.web.questEditor.stores import kotlinx.browser.window +import kotlinx.coroutines.launch import world.phantasmal.core.Severity import world.phantasmal.core.disposable.Disposer import world.phantasmal.core.disposable.disposable @@ -68,7 +69,7 @@ class AsmStore( } observe(asmAnalyser.mapDesignations) { - questEditorStore.currentQuest.value?.setMapDesignations(it) + scope.launch { questEditorStore.setMapDesignations(it) } } observe(problems) { problems -> 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 7d08226d..448e14b0 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 @@ -37,7 +37,7 @@ class QuestEditorStore( val devMode: Val = _devMode - val runner = QuestRunner() + private val runner = QuestRunner() val currentQuest: Val = _currentQuest val currentArea: Val = _currentArea val selectedEvent: Val = _selectedEvent @@ -134,12 +134,7 @@ class QuestEditorStore( _currentQuest.value = quest // Load section data. - quest.areaVariants.value.forEach { variant -> - val sections = areaStore.getSections(quest.episode, variant) - variant.setSections(sections) - setSectionOnQuestEntities(quest.npcs.value, variant, sections) - setSectionOnQuestEntities(quest.objects.value, variant, sections) - } + updateQuestEntitySections(quest) // Ensure all entities have their section initialized. quest.npcs.value.forEach { it.setSectionInitialized() } @@ -150,25 +145,6 @@ class QuestEditorStore( suspend fun getDefaultQuest(episode: Episode): QuestModel = convertQuestToModel(questLoader.loadDefaultQuest(episode), areaStore::getVariant) - private fun setSectionOnQuestEntities( - entities: List>, - variant: AreaVariantModel, - sections: List, - ) { - entities.forEach { entity -> - if (entity.areaId == variant.area.id) { - val section = sections.find { it.id == entity.sectionId.value } - - if (section == null) { - logger.warn { "Section ${entity.sectionId.value} not found." } - entity.setSectionInitialized() - } else { - entity.initializeSection(section) - } - } - } - } - fun setCurrentArea(area: AreaModel?) { val event = selectedEvent.value @@ -215,6 +191,30 @@ class QuestEditorStore( _selectedEntity.value = entity } + suspend fun setMapDesignations(mapDesignations: Map) { + currentQuest.value?.let { quest -> + quest.setMapDesignations(mapDesignations) + updateQuestEntitySections(quest) + } + } + + fun setEntitySection(entity: QuestEntityModel<*, *>, sectionId: Int) { + currentQuest.value?.let { quest -> + val variant = quest.areaVariants.value.find { it.area.id == entity.areaId } + + variant?.let { + val section = areaStore.getLoadedSections(quest.episode, variant) + ?.find { it.id == sectionId } + + if (section == null) { + entity.setSectionId(sectionId) + } else { + entity.setSection(section) + } + } + } + } + fun executeAction(action: Action) { pushAction(action) action.execute() @@ -239,4 +239,32 @@ class QuestEditorStore( fun questSaved() { undoManager.savePoint() } + + private suspend fun updateQuestEntitySections(quest: QuestModel) { + quest.areaVariants.value.forEach { variant -> + val sections = areaStore.getSections(quest.episode, variant) + variant.setSections(sections) + setSectionOnQuestEntities(quest.npcs.value, variant, sections) + setSectionOnQuestEntities(quest.objects.value, variant, sections) + } + } + + private fun setSectionOnQuestEntities( + entities: List>, + variant: AreaVariantModel, + sections: List, + ) { + entities.forEach { entity -> + if (entity.areaId == variant.area.id) { + val section = sections.find { it.id == entity.sectionId.value } + + if (section == null) { + logger.warn { "Section ${entity.sectionId.value} not found." } + entity.setSectionInitialized() + } else { + entity.setSection(section, keepRelativeTransform = true) + } + } + } + } } diff --git a/web/src/test/kotlin/world/phantasmal/web/core/undo/UndoStackTests.kt b/web/src/test/kotlin/world/phantasmal/web/core/undo/UndoStackTests.kt index ddcb2c6e..995bb723 100644 --- a/web/src/test/kotlin/world/phantasmal/web/core/undo/UndoStackTests.kt +++ b/web/src/test/kotlin/world/phantasmal/web/core/undo/UndoStackTests.kt @@ -9,7 +9,7 @@ import kotlin.test.assertTrue class UndoStackTests : WebTestSuite { @Test - fun simple_properties_and_invariants() { + fun simple_properties_and_invariants() = test { val stack = UndoStack(UndoManager()) assertFalse(stack.canUndo.value) @@ -35,7 +35,7 @@ class UndoStackTests : WebTestSuite { } @Test - fun undo() { + fun undo() = test { val stack = UndoStack(UndoManager()) var value = 3 @@ -56,7 +56,7 @@ class UndoStackTests : WebTestSuite { } @Test - fun redo() { + fun redo() = test { val stack = UndoStack(UndoManager()) var value = 3 @@ -80,7 +80,7 @@ class UndoStackTests : WebTestSuite { } @Test - fun push_then_undo_then_push_again() { + fun push_then_undo_then_push_again() = test { val stack = UndoStack(UndoManager()) var value = 3 diff --git a/web/src/test/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModelTests.kt b/web/src/test/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModelTests.kt index 3e3bcdec..9a39e8df 100644 --- a/web/src/test/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModelTests.kt +++ b/web/src/test/kotlin/world/phantasmal/web/questEditor/models/QuestEntityModelTests.kt @@ -19,13 +19,13 @@ class QuestEntityModelTests : WebTestSuite { assertTrue(entity.position.value.equals(entity.worldPosition.value)) - // When section is initialized, relative position stays the same and world position changes. - entity.initializeSection(SectionModel( + // Initialize section and keep relative position the same so world position changes. + entity.setSection(SectionModel( 20, Vector3(7.0, 7.0, 7.0), euler(.0, .0, .0), components.areaStore.getVariant(Episode.I, 0, 0)!!, - )) + ), keepRelativeTransform = true) assertEquals(5.0, entity.position.value.x) assertEquals(5.0, entity.position.value.y) diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Input.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Input.kt index bc1ca788..737b257d 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Input.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Input.kt @@ -38,12 +38,6 @@ abstract class Input( onchange = { callOnChange(this) } - onkeydown = { e -> - if (e.key == "Enter") { - callOnChange(this) - } - } - interceptInputElement(this) observe(this@Input.value) { diff --git a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Widget.kt b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Widget.kt index 98ea334c..29591ce1 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/widgets/Widget.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/widgets/Widget.kt @@ -1,7 +1,9 @@ package world.phantasmal.webui.widgets import kotlinx.browser.document +import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.launch import org.w3c.dom.Element import org.w3c.dom.HTMLElement import org.w3c.dom.HTMLStyleElement @@ -203,6 +205,10 @@ abstract class Widget( addDisposable(disposablePointerDrag(onPointerDown, onPointerMove, onPointerUp)) } + protected fun launch(block: suspend CoroutineScope.() -> Unit) { + scope.launch(block = block) + } + companion object { private val STYLE_EL by lazy { val el = document.createElement("style") as HTMLStyleElement