summaryrefslogtreecommitdiff
path: root/extractor
diff options
context:
space:
mode:
authorDaniel Weipert <git@mail.dweipert.de>2025-01-07 15:48:43 +0100
committerDaniel Weipert <git@mail.dweipert.de>2025-01-07 15:48:43 +0100
commitb75cc72c4e10bd652330b6d2bd99f3fd9129a3b3 (patch)
tree489cde2aa688273d232d66cd5d9e4251f2cea8c5 /extractor
parent35b0f811f23f029110373798b19d9d0895d907f0 (diff)
rsw with gnd step 1HEADmain
Diffstat (limited to 'extractor')
-rw-r--r--extractor/gnd_format.gd119
-rw-r--r--extractor/rsm_format.gd64
-rw-r--r--extractor/rsw_format.gd93
3 files changed, 260 insertions, 16 deletions
diff --git a/extractor/gnd_format.gd b/extractor/gnd_format.gd
index ea8d965..65635cf 100644
--- a/extractor/gnd_format.gd
+++ b/extractor/gnd_format.gd
@@ -1,6 +1,9 @@
class_name GNDFormat
+const MAP_TILE_SIZE := 10.0
+
+
## Byte Length: 4 [br]
## GRAT
var signature: String = "GRGN"
@@ -120,6 +123,67 @@ static func from_bytes(bytes: ByteStream) -> GNDFormat:
return gnd_format
+func get_cubes() -> Array:
+ var cubes := []
+ for cube in ground_mesh_cubes:
+ cubes.append({
+ "cube": cube,
+ Vector3(0, 1, 0): {
+ "surface": surfaces[cube.upwards_facing_surface_index],
+ "mesh": surfaces[cube.upwards_facing_surface_index].get_mesh({
+ Vector2(0, 0): Vector3(0, cube.top_left_height, 0),
+ Vector2(1, 0): Vector3(MAP_TILE_SIZE, cube.top_right_height, 0),
+ Vector2(1, 1): Vector3(MAP_TILE_SIZE, cube.bottom_right_height, MAP_TILE_SIZE),
+ Vector2(0, 1): Vector3(0, cube.bottom_left_height, MAP_TILE_SIZE),
+ }),
+ },
+ })
+
+ return cubes
+
+
+func convert(data_path: String) -> GridMap:
+ var grid_map := GridMap.new()
+ grid_map.cell_size = Vector3(GNDFormat.MAP_TILE_SIZE, GNDFormat.MAP_TILE_SIZE, GNDFormat.MAP_TILE_SIZE)
+ grid_map.cell_center_x = false
+ grid_map.cell_center_y = false
+ grid_map.cell_center_z = false
+
+ grid_map.position = Vector3(
+ (width * MAP_TILE_SIZE) / 2.0,
+ 0,
+ -(height * MAP_TILE_SIZE) / 2.0,
+ )
+
+ var library := MeshLibrary.new()
+ grid_map.mesh_library = library
+
+ var cubes := get_cubes()
+
+ for x in width:
+ for y in height:
+ var cube = cubes[x + y * width]
+ for surface_type in [Vector3(0, 1, 0)]:
+ var mesh: ArrayMesh = cube[surface_type].mesh
+ var material := StandardMaterial3D.new()
+ material.albedo_texture = load(
+ "%s/data/texture/%s" % [
+ data_path,
+ texture_paths[cube[surface_type].surface.texture_index]
+ ]
+ )
+
+ mesh.surface_set_material(0, material)
+
+ var id := library.get_last_unused_item_id()
+ library.create_item(id)
+ library.set_item_mesh(id, mesh)
+
+ grid_map.set_cell_item(Vector3(-x, 0, y), id)
+
+ return grid_map
+
+
class LightMapSlice:
## Byte Type: i32 [br]
## Byte Length: 4
@@ -229,6 +293,61 @@ class Surface:
)
+ func get_mesh(vertices: Dictionary[Vector2, Vector3]) -> ArrayMesh:
+ var surface_tool := SurfaceTool.new()
+ surface_tool.begin(Mesh.PRIMITIVE_TRIANGLES)
+
+ var uvs = get_uvs()
+ var color = get_vertex_color()
+
+ var tiles_per_surface := 2 * 2
+
+ surface_tool.add_triangle_fan(
+ PackedVector3Array([
+ vertices[Vector2(0, 0)],
+ vertices[Vector2(1, 0)],
+ vertices[Vector2(1, 1)],
+ ]),
+ PackedVector2Array([
+ uvs[Vector2(0, 0)] * tiles_per_surface,
+ uvs[Vector2(1, 0)] * tiles_per_surface,
+ uvs[Vector2(1, 1)] * tiles_per_surface,
+ ]),
+ PackedColorArray([
+ color, color, color,
+ ])
+ )
+ surface_tool.add_triangle_fan(
+ PackedVector3Array([
+ vertices[Vector2(0, 0)],
+ vertices[Vector2(1, 1)],
+ vertices[Vector2(0, 1)],
+ ]),
+ PackedVector2Array([
+ uvs[Vector2(0, 0)] * tiles_per_surface,
+ uvs[Vector2(1, 1)] * tiles_per_surface,
+ uvs[Vector2(0, 1)] * tiles_per_surface,
+ ]),
+ PackedColorArray([
+ color, color, color,
+ ])
+ )
+
+ return surface_tool.commit()
+
+
+ func get_uv_id() -> String:
+ # TODO: add more data?
+ return (
+ "%d %d %d %d %d %d %d %d" % [
+ u_top_left, v_top_left,
+ u_top_right, v_top_right,
+ u_bottom_right, v_bottom_right,
+ u_bottom_left, v_bottom_left
+ ]
+ ).md5_text()
+
+
static func from_bytes(bytes: ByteStream) -> Surface:
var surface = Surface.new()
diff --git a/extractor/rsm_format.gd b/extractor/rsm_format.gd
index cf83af9..2eccbd2 100644
--- a/extractor/rsm_format.gd
+++ b/extractor/rsm_format.gd
@@ -39,10 +39,12 @@ var texture_count: int
## Length: [member texture_count]
## Byte Length: 40
+## Versions: [<2.3]
var texture_names: Array[String]
## Byte Type: u8
## Byte Length: 40
+## Versions: [<2.2]
var root_node_name: String
## Byte Type: u32 [br]
@@ -90,23 +92,24 @@ static func from_bytes(bytes: ByteStream) -> RSMFormat:
rsm_format.texture_names = [] as Array[String]
for _n in rsm_format.texture_count:
- rsm_format.texture_names.append(bytes.get_string_from_utf8(40))
+ rsm_format.texture_names.append(bytes.get_string_from_ascii(40))
- rsm_format.root_node_name = bytes.get_string_from_utf8(40)
+ if version.lower_than(2, 2):
+ rsm_format.root_node_name = bytes.get_string_from_ascii(40)
if version.higher_than(2, 1): # >= 2.2
rsm_format.root_node_count = bytes.decode_u32()
rsm_format.root_node_names = [] as Array[String]
for _n in rsm_format.root_node_count:
- rsm_format.root_node_names.append(bytes.get_string_from_utf8(40))
+ rsm_format.root_node_names.append(bytes.get_string_from_ascii(40))
rsm_format.node_count = bytes.decode_u32()
rsm_format.nodes = [] as Array[ModelNode]
for _n in rsm_format.node_count:
rsm_format.nodes.append(ModelNode.from_bytes(bytes, version))
- print(inst_to_dict(rsm_format))
+ #print(inst_to_dict(rsm_format))
#print(inst_to_dict(rsm_format.nodes[0].texture_coordinates[0]))
#rsm_format.nodes[0].texture_coordinates.clear()
#print(inst_to_dict(rsm_format.nodes[0].faces[0]))
@@ -115,12 +118,16 @@ static func from_bytes(bytes: ByteStream) -> RSMFormat:
return rsm_format
-func convert() -> Node3D:
- var node := Node3D.new()
- node.name = root_node_name
+func convert(data_path: String) -> Node3D:
+ var root_node := Node3D.new()
+ root_node.name = root_node_name
#node.set_script(load("res://extractor/model.gd"))
- return node
+ for model_node in nodes:
+ var node: Node = model_node.convert(texture_names, data_path)
+ root_node.add_child(node)
+
+ return root_node
class ModelNode:
@@ -245,7 +252,7 @@ class ModelNode:
var node = ModelNode.new()
node.node_name = bytes.get_string_from_utf8(40)
- node.parent_node_name = bytes.get_string_from_utf8(40)
+ node.parent_node_name = bytes.get_string_from_ascii(40)
if version.lower_than(2, 3): # < 2.3
node.texture_count = bytes.decode_u32()
@@ -259,7 +266,7 @@ class ModelNode:
node.texture_names = [] as Array[String]
for _n in node.texture_name_count:
- node.texture_names.append(bytes.get_string_from_utf8(40))
+ node.texture_names.append(bytes.get_string_from_ascii(40))
node.offset_matrix = [] as Array[Vector3]
for _in in 3:
@@ -345,6 +352,43 @@ class ModelNode:
node.textures_keyframes.append(TexturesKeyframe.from_bytes(bytes))
return node
+
+
+ func convert(textures: Array[String], data_path: String):
+ var node := MeshInstance3D.new()
+ node.name = node_name
+
+ node.translate(translation_2)
+
+ if rotation_axis != Vector3.ZERO:
+ node.rotate(rotation_axis, rotation_angle)
+
+ node.scale = scale
+
+ var surface_tool := SurfaceTool.new()
+ surface_tool.begin(Mesh.PRIMITIVE_TRIANGLES)
+
+ var mesh := ArrayMesh.new()
+ for idx in faces.size():
+ var face := faces[idx]
+
+ surface_tool.add_triangle_fan(
+ PackedVector3Array(face.vertex_position_indices.map(func(idx):
+ return vertex_positions[idx])
+ ),
+ PackedVector2Array(face.texture_coordinate_indices.map(func(idx):
+ return texture_coordinates[idx].coordinates)
+ )
+ )
+ surface_tool.commit(mesh)
+
+ var material := StandardMaterial3D.new()
+ material.albedo_texture = load("%s/data/texture/%s" % [data_path, textures[face.texture_index]])
+ mesh.surface_set_material(idx, material)
+
+ node.mesh = mesh
+
+ return node
class TextureCoordinate:
diff --git a/extractor/rsw_format.gd b/extractor/rsw_format.gd
index 831a1cb..e38dad9 100644
--- a/extractor/rsw_format.gd
+++ b/extractor/rsw_format.gd
@@ -117,7 +117,43 @@ func convert(name: String, data_path: String) -> Node3D:
node.set_script(load("res://extractor/map.gd"))
for resource in map_resources:
- if resource is RSWFormat.SpatialAudioSource:
+ if resource is Animated3DModel:
+ var model_file_path := "%s/data/model/%s" % [data_path, resource.model_file]
+ if not FileAccess.file_exists(model_file_path):
+ continue
+
+ var rsm := RSMFormat.from_bytes(ByteStream.from_bytes(FileAccess.get_file_as_bytes(model_file_path)))
+
+ var model_root := Node3D.new()
+ model_root.name = resource.name
+ model_root.position = resource.get_position() * Vector3(-1, 1, 1)
+ model_root.rotate_x(resource.get_rotation().x)
+ model_root.rotate_y(resource.get_rotation().y)
+ model_root.rotate_z(resource.get_rotation().z)
+ model_root.scale = resource.get_scale()
+
+ node.add_child(model_root)
+ model_root.owner = node
+
+ var model := rsm.convert(data_path)
+ model_root.add_child(model)
+ model.owner = node
+ for child in model.get_children():
+ child.owner = node
+
+ elif resource is DynamicLightSource:
+ continue
+ var light := OmniLight3D.new()
+ light.name = resource.name
+ light.position = resource.get_position()
+ light.light_color = resource.get_diffuse_color()
+ light.omni_range = resource.light_range
+
+ node.add_child(light)
+ light.owner = node
+
+ elif resource is SpatialAudioSource:
+ continue
var audio_file_path := "%s/data/wav/%s" % [data_path, resource.audio_file]
if not FileAccess.file_exists(audio_file_path):
continue
@@ -128,13 +164,58 @@ func convert(name: String, data_path: String) -> Node3D:
audio.position = resource.get_position()
audio.volume_linear = resource.volume_gain
audio.max_distance = resource.audio_range
+
node.add_child(audio, true)
audio.owner = node
-
- var surface_tool := SurfaceTool.new()
- for surface: GNDFormat.Surface in gnd.surfaces:
- pass
- #surface_tool.add_vertex()
+
+ elif resource is ParticleEffectEmitter:
+ continue
+ var particles := CPUParticles3D.new()
+ particles.name = resource.name
+ particles.position = resource.get_position()
+
+ node.add_child(particles, true)
+ particles.owner = node
+
+ #var ground := MeshInstance3D.new()
+ #var mesh := ArrayMesh.new()
+ #var surface_tool := SurfaceTool.new()
+ #surface_tool.begin(Mesh.PRIMITIVE_TRIANGLES)
+ #for surface_idx in gnd.surfaces.size():
+ #if surface_idx < 1:
+ #continue
+ #
+ #var surface: GNDFormat.Surface = gnd.surfaces[surface_idx]
+ #
+ #surface_tool.add_triangle_fan(
+ #PackedVector3Array([Vector3(-0.5, -0.5, 0), Vector3(0.5, -0.5, 0), Vector3(-0.5, -0.5, -0.5)]),
+ #PackedVector2Array([surface.get_uvs()[Vector2(0, 0)], surface.get_uvs()[Vector2(1, 0)], surface.get_uvs()[Vector2(0, 1)]]),
+ #PackedColorArray([surface.get_vertex_color(),surface.get_vertex_color(),surface.get_vertex_color()])
+ #)
+ #surface_tool.add_triangle_fan(
+ #PackedVector3Array([Vector3(0.5, -0.5, 0), Vector3(0.5, -0.5, -0.5), Vector3(-0.5, -0.5, -0.5)]),
+ #PackedVector2Array([surface.get_uvs()[Vector2(1, 0)], surface.get_uvs()[Vector2(1, 1)], surface.get_uvs()[Vector2(0, 1)]]),
+ #PackedColorArray([surface.get_vertex_color(),surface.get_vertex_color(),surface.get_vertex_color()])
+ #)
+ #surface_tool.commit(mesh)
+ #var material := StandardMaterial3D.new()
+ #material.albedo_texture = load("%s/data/texture/%s" % [data_path, gnd.texture_paths[surface.texture_index]])
+ #mesh.surface_set_material(0, material)
+ #print(surface.get_uvs())
+ #break
+
+
+ #grid_map.owner = node
+
+ #ground.mesh = mesh
+ #
+ #node.add_child(ground)
+ #ground.owner = node
+
+ var grid_map := gnd.convert(data_path)
+ grid_map.name = gnd_file_path.get_basename()
+ node.add_child(grid_map)
+ grid_map.owner = node
return node