From 9653a982c02214b8504c422c8cd41b899d7ea6f0 Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Sun, 28 Nov 2021 19:11:16 +0100 Subject: [PATCH] Improvements to JS-specific code and documentation and added a unit test. --- .../core/disposable/TrackedDisposable.kt | 3 + .../world/phantasmal/core/unsafe/UnsafeMap.kt | 16 ++++- .../world/phantasmal/core/unsafe/UnsafeSet.kt | 38 +++++++++++ .../jsMain/kotlin/world/phantasmal/core/Js.kt | 33 +++++----- .../world/phantasmal/core/unsafe/UnsafeSet.kt | 15 +++++ .../world/phantasmal/core/unsafe/UnsafeMap.kt | 6 +- .../world/phantasmal/core/unsafe/UnsafeSet.kt | 27 ++++++++ .../psolib/asm/AsmTokenizationTests.kt | 23 +++++++ .../conversion/NinjaGeometryConversion.kt | 32 +++------ .../stores/HuntOptimizerStore.kt | 19 +++--- .../questEditor/loading/AreaAssetLoader.kt | 66 +++++++++---------- .../webui/dom/HTMLElementSizeCell.kt | 32 ++++----- 12 files changed, 204 insertions(+), 106 deletions(-) create mode 100644 core/src/commonMain/kotlin/world/phantasmal/core/unsafe/UnsafeSet.kt create mode 100644 core/src/jsMain/kotlin/world/phantasmal/core/unsafe/UnsafeSet.kt create mode 100644 core/src/jvmMain/kotlin/world/phantasmal/core/unsafe/UnsafeSet.kt diff --git a/core/src/commonMain/kotlin/world/phantasmal/core/disposable/TrackedDisposable.kt b/core/src/commonMain/kotlin/world/phantasmal/core/disposable/TrackedDisposable.kt index 4ec5a244..47431196 100644 --- a/core/src/commonMain/kotlin/world/phantasmal/core/disposable/TrackedDisposable.kt +++ b/core/src/commonMain/kotlin/world/phantasmal/core/disposable/TrackedDisposable.kt @@ -3,12 +3,15 @@ package world.phantasmal.core.disposable /** * A global count is kept of all undisposed instances of this class. * This count can be used to find memory leaks. + * + * Tracking is not thread-safe. */ abstract class TrackedDisposable : Disposable { var disposed = false private set init { + // Suppress this warning, because track simply adds this disposable to a set at this point. @Suppress("LeakingThis") track(this) } diff --git a/core/src/commonMain/kotlin/world/phantasmal/core/unsafe/UnsafeMap.kt b/core/src/commonMain/kotlin/world/phantasmal/core/unsafe/UnsafeMap.kt index fbad183e..89b99653 100644 --- a/core/src/commonMain/kotlin/world/phantasmal/core/unsafe/UnsafeMap.kt +++ b/core/src/commonMain/kotlin/world/phantasmal/core/unsafe/UnsafeMap.kt @@ -2,8 +2,22 @@ package world.phantasmal.core.unsafe /** * Map optimized for JS (it compiles to the built-in Map). + * * In JS, keys are compared by reference, equals and hashCode are NOT invoked. On JVM, equals and * hashCode ARE used. + * + * DO NOT USE THIS UNLESS ALL THE FOLLOWING REQUIREMENTS ARE MET: + * + * 1. It improves performance substantially. + * If it doesn't improve performance by a very noticeable amount, it's not worth the risk. + * + * 2. It's only used internally. + * E.g. in a private property which no other code can access and misuse accidentally. This way + * only a small part of the code can contain hard to discover errors. + * + * 3. The keys used do not require equals or hashCode to be called in JS. + * E.g. Int, String, objects which you consider equal if and only if they are the exact same + * instance. */ expect class UnsafeMap() { fun get(key: K): V? @@ -13,7 +27,7 @@ expect class UnsafeMap() { fun delete(key: K): Boolean } -fun UnsafeMap.getOrPut(key: K, default: () -> V): V { +inline fun UnsafeMap.getOrPut(key: K, default: () -> V): V { var value = get(key) if (value == null) { diff --git a/core/src/commonMain/kotlin/world/phantasmal/core/unsafe/UnsafeSet.kt b/core/src/commonMain/kotlin/world/phantasmal/core/unsafe/UnsafeSet.kt new file mode 100644 index 00000000..3f029642 --- /dev/null +++ b/core/src/commonMain/kotlin/world/phantasmal/core/unsafe/UnsafeSet.kt @@ -0,0 +1,38 @@ +package world.phantasmal.core.unsafe + +/** + * Set optimized for JS (it compiles to the built-in Set). + * + * In JS, values are compared by reference, equals and hashCode are NOT invoked. On JVM, equals and + * hashCode ARE used. + * + * DO NOT USE THIS UNLESS ALL THE FOLLOWING REQUIREMENTS ARE MET: + * + * 1. It improves performance substantially. + * If it doesn't improve performance by a very noticeable amount, it's not worth the risk. + * + * 2. It's only used internally. + * E.g. in a private property which no other code can access and misuse accidentally. This way + * only a small part of the code can contain hard to discover errors. + * + * 3. The values used do not require equals or hashCode to be called in JS. + * E.g. Int, String, objects which you consider equal if and only if they are the exact same + * instance. + */ +expect class UnsafeSet { + constructor() + constructor(values: Array) + + val size: Int + + fun add(value: T): UnsafeSet + fun clear() + fun delete(value: T): Boolean + fun has(value: T): Boolean + fun forEach(callback: (value: T) -> Unit) +} + +/** + * See the disclaimer at [UnsafeSet]. + */ +fun unsafeSetOf(vararg values: T): UnsafeSet = UnsafeSet(values) diff --git a/core/src/jsMain/kotlin/world/phantasmal/core/Js.kt b/core/src/jsMain/kotlin/world/phantasmal/core/Js.kt index 0caaa586..923d4ca1 100644 --- a/core/src/jsMain/kotlin/world/phantasmal/core/Js.kt +++ b/core/src/jsMain/kotlin/world/phantasmal/core/Js.kt @@ -2,6 +2,9 @@ package world.phantasmal.core +import org.w3c.dom.DOMRectReadOnly +import org.w3c.dom.Element + external interface JsArray { val length: Int @@ -43,23 +46,19 @@ inline val JsPair<*, T>.second: T get() = asDynamic()[1].unsafeCast() inline operator fun JsPair.component1(): T = first inline operator fun JsPair<*, T>.component2(): T = second -@Suppress("UNUSED_PARAMETER") -inline fun objectEntries(jsObject: dynamic): Array> = - js("Object.entries(jsObject)").unsafeCast>>() - -external interface JsSet { - val size: Int - - fun add(value: T): JsSet - fun clear() - fun delete(value: T): Boolean - fun has(value: T): Boolean - fun forEach(callback: (value: T) -> Unit) +@JsName("Object") +external class JsObject { + companion object { + fun entries(jsObject: dynamic): Array> + } } -inline fun emptyJsSet(): JsSet = - js("new Set()").unsafeCast>() +external class ResizeObserver(callback: (entries: Array) -> Unit) { + fun observe(target: Element) + fun unobserve(target: Element) + fun disconnect() +} -@Suppress("UNUSED_PARAMETER") -inline fun jsSetOf(vararg values: T): JsSet = - js("new Set(values)").unsafeCast>() +external interface ResizeObserverEntry { + val contentRect: DOMRectReadOnly +} diff --git a/core/src/jsMain/kotlin/world/phantasmal/core/unsafe/UnsafeSet.kt b/core/src/jsMain/kotlin/world/phantasmal/core/unsafe/UnsafeSet.kt new file mode 100644 index 00000000..e20fad0e --- /dev/null +++ b/core/src/jsMain/kotlin/world/phantasmal/core/unsafe/UnsafeSet.kt @@ -0,0 +1,15 @@ +package world.phantasmal.core.unsafe + +@JsName("Set") +actual external class UnsafeSet { + actual constructor() + actual constructor(values: Array) + + actual val size: Int + + actual fun add(value: T): UnsafeSet + actual fun clear() + actual fun delete(value: T): Boolean + actual fun has(value: T): Boolean + actual fun forEach(callback: (value: T) -> Unit) +} diff --git a/core/src/jvmMain/kotlin/world/phantasmal/core/unsafe/UnsafeMap.kt b/core/src/jvmMain/kotlin/world/phantasmal/core/unsafe/UnsafeMap.kt index 5335b0ae..4806166d 100644 --- a/core/src/jvmMain/kotlin/world/phantasmal/core/unsafe/UnsafeMap.kt +++ b/core/src/jvmMain/kotlin/world/phantasmal/core/unsafe/UnsafeMap.kt @@ -1,14 +1,16 @@ package world.phantasmal.core.unsafe actual class UnsafeMap { - private val map = HashMap() + private val map = LinkedHashMap() actual fun get(key: K): V? = map[key] actual fun has(key: K): Boolean = key in map actual fun forEach(callback: (value: V, key: K) -> Unit) { - map.forEach { (k, v) -> callback(v, k) } + for ((k, v) in map) { + callback(v, k) + } } actual fun set(key: K, value: V) { diff --git a/core/src/jvmMain/kotlin/world/phantasmal/core/unsafe/UnsafeSet.kt b/core/src/jvmMain/kotlin/world/phantasmal/core/unsafe/UnsafeSet.kt new file mode 100644 index 00000000..35000cf5 --- /dev/null +++ b/core/src/jvmMain/kotlin/world/phantasmal/core/unsafe/UnsafeSet.kt @@ -0,0 +1,27 @@ +package world.phantasmal.core.unsafe + +actual class UnsafeSet(private val set: LinkedHashSet) { + actual constructor() : this(LinkedHashSet()) + actual constructor(values: Array) : this(LinkedHashSet().apply { addAll(values) }) + + actual val size: Int get() = set.size + + actual fun add(value: T): UnsafeSet { + set.add(value) + return this + } + + actual fun clear() { + set.clear() + } + + actual fun delete(value: T): Boolean = set.remove(value) + + actual fun has(value: T): Boolean = value in set + + actual fun forEach(callback: (value: T) -> Unit) { + for (v in set) { + callback(v) + } + } +} diff --git a/psolib/src/commonTest/kotlin/world/phantasmal/psolib/asm/AsmTokenizationTests.kt b/psolib/src/commonTest/kotlin/world/phantasmal/psolib/asm/AsmTokenizationTests.kt index f78b7940..4f3d1129 100644 --- a/psolib/src/commonTest/kotlin/world/phantasmal/psolib/asm/AsmTokenizationTests.kt +++ b/psolib/src/commonTest/kotlin/world/phantasmal/psolib/asm/AsmTokenizationTests.kt @@ -107,4 +107,27 @@ class AsmTokenizationTests : LibTestSuite { assertEquals(len, this.len) assertFalse(nextToken()) } + + @Test + fun valid_identifiers_are_parsed_as_Ident_tokens() { + val tokenizer = LineTokenizer() + + tokenizer.testIdent(" opcode_mnemonic ", "opcode_mnemonic", col = 3, len = 15) + tokenizer.testIdent("inst_1", "inst_1", col = 1, len = 6) + } + + private fun LineTokenizer.testIdent( + line: String, + value: String, + col: Int, + len: Int, + ) { + tokenize(line) + assertTrue(nextToken()) + assertEquals(Token.Ident, this.type) + assertEquals(value, this.strValue) + assertEquals(col, this.col) + assertEquals(len, this.len) + assertFalse(nextToken()) + } } diff --git a/web/src/main/kotlin/world/phantasmal/web/core/rendering/conversion/NinjaGeometryConversion.kt b/web/src/main/kotlin/world/phantasmal/web/core/rendering/conversion/NinjaGeometryConversion.kt index 790843e5..439f284e 100644 --- a/web/src/main/kotlin/world/phantasmal/web/core/rendering/conversion/NinjaGeometryConversion.kt +++ b/web/src/main/kotlin/world/phantasmal/web/core/rendering/conversion/NinjaGeometryConversion.kt @@ -14,21 +14,8 @@ import world.phantasmal.web.core.dot import world.phantasmal.web.core.toQuaternion import world.phantasmal.web.externals.three.* import world.phantasmal.webui.obj -import kotlin.collections.List -import kotlin.collections.MutableList import kotlin.collections.component1 import kotlin.collections.component2 -import kotlin.collections.forEach -import kotlin.collections.forEachIndexed -import kotlin.collections.getOrPut -import kotlin.collections.indices -import kotlin.collections.iterator -import kotlin.collections.last -import kotlin.collections.map -import kotlin.collections.mutableListOf -import kotlin.collections.mutableMapOf -import kotlin.collections.sum -import kotlin.collections.withIndex private val logger = KotlinLogging.logger {} @@ -92,10 +79,7 @@ private val tmpNormal = Vector3() private val tmpVec = Vector3() private val tmpNormalMatrix = Matrix3() -interface AreaObjectUserData { - var sectionId: Int - var areaObject: AreaObject -} +class AreaObjectUserData(val sectionId: Int, val areaObject: AreaObject) fun ninjaObjectToMesh( ninjaObject: NinjaObject<*, *>, @@ -141,6 +125,7 @@ fun ninjaObjectToMeshBuilder( NinjaToMeshConverter(builder).convert(ninjaObject) } +/** The returned group is not copyable because it contains non-serializable user data. */ fun renderGeometryToGroup( renderGeometry: AreaGeometry, textures: List, @@ -221,6 +206,7 @@ fun AreaObject.fingerPrint(): String = append(radius.toRawBits().toUInt().toString(36)) } +/** The returned mesh is not copyable because it contains non-serializable user data. */ private fun areaObjectToMesh( textures: List, textureCache: UnsafeMap, @@ -230,9 +216,10 @@ private fun areaObjectToMesh( areaObj: AreaObject, processMesh: (AreaSection, AreaObject, Mesh) -> Unit, ): Mesh { - var mesh = meshCache.get(areaObj.xjObject) + val cachedMesh = meshCache.get(areaObj.xjObject) + val mesh: Mesh - if (mesh == null) { + if (cachedMesh == null) { val builder = MeshBuilder(textures, textureCache) ninjaObjectToMeshBuilder(areaObj.xjObject, builder) @@ -248,13 +235,10 @@ private fun areaObjectToMesh( } else { // If we already have a mesh for this XjObject, make a copy and reuse the existing buffer // geometry and materials. - mesh = Mesh(mesh.geometry, mesh.material.unsafeCast>()) + mesh = Mesh(cachedMesh.geometry, cachedMesh.material.unsafeCast>()) } - val userData = mesh.userData.unsafeCast() - userData.sectionId = section.id - userData.areaObject = areaObj - + mesh.userData = AreaObjectUserData(section.id, areaObj) mesh.position.setFromVec3(section.position) mesh.rotation.setFromVec3(section.rotation) mesh.updateMatrixWorld() diff --git a/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/stores/HuntOptimizerStore.kt b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/stores/HuntOptimizerStore.kt index d27750d7..34af549c 100644 --- a/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/stores/HuntOptimizerStore.kt +++ b/web/src/main/kotlin/world/phantasmal/web/huntOptimizer/stores/HuntOptimizerStore.kt @@ -4,13 +4,16 @@ import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch import kotlinx.coroutines.withContext import mu.KotlinLogging -import world.phantasmal.core.* +import world.phantasmal.core.JsObject +import world.phantasmal.core.component1 +import world.phantasmal.core.component2 import world.phantasmal.core.unsafe.UnsafeMap -import world.phantasmal.psolib.fileFormats.quest.NpcType +import world.phantasmal.core.unsafe.UnsafeSet import world.phantasmal.observable.cell.Cell import world.phantasmal.observable.cell.list.ListCell import world.phantasmal.observable.cell.list.mutableListCell import world.phantasmal.observable.cell.mutableCell +import world.phantasmal.psolib.fileFormats.quest.NpcType import world.phantasmal.web.core.models.Server import world.phantasmal.web.core.stores.EnemyDropTable import world.phantasmal.web.core.stores.ItemDropStore @@ -138,7 +141,7 @@ class HuntOptimizerStore( // Each variable has a matching FullMethod. val fullMethods = UnsafeMap() - val wantedItemTypeIds = emptyJsSet() + val wantedItemTypeIds = UnsafeSet() for (wanted in filteredWantedItems) { wantedItemTypeIds.add(wanted.itemType.id) @@ -191,7 +194,7 @@ class HuntOptimizerStore( private fun createFullMethods( dropTable: EnemyDropTable, - wantedItemTypeIds: JsSet, + wantedItemTypeIds: UnsafeSet, method: HuntMethodModel, defaultCounts: UnsafeMap, splitPanArms: Boolean, @@ -260,7 +263,7 @@ class HuntOptimizerStore( } private fun solve( - wantedItemTypeIds: JsSet, + wantedItemTypeIds: UnsafeSet, constraints: dynamic, variables: dynamic, fullMethods: UnsafeMap, @@ -277,13 +280,13 @@ class HuntOptimizerStore( } // Loop over the entries in result, ignore standard properties that aren't variables. - return objectEntries(result).mapNotNull { (variableName, runsOrOther) -> + return JsObject.entries(result).mapNotNull { (variableName, runsOrOther) -> fullMethods.get(variableName)?.let { fullMethod -> val runs = runsOrOther as Double val variable = variables[variableName] val itemTypeIdToCount: Map = - objectEntries(variable) + JsObject.entries(variable) .mapNotNull { (itemTypeIdStr, expectedAmount) -> itemTypeIdStr.toIntOrNull()?.let { itemTypeId -> if (wantedItemTypeIds.has(itemTypeId)) { @@ -318,7 +321,7 @@ class HuntOptimizerStore( if (v == null) { matchFound = false } else { - for ((itemName, expectedAmount) in objectEntries(variable)) { + for ((itemName, expectedAmount) in JsObject.entries(variable)) { if (v[itemName] != expectedAmount.unsafeCast()) { matchFound = false break 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 640126a9..f66f268a 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 @@ -2,6 +2,8 @@ package world.phantasmal.web.questEditor.loading import org.khronos.webgl.ArrayBuffer import world.phantasmal.core.* +import world.phantasmal.core.unsafe.UnsafeSet +import world.phantasmal.core.unsafe.unsafeSetOf import world.phantasmal.psolib.Endianness import world.phantasmal.psolib.Episode import world.phantasmal.psolib.cursor.cursor @@ -20,9 +22,7 @@ import world.phantasmal.webui.DisposableContainer import kotlin.math.PI import kotlin.math.cos -interface AreaUserData { - var section: SectionModel? -} +class AreaUserData(val section: SectionModel?) /** * Loads and caches area assets. @@ -65,9 +65,11 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine suspend fun loadSections(episode: Episode, areaVariant: AreaVariantModel): List = cache.get(EpisodeAndAreaVariant(episode, areaVariant)).sections + /** The returned object is not copyable because it contains non-serializable user data. */ suspend fun loadRenderGeometry(episode: Episode, areaVariant: AreaVariantModel): Object3D = cache.get(EpisodeAndAreaVariant(episode, areaVariant)).renderGeometry + /** The returned object is not copyable because it contains non-serializable user data. */ suspend fun loadCollisionGeometry( episode: Episode, areaVariant: AreaVariantModel, @@ -94,14 +96,14 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine tmpIntersections.asJsArray().splice(0) val intersection1 = raycaster .intersectObject(renderGeom, true, tmpIntersections) - .find { (it.`object`.userData.unsafeCast()).section != null } + .find { it.`object`.userData is AreaUserData } // Cast a ray upward from the center of the section. raycaster.set(origin, UP) tmpIntersections.asJsArray().splice(0) val intersection2 = raycaster .intersectObject(renderGeom, true, tmpIntersections) - .find { (it.`object`.userData.unsafeCast()).section != null } + .find { it.`object`.userData is AreaUserData } // Choose the nearest intersection if we have 2. val intersection = @@ -113,9 +115,7 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine } if (intersection != null) { - val cud = collisionMesh.userData.unsafeCast() - val rud = intersection.`object`.userData.unsafeCast() - cud.section = rud.section + collisionMesh.userData = intersection.`object`.userData } } } @@ -194,7 +194,7 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine ) } - (mesh.userData.unsafeCast()).section = sectionModel + mesh.userData = AreaUserData(sectionModel) } } @@ -241,12 +241,12 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine * regular geometry. Might not be necessary anymore once order-independent rendering is * implemented. */ - private val renderOnTopTextures: JsSet = emptyJsSet(), + private val renderOnTopTextures: UnsafeSet = UnsafeSet(), /** * Set of [AreaObject] finger prints. * These objects should be hidden because they cover floors and other useful geometry. */ - private val hiddenObjects: JsSet = emptyJsSet(), + private val hiddenObjects: UnsafeSet = UnsafeSet(), ) { fun shouldRenderOnTop(obj: XjObject): Boolean { obj.model?.meshes?.let { meshes -> @@ -330,7 +330,7 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine private val MANUAL_FIXES: Map, Fix> = mutableMapOf( // Pioneer 2 Pair(Episode.I, 0) to Fix( - renderOnTopTextures = jsSetOf( + renderOnTopTextures = unsafeSetOf( 20, 40, 41, @@ -360,7 +360,7 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine 234, 243, ), - hiddenObjects = jsSetOf( + hiddenObjects = unsafeSetOf( "s_m_0_6a_d_iu5sg6", "s_m_0_4b_7_ioh738", "s_k_0_1s_3_irasis", @@ -398,9 +398,9 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine ), // Forest 1 Pair(Episode.I, 1) to Fix( - renderOnTopTextures = jsSetOf(12, 24, 41), + renderOnTopTextures = unsafeSetOf(12, 24, 41), // Hiding trees and their shadows doesn't seem to improve things. -// hiddenObjects = jsSetOf( +// hiddenObjects = unsafeSetOf( // // Trees // "s_n_0_2i_2_im1teq", // "s_n_0_2i_2_im1tep", @@ -425,8 +425,8 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine ), // Cave 1 Pair(Episode.I, 3) to Fix( - renderOnTopTextures = jsSetOf(89), - hiddenObjects = jsSetOf( + renderOnTopTextures = unsafeSetOf(89), + hiddenObjects = unsafeSetOf( "s_n_0_8_1_iqrqjj", "s_i_0_b5_1_is7ajh", "s_n_0_24_1_in5ce2", @@ -463,7 +463,7 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine ), // Cave 2 Pair(Episode.I, 4) to Fix( - hiddenObjects = jsSetOf( + hiddenObjects = unsafeSetOf( "s_n_0_4j_1_irf90i", "s_n_0_5i_1_iqqrft", "s_n_0_g_1_iipv9r", @@ -490,7 +490,7 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine ), // Cave 3 Pair(Episode.I, 5) to Fix( - hiddenObjects = jsSetOf( + hiddenObjects = unsafeSetOf( "s_n_0_2o_5_inun1c", "s_n_0_5y_2_ipyair", "s_n_0_6s_1_ineank", @@ -540,7 +540,7 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine ), // Mine 1 Pair(Episode.I, 6) to Fix( - hiddenObjects = jsSetOf( + hiddenObjects = unsafeSetOf( "s_n_0_2e_2_iqfpg8", "s_n_0_d_1_iruof6", "s_n_0_o_1_im9ta5", @@ -628,8 +628,8 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine ), // Mine 2 Pair(Episode.I, 7) to Fix( - renderOnTopTextures = jsSetOf(0, 1, 7, 17, 23, 56, 57, 58, 59, 60, 83), - hiddenObjects = jsSetOf( + renderOnTopTextures = unsafeSetOf(0, 1, 7, 17, 23, 56, 57, 58, 59, 60, 83), + hiddenObjects = unsafeSetOf( "s_n_0_22_4_imqetn", "s_n_0_25_4_imqeto", "s_n_0_26_4_imqeto", @@ -719,8 +719,8 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine ), // Ruins 1 Pair(Episode.I, 8) to Fix( - renderOnTopTextures = jsSetOf(1, 21, 22, 27, 28, 43, 51, 59, 70, 72, 75), - hiddenObjects = jsSetOf( + renderOnTopTextures = unsafeSetOf(1, 21, 22, 27, 28, 43, 51, 59, 70, 72, 75), + hiddenObjects = unsafeSetOf( "s_n_0_2p_4_iohs6r", "s_n_0_2q_4_iohs6r", "s_m_0_l_1_io448k", @@ -728,18 +728,18 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine ), // Ruins 2 Pair(Episode.I, 9) to Fix( - hiddenObjects = jsSetOf( + hiddenObjects = unsafeSetOf( "s_m_0_l_1_io448k", ), ), // Lab Pair(Episode.II, 0) to Fix( - renderOnTopTextures = jsSetOf(36, 37, 38, 48, 60, 67, 79, 80), + renderOnTopTextures = unsafeSetOf(36, 37, 38, 48, 60, 67, 79, 80), ), // VR Spaceship Alpha Pair(Episode.II, 3) to Fix( - renderOnTopTextures = jsSetOf(7, 59), - hiddenObjects = jsSetOf( + renderOnTopTextures = unsafeSetOf(7, 59), + hiddenObjects = unsafeSetOf( "s_l_0_45_5_ing07n", "s_n_0_45_5_ing07k", "s_n_0_g2_b_im2en1", @@ -774,12 +774,12 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine ), // Central Control Area Pair(Episode.II, 5) to Fix( - renderOnTopTextures = jsSetOf(*((0..59).toSet() + setOf(69, 77)).toTypedArray()), + renderOnTopTextures = unsafeSetOf(*((0..59).toSet() + setOf(69, 77)).toTypedArray()), ), // Jungle Area East Pair(Episode.II, 6) to Fix( - renderOnTopTextures = jsSetOf(0, 1, 2, 18, 21, 24), - hiddenObjects = jsSetOf( + renderOnTopTextures = unsafeSetOf(0, 1, 2, 18, 21, 24), + hiddenObjects = unsafeSetOf( "a_m_0_1i_1_isf1hw", "a_m_0_1i_1_isfvf0", "a_m_0_1i_1_ise7ew", @@ -802,8 +802,8 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine ), // Subterranean Desert 1 Pair(Episode.IV, 6) to Fix( - renderOnTopTextures = jsSetOf(48, 50, 58, 66, 80, 81, 92, 93, 94, 99, 100, 103), - hiddenObjects = jsSetOf( + renderOnTopTextures = unsafeSetOf(48, 50, 58, 66, 80, 81, 92, 93, 94, 99, 100, 103), + hiddenObjects = unsafeSetOf( "s_v_f_16u_b_j2s5tx", "s_v_d_84_f_j046sf", "s_v_1v_205_2n_jb17vl", diff --git a/webui/src/main/kotlin/world/phantasmal/webui/dom/HTMLElementSizeCell.kt b/webui/src/main/kotlin/world/phantasmal/webui/dom/HTMLElementSizeCell.kt index 5f67ae5b..01853f34 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/dom/HTMLElementSizeCell.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/dom/HTMLElementSizeCell.kt @@ -1,6 +1,8 @@ package world.phantasmal.webui.dom import org.w3c.dom.HTMLElement +import world.phantasmal.core.ResizeObserver +import world.phantasmal.core.ResizeObserverEntry import world.phantasmal.core.unsafe.unsafeAssertNotNull import world.phantasmal.observable.ChangeEvent import world.phantasmal.observable.Dependent @@ -9,20 +11,15 @@ import world.phantasmal.observable.cell.AbstractCell data class Size(val width: Double, val height: Double) class HTMLElementSizeCell(element: HTMLElement? = null) : AbstractCell() { - private var resizeObserver: dynamic = null + private var resizeObserver: ResizeObserver? = null private var _value: Size? = null var element: HTMLElement? = element set(element) { - if (resizeObserver != null) { - if (field != null) { - resizeObserver.unobserve(field) - } - - if (element != null) { - resizeObserver.observe(element) - } + resizeObserver?.let { resizeObserver -> + field?.let(resizeObserver::unobserve) + element?.let(resizeObserver::observe) } field = element @@ -40,14 +37,10 @@ class HTMLElementSizeCell(element: HTMLElement? = null) : AbstractCell() { override fun addDependent(dependent: Dependent) { if (dependents.isEmpty()) { if (resizeObserver == null) { - @Suppress("UNUSED_VARIABLE") - val resize = ::resizeCallback - resizeObserver = js("new ResizeObserver(resize);") + resizeObserver = ResizeObserver(::resizeCallback) } - if (element != null) { - resizeObserver.observe(element) - } + element?.let(unsafeAssertNotNull(resizeObserver)::observe) _value = getSize() } @@ -59,7 +52,7 @@ class HTMLElementSizeCell(element: HTMLElement? = null) : AbstractCell() { super.removeDependent(dependent) if (dependents.isEmpty()) { - resizeObserver.disconnect() + resizeObserver?.disconnect() } } @@ -72,12 +65,9 @@ class HTMLElementSizeCell(element: HTMLElement? = null) : AbstractCell() { ?.let { Size(it.offsetWidth.toDouble(), it.offsetHeight.toDouble()) } ?: Size(0.0, 0.0) - private fun resizeCallback(entries: Array) { + private fun resizeCallback(entries: Array) { val entry = entries.first() - val newValue = Size( - entry.contentRect.width.unsafeCast(), - entry.contentRect.height.unsafeCast(), - ) + val newValue = Size(entry.contentRect.width, entry.contentRect.height) if (newValue != _value) { emitMightChange()