mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 15:28:29 +08:00
Fixed bug in NJ bone weight calculation. Improved NJ parser by avoiding reparsing of chunks.
This commit is contained in:
parent
47598a2670
commit
0ff4752949
@ -55,6 +55,8 @@ private fun <Model : NinjaModel, Context> parseSiblingObjects(
|
|||||||
val zxyRotationOrder = evalFlags.isBitSet(5)
|
val zxyRotationOrder = evalFlags.isBitSet(5)
|
||||||
val skip = evalFlags.isBitSet(6)
|
val skip = evalFlags.isBitSet(6)
|
||||||
val shapeSkip = evalFlags.isBitSet(7)
|
val shapeSkip = evalFlags.isBitSet(7)
|
||||||
|
val clip = evalFlags.isBitSet(8)
|
||||||
|
val modifier = evalFlags.isBitSet(9)
|
||||||
|
|
||||||
val modelOffset = cursor.int()
|
val modelOffset = cursor.int()
|
||||||
val pos = cursor.vec3Float()
|
val pos = cursor.vec3Float()
|
||||||
@ -98,6 +100,8 @@ private fun <Model : NinjaModel, Context> parseSiblingObjects(
|
|||||||
zxyRotationOrder,
|
zxyRotationOrder,
|
||||||
skip,
|
skip,
|
||||||
shapeSkip,
|
shapeSkip,
|
||||||
|
clip,
|
||||||
|
modifier,
|
||||||
),
|
),
|
||||||
model,
|
model,
|
||||||
pos,
|
pos,
|
||||||
|
@ -63,6 +63,8 @@ class NinjaEvaluationFlags(
|
|||||||
var zxyRotationOrder: Boolean,
|
var zxyRotationOrder: Boolean,
|
||||||
var skip: Boolean,
|
var skip: Boolean,
|
||||||
var shapeSkip: Boolean,
|
var shapeSkip: Boolean,
|
||||||
|
val clip: Boolean,
|
||||||
|
val modifier: Boolean,
|
||||||
)
|
)
|
||||||
|
|
||||||
sealed class NinjaModel
|
sealed class NinjaModel
|
||||||
@ -83,7 +85,7 @@ class NjModel(
|
|||||||
class NjVertex(
|
class NjVertex(
|
||||||
val position: Vec3,
|
val position: Vec3,
|
||||||
val normal: Vec3?,
|
val normal: Vec3?,
|
||||||
val boneWeight: Float,
|
val boneWeight: Float?,
|
||||||
val boneWeightStatus: Int,
|
val boneWeightStatus: Int,
|
||||||
val calcContinue: Boolean,
|
val calcContinue: Boolean,
|
||||||
)
|
)
|
||||||
@ -111,19 +113,23 @@ class NjMeshVertex(
|
|||||||
val texCoords: Vec2?,
|
val texCoords: Vec2?,
|
||||||
)
|
)
|
||||||
|
|
||||||
sealed class NjChunk(val typeId: UByte) {
|
sealed class NjChunk(val typeId: Int) {
|
||||||
class Unknown(typeId: UByte) : NjChunk(typeId)
|
class Unknown(typeId: Int) : NjChunk(typeId)
|
||||||
|
|
||||||
object Null : NjChunk(0u)
|
object Null : NjChunk(0)
|
||||||
|
|
||||||
class Bits(typeId: UByte, val srcAlpha: Int, val dstAlpha: Int) : NjChunk(typeId)
|
class BlendAlpha(val srcAlpha: Int, val dstAlpha: Int) : NjChunk(1)
|
||||||
|
|
||||||
class CachePolygonList(val cacheIndex: Int, val offset: Int) : NjChunk(4u)
|
class MipmapDAdjust(val adjust: Int) : NjChunk(2)
|
||||||
|
|
||||||
class DrawPolygonList(val cacheIndex: Int) : NjChunk(5u)
|
class SpecularExponent(val specular: Int) : NjChunk(3)
|
||||||
|
|
||||||
|
class CachePolygonList(val cacheIndex: Int) : NjChunk(4)
|
||||||
|
|
||||||
|
class DrawPolygonList(val cacheIndex: Int) : NjChunk(5)
|
||||||
|
|
||||||
class Tiny(
|
class Tiny(
|
||||||
typeId: UByte,
|
typeId: Int,
|
||||||
val flipU: Boolean,
|
val flipU: Boolean,
|
||||||
val flipV: Boolean,
|
val flipV: Boolean,
|
||||||
val clampU: Boolean,
|
val clampU: Boolean,
|
||||||
@ -135,7 +141,7 @@ sealed class NjChunk(val typeId: UByte) {
|
|||||||
) : NjChunk(typeId)
|
) : NjChunk(typeId)
|
||||||
|
|
||||||
class Material(
|
class Material(
|
||||||
typeId: UByte,
|
typeId: Int,
|
||||||
val srcAlpha: Int,
|
val srcAlpha: Int,
|
||||||
val dstAlpha: Int,
|
val dstAlpha: Int,
|
||||||
val diffuse: NjArgb?,
|
val diffuse: NjArgb?,
|
||||||
@ -143,20 +149,20 @@ sealed class NjChunk(val typeId: UByte) {
|
|||||||
val specular: NjErgb?,
|
val specular: NjErgb?,
|
||||||
) : NjChunk(typeId)
|
) : NjChunk(typeId)
|
||||||
|
|
||||||
class Vertex(typeId: UByte, val vertices: List<NjChunkVertex>) : NjChunk(typeId)
|
class Vertex(typeId: Int, val vertices: List<NjChunkVertex>) : NjChunk(typeId)
|
||||||
|
|
||||||
class Volume(typeId: UByte) : NjChunk(typeId)
|
class Volume(typeId: Int) : NjChunk(typeId)
|
||||||
|
|
||||||
class Strip(typeId: UByte, val triangleStrips: List<NjTriangleStrip>) : NjChunk(typeId)
|
class Strip(typeId: Int, val triangleStrips: List<NjTriangleStrip>) : NjChunk(typeId)
|
||||||
|
|
||||||
object End : NjChunk(255u)
|
object End : NjChunk(255)
|
||||||
}
|
}
|
||||||
|
|
||||||
class NjChunkVertex(
|
class NjChunkVertex(
|
||||||
val index: Int,
|
val index: Int,
|
||||||
val position: Vec3,
|
val position: Vec3,
|
||||||
val normal: Vec3?,
|
val normal: Vec3?,
|
||||||
val boneWeight: Float,
|
val boneWeight: Float?,
|
||||||
val boneWeightStatus: Int,
|
val boneWeightStatus: Int,
|
||||||
val calcContinue: Boolean,
|
val calcContinue: Boolean,
|
||||||
)
|
)
|
||||||
|
@ -16,7 +16,7 @@ private val logger = KotlinLogging.logger {}
|
|||||||
|
|
||||||
// TODO: Simplify parser by not parsing chunks into vertices and meshes. Do the chunk to vertex/mesh
|
// TODO: Simplify parser by not parsing chunks into vertices and meshes. Do the chunk to vertex/mesh
|
||||||
// conversion at a higher level.
|
// conversion at a higher level.
|
||||||
fun parseNjModel(cursor: Cursor, cachedChunkOffsets: MutableMap<Int, Int>): NjModel {
|
fun parseNjModel(cursor: Cursor, cachedChunks: MutableMap<Int, List<NjChunk>>): NjModel {
|
||||||
val vlistOffset = cursor.int() // Vertex list
|
val vlistOffset = cursor.int() // Vertex list
|
||||||
val plistOffset = cursor.int() // Triangle strip index list
|
val plistOffset = cursor.int() // Triangle strip index list
|
||||||
val collisionSphereCenter = cursor.vec3Float()
|
val collisionSphereCenter = cursor.vec3Float()
|
||||||
@ -27,7 +27,7 @@ fun parseNjModel(cursor: Cursor, cachedChunkOffsets: MutableMap<Int, Int>): NjMo
|
|||||||
if (vlistOffset != 0) {
|
if (vlistOffset != 0) {
|
||||||
cursor.seekStart(vlistOffset)
|
cursor.seekStart(vlistOffset)
|
||||||
|
|
||||||
for (chunk in parseChunks(cursor, cachedChunkOffsets, true)) {
|
for (chunk in parseChunks(cursor)) {
|
||||||
if (chunk is NjChunk.Vertex) {
|
if (chunk is NjChunk.Vertex) {
|
||||||
for (vertex in chunk.vertices) {
|
for (vertex in chunk.vertices) {
|
||||||
while (vertices.size <= vertex.index) {
|
while (vertices.size <= vertex.index) {
|
||||||
@ -46,20 +46,59 @@ fun parseNjModel(cursor: Cursor, cachedChunkOffsets: MutableMap<Int, Int>): NjMo
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (plistOffset != 0) {
|
if (plistOffset > 0) {
|
||||||
cursor.seekStart(plistOffset)
|
cursor.seekStart(plistOffset)
|
||||||
|
|
||||||
var textureId: Int? = null
|
PolygonChunkProcessor(cachedChunks, meshes).process(parseChunks(cursor))
|
||||||
var srcAlpha: Int? = null
|
}
|
||||||
var dstAlpha: Int? = null
|
|
||||||
|
|
||||||
for (chunk in parseChunks(cursor, cachedChunkOffsets, false)) {
|
return NjModel(
|
||||||
|
vertices,
|
||||||
|
meshes,
|
||||||
|
collisionSphereCenter,
|
||||||
|
collisionSphereRadius,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
private class PolygonChunkProcessor(
|
||||||
|
private val cachedChunks: MutableMap<Int, List<NjChunk>>,
|
||||||
|
private val meshes: MutableList<NjTriangleStrip>,
|
||||||
|
) {
|
||||||
|
private var textureId: Int? = null
|
||||||
|
private var srcAlpha: Int? = null
|
||||||
|
private var dstAlpha: Int? = null
|
||||||
|
|
||||||
|
/**
|
||||||
|
* When [cacheList] is non-null we are caching chunks.
|
||||||
|
*/
|
||||||
|
private var cacheList: MutableList<NjChunk>? = null
|
||||||
|
|
||||||
|
fun process(chunks: List<NjChunk>) {
|
||||||
|
for (chunk in chunks) {
|
||||||
|
if (cacheList == null) {
|
||||||
when (chunk) {
|
when (chunk) {
|
||||||
is NjChunk.Bits -> {
|
is NjChunk.BlendAlpha -> {
|
||||||
srcAlpha = chunk.srcAlpha
|
srcAlpha = chunk.srcAlpha
|
||||||
dstAlpha = chunk.dstAlpha
|
dstAlpha = chunk.dstAlpha
|
||||||
}
|
}
|
||||||
|
|
||||||
|
is NjChunk.CachePolygonList -> {
|
||||||
|
cacheList = mutableListOf()
|
||||||
|
cachedChunks[chunk.cacheIndex] = cacheList!!
|
||||||
|
}
|
||||||
|
|
||||||
|
is NjChunk.DrawPolygonList -> {
|
||||||
|
val cached = cachedChunks[chunk.cacheIndex]
|
||||||
|
|
||||||
|
if (cached == null) {
|
||||||
|
logger.debug {
|
||||||
|
"Draw Polygon List chunk pointed to nonexistent cache index ${chunk.cacheIndex}."
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
process(cached)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
is NjChunk.Tiny -> {
|
is NjChunk.Tiny -> {
|
||||||
textureId = chunk.textureId
|
textureId = chunk.textureId
|
||||||
}
|
}
|
||||||
@ -83,90 +122,83 @@ fun parseNjModel(cursor: Cursor, cachedChunkOffsets: MutableMap<Int, Int>): NjMo
|
|||||||
// Ignore
|
// Ignore
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
cacheList!!.add(chunk)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return NjModel(
|
private fun parseChunks(cursor: Cursor): List<NjChunk> {
|
||||||
vertices,
|
|
||||||
meshes,
|
|
||||||
collisionSphereCenter,
|
|
||||||
collisionSphereRadius,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: don't reparse when DrawPolygonList chunk is encountered.
|
|
||||||
private fun parseChunks(
|
|
||||||
cursor: Cursor,
|
|
||||||
cachedChunkOffsets: MutableMap<Int, Int>,
|
|
||||||
wideEndChunks: Boolean,
|
|
||||||
): List<NjChunk> {
|
|
||||||
val chunks: MutableList<NjChunk> = mutableListOf()
|
val chunks: MutableList<NjChunk> = mutableListOf()
|
||||||
var loop = true
|
|
||||||
|
|
||||||
while (loop) {
|
do {
|
||||||
val typeId = cursor.uByte()
|
|
||||||
val flags = cursor.uByte().toInt()
|
|
||||||
val chunkStartPosition = cursor.position
|
val chunkStartPosition = cursor.position
|
||||||
|
val typeId = cursor.uByte().toInt()
|
||||||
|
val flags = cursor.uByte().toInt()
|
||||||
|
val chunkDataPosition = cursor.position
|
||||||
var size = 0
|
var size = 0
|
||||||
|
val chunk: NjChunk
|
||||||
|
|
||||||
when (typeId.toInt()) {
|
when (typeId) {
|
||||||
0 -> {
|
0 -> {
|
||||||
chunks.add(NjChunk.Null)
|
chunk = NjChunk.Null
|
||||||
}
|
}
|
||||||
in 1..3 -> {
|
1 -> {
|
||||||
chunks.add(NjChunk.Bits(
|
chunk = NjChunk.BlendAlpha(
|
||||||
typeId,
|
|
||||||
srcAlpha = (flags ushr 3) and 0b111,
|
srcAlpha = (flags ushr 3) and 0b111,
|
||||||
dstAlpha = flags and 0b111,
|
dstAlpha = flags and 0b111,
|
||||||
))
|
)
|
||||||
|
}
|
||||||
|
2 -> {
|
||||||
|
chunk = NjChunk.MipmapDAdjust(
|
||||||
|
adjust = flags and 0b1111,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
3 -> {
|
||||||
|
chunk = NjChunk.SpecularExponent(
|
||||||
|
specular = flags and 0b11111,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
4 -> {
|
4 -> {
|
||||||
val offset = cursor.position
|
chunk = NjChunk.CachePolygonList(
|
||||||
|
|
||||||
chunks.add(NjChunk.CachePolygonList(
|
|
||||||
cacheIndex = flags,
|
cacheIndex = flags,
|
||||||
offset,
|
)
|
||||||
))
|
|
||||||
|
|
||||||
cachedChunkOffsets[flags] = offset
|
|
||||||
loop = false
|
|
||||||
}
|
}
|
||||||
5 -> {
|
5 -> {
|
||||||
val cachedOffset = cachedChunkOffsets[flags]
|
chunk = NjChunk.DrawPolygonList(
|
||||||
|
|
||||||
if (cachedOffset != null) {
|
|
||||||
cursor.seekStart(cachedOffset)
|
|
||||||
chunks.addAll(parseChunks(cursor, cachedChunkOffsets, wideEndChunks))
|
|
||||||
}
|
|
||||||
|
|
||||||
chunks.add(NjChunk.DrawPolygonList(
|
|
||||||
cacheIndex = flags,
|
cacheIndex = flags,
|
||||||
))
|
)
|
||||||
}
|
}
|
||||||
in 8..9 -> {
|
in 8..9 -> {
|
||||||
size = 2
|
size = 2
|
||||||
val textureBitsAndId = cursor.uShort().toInt()
|
val textureBitsAndId = cursor.uShort().toInt()
|
||||||
|
|
||||||
chunks.add(NjChunk.Tiny(
|
chunk = NjChunk.Tiny(
|
||||||
typeId,
|
typeId,
|
||||||
flipU = typeId.isBitSet(7),
|
flipU = flags.isBitSet(7),
|
||||||
flipV = typeId.isBitSet(6),
|
flipV = flags.isBitSet(6),
|
||||||
clampU = typeId.isBitSet(5),
|
clampU = flags.isBitSet(5),
|
||||||
clampV = typeId.isBitSet(4),
|
clampV = flags.isBitSet(4),
|
||||||
mipmapDAdjust = typeId.toUInt() and 0b1111u,
|
mipmapDAdjust = flags.toUInt() and 0b1111u,
|
||||||
filterMode = textureBitsAndId ushr 14,
|
filterMode = textureBitsAndId ushr 14,
|
||||||
superSample = (textureBitsAndId and 0x40) != 0,
|
superSample = (textureBitsAndId and 0x40) != 0,
|
||||||
textureId = textureBitsAndId and 0x1fff,
|
textureId = textureBitsAndId and 0x1FFF,
|
||||||
))
|
)
|
||||||
}
|
}
|
||||||
in 17..31 -> {
|
in 17..31 -> {
|
||||||
size = 2 + 2 * cursor.short()
|
val bodySize = 2 * cursor.short()
|
||||||
|
size = 2 + bodySize
|
||||||
|
|
||||||
var diffuse: NjArgb? = null
|
var diffuse: NjArgb? = null
|
||||||
var ambient: NjArgb? = null
|
var ambient: NjArgb? = null
|
||||||
var specular: NjErgb? = null
|
var specular: NjErgb? = null
|
||||||
|
|
||||||
if (flags.isBitSet(0)) {
|
if (typeId == 24) {
|
||||||
|
// Skip bump map data.
|
||||||
|
cursor.seek(bodySize)
|
||||||
|
} else {
|
||||||
|
if (typeId.isBitSet(0)) {
|
||||||
diffuse = NjArgb(
|
diffuse = NjArgb(
|
||||||
b = cursor.uByte().toFloat() / 255f,
|
b = cursor.uByte().toFloat() / 255f,
|
||||||
g = cursor.uByte().toFloat() / 255f,
|
g = cursor.uByte().toFloat() / 255f,
|
||||||
@ -175,7 +207,7 @@ private fun parseChunks(
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (flags.isBitSet(1)) {
|
if (typeId.isBitSet(1)) {
|
||||||
ambient = NjArgb(
|
ambient = NjArgb(
|
||||||
b = cursor.uByte().toFloat() / 255f,
|
b = cursor.uByte().toFloat() / 255f,
|
||||||
g = cursor.uByte().toFloat() / 255f,
|
g = cursor.uByte().toFloat() / 255f,
|
||||||
@ -184,7 +216,7 @@ private fun parseChunks(
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (flags.isBitSet(2)) {
|
if (typeId.isBitSet(2)) {
|
||||||
specular = NjErgb(
|
specular = NjErgb(
|
||||||
b = cursor.uByte(),
|
b = cursor.uByte(),
|
||||||
g = cursor.uByte(),
|
g = cursor.uByte(),
|
||||||
@ -192,59 +224,72 @@ private fun parseChunks(
|
|||||||
e = cursor.uByte(),
|
e = cursor.uByte(),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
chunks.add(NjChunk.Material(
|
chunk = NjChunk.Material(
|
||||||
typeId,
|
typeId,
|
||||||
srcAlpha = (flags ushr 3) and 0b111,
|
srcAlpha = (flags ushr 3) and 0b111,
|
||||||
dstAlpha = flags and 0b111,
|
dstAlpha = flags and 0b111,
|
||||||
diffuse,
|
diffuse,
|
||||||
ambient,
|
ambient,
|
||||||
specular,
|
specular,
|
||||||
))
|
)
|
||||||
}
|
}
|
||||||
in 32..50 -> {
|
in 32..50 -> {
|
||||||
size = 2 + 4 * cursor.short()
|
size = 2 + 4 * cursor.short()
|
||||||
chunks.add(NjChunk.Vertex(
|
chunk = NjChunk.Vertex(
|
||||||
typeId,
|
typeId,
|
||||||
vertices = parseVertexChunk(cursor, typeId, flags),
|
vertices = parseVertexChunk(cursor, typeId, flags),
|
||||||
))
|
)
|
||||||
}
|
}
|
||||||
in 56..58 -> {
|
in 56..58 -> {
|
||||||
size = 2 + 2 * cursor.short()
|
size = 2 + 2 * cursor.short()
|
||||||
chunks.add(NjChunk.Volume(
|
chunk = NjChunk.Volume(
|
||||||
typeId,
|
typeId,
|
||||||
))
|
)
|
||||||
|
|
||||||
|
// Skip volume information.
|
||||||
|
cursor.seek(2 * cursor.short())
|
||||||
}
|
}
|
||||||
in 64..75 -> {
|
in 64..75 -> {
|
||||||
size = 2 + 2 * cursor.short()
|
size = 2 + 2 * cursor.short()
|
||||||
chunks.add(NjChunk.Strip(
|
chunk = NjChunk.Strip(
|
||||||
typeId,
|
typeId,
|
||||||
triangleStrips = parseTriangleStripChunk(cursor, typeId, flags),
|
triangleStrips = parseTriangleStripChunk(cursor, typeId, flags),
|
||||||
))
|
)
|
||||||
}
|
}
|
||||||
255 -> {
|
255 -> {
|
||||||
size = if (wideEndChunks) 2 else 0
|
chunk = NjChunk.End
|
||||||
chunks.add(NjChunk.End)
|
|
||||||
loop = false
|
|
||||||
}
|
}
|
||||||
else -> {
|
else -> {
|
||||||
size = 2 + 2 * cursor.short()
|
val bodySize = 2 * cursor.short()
|
||||||
chunks.add(NjChunk.Unknown(
|
size = 2 + bodySize
|
||||||
|
chunk = NjChunk.Unknown(
|
||||||
typeId,
|
typeId,
|
||||||
))
|
)
|
||||||
|
// Skip unknown data.
|
||||||
|
cursor.seek(bodySize)
|
||||||
logger.warn { "Unknown chunk type $typeId at offset ${chunkStartPosition}." }
|
logger.warn { "Unknown chunk type $typeId at offset ${chunkStartPosition}." }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
cursor.seekStart(chunkStartPosition + size)
|
chunks.add(chunk)
|
||||||
|
|
||||||
|
val bytesRead = cursor.position - chunkDataPosition
|
||||||
|
|
||||||
|
check(bytesRead <= size) {
|
||||||
|
"Expected to read at most $size bytes, actually read $bytesRead."
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cursor.seekStart(chunkDataPosition + size)
|
||||||
|
} while (chunk != NjChunk.End)
|
||||||
|
|
||||||
return chunks
|
return chunks
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun parseVertexChunk(
|
private fun parseVertexChunk(
|
||||||
cursor: Cursor,
|
cursor: Cursor,
|
||||||
chunkTypeId: UByte,
|
chunkTypeId: Int,
|
||||||
flags: Int,
|
flags: Int,
|
||||||
): List<NjChunkVertex> {
|
): List<NjChunkVertex> {
|
||||||
val boneWeightStatus = flags and 0b11
|
val boneWeightStatus = flags and 0b11
|
||||||
@ -259,60 +304,79 @@ private fun parseVertexChunk(
|
|||||||
var vertexIndex = index + i
|
var vertexIndex = index + i
|
||||||
val position = cursor.vec3Float()
|
val position = cursor.vec3Float()
|
||||||
var normal: Vec3? = null
|
var normal: Vec3? = null
|
||||||
var boneWeight = 1f
|
var boneWeight: Float? = null
|
||||||
|
|
||||||
when (chunkTypeId.toInt()) {
|
when (chunkTypeId) {
|
||||||
32 -> {
|
32 -> {
|
||||||
// NJDCVSH
|
// NJD_CV_SH
|
||||||
cursor.seek(4) // Always 1.0
|
cursor.seek(4) // Always 1.0
|
||||||
}
|
}
|
||||||
33 -> {
|
33 -> {
|
||||||
// NJDCVVNSH
|
// NJD_CV_VN_SH
|
||||||
cursor.seek(4) // Always 1.0
|
cursor.seek(4) // Always 1.0
|
||||||
normal = cursor.vec3Float()
|
normal = cursor.vec3Float()
|
||||||
cursor.seek(4) // Always 0.0
|
cursor.seek(4) // Always 0.0
|
||||||
}
|
}
|
||||||
|
34 -> {
|
||||||
|
// NJD_CV
|
||||||
|
// Nothing to do.
|
||||||
|
}
|
||||||
in 35..40 -> {
|
in 35..40 -> {
|
||||||
if (chunkTypeId == (37u).toUByte()) {
|
if (chunkTypeId == 37) {
|
||||||
// NJDCVNF
|
// NJD_CV_NF
|
||||||
// NinjaFlags32
|
// NinjaFlags32
|
||||||
vertexIndex = index + cursor.uShort()
|
vertexIndex = index + cursor.uShort()
|
||||||
boneWeight = cursor.uShort().toFloat() / 255f
|
boneWeight = cursor.uShort().toFloat() / 255f
|
||||||
} else {
|
} else {
|
||||||
|
// NJD_CV_D8
|
||||||
|
// NJD_CV_UF
|
||||||
|
// NJD_CV_S5
|
||||||
|
// NJD_CV_S4
|
||||||
|
// NJD_CV_IN
|
||||||
// Skip user flags and material information.
|
// Skip user flags and material information.
|
||||||
cursor.seek(4)
|
cursor.seek(4)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
41 -> {
|
41 -> {
|
||||||
|
// NJD_CV_VN
|
||||||
normal = cursor.vec3Float()
|
normal = cursor.vec3Float()
|
||||||
}
|
}
|
||||||
in 42..47 -> {
|
in 42..47 -> {
|
||||||
normal = cursor.vec3Float()
|
normal = cursor.vec3Float()
|
||||||
|
|
||||||
if (chunkTypeId == (44u).toUByte()) {
|
if (chunkTypeId == 44) {
|
||||||
// NJDCVVNNF
|
// NJD_CV_VN_NF
|
||||||
// NinjaFlags32
|
// NinjaFlags32
|
||||||
vertexIndex = index + cursor.uShort()
|
vertexIndex = index + cursor.uShort()
|
||||||
boneWeight = cursor.uShort().toFloat() / 255f
|
boneWeight = cursor.uShort().toFloat() / 255f
|
||||||
} else {
|
} else {
|
||||||
|
// NJD_CV_VN_D8
|
||||||
|
// NJD_CV_VN_UF
|
||||||
|
// NJD_CV_VN_S5
|
||||||
|
// NJD_CV_VN_S4
|
||||||
|
// NJD_CV_VN_IN
|
||||||
// Skip user flags and material information.
|
// Skip user flags and material information.
|
||||||
cursor.seek(4)
|
cursor.seek(4)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
in 48..50 -> {
|
in 48..50 -> {
|
||||||
|
// NJD_CV_VNX
|
||||||
// 32-Bit vertex normal in format: reserved(2)|x(10)|y(10)|z(10)
|
// 32-Bit vertex normal in format: reserved(2)|x(10)|y(10)|z(10)
|
||||||
val n = cursor.uInt()
|
val n = cursor.uInt()
|
||||||
normal = Vec3(
|
normal = Vec3(
|
||||||
((n shr 20) and 0x3ffu).toFloat() / 0x3ff,
|
((n shr 20) and 0x3FFu).toFloat() / 0x3FF,
|
||||||
((n shr 10) and 0x3ffu).toFloat() / 0x3ff,
|
((n shr 10) and 0x3FFu).toFloat() / 0x3FF,
|
||||||
(n and 0x3ffu).toFloat() / 0x3ff,
|
(n and 0x3FFu).toFloat() / 0x3FF,
|
||||||
)
|
)
|
||||||
|
|
||||||
if (chunkTypeId >= 49u) {
|
if (chunkTypeId >= 49) {
|
||||||
|
// NJD_CV_VNX_D8
|
||||||
|
// NJD_CV_VNX_UF
|
||||||
// Skip user flags and material information.
|
// Skip user flags and material information.
|
||||||
cursor.seek(4)
|
cursor.seek(4)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else -> error("Unexpected chunk type ID ${chunkTypeId}.")
|
||||||
}
|
}
|
||||||
|
|
||||||
vertices.add(NjChunkVertex(
|
vertices.add(NjChunkVertex(
|
||||||
@ -330,7 +394,7 @@ private fun parseVertexChunk(
|
|||||||
|
|
||||||
private fun parseTriangleStripChunk(
|
private fun parseTriangleStripChunk(
|
||||||
cursor: Cursor,
|
cursor: Cursor,
|
||||||
chunkTypeId: UByte,
|
chunkTypeId: Int,
|
||||||
flags: Int,
|
flags: Int,
|
||||||
): List<NjTriangleStrip> {
|
): List<NjTriangleStrip> {
|
||||||
val ignoreLight = flags.isBitSet(0)
|
val ignoreLight = flags.isBitSet(0)
|
||||||
@ -342,7 +406,7 @@ private fun parseTriangleStripChunk(
|
|||||||
val environmentMapping = flags.isBitSet(6)
|
val environmentMapping = flags.isBitSet(6)
|
||||||
|
|
||||||
val userOffsetAndStripCount = cursor.short().toInt()
|
val userOffsetAndStripCount = cursor.short().toInt()
|
||||||
val userFlagsSize = (userOffsetAndStripCount ushr 14)
|
val userFlagsSize = 2 * (userOffsetAndStripCount ushr 14)
|
||||||
val stripCount = userOffsetAndStripCount and 0x3FFF
|
val stripCount = userOffsetAndStripCount and 0x3FFF
|
||||||
|
|
||||||
var hasTexCoords = false
|
var hasTexCoords = false
|
||||||
@ -350,7 +414,7 @@ private fun parseTriangleStripChunk(
|
|||||||
var hasNormal = false
|
var hasNormal = false
|
||||||
var hasDoubleTexCoords = false
|
var hasDoubleTexCoords = false
|
||||||
|
|
||||||
when (chunkTypeId.toInt()) {
|
when (chunkTypeId) {
|
||||||
64 -> {
|
64 -> {
|
||||||
}
|
}
|
||||||
65, 66 -> {
|
65, 66 -> {
|
||||||
@ -381,9 +445,9 @@ private fun parseTriangleStripChunk(
|
|||||||
val strips: MutableList<NjTriangleStrip> = mutableListOf()
|
val strips: MutableList<NjTriangleStrip> = mutableListOf()
|
||||||
|
|
||||||
repeat(stripCount) {
|
repeat(stripCount) {
|
||||||
val windingFlagAndIndexCount = cursor.short()
|
val windingFlagAndIndexCount = cursor.short().toInt()
|
||||||
val clockwiseWinding = windingFlagAndIndexCount < 1
|
val clockwiseWinding = windingFlagAndIndexCount < 0
|
||||||
val indexCount = abs(windingFlagAndIndexCount.toInt())
|
val indexCount = abs(windingFlagAndIndexCount)
|
||||||
|
|
||||||
val vertices: MutableList<NjMeshVertex> = mutableListOf()
|
val vertices: MutableList<NjMeshVertex> = mutableListOf()
|
||||||
|
|
||||||
@ -414,7 +478,7 @@ private fun parseTriangleStripChunk(
|
|||||||
|
|
||||||
// User flags start at the third vertex because they are per-triangle.
|
// User flags start at the third vertex because they are per-triangle.
|
||||||
if (j >= 2) {
|
if (j >= 2) {
|
||||||
cursor.seek(2 * userFlagsSize)
|
cursor.seek(userFlagsSize)
|
||||||
}
|
}
|
||||||
|
|
||||||
vertices.add(NjMeshVertex(
|
vertices.add(NjMeshVertex(
|
||||||
|
@ -22,7 +22,7 @@ fun parseXjModel(cursor: Cursor): XjModel {
|
|||||||
|
|
||||||
val vertices = mutableListOf<XjVertex>()
|
val vertices = mutableListOf<XjVertex>()
|
||||||
|
|
||||||
if (vertexInfoCount >= 1) {
|
if (vertexInfoCount > 0) {
|
||||||
// TODO: parse all vertex info tables.
|
// TODO: parse all vertex info tables.
|
||||||
vertices.addAll(parseVertexInfoTable(cursor, vertexInfoTableOffset))
|
vertices.addAll(parseVertexInfoTable(cursor, vertexInfoTableOffset))
|
||||||
}
|
}
|
||||||
@ -102,11 +102,11 @@ private fun parseVertexInfoTable(cursor: Cursor, vertexInfoTableOffset: Int): Li
|
|||||||
|
|
||||||
private fun parseTriangleStripTable(
|
private fun parseTriangleStripTable(
|
||||||
cursor: Cursor,
|
cursor: Cursor,
|
||||||
triangle_strip_list_offset: Int,
|
triangleStripListOffset: Int,
|
||||||
triangle_strip_count: Int,
|
triangleStripCount: Int,
|
||||||
): List<XjMesh> {
|
): List<XjMesh> {
|
||||||
return (0 until triangle_strip_count).map { i ->
|
return (0 until triangleStripCount).map { i ->
|
||||||
cursor.seekStart(triangle_strip_list_offset + i * 20)
|
cursor.seekStart(triangleStripListOffset + i * 20)
|
||||||
|
|
||||||
val materialTableOffset = cursor.int()
|
val materialTableOffset = cursor.int()
|
||||||
val materialTableSize = cursor.int()
|
val materialTableSize = cursor.int()
|
||||||
@ -124,7 +124,7 @@ private fun parseTriangleStripTable(
|
|||||||
|
|
||||||
XjMesh(
|
XjMesh(
|
||||||
material,
|
material,
|
||||||
indices = List(indexCount) { indices[it].toInt() },
|
indices = indices.map { it.toInt() },
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -56,10 +56,29 @@ class MeshBuilder(
|
|||||||
fun getNormal(index: Int): Vector3 =
|
fun getNormal(index: Int): Vector3 =
|
||||||
normals[index]
|
normals[index]
|
||||||
|
|
||||||
fun vertex(position: Vector3, normal: Vector3, uv: Vector2? = null) {
|
fun vertex(
|
||||||
|
position: Vector3,
|
||||||
|
normal: Vector3,
|
||||||
|
uv: Vector2? = null,
|
||||||
|
boneIndices: IntArray? = null,
|
||||||
|
boneWeights: FloatArray? = null,
|
||||||
|
) {
|
||||||
positions.add(position)
|
positions.add(position)
|
||||||
normals.add(normal)
|
normals.add(normal)
|
||||||
uv?.let { uvs.add(uv) }
|
uv?.let { uvs.add(uv) }
|
||||||
|
|
||||||
|
if (boneIndices != null && boneWeights != null) {
|
||||||
|
require(boneIndices.size == 4)
|
||||||
|
require(boneWeights.size == 4)
|
||||||
|
|
||||||
|
for (index in boneIndices) {
|
||||||
|
this.boneIndices.add(index.toShort())
|
||||||
|
}
|
||||||
|
|
||||||
|
for (weight in boneWeights) {
|
||||||
|
this.boneWeights.add(weight)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fun index(groupIdx: Int, index: Int) {
|
fun index(groupIdx: Int, index: Int) {
|
||||||
@ -71,19 +90,6 @@ class MeshBuilder(
|
|||||||
bones.add(bone)
|
bones.add(bone)
|
||||||
}
|
}
|
||||||
|
|
||||||
fun boneWeights(indices: IntArray, weights: FloatArray) {
|
|
||||||
require(indices.size == 4)
|
|
||||||
require(weights.size == 4)
|
|
||||||
|
|
||||||
for (index in indices) {
|
|
||||||
boneIndices.add(index.toShort())
|
|
||||||
}
|
|
||||||
|
|
||||||
for (weight in weights) {
|
|
||||||
boneWeights.add(weight)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fun defaultMaterial(material: Material) {
|
fun defaultMaterial(material: Material) {
|
||||||
defaultMaterial = material
|
defaultMaterial = material
|
||||||
}
|
}
|
||||||
|
@ -100,7 +100,9 @@ private class NinjaToMeshConverter(private val builder: MeshBuilder) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!ef.skip) {
|
||||||
boneIndex++
|
boneIndex++
|
||||||
|
}
|
||||||
|
|
||||||
if (!ef.breakChildTrace) {
|
if (!ef.breakChildTrace) {
|
||||||
obj.children.forEach { child ->
|
obj.children.forEach { child ->
|
||||||
@ -132,7 +134,6 @@ private class NinjaToMeshConverter(private val builder: MeshBuilder) {
|
|||||||
normal,
|
normal,
|
||||||
vertex.boneWeight,
|
vertex.boneWeight,
|
||||||
vertex.boneWeightStatus,
|
vertex.boneWeightStatus,
|
||||||
vertex.calcContinue,
|
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -143,7 +144,7 @@ private class NinjaToMeshConverter(private val builder: MeshBuilder) {
|
|||||||
val group = builder.getGroupIndex(
|
val group = builder.getGroupIndex(
|
||||||
mesh.textureId,
|
mesh.textureId,
|
||||||
alpha = mesh.useAlpha,
|
alpha = mesh.useAlpha,
|
||||||
additiveBlending = mesh.srcAlpha != 4 || mesh.dstAlpha != 5
|
additiveBlending = mesh.srcAlpha != 4 || mesh.dstAlpha != 5,
|
||||||
)
|
)
|
||||||
var i = 0
|
var i = 0
|
||||||
|
|
||||||
@ -160,10 +161,35 @@ private class NinjaToMeshConverter(private val builder: MeshBuilder) {
|
|||||||
vertex.normal ?: meshVertex.normal?.let(::vec3ToThree) ?: DEFAULT_NORMAL
|
vertex.normal ?: meshVertex.normal?.let(::vec3ToThree) ?: DEFAULT_NORMAL
|
||||||
val index = builder.vertexCount
|
val index = builder.vertexCount
|
||||||
|
|
||||||
|
val boneIndices = IntArray(4)
|
||||||
|
val boneWeights = FloatArray(4)
|
||||||
|
|
||||||
|
if (vertex.boneWeight == null) {
|
||||||
|
boneIndices[0] = vertex.boneIndex
|
||||||
|
boneWeights[0] = 1f
|
||||||
|
} else {
|
||||||
|
for (v in vertices) {
|
||||||
|
boneIndices[v.boneWeightStatus] = v.boneIndex
|
||||||
|
boneWeights[v.boneWeightStatus] = v.boneWeight ?: 1f
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
val totalWeight = boneWeights.sum()
|
||||||
|
|
||||||
|
if (totalWeight > 0f) {
|
||||||
|
val weightFactor = 1f / totalWeight
|
||||||
|
|
||||||
|
for (j in boneWeights.indices) {
|
||||||
|
boneWeights[j] *= weightFactor
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
builder.vertex(
|
builder.vertex(
|
||||||
vertex.position,
|
vertex.position,
|
||||||
normal,
|
normal,
|
||||||
meshVertex.texCoords?.let(::vec2ToThree) ?: DEFAULT_UV
|
meshVertex.texCoords?.let(::vec2ToThree) ?: DEFAULT_UV,
|
||||||
|
boneIndices,
|
||||||
|
boneWeights,
|
||||||
)
|
)
|
||||||
|
|
||||||
if (i >= 2) {
|
if (i >= 2) {
|
||||||
@ -178,26 +204,6 @@ private class NinjaToMeshConverter(private val builder: MeshBuilder) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
val boneIndices = IntArray(4)
|
|
||||||
val boneWeights = FloatArray(4)
|
|
||||||
|
|
||||||
for (v in vertices) {
|
|
||||||
boneIndices[v.boneWeightStatus] = v.boneIndex
|
|
||||||
boneWeights[v.boneWeightStatus] = v.boneWeight
|
|
||||||
}
|
|
||||||
|
|
||||||
val totalWeight = boneWeights.sum()
|
|
||||||
|
|
||||||
if (totalWeight > 0f) {
|
|
||||||
val weightFactor = 1f / totalWeight
|
|
||||||
|
|
||||||
for (j in boneWeights.indices) {
|
|
||||||
boneWeights[j] *= weightFactor
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
builder.boneWeights(boneIndices, boneWeights)
|
|
||||||
|
|
||||||
i++
|
i++
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -290,9 +296,8 @@ private class Vertex(
|
|||||||
val boneIndex: Int,
|
val boneIndex: Int,
|
||||||
val position: Vector3,
|
val position: Vector3,
|
||||||
val normal: Vector3?,
|
val normal: Vector3?,
|
||||||
val boneWeight: Float,
|
val boneWeight: Float?,
|
||||||
val boneWeightStatus: Int,
|
val boneWeightStatus: Int,
|
||||||
val calcContinue: Boolean,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
private class VertexHolder {
|
private class VertexHolder {
|
||||||
|
Loading…
Reference in New Issue
Block a user