mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
Improved Ninja geometry to Three.js conversion speed by avoiding some GC.
This commit is contained in:
parent
f57de99577
commit
b8edf35b30
@ -2,9 +2,7 @@ package world.phantasmal.web.core.rendering.conversion
|
|||||||
|
|
||||||
import mu.KotlinLogging
|
import mu.KotlinLogging
|
||||||
import world.phantasmal.lib.fileFormats.ninja.*
|
import world.phantasmal.lib.fileFormats.ninja.*
|
||||||
import world.phantasmal.web.core.cross
|
|
||||||
import world.phantasmal.web.core.dot
|
import world.phantasmal.web.core.dot
|
||||||
import world.phantasmal.web.core.minus
|
|
||||||
import world.phantasmal.web.core.toQuaternion
|
import world.phantasmal.web.core.toQuaternion
|
||||||
import world.phantasmal.web.externals.three.*
|
import world.phantasmal.web.externals.three.*
|
||||||
|
|
||||||
@ -16,6 +14,11 @@ private val NO_TRANSLATION = Vector3(0.0, 0.0, 0.0)
|
|||||||
private val NO_ROTATION = Quaternion()
|
private val NO_ROTATION = Quaternion()
|
||||||
private val NO_SCALE = Vector3(1.0, 1.0, 1.0)
|
private val NO_SCALE = Vector3(1.0, 1.0, 1.0)
|
||||||
|
|
||||||
|
// Objects used for temporary calculations to avoid GC.
|
||||||
|
private val tmpNormal = Vector3()
|
||||||
|
private val tmpVec = Vector3()
|
||||||
|
private val tmpNormalMatrix = Matrix3()
|
||||||
|
|
||||||
fun ninjaObjectToMesh(
|
fun ninjaObjectToMesh(
|
||||||
ninjaObject: NinjaObject<*>,
|
ninjaObject: NinjaObject<*>,
|
||||||
textures: List<XvrTexture?>,
|
textures: List<XvrTexture?>,
|
||||||
@ -128,7 +131,7 @@ private class NinjaToMeshConverter(private val builder: MeshBuilder) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun convertNjModel(model: NjModel, matrix: Matrix4) {
|
private fun convertNjModel(model: NjModel, matrix: Matrix4) {
|
||||||
val normalMatrix = Matrix3().getNormalMatrix(matrix)
|
tmpNormalMatrix.getNormalMatrix(matrix)
|
||||||
|
|
||||||
val newVertices = model.vertices.map { vertex ->
|
val newVertices = model.vertices.map { vertex ->
|
||||||
vertex?.let {
|
vertex?.let {
|
||||||
@ -136,7 +139,7 @@ private class NinjaToMeshConverter(private val builder: MeshBuilder) {
|
|||||||
val normal = vertex.normal?.let(::vec3ToThree) ?: Vector3(0.0, 1.0, 0.0)
|
val normal = vertex.normal?.let(::vec3ToThree) ?: Vector3(0.0, 1.0, 0.0)
|
||||||
|
|
||||||
position.applyMatrix4(matrix)
|
position.applyMatrix4(matrix)
|
||||||
normal.applyMatrix3(normalMatrix)
|
normal.applyMatrix3(tmpNormalMatrix)
|
||||||
|
|
||||||
Vertex(
|
Vertex(
|
||||||
boneIndex,
|
boneIndex,
|
||||||
@ -218,14 +221,14 @@ private class NinjaToMeshConverter(private val builder: MeshBuilder) {
|
|||||||
|
|
||||||
private fun convertXjModel(model: XjModel, matrix: Matrix4) {
|
private fun convertXjModel(model: XjModel, matrix: Matrix4) {
|
||||||
val indexOffset = builder.vertexCount
|
val indexOffset = builder.vertexCount
|
||||||
val normalMatrix = Matrix3().getNormalMatrix(matrix)
|
tmpNormalMatrix.getNormalMatrix(matrix)
|
||||||
|
|
||||||
for (vertex in model.vertices) {
|
for (vertex in model.vertices) {
|
||||||
val p = vec3ToThree(vertex.position)
|
val p = vec3ToThree(vertex.position)
|
||||||
p.applyMatrix4(matrix)
|
p.applyMatrix4(matrix)
|
||||||
|
|
||||||
val n = vertex.normal?.let(::vec3ToThree) ?: Vector3(0.0, 1.0, 0.0)
|
val n = vertex.normal?.let(::vec3ToThree) ?: Vector3(0.0, 1.0, 0.0)
|
||||||
n.applyMatrix3(normalMatrix)
|
n.applyMatrix3(tmpNormalMatrix)
|
||||||
|
|
||||||
val uv = vertex.uv?.let(::vec2ToThree) ?: DEFAULT_UV
|
val uv = vertex.uv?.let(::vec2ToThree) ?: DEFAULT_UV
|
||||||
|
|
||||||
@ -263,16 +266,20 @@ private class NinjaToMeshConverter(private val builder: MeshBuilder) {
|
|||||||
// Calculate a surface normal and reverse the vertex winding if at least 2 of the
|
// Calculate a surface normal and reverse the vertex winding if at least 2 of the
|
||||||
// vertex normals point in the opposite direction. This hack fixes the winding for
|
// vertex normals point in the opposite direction. This hack fixes the winding for
|
||||||
// most models.
|
// most models.
|
||||||
val normal = (pb - pa) cross (pc - pa)
|
tmpNormal.copy(pb)
|
||||||
|
tmpNormal.sub(pa)
|
||||||
|
tmpVec.copy(pc)
|
||||||
|
tmpVec.sub(pa)
|
||||||
|
tmpNormal.cross(tmpVec)
|
||||||
|
|
||||||
if (clockwise) {
|
if (clockwise) {
|
||||||
normal.negate()
|
tmpNormal.negate()
|
||||||
}
|
}
|
||||||
|
|
||||||
val oppositeCount =
|
val oppositeCount =
|
||||||
(if (normal dot na < 0) 1 else 0) +
|
(if (tmpNormal dot na < 0) 1 else 0) +
|
||||||
(if (normal dot nb < 0) 1 else 0) +
|
(if (tmpNormal dot nb < 0) 1 else 0) +
|
||||||
(if (normal dot nc < 0) 1 else 0)
|
(if (tmpNormal dot nc < 0) 1 else 0)
|
||||||
|
|
||||||
if (oppositeCount >= 2) {
|
if (oppositeCount >= 2) {
|
||||||
clockwise = !clockwise
|
clockwise = !clockwise
|
||||||
|
Loading…
Reference in New Issue
Block a user