RoomManager
Inherits: Spatial < Node < Object
The RoomManager node is used to control the portal culling system.
Description
In order to utilize the portal occlusion culling system, you must build your level using Rooms and Portals. Before these can be used at runtime, they must undergo a short conversion process to build the room graph
, runtime data needed for portal culling. The room graph
is controlled by the RoomManager
node, and the RoomManager
also contains settings that are common throughout the portal system.
Properties
| ||
| ||
| ||
| ||
| ||
| ||
| ||
| ||
process_priority |
| |
| ||
| ||
| ||
| ||
| ||
|
Methods
void | rooms_clear ( ) |
void | rooms_convert ( ) |
Enumerations
enum PVSMode:
PVS_MODE_DISABLED = 0 —- Use only Portals at runtime to determine visibility. PVS will not be generated at Rooms conversion, and gameplay notifications cannot be used.
PVS_MODE_PARTIAL = 1 —- Use a combination of PVS and Portals to determine visibility (this is usually fastest and most accurate).
PVS_MODE_FULL = 2 —- Use only the PVS (potentially visible set) of Rooms to determine visibility.
Property Descriptions
- bool active
Default |
|
Setter | rooms_set_active(value) |
Getter | rooms_get_active() |
Switches the portal culling system on and off.
It is important to note that when portal culling is active, it is responsible for all the 3d culling. Some editor functionality may be more difficult to use, so switching the active flag is intended to be used to make sure your Room / Portal layout works within the editor.
Switching to active
will have no effect when the room graph
is unloaded (the rooms have not yet been converted).
- bool debug_sprawl
Default |
|
Setter | set_debug_sprawl(value) |
Getter | get_debug_sprawl() |
Large objects can ‘sprawl’ over (be present in) more than one room. It can be useful to visualize which objects are sprawling outside the current room.
Toggling this setting turns this debug view on and off.
- float default_portal_margin
Default |
|
Setter | set_default_portal_margin(value) |
Getter | get_default_portal_margin() |
Usually we don’t want objects that only just cross a boundary into an adjacent Room to sprawl into that room. To prevent this, each Portal has an extra margin, or tolerance zone where objects can enter without sprawling to a neighbouring room.
In most cases you can set this here for all portals. It is possible to override the margin for each portal.
- bool gameplay_monitor
Default |
|
Setter | set_gameplay_monitor_enabled(value) |
Getter | get_gameplay_monitor_enabled() |
When using a partial or full PVS, the gameplay monitor allows you to receive callbacks when roaming objects or rooms enter or exit the gameplay area. The gameplay area is defined as either the primary, or secondary PVS.
These callbacks allow you to, for example, reduce processing for objects that are far from the player, or turn on and off AI.
You can either choose to receive callbacks as notifications through the _notification
function, or as signals.
NOTIFICATION_ENTER_GAMEPLAY
NOTIFICATION_EXIT_GAMEPLAY
Signals: "gameplay_entered"
, "gameplay_exited"
- bool merge_meshes
Default |
|
Setter | set_merge_meshes(value) |
Getter | get_merge_meshes() |
If enabled, the system will attempt to merge similar meshes (particularly in terms of materials) within Rooms during conversion. This can significantly reduce the number of drawcalls and state changes required during rendering, albeit at a cost of reduced culling granularity.
Note: This operates at runtime during the conversion process, and will only operate on exported or running projects, in order to prevent accidental alteration to the scene and loss of data.
- int overlap_warning_threshold
Default |
|
Setter | set_overlap_warning_threshold(value) |
Getter | get_overlap_warning_threshold() |
When converting rooms, the editor will warn you if overlap is detected between rooms. Overlap can interfere with determining the room that cameras and objects are within. A small amount can be acceptable, depending on your level. Here you can alter the threshold at which the editor warning appears. There are no other side effects.
- int portal_depth_limit
Default |
|
Setter | set_portal_depth_limit(value) |
Getter | get_portal_depth_limit() |
Portal rendering is recursive - each time a portal is seen through an earlier portal there is some cost. For this reason, and to prevent the possibility of infinite loops, this setting provides a hard limit on the recursion depth.
Note: This value is unused when using Full
PVS mode.
- NodePath preview_camera
Default |
|
Setter | set_preview_camera_path(value) |
Getter | get_preview_camera_path() |
Portal culling normally operates using the current Camera / Cameras, however for debugging purposes within the editor, you can use this setting to override this behaviour and force it to use a particular camera to get a better idea of what the occlusion culling is doing.
- PVSMode pvs_mode
Default |
|
Setter | set_pvs_mode(value) |
Getter | get_pvs_mode() |
Optionally during conversion the potentially visible set (PVS) of rooms that are potentially visible from each room can be calculated. This can be used either to aid in dynamic portal culling, or to totally replace portal culling.
In Full
PVS Mode, all objects within the potentially visible rooms will be frustum culled, and rendered if they are within the view frustum.
- float roaming_expansion_margin
Default |
|
Setter | set_roaming_expansion_margin(value) |
Getter | get_roaming_expansion_margin() |
In order to reduce processing for roaming objects, an expansion is applied to their AABB as they move. This expanded volume is used to calculate which rooms the roaming object is within. If the object’s exact AABB is still within this expanded volume on the next move, there is no need to reprocess the object, which can save considerable CPU.
The downside is that if the expansion is too much, the object may end up unexpectedly sprawling into neighbouring rooms and showing up where it might otherwise be culled.
In order to balance roaming performance against culling accuracy, this expansion margin can be customized by the user. It will typically depend on your room and object sizes, and movement speeds. The default value should work reasonably in most circumstances.
- float room_simplify
Default |
|
Setter | set_room_simplify(value) |
Getter | get_room_simplify() |
During the conversion process, the geometry of objects within Rooms, or a custom specified manual bound, are used to generate a convex hull bound.
This convex hull is required in the visibility system, and is used for many purposes. Most importantly, it is used to decide whether the Camera (or an object) is within a Room. The convex hull generating algorithm is good, but occasionally it can create too many (or too few) planes to give a good representation of the room volume.
The room_simplify
value can be used to gain fine control over this process. It determines how similar planes can be for them to be considered the same (and duplicates removed). The value can be set between 0 (no simplification) and 1 (maximum simplification).
The value set here is the default for all rooms, but individual rooms can override this value if desired.
The room convex hulls are shown as a wireframe in the editor.
- NodePath roomlist
Default |
|
Setter | set_roomlist_path(value) |
Getter | get_roomlist_path() |
For the Room conversion process to succeed, you must point the RoomManager
to the parent Node of your Rooms and RoomGroups, which we refer to as the roomlist
(the roomlist is not a special node type, it is normally just a Spatial).
- bool show_margins
Default |
|
Setter | set_show_margins(value) |
Getter | get_show_margins() |
Shows the Portal margins when the portal gizmo is used in the editor.
- bool use_secondary_pvs
Default |
|
Setter | set_use_secondary_pvs(value) |
Getter | get_use_secondary_pvs() |
When receiving gameplay callbacks when objects enter and exit gameplay, the gameplay area can be defined by either the primary PVS (potentially visible set) of Rooms, or the secondary PVS (the primary PVS and their neighbouring Rooms).
Sometimes using the larger gameplay area of the secondary PVS may be preferable.
Method Descriptions
- void rooms_clear ( )
This function clears all converted data from the room graph. Use this before unloading a level, when transitioning from level to level, or returning to a main menu.
- void rooms_convert ( )
This is the most important function in the whole portal culling system. Without it, the system cannot function.
First it goes through every Room that is a child of the room list
node (and RoomGroups within) and converts and adds it to the room graph
.
This works for both Room nodes, and Spatial nodes that follow a special naming convention. They should begin with the prefix ‘Room_‘, followed by the name you wish to give the room, e.g. ‘Room_lounge’. This will automatically convert such Spatials to Room nodes for you. This is useful if you want to build you entire room system in e.g. Blender, and reimport multiple times as you work on the level.
The conversion will try to assign VisualInstances that are children and grandchildren of the Room to the room. These should be given a suitable portal mode
(see the CullInstance documentation). The default portal mode
is STATIC
- objects which are not expected to move while the level is played, which will typically be most objects.
The conversion will usually use the geometry of these VisualInstances (and the Portals) to calculate a convex hull bound for the room. These bounds will be shown in the editor with a wireframe. Alternatively you can specify a manual custom bound for any room, see the Room documentation.
By definition, Cameras within a room can see everything else within the room (that is one advantage to using convex hulls). However, in order to see from one room into adjacent rooms, you must place Portals, which represent openings that the camera can see through, like windows and doors.
Portals are really just specialized MeshInstances. In fact you will usually first create a portal by creating a MeshInstance, especially a plane
mesh instance. You would move the plane in the editor to cover a window or doorway, with the front face pointing outward from the room. To let the conversion process know you want this mesh to be a portal, again we use a special naming convention. MeshInstances to be converted to a Portal should start with the prefix ‘Portal_‘.
You now have a choice - you can leave the name as ‘Portal_‘ and allow the system to automatically detect the nearest Room to link. In most cases this will work fine.
An alternative method is to specify the Room to link to manually, appending a suffix to the portal name, which should be the name of the room you intend to link to. For example ‘Portal_lounge’ will attempt to link to the room named ‘Room_lounge’.
There is a special case here - Godot does not allow two nodes to share the same name. What if you want to manually have more than one portal leading into the same room? Surely they will need to both be called, e.g. ‘Portal_lounge’?
The solution is a wildcard character. After the room name, if you use the character ‘\‘, this character and anything following it will be ignored. So you can use for example ‘Portal_lounge*0’, ‘Portal_lounge*1’* etc.
Note that Portals that have already been converted to Portal nodes (rather than MeshInstances) still need to follow the same naming convention, as they will be relinked each time during conversion.
It is recommended that you only place objects in rooms that are desired to stay within those rooms - i.e. portal mode
s STATIC
or DYNAMIC
(not crossing portals). GLOBAL
and ROAMING
objects are best placed in another part of the scene tree, to avoid confusion. See CullInstance for a full description of portal modes.