mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-04 22:58:29 +08:00
Added horizontal entity translation.
This commit is contained in:
parent
990a8c144f
commit
bb6f4aa352
@ -39,7 +39,7 @@ dependencies {
|
|||||||
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.kotlinx:kotlinx-serialization-core-js:1.0.0")
|
||||||
implementation(npm("golden-layout", "1.5.9"))
|
implementation(npm("golden-layout", "1.5.9"))
|
||||||
implementation(npm("@babylonjs/core", "4.1.0"))
|
implementation(npm("@babylonjs/core", "4.2.0-rc.5"))
|
||||||
|
|
||||||
testImplementation(kotlin("test-js"))
|
testImplementation(kotlin("test-js"))
|
||||||
testImplementation(project(":test-utils"))
|
testImplementation(project(":test-utils"))
|
||||||
|
@ -8,6 +8,8 @@ import kotlinx.browser.window
|
|||||||
import kotlinx.coroutines.CoroutineScope
|
import kotlinx.coroutines.CoroutineScope
|
||||||
import kotlinx.coroutines.SupervisorJob
|
import kotlinx.coroutines.SupervisorJob
|
||||||
import kotlinx.coroutines.cancel
|
import kotlinx.coroutines.cancel
|
||||||
|
import mu.KotlinLoggingConfiguration
|
||||||
|
import mu.KotlinLoggingLevel
|
||||||
import org.w3c.dom.PopStateEvent
|
import org.w3c.dom.PopStateEvent
|
||||||
import world.phantasmal.core.disposable.Disposable
|
import world.phantasmal.core.disposable.Disposable
|
||||||
import world.phantasmal.core.disposable.Disposer
|
import world.phantasmal.core.disposable.Disposer
|
||||||
@ -30,6 +32,10 @@ fun main() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun init(): Disposable {
|
private fun init(): Disposable {
|
||||||
|
if (window.location.hostname == "localhost") {
|
||||||
|
KotlinLoggingConfiguration.LOG_LEVEL = KotlinLoggingLevel.TRACE
|
||||||
|
}
|
||||||
|
|
||||||
val disposer = Disposer()
|
val disposer = Disposer()
|
||||||
|
|
||||||
val rootElement = document.body!!.root()
|
val rootElement = document.body!!.root()
|
||||||
|
@ -4,6 +4,9 @@ import world.phantasmal.web.externals.babylon.Matrix
|
|||||||
import world.phantasmal.web.externals.babylon.Quaternion
|
import world.phantasmal.web.externals.babylon.Quaternion
|
||||||
import world.phantasmal.web.externals.babylon.Vector3
|
import world.phantasmal.web.externals.babylon.Vector3
|
||||||
|
|
||||||
|
operator fun Vector3.plus(other: Vector3): Vector3 =
|
||||||
|
add(other)
|
||||||
|
|
||||||
operator fun Vector3.plusAssign(other: Vector3) {
|
operator fun Vector3.plusAssign(other: Vector3) {
|
||||||
addInPlace(other)
|
addInPlace(other)
|
||||||
}
|
}
|
||||||
@ -15,6 +18,9 @@ operator fun Vector3.minusAssign(other: Vector3) {
|
|||||||
subtractInPlace(other)
|
subtractInPlace(other)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
operator fun Vector3.times(scalar: Double): Vector3 =
|
||||||
|
scale(scalar)
|
||||||
|
|
||||||
infix fun Vector3.dot(other: Vector3): Double =
|
infix fun Vector3.dot(other: Vector3): Double =
|
||||||
Vector3.Dot(this, other)
|
Vector3.Dot(this, other)
|
||||||
|
|
||||||
@ -41,3 +47,21 @@ fun Matrix.multiply3x3(v: Vector3) {
|
|||||||
operator fun Quaternion.timesAssign(other: Quaternion) {
|
operator fun Quaternion.timesAssign(other: Quaternion) {
|
||||||
multiplyInPlace(other)
|
multiplyInPlace(other)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a new quaternion that's the inverse of this quaternion.
|
||||||
|
*/
|
||||||
|
fun Quaternion.inverse(): Quaternion = Quaternion.Inverse(this)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transforms [p] by this versor.
|
||||||
|
*/
|
||||||
|
fun Quaternion.transform(p: Vector3) {
|
||||||
|
p.rotateByQuaternionToRef(this, p)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a new point equal to [p] transformed by this versor.
|
||||||
|
*/
|
||||||
|
fun Quaternion.transformed(p: Vector3): Vector3 =
|
||||||
|
p.rotateByQuaternionToRef(this, Vector3.Zero())
|
||||||
|
@ -35,6 +35,7 @@ external class Vector3(x: Double, y: Double, z: Double) {
|
|||||||
|
|
||||||
fun set(x: Double, y: Double, z: Double): Vector2
|
fun set(x: Double, y: Double, z: Double): Vector2
|
||||||
fun toQuaternion(): Quaternion
|
fun toQuaternion(): Quaternion
|
||||||
|
fun add(otherVector: Vector3): Vector3
|
||||||
fun addInPlace(otherVector: Vector3): Vector3
|
fun addInPlace(otherVector: Vector3): Vector3
|
||||||
fun addInPlaceFromFloats(x: Double, y: Double, z: Double): Vector3
|
fun addInPlaceFromFloats(x: Double, y: Double, z: Double): Vector3
|
||||||
fun subtract(otherVector: Vector3): Vector3
|
fun subtract(otherVector: Vector3): Vector3
|
||||||
@ -42,6 +43,19 @@ external class Vector3(x: Double, y: Double, z: Double) {
|
|||||||
fun negate(): Vector3
|
fun negate(): Vector3
|
||||||
fun negateInPlace(): Vector3
|
fun negateInPlace(): Vector3
|
||||||
fun cross(other: Vector3): Vector3
|
fun cross(other: Vector3): Vector3
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a new Vector3 set with the current Vector3 coordinates multiplied by the float "scale"
|
||||||
|
*/
|
||||||
|
fun scale(scale: Double): Vector3
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Multiplies the Vector3 coordinates by the float "scale"
|
||||||
|
*
|
||||||
|
* @return the current updated Vector3
|
||||||
|
*/
|
||||||
|
fun scaleInPlace(scale: Double): Vector3
|
||||||
|
|
||||||
fun rotateByQuaternionToRef(quaternion: Quaternion, result: Vector3): Vector3
|
fun rotateByQuaternionToRef(quaternion: Quaternion, result: Vector3): Vector3
|
||||||
fun clone(): Vector3
|
fun clone(): Vector3
|
||||||
fun copyFrom(source: Vector3): Vector3
|
fun copyFrom(source: Vector3): Vector3
|
||||||
@ -94,6 +108,7 @@ external class Quaternion(
|
|||||||
fun FromEulerAngles(x: Double, y: Double, z: Double): Quaternion
|
fun FromEulerAngles(x: Double, y: Double, z: Double): 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
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -105,6 +120,8 @@ external class Matrix {
|
|||||||
fun equals(value: Matrix): Boolean
|
fun equals(value: Matrix): Boolean
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
|
val IdentityReadOnly: Matrix
|
||||||
|
|
||||||
fun Identity(): Matrix
|
fun Identity(): Matrix
|
||||||
fun Compose(scale: Vector3, rotation: Quaternion, translation: Vector3): Matrix
|
fun Compose(scale: Vector3, rotation: Quaternion, translation: Vector3): Matrix
|
||||||
}
|
}
|
||||||
@ -157,7 +174,17 @@ external class Engine(
|
|||||||
antialias: Boolean = definedExternally,
|
antialias: Boolean = definedExternally,
|
||||||
) : ThinEngine
|
) : ThinEngine
|
||||||
|
|
||||||
external class Ray(origin: Vector3, direction: Vector3, length: Double = definedExternally)
|
external class Ray(origin: Vector3, direction: Vector3, length: Double = definedExternally) {
|
||||||
|
var origin: Vector3
|
||||||
|
var direction: Vector3
|
||||||
|
var length: Double
|
||||||
|
|
||||||
|
fun intersectsPlane(plane: Plane): Double?
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
fun Zero(): Ray
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
external class PickingInfo {
|
external class PickingInfo {
|
||||||
val bu: Double
|
val bu: Double
|
||||||
@ -191,6 +218,31 @@ external class Scene(engine: Engine) {
|
|||||||
fun removeLight(toRemove: Light)
|
fun removeLight(toRemove: Light)
|
||||||
fun removeMesh(toRemove: TransformNode, recursive: Boolean? = definedExternally)
|
fun removeMesh(toRemove: TransformNode, recursive: Boolean? = definedExternally)
|
||||||
fun removeTransformNode(toRemove: TransformNode)
|
fun removeTransformNode(toRemove: TransformNode)
|
||||||
|
|
||||||
|
fun createPickingRay(
|
||||||
|
x: Double,
|
||||||
|
y: Double,
|
||||||
|
world: Matrix,
|
||||||
|
camera: Camera?,
|
||||||
|
cameraViewSpace: Boolean = definedExternally,
|
||||||
|
): Ray
|
||||||
|
|
||||||
|
fun createPickingRayToRef(
|
||||||
|
x: Double,
|
||||||
|
y: Double,
|
||||||
|
world: Matrix,
|
||||||
|
result: Ray,
|
||||||
|
camera: Camera?,
|
||||||
|
cameraViewSpace: Boolean = definedExternally,
|
||||||
|
): Scene
|
||||||
|
|
||||||
|
fun createPickingRayInCameraSpaceToRef(
|
||||||
|
x: Double,
|
||||||
|
y: Double,
|
||||||
|
result: Ray,
|
||||||
|
camera: Camera = definedExternally,
|
||||||
|
): Scene
|
||||||
|
|
||||||
fun pick(
|
fun pick(
|
||||||
x: Double,
|
x: Double,
|
||||||
y: Double,
|
y: Double,
|
||||||
@ -233,10 +285,9 @@ open external class Node {
|
|||||||
var metadata: Any?
|
var metadata: Any?
|
||||||
var parent: Node?
|
var parent: Node?
|
||||||
|
|
||||||
|
fun isEnabled(checkAncestors: Boolean = definedExternally): Boolean
|
||||||
fun setEnabled(value: Boolean)
|
fun setEnabled(value: Boolean)
|
||||||
fun getViewMatrix(force: Boolean = definedExternally): Matrix
|
fun getWorldMatrix(): Matrix
|
||||||
fun getProjectionMatrix(force: Boolean = definedExternally): Matrix
|
|
||||||
fun getTransformationMatrix(): Matrix
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Releases resources associated with this node.
|
* Releases resources associated with this node.
|
||||||
@ -257,7 +308,11 @@ open external class Camera : Node {
|
|||||||
val onViewMatrixChangedObservable: Observable<Camera>
|
val onViewMatrixChangedObservable: Observable<Camera>
|
||||||
val onAfterCheckInputsObservable: Observable<Camera>
|
val onAfterCheckInputsObservable: Observable<Camera>
|
||||||
|
|
||||||
|
fun getViewMatrix(force: Boolean = definedExternally): Matrix
|
||||||
|
fun getProjectionMatrix(force: Boolean = definedExternally): Matrix
|
||||||
|
fun getTransformationMatrix(): Matrix
|
||||||
fun attachControl(noPreventDefault: Boolean = definedExternally)
|
fun attachControl(noPreventDefault: Boolean = definedExternally)
|
||||||
|
fun detachControl()
|
||||||
fun storeState(): Camera
|
fun storeState(): Camera
|
||||||
fun restoreState(): Boolean
|
fun restoreState(): Boolean
|
||||||
}
|
}
|
||||||
@ -290,6 +345,7 @@ external class ArcRotateCamera(
|
|||||||
var pinchDeltaPercentage: Double
|
var pinchDeltaPercentage: Double
|
||||||
var wheelDeltaPercentage: Double
|
var wheelDeltaPercentage: Double
|
||||||
var lowerBetaLimit: Double
|
var lowerBetaLimit: Double
|
||||||
|
val inputs: ArcRotateCameraInputsManager
|
||||||
|
|
||||||
fun attachControl(
|
fun attachControl(
|
||||||
element: HTMLCanvasElement,
|
element: HTMLCanvasElement,
|
||||||
@ -299,6 +355,13 @@ external class ArcRotateCamera(
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
open external class CameraInputsManager<TCamera : Camera> {
|
||||||
|
fun attachElement(noPreventDefault: Boolean = definedExternally)
|
||||||
|
fun detachElement(disconnect: Boolean = definedExternally)
|
||||||
|
}
|
||||||
|
|
||||||
|
external class ArcRotateCameraInputsManager : CameraInputsManager<ArcRotateCamera>
|
||||||
|
|
||||||
abstract external class Light : Node
|
abstract external class Light : Node
|
||||||
|
|
||||||
external class HemisphericLight(name: String, direction: Vector3, scene: Scene) : Light {
|
external class HemisphericLight(name: String, direction: Vector3, scene: Scene) : Light {
|
||||||
@ -389,6 +452,18 @@ external class MeshBuilder {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
external class Plane(a: Double, b: Double, c: Double, d: Double) {
|
||||||
|
var normal: Vector3
|
||||||
|
var d: Double
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
/**
|
||||||
|
* Note : the vector "normal" is updated because normalized.
|
||||||
|
*/
|
||||||
|
fun FromPositionAndNormal(origin: Vector3, normal: Vector3): Plane
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
external class VertexData {
|
external class VertexData {
|
||||||
var positions: Float32Array? // number[] | Float32Array
|
var positions: Float32Array? // number[] | Float32Array
|
||||||
var normals: Float32Array? // number[] | Float32Array
|
var normals: Float32Array? // number[] | Float32Array
|
||||||
|
@ -68,7 +68,7 @@ class AreaAssetLoader(
|
|||||||
scope.async {
|
scope.async {
|
||||||
val buffer = getAreaAsset(episode, areaVariant, AssetType.Collision)
|
val buffer = getAreaAsset(episode, areaVariant, AssetType.Collision)
|
||||||
val obj = parseAreaCollisionGeometry(buffer.cursor(Endianness.Little))
|
val obj = parseAreaCollisionGeometry(buffer.cursor(Endianness.Little))
|
||||||
areaCollisionGeometryToTransformNode(scene, obj)
|
areaCollisionGeometryToTransformNode(scene, obj, episode, areaVariant)
|
||||||
}
|
}
|
||||||
}.await()
|
}.await()
|
||||||
|
|
||||||
@ -226,10 +226,15 @@ private fun areaGeometryToTransformNodeAndSections(
|
|||||||
private fun areaCollisionGeometryToTransformNode(
|
private fun areaCollisionGeometryToTransformNode(
|
||||||
scene: Scene,
|
scene: Scene,
|
||||||
obj: CollisionObject,
|
obj: CollisionObject,
|
||||||
|
episode: Episode,
|
||||||
|
areaVariant: AreaVariantModel,
|
||||||
): TransformNode {
|
): TransformNode {
|
||||||
val node = TransformNode("Collision Geometry", scene)
|
val node = TransformNode(
|
||||||
|
"Collision Geometry $episode-${areaVariant.area.id}-${areaVariant.id}",
|
||||||
|
scene
|
||||||
|
)
|
||||||
|
|
||||||
for (collisionMesh in obj.meshes) {
|
obj.meshes.forEachIndexed { i, collisionMesh ->
|
||||||
val builder = VertexDataBuilder()
|
val builder = VertexDataBuilder()
|
||||||
|
|
||||||
for (triangle in collisionMesh.triangles) {
|
for (triangle in collisionMesh.triangles) {
|
||||||
@ -260,7 +265,11 @@ private fun areaCollisionGeometryToTransformNode(
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (builder.vertexCount > 0) {
|
if (builder.vertexCount > 0) {
|
||||||
val mesh = Mesh("Collision Geometry", scene, parent = node)
|
val mesh = Mesh(
|
||||||
|
"Collision Geometry $episode-${areaVariant.area.id}-${areaVariant.id}-$i",
|
||||||
|
scene,
|
||||||
|
parent = node
|
||||||
|
)
|
||||||
builder.build().applyToMesh(mesh)
|
builder.build().applyToMesh(mesh)
|
||||||
mesh.metadata = CollisionMetadata()
|
mesh.metadata = CollisionMetadata()
|
||||||
}
|
}
|
||||||
|
@ -5,10 +5,9 @@ import world.phantasmal.lib.fileFormats.quest.EntityType
|
|||||||
import world.phantasmal.lib.fileFormats.quest.QuestEntity
|
import world.phantasmal.lib.fileFormats.quest.QuestEntity
|
||||||
import world.phantasmal.observable.value.Val
|
import world.phantasmal.observable.value.Val
|
||||||
import world.phantasmal.observable.value.mutableVal
|
import world.phantasmal.observable.value.mutableVal
|
||||||
import world.phantasmal.web.core.plusAssign
|
import world.phantasmal.web.core.*
|
||||||
import world.phantasmal.web.core.rendering.conversion.babylonToVec3
|
import world.phantasmal.web.core.rendering.conversion.babylonToVec3
|
||||||
import world.phantasmal.web.core.rendering.conversion.vec3ToBabylon
|
import world.phantasmal.web.core.rendering.conversion.vec3ToBabylon
|
||||||
import world.phantasmal.web.core.timesAssign
|
|
||||||
import world.phantasmal.web.externals.babylon.Quaternion
|
import world.phantasmal.web.externals.babylon.Quaternion
|
||||||
import world.phantasmal.web.externals.babylon.Vector3
|
import world.phantasmal.web.externals.babylon.Vector3
|
||||||
import kotlin.math.PI
|
import kotlin.math.PI
|
||||||
@ -75,11 +74,24 @@ abstract class QuestEntityModel<Type : EntityType, Entity : QuestEntity<Type>>(
|
|||||||
val section = section.value
|
val section = section.value
|
||||||
|
|
||||||
_worldPosition.value =
|
_worldPosition.value =
|
||||||
|
section?.rotationQuaternion?.transformed(pos)?.also {
|
||||||
|
it += section.position
|
||||||
|
} ?: pos
|
||||||
|
}
|
||||||
|
|
||||||
|
fun setWorldPosition(pos: Vector3) {
|
||||||
|
_worldPosition.value = pos
|
||||||
|
|
||||||
|
val section = section.value
|
||||||
|
|
||||||
|
val relPos =
|
||||||
if (section == null) pos
|
if (section == null) pos
|
||||||
else Vector3.Zero().also { worldPos ->
|
else (pos - section.position).also {
|
||||||
pos.rotateByQuaternionToRef(section.rotationQuaternion, worldPos)
|
section.inverseRotationQuaternion.transform(it)
|
||||||
worldPos += section.position
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
entity.position = babylonToVec3(relPos)
|
||||||
|
_position.value = relPos
|
||||||
}
|
}
|
||||||
|
|
||||||
fun setRotation(rot: Vector3) {
|
fun setRotation(rot: Vector3) {
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
package world.phantasmal.web.questEditor.models
|
package world.phantasmal.web.questEditor.models
|
||||||
|
|
||||||
|
import world.phantasmal.web.core.inverse
|
||||||
import world.phantasmal.web.externals.babylon.Quaternion
|
import world.phantasmal.web.externals.babylon.Quaternion
|
||||||
import world.phantasmal.web.externals.babylon.Vector3
|
import world.phantasmal.web.externals.babylon.Vector3
|
||||||
|
|
||||||
@ -17,4 +18,6 @@ class SectionModel(
|
|||||||
|
|
||||||
val rotationQuaternion: Quaternion =
|
val rotationQuaternion: Quaternion =
|
||||||
Quaternion.FromEulerAngles(rotation.x, rotation.y, rotation.z)
|
Quaternion.FromEulerAngles(rotation.x, rotation.y, rotation.z)
|
||||||
|
|
||||||
|
val inverseRotationQuaternion: Quaternion = rotationQuaternion.inverse()
|
||||||
}
|
}
|
||||||
|
@ -20,8 +20,10 @@ class AreaMeshManager(
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
val geom = areaAssetLoader.loadCollisionGeometry(episode, areaVariant)
|
val geom = areaAssetLoader.loadCollisionGeometry(episode, areaVariant)
|
||||||
geom.setEnabled(true)
|
// Call setEnabled(false) on renderer.collisionGeometry before calling setEnabled(true)
|
||||||
|
// on geom, because they can refer to the same object.
|
||||||
renderer.collisionGeometry?.setEnabled(false)
|
renderer.collisionGeometry?.setEnabled(false)
|
||||||
|
geom.setEnabled(true)
|
||||||
renderer.collisionGeometry = geom
|
renderer.collisionGeometry = geom
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
logger.error(e) {
|
logger.error(e) {
|
||||||
|
@ -16,12 +16,6 @@ class QuestRenderer(canvas: HTMLCanvasElement, engine: Engine) : Renderer(canvas
|
|||||||
|
|
||||||
init {
|
init {
|
||||||
with(camera) {
|
with(camera) {
|
||||||
attachControl(
|
|
||||||
canvas,
|
|
||||||
noPreventDefault = false,
|
|
||||||
useCtrlForPanning = false,
|
|
||||||
panningMouseButton = 0
|
|
||||||
)
|
|
||||||
inertia = 0.0
|
inertia = 0.0
|
||||||
angularSensibilityX = 200.0
|
angularSensibilityX = 200.0
|
||||||
angularSensibilityY = 200.0
|
angularSensibilityY = 200.0
|
||||||
@ -38,6 +32,8 @@ class QuestRenderer(canvas: HTMLCanvasElement, engine: Engine) : Renderer(canvas
|
|||||||
updatePanningSensibility()
|
updatePanningSensibility()
|
||||||
})
|
})
|
||||||
|
|
||||||
|
enableCameraControls()
|
||||||
|
|
||||||
camera.storeState()
|
camera.storeState()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -46,6 +42,19 @@ class QuestRenderer(canvas: HTMLCanvasElement, engine: Engine) : Renderer(canvas
|
|||||||
camera.restoreState()
|
camera.restoreState()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun enableCameraControls() {
|
||||||
|
camera.attachControl(
|
||||||
|
canvas,
|
||||||
|
noPreventDefault = false,
|
||||||
|
useCtrlForPanning = false,
|
||||||
|
panningMouseButton = 0
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun disableCameraControls() {
|
||||||
|
camera.detachControl()
|
||||||
|
}
|
||||||
|
|
||||||
override fun render() {
|
override fun render() {
|
||||||
camera.minZ = max(0.01, camera.radius / 100)
|
camera.minZ = max(0.01, camera.radius / 100)
|
||||||
camera.maxZ = max(2_000.0, 10 * camera.radius)
|
camera.maxZ = max(2_000.0, 10 * camera.radius)
|
||||||
@ -53,8 +62,8 @@ class QuestRenderer(canvas: HTMLCanvasElement, engine: Engine) : Renderer(canvas
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Make "panningSensibility" an inverse function of radius to make panning work "sensibly"
|
* Make "panningSensibility" an inverse function of radius to make panning work "sensibly" at
|
||||||
* at all distances.
|
* all distances.
|
||||||
*/
|
*/
|
||||||
private fun updatePanningSensibility() {
|
private fun updatePanningSensibility() {
|
||||||
camera.panningSensibility = 1_000 / camera.radius
|
camera.panningSensibility = 1_000 / camera.radius
|
||||||
|
@ -5,25 +5,31 @@ import mu.KotlinLogging
|
|||||||
import org.w3c.dom.pointerevents.PointerEvent
|
import org.w3c.dom.pointerevents.PointerEvent
|
||||||
import world.phantasmal.core.disposable.Disposable
|
import world.phantasmal.core.disposable.Disposable
|
||||||
import world.phantasmal.web.core.minus
|
import world.phantasmal.web.core.minus
|
||||||
|
import world.phantasmal.web.core.plusAssign
|
||||||
|
import world.phantasmal.web.core.times
|
||||||
import world.phantasmal.web.externals.babylon.*
|
import world.phantasmal.web.externals.babylon.*
|
||||||
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.stores.QuestEditorStore
|
import world.phantasmal.web.questEditor.stores.QuestEditorStore
|
||||||
import world.phantasmal.webui.DisposableContainer
|
import world.phantasmal.webui.DisposableContainer
|
||||||
import world.phantasmal.webui.dom.disposableListener
|
import world.phantasmal.webui.dom.disposableListener
|
||||||
|
|
||||||
private val logger = KotlinLogging.logger {}
|
private val logger = KotlinLogging.logger {}
|
||||||
|
|
||||||
|
private val ZERO_VECTOR = Vector3.Zero()
|
||||||
private val DOWN_VECTOR = Vector3.Down()
|
private val DOWN_VECTOR = Vector3.Down()
|
||||||
|
|
||||||
class UserInputManager(
|
class UserInputManager(
|
||||||
private val questEditorStore: QuestEditorStore,
|
questEditorStore: QuestEditorStore,
|
||||||
private val renderer: QuestRenderer,
|
private val renderer: QuestRenderer,
|
||||||
) : DisposableContainer() {
|
) : DisposableContainer() {
|
||||||
|
private val stateContext = StateContext(questEditorStore, renderer)
|
||||||
private val pointerPosition = Vector2.Zero()
|
private val pointerPosition = Vector2.Zero()
|
||||||
private val lastPointerPosition = Vector2.Zero()
|
private val lastPointerPosition = Vector2.Zero()
|
||||||
private var movedSinceLastPointerDown = false
|
private var movedSinceLastPointerDown = false
|
||||||
private var state: State
|
private var state: State
|
||||||
private var onPointerUpListener: Disposable? = null
|
private var onPointerUpListener: Disposable? = null
|
||||||
|
private var onPointerMoveListener: Disposable? = null
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Whether entity transformations, deletions, etc. are enabled or not.
|
* Whether entity transformations, deletions, etc. are enabled or not.
|
||||||
@ -33,44 +39,79 @@ class UserInputManager(
|
|||||||
set(enabled) {
|
set(enabled) {
|
||||||
field = enabled
|
field = enabled
|
||||||
state.cancel()
|
state.cancel()
|
||||||
state = IdleState(questEditorStore, renderer, enabled)
|
state = IdleState(stateContext, enabled)
|
||||||
}
|
}
|
||||||
|
|
||||||
init {
|
init {
|
||||||
state = IdleState(questEditorStore, renderer, entityManipulationEnabled)
|
state = IdleState(stateContext, entityManipulationEnabled)
|
||||||
|
|
||||||
observe(questEditorStore.selectedEntity) { state.cancel() }
|
observe(questEditorStore.selectedEntity) { state.cancel() }
|
||||||
|
|
||||||
addDisposables(
|
addDisposables(
|
||||||
disposableListener(renderer.canvas, "pointerdown", ::onPointerDown)
|
disposableListener(renderer.canvas, "pointerdown", ::onPointerDown)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
onPointerMoveListener = disposableListener(document, "pointermove", ::onPointerMove)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun internalDispose() {
|
||||||
|
onPointerUpListener?.dispose()
|
||||||
|
onPointerMoveListener?.dispose()
|
||||||
|
super.internalDispose()
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun onPointerDown(e: PointerEvent) {
|
private fun onPointerDown(e: PointerEvent) {
|
||||||
processPointerEvent(e)
|
processPointerEvent(e)
|
||||||
|
|
||||||
state = state.processEvent(PointerDownEvt(
|
state = state.processEvent(
|
||||||
e.buttons.toInt(),
|
PointerDownEvt(
|
||||||
movedSinceLastPointerDown
|
e.buttons.toInt(),
|
||||||
))
|
shiftKeyDown = e.shiftKey,
|
||||||
|
movedSinceLastPointerDown,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
onPointerUpListener = disposableListener(document, "pointerup", ::onPointerUp)
|
onPointerUpListener = disposableListener(document, "pointerup", ::onPointerUp)
|
||||||
|
|
||||||
|
// Stop listening to canvas move events and start listening to document move events.
|
||||||
|
onPointerMoveListener?.dispose()
|
||||||
|
onPointerMoveListener = disposableListener(document, "pointermove", ::onPointerMove)
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun onPointerUp(e: PointerEvent) {
|
private fun onPointerUp(e: PointerEvent) {
|
||||||
try {
|
try {
|
||||||
processPointerEvent(e)
|
processPointerEvent(e)
|
||||||
|
|
||||||
state = state.processEvent(PointerUpEvt(
|
state = state.processEvent(
|
||||||
e.buttons.toInt(),
|
PointerUpEvt(
|
||||||
movedSinceLastPointerDown
|
e.buttons.toInt(),
|
||||||
))
|
shiftKeyDown = e.shiftKey,
|
||||||
|
movedSinceLastPointerDown,
|
||||||
|
)
|
||||||
|
)
|
||||||
} finally {
|
} finally {
|
||||||
onPointerUpListener?.dispose()
|
onPointerUpListener?.dispose()
|
||||||
onPointerUpListener = null
|
onPointerUpListener = null
|
||||||
|
|
||||||
|
// Stop listening to document move events and start listening to canvas move events.
|
||||||
|
onPointerMoveListener?.dispose()
|
||||||
|
onPointerMoveListener =
|
||||||
|
disposableListener(renderer.canvas, "pointermove", ::onPointerMove)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun onPointerMove(e: PointerEvent) {
|
||||||
|
processPointerEvent(e)
|
||||||
|
|
||||||
|
state = state.processEvent(
|
||||||
|
PointerMoveEvt(
|
||||||
|
e.buttons.toInt(),
|
||||||
|
shiftKeyDown = e.shiftKey,
|
||||||
|
movedSinceLastPointerDown,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
private fun processPointerEvent(e: PointerEvent) {
|
private fun processPointerEvent(e: PointerEvent) {
|
||||||
val rect = renderer.canvas.getBoundingClientRect()
|
val rect = renderer.canvas.getBoundingClientRect()
|
||||||
pointerPosition.set(e.clientX - rect.left, e.clientY - rect.top)
|
pointerPosition.set(e.clientX - rect.left, e.clientY - rect.top)
|
||||||
@ -90,20 +131,134 @@ class UserInputManager(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private class StateContext(
|
||||||
|
private val questEditorStore: QuestEditorStore,
|
||||||
|
val renderer: QuestRenderer,
|
||||||
|
) {
|
||||||
|
private val plane = Plane.FromPositionAndNormal(Vector3.Up(), Vector3.Up())
|
||||||
|
private val ray = Ray.Zero()
|
||||||
|
|
||||||
|
val scene = renderer.scene
|
||||||
|
|
||||||
|
fun setSelectedEntity(entity: QuestEntityModel<*, *>?) {
|
||||||
|
questEditorStore.setSelectedEntity(entity)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun translate(
|
||||||
|
entity: QuestEntityModel<*, *>,
|
||||||
|
dragAdjust: Vector3,
|
||||||
|
grabOffset: Vector3,
|
||||||
|
vertically: Boolean,
|
||||||
|
) {
|
||||||
|
if (vertically) {
|
||||||
|
// TODO: Vertical translation.
|
||||||
|
} else {
|
||||||
|
translateEntityHorizontally(entity, dragAdjust, grabOffset)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If the drag-adjusted pointer is over the ground, translate an entity horizontally across the
|
||||||
|
* ground. Otherwise translate the entity over the horizontal plane that intersects its origin.
|
||||||
|
*/
|
||||||
|
private fun translateEntityHorizontally(
|
||||||
|
entity: QuestEntityModel<*, *>,
|
||||||
|
dragAdjust: Vector3,
|
||||||
|
grabOffset: Vector3,
|
||||||
|
) {
|
||||||
|
val pick = pickGround(scene.pointerX, scene.pointerY, dragAdjust)
|
||||||
|
|
||||||
|
if (pick == null) {
|
||||||
|
// If the pointer is not over the ground, we translate the entity across the horizontal
|
||||||
|
// plane in which the entity's origin lies.
|
||||||
|
scene.createPickingRayToRef(
|
||||||
|
scene.pointerX,
|
||||||
|
scene.pointerY,
|
||||||
|
Matrix.IdentityReadOnly,
|
||||||
|
ray,
|
||||||
|
renderer.camera
|
||||||
|
)
|
||||||
|
|
||||||
|
plane.d = -entity.worldPosition.value.y + grabOffset.y
|
||||||
|
|
||||||
|
ray.intersectsPlane(plane)?.let { distance ->
|
||||||
|
// Compute the intersection point.
|
||||||
|
val pos = ray.direction * distance
|
||||||
|
pos += ray.origin
|
||||||
|
// Compute the entity's new world position.
|
||||||
|
pos.x += grabOffset.x
|
||||||
|
pos.y = entity.worldPosition.value.y
|
||||||
|
pos.z += grabOffset.z
|
||||||
|
|
||||||
|
entity.setWorldPosition(pos)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// TODO: Set entity section.
|
||||||
|
entity.setWorldPosition(
|
||||||
|
Vector3(
|
||||||
|
pick.pickedPoint!!.x,
|
||||||
|
pick.pickedPoint.y + grabOffset.y - dragAdjust.y,
|
||||||
|
pick.pickedPoint.z,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun pickGround(x: Double, y: Double, dragAdjust: Vector3 = ZERO_VECTOR): PickingInfo? {
|
||||||
|
scene.createPickingRayToRef(
|
||||||
|
x,
|
||||||
|
y,
|
||||||
|
Matrix.IdentityReadOnly,
|
||||||
|
ray,
|
||||||
|
renderer.camera
|
||||||
|
)
|
||||||
|
|
||||||
|
ray.origin += dragAdjust
|
||||||
|
|
||||||
|
val pickingInfoArray = scene.multiPickWithRay(
|
||||||
|
ray,
|
||||||
|
{ it.isEnabled() && it.metadata is CollisionMetadata },
|
||||||
|
)
|
||||||
|
|
||||||
|
if (pickingInfoArray != null) {
|
||||||
|
for (pickingInfo in pickingInfoArray) {
|
||||||
|
pickingInfo.getNormal()?.let { n ->
|
||||||
|
// Don't allow entities to be placed on very steep terrain. E.g. walls.
|
||||||
|
// TODO: make use of the flags field in the collision data.
|
||||||
|
if (n.y > 0.75) {
|
||||||
|
return pickingInfo
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private sealed class Evt
|
private sealed class Evt
|
||||||
|
|
||||||
private sealed class PointerEvt : Evt() {
|
private sealed class PointerEvt : Evt() {
|
||||||
abstract val buttons: Int
|
abstract val buttons: Int
|
||||||
|
abstract val shiftKeyDown: Boolean
|
||||||
abstract val movedSinceLastPointerDown: Boolean
|
abstract val movedSinceLastPointerDown: Boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
private class PointerDownEvt(
|
private class PointerDownEvt(
|
||||||
override val buttons: Int,
|
override val buttons: Int,
|
||||||
|
override val shiftKeyDown: Boolean,
|
||||||
override val movedSinceLastPointerDown: Boolean,
|
override val movedSinceLastPointerDown: Boolean,
|
||||||
) : PointerEvt()
|
) : PointerEvt()
|
||||||
|
|
||||||
private class PointerUpEvt(
|
private class PointerUpEvt(
|
||||||
override val buttons: Int,
|
override val buttons: Int,
|
||||||
|
override val shiftKeyDown: Boolean,
|
||||||
|
override val movedSinceLastPointerDown: Boolean,
|
||||||
|
) : PointerEvt()
|
||||||
|
|
||||||
|
private class PointerMoveEvt(
|
||||||
|
override val buttons: Int,
|
||||||
|
override val shiftKeyDown: Boolean,
|
||||||
override val movedSinceLastPointerDown: Boolean,
|
override val movedSinceLastPointerDown: Boolean,
|
||||||
) : PointerEvt()
|
) : PointerEvt()
|
||||||
|
|
||||||
@ -113,13 +268,13 @@ private class Pick(
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Vector that points from the grabbing point (somewhere on the model's surface) to the entity's
|
* Vector that points from the grabbing point (somewhere on the model's surface) to the entity's
|
||||||
* position.
|
* origin.
|
||||||
*/
|
*/
|
||||||
val grabOffset: Vector3,
|
val grabOffset: Vector3,
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Vector that points from the grabbing point to the terrain point directly under the entity's
|
* Vector that points from the grabbing point to the terrain point directly under the entity's
|
||||||
* position.
|
* origin.
|
||||||
*/
|
*/
|
||||||
val dragAdjust: Vector3,
|
val dragAdjust: Vector3,
|
||||||
)
|
)
|
||||||
@ -139,24 +294,28 @@ private abstract class State {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private class IdleState(
|
private class IdleState(
|
||||||
private val questEditorStore: QuestEditorStore,
|
private val ctx: StateContext,
|
||||||
private val renderer: QuestRenderer,
|
|
||||||
private val entityManipulationEnabled: Boolean,
|
private val entityManipulationEnabled: Boolean,
|
||||||
) : State() {
|
) : State() {
|
||||||
override fun processEvent(event: Evt): State =
|
override fun processEvent(event: Evt): State {
|
||||||
when (event) {
|
when (event) {
|
||||||
is PointerDownEvt -> {
|
is PointerDownEvt -> {
|
||||||
pickEntity()?.let { pick ->
|
pickEntity()?.let { pick ->
|
||||||
when (event.buttons) {
|
when (event.buttons) {
|
||||||
1 -> {
|
1 -> {
|
||||||
questEditorStore.setSelectedEntity(pick.entity)
|
ctx.setSelectedEntity(pick.entity)
|
||||||
|
|
||||||
if (entityManipulationEnabled) {
|
if (entityManipulationEnabled) {
|
||||||
// TODO: Enter TranslationState.
|
return TranslationState(
|
||||||
|
ctx,
|
||||||
|
pick.entity,
|
||||||
|
pick.dragAdjust,
|
||||||
|
pick.grabOffset
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
2 -> {
|
2 -> {
|
||||||
questEditorStore.setSelectedEntity(pick.entity)
|
ctx.setSelectedEntity(pick.entity)
|
||||||
|
|
||||||
if (entityManipulationEnabled) {
|
if (entityManipulationEnabled) {
|
||||||
// TODO: Enter RotationState.
|
// TODO: Enter RotationState.
|
||||||
@ -164,8 +323,6 @@ private class IdleState(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
this
|
|
||||||
}
|
}
|
||||||
|
|
||||||
is PointerUpEvt -> {
|
is PointerUpEvt -> {
|
||||||
@ -173,13 +330,18 @@ private class IdleState(
|
|||||||
|
|
||||||
// If the user clicks on nothing, deselect the currently selected entity.
|
// If the user clicks on nothing, deselect the currently selected entity.
|
||||||
if (!event.movedSinceLastPointerDown && pickEntity() == null) {
|
if (!event.movedSinceLastPointerDown && pickEntity() == null) {
|
||||||
questEditorStore.setSelectedEntity(null)
|
ctx.setSelectedEntity(null)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
this
|
else -> {
|
||||||
|
// Do nothing.
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return this
|
||||||
|
}
|
||||||
|
|
||||||
override fun cancel() {
|
override fun cancel() {
|
||||||
// Do nothing.
|
// Do nothing.
|
||||||
}
|
}
|
||||||
@ -187,14 +349,17 @@ private class IdleState(
|
|||||||
private fun updateCameraTarget() {
|
private fun updateCameraTarget() {
|
||||||
// If the user moved the camera, try setting the camera
|
// If the user moved the camera, try setting the camera
|
||||||
// target to a better point.
|
// target to a better point.
|
||||||
pickGround()?.pickedPoint?.let { newTarget ->
|
ctx.pickGround(
|
||||||
renderer.camera.target = newTarget
|
ctx.renderer.engine.getRenderWidth() / 2,
|
||||||
|
ctx.renderer.engine.getRenderHeight() / 2,
|
||||||
|
)?.pickedPoint?.let { newTarget ->
|
||||||
|
ctx.renderer.camera.target = newTarget
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun pickEntity(): Pick? {
|
private fun pickEntity(): Pick? {
|
||||||
// Find the nearest object and NPC under the pointer.
|
// Find the nearest object and NPC under the pointer.
|
||||||
val pickInfo = renderer.scene.pick(renderer.scene.pointerX, renderer.scene.pointerY)
|
val pickInfo = ctx.scene.pick(ctx.scene.pointerX, ctx.scene.pointerY)
|
||||||
if (pickInfo?.pickedMesh == null) return null
|
if (pickInfo?.pickedMesh == null) return null
|
||||||
|
|
||||||
val entity = (pickInfo.pickedMesh.metadata as? EntityMetadata)?.entity
|
val entity = (pickInfo.pickedMesh.metadata as? EntityMetadata)?.entity
|
||||||
@ -208,9 +373,9 @@ private class IdleState(
|
|||||||
val dragAdjust = grabOffset.clone()
|
val dragAdjust = grabOffset.clone()
|
||||||
|
|
||||||
// Find vertical distance to the ground.
|
// Find vertical distance to the ground.
|
||||||
renderer.scene.pickWithRay(
|
ctx.scene.pickWithRay(
|
||||||
Ray(pickInfo.pickedMesh.position, DOWN_VECTOR),
|
Ray(pickInfo.pickedMesh.position, DOWN_VECTOR),
|
||||||
{ it.metadata is CollisionMetadata },
|
{ it.isEnabled() && it.metadata is CollisionMetadata },
|
||||||
)?.let { groundPick ->
|
)?.let { groundPick ->
|
||||||
dragAdjust.y -= groundPick.distance
|
dragAdjust.y -= groundPick.distance
|
||||||
}
|
}
|
||||||
@ -222,26 +387,81 @@ private class IdleState(
|
|||||||
dragAdjust,
|
dragAdjust,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private fun pickGround(): PickingInfo? {
|
private class TranslationState(
|
||||||
renderer.scene.multiPick(
|
private val ctx: StateContext,
|
||||||
renderer.engine.getRenderWidth() / 2,
|
private val entity: QuestEntityModel<*, *>,
|
||||||
renderer.engine.getRenderHeight() / 2,
|
private val dragAdjust: Vector3,
|
||||||
{ it.metadata is CollisionMetadata },
|
private val grabOffset: Vector3,
|
||||||
renderer.camera,
|
) : State() {
|
||||||
)?.let { pickingInfoArray ->
|
private val initialSection: SectionModel? = entity.section.value
|
||||||
// Don't allow entities to be placed on very steep terrain.
|
private val initialPosition: Vector3 = entity.worldPosition.value
|
||||||
// E.g. walls.
|
private var cancelled = false
|
||||||
// TODO: make use of the flags field in the collision data.
|
|
||||||
for (pickingInfo in pickingInfoArray) {
|
init {
|
||||||
pickingInfo.getNormal()?.let { n ->
|
ctx.renderer.disableCameraControls()
|
||||||
if (n.y > 0.75) {
|
}
|
||||||
return pickingInfo
|
|
||||||
|
override fun processEvent(event: Evt): State =
|
||||||
|
when (event) {
|
||||||
|
is PointerMoveEvt -> {
|
||||||
|
if (cancelled) {
|
||||||
|
IdleState(ctx, entityManipulationEnabled = true)
|
||||||
|
} else {
|
||||||
|
if (event.movedSinceLastPointerDown) {
|
||||||
|
ctx.translate(
|
||||||
|
entity,
|
||||||
|
dragAdjust,
|
||||||
|
grabOffset,
|
||||||
|
vertically = event.shiftKeyDown,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
is PointerUpEvt -> {
|
||||||
|
ctx.renderer.enableCameraControls()
|
||||||
|
|
||||||
|
if (!cancelled && event.movedSinceLastPointerDown) {
|
||||||
|
// TODO
|
||||||
|
// questEditorStore.undo
|
||||||
|
// .push(
|
||||||
|
// new TranslateEntityAction (
|
||||||
|
// this.questEditorStore,
|
||||||
|
// this.entity,
|
||||||
|
// this.initialSection,
|
||||||
|
// this.entity.section.
|
||||||
|
// val ,
|
||||||
|
// this.initial_position,
|
||||||
|
// this.entity.world_position.
|
||||||
|
// val ,
|
||||||
|
// true,
|
||||||
|
// ),
|
||||||
|
// )
|
||||||
|
// .redo()
|
||||||
|
}
|
||||||
|
|
||||||
|
IdleState(ctx, entityManipulationEnabled = true)
|
||||||
|
}
|
||||||
|
|
||||||
|
else -> {
|
||||||
|
if (cancelled) {
|
||||||
|
IdleState(ctx, entityManipulationEnabled = true)
|
||||||
|
} else this
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return null
|
override fun cancel() {
|
||||||
|
cancelled = true
|
||||||
|
ctx.renderer.enableCameraControls()
|
||||||
|
|
||||||
|
initialSection?.let {
|
||||||
|
entity.setSection(initialSection)
|
||||||
|
}
|
||||||
|
|
||||||
|
entity.setWorldPosition(initialPosition)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user