enum RENDER_TYPE { none = 0, partial = 1, full = 2 } #region globalvar globalvar UPDATE, RENDER_QUEUE, RENDER_ORDER, UPDATE_RENDER_ORDER; UPDATE_RENDER_ORDER = false; global.FLAG.render = 0; global.group_io = [ "Node_Group_Input", "Node_Group_Output", "Node_Feedback_Input", "Node_Feedback_Output", "Node_Iterator_Input", "Node_Iterator_Output", "Node_Iterator_Each_Input", "Node_Iterator_Each_Output", "Node_Iterator_Filter_Input", "Node_Iterator_Filter_Output", "Node_Tunnel_In", "Node_Tunnel_Out" ]; #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; #endregion function __nodeLeafList(_list) { #region var nodes = []; var nodeNames = []; for( var i = 0, n = ds_list_size(_list); i < n; i++ ) { var _node = _list[| i]; if(!_node.active) continue; if(!_node.isRenderActive()) continue; var _startNode = _node.isRenderable(); if(_startNode) { 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 __nodeIsLoop(_node) { #region switch(instanceof(_node)) { case "Node_Iterate" : case "Node_Iterate_Each" : case "Node_Iterate_Filter" : case "Node_Iterate_Sort" : return true; } return false; } #endregion function __nodeInLoop(_node) { #region var gr = _node.group; while(gr != noone) { if(__nodeIsLoop(gr)) return true; gr = gr.group; } return false; } #endregion function ResetAllNodesRender() { #region LOG_IF(global.FLAG.render == 1, $"XXXXXXXXXXXXXXXXXXXX RESETTING ALL NODES [frame {CURRENT_FRAME}] XXXXXXXXXXXXXXXXXXXX"); var _key = ds_map_find_first(PROJECT.nodeMap); var amo = ds_map_size(PROJECT.nodeMap); repeat(amo) { var _node = PROJECT.nodeMap[? _key]; _node.setRenderStatus(false); for( var i = 0, n = ds_list_size(_node.inputs); i < n; i++ ) _node.inputs[| i].resetCache(); _key = ds_map_find_next(PROJECT.nodeMap, _key); } } #endregion function NodeTopoSort() { #region var _key = ds_map_find_first(PROJECT.nodeMap); var amo = ds_map_size(PROJECT.nodeMap); repeat(amo) { var _node = PROJECT.nodeMap[? _key]; _node.topoSorted = false; _key = ds_map_find_next(PROJECT.nodeMap, _key); } ds_list_clear(PROJECT.nodeTopo); __sortGraph(PROJECT.nodeTopo, PROJECT.nodes); LOG_IF(global.FLAG.render == 1, $"+++++++ Topo Sort Completed: {ds_list_size(PROJECT.nodeTopo)} nodes sorted +++++++"); } #endregion function __sortGraph(_list, _nodeList) { #region var _root = []; for( var i = 0, n = ds_list_size(_nodeList); i < n; i++ ) { var _node = _nodeList[| i]; var _isRoot = true; for( var j = 0, m = ds_list_size(_node.outputs); j < m; j++ ) { if(!array_empty(_node.outputs[| j].getJunctionTo())) { _isRoot = false; break; } } if(_isRoot) array_push(_root, _node); } var _st = ds_stack_create(); for( var i = 0, n = array_length(_root); i < n; i++ ) ds_stack_push(_st, _root[i]); while(!ds_stack_empty(_st)) { var _node = ds_stack_pop(_st); if(_node.topoSorted) continue; var _isLeaf = true; var _childs = []; for( var i = 0, n = ds_list_size(_node.inputs); i < n; i++ ) { var _in = _node.inputs[| i]; if(_in.value_from == noone) continue; if(_in.value_from.node.topoSorted) continue; array_push(_childs, _in.value_from.node); _isLeaf = false; } if(_isLeaf) { ds_list_add(_list, _node); _node.topoSorted = true; if(is_instanceof(_node, Node_Collection)) __sortGraph(_list, _node.nodes); } else { ds_stack_push(_st, _node); for( var i = 0, n = array_length(_childs); i < n; i++ ) ds_stack_push(_st, _childs[i]); } } } #endregion function Render(partial = false, runAction = false) { #region LOG_BLOCK_START(); LOG_IF(global.FLAG.render, $"============================== RENDER START [{partial? "PARTIAL" : "FULL"}] [frame {CURRENT_FRAME}] =============================="); try { var t = get_timer(); var t1 = get_timer(); var _render_time = 0; var _leaf_time = 0; var rendering = noone; var error = 0; var reset_all = !partial || ALWAYS_FULL; if(reset_all) { var _key = ds_map_find_first(PROJECT.nodeMap); var amo = ds_map_size(PROJECT.nodeMap); repeat(amo) { var _node = PROJECT.nodeMap[? _key]; _node.setRenderStatus(false); _key = ds_map_find_next(PROJECT.nodeMap, _key); } } // get leaf node LOG_IF(global.FLAG.render == 1, $"----- Finding leaf from {ds_list_size(PROJECT.nodeTopo)} nodes -----"); RENDER_QUEUE.clear(); for( var i = 0, n = ds_list_size(PROJECT.nodeTopo); i < n; i++ ) { var _node = PROJECT.nodeTopo[| i]; _node.passiveDynamic = false; } for( var i = 0, n = ds_list_size(PROJECT.nodeTopo); i < n; i++ ) { var _node = PROJECT.nodeTopo[| i]; if(is_undefined(_node)) { LOG_IF(global.FLAG.render == 1, $"Skip undefiend [{_node}]"); continue; } if(!is_instanceof(_node, Node)) { LOG_IF(global.FLAG.render == 1, $"Skip non-node [{_node}]"); continue; } _node.render_time = 0; if(array_exists(global.group_io, instanceof(_node))) { LOG_IF(global.FLAG.render == 1, $"Skip group IO [{_node.internalName}]"); continue; } if(!_node.active) { LOG_IF(global.FLAG.render == 1, $"Skip inactive [{_node.internalName}]"); continue; } if(!_node.isRenderActive()) { LOG_IF(global.FLAG.render == 1, $"Skip non-renderActive [{_node.internalName}]"); continue; } if(!_node.attributes.update_graph) { LOG_IF(global.FLAG.render == 1, $"Skip non-auto update [{_node.internalName}]"); continue; } if(__nodeInLoop(_node)) { LOG_IF(global.FLAG.render == 1, $"Skip in-loop [{_node.internalName}]"); continue; } if(_node.passiveDynamic) { _node.forwardPassiveDynamic(); LOG_IF(global.FLAG.render == 1, $"Skip passive dynamic [{_node.internalName}]"); continue; } if(_node.rendered && !_node.isActiveDynamic()) { LOG_IF(global.FLAG.render == 1, $"Skip rendered static [{_node.internalName}]"); continue; } LOG_IF(global.FLAG.render == 1, $" Found leaf [{_node.internalName}]"); RENDER_QUEUE.enqueue(_node); _node.forwardPassiveDynamic(); } _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 =================="); // render forward while(!RENDER_QUEUE.empty()) { LOG_BLOCK_START(); LOG_IF(global.FLAG.render == 1, $"➤➤➤➤➤➤ CURRENT RENDER QUEUE {RENDER_QUEUE} [{RENDER_QUEUE.size()}] "); rendering = RENDER_QUEUE.dequeue(); var renderable = rendering.isRenderable(); LOG_IF(global.FLAG.render == 1, $"Rendering {rendering.internalName} ({rendering.display_name}) : {renderable? "Update" : "Pass"}"); if(renderable) { var _render_pt = get_timer(); rendering.doUpdate(); _render_time += get_timer() - _render_pt; var nextNodes = rendering.getNextNodes(); for( var i = 0, n = array_length(nextNodes); i < n; i++ ) { RENDER_QUEUE.enqueue(nextNodes[i]); } if(runAction && rendering.hasInspector1Update()) rendering.inspector1Update(); } LOG_BLOCK_END(); } _render_time /= 1000; LOG_IF(global.FLAG.render >= 1, $"=== RENDER 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 ==="); } catch(e) { noti_warning(exception_print(e)); } LOG_END(); } #endregion function NodeListSort(_list, _nodeList) { #region for( var i = 0, n = ds_list_size(_nodeList); i < n; i++ ) _nodeList[| i].topoSorted = false; ds_list_clear(_list); __sortGraph(_list, _nodeList); } #endregion function __renderListReset(list) { #region for( var i = 0; i < ds_list_size(list); i++ ) { list[| i].setRenderStatus(false); if(struct_has(list[| i], "nodes")) __renderListReset(list[| i].nodes); } } #endregion function RenderList(list, skipInLoop = true) { #region LOG_BLOCK_START(); LOG_IF(global.FLAG.render == 1, "=============== RENDER LIST START ==============="); var queue = ds_queue_create(); try { var rendering = noone; var error = 0; var t = current_time; __renderListReset(list); // get leaf node for( var i = 0, n = ds_list_size(list); i < n; i++ ) { var _node = list[| i]; _node.passiveDynamic = false; } for( var i = 0, n = ds_list_size(list); i < n; i++ ) { var _node = list[| i]; if(is_undefined(_node)) { LOG_IF(global.FLAG.render == 1, $"Skip undefiend {_node}"); continue; } if(!is_instanceof(_node, Node)) { LOG_IF(global.FLAG.render == 1, $"Skip non-node {_node}"); continue; } _node.render_time = 0; if(array_exists(global.group_io, instanceof(_node))) { LOG_IF(global.FLAG.render == 1, $"Skip group IO {_node.internalName}"); continue; } if(!_node.active) { LOG_IF(global.FLAG.render == 1, $"Skip inactive {_node.internalName}"); continue; } if(!_node.isRenderActive()) { LOG_IF(global.FLAG.render == 1, $"Skip non-renderActive {_node.internalName}"); continue; } if(!_node.attributes.update_graph) { LOG_IF(global.FLAG.render == 1, $"Skip non-auto update {_node.internalName}"); continue; } if(skipInLoop && __nodeInLoop(_node)) { LOG_IF(global.FLAG.render == 1, $"Skip in-loop {_node.internalName}"); continue; } if(_node.passiveDynamic) { _node.forwardPassiveDynamic(); LOG_IF(global.FLAG.render == 1, $"Skip passive dynamic {_node.internalName}"); continue; } if(_node.rendered && !_node.isActiveDynamic()) { LOG_IF(global.FLAG.render == 1, $"Skip rendered static {_node.internalName}"); continue; } //if() { LOG_IF(global.FLAG.render == 1, $"Skip static {_node.internalName}"); continue; } LOG_BLOCK_START(); LOG_IF(global.FLAG.render == 1, $"Found leaf {_node.internalName}"); ds_queue_enqueue(queue, _node); _node.forwardPassiveDynamic(); LOG_BLOCK_END(); } LOG_IF(global.FLAG.render == 1, "Get leaf complete: found " + string(ds_queue_size(queue)) + " leaves."); LOG_IF(global.FLAG.render == 1, "=== Start rendering ==="); // render forward while(!ds_queue_empty(queue)) { LOG_BLOCK_START(); rendering = ds_queue_dequeue(queue) if(!ds_list_exist(list, rendering)) continue; var renderable = rendering.isRenderable(); LOG_IF(global.FLAG.render == 1, $"Rendering {rendering.internalName} ({rendering.display_name}) : {renderable? "Update" : "Pass"}"); if(renderable) { rendering.doUpdate(); var nextNodes = rendering.getNextNodes(); for( var i = 0, n = array_length(nextNodes); i < n; i++ ) ds_queue_enqueue(queue, nextNodes[i]); } LOG_BLOCK_END(); } } catch(e) { noti_warning(exception_print(e)); } LOG_IF(global.FLAG.render == 1, "=== RENDER COMPLETE ===\n"); LOG_END(); ds_queue_destroy(queue); } #endregion function RenderListAction(list, context = PANEL_GRAPH.getCurrentContext()) { #region printIf(global.FLAG.render, "=== RENDER LIST ACTION START [frame " + string(CURRENT_FRAME) + "] ==="); try { var rendering = noone; var error = 0; var t = current_time; __renderListReset(list); // get leaf node RENDER_QUEUE.clear(); for( var i = 0; i < ds_list_size(list); i++ ) { var _node = list[| i]; if(is_undefined(_node)) continue; if(!is_struct(_node)) continue; if(!_node.active) continue; if(!_node.isRenderActive()) continue; if(_node.rendered) continue; if(_node.isRenderable()) { RENDER_QUEUE.enqueue(_node); printIf(global.FLAG.render, $" > Push {_node.internalName} node to queue"); } } // render forward while(!RENDER_QUEUE.empty()) { LOG_BLOCK_START(); LOG_IF(global.FLAG.render == 1, $"➤➤➤➤➤➤ CURRENT RENDER QUEUE {RENDER_QUEUE}"); rendering = RENDER_QUEUE.dequeue(); if(!ds_list_exist(list, rendering)) continue; var renderable = rendering.isRenderable(); LOG_IF(global.FLAG.render == 1, $"Rendering {rendering.internalName} ({rendering.display_name}) : {renderable? "Update" : "Pass"}"); if(renderable) { rendering.doUpdate(); var nextNodes = rendering.getNextNodes(); for( var i = 0, n = array_length(nextNodes); i < n; i++ ) RENDER_QUEUE.enqueue(nextNodes[i]); if(runAction && rendering.hasInspector1Update()) rendering.inspector1Update(); } else if(rendering.isRenderActive()) { RENDER_QUEUE.enqueue(rendering); } LOG_BLOCK_END(); } printIf(global.FLAG.render, "=== RENDER COMPLETE IN {" + string(current_time - t) + "ms} ===\n"); } catch(e) { noti_warning(exception_print(e)); } } #endregion