function Node_Collection_Inline(_x, _y, _group = noone) : Node(_x, _y, _group) constructor { attributes.members = []; nodes = []; group_vertex = []; group_dragging = false; group_adding = false; vertex_hash = ""; modifiable = true; managedRenderOrder = false; group_hovering = false; group_hover_al = 0; selectable = false; input_node_type = noone; output_node_type = noone; add_point = false; point_x = 0; point_y = 0; junction_x = 0; junction_y = 0; is_root = true; bbox = [ 0, 0, 0, 0 ]; static topoSortable = function() { return false; } static removeNode = function(node) { array_remove(attributes.members, node.node_id); array_remove(nodes, node); if(node.inline_context == self) node.inline_context = noone; onRemoveNode(node); // print($"Pose remove node : {array_length(nodes)}"); } static onRemoveNode = function(node) {} static addNode = function(node) { if(node.inline_context != noone && node.inline_context != self) node.inline_context.removeNode(node); node.inline_context = self; array_push_unique(attributes.members, node.node_id); array_push_unique(nodes, node); refreshGroupBG(); onAddNode(node); // print($"Post add node : {array_length(nodes)}"); } static addPoint = function(_x, _y) { add_point = true; point_x = _x; point_y = _y; } static onAddNode = function(node) {} static resetRender = function(_clearCache = false) { LOG_LINE_IF(global.FLAG.render == 1, $"Reset Render for {INAME}"); setRenderStatus(false); if(_clearCache) clearInputCache(); for( var i = 0; i < array_length(nodes); i++ ) nodes[i].resetRender(_clearCache); } /////////////////////////////////////////////////////////////////////////////////////// static ccw = function(a, b, c) { return (b[0] - a[0]) * (c[1] - a[1]) - (c[0] - a[0]) * (b[1] - a[1]); } static getNodeBorder = function(_ind, _vertex, _node) { var _rad = 6; var _stp = 30; var _nx0 = is_instanceof(_node, input_node_type)? _node.x + _node.w / 2 : _node.x - 32 + _rad; var _ny0 = _node.y - 32 + _rad; var _nx1 = is_instanceof(_node, output_node_type)? _node.x + _node.w / 2 : _node.x + _node.w + 32 - _rad; var _ny1 = _node.y + _node.h + 32 - _rad; for( var i = 0; i <= 90; i += _stp ) _vertex[_ind++] = [ _nx1 + lengthdir_x(_rad, i), _ny0 + lengthdir_y(_rad, i) ]; for( var i = 90; i <= 180; i += _stp ) _vertex[_ind++] = [ _nx0 + lengthdir_x(_rad, i), _ny0 + lengthdir_y(_rad, i) ]; for( var i = 180; i <= 270; i += _stp ) _vertex[_ind++] = [ _nx0 + lengthdir_x(_rad, i), _ny1 + lengthdir_y(_rad, i) ]; for( var i = 270; i <= 360; i += _stp ) _vertex[_ind++] = [ _nx1 + lengthdir_x(_rad, i), _ny1 + lengthdir_y(_rad, i) ]; return _ind; } static refreshMember = function() { nodes = []; array_foreach(attributes.members, function(m) /*=>*/ { if(ds_map_exists(PROJECT.nodeMap, m)) addNode(PROJECT.nodeMap[? m]); }) } static refreshGroupBG = function() { var _hash = ""; var _ind = 0; for( var i = 0, n = array_length(nodes); i < n; i++ ) { var _node = nodes[i]; if(!_node.active) continue; _hash += $"{_node.x},{_node.y},{_node.w},{_node.h}|"; _ind++; } if(add_point) _hash += $"{point_x},{point_y}|"; if(_hash == "") { group_vertex = []; destroy(); return; } _hash = md5_string_utf8(_hash); if(vertex_hash == _hash) return; vertex_hash = _hash; group_vertex = []; if(_ind == 0) return; var _vtrx = array_create(_ind * 4 * (90 / 30 + 1)); var _ind = 0; for( var i = 0, n = array_length(nodes); i < n; i++ ) { var _node = nodes[i]; if(!_node.active) continue; _ind = getNodeBorder(_ind, _vtrx, _node); } if(add_point) array_push(_vtrx, [ point_x, point_y ]); __temp_minP = _vtrx[0]; __temp_minI = 0; for( var i = 0, n = array_length(_vtrx); i < n; i++ ) { var _v = _vtrx[i]; var _vx = _v[0]; var _vy = _v[1]; if(_vy > __temp_minP[1] || (_vy == __temp_minP[1] && _vx < __temp_minP[0])) { __temp_minP = _v; __temp_minI = i; } } _vtrx = array_map( _vtrx, function(a, i) { return [ a[0], a[1], i == __temp_minI? -999 : point_direction(__temp_minP[0], __temp_minP[1], a[0], a[1]) + 360 ] }); array_sort(_vtrx, function(a0, a1) { return a0[2] == a1[2]? sign(a0[0] - a1[0]) : sign(a0[2] - a1[2]); }); var _linS = 0; for( var i = 1, n = array_length(_vtrx); i < n; i++ ) { if(_vtrx[i][1] != _vtrx[0][1]) break; _linS = i; } array_delete(_vtrx, 1, _linS - 1); group_vertex = [ _vtrx[0], _vtrx[1] ]; junction_x = _vtrx[0][0]; junction_y = _vtrx[0][1]; var minx = _vtrx[0][0]; var miny = _vtrx[0][1]; var maxx = _vtrx[0][0]; var maxy = _vtrx[0][1]; for( var i = 2, n = array_length(_vtrx); i < n; i++ ) { var _v = _vtrx[i]; while( array_length(group_vertex) >= 2 && ccw( group_vertex[array_length(group_vertex) - 2], group_vertex[array_length(group_vertex) - 1], _v ) >= 0 ) array_pop(group_vertex); array_push(group_vertex, _v); } for( var i = array_length(group_vertex) - 1; i >= 0; i-- ) { var n = array_length(group_vertex); if(n < 4) break; var v0 = group_vertex[(i - 1 + n) % n]; var v1 = group_vertex[i]; var v2 = group_vertex[(i + 1) % n]; var a0 = point_direction(v1[0], v1[1], v0[0], v0[1]); var a1 = point_direction(v1[0], v1[1], v2[0], v2[1]); var d = angle_difference(a0, a1); if(min(abs(d), abs(d - 180)) <= 2) array_delete(group_vertex, i, 1); if(v1[0] <= junction_x && v1[1] <= junction_y) { junction_x = v1[0]; junction_y = v1[1] + 8; } minx = min(minx, v1[0]); miny = min(miny, v1[1]); maxx = max(maxx, v1[0]); maxy = max(maxy, v1[1]); } bbox = [ minx, miny, maxx, maxy ]; } static groupCheck = function(_x, _y, _s, _mx, _my) { if(array_length(group_vertex) < 3) return; if(!modifiable) return; var _m = [ _mx / _s - _x, _my / _s - _y ]; group_adding = false; if(PANEL_GRAPH.node_dragging && PANEL_GRAPH.frame_hovering == self) { var _list = PANEL_GRAPH.nodes_selecting; PANEL_GRAPH.addKeyOverlay("Inline group", [[ "Shift", "Add/remove" ]]); if(!array_empty(_list) && key_mod_down(SHIFT)) { var _remove = _list[0].inline_context == self; if(_remove) { for( var i = 0, n = array_length(_list); i < n; i++ ) if(_list[i].manual_ungroupable) removeNode(_list[i]); } else { group_adding = true; for( var i = 0, n = array_length(_list); i < n; i++ ) if(_list[i].manual_ungroupable) addNode(_list[i]); } } } if(group_dragging && mouse_release(mb_left)) { refreshMember(); refreshGroupBG(); group_dragging = false; } } static pointIn = function(_x, _y, _mx, _my, _s) { return false; } static cullCheck = function(_x, _y, _s, minx, miny, maxx, maxy) { return true; } static drawNodeBG = function(_x, _y, _mx, _my, _s) { refreshGroupBG(); if(array_length(group_vertex) < 3) return false; var _hov = false; var _color = getColor(); var _sel = inspecting || add_point; inspecting = false; draw_set_color(_color); group_hover_al = lerp_float(group_hover_al, group_hovering, 4); group_hovering = 0; draw_set_alpha(_sel? 0.1 : lerp(0.025, 0.05, group_hover_al)); draw_primitive_begin(pr_trianglelist); var a = group_vertex[0]; var b = group_vertex[1]; var c; for( var i = 2, n = array_length(group_vertex); i < n; i++ ) { c = group_vertex[i]; var v0x = _x + a[0] * _s; var v0y = _y + a[1] * _s; var v1x = _x + b[0] * _s; var v1y = _y + b[1] * _s; var v2x = _x + c[0] * _s; var v2y = _y + c[1] * _s; draw_vertex(round(v0x), round(v0y)); draw_vertex(round(v1x), round(v1y)); draw_vertex(round(v2x), round(v2y)); if(!_hov && point_in_triangle(_mx, _my, v0x, v0y, v1x, v1y, v2x, v2y)) { group_hovering = 1 + (PANEL_GRAPH._frame_hovering == self && key_mod_press(SHIFT)) * 2; _hov = true; } b = group_vertex[i]; } draw_primitive_end(); draw_set_alpha(_sel? 1 : lerp(0.4, 0.65, group_hover_al)); draw_primitive_begin(pr_linestrip); for( var i = 0, n = array_length(group_vertex); i < n; i++ ) { var a = group_vertex[i]; draw_vertex(_x + a[0] * _s, _y + a[1] * _s); } a = group_vertex[0]; draw_vertex(_x + a[0] * _s, _y + a[1] * _s); draw_primitive_end(); draw_set_alpha(1); add_point = false; return _hov; } static drawNode = function(_draw, _x, _y, _mx, _my, _s, display_parameter = noone) { var xx = x * _s + _x; var yy = y * _s + _y; return drawJunctions(_draw, xx, yy, _mx, _my, _s) } static drawBadge = function(_x, _y, _s) {} static drawJunctions = function(_draw, _x, _y, _mx, _my, _s) {} static postDeserialize = function() { refreshMember(); } static junctionIsInside = function(junc) { if(!modifiable) return false; if(is_instanceof(junc.node, input_node_type) && junc.connect_type == CONNECT_TYPE.input) return false; if(is_instanceof(junc.node, output_node_type) && junc.connect_type == CONNECT_TYPE.output) return false; return true; } }