2023-02-14 13:44:46 +01:00
global.loop_nodes = [ "Node_Iterate", "Node_Iterate_Each" ];
2023-10-04 05:52:20 +02:00
#macro INAME internalName == ""? name : internalName
2023-10-02 14:41:44 +02:00
enum CACHE_USE {
none,
manual,
auto
}
2023-03-07 14:29:47 +01:00
function Node(_x, _y, _group = PANEL_GRAPH.getCurrentContext()) : __Node_Base(_x, _y) constructor {
2023-08-16 20:16:31 +02:00
#region ---- main & active ----
active = true;
renderActive = true;
2023-03-24 09:32:08 +01:00
2023-08-16 20:16:31 +02:00
node_id = UUID_generate();
2023-07-25 20:12:40 +02:00
2023-08-16 20:16:31 +02:00
group = _group;
2023-09-11 16:08:58 +02:00
manual_deletable = true;
2023-08-16 20:16:31 +02:00
destroy_when_upgroup = false;
ds_list_add(PANEL_GRAPH.getNodeList(_group), self);
2022-12-13 09:20:36 +01:00
2023-08-16 20:16:31 +02:00
active_index = -1;
active_range = [ 0, PROJECT.animator.frames_total - 1 ];
#endregion
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
static resetInternalName = function() { #region
2023-06-17 18:59:20 +02:00
var str = string_replace_all(name, " ", "_");
str = string_replace_all(str, "/", "");
str = string_replace_all(str, "-", "");
2023-09-28 15:10:41 +02:00
ds_map_delete(PROJECT.nodeNameMap, internalName);
2023-06-17 18:59:20 +02:00
internalName = str + string(irandom_range(10000, 99999));
2023-07-06 19:49:16 +02:00
PROJECT.nodeNameMap[? internalName] = self;
2023-08-16 20:16:31 +02:00
} #endregion
2023-06-17 18:59:20 +02:00
2023-08-16 20:16:31 +02:00
if(!LOADING && !APPENDING) { #region
2022-01-13 05:24:03 +01:00
recordAction(ACTION_TYPE.node_added, self);
2023-07-06 19:49:16 +02:00
PROJECT.nodeMap[? node_id] = self;
PROJECT.modified = true;
2023-07-25 20:12:40 +02:00
//print($"Adding node {node_id} to {PROJECT.path} [{ds_map_size(PROJECT.nodeMap)}]");
2023-05-16 21:28:16 +02:00
run_in(1, function() {
2023-07-05 15:09:52 +02:00
if(display_name != "") return;
2023-06-17 18:59:20 +02:00
resetInternalName();
2023-09-28 15:10:41 +02:00
display_name = __txt_node_name(instanceof(self), name);
2023-05-16 21:28:16 +02:00
});
2023-08-16 20:16:31 +02:00
} #endregion
2023-05-07 20:55:13 +02:00
2023-08-16 20:16:31 +02:00
#region ---- display ----
color = c_white;
icon = noone;
bg_spr = THEME.node_bg;
bg_sel_spr = THEME.node_active;
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
name = "";
display_name = "";
internalName = "";
2023-10-01 06:17:39 +02:00
onSetDisplayName = noone;
renamed = false;
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
tooltip = "";
x = _x;
y = _y;
2023-01-04 02:30:04 +01:00
2023-08-16 20:16:31 +02:00
w = 128;
h = 128;
min_h = 0;
draw_padding = 4;
auto_height = true;
display_parameter = {};
draw_name = true;
draggable = true;
draw_graph_culled = false;
badgePreview = 0;
badgeInspect = 0;
active_draw_index = -1;
draw_droppable = false;
2023-09-11 16:08:58 +02:00
junction_draw_pad_y = 32;
2023-08-16 20:16:31 +02:00
#endregion
2023-03-28 06:58:28 +02:00
2023-08-16 20:16:31 +02:00
#region ---- junctions ----
inputs = ds_list_create();
outputs = ds_list_create();
inputMap = ds_map_create();
outputMap = ds_map_create();
2023-03-28 06:58:28 +02:00
2023-10-05 06:29:20 +02:00
use_display_list = true;
2023-08-16 20:16:31 +02:00
input_display_list = -1;
output_display_list = -1;
inspector_display_list = -1;
is_dynamic_output = false;
inspectInput1 = nodeValue("Toggle execution", self, JUNCTION_CONNECT.input, VALUE_TYPE.action, false).setVisible(true, true);
inspectInput2 = nodeValue("Toggle execution", self, JUNCTION_CONNECT.input, VALUE_TYPE.action, false).setVisible(true, true);
2023-09-28 13:15:29 +02:00
autoUpdatedTrigger = true;
2023-10-06 11:51:11 +02:00
updatedInTrigger = nodeValue("Update", self, JUNCTION_CONNECT.input, VALUE_TYPE.trigger, false).setVisible(true, true);
updatedOutTrigger = nodeValue("Updated", self, JUNCTION_CONNECT.output, VALUE_TYPE.trigger, false).setVisible(true, true);
updatedInTrigger.tags = VALUE_TAG.updateInTrigger;
updatedOutTrigger.tags = VALUE_TAG.updateOutTrigger;
2023-09-28 13:15:29 +02:00
2023-08-16 20:16:31 +02:00
insp1UpdateTooltip = __txtx("panel_inspector_execute", "Execute node");
insp1UpdateIcon = [ THEME.sequence_control, 1, COLORS._main_value_positive ];
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
insp2UpdateTooltip = __txtx("panel_inspector_execute", "Execute node");
insp2UpdateIcon = [ THEME.sequence_control, 1, COLORS._main_value_positive ];
is_dynamic_input = false;
2023-09-16 20:26:35 +02:00
auto_input = false;
2023-08-16 20:16:31 +02:00
input_display_len = 0;
input_fix_len = 0;
data_length = 1;
2023-10-02 08:57:44 +02:00
inputs_data = [];
input_hash = "";
2023-10-02 14:41:44 +02:00
input_hash_raw = "";
2023-10-02 08:57:44 +02:00
anim_last_step = false;
2023-08-16 20:16:31 +02:00
#endregion
2023-04-16 11:53:46 +02:00
2023-08-16 20:16:31 +02:00
#region --- attributes ----
2023-09-28 13:15:29 +02:00
attributes = {
2023-10-06 11:51:11 +02:00
update_graph: true,
2023-09-28 13:15:29 +02:00
show_update_trigger: false
};
attributeEditors = [
2023-10-06 11:51:11 +02:00
"Node update",
["Auto update", function() { return attributes.update_graph; }, new checkBox(function() { attributes.update_graph = !attributes.update_graph; }) ],
["Update trigger", function() { return attributes.show_update_trigger; }, new checkBox(function() { attributes.show_update_trigger = !attributes.show_update_trigger; }) ],
2023-09-28 13:15:29 +02:00
];
2023-08-16 20:16:31 +02:00
#endregion
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
#region ---- preview ----
show_input_name = false;
show_output_name = false;
inspecting = false;
previewing = 0;
preview_surface = noone;
preview_amount = 0;
previewable = true;
preview_speed = 0;
preview_index = 0;
preview_channel = 0;
preview_alpha = 1;
preview_x = 0;
preview_y = 0;
preview_mx = 0;
preview_my = 0;
#endregion
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
#region ---- rendering ----
rendered = false;
update_on_frame = false;
render_time = 0;
2023-10-02 14:41:44 +02:00
render_cached = false;
2023-08-16 20:16:31 +02:00
auto_render_time = true;
updated = false;
2023-10-03 07:14:28 +02:00
use_cache = CACHE_USE.none;
cached_manual = false;
2023-08-16 20:16:31 +02:00
clearCacheOnChange = true;
cached_output = [];
cache_result = [];
temp_surface = [];
#endregion
2023-03-26 07:13:36 +02:00
2023-08-16 20:16:31 +02:00
#region ---- timeline ----
anim_priority = ds_map_size(PROJECT.nodeMap);
anim_show = true;
dopesheet_color = COLORS.panel_animation_dope_blend_default;
dopesheet_y = 0;
#endregion
2023-03-26 07:13:36 +02:00
2023-08-16 20:16:31 +02:00
#region ---- notification ----
value_validation = array_create(3);
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
error_noti_update = noone;
error_update_enabled = false;
manual_updated = false;
#endregion
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
#region ---- tools ----
tools = -1;
isTool = false;
tool_settings = [];
tool_attribute = {};
#endregion
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
#region ---- 3d ----
is_3D = false;
#endregion
2022-01-13 05:24:03 +01:00
2023-08-05 14:00:33 +02:00
static createNewInput = noone;
2023-08-16 20:16:31 +02:00
static initTooltip = function() { #region
2023-08-08 11:42:01 +02:00
var type_self/*:string*/ = instanceof(self);
if(!struct_has(global.NODE_GUIDE, type_self)) return;
2023-03-05 07:16:44 +01:00
2023-08-08 11:42:01 +02:00
var _n = global.NODE_GUIDE[$ type_self];
2023-03-05 07:16:44 +01:00
var _ins = _n.inputs;
var _ots = _n.outputs;
var amo = min(ds_list_size(inputs), array_length(_ins));
for( var i = 0; i < amo; i++ ) {
inputs[| i].name = _ins[i].name;
inputs[| i].tooltip = _ins[i].tooltip;
}
var amo = min(ds_list_size(outputs), array_length(_ots));
for( var i = 0; i < amo; i++ ) {
outputs[| i].name = _ots[i].name;
outputs[| i].tooltip = _ots[i].tooltip;
}
2023-08-16 20:16:31 +02:00
} #endregion
2023-03-05 07:16:44 +01:00
run_in(1, initTooltip);
2023-08-16 20:16:31 +02:00
static resetDefault = function() { #region
2023-02-14 02:51:14 +01:00
var folder = instanceof(self);
if(!ds_map_exists(global.PRESETS_MAP, folder)) return;
var pres = global.PRESETS_MAP[? folder];
2023-07-25 20:12:40 +02:00
for( var i = 0, n = array_length(pres); i < n; i++ ) {
2023-02-14 02:51:14 +01:00
var preset = pres[i];
if(preset.name != "_default") continue;
deserialize(preset.content, true, true);
applyDeserialize(true);
}
doUpdate();
2023-08-16 20:16:31 +02:00
} #endregion
2023-08-06 16:00:59 +02:00
if(!APPENDING && !LOADING)
run_in(1, method(self, resetDefault));
2022-09-27 06:37:28 +02:00
2023-08-16 20:16:31 +02:00
static getInputJunctionIndex = function(index) { #region
2023-10-05 06:29:20 +02:00
if(input_display_list == -1 || !use_display_list)
2022-09-27 06:37:28 +02:00
return index;
var jun_list_arr = input_display_list[index];
if(is_array(jun_list_arr)) return noone;
if(is_struct(jun_list_arr)) return noone;
return jun_list_arr;
2023-08-16 20:16:31 +02:00
} #endregion
2022-08-30 07:36:37 +02:00
2023-08-16 20:16:31 +02:00
static getOutputJunctionIndex = function(index) { #region
2022-12-13 09:20:36 +01:00
if(output_display_list == -1)
return index;
return output_display_list[index];
2023-08-16 20:16:31 +02:00
} #endregion
2022-12-13 09:20:36 +01:00
2023-08-16 20:16:31 +02:00
static setHeight = function() { #region
2023-09-11 16:08:58 +02:00
var _hi = ui(junction_draw_pad_y);
var _ho = ui(junction_draw_pad_y);
2023-01-25 06:49:00 +01:00
2023-09-09 13:52:16 +02:00
for( var i = 0; i < ds_list_size(inputs); i++ )
2022-01-13 05:24:03 +01:00
if(inputs[| i].isVisible()) _hi += 24;
2022-11-01 03:06:03 +01:00
2023-09-09 13:52:16 +02:00
for( var i = 0; i < ds_list_size(outputs); i++ )
2022-01-13 05:24:03 +01:00
if(outputs[| i].isVisible()) _ho += 24;
2023-02-14 02:51:14 +01:00
h = max(min_h, (preview_surface && previewable)? 128 : 0, _hi, _ho);
2023-08-16 20:16:31 +02:00
} #endregion
2023-02-14 02:51:14 +01:00
2023-08-16 20:16:31 +02:00
static setDisplayName = function(_name) { #region
2023-10-01 06:17:39 +02:00
renamed = true;
2023-04-15 14:48:29 +02:00
display_name = _name;
internalName = string_replace_all(display_name, " ", "_");
refreshNodeMap();
2023-06-13 14:42:06 +02:00
if(onSetDisplayName != noone)
onSetDisplayName();
2023-09-11 16:08:58 +02:00
return self;
2023-08-16 20:16:31 +02:00
} #endregion
2023-04-15 14:48:29 +02:00
2023-09-16 20:26:35 +02:00
static setIsDynamicInput = function(_data_length = 1, _auto_input = true) { #region
is_dynamic_input = true;
auto_input = _auto_input;
2023-08-05 14:00:33 +02:00
input_display_len = input_display_list == -1? 0 : array_length(input_display_list);
input_fix_len = ds_list_size(inputs);
data_length = _data_length;
2023-08-16 20:16:31 +02:00
} #endregion
2023-08-05 14:00:33 +02:00
2023-08-16 20:16:31 +02:00
static getOutput = function(junc = noone) { #region
2023-02-14 02:51:14 +01:00
for( var i = 0; i < ds_list_size(outputs); i++ ) {
if(!outputs[| i].visible) continue;
if(junc != noone && !junc.isConnectable(outputs[| i], true)) continue;
return outputs[| i];
}
return noone;
2023-08-16 20:16:31 +02:00
} #endregion
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
static getInput = function(junc = noone) { #region
2023-01-25 06:49:00 +01:00
for( var i = 0; i < ds_list_size(inputs); i++ ) {
if(!inputs[| i].visible) continue;
if(inputs[| i].value_from != noone) continue;
if(junc != noone && !inputs[| i].isConnectable(junc, true)) continue;
return inputs[| i];
}
return noone;
2023-08-16 20:16:31 +02:00
} #endregion
2023-01-25 06:49:00 +01:00
2023-08-16 20:16:31 +02:00
static getFullName = function() { #region
2023-02-14 02:51:14 +01:00
return display_name == ""? name : "[" + name + "] " + display_name;
2023-08-16 20:16:31 +02:00
} #endregion
2023-02-14 02:51:14 +01:00
2023-08-16 20:16:31 +02:00
static addInput = function(junctionFrom) { #region
2023-01-25 06:49:00 +01:00
var targ = getInput(junctionFrom);
if(targ == noone) return;
targ.setFrom(junctionFrom);
2023-08-16 20:16:31 +02:00
} #endregion
2023-01-25 06:49:00 +01:00
2023-08-16 20:16:31 +02:00
static isAnimated = function() { #region
2023-10-02 08:57:44 +02:00
if(update_on_frame) return true;
for(var i = 0; i < ds_list_size(inputs); i++)
if(inputs[| i].isAnimated()) return true;
2023-01-25 06:49:00 +01:00
return false;
2023-08-16 20:16:31 +02:00
} #endregion
2023-01-25 06:49:00 +01:00
2023-08-16 20:16:31 +02:00
static isInLoop = function() { #region
2023-02-14 13:44:46 +01:00
return array_exists(global.loop_nodes, instanceof(group));
2023-08-16 20:16:31 +02:00
} #endregion
2023-02-14 13:44:46 +01:00
2023-08-16 20:16:31 +02:00
static move = function(_x, _y) { #region
2023-02-23 07:02:19 +01:00
if(x == _x && y == _y) return;
2022-01-13 05:24:03 +01:00
x = _x;
2023-08-08 18:45:00 +02:00
y = _y;
2023-07-06 19:49:16 +02:00
if(!LOADING) PROJECT.modified = true;
2023-08-16 20:16:31 +02:00
} #endregion
2023-02-14 02:51:14 +01:00
2023-08-16 20:16:31 +02:00
#region ++++ inspector update ++++
2023-03-28 06:58:28 +02:00
static inspector1Update = function() {
2023-02-14 02:51:14 +01:00
if(error_update_enabled && error_noti_update != noone)
noti_remove(error_noti_update);
error_noti_update = noone;
2023-03-28 06:58:28 +02:00
onInspector1Update();
2023-02-14 02:51:14 +01:00
}
2023-03-28 06:58:28 +02:00
static onInspector1Update = noone;
static hasInspector1Update = function() { return onInspector1Update != noone; }
2023-02-14 02:51:14 +01:00
2023-02-23 07:02:19 +01:00
static inspector2Update = function() { onInspector2Update(); }
2023-02-14 02:51:14 +01:00
static onInspector2Update = noone;
static hasInspector2Update = function() { return onInspector2Update != noone; }
2023-08-16 20:16:31 +02:00
#endregion
2022-12-23 04:45:52 +01:00
2023-08-16 20:16:31 +02:00
static stepBegin = function() { #region
2023-03-28 06:58:28 +02:00
if(use_cache) cacheArrayCheck();
2022-01-25 04:05:30 +01:00
2022-01-13 05:24:03 +01:00
if(auto_height)
setHeight();
2022-12-16 09:18:09 +01:00
doStepBegin();
2023-03-28 06:58:28 +02:00
if(hasInspector1Update()) inspectInput1.name = insp1UpdateTooltip;
if(hasInspector2Update()) inspectInput2.name = insp2UpdateTooltip;
2023-09-28 13:15:29 +02:00
2023-10-06 11:51:11 +02:00
if(updatedInTrigger.getValue()) {
getInputs();
update();
updatedInTrigger.setValue(false);
}
updatedOutTrigger.setValue(false);
2023-08-16 20:16:31 +02:00
} #endregion
2022-12-16 09:18:09 +01:00
static doStepBegin = function() {}
2023-08-16 20:16:31 +02:00
static triggerCheck = function() { _triggerCheck(); }
2023-06-13 14:42:06 +02:00
2023-08-16 20:16:31 +02:00
static _triggerCheck = function() { #region
2023-06-04 18:28:29 +02:00
for( var i = 0; i < ds_list_size(inputs); i++ ) {
if(inputs[| i].type != VALUE_TYPE.trigger) continue;
if(!is_instanceof(inputs[| i].editWidget, buttonClass)) continue;
var trig = inputs[| i].getValue();
2023-10-03 07:14:28 +02:00
if(trig && !inputs[| i].display_data.output) {
2023-06-13 14:42:06 +02:00
inputs[| i].editWidget.onClick();
inputs[| i].setValue(false);
}
}
if(hasInspector1Update()) {
var trig = inspectInput1.getValue();
if(trig) {
2023-08-19 12:42:50 +02:00
onInspector1Update();
2023-06-13 14:42:06 +02:00
inspectInput1.setValue(false);
}
}
if(hasInspector2Update()) {
var trig = inspectInput2.getValue();
if(trig) {
2023-08-19 12:42:50 +02:00
onInspector2Update();
2023-06-13 14:42:06 +02:00
inspectInput2.setValue(false);
}
2023-06-04 18:28:29 +02:00
}
2023-08-16 20:16:31 +02:00
} #endregion
2023-06-04 18:28:29 +02:00
2022-01-13 05:24:03 +01:00
static step = function() {}
static focusStep = function() {}
2023-07-25 20:12:40 +02:00
static inspectorStep = function() {}
2022-01-13 05:24:03 +01:00
2023-10-02 10:45:30 +02:00
static getInputData = function(index, def = 0) { #region
gml_pragma("forceinline");
return array_safe_get(inputs_data, index, def);
} #endregion
2023-10-02 08:57:44 +02:00
static getInputs = function() { #region
2023-10-02 10:45:30 +02:00
inputs_data = array_create(ds_list_size(inputs), undefined);
2023-10-04 05:52:20 +02:00
2023-10-06 11:51:11 +02:00
for(var i = 0; i < ds_list_size(inputs); i++)
2023-10-02 10:45:30 +02:00
inputs_data[i] = inputs[| i].getValue(,,, true);
2023-10-02 14:41:44 +02:00
} #endregion
static forceUpdate = function() { #region
input_hash = "";
doUpdate();
2023-10-02 08:57:44 +02:00
} #endregion
2023-08-16 20:16:31 +02:00
static doUpdate = function() { #region
2023-09-28 13:15:29 +02:00
if(SAFE_MODE) return;
2023-06-05 18:27:53 +02:00
if(NODE_EXTRACT) return;
2023-10-04 05:52:20 +02:00
2023-10-02 14:41:44 +02:00
var render_timer = get_timer();
2023-10-02 08:57:44 +02:00
2023-10-03 07:14:28 +02:00
if(cached_manual || (use_cache == CACHE_USE.auto && recoverCache())) {
2023-10-02 14:41:44 +02:00
render_cached = true;
2023-10-04 05:52:20 +02:00
if(!is_instanceof(self, Node_Collection)) setRenderStatus(true);
2023-10-02 14:41:44 +02:00
} else {
render_cached = false;
var sBase = surface_get_target();
2023-10-04 05:52:20 +02:00
//var _hash = input_hash;
2023-10-02 14:41:44 +02:00
getInputs();
2023-10-04 05:52:20 +02:00
//input_hash = md5_string_unicode(input_hash_raw);
anim_last_step = isAnimated() || /*_hash != input_hash || */!rendered;
2023-10-02 08:57:44 +02:00
2023-10-02 14:41:44 +02:00
LOG_BLOCK_START();
2023-10-05 06:29:20 +02:00
LOG_IF(global.FLAG.render == 1, $">>>>>>>>>> DoUpdate called from {INAME} [{anim_last_step}] <<<<<<<<<<");
2023-10-04 05:52:20 +02:00
if(!is_instanceof(self, Node_Collection)) setRenderStatus(true);
2023-10-02 14:41:44 +02:00
try {
2023-10-04 05:52:20 +02:00
if(anim_last_step) update(); // Update only if input hash differs from previous.
2023-10-02 14:41:44 +02:00
} catch(exception) {
var sCurr = surface_get_target();
while(surface_get_target() != sBase)
surface_reset_target();
2023-02-14 02:51:14 +01:00
2023-10-02 14:41:44 +02:00
log_warning("RENDER", exception_print(exception), self);
}
2023-02-14 02:51:14 +01:00
}
2023-03-28 06:58:28 +02:00
2023-10-03 07:14:28 +02:00
cached_manual = false;
2023-10-02 10:45:30 +02:00
2023-10-03 11:27:36 +02:00
if(!use_cache && PROJECT.onion_skin.enabled) {
2023-07-08 20:29:23 +02:00
for( var i = 0; i < ds_list_size(outputs); i++ ) {
if(outputs[| i].type != VALUE_TYPE.surface) continue;
cacheCurrentFrame(outputs[| i].getValue());
break;
}
}
2023-03-28 06:58:28 +02:00
if(hasInspector1Update()) {
var trigger = inspectInput1.getValue();
if(trigger) onInspector1Update();
}
if(hasInspector2Update()) {
var trigger = inspectInput2.getValue();
if(trigger) onInspector2Update();
}
2023-09-28 13:15:29 +02:00
2023-10-06 11:51:11 +02:00
updatedOutTrigger.setValue(true);
2023-10-03 07:14:28 +02:00
2023-10-04 05:52:20 +02:00
if(!is_instanceof(self, Node_Collection))
2023-10-03 07:14:28 +02:00
render_time = get_timer() - render_timer;
2023-04-03 11:55:45 +02:00
LOG_BLOCK_END();
2023-08-16 20:16:31 +02:00
} #endregion
2023-02-14 02:51:14 +01:00
2023-08-16 20:16:31 +02:00
static valueUpdate = function(index) { #region
2023-02-14 02:51:14 +01:00
if(error_update_enabled && error_noti_update == noone)
error_noti_update = noti_error(getFullName() + " node require manual execution.",, self);
onValueUpdate(index);
2023-08-16 20:16:31 +02:00
} #endregion
2022-01-13 05:24:03 +01:00
2023-02-14 02:51:14 +01:00
static onValueUpdate = function(index = 0) {}
2023-01-09 03:14:20 +01:00
static onValueFromUpdate = function(index) {}
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
static triggerRender = function() { #region
2023-04-03 11:55:45 +02:00
LOG_BLOCK_START();
2023-10-05 06:29:20 +02:00
LOG_IF(global.FLAG.render == 1, $"Trigger render for {INAME}");
2023-03-28 06:58:28 +02:00
2023-03-26 07:13:36 +02:00
setRenderStatus(false);
UPDATE |= RENDER_TYPE.partial;
2023-06-01 10:32:21 +02:00
if(is_instanceof(group, Node_Collection) && group.reset_all_child) {
group.resetRender();
} else {
resetRender();
var nodes = getNextNodesRaw();
for(var i = 0; i < array_length(nodes); i++)
nodes[i].triggerRender();
}
2023-04-03 11:55:45 +02:00
LOG_BLOCK_END();
2023-08-16 20:16:31 +02:00
} #endregion
2023-06-01 10:32:21 +02:00
2023-10-06 11:51:11 +02:00
static resetRenderForward = function() { #region
2023-10-05 06:29:20 +02:00
setRenderStatus(false);
for( var i = 0, n = ds_list_size(outputs); i < n; i++ ) {
var _outp = outputs[| i];
for(var j = 0; j < ds_list_size(_outp.value_to); j++) {
var _to = _outp.value_to[| j];
if(!_to.node.active || _to.value_from == noone) continue;
if(_to.value_from != self) continue;
if(!_to.node.rendered) continue;
_to.node.resetRenderForward();
}
}
2023-10-06 11:51:11 +02:00
} #endregion
2023-10-05 06:29:20 +02:00
2023-06-01 10:32:21 +02:00
static resetRender = function() { setRenderStatus(false); }
2023-07-06 19:49:16 +02:00
static isRenderActive = function() { return renderActive || (PREF_MAP[? "render_all_export"] && PROJECT.animator.rendering); }
2023-06-01 10:32:21 +02:00
2023-08-16 20:16:31 +02:00
static isRenderable = function(log = false) { #region //Check if every input is ready (updated)
2023-03-26 07:13:36 +02:00
if(!active) return false;
2023-06-17 14:30:49 +02:00
if(!isRenderActive()) return false;
2022-01-24 02:21:25 +01:00
2023-06-01 10:32:21 +02:00
//if(group && struct_has(group, "iterationStatus") && group.iterationStatus() == ITERATION_STATUS.complete) return false;
2023-04-21 19:08:10 +02:00
2022-01-24 02:21:25 +01:00
for(var j = 0; j < ds_list_size(inputs); j++) {
var _in = inputs[| j];
2023-03-26 07:13:36 +02:00
if( _in.type == VALUE_TYPE.node) continue;
2023-02-19 13:49:20 +01:00
var val_from = _in.value_from;
2023-06-17 14:30:49 +02:00
if( val_from == noone) continue;
if(!val_from.node.active) continue;
if(!val_from.node.isRenderActive()) continue;
2023-07-28 19:41:57 +02:00
if!(val_from.node.rendered || val_from.node.update_on_frame) {
2023-10-05 06:29:20 +02:00
LOG_LINE_IF(global.FLAG.render == 1, $"Node {INAME} is not renderable because input {val_from.node.internalName} is not rendered ({val_from.node.rendered})");
2022-12-23 04:45:52 +01:00
return false;
2023-04-04 13:44:37 +02:00
}
2022-01-24 02:21:25 +01:00
}
return true;
2023-08-16 20:16:31 +02:00
} #endregion
2022-01-24 02:21:25 +01:00
2023-04-03 11:55:45 +02:00
static getNextNodesRaw = function() { return getNextNodes(); }
2023-08-16 20:16:31 +02:00
static getNextNodes = function() { #region
2023-03-28 06:58:28 +02:00
var nodes = [];
2023-06-01 10:32:21 +02:00
var nodeNames = [];
2023-03-28 06:58:28 +02:00
2023-04-03 11:55:45 +02:00
LOG_BLOCK_START();
2023-10-05 06:29:20 +02:00
LOG_IF(global.FLAG.render == 1, $"→→→→→ Call get next node from: {INAME}");
2023-04-03 11:55:45 +02:00
LOG_BLOCK_START();
2022-01-13 05:24:03 +01:00
for(var i = 0; i < ds_list_size(outputs); i++) {
2023-03-26 07:13:36 +02:00
var _ot = outputs[| i];
2023-06-01 10:32:21 +02:00
if(!_ot.forward) continue;
2023-03-26 07:13:36 +02:00
2023-06-01 10:32:21 +02:00
var _tos = _ot.getJunctionTo();
for( var j = 0; j < array_length(_tos); j++ ) {
var _to = _tos[j];
2023-03-26 07:13:36 +02:00
2023-03-28 06:58:28 +02:00
array_push(nodes, _to.node);
2023-06-01 10:32:21 +02:00
array_push(nodeNames, _to.node.internalName);
2023-10-05 06:29:20 +02:00
//LOG_IF(global.FLAG.render == 1, $"→→ Check output: {_ot.name} connect to node {_to.node.internalName}");
2022-01-13 05:24:03 +01:00
}
2023-03-26 07:13:36 +02:00
}
2023-03-28 06:58:28 +02:00
2023-10-05 06:29:20 +02:00
LOG_IF(global.FLAG.render == 1, $"→→ Push {nodeNames} to queue.");
2023-06-01 10:32:21 +02:00
2023-04-03 11:55:45 +02:00
LOG_BLOCK_END();
LOG_BLOCK_END();
2023-03-28 06:58:28 +02:00
return nodes;
2023-08-16 20:16:31 +02:00
} #endregion
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
static isTerminal = function() { #region
2023-07-21 12:40:20 +02:00
for( var i = 0; i < ds_list_size(outputs); i++ ) {
var _to = outputs[| i].getJunctionTo();
if(array_length(_to)) return false;
}
return true;
2023-08-16 20:16:31 +02:00
} #endregion
2023-07-21 12:40:20 +02:00
2022-09-23 13:28:42 +02:00
static onInspect = function() {}
2023-08-16 20:16:31 +02:00
static setRenderStatus = function(result) { #region
2023-10-05 06:29:20 +02:00
gml_pragma("forceinline");
LOG_LINE_IF(global.FLAG.render == 1, $"Set render status for {INAME} : {result}");
2022-12-12 09:08:03 +01:00
2023-04-03 11:55:45 +02:00
rendered = result;
2023-08-16 20:16:31 +02:00
} #endregion
2022-01-23 04:08:16 +01:00
2023-08-16 20:16:31 +02:00
static pointIn = function(_x, _y, _mx, _my, _s) { #region
2022-11-01 03:06:03 +01:00
var xx = x * _s + _x;
var yy = y * _s + _y;
2022-01-13 05:24:03 +01:00
2022-11-01 03:06:03 +01:00
return point_in_rectangle(_mx, _my, xx, yy, xx + w * _s, yy + h * _s);
2023-08-16 20:16:31 +02:00
} #endregion
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
static cullCheck = function(_x, _y, _s, minx, miny, maxx, maxy) { #region
2023-06-17 14:30:49 +02:00
var x0 = x * _s + _x;
var y0 = y * _s + _y;
var x1 = (x + w) * _s + _x;
var y1 = (y + h) * _s + _y;
draw_graph_culled = !rectangle_in_rectangle(minx, miny, maxx, maxy, x0, y0, x1, y1);
2023-08-16 20:16:31 +02:00
} #endregion
2023-06-17 14:30:49 +02:00
2023-08-16 20:16:31 +02:00
static preDraw = function(_x, _y, _s) { #region
2022-01-26 06:57:34 +01:00
var xx = x * _s + _x;
var yy = y * _s + _y;
2022-09-21 06:09:40 +02:00
var jun;
2022-12-13 09:20:36 +01:00
2023-03-28 06:58:28 +02:00
var inspCount = hasInspector1Update() + hasInspector2Update();
var ind = 1;
if(hasInspector1Update()) {
inspectInput1.x = xx + w * _s * ind / (inspCount + 1);
inspectInput1.y = yy;
ind++;
}
if(hasInspector2Update()) {
inspectInput2.x = xx + w * _s * ind / (inspCount + 1);
inspectInput2.y = yy;
ind++;
}
2023-10-06 11:51:11 +02:00
updatedInTrigger.x = xx;
updatedInTrigger.y = yy + 10;
updatedOutTrigger.x = xx + w * _s;
updatedOutTrigger.y = yy + 10;
2023-09-28 13:15:29 +02:00
2023-10-05 06:29:20 +02:00
var inamo = (input_display_list == -1 || !use_display_list)? ds_list_size(inputs) : array_length(input_display_list);
2023-09-11 16:08:58 +02:00
var _in = yy + ui(junction_draw_pad_y) * _s;
2022-01-16 14:28:57 +01:00
2022-12-13 09:20:36 +01:00
for(var i = 0; i < inamo; i++) {
2022-09-27 06:37:28 +02:00
var idx = getInputJunctionIndex(i);
if(idx == noone) continue;
2022-01-16 14:28:57 +01:00
2022-09-27 06:37:28 +02:00
jun = ds_list_get(inputs, idx, noone);
2023-02-14 02:51:14 +01:00
if(jun == noone || is_undefined(jun)) continue;
2022-01-26 06:57:34 +01:00
jun.x = xx;
2022-01-18 05:31:19 +01:00
jun.y = _in;
_in += 24 * _s * jun.isVisible();
2022-01-16 14:28:57 +01:00
}
2022-12-13 09:20:36 +01:00
var outamo = output_display_list == -1? ds_list_size(outputs) : array_length(output_display_list);
2023-03-28 06:58:28 +02:00
xx = xx + w * _s;
2023-09-11 16:08:58 +02:00
_in = yy + ui(junction_draw_pad_y) * _s;
2022-12-13 09:20:36 +01:00
for(var i = 0; i < outamo; i++) {
var idx = getOutputJunctionIndex(i);
jun = outputs[| idx];
2022-01-16 14:28:57 +01:00
2022-01-26 06:57:34 +01:00
jun.x = xx;
2022-01-18 05:31:19 +01:00
jun.y = _in;
_in += 24 * _s * jun.isVisible();
2022-01-16 14:28:57 +01:00
}
2023-08-16 20:16:31 +02:00
} #endregion
2022-01-16 14:28:57 +01:00
2023-08-16 20:16:31 +02:00
static drawNodeBase = function(xx, yy, _s) { #region
2023-06-17 14:30:49 +02:00
if(draw_graph_culled) return;
2023-02-14 02:51:14 +01:00
if(!active) return;
2023-02-19 13:49:20 +01:00
var aa = 0.25 + 0.5 * renderActive;
draw_sprite_stretched_ext(bg_spr, 0, xx, yy, w * _s, h * _s, color, aa);
2023-08-16 20:16:31 +02:00
} #endregion
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
static drawGetBbox = function(xx, yy, _s) { #region
var pad_label = draw_name && display_parameter.avoid_label;
2023-08-22 11:51:45 +02:00
var _w = w;
var _h = h;
2023-08-16 20:16:31 +02:00
_w *= display_parameter.preview_scale / 100 * _s;
_h *= display_parameter.preview_scale / 100 * _s;
2023-08-22 11:51:45 +02:00
_w -= draw_padding * 2;
_h -= draw_padding * 2 + 20 * pad_label;
2023-08-16 20:16:31 +02:00
var _xc = xx + w * _s / 2;
var _yc = yy + (h * _s + 20 * pad_label) / 2;
var x0 = _xc - _w / 2;
var x1 = _xc + _w / 2;
var y0 = _yc - _h / 2;
var y1 = _yc + _h / 2;
2023-01-04 02:30:04 +01:00
2023-06-13 14:42:06 +02:00
return BBOX().fromPoints(x0, y0, x1, y1);
2023-08-16 20:16:31 +02:00
} #endregion
2023-01-04 02:30:04 +01:00
2023-08-16 20:16:31 +02:00
static drawNodeName = function(xx, yy, _s) { #region
2023-06-17 14:30:49 +02:00
if(draw_graph_culled) return;
2023-02-14 02:51:14 +01:00
if(!active) return;
2023-02-19 13:49:20 +01:00
2023-01-04 02:30:04 +01:00
draw_name = false;
2023-02-14 02:51:14 +01:00
var _name = display_name == ""? name : display_name;
if(_name == "") return;
2023-01-25 06:49:00 +01:00
if(_s < 0.75) return;
2023-01-04 02:30:04 +01:00
draw_name = true;
2022-12-10 05:06:01 +01:00
2023-02-19 13:49:20 +01:00
var aa = 0.25 + 0.5 * renderActive;
draw_sprite_stretched_ext(THEME.node_bg_name, 0, xx, yy, w * _s, ui(20), color, aa);
2022-12-10 05:06:01 +01:00
var cc = COLORS._main_text;
2023-04-07 21:25:27 +02:00
if(PREF_MAP[? "node_show_render_status"] && !rendered)
2023-03-26 07:13:36 +02:00
cc = isRenderable()? COLORS._main_value_positive : COLORS._main_value_negative;
2022-12-12 09:08:03 +01:00
2022-12-10 05:06:01 +01:00
draw_set_text(f_p1, fa_left, fa_center, cc);
2023-03-28 06:58:28 +02:00
if(hasInspector1Update()) icon = THEME.refresh_s;
2022-12-12 09:08:03 +01:00
var ts = clamp(power(_s, 0.5), 0.5, 1);
2023-01-04 02:30:04 +01:00
2023-02-19 13:49:20 +01:00
var aa = 0.5 + 0.5 * renderActive;
draw_set_alpha(aa);
2023-01-04 02:30:04 +01:00
if(icon && _s > 0.75) {
2023-02-19 13:49:20 +01:00
draw_sprite_ui_uniform(icon, 0, xx + ui(12), yy + ui(10),,, aa);
2023-02-14 02:51:14 +01:00
draw_text_cut(xx + ui(24), yy + ui(10), _name, w * _s - ui(24), ts);
2023-01-04 02:30:04 +01:00
} else
2023-02-14 02:51:14 +01:00
draw_text_cut(xx + ui(8), yy + ui(10), _name, w * _s - ui(8), ts);
2023-02-19 13:49:20 +01:00
draw_set_alpha(1);
2023-08-16 20:16:31 +02:00
} #endregion
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
static drawJunctions = function(_x, _y, _mx, _my, _s) { #region
2023-02-14 02:51:14 +01:00
if(!active) return;
2022-01-13 05:24:03 +01:00
var hover = noone;
2022-09-23 13:28:42 +02:00
var amo = input_display_list == -1? ds_list_size(inputs) : array_length(input_display_list);
2022-09-21 06:09:40 +02:00
var jun;
2022-01-13 05:24:03 +01:00
for(var i = 0; i < amo; i++) {
2022-09-27 06:37:28 +02:00
var ind = getInputJunctionIndex(i);
if(ind == noone) continue;
jun = ds_list_get(inputs, ind, noone);
2023-02-14 02:51:14 +01:00
if(jun == noone || is_undefined(jun)) continue;
2022-01-13 05:24:03 +01:00
2022-09-27 06:37:28 +02:00
if(jun.drawJunction(_s, _mx, _my))
2022-01-26 06:57:34 +01:00
hover = jun;
2022-01-13 05:24:03 +01:00
}
for(var i = 0; i < ds_list_size(outputs); i++) {
2022-09-21 06:09:40 +02:00
jun = outputs[| i];
2022-01-13 05:24:03 +01:00
2022-09-27 06:37:28 +02:00
if(jun.drawJunction(_s, _mx, _my))
2022-01-26 06:57:34 +01:00
hover = jun;
2022-01-13 05:24:03 +01:00
}
2023-03-28 06:58:28 +02:00
if(hasInspector1Update() && inspectInput1.drawJunction(_s, _mx, _my))
hover = inspectInput1;
if(hasInspector2Update() && inspectInput2.drawJunction(_s, _mx, _my))
hover = inspectInput2;
2023-10-06 11:51:11 +02:00
if(attributes.show_update_trigger) {
if(updatedInTrigger.drawJunction(_s, _mx, _my)) hover = updatedInTrigger;
if(updatedOutTrigger.drawJunction(_s, _mx, _my)) hover = updatedOutTrigger;
}
2023-09-28 13:15:29 +02:00
2022-01-13 05:24:03 +01:00
return hover;
2023-08-16 20:16:31 +02:00
} #endregion
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
static drawJunctionNames = function(_x, _y, _mx, _my, _s) { #region
2023-06-17 14:30:49 +02:00
if(draw_graph_culled) return;
2023-02-14 02:51:14 +01:00
if(!active) return;
2022-09-27 06:37:28 +02:00
var amo = input_display_list == -1? ds_list_size(inputs) : array_length(input_display_list);
var jun;
var xx = x * _s + _x;
var yy = y * _s + _y;
2023-02-23 07:02:19 +01:00
show_input_name = PANEL_GRAPH.pHOVER && point_in_rectangle(_mx, _my, xx - 8 * _s, yy + 20 * _s, xx + 8 * _s, yy + h * _s);
show_output_name = PANEL_GRAPH.pHOVER && point_in_rectangle(_mx, _my, xx + (w - 8) * _s, yy + 20 * _s, xx + (w + 8) * _s, yy + h * _s);
2022-09-27 06:37:28 +02:00
if(show_input_name) {
for(var i = 0; i < amo; i++) {
var ind = getInputJunctionIndex(i);
if(ind == noone) continue;
2023-03-07 14:29:47 +01:00
if(!inputs[| ind]) continue;
2022-09-27 06:37:28 +02:00
inputs[| ind].drawNameBG(_s);
}
for(var i = 0; i < amo; i++) {
var ind = getInputJunctionIndex(i);
if(ind == noone) continue;
2023-03-07 14:29:47 +01:00
if(!inputs[| ind]) continue;
2022-09-27 06:37:28 +02:00
inputs[| ind].drawName(_s, _mx, _my);
}
}
if(show_output_name) {
2023-02-23 07:02:19 +01:00
for(var i = 0; i < ds_list_size(outputs); i++)
2022-09-27 06:37:28 +02:00
outputs[| i].drawNameBG(_s);
2023-02-23 07:02:19 +01:00
for(var i = 0; i < ds_list_size(outputs); i++)
2022-09-27 06:37:28 +02:00
outputs[| i].drawName(_s, _mx, _my);
}
2023-03-28 06:58:28 +02:00
if(hasInspector1Update() && PANEL_GRAPH.pHOVER && point_in_circle(_mx, _my, inspectInput1.x, inspectInput1.y, 10)) {
inspectInput1.drawNameBG(_s);
inspectInput1.drawName(_s, _mx, _my);
}
if(hasInspector2Update() && PANEL_GRAPH.pHOVER && point_in_circle(_mx, _my, inspectInput2.x, inspectInput2.y, 10)) {
inspectInput2.drawNameBG(_s);
inspectInput2.drawName(_s, _mx, _my);
}
2023-08-16 20:16:31 +02:00
} #endregion
2022-09-27 06:37:28 +02:00
2023-08-31 18:49:57 +02:00
static drawConnections = function(params = {}) { #region
2023-02-14 02:51:14 +01:00
if(!active) return;
2022-01-25 04:05:30 +01:00
var hovering = noone;
2023-03-05 07:16:44 +01:00
var drawLineIndex = 1;
for(var i = 0; i < ds_list_size(outputs); i++) {
var jun = outputs[| i];
var connected = false;
for( var j = 0; j < ds_list_size(jun.value_to); j++ ) {
if(jun.value_to[| j].value_from == jun)
connected = true;
}
if(connected) {
jun.drawLineIndex = drawLineIndex;
drawLineIndex += 0.5;
}
}
2023-03-28 06:58:28 +02:00
var st = 0;
2023-06-17 14:30:49 +02:00
if(hasInspector1Update()) st = -1;
2023-03-28 06:58:28 +02:00
if(hasInspector2Update()) st = -2;
2023-06-17 14:30:49 +02:00
var _inputs = [];
2023-03-05 07:16:44 +01:00
var drawLineIndex = 1;
2023-06-17 14:30:49 +02:00
for(var i = st; i < ds_list_size(inputs); i++) {
var jun;
if(i == -1) jun = inspectInput1;
else if(i == -2) jun = inspectInput2;
else jun = inputs[| i];
if(jun.value_from == noone) continue;
if(!jun.value_from.node.active) continue;
if(!jun.isVisible()) continue;
if(i >= 0)
array_push(_inputs, jun);
}
var len = array_length(_inputs);
for( var i = 0; i < len; i++ )
_inputs[i].drawLineIndex = 1 + (i > len / 2? (len - 1 - i) : i) * 0.5;
2023-03-28 06:58:28 +02:00
for(var i = st; i < ds_list_size(inputs); i++) {
var jun;
if(i == -1) jun = inspectInput1;
else if(i == -2) jun = inspectInput2;
else jun = inputs[| i];
2023-08-31 18:49:57 +02:00
var hov = jun.drawConnections(params);
2023-07-25 20:12:40 +02:00
if(hov) hovering = hov;
2022-01-13 05:24:03 +01:00
}
2022-01-25 04:05:30 +01:00
2023-10-06 11:51:11 +02:00
if(attributes.show_update_trigger) {
if(updatedInTrigger.drawConnections(params)) hovering = updatedInTrigger;
if(updatedOutTrigger.drawConnections(params)) hovering = updatedOutTrigger;
}
2022-01-25 04:05:30 +01:00
return hovering;
2023-08-16 20:16:31 +02:00
} #endregion
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
static getGraphPreviewSurface = function() { #region
2023-08-10 11:51:16 +02:00
var _node = outputs[| preview_channel];
2023-09-09 13:52:16 +02:00
switch(_node.type) {
case VALUE_TYPE.surface :
2023-09-16 20:26:35 +02:00
case VALUE_TYPE.dynaSurface :
2023-09-09 13:52:16 +02:00
return _node.getValue();
}
return noone;
2023-08-16 20:16:31 +02:00
} #endregion
2023-08-10 11:51:16 +02:00
2023-08-16 20:16:31 +02:00
static drawPreview = function(xx, yy, _s) { #region
2023-06-17 14:30:49 +02:00
if(draw_graph_culled) return;
2023-02-14 02:51:14 +01:00
if(!active) return;
2023-02-20 10:16:31 +01:00
2023-08-10 11:51:16 +02:00
var surf = getGraphPreviewSurface();
if(surf == noone) return;
2023-02-20 10:16:31 +01:00
2023-01-25 06:49:00 +01:00
preview_amount = 0;
2022-01-13 05:24:03 +01:00
if(is_array(surf)) {
if(array_length(surf) == 0) return;
2023-01-25 06:49:00 +01:00
preview_amount = array_length(surf);
2022-01-13 05:24:03 +01:00
if(preview_speed != 0) {
preview_index += preview_speed;
if(preview_index <= 0)
preview_index = array_length(surf) - 1;
}
if(floor(preview_index) > array_length(surf) - 1) preview_index = 0;
2022-01-16 05:17:35 +01:00
surf = surf[preview_index];
2022-01-13 05:24:03 +01:00
}
2023-02-14 02:51:14 +01:00
preview_surface = is_surface(surf)? surf : noone;
2023-03-26 07:13:36 +02:00
if(preview_surface == noone) return;
2022-11-03 11:44:49 +01:00
2023-08-16 20:16:31 +02:00
var bbox = drawGetBbox(xx, yy, _s);
var aa = 0.5 + 0.5 * renderActive;
2023-02-19 13:49:20 +01:00
2023-08-16 20:16:31 +02:00
draw_surface_bbox(preview_surface, bbox, c_white, aa);
} #endregion
2023-01-25 06:49:00 +01:00
2023-08-16 20:16:31 +02:00
static getNodeDimension = function(showFormat = true) { #region
2023-01-25 06:49:00 +01:00
if(!is_surface(preview_surface)) {
if(ds_list_size(outputs))
2023-03-19 09:17:39 +01:00
return "[" + array_shape(outputs[| 0].getValue()) + "]";
2023-01-25 06:49:00 +01:00
return "";
}
2023-09-08 21:37:36 +02:00
var pw = surface_get_width_safe(preview_surface);
var ph = surface_get_height_safe(preview_surface);
2023-03-19 09:17:39 +01:00
var format = surface_get_format(preview_surface);
var txt = "[" + string(pw) + " x " + string(ph) + " ";
if(preview_amount) txt = string(preview_amount) + " x " + txt;
switch(format) {
2023-04-05 20:13:27 +02:00
case surface_rgba4unorm : txt += showFormat? "4RGBA" : "4R"; break;
case surface_rgba8unorm : txt += showFormat? "8RGBA" : "8R"; break;
case surface_rgba16float : txt += showFormat? "16RGBA" : "16R"; break;
case surface_rgba32float : txt += showFormat? "32RGBA" : "32R"; break;
case surface_r8unorm : txt += showFormat? "8BW" : "8B"; break;
case surface_r16float : txt += showFormat? "16BW" : "16B"; break;
case surface_r32float : txt += showFormat? "32BW" : "32B"; break;
2023-03-19 09:17:39 +01:00
}
2023-04-05 20:13:27 +02:00
2023-03-19 09:17:39 +01:00
txt += "]";
2023-01-25 06:49:00 +01:00
return txt;
2023-08-16 20:16:31 +02:00
} #endregion
2023-01-25 06:49:00 +01:00
2023-08-16 20:16:31 +02:00
static drawDimension = function(xx, yy, _s) { #region
2023-06-17 14:30:49 +02:00
if(draw_graph_culled) return;
2023-02-14 02:51:14 +01:00
if(!active) return;
2023-01-25 06:49:00 +01:00
if(_s * w < 64) return;
2023-07-23 20:21:35 +02:00
draw_set_text(f_p2, fa_center, fa_top, COLORS.panel_graph_node_dimension);
2023-01-25 06:49:00 +01:00
var tx = xx + w * _s / 2;
2023-07-23 20:21:35 +02:00
var ty = yy + (h + 4) * _s - 2;
2023-01-25 06:49:00 +01:00
2023-08-16 20:16:31 +02:00
if(display_parameter.show_dimension) {
2023-04-05 20:13:27 +02:00
var txt = string(getNodeDimension(_s > 0.65));
2023-01-25 06:49:00 +01:00
draw_text(round(tx), round(ty), txt);
2023-07-23 20:21:35 +02:00
ty += string_height(txt) - 2;
2023-01-25 06:49:00 +01:00
}
2023-07-23 20:21:35 +02:00
draw_set_font(f_p3);
2023-08-16 20:16:31 +02:00
if(display_parameter.show_compute) {
2023-10-02 14:41:44 +02:00
var rt = 0, unit = "";
if(render_time == 0) {
draw_set_color(COLORS._main_text_sub);
unit = "us";
} else if(render_time < 1000) {
2023-01-25 06:49:00 +01:00
rt = round(render_time / 10) * 10;
unit = "us";
draw_set_color(COLORS.speed[2]);
} else if(render_time < 1000000) {
rt = string_format(render_time / 1000, -1, 2);
unit = "ms";
draw_set_color(COLORS.speed[1]);
} else {
rt = string_format(render_time / 1000000, -1, 2);
unit = "s";
draw_set_color(COLORS.speed[0]);
2022-01-16 05:17:35 +01:00
}
2023-10-02 14:41:44 +02:00
2023-10-03 07:14:28 +02:00
if(render_cached || !anim_last_step) draw_set_color(COLORS._main_text_sub);
2023-10-02 14:41:44 +02:00
2023-01-25 06:49:00 +01:00
draw_text(round(tx), round(ty), string(rt) + " " + unit);
2022-01-13 05:24:03 +01:00
}
2023-08-16 20:16:31 +02:00
} #endregion
2022-01-13 05:24:03 +01:00
2023-09-28 13:15:29 +02:00
static drawNode = function(_x, _y, _mx, _my, _s, display_parameter = noone) { #region
2023-06-17 14:30:49 +02:00
if(draw_graph_culled) return;
2023-02-14 02:51:14 +01:00
if(!active) return;
2023-09-28 13:15:29 +02:00
if(display_parameter != noone)
self.display_parameter = display_parameter;
2022-01-13 05:24:03 +01:00
var xx = x * _s + _x;
var yy = y * _s + _y;
2022-08-30 07:36:37 +02:00
2023-03-26 07:13:36 +02:00
preview_mx = _mx;
preview_my = _my;
2023-02-14 02:51:14 +01:00
if(value_validation[VALIDATION.error] || error_noti_update != noone)
2023-01-01 02:06:02 +01:00
draw_sprite_stretched_ext(THEME.node_glow, 0, xx - 9, yy - 9, w * _s + 18, h * _s + 18, COLORS._main_value_negative, 1);
2022-08-30 07:36:37 +02:00
2022-01-13 05:24:03 +01:00
drawNodeBase(xx, yy, _s);
2023-10-06 11:51:11 +02:00
if(previewable) {
2023-01-01 02:06:02 +01:00
if(preview_channel >= ds_list_size(outputs))
preview_channel = 0;
2023-01-25 06:49:00 +01:00
drawPreview(xx, yy, _s);
2023-02-14 02:51:14 +01:00
}
2023-01-25 06:49:00 +01:00
drawDimension(xx, yy, _s);
2023-01-01 02:06:02 +01:00
2023-03-05 07:16:44 +01:00
onDrawNode(xx, yy, _mx, _my, _s, PANEL_GRAPH.node_hovering == self, PANEL_GRAPH.node_focus == self);
2022-12-16 09:18:09 +01:00
drawNodeName(xx, yy, _s);
2023-03-05 07:16:44 +01:00
2022-01-13 05:24:03 +01:00
if(active_draw_index > -1) {
2023-01-25 06:49:00 +01:00
draw_sprite_stretched_ext(bg_sel_spr, 0, xx, yy, round(w * _s), round(h * _s), active_draw_index > 1? COLORS.node_border_file_drop : COLORS._main_accent, 1);
2022-01-13 05:24:03 +01:00
active_draw_index = -1;
}
2023-03-26 07:13:36 +02:00
if(draw_droppable)
draw_sprite_stretched_ext(THEME.ui_panel_active, 0, xx, yy, w * _s, h * _s, COLORS._main_value_positive, 1);
draw_droppable = false;
2022-09-21 06:09:40 +02:00
return drawJunctions(xx, yy, _mx, _my, _s);
2023-08-16 20:16:31 +02:00
} #endregion
2023-03-28 06:58:28 +02:00
2023-01-25 06:49:00 +01:00
static onDrawNodeBehind = function(_x, _y, _mx, _my, _s) {}
2023-03-28 06:58:28 +02:00
2023-03-05 07:16:44 +01:00
static onDrawNode = function(xx, yy, _mx, _my, _s, _hover = false, _focus = false) {}
2022-01-13 05:24:03 +01:00
2023-06-13 21:24:05 +02:00
static onDrawHover = function(_x, _y, _mx, _my, _s) {}
2023-08-16 20:16:31 +02:00
static drawBadge = function(_x, _y, _s) { #region
2023-02-14 02:51:14 +01:00
if(!active) return;
2022-01-13 05:24:03 +01:00
var xx = x * _s + _x + w * _s;
var yy = y * _s + _y;
2023-03-28 06:58:28 +02:00
badgePreview = lerp_float(badgePreview, !!previewing, 2);
badgeInspect = lerp_float(badgeInspect, inspecting, 2);
if(badgePreview > 0) {
draw_sprite_ext(THEME.node_state, 0, xx, yy, badgePreview, badgePreview, 0, c_white, 1);
xx -= 28 * badgePreview;
2022-01-13 05:24:03 +01:00
}
2023-03-28 06:58:28 +02:00
if(badgeInspect > 0) {
draw_sprite_ext(THEME.node_state, 1, xx, yy, badgeInspect, badgeInspect, 0, c_white, 1);
xx -= 28 * badgeInspect;
2022-01-13 05:24:03 +01:00
}
2023-07-17 19:58:33 +02:00
if(isTool) {
draw_sprite_ext(THEME.node_state, 2, xx, yy, 1, 1, 0, c_white, 1);
xx -= 28 * 2;
}
2022-01-13 05:24:03 +01:00
inspecting = false;
2022-08-30 07:36:37 +02:00
previewing = 0;
2023-08-16 20:16:31 +02:00
} #endregion
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
static drawActive = function(_x, _y, _s, ind = 0) { active_draw_index = ind; }
2022-01-13 05:24:03 +01:00
2022-12-19 13:35:30 +01:00
static drawOverlay = function(active, _x, _y, _s, _mx, _my, _snx, _sny) {}
2022-01-13 05:24:03 +01:00
2023-05-22 20:31:55 +02:00
static drawAnimationTimeline = function(_w, _h, _s) {}
2023-02-14 02:51:14 +01:00
static enable = function() { active = true; }
static disable = function() { active = false; }
2023-07-08 20:29:23 +02:00
2023-08-16 20:16:31 +02:00
static destroy = function(_merge = false) { #region
2023-02-14 02:51:14 +01:00
if(!active) return;
disable();
2023-09-19 20:43:02 +02:00
if(PANEL_GRAPH.node_hover == self) PANEL_GRAPH.node_hover = noone;
if(PANEL_GRAPH.node_focus == self) PANEL_GRAPH.node_focus = noone;
if(PANEL_INSPECTOR.inspecting == self) PANEL_INSPECTOR.inspecting = noone;
2023-07-17 19:58:33 +02:00
PANEL_PREVIEW.removeNodePreview(self);
2022-01-13 05:24:03 +01:00
PANEL_ANIMATION.updatePropertyList();
for(var i = 0; i < ds_list_size(outputs); i++) {
var jun = outputs[| i];
2022-01-26 06:57:34 +01:00
2022-01-13 05:24:03 +01:00
for(var j = 0; j < ds_list_size(jun.value_to); j++) {
2022-01-26 06:57:34 +01:00
var _vt = jun.value_to[| j];
2023-02-14 02:51:14 +01:00
if(_vt.value_from == noone) break;
if(_vt.value_from.node != self) break;
2022-01-26 06:57:34 +01:00
_vt.removeFrom(false);
2023-02-14 02:51:14 +01:00
if(!_merge) continue;
for( var k = 0; k < ds_list_size(inputs); k++ ) {
if(inputs[| k].value_from == noone) continue;
if(_vt.setFrom(inputs[| k].value_from)) break;
2022-01-26 06:57:34 +01:00
}
2022-01-13 05:24:03 +01:00
}
2022-01-26 06:57:34 +01:00
ds_list_clear(jun.value_to);
2022-01-13 05:24:03 +01:00
}
2023-02-23 07:02:19 +01:00
for( var i = 0; i < ds_list_size(inputs); i++ )
inputs[| i].destroy();
for( var i = 0; i < ds_list_size(outputs); i++ )
outputs[| i].destroy();
2022-01-13 05:24:03 +01:00
onDestroy();
2023-10-02 10:45:30 +02:00
RENDER_ALL_REORDER
2023-08-16 20:16:31 +02:00
} #endregion
2023-02-14 02:51:14 +01:00
2023-08-16 20:16:31 +02:00
static restore = function() { #region
2023-02-14 02:51:14 +01:00
if(active) return;
enable();
2023-07-06 19:49:16 +02:00
ds_list_add(group == noone? PROJECT.nodes : group.getNodeList(), self);
2023-08-16 20:16:31 +02:00
} #endregion
2023-02-14 02:51:14 +01:00
2023-08-16 20:16:31 +02:00
static onValidate = function() { #region
2022-11-03 11:44:49 +01:00
value_validation[VALIDATION.pass] = 0;
value_validation[VALIDATION.warning] = 0;
value_validation[VALIDATION.error] = 0;
2022-08-30 07:36:37 +02:00
for( var i = 0; i < ds_list_size(inputs); i++ ) {
var jun = inputs[| i];
if(jun.value_validation)
2022-11-03 11:44:49 +01:00
value_validation[jun.value_validation]++;
2022-08-30 07:36:37 +02:00
}
2023-08-16 20:16:31 +02:00
} #endregion
2022-08-30 07:36:37 +02:00
2022-01-13 05:24:03 +01:00
static onDestroy = function() {}
2023-08-16 20:16:31 +02:00
static clearInputCache = function() { #region
2023-02-17 11:31:33 +01:00
for( var i = 0; i < ds_list_size(inputs); i++ )
2023-02-14 02:51:14 +01:00
inputs[| i].cache_value[0] = false;
2023-08-16 20:16:31 +02:00
} #endregion
2023-02-14 02:51:14 +01:00
2023-08-16 20:16:31 +02:00
static cacheArrayCheck = function() { #region
2023-07-06 19:49:16 +02:00
if(array_length(cached_output) != PROJECT.animator.frames_total)
array_resize(cached_output, PROJECT.animator.frames_total);
if(array_length(cache_result) != PROJECT.animator.frames_total)
array_resize(cache_result, PROJECT.animator.frames_total);
2023-08-16 20:16:31 +02:00
} #endregion
2022-12-10 05:06:01 +01:00
2023-08-16 20:16:31 +02:00
static cacheCurrentFrame = function(_frame) { #region
2022-12-10 05:06:01 +01:00
cacheArrayCheck();
2023-07-06 19:49:16 +02:00
if(PROJECT.animator.current_frame < 0) return;
if(PROJECT.animator.current_frame >= array_length(cached_output)) return;
2022-01-13 05:24:03 +01:00
2023-07-06 19:49:16 +02:00
surface_array_free(cached_output[PROJECT.animator.current_frame]);
cached_output[PROJECT.animator.current_frame] = surface_array_clone(_frame);
2022-12-10 05:06:01 +01:00
2023-07-06 19:49:16 +02:00
array_safe_set(cache_result, PROJECT.animator.current_frame, true);
2023-05-03 21:42:17 +02:00
2023-07-06 19:49:16 +02:00
return cached_output[PROJECT.animator.current_frame];
2023-08-16 20:16:31 +02:00
} #endregion
2023-02-19 13:49:20 +01:00
2023-08-16 20:16:31 +02:00
static cacheExist = function(frame = PROJECT.animator.current_frame) { #region
2023-03-02 07:59:14 +01:00
if(frame < 0) return false;
2022-12-10 05:06:01 +01:00
if(frame >= array_length(cached_output)) return false;
if(frame >= array_length(cache_result)) return false;
2023-02-23 07:02:19 +01:00
if(!array_safe_get(cache_result, frame, false)) return false;
2023-05-03 21:42:17 +02:00
var s = array_safe_get(cached_output, frame);
return is_array(s) || surface_exists(s);
2023-08-16 20:16:31 +02:00
} #endregion
2022-12-10 05:06:01 +01:00
2023-08-16 20:16:31 +02:00
static getCacheFrame = function(frame = PROJECT.animator.current_frame) { #region
2023-03-02 07:59:14 +01:00
if(frame < 0) return false;
2023-02-19 13:49:20 +01:00
if(!cacheExist(frame)) return noone;
var surf = array_safe_get(cached_output, frame);
return surf;
2023-08-16 20:16:31 +02:00
} #endregion
2023-02-19 13:49:20 +01:00
2023-08-16 20:16:31 +02:00
static recoverCache = function(frame = PROJECT.animator.current_frame) { #region
2022-12-10 05:06:01 +01:00
if(!cacheExist(frame)) return false;
2023-07-06 19:49:16 +02:00
var _s = cached_output[PROJECT.animator.current_frame];
2023-03-02 07:59:14 +01:00
outputs[| 0].setValue(_s);
2022-11-22 14:25:39 +01:00
return true;
2023-08-16 20:16:31 +02:00
} #endregion
static clearCache = function() { #region
2023-07-25 20:12:40 +02:00
clearInputCache();
2023-07-30 13:56:22 +02:00
if(!clearCacheOnChange) return;
2023-02-23 07:02:19 +01:00
if(!use_cache) return;
2023-06-17 14:30:49 +02:00
if(!isRenderActive()) return;
2023-02-19 13:49:20 +01:00
2023-07-06 19:49:16 +02:00
if(array_length(cached_output) != PROJECT.animator.frames_total)
array_resize(cached_output, PROJECT.animator.frames_total);
2022-01-13 05:24:03 +01:00
for(var i = 0; i < array_length(cached_output); i++) {
var _s = cached_output[i];
2022-11-22 14:25:39 +01:00
if(is_surface(_s))
2022-01-13 05:24:03 +01:00
surface_free(_s);
cached_output[i] = 0;
2022-12-10 05:06:01 +01:00
cache_result[i] = false;
2022-01-13 05:24:03 +01:00
}
2023-08-16 20:16:31 +02:00
} #endregion
2023-07-08 20:29:23 +02:00
2023-08-16 20:16:31 +02:00
static clearCacheForward = function() { _clearCacheForward(); }
2023-07-08 20:29:23 +02:00
2023-08-16 20:16:31 +02:00
static _clearCacheForward = function() { #region
2023-06-17 14:30:49 +02:00
if(!isRenderActive()) return;
2023-03-08 07:35:51 +01:00
2023-02-23 07:02:19 +01:00
clearCache();
2023-07-25 20:12:40 +02:00
var arr = getNextNodesRaw();
for( var i = 0, n = array_length(arr); i < n; i++ )
arr[i]._clearCacheForward();
//for( var i = 0; i < ds_list_size(outputs); i++ )
//for( var j = 0; j < ds_list_size(outputs[| i].value_to); j++ )
// outputs[| i].value_to[| j].node._clearCacheForward();
2023-08-16 20:16:31 +02:00
} #endregion
2023-07-25 20:12:40 +02:00
2023-08-16 20:16:31 +02:00
static clearInputCache = function() { #region
2023-07-25 20:12:40 +02:00
for( var i = 0; i < ds_list_size(inputs); i++ )
inputs[| i].resetCache();
2023-08-16 20:16:31 +02:00
} #endregion
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
static checkConnectGroup = function(_type = "group") { #region
2022-12-23 04:45:52 +01:00
var _y = y;
2023-02-14 02:51:14 +01:00
var nodes = [];
2022-01-13 05:24:03 +01:00
for(var i = 0; i < ds_list_size(inputs); i++) {
var _in = inputs[| i];
2022-12-23 04:45:52 +01:00
if(_in.value_from == noone) continue;
if(_in.value_from.node.group == group) continue;
var input_node = noone;
2023-02-14 02:51:14 +01:00
2022-12-23 04:45:52 +01:00
switch(_type) {
case "group" : input_node = new Node_Group_Input(x - w - 64, _y, group); break;
case "loop" : input_node = new Node_Iterator_Input(x - w - 64, _y, group); break;
case "feedback" : input_node = new Node_Feedback_Input(x - w - 64, _y, group); break;
}
2022-01-13 05:24:03 +01:00
2022-12-23 04:45:52 +01:00
if(input_node == noone) continue;
2023-02-14 02:51:14 +01:00
array_push(nodes, input_node);
2022-12-23 04:45:52 +01:00
input_node.inputs[| 2].setValue(_in.type);
input_node.inParent.setFrom(_in.value_from);
input_node.onValueUpdate(0);
_in.setFrom(input_node.outputs[| 0]);
_y += 64;
}
2022-01-13 05:24:03 +01:00
for(var i = 0; i < ds_list_size(outputs); i++) {
var _ou = outputs[| i];
for(var j = 0; j < ds_list_size(_ou.value_to); j++) {
var _to = _ou.value_to[| j];
2022-12-12 09:08:03 +01:00
if(_to.value_from != _ou) continue;
if(!_to.node.active) continue;
if(_to.node.group == group) continue;
var output_node = noone;
2023-02-14 02:51:14 +01:00
2022-12-12 09:08:03 +01:00
switch(_type) {
2022-12-23 04:45:52 +01:00
case "group" : output_node = new Node_Group_Output(x + w + 64, y, group); break;
case "loop" : output_node = new Node_Iterator_Output(x + w + 64, y, group); break;
case "feedback" : output_node = new Node_Feedback_Output(x + w + 64, y, group); break;
2022-12-12 09:08:03 +01:00
}
2022-01-25 10:58:11 +01:00
2022-12-12 09:08:03 +01:00
if(output_node == noone) continue;
2022-12-23 04:45:52 +01:00
2023-02-14 02:51:14 +01:00
array_push(nodes, output_node);
2022-12-12 09:08:03 +01:00
_to.setFrom(output_node.outParent);
output_node.inputs[| 0].setFrom(_ou);
2022-01-13 05:24:03 +01:00
}
}
2023-02-14 02:51:14 +01:00
return nodes;
2023-08-16 20:16:31 +02:00
} #endregion
2022-01-13 05:24:03 +01:00
2023-08-16 20:16:31 +02:00
static isNotUsingTool = function() { return PANEL_PREVIEW.tool_current == noone; }
2023-03-11 01:40:17 +01:00
2023-08-16 20:16:31 +02:00
static isUsingTool = function(index = undefined, subtool = noone) { #region
2023-03-11 01:40:17 +01:00
if(tools == -1)
return false;
2023-07-15 20:01:29 +02:00
var _tool = PANEL_PREVIEW.tool_current;
2023-10-04 07:21:45 +02:00
if(_tool == noone) //not using any tool
2023-07-15 20:01:29 +02:00
return false;
2023-10-04 07:21:45 +02:00
if(index == undefined) //using any tool
2023-08-16 20:16:31 +02:00
return true;
2023-07-15 20:01:29 +02:00
if(is_real(index) && _tool != tools[index])
2023-03-11 01:40:17 +01:00
return false;
2023-07-15 20:01:29 +02:00
if(is_string(index) && _tool.getName(_tool.selecting) != index)
return false;
2023-03-11 01:40:17 +01:00
if(subtool == noone)
return true;
2023-07-15 20:01:29 +02:00
return _tool.selecting == subtool;
2023-08-16 20:16:31 +02:00
} #endregion
2023-03-11 01:40:17 +01:00
2023-08-16 20:16:31 +02:00
static clone = function(target = PANEL_GRAPH.getCurrentContext()) { #region
2023-01-17 08:11:55 +01:00
CLONING = true;
2022-12-12 09:08:03 +01:00
var _type = instanceof(self);
2023-02-14 02:51:14 +01:00
var _node = nodeBuild(_type, x, y, target);
2023-01-17 08:11:55 +01:00
CLONING = false;
2023-03-07 14:29:47 +01:00
2023-07-06 19:49:16 +02:00
PROJECT.version = SAVE_VERSION;
2023-01-17 08:11:55 +01:00
if(!_node) return;
2023-03-07 14:29:47 +01:00
CLONING = true;
2022-12-21 02:30:23 +01:00
var _nid = _node.node_id;
2023-02-14 02:51:14 +01:00
_node.deserialize(serialize());
_node.postDeserialize();
2022-12-21 02:30:23 +01:00
_node.applyDeserialize();
_node.node_id = _nid;
2022-12-18 03:20:38 +01:00
2023-07-06 19:49:16 +02:00
PROJECT.nodeMap[? node_id] = self;
PROJECT.nodeMap[? _nid] = _node;
2022-12-21 02:30:23 +01:00
PANEL_ANIMATION.updatePropertyList();
2023-03-07 14:29:47 +01:00
CLONING = false;
2022-12-12 09:08:03 +01:00
2023-02-28 09:43:01 +01:00
onClone(_node, target);
2023-01-25 06:49:00 +01:00
2022-12-12 09:08:03 +01:00
return _node;
2023-08-16 20:16:31 +02:00
} #endregion
2022-12-12 09:08:03 +01:00
2023-02-28 09:43:01 +01:00
static onClone = function(_NewNode, target = PANEL_GRAPH.getCurrentContext()) {}
2023-01-25 06:49:00 +01:00
2023-08-16 20:16:31 +02:00
static droppable = function(dragObj) { #region
2023-03-26 07:13:36 +02:00
for( var i = 0; i < ds_list_size(inputs); i++ ) {
if(dragObj.type == inputs[| i].drop_key)
return true;
}
return false;
2023-08-16 20:16:31 +02:00
} #endregion
2023-03-26 07:13:36 +02:00
2023-09-28 13:15:29 +02:00
on_drop_file = noone;
2023-08-16 20:16:31 +02:00
static onDrop = function(dragObj) { #region
2023-09-28 13:15:29 +02:00
if(dragObj.type == "Asset" && is_callable(on_drop_file)) {
on_drop_file(dragObj.data.path);
return;
}
2023-03-26 07:13:36 +02:00
for( var i = 0; i < ds_list_size(inputs); i++ ) {
if(dragObj.type == inputs[| i].drop_key) {
inputs[| i].setValue(dragObj.data);
return;
}
}
2023-08-16 20:16:31 +02:00
} #endregion
2023-03-25 12:27:04 +01:00
2023-08-17 16:56:54 +02:00
static getPreviewValues = function() { #region
2023-08-31 18:49:57 +02:00
if(preview_channel >= ds_list_size(outputs)) return noone;
2023-09-09 13:52:16 +02:00
switch(outputs[| preview_channel].type) {
case VALUE_TYPE.surface :
2023-09-16 20:26:35 +02:00
case VALUE_TYPE.dynaSurface :
2023-09-09 13:52:16 +02:00
break;
default :
return;
}
2023-08-31 18:49:57 +02:00
2023-08-17 16:56:54 +02:00
return outputs[| preview_channel].getValue();
2023-08-16 20:16:31 +02:00
} #endregion
2023-07-21 12:40:20 +02:00
2023-08-16 20:16:31 +02:00
static getPreviewBoundingBox = function() { #region
2023-08-17 16:56:54 +02:00
var _surf = getPreviewValues();
2023-07-10 20:14:10 +02:00
if(is_array(_surf))
_surf = array_safe_get(_surf, preview_index, noone);
if(!is_surface(_surf)) return noone;
2023-09-08 21:37:36 +02:00
return BBOX().fromWH(preview_x, preview_y, surface_get_width_safe(_surf), surface_get_height_safe(_surf));
2023-08-16 20:16:31 +02:00
} #endregion
2023-07-10 20:14:10 +02:00
2023-08-16 20:16:31 +02:00
static getTool = function() { return self; }
2023-07-17 19:58:33 +02:00
2023-08-19 12:42:50 +02:00
static getToolSettings = function() { return tool_settings; }
2023-08-16 20:16:31 +02:00
static setTool = function(tool) { #region
2023-07-17 19:58:33 +02:00
if(!tool) {
isTool = false;
return;
}
for( var i = 0; i < ds_list_size(group.nodes); i++ )
group.nodes[| i].isTool = false;
isTool = true;
2023-08-16 20:16:31 +02:00
} #endregion
2023-07-17 19:58:33 +02:00
2023-08-29 19:26:18 +02:00
static serialize = function(scale = false, preset = false) { #region
2023-06-13 14:42:06 +02:00
var _map = {};
2023-03-11 01:40:17 +01:00
//print(" > Serializing: " + name);
2022-01-13 05:24:03 +01:00
2022-12-10 05:06:01 +01:00
if(!preset) {
2023-06-13 14:42:06 +02:00
_map.id = node_id;
_map.render = renderActive;
_map.name = display_name;
_map.iname = internalName;
_map.x = x;
_map.y = y;
_map.type = instanceof(self);
_map.group = group == noone? group : group.node_id;
_map.preview = previewable;
2023-07-17 19:58:33 +02:00
_map.tool = isTool;
2022-12-10 05:06:01 +01:00
}
2022-01-13 05:24:03 +01:00
2023-08-05 14:00:33 +02:00
_map.attri = attributeSerialize();
if(is_dynamic_input) {
_map.input_fix_len = input_fix_len;
_map.data_length = data_length;
}
2022-01-13 05:24:03 +01:00
2023-06-13 14:42:06 +02:00
var _inputs = [];
2023-02-17 04:48:54 +01:00
for(var i = 0; i < ds_list_size(inputs); i++)
2023-06-13 14:42:06 +02:00
array_push(_inputs, inputs[| i].serialize(scale, preset));
_map.inputs = _inputs;
2022-01-13 05:24:03 +01:00
2023-06-13 14:42:06 +02:00
var _outputs = [];
2023-04-07 21:25:27 +02:00
for(var i = 0; i < ds_list_size(outputs); i++)
2023-06-13 14:42:06 +02:00
array_push(_outputs, outputs[| i].serialize(scale, preset));
_map.outputs = _outputs;
2023-04-07 21:25:27 +02:00
2023-06-13 14:42:06 +02:00
var _trigger = [];
array_push(_trigger, inspectInput1.serialize(scale, preset));
array_push(_trigger, inspectInput2.serialize(scale, preset));
2023-10-06 11:51:11 +02:00
array_push(_trigger, updatedInTrigger.serialize(scale, preset));
array_push(_trigger, updatedOutTrigger.serialize(scale, preset));
2023-06-13 14:42:06 +02:00
_map.inspectInputs = _trigger;
2023-10-01 06:17:39 +02:00
_map.renamed = renamed;
2023-03-28 06:58:28 +02:00
2022-01-13 05:24:03 +01:00
doSerialize(_map);
2023-02-28 09:43:01 +01:00
processSerialize(_map);
2022-01-13 05:24:03 +01:00
return _map;
2023-08-29 19:26:18 +02:00
} #endregion
2022-01-13 05:24:03 +01:00
2023-06-13 14:42:06 +02:00
static attributeSerialize = function() { return attributes; }
2022-01-13 05:24:03 +01:00
static doSerialize = function(_map) {}
2023-02-28 09:43:01 +01:00
static processSerialize = function(_map) {}
2022-01-13 05:24:03 +01:00
2022-12-12 09:08:03 +01:00
load_scale = false;
2022-01-18 05:31:19 +01:00
load_map = -1;
2023-08-29 19:26:18 +02:00
static deserialize = function(_map, scale = false, preset = false) { #region
2022-12-10 05:06:01 +01:00
load_map = _map;
2022-12-12 09:08:03 +01:00
load_scale = scale;
2022-01-19 03:05:13 +01:00
2022-12-10 05:06:01 +01:00
if(!preset) {
2023-06-13 14:42:06 +02:00
if(APPENDING) APPEND_MAP[? load_map.id] = node_id;
else node_id = load_map.id;
2023-07-06 19:49:16 +02:00
PROJECT.nodeMap[? node_id] = self;
2023-07-25 20:12:40 +02:00
//print($"D Adding node {node_id} to {PROJECT.path} [{ds_map_size(PROJECT.nodeMap)}]");
2023-03-30 07:54:25 +02:00
2023-06-13 14:42:06 +02:00
if(struct_has(load_map, "name"))
setDisplayName(load_map.name);
2023-05-16 21:28:16 +02:00
2023-06-13 14:42:06 +02:00
internalName = struct_try_get(load_map, "iname", internalName);
2023-06-17 18:59:20 +02:00
if(internalName == "")
resetInternalName();
2023-06-13 14:42:06 +02:00
_group = struct_try_get(load_map, "group", noone);
2023-03-02 07:59:14 +01:00
if(_group == -1) _group = noone;
2023-03-30 07:54:25 +02:00
2023-06-13 14:42:06 +02:00
x = struct_try_get(load_map, "x");
y = struct_try_get(load_map, "y");
renderActive = struct_try_get(load_map, "render", true);
previewable = struct_try_get(load_map, "preview", previewable);
2023-07-17 19:58:33 +02:00
isTool = struct_try_get(load_map, "tool");
2022-12-10 05:06:01 +01:00
}
2022-01-13 05:24:03 +01:00
2023-06-13 14:42:06 +02:00
if(struct_has(load_map, "attri"))
attributeDeserialize(load_map.attri);
2022-01-13 05:24:03 +01:00
2023-08-05 14:00:33 +02:00
if(is_dynamic_input) {
inputBalance();
inputGenerate();
}
2023-02-28 09:43:01 +01:00
processDeserialize();
2023-04-22 16:23:51 +02:00
if(preset) {
postDeserialize();
applyDeserialize();
triggerRender();
}
2023-10-01 06:17:39 +02:00
renamed = struct_try_get(load_map, "renamed", false);
2023-08-29 19:26:18 +02:00
} #endregion
2023-03-28 06:58:28 +02:00
2023-08-29 19:26:18 +02:00
static inputBalance = function() { #region //Cross version compatibility for dynamic input nodes
2023-08-05 14:00:33 +02:00
if(!struct_has(load_map, "data_length"))
return;
var _input_fix_len = load_map.input_fix_len;
var _data_length = load_map.data_length;
2023-08-08 18:45:00 +02:00
//print($"Balancing IO: {input_fix_len} => {load_map.input_fix_len} : {data_length} => {load_map.data_length}");
//print($"IO size before: {array_length(load_map.inputs)}");
//for( var i = 0, n = array_length(load_map.inputs); i < n; i++ )
// print($"{i}: {load_map.inputs[i].name}");
2023-08-05 14:00:33 +02:00
var _dynamic_inputs = (array_length(load_map.inputs) - _input_fix_len) / _data_length;
if(frac(_dynamic_inputs) != 0) {
noti_warning("LOAD: Uneven dynamic input.");
_dynamic_inputs = ceil(_dynamic_inputs);
}
2023-08-08 18:45:00 +02:00
2023-08-05 14:00:33 +02:00
if(_input_fix_len == input_fix_len && _data_length == data_length)
return;
var _pad_dyna = data_length - _data_length;
2023-08-08 18:45:00 +02:00
for( var i = _dynamic_inputs; i >= 1; i-- ) {
2023-08-05 14:00:33 +02:00
var _ind = _input_fix_len + i * _data_length;
2023-08-08 20:33:17 +02:00
if(_pad_dyna > 0)
repeat(_pad_dyna) array_insert(load_map.inputs, _ind, noone);
else
array_delete(load_map.inputs, _ind + _pad_dyna, -_pad_dyna);
2023-08-05 14:00:33 +02:00
}
var _pad_fix = input_fix_len - _input_fix_len;
repeat(_pad_fix)
array_insert(load_map.inputs, _input_fix_len, noone);
2023-08-08 18:45:00 +02:00
//print($"IO size after: {array_length(load_map.inputs)}");
//for( var i = 0, n = array_length(load_map.inputs); i < n; i++ )
// print($"{i}: {load_map.inputs[i] == noone? "noone" : load_map.inputs[i].name}");
2023-08-29 19:26:18 +02:00
} #endregion
2023-08-05 14:00:33 +02:00
2023-08-29 19:26:18 +02:00
static inputGenerate = function() { #region //Generate input for dynamic input nodes
2023-08-05 14:00:33 +02:00
if(createNewInput == noone)
return;
var _dynamic_inputs = (array_length(load_map.inputs) - input_fix_len) / data_length;
2023-08-08 18:45:00 +02:00
//print($"Node {name} create {_dynamic_inputs} inputs for data length {data_length}");
2023-08-05 14:00:33 +02:00
repeat(_dynamic_inputs)
createNewInput();
2023-08-29 19:26:18 +02:00
} #endregion
2022-01-13 05:24:03 +01:00
2023-08-29 19:26:18 +02:00
static attributeDeserialize = function(attr) { #region
2023-06-13 14:42:06 +02:00
struct_override(attributes, attr);
2023-10-02 10:45:30 +02:00
} #endregion
2022-01-13 05:24:03 +01:00
static postDeserialize = function() {}
2023-02-28 09:43:01 +01:00
static processDeserialize = function() {}
2023-03-11 01:40:17 +01:00
2023-08-29 19:26:18 +02:00
static applyDeserialize = function(preset = false) { #region
2023-06-13 14:42:06 +02:00
var _inputs = load_map.inputs;
var amo = min(ds_list_size(inputs), array_length(_inputs));
2022-12-22 03:09:55 +01:00
2023-03-11 01:40:17 +01:00
for(var i = 0; i < amo; i++) {
2023-07-14 20:34:35 +02:00
if(inputs[| i] == noone || _inputs[i] == noone) continue;
2023-09-29 04:22:41 +02:00
//if(name == "Particle") print($"Apply {i} : {inputs[| i].name}");
2023-06-13 14:42:06 +02:00
inputs[| i].applyDeserialize(_inputs[i], load_scale, preset);
2023-03-11 01:40:17 +01:00
}
2023-01-01 02:06:02 +01:00
2023-06-13 14:42:06 +02:00
if(struct_has(load_map, "outputs")) {
var _outputs = load_map.outputs;
2023-06-17 14:30:49 +02:00
var amo = min(ds_list_size(outputs), array_length(_outputs));
for(var i = 0; i < amo; i++) {
2023-04-07 21:25:27 +02:00
if(outputs[| i] == noone) continue;
2023-09-29 04:22:41 +02:00
2023-06-13 14:42:06 +02:00
outputs[| i].applyDeserialize(_outputs[i], load_scale, preset);
2023-04-07 21:25:27 +02:00
}
}
2023-06-13 14:42:06 +02:00
if(struct_has(load_map, "inspectInputs")) {
var insInp = load_map.inspectInputs;
inspectInput1.applyDeserialize(insInp[0], load_scale, preset);
inspectInput2.applyDeserialize(insInp[1], load_scale, preset);
2023-09-28 13:15:29 +02:00
2023-10-06 11:51:11 +02:00
if(array_length(insInp) > 2) updatedInTrigger.applyDeserialize(insInp[2], load_scale, preset);
if(array_length(insInp) > 3) updatedOutTrigger.applyDeserialize(insInp[3], load_scale, preset);
2023-03-28 06:58:28 +02:00
}
2023-01-09 03:14:20 +01:00
doApplyDeserialize();
2023-08-29 19:26:18 +02:00
} #endregion
2022-12-21 02:30:23 +01:00
2023-01-09 03:14:20 +01:00
static doApplyDeserialize = function() {}
2023-08-29 19:26:18 +02:00
static loadGroup = function(context = PANEL_GRAPH.getCurrentContext()) { #region
2023-02-28 09:43:01 +01:00
if(_group == noone) {
2023-03-26 07:13:36 +02:00
var c = context;
2023-02-28 09:43:01 +01:00
if(c != noone) c.add(self);
2022-01-13 05:24:03 +01:00
} else {
2022-01-19 03:05:13 +01:00
if(APPENDING) _group = GetAppendID(_group);
2023-07-06 19:49:16 +02:00
if(ds_map_exists(PROJECT.nodeMap, _group)) {
2023-07-21 12:40:20 +02:00
if(struct_has(PROJECT.nodeMap[? _group], "add"))
PROJECT.nodeMap[? _group].add(self);
else {
var txt = $"Group load failed. Node ID {_group} is not a group.";
throw(txt);
}
2022-01-19 03:05:13 +01:00
} else {
2023-07-21 12:40:20 +02:00
var txt = $"Group load failed. Can't find node ID {_group}";
throw(txt);
2022-01-18 05:31:19 +01:00
}
2022-01-13 05:24:03 +01:00
}
2023-08-29 19:26:18 +02:00
} #endregion
2022-01-18 05:31:19 +01:00
2023-08-29 19:26:18 +02:00
static connect = function(log = false) { #region
2022-01-13 05:24:03 +01:00
var connected = true;
2023-03-28 06:58:28 +02:00
for(var i = 0; i < ds_list_size(inputs); i++)
2022-09-21 06:09:40 +02:00
connected &= inputs[| i].connect(log);
2023-03-28 06:58:28 +02:00
2023-10-06 11:51:11 +02:00
inspectInput1.connect(log);
inspectInput2.connect(log);
updatedInTrigger.connect(log);
2023-03-28 06:58:28 +02:00
2022-01-13 05:24:03 +01:00
if(!connected) ds_queue_enqueue(CONNECTION_CONFLICT, self);
2022-09-21 06:09:40 +02:00
return connected;
2023-08-29 19:26:18 +02:00
} #endregion
2022-01-13 05:24:03 +01:00
static preConnect = function() {}
2022-01-18 05:31:19 +01:00
static postConnect = function() {}
2022-01-19 06:11:17 +01:00
2023-04-10 20:02:59 +02:00
static resetAnimation = function() {}
2023-08-16 20:16:31 +02:00
static cleanUp = function() { #region
2023-02-23 07:02:19 +01:00
for( var i = 0; i < ds_list_size(inputs); i++ )
2022-01-19 06:11:17 +01:00
inputs[| i].cleanUp();
2023-02-23 07:02:19 +01:00
for( var i = 0; i < ds_list_size(outputs); i++ )
2022-01-19 06:11:17 +01:00
outputs[| i].cleanUp();
ds_list_destroy(inputs);
ds_list_destroy(outputs);
2023-04-15 14:48:29 +02:00
ds_map_destroy(inputMap);
ds_map_destroy(outputMap);
2023-07-25 20:12:40 +02:00
for( var i = 0, n = array_length(temp_surface); i < n; i++ )
2023-02-23 07:02:19 +01:00
surface_free(temp_surface[i]);
2023-04-14 12:23:25 +02:00
onCleanUp();
2023-08-16 20:16:31 +02:00
} #endregion
2023-02-23 07:02:19 +01:00
static onCleanUp = function() {}
2023-03-19 09:17:39 +01:00
// helper function
2023-08-16 20:16:31 +02:00
static attrDepth = function() { #region
2023-06-13 14:42:06 +02:00
if(struct_has(attributes, "color_depth")) {
var form = attributes.color_depth;
2023-03-19 09:17:39 +01:00
if(inputs[| 0].type == VALUE_TYPE.surface)
form--;
if(form >= 0)
return array_safe_get(global.SURFACE_FORMAT, form, surface_rgba8unorm);
}
2023-10-02 08:57:44 +02:00
var _s = getInputData(0);
2023-03-19 09:17:39 +01:00
while(is_array(_s) && array_length(_s)) _s = _s[0];
if(!is_surface(_s))
return surface_rgba8unorm;
return surface_get_format(_s);
2023-08-16 20:16:31 +02:00
} #endregion
2023-10-02 10:45:30 +02:00
static toString = function() { return $"PixelComposerNode: {node_id}[{internalName}] {input_hash}"; }
2022-01-13 05:24:03 +01:00
}