Pixel-Composer/scripts/render_data/render_data.gml

350 lines
11 KiB
Text
Raw Normal View History

enum RENDER_TYPE {
none = 0,
partial = 1,
full = 2
}
2023-03-28 06:58:28 +02:00
#region globalvar
2023-12-25 12:51:12 +01:00
globalvar UPDATE, RENDER_QUEUE, RENDER_ORDER, UPDATE_RENDER_ORDER, LIVE_UPDATE;
LIVE_UPDATE = false;
2023-12-03 05:02:04 +01:00
UPDATE_RENDER_ORDER = false;
2023-11-17 06:24:31 +01:00
2023-10-02 10:45:30 +02:00
#macro RENDER_ALL_REORDER UPDATE_RENDER_ORDER = true; UPDATE |= RENDER_TYPE.full;
#macro RENDER_ALL UPDATE |= RENDER_TYPE.full;
#macro RENDER_PARTIAL UPDATE |= RENDER_TYPE.partial;
global.getvalue_hit = 0;
2023-03-28 06:58:28 +02:00
#endregion
2022-12-13 09:20:36 +01:00
2023-10-02 10:45:30 +02:00
function ResetAllNodesRender() { #region
2023-10-09 16:07:33 +02:00
LOG_IF(global.FLAG.render == 1, $"XXXXXXXXXXXXXXXXXXXX RESETTING ALL NODES [frame {CURRENT_FRAME}] XXXXXXXXXXXXXXXXXXXX");
2023-10-07 16:23:40 +02:00
array_foreach(PROJECT.allNodes, function(_node) {
2024-07-08 05:55:23 +02:00
if(!is_instanceof(_node, Node)) return;
2023-10-02 10:45:30 +02:00
_node.setRenderStatus(false);
2024-08-08 06:57:51 +02:00
for( var i = 0, n = array_length(_node.inputs); i < n; i++ )
_node.inputs[i].resetCache();
2024-07-08 05:55:23 +02:00
return;
});
2023-10-02 10:45:30 +02:00
} #endregion
2022-12-16 09:18:09 +01:00
2023-10-07 16:23:40 +02:00
function NodeTopoSort() { #region
LOG_IF(global.FLAG.render == 1, $"======================= RESET TOPO =======================")
2024-04-03 09:40:37 +02:00
var amo = array_length(PROJECT.allNodes);
2024-01-26 14:38:50 +01:00
var _t = get_timer();
array_foreach(PROJECT.allNodes, function(_node) {
2023-12-22 04:15:04 +01:00
_node.clearTopoSorted();
return 0;
});
2023-10-07 16:23:40 +02:00
PROJECT.nodeTopo = [];
2024-03-14 14:35:19 +01:00
__topoSort(PROJECT.nodeTopo, PROJECT.nodes);
2023-10-08 04:14:35 +02:00
LOG_IF(global.FLAG.render == 1, $"+++++++ Topo Sort Completed: {array_length(PROJECT.nodeTopo)}/{amo} nodes sorted in {(get_timer() - _t) / 1000} ms +++++++");
2023-10-07 16:23:40 +02:00
} #endregion
2024-06-25 11:53:40 +02:00
function NodeListSort(_nodeList) { #region
array_foreach(_nodeList, function(node) {
node.clearTopoSorted();
return 0;
});
2024-04-27 05:10:20 +02:00
2024-06-25 11:53:40 +02:00
var _arr = [];
__topoSort(_arr, _nodeList);
return _arr;
2024-04-03 09:40:37 +02:00
} #endregion
function __sortNode(_arr, _node) { #region
2024-02-11 14:53:33 +01:00
if(_node.topoSorted) return;
2024-04-03 09:40:37 +02:00
var _parents = [];
var _prev = _node.getPreviousNodes();
2024-02-11 14:53:33 +01:00
for( var i = 0, n = array_length(_prev); i < n; i++ ) {
var _in = _prev[i];
if(_in == noone || _in.topoSorted) continue;
2024-02-11 14:53:33 +01:00
2024-04-03 09:40:37 +02:00
array_push(_parents, _in);
2024-02-11 14:53:33 +01:00
}
2024-04-27 05:10:20 +02:00
// print($" > Checking {_node.name}: {array_length(_parents)}");
2024-02-11 14:53:33 +01:00
2024-04-03 09:40:37 +02:00
if(is_instanceof(_node, Node_Collection) && !_node.managedRenderOrder)
__topoSort(_arr, _node.nodes);
2024-04-03 09:40:37 +02:00
for( var i = 0, n = array_length(_parents); i < n; i++ )
__sortNode(_arr, _parents[i]);
2024-02-11 14:53:33 +01:00
if(!_node.topoSorted) {
array_push(_arr, _node);
2024-02-11 14:53:33 +01:00
_node.topoSorted = true;
2024-04-27 05:10:20 +02:00
// print($" > Adding > {_node.name}");
2024-02-11 14:53:33 +01:00
}
} #endregion
function __topoSort(_arr, _nodeArr) { #region
2024-04-27 05:10:20 +02:00
var _root = [];
2023-12-22 04:15:04 +01:00
var _leftOver = [];
var _global = _nodeArr == PROJECT.nodes;
__temp_nodeList = _nodeArr;
2023-10-07 16:23:40 +02:00
for( var i = 0, n = array_length(_nodeArr); i < n; i++ ) {
var _node = _nodeArr[i];
2024-04-27 05:10:20 +02:00
var _isRoot = true;
2023-10-07 16:23:40 +02:00
2023-12-22 04:15:04 +01:00
if(is_instanceof(_node, Node_Collection_Inline) && !_node.is_root) {
array_push(_leftOver, _node);
continue;
}
if(_node.attributes.show_update_trigger && !array_empty(_node.updatedOutTrigger.getJunctionTo())) {
_isRoot = false;
2023-12-20 14:02:49 +01:00
} else {
2024-08-08 06:57:51 +02:00
for( var j = 0, m = array_length(_node.outputs); j < m; j++ ) {
var _to = _node.outputs[j].getJunctionTo();
if(_global) _isRoot &= array_empty(_to);
else _isRoot &= !array_any(_to, function(_val) { return array_exists(__temp_nodeList, _val.node); } );
if(!_isRoot) break;
2023-10-07 16:23:40 +02:00
}
}
2024-04-27 05:10:20 +02:00
if(_isRoot) array_push(_root, _node);
2023-10-07 16:23:40 +02:00
}
2024-04-27 05:10:20 +02:00
// print($"Root: {_root}");
for( var i = 0, n = array_length(_root); i < n; i++ )
__sortNode(_arr, _root[i]);
2023-12-22 04:15:04 +01:00
for( var i = 0, n = array_length(_leftOver); i < n; i++ ) {
if(!_leftOver[i].topoSorted)
array_insert(_arr, 0, _leftOver[i]);
2023-12-22 04:15:04 +01:00
}
2023-10-07 16:23:40 +02:00
} #endregion
function __nodeLeafList(_arr) { #region
var nodes = [];
var nodeNames = [];
for( var i = 0, n = array_length(_arr); i < n; i++ ) {
var _node = _arr[i];
if(!_node.active) { LOG_LINE_IF(global.FLAG.render == 1, $"Reject {_node.internalName} [inactive]"); continue; }
if(!_node.isLeafList(_arr)) { LOG_LINE_IF(global.FLAG.render == 1, $"Reject {_node.internalName} [not leaf]"); continue; }
if(!_node.isRenderable()) { LOG_LINE_IF(global.FLAG.render == 1, $"Reject {_node.internalName} [not renderable]"); continue; }
array_push(nodes, _node);
array_push(nodeNames, _node.internalName);
}
LOG_LINE_IF(global.FLAG.render == 1, $"Push node {nodeNames} to queue");
return nodes;
} #endregion
function __nodeIsRenderLeaf(_node) { #region
if(is_undefined(_node)) { LOG_IF(global.FLAG.render == 1, $"Skip undefiend [{_node}]"); return false; }
if(!is_instanceof(_node, Node)) { LOG_IF(global.FLAG.render == 1, $"Skip non-node [{_node}]"); return false; }
2023-12-22 04:15:04 +01:00
2023-11-29 03:04:28 +01:00
if(_node.is_group_io) { LOG_IF(global.FLAG.render == 1, $"Skip group IO [{_node.internalName}]"); return false; }
2023-12-22 04:15:04 +01:00
if(!_node.active) { LOG_IF(global.FLAG.render == 1, $"Skip inactive [{_node.internalName}]"); return false; }
2023-11-19 14:34:56 +01:00
if(!_node.isRenderActive()) { LOG_IF(global.FLAG.render == 1, $"Skip render inactive [{_node.internalName}]"); return false; }
if(!_node.attributes.update_graph) { LOG_IF(global.FLAG.render == 1, $"Skip non-auto update [{_node.internalName}]"); return false; }
if(_node.passiveDynamic) { _node.forwardPassiveDynamic(); LOG_IF(global.FLAG.render == 1, $"Skip passive dynamic [{_node.internalName}]"); return false; }
2023-11-19 14:34:56 +01:00
if(!_node.isActiveDynamic()) { LOG_IF(global.FLAG.render == 1, $"Skip rendered static [{_node.internalName}]"); return false; }
if(_node.inline_context != noone && _node.inline_context.managedRenderOrder) return false;
return true;
} #endregion
2023-10-02 10:45:30 +02:00
function Render(partial = false, runAction = false) { #region
2024-04-30 05:57:34 +02:00
LOG_END();
LOG_BLOCK_START();
2023-10-09 16:07:33 +02:00
LOG_IF(global.FLAG.render, $"============================== RENDER START [{partial? "PARTIAL" : "FULL"}] [frame {CURRENT_FRAME}] ==============================");
2023-03-28 06:58:28 +02:00
// global.getvalue_hit = 0;
2023-02-14 02:51:14 +01:00
try {
var t = get_timer();
var t1 = get_timer();
var _render_time = 0;
var _leaf_time = 0;
2023-02-14 02:51:14 +01:00
var rendering = noone;
2023-06-01 10:32:21 +02:00
var error = 0;
var reset_all = !partial;
2024-08-08 06:57:51 +02:00
var renderable;
2023-03-28 06:58:28 +02:00
2023-06-01 10:32:21 +02:00
if(reset_all) {
LOG_IF(global.FLAG.render == 1, $"xxxxxxxxxx Resetting {array_length(PROJECT.nodeTopo)} nodes xxxxxxxxxx");
2023-12-22 04:15:04 +01:00
for (var i = 0, n = array_length(PROJECT.allNodes); i < n; i++) {
var _node = PROJECT.allNodes[i];
2023-02-14 02:51:14 +01:00
_node.setRenderStatus(false);
}
2022-12-10 05:06:01 +01:00
}
2023-10-02 10:45:30 +02:00
2023-02-14 02:51:14 +01:00
// get leaf node
LOG_IF(global.FLAG.render == 1, $"----- Finding leaf from {array_length(PROJECT.nodeTopo)} nodes -----");
2023-06-01 10:32:21 +02:00
RENDER_QUEUE.clear();
for( var i = 0, n = array_length(PROJECT.nodeTopo); i < n; i++ ) {
var _node = PROJECT.nodeTopo[i];
2023-10-07 16:23:40 +02:00
_node.passiveDynamic = false;
}
for( var i = 0, n = array_length(PROJECT.nodeTopo); i < n; i++ ) {
var _node = PROJECT.nodeTopo[i];
2023-10-02 14:41:44 +02:00
_node.render_time = 0;
if(!__nodeIsRenderLeaf(_node))
continue;
2023-10-07 16:23:40 +02:00
LOG_IF(global.FLAG.render == 1, $" Found leaf [{_node.internalName}]");
RENDER_QUEUE.enqueue(_node);
_node.forwardPassiveDynamic();
2022-12-10 05:06:01 +01:00
}
_leaf_time = get_timer() - t;
LOG_IF(global.FLAG.render >= 1, $"Get leaf complete: found {RENDER_QUEUE.size()} leaves in {(get_timer() - t) / 1000} ms."); t = get_timer();
LOG_IF(global.FLAG.render == 1, "================== Start rendering ==================");
2023-02-14 02:51:14 +01:00
// render forward
2023-06-01 10:32:21 +02:00
while(!RENDER_QUEUE.empty()) {
LOG_BLOCK_START();
LOG_IF(global.FLAG.render == 1, $"➤➤➤➤➤➤ CURRENT RENDER QUEUE {RENDER_QUEUE} [{RENDER_QUEUE.size()}] ");
2023-10-07 16:23:40 +02:00
2024-08-08 06:57:51 +02:00
rendering = RENDER_QUEUE.dequeue();
renderable = rendering.isRenderable();
LOG_IF(global.FLAG.render == 1, $"Rendering {rendering.internalName} ({rendering.display_name}) : {renderable? "Update" : "Pass"} ({rendering.rendered})");
2023-03-29 15:02:03 +02:00
2023-04-04 13:44:37 +02:00
if(renderable) {
var _render_pt = get_timer();
2024-07-07 09:08:13 +02:00
rendering.doUpdate();
_render_time += get_timer() - _render_pt;
2023-03-02 07:59:14 +01:00
2023-03-28 06:58:28 +02:00
var nextNodes = rendering.getNextNodes();
2023-10-02 10:45:30 +02:00
for( var i = 0, n = array_length(nextNodes); i < n; i++ ) {
var nextNode = nextNodes[i];
if(!is_instanceof(nextNode, __Node_Base)) continue;
if(!nextNode.isRenderable()) continue;
RENDER_QUEUE.enqueue(nextNode);
2023-10-02 10:45:30 +02:00
}
2023-03-02 07:59:14 +01:00
2023-03-28 06:58:28 +02:00
if(runAction && rendering.hasInspector1Update())
rendering.inspector1Update();
2023-12-10 14:55:05 +01:00
} else if(rendering.force_requeue)
RENDER_QUEUE.enqueue(rendering);
2023-03-29 15:02:03 +02:00
LOG_BLOCK_END();
2023-03-02 07:59:14 +01:00
}
_render_time /= 1000;
2023-11-17 06:24:31 +01:00
LOG_IF(global.FLAG.renderTime || global.FLAG.render >= 1, $"=== RENDER FRAME {CURRENT_FRAME} COMPLETE IN {(get_timer() - t1) / 1000} ms ===\n");
LOG_IF(global.FLAG.render > 1, $"=== RENDER SUMMARY STA ===");
LOG_IF(global.FLAG.render > 1, $" total time: {(get_timer() - t1) / 1000} ms");
LOG_IF(global.FLAG.render > 1, $" leaf: {_leaf_time / 1000} ms");
LOG_IF(global.FLAG.render > 1, $" render loop: {(get_timer() - t) / 1000} ms");
LOG_IF(global.FLAG.render > 1, $" render only: {_render_time} ms");
LOG_IF(global.FLAG.render > 1, $"=== RENDER SUMMARY END ===");
2023-05-08 10:50:42 +02:00
} catch(e) {
2023-03-02 07:59:14 +01:00
noti_warning(exception_print(e));
2023-05-08 10:50:42 +02:00
}
2023-03-28 06:58:28 +02:00
// print("\n============== render stat ==============");
// print($"Get value hit: {global.getvalue_hit}");
LOG_END();
2024-07-07 09:08:13 +02:00
2023-10-02 10:45:30 +02:00
} #endregion
2023-03-02 07:59:14 +01:00
function __renderListReset(arr) { #region
for( var i = 0; i < array_length(arr); i++ ) {
2024-06-25 11:53:40 +02:00
arr[i].setRenderStatus(false);
2023-03-24 05:55:34 +01:00
2024-06-25 11:53:40 +02:00
if(struct_has(arr[i], "nodes"))
__renderListReset(arr[i].nodes);
2023-03-02 07:59:14 +01:00
}
2023-10-02 10:45:30 +02:00
} #endregion
2023-03-02 07:59:14 +01:00
function RenderList(arr) { #region
2023-04-23 16:47:33 +02:00
LOG_BLOCK_START();
LOG_IF(global.FLAG.render == 1, $"=============== RENDER LIST START [{array_length(arr)}] ===============");
2023-04-23 16:47:33 +02:00
var queue = ds_queue_create();
try {
var rendering = noone;
var error = 0;
var t = current_time;
__renderListReset(arr);
2023-04-23 16:47:33 +02:00
// get leaf node
for( var i = 0, n = array_length(arr); i < n; i++ ) {
var _node = arr[i];
2023-10-07 16:23:40 +02:00
_node.passiveDynamic = false;
}
for( var i = 0, n = array_length(arr); i < n; i++ ) {
var _node = arr[i];
2023-04-23 16:47:33 +02:00
2023-12-06 05:09:39 +01:00
if(!__nodeIsRenderLeaf(_node))
continue;
2023-10-07 16:23:40 +02:00
LOG_IF(global.FLAG.render == 1, $"Found leaf {_node.internalName}");
ds_queue_enqueue(queue, _node);
_node.forwardPassiveDynamic();
2023-04-23 16:47:33 +02:00
}
LOG_IF(global.FLAG.render == 1, "Get leaf complete: found " + string(ds_queue_size(queue)) + " leaves.");
LOG_IF(global.FLAG.render == 1, "=== Start rendering ===");
2023-04-23 16:47:33 +02:00
// render forward
while(!ds_queue_empty(queue)) {
2023-06-13 14:42:06 +02:00
LOG_BLOCK_START();
2023-10-07 16:23:40 +02:00
rendering = ds_queue_dequeue(queue)
2023-06-13 14:42:06 +02:00
var renderable = rendering.isRenderable();
LOG_IF(global.FLAG.render == 1, $"Rendering {rendering.internalName} ({rendering.display_name}) : {renderable? "Update" : "Pass"}");
2023-04-23 16:47:33 +02:00
2023-06-13 14:42:06 +02:00
if(renderable) {
rendering.doUpdate();
2023-04-23 16:47:33 +02:00
2023-06-13 14:42:06 +02:00
var nextNodes = rendering.getNextNodes();
2023-12-09 09:00:35 +01:00
for( var i = 0, n = array_length(nextNodes); i < n; i++ ) {
var _node = nextNodes[i];
2024-06-25 11:53:40 +02:00
if(array_exists(arr, _node) && _node.isRenderable())
2023-12-09 09:00:35 +01:00
ds_queue_enqueue(queue, _node);
}
2023-10-07 16:23:40 +02:00
}
2023-06-13 14:42:06 +02:00
LOG_BLOCK_END();
2023-04-23 16:47:33 +02:00
}
2023-05-08 10:50:42 +02:00
} catch(e) {
2023-04-23 16:47:33 +02:00
noti_warning(exception_print(e));
2023-05-08 10:50:42 +02:00
}
2024-04-30 05:57:34 +02:00
LOG_BLOCK_END();
LOG_IF(global.FLAG.render == 1, "=== RENDER COMPLETE ===\n");
2023-04-23 16:47:33 +02:00
LOG_END();
ds_queue_destroy(queue);
2023-10-02 10:45:30 +02:00
} #endregion