godot-parkour/addons/qodot/src/nodes/qodot_map.gd

1318 lines
47 KiB
GDScript

@tool
class_name QodotMap
extends QodotNode3D
## Builds Godot scenes from .map files
##
## A QodotMap node lets you define the source file for a map, as well as specify
## the definitions for entities, textures, and materials that appear in the map.
## To use this node, select an instance of the node in the Godot editor and
## select "Quick Build", "Full Build", or "Unwrap UV2" from the toolbar.
## Alternatively, call [method manual_build] from code.
##
## @tutorial: https://qodotplugin.github.io/docs/beginner's-guide-to-qodot/
## Force reinitialization of Qodot on map build
const DEBUG := false
## How long to wait between child/owner batches
const YIELD_DURATION := 0.0
## Emitted when the build process successfully completes
signal build_complete()
## Emitted when the build process finishes a step. [code]progress[/code] is from 0.0-1.0
signal build_progress(step, progress)
## Emitted when the build process fails
signal build_failed()
## Emitted when UV2 unwrapping is completed
signal unwrap_uv2_complete()
@export_category("Map")
## Trenchbroom Map file to build a scene from
@export_global_file("*.map") var map_file := ""
## Ratio between Trenchbroom units in the .map file and Godot units.
## An inverse scale factor of 16 would cause 16 Trenchbroom units to correspond to 1 Godot unit. See [url=https://qodotplugin.github.io/docs/geometry.html#scale]Scale[/url] in the Qodot documentation.
@export var inverse_scale_factor := 16.0
@export_category("Entities")
## [QodotFGDFile] for the map.
## This resource will translate between Trenchbroom classnames and Godot scripts/scenes. See [url=https://qodotplugin.github.io/docs/entities/]Entities[/url] in the Qodot manual.
@export var entity_fgd: QodotFGDFile = load("res://addons/qodot/game_definitions/fgd/qodot_fgd.tres")
@export_category("Textures")
## Base directory for textures. When building materials, Qodot will search this directory for texture files matching the textures assigned to Trenchbroom faces.
@export_dir var base_texture_dir := "res://textures"
## File extensions to search for texture data.
@export var texture_file_extensions := PackedStringArray(["png", "jpg", "jpeg", "bmp"])
## Optional. List of worldspawn layers.
## A worldspawn layer converts any brush of a certain texture to a certain kind of node. See example 1-2.
@export var worldspawn_layers: Array[QodotWorldspawnLayer]
## Optional. Path for the clip texture, relative to [member base_texture_dir].
## Brushes textured with the clip texture will be turned into invisible but solid volumes.
@export var brush_clip_texture := "special/clip"
## Optional. Path for the skip texture, relative to [member base_texture_dir].
## Faces textured with the skip texture will not be rendered.
@export var face_skip_texture := "special/skip"
## Optional. WAD files to pull textures from.
## Quake engine games are distributed with .WAD files, which are packed texture libraries. Qodot can import these files as [QuakeWadFile]s.
@export var texture_wads: Array[QuakeWadFile]
@export_category("Materials")
## File extensions to search for Material definitions
@export var material_file_extension := "tres"
## If true, all materials will be unshaded, i.e. will ignore light. Also known as "fullbright".
@export var unshaded := false
## Material used as template when generating missing materials.
@export var default_material : Material = StandardMaterial3D.new()
## Default albedo texture (used when [member default_material] is a [ShaderMaterial])
@export var default_material_albedo_uniform := ""
@export_category("UV Unwrap")
## Texel size for UV2 unwrapping.
## A texel size of 1 will lead to a 1:1 correspondence between texture texels and lightmap texels. Larger values will produce less detailed lightmaps. To conserve memory and filesize, use the largest value that still looks good.
@export var uv_unwrap_texel_size := 1.0
@export_category("Build")
## If true, print profiling data before and after each build step
@export var print_profiling_data := false
## If true, Qodot will build a hierarchy from Trenchbroom groups, each group being a node. Otherwise, Qodot nodes will ignore Trenchbroom groups and have a flat structure.
@export var use_trenchbroom_group_hierarchy := false
## If true, stop the whole editor until build is complete
@export var block_until_complete := false
## How many nodes to set the owner of, or add children of, at once. Higher values may lead to quicker build times, but a less responsive editor.
@export var set_owner_batch_size := 1000
# Build context variables
var qodot = null
var profile_timestamps := {}
var add_child_array := []
var set_owner_array := []
var should_add_children := true
var should_set_owners := true
var texture_list := []
var texture_loader = null
var texture_dict := {}
var texture_size_dict := {}
var material_dict := {}
var entity_definitions := {}
var entity_dicts := []
var worldspawn_layer_dicts := []
var entity_mesh_dict := {}
var worldspawn_layer_mesh_dict := {}
var entity_nodes := []
var worldspawn_layer_nodes := []
var entity_mesh_instances := {}
var entity_occluder_instances := {}
var worldspawn_layer_mesh_instances := {}
var entity_collision_shapes := []
var worldspawn_layer_collision_shapes := []
# Overrides
func _ready() -> void:
if not DEBUG:
return
if not Engine.is_editor_hint():
if verify_parameters():
build_map()
# Utility
## Verify that Qodot is functioning and that [member map_file] exists. If so, build the map. If not, signal [signal build_failed]
func verify_and_build():
if verify_parameters():
build_map()
else:
emit_signal("build_failed")
## Build the map.
func manual_build():
should_add_children = false
should_set_owners = false
verify_and_build()
## Return true if parameters are valid; Qodot should be functioning and [member map_file] should exist.
func verify_parameters():
if not qodot or DEBUG:
qodot = load("res://addons/qodot/src/core/qodot.gd").new()
if not qodot:
push_error("Error: Failed to load qodot.")
return false
if map_file == "":
push_error("Error: Map file not set")
return false
if not FileAccess.file_exists(map_file):
push_error("Error: No such file %s" % map_file)
return false
return true
## Reset member variables that affect the current build
func reset_build_context():
add_child_array = []
set_owner_array = []
texture_list = []
texture_loader = null
texture_dict = {}
texture_size_dict = {}
material_dict = {}
entity_definitions = {}
entity_dicts = []
worldspawn_layer_dicts = []
entity_mesh_dict = {}
worldspawn_layer_mesh_dict = {}
entity_nodes = []
worldspawn_layer_nodes = []
entity_mesh_instances = {}
entity_occluder_instances = {}
worldspawn_layer_mesh_instances = {}
entity_collision_shapes = []
worldspawn_layer_collision_shapes = []
build_step_index = 0
build_step_count = 0
if qodot:
qodot = load("res://addons/qodot/src/core/qodot.gd").new()
## Record the start time of a build step for profiling
func start_profile(item_name: String) -> void:
if print_profiling_data:
print(item_name)
profile_timestamps[item_name] = Time.get_unix_time_from_system()
## Finish profiling for a build step; print associated timing data
func stop_profile(item_name: String) -> void:
if print_profiling_data:
if item_name in profile_timestamps:
var delta: float = Time.get_unix_time_from_system() - profile_timestamps[item_name]
print("Done in %s sec.\n" % snapped(delta, 0.01))
profile_timestamps.erase(item_name)
## Run a build step. [code]step_name[/code] is the method corresponding to the step, [code]params[/code] are parameters to pass to the step, and [code]func_name[/code] does nothing.
func run_build_step(step_name: String, params: Array = [], func_name: String = ""):
start_profile(step_name)
if func_name == "":
func_name = step_name
var result = callv(step_name, params)
stop_profile(step_name)
return result
## Add [code]node[/code] as a child of parent, or as a child of [code]below[/code] if non-null. Also queue for ownership assignment.
func add_child_editor(parent, node, below = null) -> void:
var prev_parent = node.get_parent()
if prev_parent:
prev_parent.remove_child(node)
if below:
below.add_sibling(node)
else:
parent.add_child(node)
set_owner_array.append(node)
## Set the owner of [code]node[/code] to the current scene.
func set_owner_editor(node):
var tree := get_tree()
if not tree:
return
var edited_scene_root := tree.get_edited_scene_root()
if not edited_scene_root:
return
node.set_owner(edited_scene_root)
var build_step_index := 0
var build_step_count := 0
var build_steps := []
var post_attach_steps := []
## Register a build step.
## [code]build_step[/code] is a string that corresponds to a method on this class, [code]arguments[/code] a list of arguments to pass to this method, and [code]target[/code] is a property on this class to save the return value of the build step in. If [code]post_attach[/code] is true, the step will be run after the scene hierarchy is completed.
func register_build_step(build_step: String, arguments := [], target := "", post_attach := false) -> void:
(post_attach_steps if post_attach else build_steps).append([build_step, arguments, target])
build_step_count += 1
## Run all build steps. Emits [signal build_progress] after each step.
## If [code]post_attach[/code] is true, run post-attach steps instead and signal [signal build_complete] when finished.
func run_build_steps(post_attach := false) -> void:
var target_array = post_attach_steps if post_attach else build_steps
while target_array.size() > 0:
var build_step = target_array.pop_front()
emit_signal("build_progress", build_step[0], float(build_step_index + 1) / float(build_step_count))
var scene_tree := get_tree()
if scene_tree and not block_until_complete:
await get_tree().create_timer(YIELD_DURATION).timeout
var result = run_build_step(build_step[0], build_step[1])
var target = build_step[2]
if target != "":
set(target, result)
build_step_index += 1
if scene_tree and not block_until_complete:
await get_tree().create_timer(YIELD_DURATION).timeout
if post_attach:
_build_complete()
else:
start_profile('add_children')
add_children()
## Register all steps for the build. See [method register_build_step] and [method run_build_steps]
func register_build_steps() -> void:
register_build_step('remove_children')
register_build_step('load_map')
register_build_step('fetch_texture_list', [], 'texture_list')
register_build_step('init_texture_loader', [], 'texture_loader')
register_build_step('load_textures', [], 'texture_dict')
register_build_step('build_texture_size_dict', [], 'texture_size_dict')
register_build_step('build_materials', [], 'material_dict')
register_build_step('fetch_entity_definitions', [], 'entity_definitions')
register_build_step('set_qodot_entity_definitions', [])
register_build_step('set_qodot_worldspawn_layers', [])
register_build_step('generate_geometry', [])
register_build_step('fetch_entity_dicts', [], 'entity_dicts')
register_build_step('fetch_worldspawn_layer_dicts', [], 'worldspawn_layer_dicts')
register_build_step('build_entity_nodes', [], 'entity_nodes')
register_build_step('build_worldspawn_layer_nodes', [], 'worldspawn_layer_nodes')
register_build_step('resolve_group_hierarchy', [])
register_build_step('build_entity_mesh_dict', [], 'entity_mesh_dict')
register_build_step('build_worldspawn_layer_mesh_dict', [], 'worldspawn_layer_mesh_dict')
register_build_step('build_entity_mesh_instances', [], 'entity_mesh_instances')
register_build_step('build_entity_occluder_instances', [], 'entity_occluder_instances')
register_build_step('build_worldspawn_layer_mesh_instances', [], 'worldspawn_layer_mesh_instances')
register_build_step('build_entity_collision_shape_nodes', [], 'entity_collision_shapes')
register_build_step('build_worldspawn_layer_collision_shape_nodes', [], 'worldspawn_layer_collision_shapes')
## Register all post-attach steps for the build. See [method register_build_step] and [method run_build_steps]
func register_post_attach_steps() -> void:
register_build_step('build_entity_collision_shapes', [], "", true)
register_build_step('build_worldspawn_layer_collision_shapes', [], "", true)
register_build_step('apply_entity_meshes', [], "", true)
register_build_step('apply_entity_occluders', [], "", true)
register_build_step('apply_worldspawn_layer_meshes', [], "", true)
register_build_step('apply_properties', [], "", true)
register_build_step('connect_signals', [], "", true)
register_build_step('remove_transient_nodes', [], "", true)
# Actions
## Build the map
func build_map() -> void:
reset_build_context()
print('Building %s\n' % map_file)
start_profile('build_map')
register_build_steps()
register_post_attach_steps()
run_build_steps()
## Recursively unwrap UV2s for [code]node[/code] and its children, in preparation for baked lighting.
func unwrap_uv2(node: Node = null) -> void:
var target_node = null
if node:
target_node = node
else:
target_node = self
print("Unwrapping mesh UV2s")
if target_node is MeshInstance3D:
var mesh = target_node.get_mesh()
if mesh is ArrayMesh:
mesh.lightmap_unwrap(Transform3D.IDENTITY, uv_unwrap_texel_size / inverse_scale_factor)
for child in target_node.get_children():
unwrap_uv2(child)
if not node:
print("Unwrap complete")
emit_signal("unwrap_uv2_complete")
# Build Steps
## Recursively remove and delete all children of this node
func remove_children() -> void:
for child in get_children():
remove_child(child)
child.queue_free()
## Parse and load [member map_file]
func load_map() -> void:
var file: String = map_file
qodot.load_map(file)
## Get textures found in [member map_file]
func fetch_texture_list() -> Array:
return qodot.get_texture_list() as Array
## Initialize texture loader, allowing textures in [member base_texture_dir] and [member texture_wads] to be turned into materials
func init_texture_loader() -> QodotTextureLoader:
var tex_ldr := QodotTextureLoader.new(
base_texture_dir,
texture_file_extensions,
texture_wads
)
tex_ldr.unshaded = unshaded
return tex_ldr
## Build a dictionary from Trenchbroom texture names to their corresponding Texture2D resources in Godot
func load_textures() -> Dictionary:
return texture_loader.load_textures(texture_list) as Dictionary
## Build a dictionary from Trenchbroom texture names to Godot materials
func build_materials() -> Dictionary:
return texture_loader.create_materials(texture_list, material_file_extension, default_material, default_material_albedo_uniform)
## Collect entity definitions from [member entity_fgd], as a dictionary from Trenchbroom classnames to entity definitions
func fetch_entity_definitions() -> Dictionary:
return entity_fgd.get_entity_definitions()
## Hand the Qodot C# core the entity definitions
func set_qodot_entity_definitions() -> void:
qodot.set_entity_definitions(build_libmap_entity_definitions(entity_definitions))
## Hand the Qodot C# core the worldspawn layer definitions. See [member worldspawn_layers]
func set_qodot_worldspawn_layers() -> void:
qodot.set_worldspawn_layers(build_libmap_worldspawn_layers(worldspawn_layers))
## Generate geometry from map file
func generate_geometry() -> void:
qodot.generate_geometry(texture_size_dict);
## Get a list of dictionaries representing each entity from the Qodot C# core
func fetch_entity_dicts() -> Array:
return qodot.get_entity_dicts()
## Get a list of dictionaries representing each worldspawn layer from the Qodot C# core
func fetch_worldspawn_layer_dicts() -> Array:
var layer_dicts = qodot.get_worldspawn_layer_dicts()
return layer_dicts if layer_dicts else []
## Build a dictionary from Trenchbroom textures to the sizes of their corresponding Godot textures
func build_texture_size_dict() -> Dictionary:
var texture_size_dict := {}
for tex_key in texture_dict:
var texture := texture_dict[tex_key] as Texture2D
if texture:
texture_size_dict[tex_key] = texture.get_size()
else:
texture_size_dict[tex_key] = Vector2.ONE
return texture_size_dict
## Marshall Qodot FGD definitions for transfer to libmap
func build_libmap_entity_definitions(entity_definitions: Dictionary) -> Dictionary:
var libmap_entity_definitions = {}
for classname in entity_definitions:
libmap_entity_definitions[classname] = {}
if entity_definitions[classname] is QodotFGDSolidClass:
libmap_entity_definitions[classname]['spawn_type'] = entity_definitions[classname].spawn_type
return libmap_entity_definitions
## Marshall worldspawn layer definitions for transfer to libmap
func build_libmap_worldspawn_layers(worldspawn_layers: Array) -> Array:
var libmap_worldspawn_layers := []
for worldspawn_layer in worldspawn_layers:
libmap_worldspawn_layers.append({
'name': worldspawn_layer.name,
'texture': worldspawn_layer.texture,
'node_class': worldspawn_layer.node_class,
'build_visuals': worldspawn_layer.build_visuals,
'collision_shape_type': worldspawn_layer.collision_shape_type,
'script_class': worldspawn_layer.script_class
})
return libmap_worldspawn_layers
## Build nodes from the entities in [member entity_dicts]
func build_entity_nodes() -> Array:
var entity_nodes := []
for entity_idx in range(0, entity_dicts.size()):
var entity_dict := entity_dicts[entity_idx] as Dictionary
var properties := entity_dict['properties'] as Dictionary
var node = QodotEntity.new()
var node_name = "entity_%s" % entity_idx
var should_add_child = should_add_children
if 'classname' in properties:
var classname = properties['classname']
node_name += "_" + classname
if classname in entity_definitions:
var entity_definition := entity_definitions[classname] as QodotFGDClass
if entity_definition is QodotFGDSolidClass:
if entity_definition.spawn_type == QodotFGDSolidClass.SpawnType.MERGE_WORLDSPAWN:
entity_nodes.append(null)
continue
elif use_trenchbroom_group_hierarchy and entity_definition.spawn_type == QodotFGDSolidClass.SpawnType.GROUP:
should_add_child = false
if entity_definition.node_class != "":
node.queue_free()
node = ClassDB.instantiate(entity_definition.node_class)
elif entity_definition is QodotFGDPointClass:
if entity_definition.scene_file:
var flag = PackedScene.GEN_EDIT_STATE_DISABLED
if Engine.is_editor_hint():
flag = PackedScene.GEN_EDIT_STATE_INSTANCE
node.queue_free()
node = entity_definition.scene_file.instantiate(flag)
elif entity_definition.node_class != "":
node.queue_free()
node = ClassDB.instantiate(entity_definition.node_class)
if 'rotation_degrees' in node and entity_definition.apply_rotation_on_map_build:
var angles := Vector3.ZERO
if 'angles' in properties or 'mangle' in properties:
var key := 'angles' if 'angles' in properties else 'mangle'
var angles_raw = properties[key]
if not angles_raw is Vector3:
angles_raw = angles_raw.split_floats(' ')
if angles_raw.size() > 2:
angles = Vector3(-angles_raw[0], angles_raw[1], -angles_raw[2])
if key == 'mangle':
if entity_definition.classname.begins_with('light'):
angles = Vector3(angles_raw[1], angles_raw[0], -angles_raw[2])
elif entity_definition.classname == 'info_intermission':
angles = Vector3(angles_raw[0], angles_raw[1], -angles_raw[2])
else:
push_error("Invalid vector format for \'" + key + "\' in entity \'" + classname + "\'")
elif 'angle' in properties:
var angle = properties['angle']
if not angle is float:
angle = float(angle)
angles.y += angle
angles.y += 180
node.rotation_degrees = angles
if entity_definition.script_class:
node.set_script(entity_definition.script_class)
node.name = node_name
if 'origin' in properties:
var origin_vec = Vector3.ZERO
var origin_comps = properties['origin'].split_floats(' ')
if origin_comps.size() > 2:
origin_vec = Vector3(origin_comps[1], origin_comps[2], origin_comps[0])
else:
push_error("Invalid vector format for \'origin\' in " + node.name)
if "position" in node:
if node.position is Vector3:
node.position = origin_vec / inverse_scale_factor
elif node.position is Vector2:
node.position = Vector2(origin_vec.z, -origin_vec.y)
else:
if entity_idx != 0 and "position" in node:
if node.position is Vector3:
node.position = entity_dict['center'] / inverse_scale_factor
entity_nodes.append(node)
if should_add_child:
queue_add_child(self, node)
return entity_nodes
## Build nodes from the worldspawn layers in [member worldspawn_layers]
func build_worldspawn_layer_nodes() -> Array:
var worldspawn_layer_nodes := []
for worldspawn_layer in worldspawn_layers:
var node = ClassDB.instantiate(worldspawn_layer.node_class)
node.name = "entity_0_" + worldspawn_layer.name
if worldspawn_layer.script_class:
node.set_script(worldspawn_layer.script_class)
worldspawn_layer_nodes.append(node)
queue_add_child(self, node, entity_nodes[0])
return worldspawn_layer_nodes
## Resolve entity group hierarchy, turning Trenchbroom groups into nodes and queueing their contents to be added to said nodes as children
func resolve_group_hierarchy() -> void:
if not use_trenchbroom_group_hierarchy:
return
var group_entities := {}
var owner_entities := {}
# Gather group entities and their owning children
for node_idx in range(0, entity_nodes.size()):
var node = entity_nodes[node_idx]
var properties = entity_dicts[node_idx]['properties']
if not properties: continue
if not '_tb_id' in properties and not '_tb_group' in properties:
continue
if not 'classname' in properties: continue
var classname = properties['classname']
if not classname in entity_definitions: continue
var entity_definition = entity_definitions[classname]
# TODO: Add clause on this line for point entities, which do not have a spawn type. Add as child of the current group owner.
if entity_definition.spawn_type == QodotFGDSolidClass.SpawnType.GROUP:
group_entities[node_idx] = node
else:
owner_entities[node_idx] = node
var group_to_entity_map := {}
for node_idx in owner_entities:
var node = owner_entities[node_idx]
var properties = entity_dicts[node_idx]['properties']
var tb_group = properties['_tb_group']
var parent_idx = null
var parent = null
var parent_properties = null
for group_idx in group_entities:
var group_entity = group_entities[group_idx]
var group_properties = entity_dicts[group_idx]['properties']
if group_properties['_tb_id'] == tb_group:
parent_idx = group_idx
parent = group_entity
parent_properties = group_properties
break
if parent:
group_to_entity_map[parent_idx] = node_idx
var group_to_group_map := {}
for node_idx in group_entities:
var node = group_entities[node_idx]
var properties = entity_dicts[node_idx]['properties']
if not '_tb_group' in properties:
continue
var tb_group = properties['_tb_group']
var parent_idx = null
var parent = null
var parent_properties = null
for group_idx in group_entities:
var group_entity = group_entities[group_idx]
var group_properties = entity_dicts[group_idx]['properties']
if group_properties['_tb_id'] == tb_group:
parent_idx = group_idx
parent = group_entity
parent_properties = group_properties
break
if parent:
group_to_group_map[parent_idx] = node_idx
for parent_idx in group_to_group_map:
var child_idx = group_to_group_map[parent_idx]
var parent_entity_idx = group_to_entity_map[parent_idx]
var child_entity_idx = group_to_entity_map[child_idx]
var parent = entity_nodes[parent_entity_idx]
var child = entity_nodes[child_entity_idx]
queue_add_child(parent, child, null, true)
for child_idx in group_to_entity_map:
var parent_idx = group_to_entity_map[child_idx]
var parent = entity_nodes[parent_idx]
var child = entity_nodes[child_idx]
queue_add_child(parent, child, null, true)
## Return the node associated with a Trenchbroom index. Unused.
func get_node_by_tb_id(target_id: String, entity_nodes: Dictionary):
for node_idx in entity_nodes:
var node = entity_nodes[node_idx]
if not node:
continue
if not 'properties' in node:
continue
var properties = node['properties']
if not '_tb_id' in properties:
continue
var parent_id = properties['_tb_id']
if parent_id == target_id:
return node
return null
## Build [CollisionShape3D] nodes for brush entities
func build_entity_collision_shape_nodes() -> Array:
var entity_collision_shapes_arr := []
for entity_idx in range(0, entity_nodes.size()):
var entity_collision_shapes := []
var entity_dict = entity_dicts[entity_idx]
var properties = entity_dict['properties']
var node := entity_nodes[entity_idx] as Node
var concave = false
if 'classname' in properties:
var classname = properties['classname']
if classname in entity_definitions:
var entity_definition := entity_definitions[classname] as QodotFGDSolidClass
if entity_definition:
if entity_definition.collision_shape_type == QodotFGDSolidClass.CollisionShapeType.NONE:
entity_collision_shapes_arr.append(null)
continue
elif entity_definition.collision_shape_type == QodotFGDSolidClass.CollisionShapeType.CONCAVE:
concave = true
if entity_definition.spawn_type == QodotFGDSolidClass.SpawnType.MERGE_WORLDSPAWN:
# TODO: Find the worldspawn object instead of assuming index 0
node = entity_nodes[0] as Node
if node and node is CollisionObject3D:
(node as CollisionObject3D).collision_layer = entity_definition.collision_layer
(node as CollisionObject3D).collision_mask = entity_definition.collision_mask
(node as CollisionObject3D).collision_priority = entity_definition.collision_priority
# don't create collision shapes that wont be attached to a CollisionObject3D as they are a waste
if not node or (not node is CollisionObject3D):
entity_collision_shapes_arr.append(null)
continue
if concave:
var collision_shape := CollisionShape3D.new()
collision_shape.name = "entity_%s_collision_shape" % entity_idx
entity_collision_shapes.append(collision_shape)
queue_add_child(node, collision_shape)
else:
for brush_idx in entity_dict['brush_indices']:
var collision_shape := CollisionShape3D.new()
collision_shape.name = "entity_%s_brush_%s_collision_shape" % [entity_idx, brush_idx]
entity_collision_shapes.append(collision_shape)
queue_add_child(node, collision_shape)
entity_collision_shapes_arr.append(entity_collision_shapes)
return entity_collision_shapes_arr
## Build CollisionShape3D nodes for worldspawn layers
func build_worldspawn_layer_collision_shape_nodes() -> Array:
var worldspawn_layer_collision_shapes := []
for layer_idx in range(0, worldspawn_layers.size()):
if layer_idx >= worldspawn_layer_dicts.size():
continue
var layer = worldspawn_layers[layer_idx]
var layer_dict = worldspawn_layer_dicts[layer_idx]
var node := worldspawn_layer_nodes[layer_idx] as Node
var concave = false
var shapes := []
if layer.collision_shape_type == QodotFGDSolidClass.CollisionShapeType.NONE:
worldspawn_layer_collision_shapes.append(shapes)
continue
elif layer.collision_shape_type == QodotFGDSolidClass.CollisionShapeType.CONCAVE:
concave = true
if not node:
worldspawn_layer_collision_shapes.append(shapes)
continue
if concave:
var collision_shape := CollisionShape3D.new()
collision_shape.name = "entity_0_%s_collision_shape" % layer.name
shapes.append(collision_shape)
queue_add_child(node, collision_shape)
else:
for brush_idx in layer_dict['brush_indices']:
var collision_shape := CollisionShape3D.new()
collision_shape.name = "entity_0_%s_brush_%s_collision_shape" % [layer.name, brush_idx]
shapes.append(collision_shape)
queue_add_child(node, collision_shape)
worldspawn_layer_collision_shapes.append(shapes)
return worldspawn_layer_collision_shapes
## Build the concrete [Shape3D] resources for each brush
func build_entity_collision_shapes() -> void:
for entity_idx in range(0, entity_dicts.size()):
var entity_dict := entity_dicts[entity_idx] as Dictionary
var properties = entity_dict['properties']
var entity_position: Vector3 = Vector3.ZERO
if entity_nodes[entity_idx] != null and entity_nodes[entity_idx].get("position"):
if entity_nodes[entity_idx].position is Vector3:
entity_position = entity_nodes[entity_idx].position
var entity_collision_shape = entity_collision_shapes[entity_idx]
if entity_collision_shape == null:
continue
var concave: bool = false
var shape_margin: float = 0.04
if 'classname' in properties:
var classname = properties['classname']
if classname in entity_definitions:
var entity_definition = entity_definitions[classname] as QodotFGDSolidClass
if entity_definition:
match(entity_definition.collision_shape_type):
QodotFGDSolidClass.CollisionShapeType.NONE:
continue
QodotFGDSolidClass.CollisionShapeType.CONVEX:
concave = false
QodotFGDSolidClass.CollisionShapeType.CONCAVE:
concave = true
shape_margin = entity_definition.collision_shape_margin
if entity_collision_shapes[entity_idx] == null:
continue
if concave:
qodot.gather_entity_concave_collision_surfaces(entity_idx, face_skip_texture)
else:
qodot.gather_entity_convex_collision_surfaces(entity_idx)
var entity_surfaces := qodot.fetch_surfaces(inverse_scale_factor) as Array
var entity_verts := PackedVector3Array()
for surface_idx in range(0, entity_surfaces.size()):
var surface_verts = entity_surfaces[surface_idx]
if surface_verts == null:
continue
if concave:
var vertices := surface_verts[Mesh.ARRAY_VERTEX] as PackedVector3Array
var indices := surface_verts[Mesh.ARRAY_INDEX] as PackedInt32Array
for vert_idx in indices:
entity_verts.append(vertices[vert_idx])
else:
var shape_points = PackedVector3Array()
for vertex in surface_verts[Mesh.ARRAY_VERTEX]:
if not vertex in shape_points:
shape_points.append(vertex)
var shape = ConvexPolygonShape3D.new()
shape.set_points(shape_points)
shape.margin = shape_margin
var collision_shape = entity_collision_shape[surface_idx]
collision_shape.set_shape(shape)
if concave:
if entity_verts.size() == 0:
continue
var shape = ConcavePolygonShape3D.new()
shape.set_faces(entity_verts)
shape.margin = shape_margin
var collision_shape = entity_collision_shapes[entity_idx][0]
collision_shape.set_shape(shape)
## Build the concrete [Shape3D] resources for each worldspawn layer
func build_worldspawn_layer_collision_shapes() -> void:
for layer_idx in range(0, worldspawn_layers.size()):
if layer_idx >= worldspawn_layer_dicts.size():
continue
var layer = worldspawn_layers[layer_idx]
var concave = false
match(layer.collision_shape_type):
QodotFGDSolidClass.CollisionShapeType.NONE:
continue
QodotFGDSolidClass.CollisionShapeType.CONVEX:
concave = false
QodotFGDSolidClass.CollisionShapeType.CONCAVE:
concave = true
var layer_dict = worldspawn_layer_dicts[layer_idx]
if not worldspawn_layer_collision_shapes[layer_idx]:
continue
qodot.gather_worldspawn_layer_collision_surfaces(0)
var layer_surfaces := qodot.FetchSurfaces(inverse_scale_factor) as Array
var verts := PackedVector3Array()
for i in range(0, layer_dict.brush_indices.size()):
var surface_idx = layer_dict.brush_indices[i]
var surface_verts = layer_surfaces[surface_idx]
if not surface_verts:
continue
if concave:
var vertices := surface_verts[0] as PackedVector3Array
var indices := surface_verts[8] as PackedInt32Array
for vert_idx in indices:
verts.append(vertices[vert_idx])
else:
var shape_points = PackedVector3Array()
for vertex in surface_verts[0]:
if not vertex in shape_points:
shape_points.append(vertex)
var shape = ConvexPolygonShape3D.new()
shape.set_points(shape_points)
var collision_shape = worldspawn_layer_collision_shapes[layer_idx][i]
collision_shape.set_shape(shape)
if concave:
if verts.size() == 0:
continue
var shape = ConcavePolygonShape3D.new()
shape.set_faces(verts)
var collision_shape = worldspawn_layer_collision_shapes[layer_idx][0]
collision_shape.set_shape(shape)
## Build Dictionary from entity indices to [ArrayMesh] instances
func build_entity_mesh_dict() -> Dictionary:
var meshes := {}
var texture_surf_map: Dictionary
for texture in texture_dict:
texture_surf_map[texture] = Array()
var gather_task = func(i):
var texture = texture_dict.keys()[i]
texture_surf_map[texture] = qodot.gather_texture_surfaces_mt(texture, brush_clip_texture, face_skip_texture, inverse_scale_factor)
var task_id:= WorkerThreadPool.add_group_task(gather_task, texture_dict.keys().size(), 4, true)
WorkerThreadPool.wait_for_group_task_completion(task_id)
for texture in texture_dict:
var texture_surfaces := texture_surf_map[texture] as Array
for entity_idx in range(0, texture_surfaces.size()):
var entity_dict := entity_dicts[entity_idx] as Dictionary
var properties = entity_dict['properties']
var entity_surface = texture_surfaces[entity_idx]
if 'classname' in properties:
var classname = properties['classname']
if classname in entity_definitions:
var entity_definition = entity_definitions[classname] as QodotFGDSolidClass
if entity_definition:
if entity_definition.spawn_type == QodotFGDSolidClass.SpawnType.MERGE_WORLDSPAWN:
entity_surface = null
if not entity_definition.build_visuals and not entity_definition.build_occlusion:
entity_surface = null
if entity_surface == null:
continue
if not entity_idx in meshes:
meshes[entity_idx] = ArrayMesh.new()
var mesh: ArrayMesh = meshes[entity_idx]
mesh.add_surface_from_arrays(ArrayMesh.PRIMITIVE_TRIANGLES, entity_surface)
mesh.surface_set_name(mesh.get_surface_count() - 1, texture)
mesh.surface_set_material(mesh.get_surface_count() - 1, material_dict[texture])
return meshes
## Build Dictionary from worldspawn layers (via textures) to [ArrayMesh] instances
func build_worldspawn_layer_mesh_dict() -> Dictionary:
var meshes := {}
for layer in worldspawn_layer_dicts:
var texture = layer.texture
qodot.gather_worldspawn_layer_surfaces(texture, brush_clip_texture, face_skip_texture)
var texture_surfaces := qodot.fetch_surfaces(inverse_scale_factor) as Array
var mesh: Mesh = null
if not texture in meshes:
meshes[texture] = ArrayMesh.new()
mesh = meshes[texture]
mesh.add_surface_from_arrays(ArrayMesh.PRIMITIVE_TRIANGLES, texture_surfaces[0])
mesh.surface_set_name(mesh.get_surface_count() - 1, texture)
mesh.surface_set_material(mesh.get_surface_count() - 1, material_dict[texture])
return meshes
## Build [MeshInstance3D]s from brush entities and add them to the add child queue
func build_entity_mesh_instances() -> Dictionary:
var entity_mesh_instances := {}
for entity_idx in entity_mesh_dict:
var use_in_baked_light = false
var shadow_casting_setting := GeometryInstance3D.SHADOW_CASTING_SETTING_DOUBLE_SIDED
var render_layers: int = 1
var entity_dict := entity_dicts[entity_idx] as Dictionary
var properties = entity_dict['properties']
var classname = properties['classname']
if classname in entity_definitions:
var entity_definition = entity_definitions[classname] as QodotFGDSolidClass
if entity_definition:
if not entity_definition.build_visuals:
continue
if entity_definition.use_in_baked_light:
use_in_baked_light = true
elif '_shadow' in properties:
if properties['_shadow'] == "1":
use_in_baked_light = true
shadow_casting_setting = entity_definition.shadow_casting_setting
render_layers = entity_definition.render_layers
if not entity_mesh_dict[entity_idx]:
continue
var mesh_instance := MeshInstance3D.new()
mesh_instance.name = 'entity_%s_mesh_instance' % entity_idx
mesh_instance.gi_mode = MeshInstance3D.GI_MODE_STATIC if use_in_baked_light else GeometryInstance3D.GI_MODE_DISABLED
mesh_instance.cast_shadow = shadow_casting_setting
mesh_instance.layers = render_layers
queue_add_child(entity_nodes[entity_idx], mesh_instance)
entity_mesh_instances[entity_idx] = mesh_instance
return entity_mesh_instances
func build_entity_occluder_instances() -> Dictionary:
var entity_occluder_instances := {}
for entity_idx in entity_mesh_dict:
var entity_dict := entity_dicts[entity_idx] as Dictionary
var properties = entity_dict['properties']
var classname = properties['classname']
if classname in entity_definitions:
var entity_definition = entity_definitions[classname] as QodotFGDSolidClass
if entity_definition:
if entity_definition.build_occlusion:
if not entity_mesh_dict[entity_idx]:
continue
var occluder_instance := OccluderInstance3D.new()
occluder_instance.name = 'entity_%s_occluder_instance' % entity_idx
queue_add_child(entity_nodes[entity_idx], occluder_instance)
entity_occluder_instances[entity_idx] = occluder_instance
return entity_occluder_instances
## Build Dictionary from worldspawn layers (via textures) to [MeshInstance3D]s
func build_worldspawn_layer_mesh_instances() -> Dictionary:
var worldspawn_layer_mesh_instances := {}
var idx = 0
for i in range(0, worldspawn_layers.size()):
var worldspawn_layer = worldspawn_layers[i]
var texture_name = worldspawn_layer.texture
if not texture_name in worldspawn_layer_mesh_dict:
continue
var mesh := worldspawn_layer_mesh_dict[texture_name] as Mesh
if not mesh:
continue
var mesh_instance := MeshInstance3D.new()
mesh_instance.name = 'entity_0_%s_mesh_instance' % worldspawn_layer.name
mesh_instance.gi_mode = MeshInstance3D.GI_MODE_STATIC
queue_add_child(worldspawn_layer_nodes[idx], mesh_instance)
idx += 1
worldspawn_layer_mesh_instances[texture_name] = mesh_instance
return worldspawn_layer_mesh_instances
## Assign [ArrayMesh]es to their [MeshInstance3D] counterparts
func apply_entity_meshes() -> void:
for entity_idx in entity_mesh_instances:
var mesh := entity_mesh_dict[entity_idx] as Mesh
var mesh_instance := entity_mesh_instances[entity_idx] as MeshInstance3D
if not mesh or not mesh_instance:
continue
mesh_instance.set_mesh(mesh)
queue_add_child(entity_nodes[entity_idx], mesh_instance)
func apply_entity_occluders() -> void:
for entity_idx in entity_mesh_dict:
var mesh := entity_mesh_dict[entity_idx] as Mesh
var occluder_instance : OccluderInstance3D
if entity_idx in entity_occluder_instances:
occluder_instance = entity_occluder_instances[entity_idx]
if not mesh or not occluder_instance:
continue
var verts: PackedVector3Array
var indices: PackedInt32Array
for surf_idx in range(mesh.get_surface_count()):
var vert_count := verts.size()
var surf_array := mesh.surface_get_arrays(surf_idx)
verts.append_array(surf_array[Mesh.ARRAY_VERTEX])
indices.resize(indices.size() + surf_array[Mesh.ARRAY_INDEX].size())
for new_index in surf_array[Mesh.ARRAY_INDEX]:
indices.append(new_index + vert_count)
var occluder := ArrayOccluder3D.new()
occluder.set_arrays(verts, indices)
occluder_instance.occluder = occluder
## Assign [ArrayMesh]es to their [MeshInstance3D] counterparts for worldspawn layers
func apply_worldspawn_layer_meshes() -> void:
for texture_name in worldspawn_layer_mesh_dict:
var mesh = worldspawn_layer_mesh_dict[texture_name]
var mesh_instance = worldspawn_layer_mesh_instances[texture_name]
if not mesh or not mesh_instance:
continue
mesh_instance.set_mesh(mesh)
## Add a child and its new parent to the add child queue. If [code]below[/code] is a node, add it as a child to that instead. If [code]relative[/code] is true, set the location of node relative to parent.
func queue_add_child(parent, node, below = null, relative = false) -> void:
add_child_array.append({"parent": parent, "node": node, "below": below, "relative": relative})
## Assign children to parents based on the contents of the add child queue (see [method queue_add_child])
func add_children() -> void:
while true:
for i in range(0, set_owner_batch_size):
var data = add_child_array.pop_front()
if data:
add_child_editor(data['parent'], data['node'], data['below'])
if data['relative']:
data['node'].global_transform.origin -= data['parent'].global_transform.origin
else:
add_children_complete()
return
var scene_tree := get_tree()
if scene_tree and not block_until_complete:
await get_tree().create_timer(YIELD_DURATION).timeout
## Set owners and start post-attach build steps
func add_children_complete():
stop_profile('add_children')
if should_set_owners:
start_profile('set_owners')
set_owners()
else:
run_build_steps(true)
## Set owner of nodes generated by Qodot to scene root based on [member set_owner_array]
func set_owners():
while true:
for i in range(0, set_owner_batch_size):
var node = set_owner_array.pop_front()
if node:
set_owner_editor(node)
else:
set_owners_complete()
return
var scene_tree := get_tree()
if scene_tree and not block_until_complete:
await get_tree().create_timer(YIELD_DURATION).timeout
## Finish profiling for set_owners and start post-attach build steps
func set_owners_complete():
stop_profile('set_owners')
run_build_steps(true)
## Apply Trenchbroom properties to [QodotEntity] instances, transferring Trenchbroom dictionaries to [QodotEntity.properties]
func apply_properties() -> void:
for entity_idx in range(0, entity_nodes.size()):
var entity_node = entity_nodes[entity_idx]
if not entity_node:
continue
var entity_dict := entity_dicts[entity_idx] as Dictionary
var properties := entity_dict['properties'] as Dictionary
if 'classname' in properties:
var classname = properties['classname']
if classname in entity_definitions:
var entity_definition := entity_definitions[classname] as QodotFGDClass
for property in properties:
var prop_string = properties[property]
if property in entity_definition.class_properties:
var prop_default = entity_definition.class_properties[property]
if prop_default is int:
properties[property] = prop_string.to_int()
elif prop_default is float:
properties[property] = prop_string.to_float()
elif prop_default is Vector3:
var prop_comps = prop_string.split_floats(" ")
if prop_comps.size() > 2:
properties[property] = Vector3(prop_comps[0], prop_comps[1], prop_comps[2])
else:
push_error("Invalid vector format for \'" + property + "\' in entity \'" + classname + "\': " + prop_string)
properties[property] = prop_default
elif prop_default is Color:
var prop_color = prop_default
var prop_comps = prop_string.split(" ")
if prop_comps.size() > 2:
if "." in prop_comps[0] or "." in prop_comps[1] or "." in prop_comps[2]:
prop_color.r = prop_comps[0].to_float()
prop_color.g = prop_comps[1].to_float()
prop_color.b = prop_comps[2].to_float()
else:
prop_color.r8 = prop_comps[0].to_int()
prop_color.g8 = prop_comps[1].to_int()
prop_color.b8 = prop_comps[2].to_int()
else:
push_error("Invalid color format for \'" + property + "\' in entity \'" + classname + "\': " + prop_string)
properties[property] = prop_color
elif prop_default is Dictionary:
properties[property] = prop_string.to_int()
elif prop_default is Array:
properties[property] = prop_string.to_int()
# Assign properties not defined with defaults from the entity definition
for property in entity_definitions[classname].class_properties:
if not property in properties:
var prop_default = entity_definition.class_properties[property]
# Flags
if prop_default is Array:
var prop_flags_sum := 0
for prop_flag in prop_default:
if prop_flag is Array and prop_flag.size() > 2:
if prop_flag[2] and prop_flag[1] is int:
prop_flags_sum += prop_flag[1]
properties[property] = prop_flags_sum
# Choices
elif prop_default is Dictionary:
var prop_desc = entity_definition.class_property_descriptions[property]
if prop_desc is Array and prop_desc.size() > 1 and prop_desc[1] is int:
properties[property] = prop_desc[1]
else:
properties[property] = 0
# Everything else
else:
properties[property] = prop_default
if 'properties' in entity_node:
entity_node.properties = properties
## Wire signals based on Trenchbroom [code]target[/code] and [code]targetname[/code] properties
func connect_signals() -> void:
for entity_idx in range(0, entity_nodes.size()):
var entity_node = entity_nodes[entity_idx]
if not entity_node:
continue
var entity_dict := entity_dicts[entity_idx] as Dictionary
var entity_properties := entity_dict['properties'] as Dictionary
if not 'target' in entity_properties:
continue
var target_nodes := get_nodes_by_targetname(entity_properties['target'])
for target_node in target_nodes:
connect_signal(entity_node, target_node)
## Connect a signal on [code]entity_node[/code] to [code]target_node[/code], possibly mediated by the contents of a [code]signal[/code] or [code]receiver[/code] entity
func connect_signal(entity_node: Node, target_node: Node) -> void:
if target_node.properties['classname'] == 'signal':
var signal_name = target_node.properties['signal_name']
var receiver_nodes := get_nodes_by_targetname(target_node.properties['target'])
for receiver_node in receiver_nodes:
if receiver_node.properties['classname'] != 'receiver':
continue
var receiver_name = receiver_node.properties['receiver_name']
var target_nodes := get_nodes_by_targetname(receiver_node.properties['target'])
for node in target_nodes:
entity_node.connect(signal_name,Callable(node,receiver_name),CONNECT_PERSIST)
else:
var signal_list = entity_node.get_signal_list()
for signal_dict in signal_list:
if signal_dict['name'] == 'trigger':
entity_node.connect("trigger",Callable(target_node,"use"),CONNECT_PERSIST)
break
## Remove nodes marked transient. See [member QodotFGDClass.transient_node]
func remove_transient_nodes() -> void:
for entity_idx in range(0, entity_nodes.size()):
var entity_node = entity_nodes[entity_idx]
if not entity_node:
continue
var entity_dict := entity_dicts[entity_idx] as Dictionary
var entity_properties := entity_dict['properties'] as Dictionary
if not 'classname' in entity_properties:
continue
var classname = entity_properties['classname']
if not classname in entity_definitions:
continue
var entity_definition = entity_definitions[classname]
if entity_definition.transient_node:
entity_node.get_parent().remove_child(entity_node)
entity_node.queue_free()
## Find all nodes with matching targetname property
func get_nodes_by_targetname(targetname: String) -> Array:
var nodes := []
for node_idx in range(0, entity_nodes.size()):
var node = entity_nodes[node_idx]
if not node:
continue
var entity_dict := entity_dicts[node_idx] as Dictionary
var entity_properties := entity_dict['properties'] as Dictionary
if not 'targetname' in entity_properties:
continue
if entity_properties['targetname'] == targetname:
nodes.append(node)
return nodes
# Cleanup after build is finished (internal)
func _build_complete():
reset_build_context()
stop_profile('build_map')
if not print_profiling_data:
print('\n')
print('Build complete\n')
emit_signal("build_complete")