From 078cf7fb2e6b5ffb0edb5d182ee56a613224320d Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Thu, 25 Mar 2021 16:55:31 +0100 Subject: [PATCH] Ported viewer model, body and section_id URL parameters. --- .../assetsGeneration/ephinea/UpdateDrops.kt | 2 +- .../phantasmal/web/application/Application.kt | 2 +- .../phantasmal/web/core/stores/UiStore.kt | 71 +++++++++++++++--- .../world/phantasmal/web/viewer/Viewer.kt | 4 +- .../web/viewer/stores/ViewerStore.kt | 73 ++++++++++++++++++- .../phantasmal/web/viewer/ViewerTests.kt | 2 +- 6 files changed, 136 insertions(+), 18 deletions(-) diff --git a/web/assets-generation/src/main/kotlin/world/phantasmal/web/assetsGeneration/ephinea/UpdateDrops.kt b/web/assets-generation/src/main/kotlin/world/phantasmal/web/assetsGeneration/ephinea/UpdateDrops.kt index 7762fcd2..8bb2cc8d 100644 --- a/web/assets-generation/src/main/kotlin/world/phantasmal/web/assetsGeneration/ephinea/UpdateDrops.kt +++ b/web/assets-generation/src/main/kotlin/world/phantasmal/web/assetsGeneration/ephinea/UpdateDrops.kt @@ -37,7 +37,7 @@ private fun download( difficulty: Difficulty, difficultyUrl: String, enemyDrops: MutableList, - boxDrops: MutableList, + @Suppress("UNUSED_PARAMETER") boxDrops: MutableList, ) { val doc = Jsoup.connect("https://ephinea.pioneer2.net/drop-charts/${difficultyUrl}/").get() diff --git a/web/src/main/kotlin/world/phantasmal/web/application/Application.kt b/web/src/main/kotlin/world/phantasmal/web/application/Application.kt index 3a58629d..f9bb1a9b 100644 --- a/web/src/main/kotlin/world/phantasmal/web/application/Application.kt +++ b/web/src/main/kotlin/world/phantasmal/web/application/Application.kt @@ -49,7 +49,7 @@ class Application( // The various tools Phantasmal World consists of. val tools: List = listOf( - addDisposable(Viewer(assetLoader, createThreeRenderer)), + addDisposable(Viewer(assetLoader, uiStore, createThreeRenderer)), addDisposable(QuestEditor(assetLoader, uiStore, createThreeRenderer)), addDisposable(HuntOptimizer(assetLoader, uiStore)), ) diff --git a/web/src/main/kotlin/world/phantasmal/web/core/stores/UiStore.kt b/web/src/main/kotlin/world/phantasmal/web/core/stores/UiStore.kt index 6db7a46f..4f455d5e 100644 --- a/web/src/main/kotlin/world/phantasmal/web/core/stores/UiStore.kt +++ b/web/src/main/kotlin/world/phantasmal/web/core/stores/UiStore.kt @@ -4,6 +4,7 @@ import kotlinx.browser.window import kotlinx.coroutines.launch import org.w3c.dom.events.KeyboardEvent import world.phantasmal.core.disposable.Disposable +import world.phantasmal.core.disposable.Disposer import world.phantasmal.core.disposable.disposable import world.phantasmal.observable.value.MutableVal import world.phantasmal.observable.value.Val @@ -32,7 +33,8 @@ class UiStore(private val applicationUrl: ApplicationUrl) : Store() { * Maps full paths to maps of parameters and their values. In other words we keep track of * parameter values per [applicationUrl]. */ - private val parameters: MutableMap> = mutableMapOf() + private val parameters: MutableMap>> = + mutableMapOf() private val globalKeyDownHandlers: MutableMap Unit> = mutableMapOf() @@ -43,7 +45,7 @@ class UiStore(private val applicationUrl: ApplicationUrl) : Store() { */ private val features: MutableSet = mutableSetOf() - val tools: List = PwToolType.values().toList() + private val tools: List = PwToolType.values().toList() /** * The default tool that is loaded. @@ -104,6 +106,53 @@ class UiStore(private val applicationUrl: ApplicationUrl) : Store() { } } + fun registerParameter( + tool: PwToolType, + path: String, + parameter: String, + setInitialValue: (String?) -> Unit, + value: Val, + onChange: (String?) -> Unit, + ): Disposable { + require(parameter !== FEATURES_PARAM) { + "$FEATURES_PARAM can't be set because it is a global parameter." + } + + val pathParams = parameters.getOrPut("/${tool.slug}$path", ::mutableMapOf) + val param = pathParams.getOrPut(parameter) { mutableVal(null) } + + setInitialValue(param.value) + + value.value.let { v -> + if (v != param.value) { + setParameter(tool, path, param, v, replaceUrl = true) + } + } + + return Disposer( + value.observe { + if (it.value != param.value) { + setParameter(tool, path, param, it.value, replaceUrl = false) + } + }, + param.observe { onChange(it.value) }, + ) + } + + private fun setParameter( + tool: PwToolType, + path: String, + parameter: MutableVal, + value: String?, + replaceUrl: Boolean, + ) { + parameter.value = value + + if (this.currentTool.value == tool && this.path.value == path) { + updateApplicationUrl(tool, path, replaceUrl) + } + } + fun onGlobalKeyDown( tool: PwToolType, binding: String, @@ -135,21 +184,19 @@ class UiStore(private val applicationUrl: ApplicationUrl) : Store() { val path = if (secondSlashIdx == -1) "" else fullPath.substring(secondSlashIdx) if (paramsStr != null) { - val params = mutableMapOf() + val params = parameters.getOrPut(fullPath, ::mutableMapOf) for (p in paramsStr.split("&")) { val (param, value) = p.split("=", limit = 2) - if (param == "features") { + if (param == FEATURES_PARAM) { for (feature in value.split(",")) { features.add(feature) } } else { - params[param] = value + params.getOrPut(param) { mutableVal(value) }.value = value } } - - parameters[fullPath] = params } val actualTool = tool ?: defaultTool @@ -167,11 +214,14 @@ class UiStore(private val applicationUrl: ApplicationUrl) : Store() { private fun updateApplicationUrl(tool: PwToolType, path: String, replace: Boolean) { val fullPath = "/${tool.slug}${path}" - val params: MutableMap = - parameters[fullPath]?.let { HashMap(it) } ?: mutableMapOf() + val params = mutableMapOf() + + parameters[fullPath]?.forEach { (k, v) -> + v.value?.let { params[k] = it } + } if (features.isNotEmpty()) { - params["features"] = features.joinToString(",") + params[FEATURES_PARAM] = features.joinToString(",") } val paramStr = @@ -210,6 +260,7 @@ class UiStore(private val applicationUrl: ApplicationUrl) : Store() { } companion object { + private const val FEATURES_PARAM = "features" private val SLUG_TO_PW_TOOL: Map = PwToolType.values().map { it.slug to it }.toMap() } diff --git a/web/src/main/kotlin/world/phantasmal/web/viewer/Viewer.kt b/web/src/main/kotlin/world/phantasmal/web/viewer/Viewer.kt index 06c0bb43..a02481db 100644 --- a/web/src/main/kotlin/world/phantasmal/web/viewer/Viewer.kt +++ b/web/src/main/kotlin/world/phantasmal/web/viewer/Viewer.kt @@ -5,6 +5,7 @@ import world.phantasmal.web.core.PwTool import world.phantasmal.web.core.PwToolType import world.phantasmal.web.core.loading.AssetLoader import world.phantasmal.web.core.rendering.DisposableThreeRenderer +import world.phantasmal.web.core.stores.UiStore import world.phantasmal.web.core.widgets.RendererWidget import world.phantasmal.web.viewer.controllers.CharacterClassOptionsController import world.phantasmal.web.viewer.controllers.ViewerController @@ -21,6 +22,7 @@ import world.phantasmal.webui.widgets.Widget class Viewer( private val assetLoader: AssetLoader, + private val uiStore: UiStore, private val createThreeRenderer: (HTMLCanvasElement) -> DisposableThreeRenderer, ) : DisposableContainer(), PwTool { override val toolType = PwToolType.Viewer @@ -30,7 +32,7 @@ class Viewer( val characterClassAssetLoader = addDisposable(CharacterClassAssetLoader(assetLoader)) // Stores - val viewerStore = addDisposable(ViewerStore(characterClassAssetLoader)) + val viewerStore = addDisposable(ViewerStore(characterClassAssetLoader, uiStore)) // Controllers val viewerController = addDisposable(ViewerController(viewerStore)) 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 fe667ab9..d2369d10 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 @@ -11,6 +11,8 @@ import world.phantasmal.observable.value.Val import world.phantasmal.observable.value.list.ListVal import world.phantasmal.observable.value.list.mutableListVal import world.phantasmal.observable.value.mutableVal +import world.phantasmal.web.core.PwToolType +import world.phantasmal.web.core.stores.UiStore import world.phantasmal.web.shared.dto.SectionId import world.phantasmal.web.viewer.loading.CharacterClassAssetLoader import world.phantasmal.web.viewer.models.CharacterClass @@ -18,13 +20,15 @@ import world.phantasmal.webui.stores.Store private val logger = KotlinLogging.logger {} -class ViewerStore(private val assetLoader: CharacterClassAssetLoader) : Store() { +class ViewerStore( + private val assetLoader: CharacterClassAssetLoader, + uiStore: UiStore, +) : Store() { private val _currentNinjaObject = mutableVal?>(null) private val _currentTextures = mutableListVal() - private val _currentCharacterClass = mutableVal(CharacterClass.VALUES.random()) + private val _currentCharacterClass = mutableVal(null) private val _currentSectionId = mutableVal(SectionId.VALUES.random()) - private val _currentBody = - mutableVal((0 until _currentCharacterClass.value!!.bodyStyleCount).random()) + private val _currentBody = mutableVal(0) private val _currentNinjaMotion = mutableVal(null) // Settings @@ -41,6 +45,61 @@ class ViewerStore(private val assetLoader: CharacterClassAssetLoader) : Store() val showSkeleton: Val = _showSkeleton init { + addDisposables( + uiStore.registerParameter( + PwToolType.Viewer, + path = "", + MODEL_PARAM, + setInitialValue = { initialValue -> + _currentCharacterClass.value = + CharacterClass.VALUES.find { it.slug == initialValue } + ?: CharacterClass.VALUES.random() + }, + value = currentCharacterClass.map { it?.slug }, + onChange = { newValue -> + scope.launch { + setCurrentCharacterClass(CharacterClass.VALUES.find { it.slug == newValue }) + } + }, + ), + + uiStore.registerParameter( + PwToolType.Viewer, + path = "", + BODY_PARAM, + setInitialValue = { initialValue -> + val maxBody = _currentCharacterClass.value?.bodyStyleCount?.minus(1) ?: 0 + _currentBody.value = + initialValue?.toIntOrNull()?.takeIf { it <= maxBody }?.minus(1) + ?: (0..maxBody).random() + }, + value = currentBody.map { (it + 1).toString() }, + onChange = { newValue -> + scope.launch { + setCurrentBody(newValue?.toIntOrNull()?.minus(1) ?: 0) + } + }, + ), + + uiStore.registerParameter( + PwToolType.Viewer, + path = "", + SECTION_ID_PARAM, + setInitialValue = { initialValue -> + _currentSectionId.value = + initialValue?.let(SectionId::valueOf) ?: SectionId.VALUES.random() + }, + value = currentSectionId.map { it.name }, + onChange = { newValue -> + scope.launch { + setCurrentSectionId( + newValue?.let(SectionId::valueOf) ?: SectionId.VALUES.random() + ) + } + }, + ), + ) + scope.launch(Dispatchers.Default) { loadCharacterClassNinjaObject(clearAnimation = true) } @@ -119,4 +178,10 @@ class ViewerStore(private val assetLoader: CharacterClassAssetLoader) : Store() } } } + + companion object { + private const val MODEL_PARAM = "model" + private const val BODY_PARAM = "body" + private const val SECTION_ID_PARAM = "section_id" + } } diff --git a/web/src/test/kotlin/world/phantasmal/web/viewer/ViewerTests.kt b/web/src/test/kotlin/world/phantasmal/web/viewer/ViewerTests.kt index 8143e3c2..2781deda 100644 --- a/web/src/test/kotlin/world/phantasmal/web/viewer/ViewerTests.kt +++ b/web/src/test/kotlin/world/phantasmal/web/viewer/ViewerTests.kt @@ -11,7 +11,7 @@ class ViewerTests : WebTestSuite { components.applicationUrl = TestApplicationUrl("/${PwToolType.Viewer}") val viewer = disposer.add( - Viewer(components.assetLoader, components.createThreeRenderer) + Viewer(components.assetLoader, components.uiStore, components.createThreeRenderer) ) disposer.add(viewer.initialize()) }