mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-04 06:28:28 +08:00
Worked on mipmapping and anisotropy.
This commit is contained in:
parent
423c3e252b
commit
62a49b067c
@ -13,6 +13,7 @@ import world.phantasmal.webui.obj
|
|||||||
class MeshBuilder(
|
class MeshBuilder(
|
||||||
private val textures: List<XvrTexture?> = emptyList(),
|
private val textures: List<XvrTexture?> = emptyList(),
|
||||||
private val textureCache: UnsafeMap<Int, Texture?> = UnsafeMap(),
|
private val textureCache: UnsafeMap<Int, Texture?> = UnsafeMap(),
|
||||||
|
private val anisotropy: Int = 1,
|
||||||
) {
|
) {
|
||||||
private val positions = mutableListOf<Vector3>()
|
private val positions = mutableListOf<Vector3>()
|
||||||
private val normals = mutableListOf<Vector3>()
|
private val normals = mutableListOf<Vector3>()
|
||||||
@ -197,7 +198,7 @@ class MeshBuilder(
|
|||||||
|
|
||||||
if (tex == null) {
|
if (tex == null) {
|
||||||
tex = textures.getOrNull(group.textureIndex)?.let { xvm ->
|
tex = textures.getOrNull(group.textureIndex)?.let { xvm ->
|
||||||
xvrTextureToThree(xvm)
|
xvrTextureToThree(xvm, anisotropy = anisotropy)
|
||||||
}
|
}
|
||||||
textureCache.set(group.textureIndex, tex)
|
textureCache.set(group.textureIndex, tex)
|
||||||
}
|
}
|
||||||
|
@ -8,11 +8,40 @@ import world.phantasmal.core.asArray
|
|||||||
import world.phantasmal.core.isBitSet
|
import world.phantasmal.core.isBitSet
|
||||||
import world.phantasmal.core.jsArrayOf
|
import world.phantasmal.core.jsArrayOf
|
||||||
import world.phantasmal.core.unsafe.UnsafeMap
|
import world.phantasmal.core.unsafe.UnsafeMap
|
||||||
import world.phantasmal.psolib.fileFormats.*
|
import world.phantasmal.psolib.fileFormats.AreaGeometry
|
||||||
import world.phantasmal.psolib.fileFormats.ninja.*
|
import world.phantasmal.psolib.fileFormats.AreaObject
|
||||||
|
import world.phantasmal.psolib.fileFormats.AreaSection
|
||||||
|
import world.phantasmal.psolib.fileFormats.CollisionGeometry
|
||||||
|
import world.phantasmal.psolib.fileFormats.CollisionTriangle
|
||||||
|
import world.phantasmal.psolib.fileFormats.ninja.NinjaModel
|
||||||
|
import world.phantasmal.psolib.fileFormats.ninja.NinjaObject
|
||||||
|
import world.phantasmal.psolib.fileFormats.ninja.NjModel
|
||||||
|
import world.phantasmal.psolib.fileFormats.ninja.NjObject
|
||||||
|
import world.phantasmal.psolib.fileFormats.ninja.XjModel
|
||||||
|
import world.phantasmal.psolib.fileFormats.ninja.XjObject
|
||||||
|
import world.phantasmal.psolib.fileFormats.ninja.XvrTexture
|
||||||
import world.phantasmal.web.core.dot
|
import world.phantasmal.web.core.dot
|
||||||
import world.phantasmal.web.core.toQuaternion
|
import world.phantasmal.web.core.toQuaternion
|
||||||
import world.phantasmal.web.externals.three.*
|
import world.phantasmal.web.externals.three.Bone
|
||||||
|
import world.phantasmal.web.externals.three.BufferGeometry
|
||||||
|
import world.phantasmal.web.externals.three.Color
|
||||||
|
import world.phantasmal.web.externals.three.DoubleSide
|
||||||
|
import world.phantasmal.web.externals.three.Euler
|
||||||
|
import world.phantasmal.web.externals.three.Float32BufferAttribute
|
||||||
|
import world.phantasmal.web.externals.three.Group
|
||||||
|
import world.phantasmal.web.externals.three.InstancedMesh
|
||||||
|
import world.phantasmal.web.externals.three.Material
|
||||||
|
import world.phantasmal.web.externals.three.Matrix3
|
||||||
|
import world.phantasmal.web.externals.three.Matrix4
|
||||||
|
import world.phantasmal.web.externals.three.Mesh
|
||||||
|
import world.phantasmal.web.externals.three.MeshBasicMaterial
|
||||||
|
import world.phantasmal.web.externals.three.MeshLambertMaterial
|
||||||
|
import world.phantasmal.web.externals.three.Quaternion
|
||||||
|
import world.phantasmal.web.externals.three.SkinnedMesh
|
||||||
|
import world.phantasmal.web.externals.three.Texture
|
||||||
|
import world.phantasmal.web.externals.three.Uint16BufferAttribute
|
||||||
|
import world.phantasmal.web.externals.three.Vector2
|
||||||
|
import world.phantasmal.web.externals.three.Vector3
|
||||||
import world.phantasmal.webui.obj
|
import world.phantasmal.webui.obj
|
||||||
import kotlin.collections.component1
|
import kotlin.collections.component1
|
||||||
import kotlin.collections.component2
|
import kotlin.collections.component2
|
||||||
@ -86,8 +115,9 @@ fun ninjaObjectToMesh(
|
|||||||
textures: List<XvrTexture?>,
|
textures: List<XvrTexture?>,
|
||||||
defaultMaterial: Material? = null,
|
defaultMaterial: Material? = null,
|
||||||
boundingVolumes: Boolean = false,
|
boundingVolumes: Boolean = false,
|
||||||
|
anisotropy: Int = 1,
|
||||||
): Mesh {
|
): Mesh {
|
||||||
val builder = MeshBuilder(textures)
|
val builder = MeshBuilder(textures, anisotropy = anisotropy)
|
||||||
defaultMaterial?.let { builder.defaultMaterial(defaultMaterial) }
|
defaultMaterial?.let { builder.defaultMaterial(defaultMaterial) }
|
||||||
ninjaObjectToMeshBuilder(ninjaObject, builder)
|
ninjaObjectToMeshBuilder(ninjaObject, builder)
|
||||||
return builder.buildMesh(boundingVolumes)
|
return builder.buildMesh(boundingVolumes)
|
||||||
@ -99,8 +129,9 @@ fun ninjaObjectToInstancedMesh(
|
|||||||
maxInstances: Int,
|
maxInstances: Int,
|
||||||
defaultMaterial: Material? = null,
|
defaultMaterial: Material? = null,
|
||||||
boundingVolumes: Boolean = false,
|
boundingVolumes: Boolean = false,
|
||||||
|
anisotropy: Int = 1,
|
||||||
): InstancedMesh {
|
): InstancedMesh {
|
||||||
val builder = MeshBuilder(textures)
|
val builder = MeshBuilder(textures, anisotropy = anisotropy)
|
||||||
defaultMaterial?.let { builder.defaultMaterial(defaultMaterial) }
|
defaultMaterial?.let { builder.defaultMaterial(defaultMaterial) }
|
||||||
ninjaObjectToMeshBuilder(ninjaObject, builder)
|
ninjaObjectToMeshBuilder(ninjaObject, builder)
|
||||||
return builder.buildInstancedMesh(maxInstances, boundingVolumes)
|
return builder.buildInstancedMesh(maxInstances, boundingVolumes)
|
||||||
@ -111,8 +142,9 @@ fun ninjaObjectToSkinnedMesh(
|
|||||||
textures: List<XvrTexture?>,
|
textures: List<XvrTexture?>,
|
||||||
defaultMaterial: Material? = null,
|
defaultMaterial: Material? = null,
|
||||||
boundingVolumes: Boolean = false,
|
boundingVolumes: Boolean = false,
|
||||||
|
anisotropy: Int = 1,
|
||||||
): SkinnedMesh {
|
): SkinnedMesh {
|
||||||
val builder = MeshBuilder(textures)
|
val builder = MeshBuilder(textures, anisotropy = anisotropy)
|
||||||
defaultMaterial?.let { builder.defaultMaterial(defaultMaterial) }
|
defaultMaterial?.let { builder.defaultMaterial(defaultMaterial) }
|
||||||
ninjaObjectToMeshBuilder(ninjaObject, builder)
|
ninjaObjectToMeshBuilder(ninjaObject, builder)
|
||||||
return builder.buildSkinnedMesh(boundingVolumes)
|
return builder.buildSkinnedMesh(boundingVolumes)
|
||||||
@ -129,6 +161,7 @@ fun ninjaObjectToMeshBuilder(
|
|||||||
fun renderGeometryToGroup(
|
fun renderGeometryToGroup(
|
||||||
renderGeometry: AreaGeometry,
|
renderGeometry: AreaGeometry,
|
||||||
textures: List<XvrTexture?>,
|
textures: List<XvrTexture?>,
|
||||||
|
anisotropy: Int = 1,
|
||||||
processMesh: (AreaSection, AreaObject, Mesh) -> Unit = { _, _, _ -> },
|
processMesh: (AreaSection, AreaObject, Mesh) -> Unit = { _, _, _ -> },
|
||||||
): Group {
|
): Group {
|
||||||
val group = Group()
|
val group = Group()
|
||||||
@ -145,6 +178,7 @@ fun renderGeometryToGroup(
|
|||||||
section,
|
section,
|
||||||
sectionIndex,
|
sectionIndex,
|
||||||
areaObj,
|
areaObj,
|
||||||
|
anisotropy,
|
||||||
processMesh,
|
processMesh,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
@ -159,6 +193,7 @@ fun renderGeometryToGroup(
|
|||||||
section,
|
section,
|
||||||
sectionIndex,
|
sectionIndex,
|
||||||
areaObj,
|
areaObj,
|
||||||
|
anisotropy,
|
||||||
processMesh,
|
processMesh,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
@ -214,13 +249,14 @@ private fun areaObjectToMesh(
|
|||||||
section: AreaSection,
|
section: AreaSection,
|
||||||
sectionIndex: Int,
|
sectionIndex: Int,
|
||||||
areaObj: AreaObject,
|
areaObj: AreaObject,
|
||||||
|
anisotropy: Int,
|
||||||
processMesh: (AreaSection, AreaObject, Mesh) -> Unit,
|
processMesh: (AreaSection, AreaObject, Mesh) -> Unit,
|
||||||
): Mesh {
|
): Mesh {
|
||||||
val cachedMesh = meshCache.get(areaObj.xjObject)
|
val cachedMesh = meshCache.get(areaObj.xjObject)
|
||||||
val mesh: Mesh
|
val mesh: Mesh
|
||||||
|
|
||||||
if (cachedMesh == null) {
|
if (cachedMesh == null) {
|
||||||
val builder = MeshBuilder(textures, textureCache)
|
val builder = MeshBuilder(textures, textureCache, anisotropy)
|
||||||
ninjaObjectToMeshBuilder(areaObj.xjObject, builder)
|
ninjaObjectToMeshBuilder(areaObj.xjObject, builder)
|
||||||
|
|
||||||
builder.defaultMaterial(MeshLambertMaterial(obj {
|
builder.defaultMaterial(MeshLambertMaterial(obj {
|
||||||
|
@ -6,11 +6,32 @@ import org.khronos.webgl.get
|
|||||||
import org.khronos.webgl.set
|
import org.khronos.webgl.set
|
||||||
import world.phantasmal.psolib.cursor.cursor
|
import world.phantasmal.psolib.cursor.cursor
|
||||||
import world.phantasmal.psolib.fileFormats.ninja.XvrTexture
|
import world.phantasmal.psolib.fileFormats.ninja.XvrTexture
|
||||||
import world.phantasmal.web.externals.three.*
|
import world.phantasmal.web.externals.three.CompressedPixelFormat
|
||||||
|
import world.phantasmal.web.externals.three.CompressedTexture
|
||||||
|
import world.phantasmal.web.externals.three.DataTexture
|
||||||
|
import world.phantasmal.web.externals.three.LinearFilter
|
||||||
|
import world.phantasmal.web.externals.three.Mipmap
|
||||||
|
import world.phantasmal.web.externals.three.MirroredRepeatWrapping
|
||||||
|
import world.phantasmal.web.externals.three.PixelFormat
|
||||||
|
import world.phantasmal.web.externals.three.RGBAFormat
|
||||||
|
import world.phantasmal.web.externals.three.RGBA_S3TC_DXT1_Format
|
||||||
|
import world.phantasmal.web.externals.three.RGBA_S3TC_DXT3_Format
|
||||||
|
import world.phantasmal.web.externals.three.RGBFormat
|
||||||
|
import world.phantasmal.web.externals.three.Texture
|
||||||
|
import world.phantasmal.web.externals.three.TextureDataType
|
||||||
|
import world.phantasmal.web.externals.three.TextureFilter
|
||||||
|
import world.phantasmal.web.externals.three.UnsignedShort5551Type
|
||||||
|
import world.phantasmal.web.externals.three.UnsignedShort565Type
|
||||||
import world.phantasmal.webui.obj
|
import world.phantasmal.webui.obj
|
||||||
import kotlin.math.roundToInt
|
import kotlin.math.roundToInt
|
||||||
|
|
||||||
fun xvrTextureToThree(xvr: XvrTexture, filter: TextureFilter = LinearFilter): Texture =
|
fun xvrTextureToThree(
|
||||||
|
xvr: XvrTexture,
|
||||||
|
magFilter: TextureFilter = LinearFilter,
|
||||||
|
// TODO: Use LinearMipmapLinearFilter once we figure out mipmapping.
|
||||||
|
minFilter: TextureFilter = LinearFilter,
|
||||||
|
anisotropy: Int = 1,
|
||||||
|
): Texture =
|
||||||
when (xvr.format.second) {
|
when (xvr.format.second) {
|
||||||
// D3DFMT_R5G6B5
|
// D3DFMT_R5G6B5
|
||||||
2 -> createDataTexture(
|
2 -> createDataTexture(
|
||||||
@ -19,7 +40,9 @@ fun xvrTextureToThree(xvr: XvrTexture, filter: TextureFilter = LinearFilter): Te
|
|||||||
xvr.height,
|
xvr.height,
|
||||||
RGBFormat,
|
RGBFormat,
|
||||||
UnsignedShort565Type,
|
UnsignedShort565Type,
|
||||||
filter,
|
magFilter,
|
||||||
|
minFilter,
|
||||||
|
anisotropy,
|
||||||
)
|
)
|
||||||
// D3DFMT_A1R5G5B5
|
// D3DFMT_A1R5G5B5
|
||||||
3 -> {
|
3 -> {
|
||||||
@ -38,26 +61,84 @@ fun xvrTextureToThree(xvr: XvrTexture, filter: TextureFilter = LinearFilter): Te
|
|||||||
xvr.height,
|
xvr.height,
|
||||||
RGBAFormat,
|
RGBAFormat,
|
||||||
UnsignedShort5551Type,
|
UnsignedShort5551Type,
|
||||||
filter,
|
magFilter,
|
||||||
|
minFilter,
|
||||||
|
anisotropy,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
// D3DFMT_DXT1
|
// D3DFMT_DXT1
|
||||||
6 -> createCompressedTexture(
|
6 -> {
|
||||||
Uint8Array(xvr.data.arrayBuffer, 0, (xvr.width * xvr.height) / 2),
|
val mipmaps = mutableListOf<Mipmap>()
|
||||||
xvr.width,
|
var byteOffset = 0
|
||||||
xvr.height,
|
var width = xvr.width
|
||||||
RGBA_S3TC_DXT1_Format,
|
var height = xvr.height
|
||||||
filter,
|
|
||||||
)
|
while (byteOffset < xvr.data.size && width * height > 0) {
|
||||||
|
val byteSize = (width * height) / 2
|
||||||
|
|
||||||
|
mipmaps.add(obj {
|
||||||
|
this.data = Uint8Array(xvr.data.arrayBuffer, byteOffset, byteSize)
|
||||||
|
this.width = width
|
||||||
|
this.height = height
|
||||||
|
})
|
||||||
|
|
||||||
|
byteOffset += byteSize
|
||||||
|
width /= 2
|
||||||
|
height /= 2
|
||||||
|
|
||||||
|
// TODO: Figure out what the problem with mipmaps is and remove this break.
|
||||||
|
// Do we interpret the XVR format incorrectly or is there a problem with
|
||||||
|
// Three.js/WebGL?
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
createCompressedTexture(
|
||||||
|
mipmaps.toTypedArray(),
|
||||||
|
xvr.width,
|
||||||
|
xvr.height,
|
||||||
|
RGBA_S3TC_DXT1_Format,
|
||||||
|
magFilter,
|
||||||
|
minFilter,
|
||||||
|
anisotropy,
|
||||||
|
)
|
||||||
|
}
|
||||||
// D3DFMT_DXT2
|
// D3DFMT_DXT2
|
||||||
// TODO: Correctly interpret this (DXT2 is basically DXT3 with premultiplied alpha).
|
// TODO: Correctly interpret this (DXT2 is basically DXT3 with premultiplied alpha).
|
||||||
7 -> createCompressedTexture(
|
7 -> {
|
||||||
Uint8Array(xvr.data.arrayBuffer, 0, xvr.width * xvr.height),
|
val mipmaps = mutableListOf<Mipmap>()
|
||||||
xvr.width,
|
var byteOffset = 0
|
||||||
xvr.height,
|
var width = xvr.width
|
||||||
RGBA_S3TC_DXT3_Format,
|
var height = xvr.height
|
||||||
filter,
|
|
||||||
)
|
while (byteOffset < xvr.data.size && width * height > 0) {
|
||||||
|
val byteSize = width * height
|
||||||
|
|
||||||
|
mipmaps.add(obj {
|
||||||
|
this.data = Uint8Array(xvr.data.arrayBuffer, byteOffset, byteSize)
|
||||||
|
this.width = width
|
||||||
|
this.height = height
|
||||||
|
})
|
||||||
|
|
||||||
|
byteOffset += byteSize
|
||||||
|
width /= 2
|
||||||
|
height /= 2
|
||||||
|
|
||||||
|
// TODO: Figure out what the problem with mipmaps is and remove this break.
|
||||||
|
// Do we interpret the XVR format incorrectly or is there a problem with
|
||||||
|
// Three.js/WebGL?
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
createCompressedTexture(
|
||||||
|
mipmaps.toTypedArray(),
|
||||||
|
xvr.width,
|
||||||
|
xvr.height,
|
||||||
|
RGBA_S3TC_DXT3_Format,
|
||||||
|
magFilter,
|
||||||
|
minFilter,
|
||||||
|
anisotropy,
|
||||||
|
)
|
||||||
|
}
|
||||||
// 1 -> D3DFMT_A8R8G8B8
|
// 1 -> D3DFMT_A8R8G8B8
|
||||||
// 4 -> D3DFMT_A4R4G4B4
|
// 4 -> D3DFMT_A4R4G4B4
|
||||||
// 5 -> D3DFMT_P8
|
// 5 -> D3DFMT_P8
|
||||||
@ -83,7 +164,9 @@ private fun createDataTexture(
|
|||||||
height: Int,
|
height: Int,
|
||||||
format: PixelFormat,
|
format: PixelFormat,
|
||||||
type: TextureDataType,
|
type: TextureDataType,
|
||||||
filter: TextureFilter,
|
magFilter: TextureFilter,
|
||||||
|
minFilter: TextureFilter,
|
||||||
|
anisotropy: Int,
|
||||||
): DataTexture =
|
): DataTexture =
|
||||||
DataTexture(
|
DataTexture(
|
||||||
data,
|
data,
|
||||||
@ -93,30 +176,30 @@ private fun createDataTexture(
|
|||||||
type,
|
type,
|
||||||
wrapS = MirroredRepeatWrapping,
|
wrapS = MirroredRepeatWrapping,
|
||||||
wrapT = MirroredRepeatWrapping,
|
wrapT = MirroredRepeatWrapping,
|
||||||
magFilter = filter,
|
magFilter = magFilter,
|
||||||
minFilter = filter,
|
minFilter = minFilter,
|
||||||
|
anisotropy = anisotropy,
|
||||||
)
|
)
|
||||||
|
|
||||||
private fun createCompressedTexture(
|
private fun createCompressedTexture(
|
||||||
data: Uint8Array,
|
mipmaps: Array<Mipmap>,
|
||||||
width: Int,
|
width: Int,
|
||||||
height: Int,
|
height: Int,
|
||||||
format: CompressedPixelFormat,
|
format: CompressedPixelFormat,
|
||||||
filter: TextureFilter,
|
magFilter: TextureFilter,
|
||||||
|
minFilter: TextureFilter,
|
||||||
|
anisotropy: Int,
|
||||||
): CompressedTexture {
|
): CompressedTexture {
|
||||||
val texture = CompressedTexture(
|
val texture = CompressedTexture(
|
||||||
arrayOf(obj {
|
mipmaps,
|
||||||
this.data = data
|
|
||||||
this.width = width
|
|
||||||
this.height = height
|
|
||||||
}),
|
|
||||||
width,
|
width,
|
||||||
height,
|
height,
|
||||||
format,
|
format,
|
||||||
wrapS = MirroredRepeatWrapping,
|
wrapS = MirroredRepeatWrapping,
|
||||||
wrapT = MirroredRepeatWrapping,
|
wrapT = MirroredRepeatWrapping,
|
||||||
magFilter = filter,
|
magFilter = magFilter,
|
||||||
minFilter = filter,
|
minFilter = minFilter,
|
||||||
|
anisotropy = anisotropy,
|
||||||
)
|
)
|
||||||
texture.needsUpdate = true
|
texture.needsUpdate = true
|
||||||
return texture
|
return texture
|
||||||
@ -165,12 +248,14 @@ private fun xvrTextureToUint8Array(xvr: XvrTexture): Uint8Array {
|
|||||||
b = c0b
|
b = c0b
|
||||||
a = 1.0
|
a = 1.0
|
||||||
}
|
}
|
||||||
|
|
||||||
1 -> {
|
1 -> {
|
||||||
r = c1r
|
r = c1r
|
||||||
g = c1g
|
g = c1g
|
||||||
b = c1b
|
b = c1b
|
||||||
a = 1.0
|
a = 1.0
|
||||||
}
|
}
|
||||||
|
|
||||||
2 -> {
|
2 -> {
|
||||||
if (c0 > c1) {
|
if (c0 > c1) {
|
||||||
r = (2 * c0r + c1r) / 3
|
r = (2 * c0r + c1r) / 3
|
||||||
@ -184,6 +269,7 @@ private fun xvrTextureToUint8Array(xvr: XvrTexture): Uint8Array {
|
|||||||
a = 1.0
|
a = 1.0
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
3 -> {
|
3 -> {
|
||||||
if (c0 > c1) {
|
if (c0 > c1) {
|
||||||
r = (c0r + 2 * c1r) / 3
|
r = (c0r + 2 * c1r) / 3
|
||||||
|
@ -7,6 +7,7 @@ package world.phantasmal.web.externals.three
|
|||||||
import org.khronos.webgl.Float32Array
|
import org.khronos.webgl.Float32Array
|
||||||
import org.khronos.webgl.Int32Array
|
import org.khronos.webgl.Int32Array
|
||||||
import org.khronos.webgl.Uint16Array
|
import org.khronos.webgl.Uint16Array
|
||||||
|
import org.khronos.webgl.Uint8Array
|
||||||
import org.w3c.dom.HTMLCanvasElement
|
import org.w3c.dom.HTMLCanvasElement
|
||||||
|
|
||||||
external interface Vector
|
external interface Vector
|
||||||
@ -235,6 +236,7 @@ open external class WebGLRenderer(
|
|||||||
|
|
||||||
var autoClearColor: Boolean
|
var autoClearColor: Boolean
|
||||||
var debug: WebGLDebug
|
var debug: WebGLDebug
|
||||||
|
var capabilities: WebGLCapabilities
|
||||||
|
|
||||||
override fun render(scene: Object3D, camera: Camera)
|
override fun render(scene: Object3D, camera: Camera)
|
||||||
|
|
||||||
@ -253,6 +255,10 @@ external interface WebGLDebug {
|
|||||||
var checkShaderErrors: Boolean
|
var checkShaderErrors: Boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
|
external interface WebGLCapabilities {
|
||||||
|
fun getMaxAnisotropy(): Int
|
||||||
|
}
|
||||||
|
|
||||||
open external class Object3D {
|
open external class Object3D {
|
||||||
/**
|
/**
|
||||||
* Optional name of the object (doesn't need to be unique).
|
* Optional name of the object (doesn't need to be unique).
|
||||||
@ -686,7 +692,7 @@ external class DataTexture(
|
|||||||
wrapT: Wrapping = definedExternally,
|
wrapT: Wrapping = definedExternally,
|
||||||
magFilter: TextureFilter = definedExternally,
|
magFilter: TextureFilter = definedExternally,
|
||||||
minFilter: TextureFilter = definedExternally,
|
minFilter: TextureFilter = definedExternally,
|
||||||
anisotropy: Double = definedExternally,
|
anisotropy: Int = definedExternally,
|
||||||
encoding: TextureEncoding = definedExternally,
|
encoding: TextureEncoding = definedExternally,
|
||||||
) : Texture
|
) : Texture
|
||||||
|
|
||||||
@ -707,14 +713,10 @@ external object MirroredRepeatWrapping : Wrapping
|
|||||||
external interface TextureFilter
|
external interface TextureFilter
|
||||||
external object NearestFilter : TextureFilter
|
external object NearestFilter : TextureFilter
|
||||||
external object NearestMipmapNearestFilter : TextureFilter
|
external object NearestMipmapNearestFilter : TextureFilter
|
||||||
external object NearestMipMapNearestFilter : TextureFilter
|
|
||||||
external object NearestMipmapLinearFilter : TextureFilter
|
external object NearestMipmapLinearFilter : TextureFilter
|
||||||
external object NearestMipMapLinearFilter : TextureFilter
|
|
||||||
external object LinearFilter : TextureFilter
|
external object LinearFilter : TextureFilter
|
||||||
external object LinearMipmapNearestFilter : TextureFilter
|
external object LinearMipmapNearestFilter : TextureFilter
|
||||||
external object LinearMipMapNearestFilter : TextureFilter
|
|
||||||
external object LinearMipmapLinearFilter : TextureFilter
|
external object LinearMipmapLinearFilter : TextureFilter
|
||||||
external object LinearMipMapLinearFilter : TextureFilter
|
|
||||||
|
|
||||||
external interface TextureDataType
|
external interface TextureDataType
|
||||||
external object UnsignedByteType : TextureDataType
|
external object UnsignedByteType : TextureDataType
|
||||||
@ -764,7 +766,7 @@ external object RGBM16Encoding : TextureEncoding
|
|||||||
external object RGBDEncoding : TextureEncoding
|
external object RGBDEncoding : TextureEncoding
|
||||||
|
|
||||||
external class CompressedTexture(
|
external class CompressedTexture(
|
||||||
mipmaps: Array<dynamic>, /* Should have data, height and width. */
|
mipmaps: Array<Mipmap>,
|
||||||
width: Int,
|
width: Int,
|
||||||
height: Int,
|
height: Int,
|
||||||
format: CompressedPixelFormat = definedExternally,
|
format: CompressedPixelFormat = definedExternally,
|
||||||
@ -774,10 +776,16 @@ external class CompressedTexture(
|
|||||||
wrapT: Wrapping = definedExternally,
|
wrapT: Wrapping = definedExternally,
|
||||||
magFilter: TextureFilter = definedExternally,
|
magFilter: TextureFilter = definedExternally,
|
||||||
minFilter: TextureFilter = definedExternally,
|
minFilter: TextureFilter = definedExternally,
|
||||||
anisotropy: Double = definedExternally,
|
anisotropy: Int = definedExternally,
|
||||||
encoding: TextureEncoding = definedExternally,
|
encoding: TextureEncoding = definedExternally,
|
||||||
) : Texture
|
) : Texture
|
||||||
|
|
||||||
|
external interface Mipmap {
|
||||||
|
var data: Uint8Array
|
||||||
|
var width: Int
|
||||||
|
var height: Int
|
||||||
|
}
|
||||||
|
|
||||||
external enum class MOUSE {
|
external enum class MOUSE {
|
||||||
LEFT,
|
LEFT,
|
||||||
MIDDLE,
|
MIDDLE,
|
||||||
|
@ -174,6 +174,7 @@ class AreaAssetLoader(private val assetLoader: AssetLoader) : DisposableContaine
|
|||||||
val fix = MANUAL_FIXES[Pair(episode, areaVariant.area.id)]
|
val fix = MANUAL_FIXES[Pair(episode, areaVariant.area.id)]
|
||||||
val sections = mutableMapOf<Int, SectionModel>()
|
val sections = mutableMapOf<Int, SectionModel>()
|
||||||
|
|
||||||
|
// TODO: Pass anisotropy parameter.
|
||||||
val group =
|
val group =
|
||||||
renderGeometryToGroup(renderGeometry, textures) { renderSection, areaObj, mesh ->
|
renderGeometryToGroup(renderGeometry, textures) { renderSection, areaObj, mesh ->
|
||||||
if (fix != null) {
|
if (fix != null) {
|
||||||
|
@ -7,14 +7,22 @@ import world.phantasmal.core.Success
|
|||||||
import world.phantasmal.psolib.Endianness
|
import world.phantasmal.psolib.Endianness
|
||||||
import world.phantasmal.psolib.cursor.Cursor
|
import world.phantasmal.psolib.cursor.Cursor
|
||||||
import world.phantasmal.psolib.cursor.cursor
|
import world.phantasmal.psolib.cursor.cursor
|
||||||
import world.phantasmal.psolib.fileFormats.ninja.*
|
import world.phantasmal.psolib.fileFormats.ninja.NinjaObject
|
||||||
|
import world.phantasmal.psolib.fileFormats.ninja.XvrTexture
|
||||||
|
import world.phantasmal.psolib.fileFormats.ninja.parseNj
|
||||||
|
import world.phantasmal.psolib.fileFormats.ninja.parseXj
|
||||||
|
import world.phantasmal.psolib.fileFormats.ninja.parseXvm
|
||||||
import world.phantasmal.psolib.fileFormats.quest.EntityType
|
import world.phantasmal.psolib.fileFormats.quest.EntityType
|
||||||
import world.phantasmal.psolib.fileFormats.quest.NpcType
|
import world.phantasmal.psolib.fileFormats.quest.NpcType
|
||||||
import world.phantasmal.psolib.fileFormats.quest.ObjectType
|
import world.phantasmal.psolib.fileFormats.quest.ObjectType
|
||||||
import world.phantasmal.web.core.loading.AssetLoader
|
import world.phantasmal.web.core.loading.AssetLoader
|
||||||
import world.phantasmal.web.core.rendering.conversion.ninjaObjectToInstancedMesh
|
import world.phantasmal.web.core.rendering.conversion.ninjaObjectToInstancedMesh
|
||||||
import world.phantasmal.web.core.rendering.disposeObject3DResources
|
import world.phantasmal.web.core.rendering.disposeObject3DResources
|
||||||
import world.phantasmal.web.externals.three.*
|
import world.phantasmal.web.externals.three.Color
|
||||||
|
import world.phantasmal.web.externals.three.CylinderGeometry
|
||||||
|
import world.phantasmal.web.externals.three.DoubleSide
|
||||||
|
import world.phantasmal.web.externals.three.InstancedMesh
|
||||||
|
import world.phantasmal.web.externals.three.MeshLambertMaterial
|
||||||
import world.phantasmal.webui.DisposableContainer
|
import world.phantasmal.webui.DisposableContainer
|
||||||
import world.phantasmal.webui.obj
|
import world.phantasmal.webui.obj
|
||||||
|
|
||||||
@ -56,6 +64,7 @@ class EntityAssetLoader(private val assetLoader: AssetLoader) : DisposableContai
|
|||||||
|
|
||||||
val textures = loadTextures(type, model)
|
val textures = loadTextures(type, model)
|
||||||
|
|
||||||
|
// TODO: Pass anisotropy parameter.
|
||||||
return ninjaObjectToInstancedMesh(
|
return ninjaObjectToInstancedMesh(
|
||||||
ninjaObject,
|
ninjaObject,
|
||||||
textures,
|
textures,
|
||||||
@ -226,6 +235,7 @@ private fun entityTypeToGeometryFormat(type: EntityType): GeomFormat =
|
|||||||
else -> GeomFormat.Nj
|
else -> GeomFormat.Nj
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
is ObjectType -> {
|
is ObjectType -> {
|
||||||
when (type) {
|
when (type) {
|
||||||
ObjectType.EasterEgg,
|
ObjectType.EasterEgg,
|
||||||
@ -249,6 +259,7 @@ private fun entityTypeToGeometryFormat(type: EntityType): GeomFormat =
|
|||||||
else -> GeomFormat.Xj
|
else -> GeomFormat.Xj
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
else -> {
|
else -> {
|
||||||
error("$type not supported.")
|
error("$type not supported.")
|
||||||
}
|
}
|
||||||
@ -272,6 +283,7 @@ private fun entityTypeToPath(
|
|||||||
GeomFormat.Nj -> "nj"
|
GeomFormat.Nj -> "nj"
|
||||||
GeomFormat.Xj -> "xj"
|
GeomFormat.Xj -> "xj"
|
||||||
}
|
}
|
||||||
|
|
||||||
AssetType.Texture -> "xvm"
|
AssetType.Texture -> "xvm"
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -296,26 +308,37 @@ private fun entityTypeToPath(
|
|||||||
|
|
||||||
NpcType.Hildebear2 ->
|
NpcType.Hildebear2 ->
|
||||||
entityTypeToPath(NpcType.Hildebear, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.Hildebear, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.Hildeblue2 ->
|
NpcType.Hildeblue2 ->
|
||||||
entityTypeToPath(NpcType.Hildeblue, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.Hildeblue, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.RagRappy2 ->
|
NpcType.RagRappy2 ->
|
||||||
entityTypeToPath(NpcType.RagRappy, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.RagRappy, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.Monest2 ->
|
NpcType.Monest2 ->
|
||||||
entityTypeToPath(NpcType.Monest, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.Monest, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.Mothmant2 ->
|
NpcType.Mothmant2 ->
|
||||||
entityTypeToPath(NpcType.Mothmant, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.Mothmant, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.PoisonLily2 ->
|
NpcType.PoisonLily2 ->
|
||||||
entityTypeToPath(NpcType.PoisonLily, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.PoisonLily, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.NarLily2 ->
|
NpcType.NarLily2 ->
|
||||||
entityTypeToPath(NpcType.NarLily, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.NarLily, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.GrassAssassin2 ->
|
NpcType.GrassAssassin2 ->
|
||||||
entityTypeToPath(NpcType.GrassAssassin, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.GrassAssassin, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.Dimenian2 ->
|
NpcType.Dimenian2 ->
|
||||||
entityTypeToPath(NpcType.Dimenian, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.Dimenian, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.LaDimenian2 ->
|
NpcType.LaDimenian2 ->
|
||||||
entityTypeToPath(NpcType.LaDimenian, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.LaDimenian, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.SoDimenian2 ->
|
NpcType.SoDimenian2 ->
|
||||||
entityTypeToPath(NpcType.SoDimenian, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.SoDimenian, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.DarkBelra2 ->
|
NpcType.DarkBelra2 ->
|
||||||
entityTypeToPath(NpcType.DarkBelra, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.DarkBelra, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
@ -323,26 +346,35 @@ private fun entityTypeToPath(
|
|||||||
|
|
||||||
NpcType.SavageWolf2 ->
|
NpcType.SavageWolf2 ->
|
||||||
entityTypeToPath(NpcType.SavageWolf, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.SavageWolf, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.BarbarousWolf2 ->
|
NpcType.BarbarousWolf2 ->
|
||||||
entityTypeToPath(NpcType.BarbarousWolf, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.BarbarousWolf, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.PanArms2 ->
|
NpcType.PanArms2 ->
|
||||||
entityTypeToPath(NpcType.PanArms, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.PanArms, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.Dubchic2 ->
|
NpcType.Dubchic2 ->
|
||||||
entityTypeToPath(NpcType.Dubchic, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.Dubchic, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.Gilchic2 ->
|
NpcType.Gilchic2 ->
|
||||||
entityTypeToPath(NpcType.Gilchic, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.Gilchic, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.Garanz2 ->
|
NpcType.Garanz2 ->
|
||||||
entityTypeToPath(NpcType.Garanz, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.Garanz, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.Dubswitch2 ->
|
NpcType.Dubswitch2 ->
|
||||||
entityTypeToPath(NpcType.Dubswitch, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.Dubswitch, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.Delsaber2 ->
|
NpcType.Delsaber2 ->
|
||||||
entityTypeToPath(NpcType.Delsaber, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.Delsaber, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
NpcType.ChaosSorcerer2 ->
|
NpcType.ChaosSorcerer2 ->
|
||||||
entityTypeToPath(NpcType.ChaosSorcerer, assetType, suffix, model, geomFormat)
|
entityTypeToPath(NpcType.ChaosSorcerer, assetType, suffix, model, geomFormat)
|
||||||
|
|
||||||
else -> "/npcs/${type.name}${fullSuffix}.$extension"
|
else -> "/npcs/${type.name}${fullSuffix}.$extension"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
is ObjectType -> {
|
is ObjectType -> {
|
||||||
when (type) {
|
when (type) {
|
||||||
// We don't have a model for these objects.
|
// We don't have a model for these objects.
|
||||||
@ -431,6 +463,7 @@ private fun entityTypeToPath(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
else -> {
|
else -> {
|
||||||
error("$type not supported.")
|
error("$type not supported.")
|
||||||
}
|
}
|
||||||
|
@ -8,11 +8,27 @@ import world.phantasmal.psolib.fileFormats.ninja.NjMotion
|
|||||||
import world.phantasmal.psolib.fileFormats.ninja.NjObject
|
import world.phantasmal.psolib.fileFormats.ninja.NjObject
|
||||||
import world.phantasmal.web.core.boundingSphere
|
import world.phantasmal.web.core.boundingSphere
|
||||||
import world.phantasmal.web.core.isSkinnedMesh
|
import world.phantasmal.web.core.isSkinnedMesh
|
||||||
import world.phantasmal.web.core.rendering.*
|
import world.phantasmal.web.core.rendering.DisposableThreeRenderer
|
||||||
|
import world.phantasmal.web.core.rendering.OrbitalCameraInputManager
|
||||||
|
import world.phantasmal.web.core.rendering.RenderContext
|
||||||
import world.phantasmal.web.core.rendering.Renderer
|
import world.phantasmal.web.core.rendering.Renderer
|
||||||
import world.phantasmal.web.core.rendering.conversion.*
|
import world.phantasmal.web.core.rendering.conversion.PSO_FRAME_RATE_DOUBLE
|
||||||
|
import world.phantasmal.web.core.rendering.conversion.collisionGeometryToGroup
|
||||||
|
import world.phantasmal.web.core.rendering.conversion.createAnimationClip
|
||||||
|
import world.phantasmal.web.core.rendering.conversion.ninjaObjectToMesh
|
||||||
|
import world.phantasmal.web.core.rendering.conversion.ninjaObjectToSkinnedMesh
|
||||||
|
import world.phantasmal.web.core.rendering.conversion.renderGeometryToGroup
|
||||||
|
import world.phantasmal.web.core.rendering.disposeObject3DResources
|
||||||
import world.phantasmal.web.core.times
|
import world.phantasmal.web.core.times
|
||||||
import world.phantasmal.web.externals.three.*
|
import world.phantasmal.web.externals.three.AnimationAction
|
||||||
|
import world.phantasmal.web.externals.three.AnimationClip
|
||||||
|
import world.phantasmal.web.externals.three.AnimationMixer
|
||||||
|
import world.phantasmal.web.externals.three.Clock
|
||||||
|
import world.phantasmal.web.externals.three.LineBasicMaterial
|
||||||
|
import world.phantasmal.web.externals.three.Object3D
|
||||||
|
import world.phantasmal.web.externals.three.PerspectiveCamera
|
||||||
|
import world.phantasmal.web.externals.three.SkeletonHelper
|
||||||
|
import world.phantasmal.web.externals.three.Vector3
|
||||||
import world.phantasmal.web.shared.Throttle
|
import world.phantasmal.web.shared.Throttle
|
||||||
import world.phantasmal.web.viewer.stores.NinjaGeometry
|
import world.phantasmal.web.viewer.stores.NinjaGeometry
|
||||||
import world.phantasmal.web.viewer.stores.ViewerStore
|
import world.phantasmal.web.viewer.stores.ViewerStore
|
||||||
@ -125,15 +141,26 @@ class MeshRenderer(
|
|||||||
val obj = ninjaGeometry.obj
|
val obj = ninjaGeometry.obj
|
||||||
|
|
||||||
if (obj is NjObject) {
|
if (obj is NjObject) {
|
||||||
ninjaObjectToSkinnedMesh(obj, textures, boundingVolumes = true)
|
ninjaObjectToSkinnedMesh(
|
||||||
|
obj,
|
||||||
|
textures,
|
||||||
|
boundingVolumes = true,
|
||||||
|
anisotropy = threeRenderer.capabilities.getMaxAnisotropy() / 2,
|
||||||
|
)
|
||||||
} else {
|
} else {
|
||||||
ninjaObjectToMesh(obj, textures, boundingVolumes = true)
|
ninjaObjectToMesh(
|
||||||
|
obj,
|
||||||
|
textures,
|
||||||
|
boundingVolumes = true,
|
||||||
|
anisotropy = threeRenderer.capabilities.getMaxAnisotropy() / 2,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
is NinjaGeometry.Render -> renderGeometryToGroup(
|
is NinjaGeometry.Render -> renderGeometryToGroup(
|
||||||
ninjaGeometry.geometry,
|
ninjaGeometry.geometry,
|
||||||
textures
|
textures,
|
||||||
|
anisotropy = threeRenderer.capabilities.getMaxAnisotropy() / 2,
|
||||||
)
|
)
|
||||||
|
|
||||||
is NinjaGeometry.Collision -> collisionGeometryToGroup(ninjaGeometry.geometry)
|
is NinjaGeometry.Collision -> collisionGeometryToGroup(ninjaGeometry.geometry)
|
||||||
|
@ -5,10 +5,21 @@ import org.khronos.webgl.Float32Array
|
|||||||
import org.khronos.webgl.Uint16Array
|
import org.khronos.webgl.Uint16Array
|
||||||
import org.w3c.dom.HTMLCanvasElement
|
import org.w3c.dom.HTMLCanvasElement
|
||||||
import world.phantasmal.psolib.fileFormats.ninja.XvrTexture
|
import world.phantasmal.psolib.fileFormats.ninja.XvrTexture
|
||||||
import world.phantasmal.web.core.rendering.*
|
import world.phantasmal.web.core.rendering.DisposableThreeRenderer
|
||||||
|
import world.phantasmal.web.core.rendering.OrbitalCameraInputManager
|
||||||
|
import world.phantasmal.web.core.rendering.RenderContext
|
||||||
import world.phantasmal.web.core.rendering.Renderer
|
import world.phantasmal.web.core.rendering.Renderer
|
||||||
import world.phantasmal.web.core.rendering.conversion.xvrTextureToThree
|
import world.phantasmal.web.core.rendering.conversion.xvrTextureToThree
|
||||||
import world.phantasmal.web.externals.three.*
|
import world.phantasmal.web.core.rendering.disposeObject3DResources
|
||||||
|
import world.phantasmal.web.externals.three.BufferGeometry
|
||||||
|
import world.phantasmal.web.externals.three.Color
|
||||||
|
import world.phantasmal.web.externals.three.Float32BufferAttribute
|
||||||
|
import world.phantasmal.web.externals.three.Mesh
|
||||||
|
import world.phantasmal.web.externals.three.MeshBasicMaterial
|
||||||
|
import world.phantasmal.web.externals.three.NearestFilter
|
||||||
|
import world.phantasmal.web.externals.three.OrthographicCamera
|
||||||
|
import world.phantasmal.web.externals.three.Uint16BufferAttribute
|
||||||
|
import world.phantasmal.web.externals.three.Vector3
|
||||||
import world.phantasmal.web.viewer.stores.ViewerStore
|
import world.phantasmal.web.viewer.stores.ViewerStore
|
||||||
import world.phantasmal.webui.obj
|
import world.phantasmal.webui.obj
|
||||||
import kotlin.math.ceil
|
import kotlin.math.ceil
|
||||||
@ -23,27 +34,31 @@ class TextureRenderer(
|
|||||||
) : Renderer() {
|
) : Renderer() {
|
||||||
private var meshes = listOf<Mesh>()
|
private var meshes = listOf<Mesh>()
|
||||||
|
|
||||||
override val context = addDisposable(RenderContext(
|
override val context = addDisposable(
|
||||||
createCanvas(),
|
RenderContext(
|
||||||
OrthographicCamera(
|
createCanvas(),
|
||||||
left = -400.0,
|
OrthographicCamera(
|
||||||
right = 400.0,
|
left = -400.0,
|
||||||
top = 300.0,
|
right = 400.0,
|
||||||
bottom = -300.0,
|
top = 300.0,
|
||||||
near = 1.0,
|
bottom = -300.0,
|
||||||
far = 10.0,
|
near = 1.0,
|
||||||
|
far = 10.0,
|
||||||
|
)
|
||||||
)
|
)
|
||||||
))
|
)
|
||||||
|
|
||||||
override val threeRenderer = addDisposable(createThreeRenderer(context.canvas)).renderer
|
override val threeRenderer = addDisposable(createThreeRenderer(context.canvas)).renderer
|
||||||
|
|
||||||
override val inputManager = addDisposable(OrbitalCameraInputManager(
|
override val inputManager = addDisposable(
|
||||||
context.canvas,
|
OrbitalCameraInputManager(
|
||||||
context.camera,
|
context.canvas,
|
||||||
Vector3(0.0, 0.0, 5.0),
|
context.camera,
|
||||||
screenSpacePanning = true,
|
Vector3(0.0, 0.0, 5.0),
|
||||||
enableRotate = false,
|
screenSpacePanning = true,
|
||||||
))
|
enableRotate = false,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
init {
|
init {
|
||||||
observeNow(store.currentTextures) {
|
observeNow(store.currentTextures) {
|
||||||
@ -81,7 +96,7 @@ class TextureRenderer(
|
|||||||
meshes = textures.map { xvr ->
|
meshes = textures.map { xvr ->
|
||||||
val texture =
|
val texture =
|
||||||
try {
|
try {
|
||||||
xvrTextureToThree(xvr, filter = NearestFilter)
|
xvrTextureToThree(xvr, magFilter = NearestFilter, minFilter = NearestFilter)
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
logger.error(e) { "Couldn't convert XVR texture." }
|
logger.error(e) { "Couldn't convert XVR texture." }
|
||||||
null
|
null
|
||||||
@ -120,46 +135,56 @@ class TextureRenderer(
|
|||||||
geom.setAttribute(
|
geom.setAttribute(
|
||||||
"position",
|
"position",
|
||||||
Float32BufferAttribute(
|
Float32BufferAttribute(
|
||||||
Float32Array(arrayOf(
|
Float32Array(
|
||||||
-halfWidth, -halfHeight, 0f,
|
arrayOf(
|
||||||
-halfWidth, halfHeight, 0f,
|
-halfWidth, -halfHeight, 0f,
|
||||||
halfWidth, halfHeight, 0f,
|
-halfWidth, halfHeight, 0f,
|
||||||
halfWidth, -halfHeight, 0f,
|
halfWidth, halfHeight, 0f,
|
||||||
)),
|
halfWidth, -halfHeight, 0f,
|
||||||
|
)
|
||||||
|
),
|
||||||
3,
|
3,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
geom.setAttribute(
|
geom.setAttribute(
|
||||||
"normal",
|
"normal",
|
||||||
Float32BufferAttribute(
|
Float32BufferAttribute(
|
||||||
Float32Array(arrayOf(
|
Float32Array(
|
||||||
0f, 0f, 1f,
|
arrayOf(
|
||||||
0f, 0f, 1f,
|
0f, 0f, 1f,
|
||||||
0f, 0f, 1f,
|
0f, 0f, 1f,
|
||||||
0f, 0f, 1f,
|
0f, 0f, 1f,
|
||||||
)),
|
0f, 0f, 1f,
|
||||||
|
)
|
||||||
|
),
|
||||||
3,
|
3,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
geom.setAttribute(
|
geom.setAttribute(
|
||||||
"uv",
|
"uv",
|
||||||
Float32BufferAttribute(
|
Float32BufferAttribute(
|
||||||
Float32Array(arrayOf(
|
Float32Array(
|
||||||
0f, 1f,
|
arrayOf(
|
||||||
0f, 0f,
|
0f, 1f,
|
||||||
1f, 0f,
|
0f, 0f,
|
||||||
1f, 1f,
|
1f, 0f,
|
||||||
)),
|
1f, 1f,
|
||||||
|
)
|
||||||
|
),
|
||||||
2,
|
2,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
geom.setIndex(Uint16BufferAttribute(
|
geom.setIndex(
|
||||||
Uint16Array(arrayOf(
|
Uint16BufferAttribute(
|
||||||
0, 2, 1,
|
Uint16Array(
|
||||||
2, 0, 3,
|
arrayOf(
|
||||||
)),
|
0, 2, 1,
|
||||||
1,
|
2, 0, 3,
|
||||||
))
|
)
|
||||||
|
),
|
||||||
|
1,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
geom.translate(x.toDouble(), y.toDouble(), -5.0)
|
geom.translate(x.toDouble(), y.toDouble(), -5.0)
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user