-
Notifications
You must be signed in to change notification settings - Fork 7
Entities
You'll need to create your own entities for your game. If you want to use entities from Half-Life 2 then you should implement entities logic on your own.
All entities should be in @tool
mode, extended from ValveIONode
and placed in the folder that you assigned in the config (entities_folder
).
- Create script of an entity.
- Create a 3d scene with the same name and assign the script.
- Try to import your map with entity.
Inputs of entities are just methods with a single parameter that passed from outputs
Outputs should be called by method
trigger_output
(from signal for example)
In case you don't want to extend your entity from ValveIONode
, just define static function called setup
. In this case it's not necessary to make the script in @tool
mode. Also you won't be able to use flags and other features from ValveIONode
.
class_name SomeEntity extends Node3D
var entity: Dictionary = {};
static func setup(entity_structure: Dictionary, instance: SomeEntity):
instance.entity = entity_structure; # This is required
instance.transform = ValveIONode.get_entity_transform(entity_structure);
instance.basis *= Basis(Vector3.UP, -PI / 2);
## func_button.gd
@tool
extends ValveIONode
signal interact();
const FLAG_ONCE = 2;
const FLAG_STARTS_LOCKED = 2048;
var isLocked = false;
var isUsed = false;
var sound = null;
var lockedSound = null;
## Use this instead _ready
func _entity_ready():
isLocked = has_flag(FLAG_STARTS_LOCKED);
# Once we trigger this signal we triggering the entity's outputs.
interact.connect(func ():
if isLocked:
if lockedSound:
SoundManager.PlaySound(global_position, lockedSound, 0.05);
trigger_output("OnUseLocked")
else:
if has_flag(FLAG_ONCE) and isUsed:
return;
if sound:
SoundManager.PlaySound(global_position, sound, 0.05);
trigger_output("OnPressed")
isUsed = true);
if "sound" in entity:
sound = load("res://Assets/Sounds/" + entity.sound);
if "locked_sound" in entity:
lockedSound = load("res://Assets/Sounds/" + entity.locked_sound);
# This method will be called during import
func _apply_entity(entityInfo: Dictionary):
super._apply_entity(entityInfo, vmfNode);
# Getting entity's brush geometry and assigning it
var mesh = get_mesh();
$MeshInstance3D.set_mesh(mesh);
# Generating collision for the assigned mesh.
$MeshInstance3D/StaticBody3D/CollisionShape3D.shape = mesh.create_convex_shape();
## INPUTS
func Lock(_param = null):
isLocked = true;
func Unlock(_param = null):
isLocked = false;
In case you don't want to place some entities inside the entities folder you can use aliases:
// vmf.config.json
{
"import": {
"entity_aliases": {
"npc_ghost": "res://objects/npc_ghost/npc_ghost.tscn"
}
}
}
The Base of all entities. Contains I/O logic and some useful methods for entities.
All entities extends from this node will always have these inputs so you don't need to define it.
- Toggle - toggles
enabled
field of the entity - Enable - enable the entity
- Disable - disables and blocks outputs for the entity
- Kill - removes the node from tree
Reference:
- _entity_ready
- _apply_entity
- has_flag
- trigger_output
- get_mesh
- get_entity_shape
- get_entity_convex_shape
- get_entity_trimesh_shape
- get_entity_basis
- get_movement_vector
- convert_vector
- convert_direction
- define_alias (static)
- get_target
- get_all_targets
- get_entity_transform (static)
- get_entity_basis (static)
- get_separated_collisions
- define_alias (static)
Means that all outputs and reparents are ready to use. Use this method instead of _ready
.
This method called by VMFNode during import entities and needs to make some setup for entities, such as assigning brushes, generation collisions and so on.
Don't forget to call super._apply_entity
before making any changes in the node.
func _apply_entity(entityInfo: Dictionary):
super._apply_entity(entityInfo, vmfNode);
# Getting a mesh from the solid data of the entity and assigning
var mesh = get_mesh();
$MeshInstance3D.set_mesh(mesh);
# Generating a collision shape for the mesh
$MeshInstance3D/StaticBody3D/CollisionShape3D.shape = mesh.create_convex_shape();
Checks the spawnflags
field of the entity.
const FLAG_STARTS_LOCKED = 2048;
var isLocked = false;
func _entity_ready():
if has_flag(FLAG_STARTS_LOCKED):
isLocked = true;
Triggers outputs that defined in the entity.
interact.connect(func():
if isLocked:
trigger_output("OnUseLocked");
else:
trigger_output("OnPressed"));
Returns entity's solids as ArrayMesh. cleanup
means that the mesh will be cleaned from unnecessary faces
Returns optimized collision shape for the entity's solids (trimesh or convex. Depends of brushes count inside of entity). Uses CSGMesh for shape generation
Returns a convex shape for the entity's brushes
Returns optimized trimesh shape for the entity's brushes
Returns rotation state for specified entity.
Returns directional vector from specified vector. If an entity has movement direction property (i.e. func_door, func_button) use this function to convert the direction.
Converts Vector3 of position from Z-up to Y-up.
Converts Vector3 of rotation from Z-up to Y-up.
Defines global alias to node for using in I/O.
# player.gd
func _entity_ready():
ValveIONode.define_alias('!player', self);
Returns first node by target name assigned in entity.
Returns all nodes by target name assigned in entities.
Returns a list of collisions for each brush.
Returns Transform3D of entity
Returns Basis of entity