mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-04 06:28:28 +08:00
Made basic quest properties editable and added the entity detail widget.
This commit is contained in:
parent
4e65cc1882
commit
c82396326c
@ -13,8 +13,8 @@ tasks.wrapper {
|
|||||||
subprojects {
|
subprojects {
|
||||||
project.extra["coroutinesVersion"] = "1.3.9"
|
project.extra["coroutinesVersion"] = "1.3.9"
|
||||||
project.extra["kotlinLoggingVersion"] = "2.0.2"
|
project.extra["kotlinLoggingVersion"] = "2.0.2"
|
||||||
project.extra["ktorVersion"] = "1.4.1"
|
project.extra["ktorVersion"] = "1.4.2"
|
||||||
project.extra["serializationVersion"] = "1.0.0"
|
project.extra["serializationVersion"] = "1.4.10"
|
||||||
project.extra["slf4jVersion"] = "1.7.30"
|
project.extra["slf4jVersion"] = "1.7.30"
|
||||||
|
|
||||||
repositories {
|
repositories {
|
||||||
|
@ -1,5 +1,20 @@
|
|||||||
package world.phantasmal.core.math
|
package world.phantasmal.core.math
|
||||||
|
|
||||||
|
import kotlin.math.PI
|
||||||
|
|
||||||
|
private const val TO_DEG = 180 / PI
|
||||||
|
private const val TO_RAD = PI / 180
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts radians to degrees.
|
||||||
|
*/
|
||||||
|
fun radToDeg(rad: Double): Double = rad * TO_DEG
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts degrees to radians.
|
||||||
|
*/
|
||||||
|
fun degToRad(deg: Double): Double = deg * TO_RAD
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the floored modulus of its arguments. The computed value will have the same sign as the
|
* Returns the floored modulus of its arguments. The computed value will have the same sign as the
|
||||||
* [divisor].
|
* [divisor].
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
@file:JvmName("PrimitiveExtensionsJvm")
|
||||||
|
|
||||||
package world.phantasmal.core
|
package world.phantasmal.core
|
||||||
|
|
||||||
import java.lang.Float.intBitsToFloat
|
import java.lang.Float.intBitsToFloat
|
||||||
|
@ -260,7 +260,7 @@ private fun StringBuilder.appendArgs(params: List<Param>, args: List<ArgWithType
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun StringBuilder.appendStringArg(value: String) {
|
private fun StringBuilder.appendStringArg(value: String): StringBuilder {
|
||||||
append("\"")
|
append("\"")
|
||||||
|
|
||||||
for (char in value) {
|
for (char in value) {
|
||||||
@ -274,9 +274,10 @@ private fun StringBuilder.appendStringArg(value: String) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
append("\"")
|
append("\"")
|
||||||
|
return this
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun StringBuilder.appendStringSegment(value: String) {
|
private fun StringBuilder.appendStringSegment(value: String): StringBuilder {
|
||||||
append("\"")
|
append("\"")
|
||||||
|
|
||||||
var i = 0
|
var i = 0
|
||||||
@ -307,4 +308,5 @@ private fun StringBuilder.appendStringSegment(value: String) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
append("\"")
|
append("\"")
|
||||||
|
return this
|
||||||
}
|
}
|
||||||
|
@ -50,4 +50,7 @@ interface Val<out T> : Observable<T> {
|
|||||||
*/
|
*/
|
||||||
fun <R> flatMap(transform: (T) -> Val<R>): Val<R> =
|
fun <R> flatMap(transform: (T) -> Val<R>): Val<R> =
|
||||||
FlatMappedVal(listOf(this)) { transform(value) }
|
FlatMappedVal(listOf(this)) { transform(value) }
|
||||||
|
|
||||||
|
fun <R> flatMapNull(transform: (T) -> Val<R>?): Val<R?> =
|
||||||
|
FlatMappedVal(listOf(this)) { transform(value) ?: nullVal() }
|
||||||
}
|
}
|
||||||
|
@ -30,6 +30,7 @@ kotlin {
|
|||||||
|
|
||||||
val kotlinLoggingVersion: String by project.extra
|
val kotlinLoggingVersion: String by project.extra
|
||||||
val ktorVersion: String by project.extra
|
val ktorVersion: String by project.extra
|
||||||
|
val serializationVersion: String by project.extra
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
implementation(project(":lib"))
|
implementation(project(":lib"))
|
||||||
@ -38,12 +39,13 @@ dependencies {
|
|||||||
implementation("io.github.microutils:kotlin-logging-js:$kotlinLoggingVersion")
|
implementation("io.github.microutils:kotlin-logging-js:$kotlinLoggingVersion")
|
||||||
implementation("io.ktor:ktor-client-core-js:$ktorVersion")
|
implementation("io.ktor:ktor-client-core-js:$ktorVersion")
|
||||||
implementation("io.ktor:ktor-client-serialization-js:$ktorVersion")
|
implementation("io.ktor:ktor-client-serialization-js:$ktorVersion")
|
||||||
implementation("org.jetbrains.kotlinx:kotlinx-serialization-core-js:1.0.0")
|
implementation("org.jetbrains.kotlin:kotlin-serialization:$serializationVersion")
|
||||||
implementation(npm("@babylonjs/core", "^4.2.0-rc.5"))
|
implementation(npm("@babylonjs/core", "^4.2.0"))
|
||||||
implementation(npm("golden-layout", "^1.5.9"))
|
implementation(npm("golden-layout", "^1.5.9"))
|
||||||
implementation(npm("monaco-editor", "^0.21.2"))
|
implementation(npm("monaco-editor", "^0.21.2"))
|
||||||
|
|
||||||
implementation(devNpm("file-loader", "^6.0.0"))
|
implementation(devNpm("file-loader", "^6.0.0"))
|
||||||
|
implementation(devNpm("monaco-editor-webpack-plugin", "^2.0.0"))
|
||||||
|
|
||||||
testImplementation(kotlin("test-js"))
|
testImplementation(kotlin("test-js"))
|
||||||
testImplementation(project(":test-utils"))
|
testImplementation(project(":test-utils"))
|
||||||
|
50
web/src/main/kotlin/world/phantasmal/web/LogFormatter.kt
Normal file
50
web/src/main/kotlin/world/phantasmal/web/LogFormatter.kt
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
package world.phantasmal.web
|
||||||
|
|
||||||
|
import mu.DefaultMessageFormatter
|
||||||
|
import mu.Formatter
|
||||||
|
import mu.KotlinLoggingLevel
|
||||||
|
import mu.Marker
|
||||||
|
import kotlin.js.Date
|
||||||
|
|
||||||
|
class LogFormatter : Formatter {
|
||||||
|
override fun formatMessage(
|
||||||
|
level: KotlinLoggingLevel,
|
||||||
|
loggerName: String,
|
||||||
|
msg: () -> Any?,
|
||||||
|
): String =
|
||||||
|
time() + DefaultMessageFormatter.formatMessage(level, loggerName, msg)
|
||||||
|
|
||||||
|
override fun formatMessage(
|
||||||
|
level: KotlinLoggingLevel,
|
||||||
|
loggerName: String,
|
||||||
|
t: Throwable?,
|
||||||
|
msg: () -> Any?,
|
||||||
|
): String =
|
||||||
|
time() + DefaultMessageFormatter.formatMessage(level, loggerName, t, msg)
|
||||||
|
|
||||||
|
override fun formatMessage(
|
||||||
|
level: KotlinLoggingLevel,
|
||||||
|
loggerName: String,
|
||||||
|
marker: Marker?,
|
||||||
|
msg: () -> Any?,
|
||||||
|
): String =
|
||||||
|
time() + DefaultMessageFormatter.formatMessage(level, loggerName, marker, msg)
|
||||||
|
|
||||||
|
override fun formatMessage(
|
||||||
|
level: KotlinLoggingLevel,
|
||||||
|
loggerName: String,
|
||||||
|
marker: Marker?,
|
||||||
|
t: Throwable?,
|
||||||
|
msg: () -> Any?,
|
||||||
|
): String =
|
||||||
|
time() + DefaultMessageFormatter.formatMessage(level, loggerName, marker, t, msg)
|
||||||
|
|
||||||
|
private fun time(): String {
|
||||||
|
val date = Date()
|
||||||
|
val h = date.getHours().toString().padStart(2, '0')
|
||||||
|
val m = date.getMinutes().toString().padStart(2, '0')
|
||||||
|
val s = date.getSeconds().toString().padStart(2, '0')
|
||||||
|
val ms = date.getMilliseconds().toString().padStart(3, '0')
|
||||||
|
return "$h:$m:$s.$ms "
|
||||||
|
}
|
||||||
|
}
|
@ -32,6 +32,8 @@ fun main() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun init(): Disposable {
|
private fun init(): Disposable {
|
||||||
|
KotlinLoggingConfiguration.FORMATTER = LogFormatter()
|
||||||
|
|
||||||
if (window.location.hostname == "localhost") {
|
if (window.location.hostname == "localhost") {
|
||||||
KotlinLoggingConfiguration.LOG_LEVEL = KotlinLoggingLevel.TRACE
|
KotlinLoggingConfiguration.LOG_LEVEL = KotlinLoggingLevel.TRACE
|
||||||
}
|
}
|
||||||
|
@ -53,6 +53,13 @@ operator fun Quaternion.timesAssign(other: Quaternion) {
|
|||||||
*/
|
*/
|
||||||
fun Quaternion.inverse(): Quaternion = Quaternion.Inverse(this)
|
fun Quaternion.inverse(): Quaternion = Quaternion.Inverse(this)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Inverts this quaternion.
|
||||||
|
*/
|
||||||
|
fun Quaternion.invert() {
|
||||||
|
Quaternion.InverseToRef(this, this)
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Transforms [p] by this versor.
|
* Transforms [p] by this versor.
|
||||||
*/
|
*/
|
||||||
|
@ -109,6 +109,7 @@ external class Quaternion(
|
|||||||
fun FromEulerAnglesToRef(x: Double, y: Double, z: Double, result: Quaternion): Quaternion
|
fun FromEulerAnglesToRef(x: Double, y: Double, z: Double, result: Quaternion): Quaternion
|
||||||
fun RotationYawPitchRoll(yaw: Double, pitch: Double, roll: Double): Quaternion
|
fun RotationYawPitchRoll(yaw: Double, pitch: Double, roll: Double): Quaternion
|
||||||
fun Inverse(q: Quaternion): Quaternion
|
fun Inverse(q: Quaternion): Quaternion
|
||||||
|
fun InverseToRef(q: Quaternion, result: Quaternion): Quaternion
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -8,10 +8,7 @@ import world.phantasmal.web.core.PwToolType
|
|||||||
import world.phantasmal.web.core.loading.AssetLoader
|
import world.phantasmal.web.core.loading.AssetLoader
|
||||||
import world.phantasmal.web.core.stores.UiStore
|
import world.phantasmal.web.core.stores.UiStore
|
||||||
import world.phantasmal.web.externals.babylon.Engine
|
import world.phantasmal.web.externals.babylon.Engine
|
||||||
import world.phantasmal.web.questEditor.controllers.AssemblyEditorController
|
import world.phantasmal.web.questEditor.controllers.*
|
||||||
import world.phantasmal.web.questEditor.controllers.NpcCountsController
|
|
||||||
import world.phantasmal.web.questEditor.controllers.QuestEditorToolbarController
|
|
||||||
import world.phantasmal.web.questEditor.controllers.QuestInfoController
|
|
||||||
import world.phantasmal.web.questEditor.loading.AreaAssetLoader
|
import world.phantasmal.web.questEditor.loading.AreaAssetLoader
|
||||||
import world.phantasmal.web.questEditor.loading.EntityAssetLoader
|
import world.phantasmal.web.questEditor.loading.EntityAssetLoader
|
||||||
import world.phantasmal.web.questEditor.loading.QuestLoader
|
import world.phantasmal.web.questEditor.loading.QuestLoader
|
||||||
@ -56,6 +53,7 @@ class QuestEditor(
|
|||||||
))
|
))
|
||||||
val questInfoController = addDisposable(QuestInfoController(questEditorStore))
|
val questInfoController = addDisposable(QuestInfoController(questEditorStore))
|
||||||
val npcCountsController = addDisposable(NpcCountsController(questEditorStore))
|
val npcCountsController = addDisposable(NpcCountsController(questEditorStore))
|
||||||
|
val entityInfoController = addDisposable(EntityInfoController(questEditorStore))
|
||||||
val assemblyEditorController = addDisposable(AssemblyEditorController(assemblyEditorStore))
|
val assemblyEditorController = addDisposable(AssemblyEditorController(assemblyEditorStore))
|
||||||
|
|
||||||
// Rendering
|
// Rendering
|
||||||
@ -76,6 +74,7 @@ class QuestEditor(
|
|||||||
{ s -> QuestEditorToolbarWidget(s, toolbarController) },
|
{ s -> QuestEditorToolbarWidget(s, toolbarController) },
|
||||||
{ s -> QuestInfoWidget(s, questInfoController) },
|
{ s -> QuestInfoWidget(s, questInfoController) },
|
||||||
{ s -> NpcCountsWidget(s, npcCountsController) },
|
{ s -> NpcCountsWidget(s, npcCountsController) },
|
||||||
|
{ s -> EntityInfoWidget(s, entityInfoController) },
|
||||||
{ s -> QuestEditorRendererWidget(s, canvas, renderer) },
|
{ s -> QuestEditorRendererWidget(s, canvas, renderer) },
|
||||||
{ s -> AssemblyEditorWidget(s, assemblyEditorController) },
|
{ s -> AssemblyEditorWidget(s, assemblyEditorController) },
|
||||||
)
|
)
|
||||||
|
@ -0,0 +1,20 @@
|
|||||||
|
package world.phantasmal.web.questEditor.actions
|
||||||
|
|
||||||
|
import world.phantasmal.web.core.actions.Action
|
||||||
|
import world.phantasmal.web.questEditor.models.QuestModel
|
||||||
|
|
||||||
|
class EditIdAction(
|
||||||
|
private val quest: QuestModel,
|
||||||
|
private val newId: Int,
|
||||||
|
private val oldId: Int,
|
||||||
|
) : Action {
|
||||||
|
override val description: String = "Edit ID"
|
||||||
|
|
||||||
|
override fun execute() {
|
||||||
|
quest.setId(newId)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun undo() {
|
||||||
|
quest.setId(oldId)
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,20 @@
|
|||||||
|
package world.phantasmal.web.questEditor.actions
|
||||||
|
|
||||||
|
import world.phantasmal.web.core.actions.Action
|
||||||
|
import world.phantasmal.web.questEditor.models.QuestModel
|
||||||
|
|
||||||
|
class EditLongDescriptionAction(
|
||||||
|
private val quest: QuestModel,
|
||||||
|
private val newLongDescription: String,
|
||||||
|
private val oldLongDescription: String,
|
||||||
|
) : Action {
|
||||||
|
override val description: String = "Edit long description"
|
||||||
|
|
||||||
|
override fun execute() {
|
||||||
|
quest.setLongDescription(newLongDescription)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun undo() {
|
||||||
|
quest.setLongDescription(oldLongDescription)
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,20 @@
|
|||||||
|
package world.phantasmal.web.questEditor.actions
|
||||||
|
|
||||||
|
import world.phantasmal.web.core.actions.Action
|
||||||
|
import world.phantasmal.web.questEditor.models.QuestModel
|
||||||
|
|
||||||
|
class EditNameAction(
|
||||||
|
private val quest: QuestModel,
|
||||||
|
private val newName: String,
|
||||||
|
private val oldName: String,
|
||||||
|
) : Action {
|
||||||
|
override val description: String = "Edit name"
|
||||||
|
|
||||||
|
override fun execute() {
|
||||||
|
quest.setName(newName)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun undo() {
|
||||||
|
quest.setName(oldName)
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,20 @@
|
|||||||
|
package world.phantasmal.web.questEditor.actions
|
||||||
|
|
||||||
|
import world.phantasmal.web.core.actions.Action
|
||||||
|
import world.phantasmal.web.questEditor.models.QuestModel
|
||||||
|
|
||||||
|
class EditShortDescriptionAction(
|
||||||
|
private val quest: QuestModel,
|
||||||
|
private val newShortDescription: String,
|
||||||
|
private val oldShortDescription: String,
|
||||||
|
) : Action {
|
||||||
|
override val description: String = "Edit short description"
|
||||||
|
|
||||||
|
override fun execute() {
|
||||||
|
quest.setShortDescription(newShortDescription)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun undo() {
|
||||||
|
quest.setShortDescription(oldShortDescription)
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,35 @@
|
|||||||
|
package world.phantasmal.web.questEditor.actions
|
||||||
|
|
||||||
|
import world.phantasmal.web.core.actions.Action
|
||||||
|
import world.phantasmal.web.externals.babylon.Vector3
|
||||||
|
import world.phantasmal.web.questEditor.models.QuestEntityModel
|
||||||
|
|
||||||
|
class RotateEntityAction(
|
||||||
|
private val setSelectedEntity: (QuestEntityModel<*, *>) -> Unit,
|
||||||
|
private val entity: QuestEntityModel<*, *>,
|
||||||
|
private val newRotation: Vector3,
|
||||||
|
private val oldRotation: Vector3,
|
||||||
|
private val world: Boolean,
|
||||||
|
) : Action {
|
||||||
|
override val description: String = "Rotate ${entity.type.simpleName}"
|
||||||
|
|
||||||
|
override fun execute() {
|
||||||
|
setSelectedEntity(entity)
|
||||||
|
|
||||||
|
if (world) {
|
||||||
|
entity.setWorldRotation(newRotation)
|
||||||
|
} else {
|
||||||
|
entity.setRotation(newRotation)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun undo() {
|
||||||
|
setSelectedEntity(entity)
|
||||||
|
|
||||||
|
if (world) {
|
||||||
|
entity.setWorldRotation(oldRotation)
|
||||||
|
} else {
|
||||||
|
entity.setRotation(oldRotation)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -8,10 +8,10 @@ import world.phantasmal.web.questEditor.models.SectionModel
|
|||||||
class TranslateEntityAction(
|
class TranslateEntityAction(
|
||||||
private val setSelectedEntity: (QuestEntityModel<*, *>) -> Unit,
|
private val setSelectedEntity: (QuestEntityModel<*, *>) -> Unit,
|
||||||
private val entity: QuestEntityModel<*, *>,
|
private val entity: QuestEntityModel<*, *>,
|
||||||
private val oldSection: SectionModel?,
|
|
||||||
private val newSection: SectionModel?,
|
private val newSection: SectionModel?,
|
||||||
private val oldPosition: Vector3,
|
private val oldSection: SectionModel?,
|
||||||
private val newPosition: Vector3,
|
private val newPosition: Vector3,
|
||||||
|
private val oldPosition: Vector3,
|
||||||
private val world: Boolean,
|
private val world: Boolean,
|
||||||
) : Action {
|
) : Action {
|
||||||
override val description: String = "Move ${entity.type.simpleName}"
|
override val description: String = "Move ${entity.type.simpleName}"
|
||||||
|
@ -0,0 +1,117 @@
|
|||||||
|
package world.phantasmal.web.questEditor.controllers
|
||||||
|
|
||||||
|
import world.phantasmal.core.math.degToRad
|
||||||
|
import world.phantasmal.core.math.radToDeg
|
||||||
|
import world.phantasmal.observable.value.Val
|
||||||
|
import world.phantasmal.observable.value.isNull
|
||||||
|
import world.phantasmal.observable.value.value
|
||||||
|
import world.phantasmal.web.externals.babylon.Vector3
|
||||||
|
import world.phantasmal.web.questEditor.actions.RotateEntityAction
|
||||||
|
import world.phantasmal.web.questEditor.actions.TranslateEntityAction
|
||||||
|
import world.phantasmal.web.questEditor.models.QuestEntityModel
|
||||||
|
import world.phantasmal.web.questEditor.models.QuestNpcModel
|
||||||
|
import world.phantasmal.web.questEditor.stores.QuestEditorStore
|
||||||
|
import world.phantasmal.webui.controllers.Controller
|
||||||
|
|
||||||
|
class EntityInfoController(private val store: QuestEditorStore) : Controller() {
|
||||||
|
val unavailable: Val<Boolean> = store.selectedEntity.isNull()
|
||||||
|
val enabled: Val<Boolean> = store.questEditingEnabled
|
||||||
|
|
||||||
|
val type: Val<String> = store.selectedEntity.map {
|
||||||
|
it?.let { if (it is QuestNpcModel) "NPC" else "Object" } ?: ""
|
||||||
|
}
|
||||||
|
|
||||||
|
val name: Val<String> = store.selectedEntity.map { it?.type?.simpleName ?: "" }
|
||||||
|
|
||||||
|
val sectionId: Val<String> = store.selectedEntity
|
||||||
|
.flatMapNull { it?.sectionId }
|
||||||
|
.map { it?.toString() ?: "" }
|
||||||
|
|
||||||
|
val wave: Val<String> = store.selectedEntity
|
||||||
|
.flatMapNull { entity -> (entity as? QuestNpcModel)?.wave?.flatMapNull { it?.id } }
|
||||||
|
.map { it?.toString() ?: "" }
|
||||||
|
|
||||||
|
val waveHidden: Val<Boolean> = store.selectedEntity.map { it !is QuestNpcModel }
|
||||||
|
|
||||||
|
private val pos: Val<Vector3> = store.selectedEntity.flatMap { it?.position ?: DEFAULT_VECTOR }
|
||||||
|
val posX: Val<Double> = pos.map { it.x }
|
||||||
|
val posY: Val<Double> = pos.map { it.y }
|
||||||
|
val posZ: Val<Double> = pos.map { it.z }
|
||||||
|
|
||||||
|
private val rot: Val<Vector3> = store.selectedEntity.flatMap { it?.rotation ?: DEFAULT_VECTOR }
|
||||||
|
val rotX: Val<Double> = rot.map { radToDeg(it.x) }
|
||||||
|
val rotY: Val<Double> = rot.map { radToDeg(it.y) }
|
||||||
|
val rotZ: Val<Double> = rot.map { radToDeg(it.z) }
|
||||||
|
|
||||||
|
fun setPosX(x: Double) {
|
||||||
|
store.selectedEntity.value?.let { entity ->
|
||||||
|
val pos = entity.position.value
|
||||||
|
setPos(entity, x, pos.y, pos.z)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun setPosY(y: Double) {
|
||||||
|
store.selectedEntity.value?.let { entity ->
|
||||||
|
val pos = entity.position.value
|
||||||
|
setPos(entity, pos.x, y, pos.z)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun setPosZ(z: Double) {
|
||||||
|
store.selectedEntity.value?.let { entity ->
|
||||||
|
val pos = entity.position.value
|
||||||
|
setPos(entity, pos.x, pos.y, z)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun setPos(entity: QuestEntityModel<*, *>, x: Double, y: Double, z: Double) {
|
||||||
|
if (!enabled.value) return
|
||||||
|
|
||||||
|
store.executeAction(TranslateEntityAction(
|
||||||
|
setSelectedEntity = store::setSelectedEntity,
|
||||||
|
entity,
|
||||||
|
entity.section.value,
|
||||||
|
entity.section.value,
|
||||||
|
Vector3(x, y, z),
|
||||||
|
entity.position.value,
|
||||||
|
false,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
|
fun setRotX(x: Double) {
|
||||||
|
store.selectedEntity.value?.let { entity ->
|
||||||
|
val rot = entity.rotation.value
|
||||||
|
setRot(entity, degToRad(x), rot.y, rot.z)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun setRotY(y: Double) {
|
||||||
|
store.selectedEntity.value?.let { entity ->
|
||||||
|
val rot = entity.rotation.value
|
||||||
|
setRot(entity, rot.x, degToRad(y), rot.z)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun setRotZ(z: Double) {
|
||||||
|
store.selectedEntity.value?.let { entity ->
|
||||||
|
val rot = entity.rotation.value
|
||||||
|
setRot(entity, rot.x, rot.y, degToRad(z))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun setRot(entity: QuestEntityModel<*, *>, x: Double, y: Double, z: Double) {
|
||||||
|
if (!enabled.value) return
|
||||||
|
|
||||||
|
store.executeAction(RotateEntityAction(
|
||||||
|
setSelectedEntity = store::setSelectedEntity,
|
||||||
|
entity,
|
||||||
|
Vector3(x, y, z),
|
||||||
|
entity.rotation.value,
|
||||||
|
false,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
private val DEFAULT_VECTOR = value(Vector3.Zero())
|
||||||
|
}
|
||||||
|
}
|
@ -3,10 +3,14 @@ package world.phantasmal.web.questEditor.controllers
|
|||||||
import world.phantasmal.observable.value.Val
|
import world.phantasmal.observable.value.Val
|
||||||
import world.phantasmal.observable.value.isNull
|
import world.phantasmal.observable.value.isNull
|
||||||
import world.phantasmal.observable.value.value
|
import world.phantasmal.observable.value.value
|
||||||
|
import world.phantasmal.web.questEditor.actions.EditIdAction
|
||||||
|
import world.phantasmal.web.questEditor.actions.EditLongDescriptionAction
|
||||||
|
import world.phantasmal.web.questEditor.actions.EditNameAction
|
||||||
|
import world.phantasmal.web.questEditor.actions.EditShortDescriptionAction
|
||||||
import world.phantasmal.web.questEditor.stores.QuestEditorStore
|
import world.phantasmal.web.questEditor.stores.QuestEditorStore
|
||||||
import world.phantasmal.webui.controllers.Controller
|
import world.phantasmal.webui.controllers.Controller
|
||||||
|
|
||||||
class QuestInfoController(store: QuestEditorStore) : Controller() {
|
class QuestInfoController(private val store: QuestEditorStore) : Controller() {
|
||||||
val unavailable: Val<Boolean> = store.currentQuest.isNull()
|
val unavailable: Val<Boolean> = store.currentQuest.isNull()
|
||||||
val enabled: Val<Boolean> = store.questEditingEnabled
|
val enabled: Val<Boolean> = store.questEditingEnabled
|
||||||
|
|
||||||
@ -17,4 +21,40 @@ class QuestInfoController(store: QuestEditorStore) : Controller() {
|
|||||||
store.currentQuest.flatMap { it?.shortDescription ?: value("") }
|
store.currentQuest.flatMap { it?.shortDescription ?: value("") }
|
||||||
val longDescription: Val<String> =
|
val longDescription: Val<String> =
|
||||||
store.currentQuest.flatMap { it?.longDescription ?: value("") }
|
store.currentQuest.flatMap { it?.longDescription ?: value("") }
|
||||||
|
|
||||||
|
fun setId(id: Int) {
|
||||||
|
if (!enabled.value) return
|
||||||
|
|
||||||
|
store.currentQuest.value?.let { quest ->
|
||||||
|
store.executeAction(EditIdAction(quest, id, quest.id.value))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun setName(name: String) {
|
||||||
|
if (!enabled.value) return
|
||||||
|
|
||||||
|
store.currentQuest.value?.let { quest ->
|
||||||
|
store.executeAction(EditNameAction(quest, name, quest.name.value))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun setShortDescription(shortDescription: String) {
|
||||||
|
if (!enabled.value) return
|
||||||
|
|
||||||
|
store.currentQuest.value?.let { quest ->
|
||||||
|
store.executeAction(
|
||||||
|
EditShortDescriptionAction(quest, shortDescription, quest.shortDescription.value)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun setLongDescription(longDescription: String) {
|
||||||
|
if (!enabled.value) return
|
||||||
|
|
||||||
|
store.currentQuest.value?.let { quest ->
|
||||||
|
store.executeAction(
|
||||||
|
EditLongDescriptionAction(quest, longDescription, quest.longDescription.value)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -98,6 +98,7 @@ abstract class QuestEntityModel<Type : EntityType, Entity : QuestEntity<Type>>(
|
|||||||
floorModEuler(rot)
|
floorModEuler(rot)
|
||||||
|
|
||||||
entity.rotation = babylonToVec3(rot)
|
entity.rotation = babylonToVec3(rot)
|
||||||
|
_rotation.value = rot
|
||||||
|
|
||||||
val section = section.value
|
val section = section.value
|
||||||
|
|
||||||
@ -118,6 +119,34 @@ abstract class QuestEntityModel<Type : EntityType, Entity : QuestEntity<Type>>(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun setWorldRotation(rot: Vector3) {
|
||||||
|
floorModEuler(rot)
|
||||||
|
|
||||||
|
_worldRotation.value = rot
|
||||||
|
|
||||||
|
val section = section.value
|
||||||
|
|
||||||
|
val relRot = if (section == null) {
|
||||||
|
rot
|
||||||
|
} else {
|
||||||
|
Quaternion.FromEulerAnglesToRef(rot.x, rot.y, rot.z, q1)
|
||||||
|
Quaternion.FromEulerAnglesToRef(
|
||||||
|
section.rotation.x,
|
||||||
|
section.rotation.y,
|
||||||
|
section.rotation.z,
|
||||||
|
q2
|
||||||
|
)
|
||||||
|
q2.invert()
|
||||||
|
q1 *= q2
|
||||||
|
val relRot = q1.toEulerAngles()
|
||||||
|
floorModEuler(relRot)
|
||||||
|
relRot
|
||||||
|
}
|
||||||
|
|
||||||
|
entity.rotation = babylonToVec3(relRot)
|
||||||
|
_rotation.value = relRot
|
||||||
|
}
|
||||||
|
|
||||||
private fun floorModEuler(euler: Vector3) {
|
private fun floorModEuler(euler: Vector3) {
|
||||||
euler.set(
|
euler.set(
|
||||||
floorMod(euler.x, 2 * PI),
|
floorMod(euler.x, 2 * PI),
|
||||||
|
@ -1,3 +1,14 @@
|
|||||||
package world.phantasmal.web.questEditor.models
|
package world.phantasmal.web.questEditor.models
|
||||||
|
|
||||||
class WaveModel
|
import world.phantasmal.observable.value.Val
|
||||||
|
import world.phantasmal.observable.value.mutableVal
|
||||||
|
|
||||||
|
class WaveModel(id: Int, areaId: Int, sectionId: Int) {
|
||||||
|
private val _id = mutableVal(id)
|
||||||
|
private val _areaId = mutableVal(areaId)
|
||||||
|
private val _sectionId = mutableVal(sectionId)
|
||||||
|
|
||||||
|
val id: Val<Int> = _id
|
||||||
|
val areaId: Val<Int> = _areaId
|
||||||
|
val sectionId: Val<Int> = _sectionId
|
||||||
|
}
|
||||||
|
@ -8,6 +8,7 @@ import world.phantasmal.web.core.minus
|
|||||||
import world.phantasmal.web.core.plusAssign
|
import world.phantasmal.web.core.plusAssign
|
||||||
import world.phantasmal.web.core.times
|
import world.phantasmal.web.core.times
|
||||||
import world.phantasmal.web.externals.babylon.*
|
import world.phantasmal.web.externals.babylon.*
|
||||||
|
import world.phantasmal.web.questEditor.actions.TranslateEntityAction
|
||||||
import world.phantasmal.web.questEditor.models.QuestEntityModel
|
import world.phantasmal.web.questEditor.models.QuestEntityModel
|
||||||
import world.phantasmal.web.questEditor.models.SectionModel
|
import world.phantasmal.web.questEditor.models.SectionModel
|
||||||
import world.phantasmal.web.questEditor.stores.QuestEditorStore
|
import world.phantasmal.web.questEditor.stores.QuestEditorStore
|
||||||
@ -159,20 +160,21 @@ private class StateContext(
|
|||||||
|
|
||||||
fun finalizeTranslation(
|
fun finalizeTranslation(
|
||||||
entity: QuestEntityModel<*, *>,
|
entity: QuestEntityModel<*, *>,
|
||||||
oldSection: SectionModel?,
|
|
||||||
newSection: SectionModel?,
|
newSection: SectionModel?,
|
||||||
oldPosition: Vector3,
|
oldSection: SectionModel?,
|
||||||
newPosition: Vector3,
|
newPosition: Vector3,
|
||||||
|
oldPosition: Vector3,
|
||||||
world: Boolean,
|
world: Boolean,
|
||||||
) {
|
) {
|
||||||
questEditorStore.translateEntity(
|
questEditorStore.executeAction(TranslateEntityAction(
|
||||||
|
::setSelectedEntity,
|
||||||
entity,
|
entity,
|
||||||
oldSection,
|
|
||||||
newSection,
|
newSection,
|
||||||
oldPosition,
|
oldSection,
|
||||||
newPosition,
|
newPosition,
|
||||||
world
|
oldPosition,
|
||||||
)
|
world,
|
||||||
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -446,10 +448,10 @@ private class TranslationState(
|
|||||||
if (!cancelled && event.movedSinceLastPointerDown) {
|
if (!cancelled && event.movedSinceLastPointerDown) {
|
||||||
ctx.finalizeTranslation(
|
ctx.finalizeTranslation(
|
||||||
entity,
|
entity,
|
||||||
initialSection,
|
|
||||||
entity.section.value,
|
entity.section.value,
|
||||||
initialPosition,
|
initialSection,
|
||||||
entity.worldPosition.value,
|
entity.worldPosition.value,
|
||||||
|
initialPosition,
|
||||||
true,
|
true,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -4,13 +4,11 @@ import kotlinx.coroutines.CoroutineScope
|
|||||||
import mu.KotlinLogging
|
import mu.KotlinLogging
|
||||||
import world.phantasmal.observable.value.*
|
import world.phantasmal.observable.value.*
|
||||||
import world.phantasmal.web.core.PwToolType
|
import world.phantasmal.web.core.PwToolType
|
||||||
import world.phantasmal.web.core.stores.UiStore
|
|
||||||
import world.phantasmal.web.core.actions.Action
|
import world.phantasmal.web.core.actions.Action
|
||||||
|
import world.phantasmal.web.core.stores.UiStore
|
||||||
import world.phantasmal.web.core.undo.UndoManager
|
import world.phantasmal.web.core.undo.UndoManager
|
||||||
import world.phantasmal.web.core.undo.UndoStack
|
import world.phantasmal.web.core.undo.UndoStack
|
||||||
import world.phantasmal.web.externals.babylon.Vector3
|
|
||||||
import world.phantasmal.web.questEditor.QuestRunner
|
import world.phantasmal.web.questEditor.QuestRunner
|
||||||
import world.phantasmal.web.questEditor.actions.TranslateEntityAction
|
|
||||||
import world.phantasmal.web.questEditor.models.*
|
import world.phantasmal.web.questEditor.models.*
|
||||||
import world.phantasmal.webui.stores.Store
|
import world.phantasmal.webui.stores.Store
|
||||||
|
|
||||||
@ -128,22 +126,15 @@ class QuestEditorStore(
|
|||||||
_selectedEntity.value = entity
|
_selectedEntity.value = entity
|
||||||
}
|
}
|
||||||
|
|
||||||
fun translateEntity(
|
fun executeAction(action: Action) {
|
||||||
entity: QuestEntityModel<*, *>,
|
require(questEditingEnabled.value) {
|
||||||
oldSection: SectionModel?,
|
val reason = when {
|
||||||
newSection: SectionModel?,
|
currentQuest.value == null -> " (no current quest)"
|
||||||
oldPosition: Vector3,
|
runner.running.value -> " (QuestRunner is running)"
|
||||||
newPosition: Vector3,
|
else -> ""
|
||||||
world: Boolean,
|
}
|
||||||
) {
|
"Quest editing is disabled at the moment$reason."
|
||||||
mainUndo.push(TranslateEntityAction(
|
}
|
||||||
::setSelectedEntity,
|
mainUndo.push(action).execute()
|
||||||
entity,
|
|
||||||
oldSection,
|
|
||||||
newSection,
|
|
||||||
oldPosition,
|
|
||||||
newPosition,
|
|
||||||
world,
|
|
||||||
)).execute()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,159 @@
|
|||||||
|
package world.phantasmal.web.questEditor.widgets
|
||||||
|
|
||||||
|
import kotlinx.coroutines.CoroutineScope
|
||||||
|
import org.w3c.dom.Node
|
||||||
|
import world.phantasmal.web.core.widgets.UnavailableWidget
|
||||||
|
import world.phantasmal.web.questEditor.controllers.EntityInfoController
|
||||||
|
import world.phantasmal.webui.dom.*
|
||||||
|
import world.phantasmal.webui.widgets.DoubleInput
|
||||||
|
import world.phantasmal.webui.widgets.Widget
|
||||||
|
|
||||||
|
class EntityInfoWidget(
|
||||||
|
scope: CoroutineScope,
|
||||||
|
private val ctrl: EntityInfoController,
|
||||||
|
) : Widget(scope, enabled = ctrl.enabled) {
|
||||||
|
override fun Node.createElement() =
|
||||||
|
div {
|
||||||
|
className = "pw-quest-editor-entity-info"
|
||||||
|
tabIndex = -1
|
||||||
|
|
||||||
|
table {
|
||||||
|
hidden(ctrl.unavailable)
|
||||||
|
|
||||||
|
tr {
|
||||||
|
th { textContent = "Type:" }
|
||||||
|
td { text(ctrl.type) }
|
||||||
|
}
|
||||||
|
tr {
|
||||||
|
th { textContent = "Name:" }
|
||||||
|
td { text(ctrl.name) }
|
||||||
|
}
|
||||||
|
tr {
|
||||||
|
th { textContent = "Section:" }
|
||||||
|
td { text(ctrl.sectionId) }
|
||||||
|
}
|
||||||
|
tr {
|
||||||
|
hidden(ctrl.waveHidden)
|
||||||
|
|
||||||
|
th { textContent = "Wave:" }
|
||||||
|
td { text(ctrl.wave) }
|
||||||
|
}
|
||||||
|
tr {
|
||||||
|
th { colSpan = 2; textContent = "Position:" }
|
||||||
|
}
|
||||||
|
tr {
|
||||||
|
th { className = COORD_CLASS; textContent = "X:" }
|
||||||
|
td {
|
||||||
|
addChild(DoubleInput(
|
||||||
|
this@EntityInfoWidget.scope,
|
||||||
|
value = ctrl.posX,
|
||||||
|
onChange = ctrl::setPosX,
|
||||||
|
roundTo = 3,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tr {
|
||||||
|
th { className = COORD_CLASS; textContent = "Y:" }
|
||||||
|
td {
|
||||||
|
addChild(DoubleInput(
|
||||||
|
this@EntityInfoWidget.scope,
|
||||||
|
value = ctrl.posY,
|
||||||
|
onChange = ctrl::setPosY,
|
||||||
|
roundTo = 3,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tr {
|
||||||
|
th { className = COORD_CLASS; textContent = "Z:" }
|
||||||
|
td {
|
||||||
|
addChild(DoubleInput(
|
||||||
|
this@EntityInfoWidget.scope,
|
||||||
|
value = ctrl.posZ,
|
||||||
|
onChange = ctrl::setPosZ,
|
||||||
|
roundTo = 3,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tr {
|
||||||
|
th { colSpan = 2; textContent = "Rotation:" }
|
||||||
|
}
|
||||||
|
tr {
|
||||||
|
th { className = COORD_CLASS; textContent = "X:" }
|
||||||
|
td {
|
||||||
|
addChild(DoubleInput(
|
||||||
|
this@EntityInfoWidget.scope,
|
||||||
|
value = ctrl.rotX,
|
||||||
|
onChange = ctrl::setRotX,
|
||||||
|
roundTo = 3,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tr {
|
||||||
|
th { className = COORD_CLASS; textContent = "Y:" }
|
||||||
|
td {
|
||||||
|
addChild(DoubleInput(
|
||||||
|
this@EntityInfoWidget.scope,
|
||||||
|
value = ctrl.rotY,
|
||||||
|
onChange = ctrl::setRotY,
|
||||||
|
roundTo = 3,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tr {
|
||||||
|
th { className = COORD_CLASS; textContent = "Z:" }
|
||||||
|
td {
|
||||||
|
addChild(DoubleInput(
|
||||||
|
this@EntityInfoWidget.scope,
|
||||||
|
value = ctrl.rotZ,
|
||||||
|
onChange = ctrl::setRotZ,
|
||||||
|
roundTo = 3,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
addChild(UnavailableWidget(
|
||||||
|
scope,
|
||||||
|
visible = ctrl.unavailable,
|
||||||
|
message = "No entity selected.",
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
private const val COORD_CLASS = "pw-quest-editor-entity-info-coord"
|
||||||
|
|
||||||
|
init {
|
||||||
|
@Suppress("CssUnusedSymbol")
|
||||||
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-quest-editor-entity-info {
|
||||||
|
outline: none;
|
||||||
|
box-sizing: border-box;
|
||||||
|
padding: 3px;
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pw-quest-editor-entity-info table {
|
||||||
|
table-layout: fixed;
|
||||||
|
width: 100%;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pw-quest-editor-entity-info th {
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pw-quest-editor-entity-info th.pw-quest-editor-entity-info-coord {
|
||||||
|
padding-left: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pw-quest-editor-entity-info .pw-number-input {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pw-quest-editor-entity-info table.pw-quest-editor-entity-info-specific-props {
|
||||||
|
margin-top: -2px;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -18,13 +18,14 @@ private class TestWidget(scope: CoroutineScope) : Widget(scope) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Takes ownership of the widgets created by the given createWidget functions.
|
* Takes ownership of the widgets created by the given creation functions.
|
||||||
*/
|
*/
|
||||||
class QuestEditorWidget(
|
class QuestEditorWidget(
|
||||||
scope: CoroutineScope,
|
scope: CoroutineScope,
|
||||||
private val createToolbar: (CoroutineScope) -> Widget,
|
private val createToolbar: (CoroutineScope) -> Widget,
|
||||||
private val createQuestInfoWidget: (CoroutineScope) -> Widget,
|
private val createQuestInfoWidget: (CoroutineScope) -> Widget,
|
||||||
private val createNpcCountsWidget: (CoroutineScope) -> Widget,
|
private val createNpcCountsWidget: (CoroutineScope) -> Widget,
|
||||||
|
private val createEntityInfoWidget: (CoroutineScope) -> Widget,
|
||||||
private val createQuestRendererWidget: (CoroutineScope) -> Widget,
|
private val createQuestRendererWidget: (CoroutineScope) -> Widget,
|
||||||
private val createAssemblyEditorWidget: (CoroutineScope) -> Widget,
|
private val createAssemblyEditorWidget: (CoroutineScope) -> Widget,
|
||||||
) : Widget(scope) {
|
) : Widget(scope) {
|
||||||
@ -57,7 +58,7 @@ class QuestEditorWidget(
|
|||||||
DockedWidget(
|
DockedWidget(
|
||||||
title = "Entity",
|
title = "Entity",
|
||||||
id = "entity_info",
|
id = "entity_info",
|
||||||
createWidget = ::TestWidget
|
createWidget = createEntityInfoWidget
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
),
|
),
|
||||||
|
@ -32,7 +32,8 @@ class QuestInfoWidget(
|
|||||||
addChild(IntInput(
|
addChild(IntInput(
|
||||||
this@QuestInfoWidget.scope,
|
this@QuestInfoWidget.scope,
|
||||||
enabled = ctrl.enabled,
|
enabled = ctrl.enabled,
|
||||||
valueVal = ctrl.id,
|
value = ctrl.id,
|
||||||
|
onChange = ctrl::setId,
|
||||||
min = 0,
|
min = 0,
|
||||||
step = 1,
|
step = 1,
|
||||||
))
|
))
|
||||||
@ -44,7 +45,8 @@ class QuestInfoWidget(
|
|||||||
addChild(TextInput(
|
addChild(TextInput(
|
||||||
this@QuestInfoWidget.scope,
|
this@QuestInfoWidget.scope,
|
||||||
enabled = ctrl.enabled,
|
enabled = ctrl.enabled,
|
||||||
valueVal = ctrl.name,
|
value = ctrl.name,
|
||||||
|
onChange = ctrl::setName,
|
||||||
maxLength = 32,
|
maxLength = 32,
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
@ -61,7 +63,8 @@ class QuestInfoWidget(
|
|||||||
addChild(TextArea(
|
addChild(TextArea(
|
||||||
this@QuestInfoWidget.scope,
|
this@QuestInfoWidget.scope,
|
||||||
enabled = ctrl.enabled,
|
enabled = ctrl.enabled,
|
||||||
valueVal = ctrl.shortDescription,
|
value = ctrl.shortDescription,
|
||||||
|
onChange = ctrl::setShortDescription,
|
||||||
maxLength = 128,
|
maxLength = 128,
|
||||||
fontFamily = "\"Courier New\", monospace",
|
fontFamily = "\"Courier New\", monospace",
|
||||||
cols = 25,
|
cols = 25,
|
||||||
@ -81,7 +84,8 @@ class QuestInfoWidget(
|
|||||||
addChild(TextArea(
|
addChild(TextArea(
|
||||||
this@QuestInfoWidget.scope,
|
this@QuestInfoWidget.scope,
|
||||||
enabled = ctrl.enabled,
|
enabled = ctrl.enabled,
|
||||||
valueVal = ctrl.longDescription,
|
value = ctrl.longDescription,
|
||||||
|
onChange = ctrl::setLongDescription,
|
||||||
maxLength = 288,
|
maxLength = 288,
|
||||||
fontFamily = "\"Courier New\", monospace",
|
fontFamily = "\"Courier New\", monospace",
|
||||||
cols = 25,
|
cols = 25,
|
||||||
|
@ -0,0 +1,119 @@
|
|||||||
|
package world.phantasmal.web.questEditor.controllers
|
||||||
|
|
||||||
|
import world.phantasmal.lib.fileFormats.Vec3
|
||||||
|
import world.phantasmal.lib.fileFormats.quest.Episode
|
||||||
|
import world.phantasmal.lib.fileFormats.quest.NpcType
|
||||||
|
import world.phantasmal.lib.fileFormats.quest.QuestNpc
|
||||||
|
import world.phantasmal.testUtils.assertCloseTo
|
||||||
|
import world.phantasmal.web.questEditor.models.QuestNpcModel
|
||||||
|
import world.phantasmal.web.questEditor.models.WaveModel
|
||||||
|
import world.phantasmal.web.test.WebTestSuite
|
||||||
|
import world.phantasmal.web.test.createQuestModel
|
||||||
|
import world.phantasmal.web.test.createQuestNpcModel
|
||||||
|
import kotlin.math.PI
|
||||||
|
import kotlin.test.Test
|
||||||
|
import kotlin.test.assertEquals
|
||||||
|
import kotlin.test.assertFalse
|
||||||
|
import kotlin.test.assertTrue
|
||||||
|
|
||||||
|
class EntityInfoControllerTests : WebTestSuite() {
|
||||||
|
@Test
|
||||||
|
fun test_unavailable_and_enabled() = asyncTest {
|
||||||
|
val ctrl = EntityInfoController(components.questEditorStore)
|
||||||
|
|
||||||
|
assertTrue(ctrl.unavailable.value)
|
||||||
|
assertFalse(ctrl.enabled.value)
|
||||||
|
|
||||||
|
val npc = createQuestNpcModel(NpcType.Principal, Episode.I)
|
||||||
|
components.questEditorStore.setCurrentQuest(createQuestModel(npcs = listOf(npc)))
|
||||||
|
|
||||||
|
assertTrue(ctrl.unavailable.value)
|
||||||
|
assertTrue(ctrl.enabled.value)
|
||||||
|
|
||||||
|
components.questEditorStore.setSelectedEntity(npc)
|
||||||
|
|
||||||
|
assertFalse(ctrl.unavailable.value)
|
||||||
|
assertTrue(ctrl.enabled.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun can_read_regular_properties() = asyncTest {
|
||||||
|
val ctrl = EntityInfoController(components.questEditorStore)
|
||||||
|
|
||||||
|
val questNpc = QuestNpc(NpcType.Booma, Episode.I, areaId = 10, wave = 5)
|
||||||
|
questNpc.sectionId = 7
|
||||||
|
questNpc.position = Vec3(8f, 16f, 32f)
|
||||||
|
questNpc.rotation = Vec3(PI.toFloat() / 4, PI.toFloat() / 2, PI.toFloat())
|
||||||
|
val npc = QuestNpcModel(questNpc, WaveModel(5, 10, 7))
|
||||||
|
components.questEditorStore.setCurrentQuest(createQuestModel(npcs = listOf(npc)))
|
||||||
|
components.questEditorStore.setSelectedEntity(npc)
|
||||||
|
|
||||||
|
assertEquals("NPC", ctrl.type.value)
|
||||||
|
assertEquals("Booma", ctrl.name.value)
|
||||||
|
assertEquals("7", ctrl.sectionId.value)
|
||||||
|
assertEquals("5", ctrl.wave.value)
|
||||||
|
assertFalse(ctrl.waveHidden.value)
|
||||||
|
assertEquals(8.0, ctrl.posX.value)
|
||||||
|
assertEquals(16.0, ctrl.posY.value)
|
||||||
|
assertEquals(32.0, ctrl.posZ.value)
|
||||||
|
assertCloseTo(45.0, ctrl.rotX.value)
|
||||||
|
assertCloseTo(90.0, ctrl.rotY.value)
|
||||||
|
assertCloseTo(180.0, ctrl.rotZ.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun can_set_regular_properties_undo_and_redo() = asyncTest {
|
||||||
|
val ctrl = EntityInfoController(components.questEditorStore)
|
||||||
|
|
||||||
|
val npc = createQuestNpcModel(NpcType.Principal, Episode.I)
|
||||||
|
components.questEditorStore.setCurrentQuest(createQuestModel(npcs = listOf(npc)))
|
||||||
|
components.questEditorStore.setSelectedEntity(npc)
|
||||||
|
|
||||||
|
ctrl.setPosX(3.15)
|
||||||
|
ctrl.setPosY(4.15)
|
||||||
|
ctrl.setPosZ(5.15)
|
||||||
|
|
||||||
|
ctrl.setRotX(50.0)
|
||||||
|
ctrl.setRotY(25.4)
|
||||||
|
ctrl.setRotZ(12.5)
|
||||||
|
|
||||||
|
assertEquals(3.15, ctrl.posX.value)
|
||||||
|
assertEquals(4.15, ctrl.posY.value)
|
||||||
|
assertEquals(5.15, ctrl.posZ.value)
|
||||||
|
|
||||||
|
assertCloseTo(50.0, ctrl.rotX.value)
|
||||||
|
assertCloseTo(25.4, ctrl.rotY.value)
|
||||||
|
assertCloseTo(12.5, ctrl.rotZ.value)
|
||||||
|
|
||||||
|
components.questEditorStore.makeMainUndoCurrent()
|
||||||
|
components.questEditorStore.undo()
|
||||||
|
components.questEditorStore.undo()
|
||||||
|
components.questEditorStore.undo()
|
||||||
|
components.questEditorStore.undo()
|
||||||
|
components.questEditorStore.undo()
|
||||||
|
components.questEditorStore.undo()
|
||||||
|
|
||||||
|
assertEquals(0.0, ctrl.posX.value)
|
||||||
|
assertEquals(0.0, ctrl.posY.value)
|
||||||
|
assertEquals(0.0, ctrl.posZ.value)
|
||||||
|
|
||||||
|
assertEquals(0.0, ctrl.rotX.value)
|
||||||
|
assertEquals(0.0, ctrl.rotY.value)
|
||||||
|
assertEquals(0.0, ctrl.rotZ.value)
|
||||||
|
|
||||||
|
components.questEditorStore.redo()
|
||||||
|
components.questEditorStore.redo()
|
||||||
|
components.questEditorStore.redo()
|
||||||
|
components.questEditorStore.redo()
|
||||||
|
components.questEditorStore.redo()
|
||||||
|
components.questEditorStore.redo()
|
||||||
|
|
||||||
|
assertEquals(3.15, ctrl.posX.value)
|
||||||
|
assertEquals(4.15, ctrl.posY.value)
|
||||||
|
assertEquals(5.15, ctrl.posZ.value)
|
||||||
|
|
||||||
|
assertCloseTo(50.0, ctrl.rotX.value)
|
||||||
|
assertCloseTo(25.4, ctrl.rotY.value)
|
||||||
|
assertCloseTo(12.5, ctrl.rotZ.value)
|
||||||
|
}
|
||||||
|
}
|
@ -6,6 +6,8 @@ import world.phantasmal.core.Severity
|
|||||||
import world.phantasmal.lib.fileFormats.quest.Episode
|
import world.phantasmal.lib.fileFormats.quest.Episode
|
||||||
import world.phantasmal.lib.fileFormats.quest.NpcType
|
import world.phantasmal.lib.fileFormats.quest.NpcType
|
||||||
import world.phantasmal.web.externals.babylon.Vector3
|
import world.phantasmal.web.externals.babylon.Vector3
|
||||||
|
import world.phantasmal.web.questEditor.actions.EditNameAction
|
||||||
|
import world.phantasmal.web.questEditor.actions.TranslateEntityAction
|
||||||
import world.phantasmal.web.test.WebTestSuite
|
import world.phantasmal.web.test.WebTestSuite
|
||||||
import world.phantasmal.web.test.createQuestModel
|
import world.phantasmal.web.test.createQuestModel
|
||||||
import world.phantasmal.web.test.createQuestNpcModel
|
import world.phantasmal.web.test.createQuestNpcModel
|
||||||
@ -69,7 +71,8 @@ class QuestEditorToolbarControllerTests : WebTestSuite() {
|
|||||||
|
|
||||||
// Load quest.
|
// Load quest.
|
||||||
val npc = createQuestNpcModel(NpcType.Scientist, Episode.I)
|
val npc = createQuestNpcModel(NpcType.Scientist, Episode.I)
|
||||||
components.questEditorStore.setCurrentQuest(createQuestModel(npcs = listOf(npc)))
|
val quest = createQuestModel(name = "Old Name", npcs = listOf(npc))
|
||||||
|
components.questEditorStore.setCurrentQuest(quest)
|
||||||
|
|
||||||
assertEquals(nothingToUndo, ctrl.undoTooltip.value)
|
assertEquals(nothingToUndo, ctrl.undoTooltip.value)
|
||||||
assertFalse(ctrl.undoEnabled.value)
|
assertFalse(ctrl.undoEnabled.value)
|
||||||
@ -78,16 +81,9 @@ class QuestEditorToolbarControllerTests : WebTestSuite() {
|
|||||||
assertFalse(ctrl.redoEnabled.value)
|
assertFalse(ctrl.redoEnabled.value)
|
||||||
|
|
||||||
// Add an action to the undo stack.
|
// Add an action to the undo stack.
|
||||||
components.questEditorStore.translateEntity(
|
components.questEditorStore.executeAction(EditNameAction(quest, "New Name", quest.name.value))
|
||||||
npc,
|
|
||||||
null,
|
|
||||||
null,
|
|
||||||
Vector3.Zero(),
|
|
||||||
Vector3.Up(),
|
|
||||||
true,
|
|
||||||
)
|
|
||||||
|
|
||||||
assertEquals("Undo \"Move Scientist\" (Ctrl-Z)", ctrl.undoTooltip.value)
|
assertEquals("Undo \"Edit name\" (Ctrl-Z)", ctrl.undoTooltip.value)
|
||||||
assertTrue(ctrl.undoEnabled.value)
|
assertTrue(ctrl.undoEnabled.value)
|
||||||
|
|
||||||
assertEquals(nothingToRedo, ctrl.redoTooltip.value)
|
assertEquals(nothingToRedo, ctrl.redoTooltip.value)
|
||||||
@ -99,7 +95,7 @@ class QuestEditorToolbarControllerTests : WebTestSuite() {
|
|||||||
assertEquals(nothingToUndo, ctrl.undoTooltip.value)
|
assertEquals(nothingToUndo, ctrl.undoTooltip.value)
|
||||||
assertFalse(ctrl.undoEnabled.value)
|
assertFalse(ctrl.undoEnabled.value)
|
||||||
|
|
||||||
assertEquals("Redo \"Move Scientist\" (Ctrl-Shift-Z)", ctrl.redoTooltip.value)
|
assertEquals("Redo \"Edit name\" (Ctrl-Shift-Z)", ctrl.redoTooltip.value)
|
||||||
assertTrue(ctrl.redoEnabled.value)
|
assertTrue(ctrl.redoEnabled.value)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -33,4 +33,56 @@ class QuestInfoControllerTests : WebTestSuite() {
|
|||||||
assertEquals("A short description.", ctrl.shortDescription.value)
|
assertEquals("A short description.", ctrl.shortDescription.value)
|
||||||
assertEquals("A long description.", ctrl.longDescription.value)
|
assertEquals("A long description.", ctrl.longDescription.value)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun can_edit_simple_properties_undo_edits_and_redo_edits() = asyncTest {
|
||||||
|
val store = components.questEditorStore
|
||||||
|
val ctrl = disposer.add(QuestInfoController(store))
|
||||||
|
|
||||||
|
store.setCurrentQuest(createQuestModel(
|
||||||
|
id = 1,
|
||||||
|
name = "name 1",
|
||||||
|
shortDescription = "short 1",
|
||||||
|
longDescription = "long 1",
|
||||||
|
episode = Episode.II
|
||||||
|
))
|
||||||
|
|
||||||
|
assertTrue(ctrl.enabled.value)
|
||||||
|
|
||||||
|
assertEquals(1, ctrl.id.value)
|
||||||
|
assertEquals("name 1", ctrl.name.value)
|
||||||
|
assertEquals("short 1", ctrl.shortDescription.value)
|
||||||
|
assertEquals("long 1", ctrl.longDescription.value)
|
||||||
|
|
||||||
|
ctrl.setId(2)
|
||||||
|
ctrl.setName("name 2")
|
||||||
|
ctrl.setShortDescription("short 2")
|
||||||
|
ctrl.setLongDescription("long 2")
|
||||||
|
|
||||||
|
assertEquals(2, ctrl.id.value)
|
||||||
|
assertEquals("name 2", ctrl.name.value)
|
||||||
|
assertEquals("short 2", ctrl.shortDescription.value)
|
||||||
|
assertEquals("long 2", ctrl.longDescription.value)
|
||||||
|
|
||||||
|
store.makeMainUndoCurrent()
|
||||||
|
store.undo()
|
||||||
|
store.undo()
|
||||||
|
store.undo()
|
||||||
|
store.undo()
|
||||||
|
|
||||||
|
assertEquals(1, ctrl.id.value)
|
||||||
|
assertEquals("name 1", ctrl.name.value)
|
||||||
|
assertEquals("short 1", ctrl.shortDescription.value)
|
||||||
|
assertEquals("long 1", ctrl.longDescription.value)
|
||||||
|
|
||||||
|
store.redo()
|
||||||
|
store.redo()
|
||||||
|
store.redo()
|
||||||
|
store.redo()
|
||||||
|
|
||||||
|
assertEquals(2, ctrl.id.value)
|
||||||
|
assertEquals("name 2", ctrl.name.value)
|
||||||
|
assertEquals("short 2", ctrl.shortDescription.value)
|
||||||
|
assertEquals("long 2", ctrl.longDescription.value)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,12 @@
|
|||||||
|
const MonacoWebpackPlugin = require("monaco-editor-webpack-plugin");
|
||||||
|
|
||||||
config.module.rules.push({
|
config.module.rules.push({
|
||||||
test: /\.(gif|jpg|png|svg|ttf)$/,
|
test: /\.(gif|jpg|png|svg|ttf)$/,
|
||||||
loader: "file-loader",
|
loader: "file-loader",
|
||||||
});
|
});
|
||||||
|
|
||||||
|
config.plugins.push(
|
||||||
|
new MonacoWebpackPlugin({
|
||||||
|
languages: [],
|
||||||
|
})
|
||||||
|
);
|
||||||
|
@ -5,6 +5,8 @@ import org.w3c.dom.HTMLInputElement
|
|||||||
import world.phantasmal.observable.value.Val
|
import world.phantasmal.observable.value.Val
|
||||||
import world.phantasmal.observable.value.nullVal
|
import world.phantasmal.observable.value.nullVal
|
||||||
import world.phantasmal.observable.value.trueVal
|
import world.phantasmal.observable.value.trueVal
|
||||||
|
import world.phantasmal.observable.value.value
|
||||||
|
import kotlin.math.abs
|
||||||
import kotlin.math.pow
|
import kotlin.math.pow
|
||||||
import kotlin.math.round
|
import kotlin.math.round
|
||||||
|
|
||||||
@ -16,8 +18,7 @@ class DoubleInput(
|
|||||||
label: String? = null,
|
label: String? = null,
|
||||||
labelVal: Val<String>? = null,
|
labelVal: Val<String>? = null,
|
||||||
preferredLabelPosition: LabelPosition = LabelPosition.Before,
|
preferredLabelPosition: LabelPosition = LabelPosition.Before,
|
||||||
value: Double? = null,
|
value: Val<Double> = value(0.0),
|
||||||
valueVal: Val<Double>? = null,
|
|
||||||
onChange: (Double) -> Unit = {},
|
onChange: (Double) -> Unit = {},
|
||||||
roundTo: Int = 2,
|
roundTo: Int = 2,
|
||||||
) : NumberInput<Double>(
|
) : NumberInput<Double>(
|
||||||
@ -29,7 +30,6 @@ class DoubleInput(
|
|||||||
labelVal,
|
labelVal,
|
||||||
preferredLabelPosition,
|
preferredLabelPosition,
|
||||||
value,
|
value,
|
||||||
valueVal,
|
|
||||||
onChange,
|
onChange,
|
||||||
min = null,
|
min = null,
|
||||||
max = null,
|
max = null,
|
||||||
@ -43,4 +43,7 @@ class DoubleInput(
|
|||||||
override fun setInputValue(input: HTMLInputElement, value: Double) {
|
override fun setInputValue(input: HTMLInputElement, value: Double) {
|
||||||
input.valueAsNumber = round(value * roundingFactor) / roundingFactor
|
input.valueAsNumber = round(value * roundingFactor) / roundingFactor
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override fun valuesEqual(a: Double, b: Double): Boolean =
|
||||||
|
abs(a - b) * roundingFactor < 1.0
|
||||||
}
|
}
|
||||||
|
@ -18,8 +18,7 @@ abstract class Input<T>(
|
|||||||
private val className: String,
|
private val className: String,
|
||||||
private val inputClassName: String,
|
private val inputClassName: String,
|
||||||
private val inputType: String,
|
private val inputType: String,
|
||||||
private val value: T?,
|
private val value: Val<T>,
|
||||||
private val valueVal: Val<T>?,
|
|
||||||
private val onChange: (T) -> Unit,
|
private val onChange: (T) -> Unit,
|
||||||
private val maxLength: Int?,
|
private val maxLength: Int?,
|
||||||
private val min: Int?,
|
private val min: Int?,
|
||||||
@ -34,6 +33,8 @@ abstract class Input<T>(
|
|||||||
labelVal,
|
labelVal,
|
||||||
preferredLabelPosition,
|
preferredLabelPosition,
|
||||||
) {
|
) {
|
||||||
|
private var settingValue = false
|
||||||
|
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
span {
|
span {
|
||||||
classList.add("pw-input", this@Input.className)
|
classList.add("pw-input", this@Input.className)
|
||||||
@ -44,18 +45,16 @@ abstract class Input<T>(
|
|||||||
|
|
||||||
observe(this@Input.enabled) { disabled = !it }
|
observe(this@Input.enabled) { disabled = !it }
|
||||||
|
|
||||||
onchange = { onChange(getInputValue(this)) }
|
onchange = { callOnChange(this) }
|
||||||
|
|
||||||
onkeydown = { e ->
|
onkeydown = { e ->
|
||||||
if (e.key == "Enter") {
|
if (e.key == "Enter") {
|
||||||
onChange(getInputValue(this))
|
callOnChange(this)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (valueVal != null) {
|
observe(this@Input.value) {
|
||||||
observe(valueVal) { setInputValue(this, it) }
|
setInputValue(this, it)
|
||||||
} else if (this@Input.value != null) {
|
|
||||||
setInputValue(this, this@Input.value)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
this@Input.maxLength?.let { maxLength = it }
|
this@Input.maxLength?.let { maxLength = it }
|
||||||
@ -69,6 +68,17 @@ abstract class Input<T>(
|
|||||||
|
|
||||||
protected abstract fun setInputValue(input: HTMLInputElement, value: T)
|
protected abstract fun setInputValue(input: HTMLInputElement, value: T)
|
||||||
|
|
||||||
|
private fun callOnChange(input: HTMLInputElement) {
|
||||||
|
val v = getInputValue(input)
|
||||||
|
|
||||||
|
if (!valuesEqual(v, this@Input.value.value)) {
|
||||||
|
onChange(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected open fun valuesEqual(a: T, b: T): Boolean =
|
||||||
|
a == b
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
init {
|
init {
|
||||||
@Suppress("CssUnusedSymbol", "CssUnresolvedCustomProperty")
|
@Suppress("CssUnusedSymbol", "CssUnresolvedCustomProperty")
|
||||||
|
@ -5,6 +5,7 @@ import org.w3c.dom.HTMLInputElement
|
|||||||
import world.phantasmal.observable.value.Val
|
import world.phantasmal.observable.value.Val
|
||||||
import world.phantasmal.observable.value.nullVal
|
import world.phantasmal.observable.value.nullVal
|
||||||
import world.phantasmal.observable.value.trueVal
|
import world.phantasmal.observable.value.trueVal
|
||||||
|
import world.phantasmal.observable.value.value
|
||||||
|
|
||||||
class IntInput(
|
class IntInput(
|
||||||
scope: CoroutineScope,
|
scope: CoroutineScope,
|
||||||
@ -14,8 +15,7 @@ class IntInput(
|
|||||||
label: String? = null,
|
label: String? = null,
|
||||||
labelVal: Val<String>? = null,
|
labelVal: Val<String>? = null,
|
||||||
preferredLabelPosition: LabelPosition = LabelPosition.Before,
|
preferredLabelPosition: LabelPosition = LabelPosition.Before,
|
||||||
value: Int? = null,
|
value: Val<Int> = value(0),
|
||||||
valueVal: Val<Int>? = null,
|
|
||||||
onChange: (Int) -> Unit = {},
|
onChange: (Int) -> Unit = {},
|
||||||
min: Int? = null,
|
min: Int? = null,
|
||||||
max: Int? = null,
|
max: Int? = null,
|
||||||
@ -29,7 +29,6 @@ class IntInput(
|
|||||||
labelVal,
|
labelVal,
|
||||||
preferredLabelPosition,
|
preferredLabelPosition,
|
||||||
value,
|
value,
|
||||||
valueVal,
|
|
||||||
onChange,
|
onChange,
|
||||||
min,
|
min,
|
||||||
max,
|
max,
|
||||||
|
@ -11,8 +11,7 @@ abstract class NumberInput<T : Number>(
|
|||||||
label: String?,
|
label: String?,
|
||||||
labelVal: Val<String>?,
|
labelVal: Val<String>?,
|
||||||
preferredLabelPosition: LabelPosition,
|
preferredLabelPosition: LabelPosition,
|
||||||
value: T?,
|
value: Val<T>,
|
||||||
valueVal: Val<T>?,
|
|
||||||
onChange: (T) -> Unit,
|
onChange: (T) -> Unit,
|
||||||
min: Int?,
|
min: Int?,
|
||||||
max: Int?,
|
max: Int?,
|
||||||
@ -29,7 +28,6 @@ abstract class NumberInput<T : Number>(
|
|||||||
inputClassName = "pw-number-input-inner",
|
inputClassName = "pw-number-input-inner",
|
||||||
inputType = "number",
|
inputType = "number",
|
||||||
value,
|
value,
|
||||||
valueVal,
|
|
||||||
onChange,
|
onChange,
|
||||||
maxLength = null,
|
maxLength = null,
|
||||||
min,
|
min,
|
||||||
|
@ -5,6 +5,7 @@ import org.w3c.dom.Node
|
|||||||
import world.phantasmal.observable.value.Val
|
import world.phantasmal.observable.value.Val
|
||||||
import world.phantasmal.observable.value.nullVal
|
import world.phantasmal.observable.value.nullVal
|
||||||
import world.phantasmal.observable.value.trueVal
|
import world.phantasmal.observable.value.trueVal
|
||||||
|
import world.phantasmal.observable.value.value
|
||||||
import world.phantasmal.webui.dom.div
|
import world.phantasmal.webui.dom.div
|
||||||
import world.phantasmal.webui.dom.textarea
|
import world.phantasmal.webui.dom.textarea
|
||||||
|
|
||||||
@ -16,9 +17,8 @@ class TextArea(
|
|||||||
label: String? = null,
|
label: String? = null,
|
||||||
labelVal: Val<String>? = null,
|
labelVal: Val<String>? = null,
|
||||||
preferredLabelPosition: LabelPosition = LabelPosition.Before,
|
preferredLabelPosition: LabelPosition = LabelPosition.Before,
|
||||||
private val value: String? = null,
|
private val value: Val<String> = value(""),
|
||||||
private val valueVal: Val<String>? = null,
|
private val onChange: ((String) -> Unit)? = null,
|
||||||
private val setValue: ((String) -> Unit)? = null,
|
|
||||||
private val maxLength: Int? = null,
|
private val maxLength: Int? = null,
|
||||||
private val fontFamily: String? = null,
|
private val fontFamily: String? = null,
|
||||||
private val rows: Int? = null,
|
private val rows: Int? = null,
|
||||||
@ -41,15 +41,11 @@ class TextArea(
|
|||||||
|
|
||||||
observe(this@TextArea.enabled) { disabled = !it }
|
observe(this@TextArea.enabled) { disabled = !it }
|
||||||
|
|
||||||
if (setValue != null) {
|
if (onChange != null) {
|
||||||
onchange = { setValue.invoke(value) }
|
onchange = { onChange.invoke(value) }
|
||||||
}
|
}
|
||||||
|
|
||||||
if (valueVal != null) {
|
observe(this@TextArea.value) { value = it }
|
||||||
observe(valueVal) { value = it }
|
|
||||||
} else if (this@TextArea.value != null) {
|
|
||||||
value = this@TextArea.value
|
|
||||||
}
|
|
||||||
|
|
||||||
this@TextArea.maxLength?.let { maxLength = it }
|
this@TextArea.maxLength?.let { maxLength = it }
|
||||||
fontFamily?.let { style.fontFamily = it }
|
fontFamily?.let { style.fontFamily = it }
|
||||||
|
@ -5,6 +5,7 @@ import org.w3c.dom.HTMLInputElement
|
|||||||
import world.phantasmal.observable.value.Val
|
import world.phantasmal.observable.value.Val
|
||||||
import world.phantasmal.observable.value.nullVal
|
import world.phantasmal.observable.value.nullVal
|
||||||
import world.phantasmal.observable.value.trueVal
|
import world.phantasmal.observable.value.trueVal
|
||||||
|
import world.phantasmal.observable.value.value
|
||||||
|
|
||||||
class TextInput(
|
class TextInput(
|
||||||
scope: CoroutineScope,
|
scope: CoroutineScope,
|
||||||
@ -14,8 +15,7 @@ class TextInput(
|
|||||||
label: String? = null,
|
label: String? = null,
|
||||||
labelVal: Val<String>? = null,
|
labelVal: Val<String>? = null,
|
||||||
preferredLabelPosition: LabelPosition = LabelPosition.Before,
|
preferredLabelPosition: LabelPosition = LabelPosition.Before,
|
||||||
value: String? = null,
|
value: Val<String> = value(""),
|
||||||
valueVal: Val<String>? = null,
|
|
||||||
onChange: (String) -> Unit = {},
|
onChange: (String) -> Unit = {},
|
||||||
maxLength: Int? = null,
|
maxLength: Int? = null,
|
||||||
) : Input<String>(
|
) : Input<String>(
|
||||||
@ -30,7 +30,6 @@ class TextInput(
|
|||||||
inputClassName = "pw-number-text-inner",
|
inputClassName = "pw-number-text-inner",
|
||||||
inputType = "text",
|
inputType = "text",
|
||||||
value,
|
value,
|
||||||
valueVal,
|
|
||||||
onChange,
|
onChange,
|
||||||
maxLength,
|
maxLength,
|
||||||
min = null,
|
min = null,
|
||||||
|
Loading…
Reference in New Issue
Block a user