2019-11-11 10:03:52 +00:00
|
|
|
""" Contains functions for saving a Scene to a .msh file. """
|
|
|
|
|
|
|
|
from itertools import islice
|
2019-11-15 03:28:09 +00:00
|
|
|
from typing import Dict
|
2019-11-11 10:03:52 +00:00
|
|
|
from .msh_scene import Scene, create_scene_aabb
|
|
|
|
from .msh_model import *
|
2019-11-15 03:28:09 +00:00
|
|
|
from .msh_material import *
|
2019-11-11 10:03:52 +00:00
|
|
|
from .msh_writer import Writer
|
|
|
|
from .msh_utilities import *
|
|
|
|
|
2020-10-11 16:11:15 +00:00
|
|
|
from .crc import *
|
|
|
|
|
2020-10-20 03:18:08 +00:00
|
|
|
def save_scene(output_file, scene: Scene, is_animated: bool):
|
2019-11-11 10:03:52 +00:00
|
|
|
""" Saves scene to the supplied file. """
|
|
|
|
|
|
|
|
with Writer(file=output_file, chunk_id="HEDR") as hedr:
|
|
|
|
with hedr.create_child("MSH2") as msh2:
|
|
|
|
|
|
|
|
with msh2.create_child("SINF") as sinf:
|
|
|
|
_write_sinf(sinf, scene)
|
|
|
|
|
2020-10-16 00:40:27 +00:00
|
|
|
model_index: Dict[str, int] = {model.name:i for i, model in enumerate(scene.models)}
|
2019-11-15 03:28:09 +00:00
|
|
|
material_index: Dict[str, int] = {}
|
|
|
|
|
2019-11-11 10:03:52 +00:00
|
|
|
with msh2.create_child("MATL") as matl:
|
2019-11-15 03:28:09 +00:00
|
|
|
material_index = _write_matl_and_get_material_index(matl, scene)
|
2019-11-11 10:03:52 +00:00
|
|
|
|
|
|
|
for index, model in enumerate(scene.models):
|
2020-10-20 03:18:08 +00:00
|
|
|
|
|
|
|
print(model.name)
|
|
|
|
|
2019-11-11 10:03:52 +00:00
|
|
|
with msh2.create_child("MODL") as modl:
|
2020-10-16 00:40:27 +00:00
|
|
|
_write_modl(modl, model, index, material_index, model_index)
|
2019-11-11 10:03:52 +00:00
|
|
|
|
2020-10-20 03:18:08 +00:00
|
|
|
if is_animated:
|
|
|
|
with hedr.create_child("SKL2") as skl2:
|
|
|
|
_write_skl2(skl2, scene)
|
|
|
|
|
|
|
|
with hedr.create_child("BLN2") as bln2:
|
|
|
|
_write_bln2(bln2, scene)
|
|
|
|
|
2020-10-16 18:56:03 +00:00
|
|
|
with hedr.create_child("ANM2") as anm2: #simple for now
|
|
|
|
for anim in scene.anims:
|
|
|
|
_write_anm2(anm2, anim)
|
2020-10-11 16:11:15 +00:00
|
|
|
|
2019-11-11 10:03:52 +00:00
|
|
|
with hedr.create_child("CL1L"):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def _write_sinf(sinf: Writer, scene: Scene):
|
|
|
|
with sinf.create_child("NAME") as name:
|
|
|
|
name.write_string(scene.name)
|
|
|
|
|
|
|
|
with sinf.create_child("FRAM") as fram:
|
2020-10-20 03:18:08 +00:00
|
|
|
min_index = 0
|
|
|
|
max_index = 1
|
|
|
|
|
|
|
|
if scene.anims and len(scene.anims) > 0:
|
|
|
|
min_index = min([anim.start_index for anim in scene.anims])
|
|
|
|
max_index = min([anim.end_index for anim in scene.anims])
|
|
|
|
|
|
|
|
fram.write_i32(min_index, max_index)
|
|
|
|
fram.write_f32(10.0)
|
2019-11-11 10:03:52 +00:00
|
|
|
|
|
|
|
with sinf.create_child("BBOX") as bbox:
|
|
|
|
aabb = create_scene_aabb(scene)
|
|
|
|
|
|
|
|
bbox_position = div_vec(add_vec(aabb.min_, aabb.max_), Vector((2.0, 2.0, 2.0)))
|
|
|
|
bbox_size = div_vec(sub_vec(aabb.max_, aabb.min_), Vector((2.0, 2.0, 2.0)))
|
|
|
|
bbox_length = bbox_size.length
|
|
|
|
|
|
|
|
bbox.write_f32(0.0, 0.0, 0.0, 1.0)
|
|
|
|
bbox.write_f32(bbox_position.x, bbox_position.y, bbox_position.z)
|
|
|
|
bbox.write_f32(bbox_size.x, bbox_size.y, bbox_size.z, bbox_length)
|
|
|
|
|
2019-11-15 03:28:09 +00:00
|
|
|
def _write_matl_and_get_material_index(matl: Writer, scene: Scene):
|
|
|
|
material_index: Dict[str, int] = {}
|
|
|
|
|
|
|
|
if len(scene.materials) > 0:
|
|
|
|
matl.write_u32(len(scene.materials)) # Material count.
|
|
|
|
|
|
|
|
for index, name_material in enumerate(scene.materials.items()):
|
|
|
|
with matl.create_child("MATD") as matd:
|
|
|
|
material_index[name_material[0]] = index
|
|
|
|
_write_matd(matd, name_material[0], name_material[1])
|
|
|
|
else:
|
|
|
|
matl.write_u32(1) # Material count.
|
|
|
|
|
|
|
|
default_material_name = f"{scene.name}Material"
|
|
|
|
material_index[default_material_name] = 0
|
|
|
|
|
|
|
|
with matl.create_child("MATD") as matd:
|
|
|
|
_write_matd(matd, default_material_name, Material())
|
|
|
|
|
|
|
|
return material_index
|
|
|
|
|
|
|
|
def _write_matd(matd: Writer, material_name: str, material: Material):
|
|
|
|
with matd.create_child("NAME") as name:
|
|
|
|
name.write_string(material_name)
|
|
|
|
with matd.create_child("DATA") as data:
|
|
|
|
data.write_f32(1.0, 1.0, 1.0, 1.0) # Diffuse Color (Seams to get ignored by modelmunge)
|
|
|
|
data.write_f32(material.specular_color[0], material.specular_color[1],
|
|
|
|
material.specular_color[2], 1.0)
|
|
|
|
data.write_f32(0.0, 0.0, 0.0, 1.0) # Ambient Color (Seams to get ignored by modelmunge and Zero(?))
|
|
|
|
data.write_f32(50.0) # Specular Exponent/Decay (Gets ignored by RedEngine in SWBFII for all known materials)
|
|
|
|
with matd.create_child("ATRB") as atrb:
|
|
|
|
atrb.write_u8(material.flags.value)
|
|
|
|
atrb.write_u8(material.rendertype.value)
|
|
|
|
atrb.write_u8(material.data[0], material.data[1])
|
|
|
|
|
|
|
|
with matd.create_child("TX0D") as tx0d:
|
|
|
|
tx0d.write_string(material.texture0)
|
|
|
|
if material.texture1 or material.texture2 or material.texture3:
|
|
|
|
with matd.create_child("TX1D") as tx1d:
|
|
|
|
tx1d.write_string(material.texture1)
|
|
|
|
|
|
|
|
if material.texture2 or material.texture3:
|
|
|
|
with matd.create_child("TX2D") as tx2d:
|
|
|
|
tx2d.write_string(material.texture2)
|
|
|
|
|
|
|
|
if material.texture3:
|
|
|
|
with matd.create_child("TX3D") as tx3d:
|
|
|
|
tx3d.write_string(material.texture3)
|
|
|
|
|
2020-10-16 00:40:27 +00:00
|
|
|
def _write_modl(modl: Writer, model: Model, index: int, material_index: Dict[str, int], model_index: Dict[str, int]):
|
2019-11-11 10:03:52 +00:00
|
|
|
with modl.create_child("MTYP") as mtyp:
|
|
|
|
mtyp.write_u32(model.model_type.value)
|
|
|
|
|
|
|
|
with modl.create_child("MNDX") as mndx:
|
|
|
|
mndx.write_u32(index)
|
|
|
|
|
|
|
|
with modl.create_child("NAME") as name:
|
|
|
|
name.write_string(model.name)
|
|
|
|
|
|
|
|
if model.parent:
|
|
|
|
with modl.create_child("PRNT") as prnt:
|
|
|
|
prnt.write_string(model.parent)
|
|
|
|
|
|
|
|
if model.hidden:
|
|
|
|
with modl.create_child("FLGS") as flgs:
|
|
|
|
flgs.write_u32(1)
|
|
|
|
|
|
|
|
with modl.create_child("TRAN") as tran:
|
|
|
|
_write_tran(tran, model.transform)
|
|
|
|
|
|
|
|
if model.geometry is not None:
|
|
|
|
with modl.create_child("GEOM") as geom:
|
2020-10-20 03:18:08 +00:00
|
|
|
|
|
|
|
with geom.create_child("BBOX") as bbox:
|
|
|
|
bbox.write_f32(0.0, 0.0, 0.0, 1.0)
|
|
|
|
bbox.write_f32(0, 0, 0)
|
|
|
|
bbox.write_f32(1.0,1.0,1.0,2.0)
|
|
|
|
|
2019-11-11 10:03:52 +00:00
|
|
|
for segment in model.geometry:
|
|
|
|
with geom.create_child("SEGM") as segm:
|
2019-11-15 03:28:09 +00:00
|
|
|
_write_segm(segm, segment, material_index)
|
2019-11-11 10:03:52 +00:00
|
|
|
|
2020-10-16 00:40:27 +00:00
|
|
|
if model.bone_map:
|
|
|
|
with geom.create_child("ENVL") as envl:
|
|
|
|
_write_envl(envl, model, model_index)
|
2019-11-11 10:03:52 +00:00
|
|
|
|
2019-11-15 09:40:47 +00:00
|
|
|
if model.collisionprimitive is not None:
|
|
|
|
with modl.create_child("SWCI") as swci:
|
|
|
|
swci.write_u32(model.collisionprimitive.shape.value)
|
|
|
|
swci.write_f32(model.collisionprimitive.radius)
|
|
|
|
swci.write_f32(model.collisionprimitive.height)
|
|
|
|
swci.write_f32(model.collisionprimitive.length)
|
2019-11-11 10:03:52 +00:00
|
|
|
|
|
|
|
def _write_tran(tran: Writer, transform: ModelTransform):
|
|
|
|
tran.write_f32(1.0, 1.0, 1.0) # Scale, ignored by modelmunge
|
|
|
|
tran.write_f32(transform.rotation.x, transform.rotation.y, transform.rotation.z, transform.rotation.w)
|
|
|
|
tran.write_f32(transform.translation.x, transform.translation.y, transform.translation.z)
|
|
|
|
|
2019-11-15 03:28:09 +00:00
|
|
|
def _write_segm(segm: Writer, segment: GeometrySegment, material_index: Dict[str, int]):
|
2019-11-11 10:03:52 +00:00
|
|
|
|
|
|
|
with segm.create_child("MATI") as mati:
|
2019-11-15 03:28:09 +00:00
|
|
|
mati.write_u32(material_index.get(segment.material_name, 0))
|
2019-11-11 10:03:52 +00:00
|
|
|
|
|
|
|
with segm.create_child("POSL") as posl:
|
|
|
|
posl.write_u32(len(segment.positions))
|
|
|
|
|
|
|
|
for position in segment.positions:
|
|
|
|
posl.write_f32(position.x, position.y, position.z)
|
|
|
|
|
2020-10-16 00:40:27 +00:00
|
|
|
if segment.weights:
|
|
|
|
with segm.create_child("WGHT") as wght:
|
|
|
|
_write_wght(wght, segment.weights)
|
|
|
|
|
2019-11-11 10:03:52 +00:00
|
|
|
with segm.create_child("NRML") as nrml:
|
|
|
|
nrml.write_u32(len(segment.normals))
|
|
|
|
|
2020-10-20 03:18:08 +00:00
|
|
|
for i,normal in enumerate(segment.normals):
|
2019-11-11 10:03:52 +00:00
|
|
|
nrml.write_f32(normal.x, normal.y, normal.z)
|
|
|
|
|
|
|
|
if segment.colors is not None:
|
|
|
|
with segm.create_child("CLRL") as clrl:
|
|
|
|
clrl.write_u32(len(segment.colors))
|
|
|
|
|
|
|
|
for color in segment.colors:
|
|
|
|
clrl.write_u32(pack_color(color))
|
|
|
|
|
|
|
|
with segm.create_child("UV0L") as uv0l:
|
|
|
|
uv0l.write_u32(len(segment.texcoords))
|
|
|
|
|
|
|
|
for texcoord in segment.texcoords:
|
|
|
|
uv0l.write_f32(texcoord.x, texcoord.y)
|
|
|
|
|
|
|
|
with segm.create_child("NDXL") as ndxl:
|
|
|
|
ndxl.write_u32(len(segment.polygons))
|
|
|
|
|
|
|
|
for polygon in segment.polygons:
|
|
|
|
ndxl.write_u16(len(polygon))
|
|
|
|
|
|
|
|
for index in polygon:
|
|
|
|
ndxl.write_u16(index)
|
|
|
|
|
|
|
|
with segm.create_child("NDXT") as ndxt:
|
|
|
|
ndxt.write_u32(len(segment.triangles))
|
|
|
|
|
|
|
|
for triangle in segment.triangles:
|
|
|
|
ndxt.write_u16(triangle[0], triangle[1], triangle[2])
|
|
|
|
|
|
|
|
with segm.create_child("STRP") as strp:
|
|
|
|
strp.write_u32(sum(len(strip) for strip in segment.triangle_strips))
|
|
|
|
|
|
|
|
for strip in segment.triangle_strips:
|
|
|
|
strp.write_u16(strip[0] | 0x8000, strip[1] | 0x8000)
|
|
|
|
|
|
|
|
for index in islice(strip, 2, len(strip)):
|
|
|
|
strp.write_u16(index)
|
2020-10-11 16:11:15 +00:00
|
|
|
|
2020-10-16 18:56:03 +00:00
|
|
|
'''
|
|
|
|
SKINNING CHUNKS
|
|
|
|
'''
|
|
|
|
def _write_wght(wght: Writer, weights: List[List[VertexWeight]]):
|
|
|
|
wght.write_u32(len(weights))
|
|
|
|
|
2020-10-20 03:18:08 +00:00
|
|
|
print("Writing WGHT: ")
|
|
|
|
|
2020-10-16 18:56:03 +00:00
|
|
|
for weight_list in weights:
|
|
|
|
weight_list += [VertexWeight(0.0, 0)] * 4
|
|
|
|
weight_list = sorted(weight_list, key=lambda w: w.weight, reverse=True)
|
|
|
|
weight_list = weight_list[:4]
|
|
|
|
|
|
|
|
total_weight = max(sum(map(lambda w: w.weight, weight_list)), 1e-5)
|
|
|
|
|
2020-10-20 03:18:08 +00:00
|
|
|
print_str = ""
|
|
|
|
|
2020-10-16 18:56:03 +00:00
|
|
|
for weight in weight_list:
|
2020-10-20 03:18:08 +00:00
|
|
|
|
|
|
|
print_str += "({}, {}) ".format(weight.bone, weight.weight / total_weight)
|
|
|
|
|
2020-10-16 18:56:03 +00:00
|
|
|
wght.write_i32(weight.bone)
|
|
|
|
wght.write_f32(weight.weight / total_weight)
|
|
|
|
|
2020-10-20 03:18:08 +00:00
|
|
|
print(" {}".format(print_str))
|
|
|
|
|
2020-10-16 18:56:03 +00:00
|
|
|
def _write_envl(envl: Writer, model: Model, model_index: Dict[str, int]):
|
|
|
|
envl.write_u32(len(model.bone_map))
|
|
|
|
|
2020-10-20 03:18:08 +00:00
|
|
|
print("Writing ENVL: ")
|
|
|
|
|
2020-10-16 18:56:03 +00:00
|
|
|
for bone_name in model.bone_map:
|
2020-10-20 03:18:08 +00:00
|
|
|
print(" {:10} Index: {}".format(bone_name, model_index[bone_name]))
|
2020-10-16 18:56:03 +00:00
|
|
|
envl.write_u32(model_index[bone_name])
|
2020-10-11 16:11:15 +00:00
|
|
|
|
2020-10-16 18:56:03 +00:00
|
|
|
'''
|
2020-10-20 03:18:08 +00:00
|
|
|
SKELETON CHUNKS
|
|
|
|
'''
|
|
|
|
def _write_bln2(bln2: Writer, scene: Scene):
|
|
|
|
bones = scene.anims[0].bone_transforms.keys()
|
|
|
|
bln2.write_u32(len(bones))
|
|
|
|
|
|
|
|
for boneName in bones:
|
|
|
|
bln2.write_u32(crc(boneName), 0)
|
|
|
|
|
|
|
|
def _write_skl2(skl2: Writer, scene: Scene):
|
|
|
|
bones = scene.anims[0].bone_transforms.keys()
|
|
|
|
skl2.write_u32(len(bones))
|
|
|
|
|
|
|
|
for boneName in bones:
|
|
|
|
skl2.write_u32(crc(boneName), 0) #default values
|
|
|
|
skl2.write_f32(1.0, 0.0, 0.0) #from docs
|
|
|
|
|
|
|
|
'''
|
|
|
|
ANIMATION CHUNK
|
2020-10-16 18:56:03 +00:00
|
|
|
'''
|
2020-10-11 16:11:15 +00:00
|
|
|
def _write_anm2(anm2: Writer, anim: Animation):
|
|
|
|
|
|
|
|
with anm2.create_child("CYCL") as cycl:
|
|
|
|
|
|
|
|
cycl.write_u32(1)
|
|
|
|
cycl.write_string(anim.name)
|
|
|
|
|
2020-10-20 03:18:08 +00:00
|
|
|
for _ in range(63 - len(anim.name)):
|
2020-10-11 16:11:15 +00:00
|
|
|
cycl.write_u8(0)
|
|
|
|
|
2020-10-20 03:18:08 +00:00
|
|
|
cycl.write_f32(anim.framerate)
|
2020-10-11 16:11:15 +00:00
|
|
|
cycl.write_u32(0) #what does play style refer to?
|
2020-10-20 03:18:08 +00:00
|
|
|
cycl.write_u32(anim.start_index, anim.end_index) #first frame indices
|
2020-10-11 16:11:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
with anm2.create_child("KFR3") as kfr3:
|
|
|
|
|
|
|
|
kfr3.write_u32(len(anim.bone_transforms.keys()))
|
|
|
|
|
|
|
|
for boneName in anim.bone_transforms.keys():
|
|
|
|
kfr3.write_u32(crc(boneName))
|
|
|
|
kfr3.write_u32(0) #what is keyframe type?
|
|
|
|
|
2020-10-20 03:18:08 +00:00
|
|
|
num_frames = 1 + anim.end_index - anim.start_index
|
|
|
|
kfr3.write_u32(num_frames, num_frames) #basic testing
|
2020-10-11 16:11:15 +00:00
|
|
|
|
|
|
|
for i, xform in enumerate(anim.bone_transforms[boneName]):
|
|
|
|
kfr3.write_u32(i)
|
|
|
|
kfr3.write_f32(xform.translation.x, xform.translation.y, xform.translation.z)
|
|
|
|
|
|
|
|
for i, xform in enumerate(anim.bone_transforms[boneName]):
|
|
|
|
kfr3.write_u32(i)
|
|
|
|
kfr3.write_f32(xform.rotation.x, xform.rotation.y, xform.rotation.z, xform.rotation.w)
|
|
|
|
|
|
|
|
|
|
|
|
|