enum NODE_COMPOSE_DRAG { move, rotate, scale } enum COMPOSE_OUTPUT_SCALING { first, largest, constant } function Node_Composite(_x, _y, _group = noone) : Node_Processor(_x, _y, _group) constructor { name = "Composite"; shader = sh_blend_normal_dim; uniform_dim = shader_get_uniform(shader, "dimension"); uniform_pos = shader_get_uniform(shader, "position"); uniform_sca = shader_get_uniform(shader, "scale"); uniform_rot = shader_get_uniform(shader, "rotation"); uniform_for = shader_get_sampler_index(shader, "fore"); inputs[| 0] = nodeValue("Padding", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, [ 0, 0, 0, 0 ]) .setDisplay(VALUE_DISPLAY.padding); inputs[| 1] = nodeValue("Output dimension", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, COMPOSE_OUTPUT_SCALING.first) .setDisplay(VALUE_DISPLAY.enum_scroll, [ "First surface", "Largest surface", "Constant" ]); inputs[| 2] = nodeValue("Dimension", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, def_surf_size2) .setDisplay(VALUE_DISPLAY.vector) .setVisible(false); attribute_surface_depth(); attribute_interpolation(); input_fix_len = ds_list_size(inputs); data_length = 4; attributes[? "layer_visible"] = ds_list_create(); attributes[? "layer_selectable"] = ds_list_create(); hold_visibility = true; hold_select = true; layer_dragging = noone; layer_remove = -1; layer_renderer = new Inspector_Custom_Renderer(function(_x, _y, _w, _m, _hover, _focus) { var amo = (ds_list_size(inputs) - input_fix_len) / data_length - 1; if(array_length(current_data) != ds_list_size(inputs)) return 0; var lh = 32; var _h = 8 + max(1, amo) * (lh + 4) + 8; layer_renderer.h = _h; draw_sprite_stretched_ext(THEME.ui_panel_bg, 1, _x, _y, _w, _h, COLORS.node_composite_bg_blend, 1); var _vis = attributes[? "layer_visible"]; var _sel = attributes[? "layer_selectable"]; var ly = _y + 8; var ssh = lh - 6; var hoverIndex = noone; draw_set_color(COLORS.node_composite_separator); draw_line(_x + 16, ly, _x + _w - 16, ly); layer_remove = -1; for(var i = 0; i < amo; i++) { var ind = amo - i - 1; var index = input_fix_len + ind * data_length; var _surf = current_data[index + 0]; var _pos = current_data[index + 1]; var _bx = _x + _w - 24; var _cy = ly + i * (lh + 4); if(point_in_circle(_m[0], _m[1], _bx, _cy + lh / 2, 16)) { draw_sprite_ui_uniform(THEME.icon_delete, 3, _bx, _cy + lh / 2, 1, COLORS._main_value_negative); if(mouse_press(mb_left, _focus)) layer_remove = ind; } else draw_sprite_ui_uniform(THEME.icon_delete, 3, _bx, _cy + lh / 2, 1, COLORS._main_icon); if(!is_surface(_surf)) continue; var aa = (ind != layer_dragging || layer_dragging == noone)? 1 : 0.5; var vis = _vis[| ind]; var sel = _sel[| ind]; var hover = point_in_rectangle(_m[0], _m[1], _x, _cy, _x + _w, _cy + lh); draw_set_color(COLORS.node_composite_separator); draw_line(_x + 16, _cy + lh + 2, _x + _w - 16, _cy + lh + 2); var _bx = _x + 24 * 2 + 8; if(point_in_circle(_m[0], _m[1], _bx, _cy + lh / 2, 12)) { draw_sprite_ui_uniform(THEME.junc_visible, vis, _bx, _cy + lh / 2, 1, c_white); if(mouse_press(mb_left, _focus)) hold_visibility = !_vis[| ind]; if(mouse_click(mb_left, _focus) && _vis[| ind] != hold_visibility) { _vis[| ind] = hold_visibility; doUpdate(); } } else draw_sprite_ui_uniform(THEME.junc_visible, vis, _bx, _cy + lh / 2, 1, COLORS._main_icon, 0.5 + 0.5 * vis); _bx += 24 + 8; if(point_in_circle(_m[0], _m[1], _bx, _cy + lh / 2, 12)) { draw_sprite_ui_uniform(THEME.cursor_select, sel, _bx, _cy + lh / 2, 1, c_white); if(mouse_press(mb_left, _focus)) hold_select = !_sel[| ind]; if(mouse_click(mb_left, _focus) && _sel[| ind] != hold_select) _sel[| ind] = hold_select; } else draw_sprite_ui_uniform(THEME.cursor_select, sel, _bx, _cy + lh / 2, 1, COLORS._main_icon, 0.5 + 0.5 * sel); draw_set_color(COLORS.node_composite_bg); var _sx0 = _bx + 24; var _sx1 = _sx0 + ssh; var _sy0 = _cy + 3; var _sy1 = _sy0 + ssh; draw_rectangle(_sx0, _sy0, _sx1, _sy1, true); var _ssw = surface_get_width(_surf); var _ssh = surface_get_height(_surf); var _sss = min(ssh / _ssw, ssh / _ssh); draw_surface_ext_safe(_surf, _sx0, _sy0, _sss, _sss, 0, c_white, 1); draw_set_text(f_p1, fa_left, fa_center, hover? COLORS._main_text : COLORS._main_text); draw_set_alpha(aa); draw_text(_sx1 + 12, _cy + lh / 2, inputs[| index].name); draw_set_alpha(1); if(_hover && point_in_rectangle(_m[0], _m[1], _x, _cy, _x + _w, _cy + lh)) { hoverIndex = ind; if(layer_dragging != noone) { draw_set_color(COLORS._main_accent); if(layer_dragging > ind) draw_line_width(_x + 16, _cy + lh + 2, _x + _w - 16, _cy + lh + 2, 2); else if(layer_dragging < ind) draw_line_width(_x + 16, _cy - 2, _x + _w - 16, _cy - 2, 2); } } if(layer_dragging == noone || layer_dragging == ind) { var _bx = _x + 24; if(point_in_circle(_m[0], _m[1], _bx, _cy + lh / 2, 16)) { draw_sprite_ui_uniform(THEME.hamburger, 3, _bx, _cy + lh / 2, .75, c_white); if(mouse_press(mb_left, _focus)) layer_dragging = ind; } else draw_sprite_ui_uniform(THEME.hamburger, 3, _bx, _cy + lh / 2, .75, COLORS._main_icon); } } if(layer_dragging != noone && mouse_release(mb_left)) { if(layer_dragging != hoverIndex && hoverIndex != noone) { var index = input_fix_len + layer_dragging * data_length; var targt = input_fix_len + hoverIndex * data_length; var _vis = attributes[? "layer_visible"]; var _sel = attributes[? "layer_selectable"]; var ext = []; var vis = _vis[| layer_dragging]; ds_list_delete(_vis, layer_dragging); ds_list_insert(_vis, hoverIndex, vis); var sel = _sel[| layer_dragging]; ds_list_delete(_sel, layer_dragging); ds_list_insert(_sel, hoverIndex, sel); for( var i = 0; i < data_length; i++ ) { ext[i] = inputs[| index]; ds_list_delete(inputs, index); } for( var i = 0; i < data_length; i++ ) { ds_list_insert(inputs, targt + i, ext[i]); } doUpdate(); } layer_dragging = noone; } return _h; }); input_display_list = [ ["Output", true], 0, 1, 2, ["Layers", false], layer_renderer, ["Surfaces", true], ]; input_display_list_len = array_length(input_display_list); function deleteLayer(index) { var idx = input_fix_len + index * data_length; for( var i = 0; i < data_length; i++ ) { ds_list_delete(inputs, idx); array_remove(input_display_list, idx + i); } for( var i = input_display_list_len; i < array_length(input_display_list); i++ ) { if(input_display_list[i] > idx) input_display_list[i] = input_display_list[i] - data_length; } doUpdate(); } function createNewSurface() { var index = ds_list_size(inputs); var _s = floor((index - input_fix_len) / data_length); inputs[| index + 0] = nodeValue(_s? ("Surface " + string(_s)) : "Background", self, JUNCTION_CONNECT.input, VALUE_TYPE.surface, 0); inputs[| index + 0].surface_index = index; inputs[| index + 0].hover_effect = 0; inputs[| index + 1] = nodeValue("Position " + string(_s), self, JUNCTION_CONNECT.input, VALUE_TYPE.float, [ 0, 0 ] ) .setDisplay(VALUE_DISPLAY.vector) .setUnitRef(function(index) { return [ overlay_w, overlay_h ]; }); inputs[| index + 1].surface_index = index; inputs[| index + 2] = nodeValue("Rotation " + string(_s), self, JUNCTION_CONNECT.input, VALUE_TYPE.float, 0 ) .setDisplay(VALUE_DISPLAY.rotation); inputs[| index + 2].surface_index = index; inputs[| index + 3] = nodeValue("Scale " + string(_s), self, JUNCTION_CONNECT.input, VALUE_TYPE.float, [ 1, 1 ] ) .setDisplay(VALUE_DISPLAY.vector); inputs[| index + 3].surface_index = index; array_push(input_display_list, index + 0); array_push(input_display_list, index + 1); array_push(input_display_list, index + 2); array_push(input_display_list, index + 3); while(_s >= ds_list_size(attributes[? "layer_visible"])) ds_list_add(attributes[? "layer_visible"], true); while(_s >= ds_list_size(attributes[? "layer_selectable"])) ds_list_add(attributes[? "layer_selectable"], true); } if(!LOADING && !APPENDING) createNewSurface(); //function getInput() { return inputs[| ds_list_size(inputs) - data_length]; } outputs[| 0] = nodeValue("Surface out", self, JUNCTION_CONNECT.output, VALUE_TYPE.surface, noone); outputs[| 1] = nodeValue("Atlas data", self, JUNCTION_CONNECT.output, VALUE_TYPE.atlas, []) .rejectArrayProcess(); temp_surface = [ surface_create(1, 1), surface_create(1, 1) ]; surf_dragging = -1; input_dragging = -1; drag_type = 0; dragging_sx = 0; dragging_sy = 0; dragging_mx = 0; dragging_my = 0; rot_anc_x = 0; rot_anc_y = 0; overlay_w = 0; overlay_h = 0; atlas_data = []; static getInputAmount = function() { return input_fix_len + (ds_list_size(inputs) - input_fix_len) / data_length; } static getInputIndex = function(index) { if(index < input_fix_len) return index; return input_fix_len + (index - input_fix_len) * data_length; } static setHeight = function() { var _hi = ui(32); var _ho = ui(32); for( var i = 0; i < getInputAmount(); i++ ) if(inputs[| getInputIndex(i)].isVisible()) _hi += 24; for( var i = 0; i < ds_list_size(outputs); i++ ) if(outputs[| i].isVisible()) _ho += 24; h = max(min_h, (preview_surface && previewable)? 128 : 0, _hi, _ho); } static drawJunctions = function(_x, _y, _mx, _my, _s) { if(!active) return; var hover = noone; var amo = array_length(input_display_list); var hov = PANEL_GRAPH._junction_hovering; var ind = -1; if(hov != noone && struct_has(hov, "surface_index")) ind = hov.surface_index; for( var i = 0; i < getInputAmount(); i++ ) { var idx = getInputIndex(i); if(!inputs[| idx].isVisible()) continue; if(inputs[| idx].drawJunction(_s, _mx, _my, 1.5)) hover = inputs[| idx]; if(idx >= input_fix_len && inputs[| idx].hover_effect > 0) { var _px0 = 999999; var _py0 = 999999; var _px1 = -999999; var _py1 = -999999; var _drw = false; var _hov = inputs[| idx].hover_effect; for( var j = 1; j < data_length; j++ ) { if(!inputs[| idx + j].isVisible()) continue; _px0 = min( _px0, inputs[| idx + j].x ); _py0 = min( _py0, inputs[| idx + j].y ); _px1 = max( _px1, inputs[| idx + j].x ); _py1 = max( _py1, inputs[| idx + j].y ); _drw = true; } if(!_drw) continue; //if(_hov > 0.5) { // var pilx = _px0 - 16 * _s; // var pily = _py0 - 16 * _s; // var pilw = _px1 - _px0 + 32 * _s; // var pilh = _py1 - _py0 + 32 * _s; // draw_sprite_stretched_ext(THEME.node_bg_pill, 0, pilx, pily, pilw, pilh, COLORS._main_icon_dark, (_hov - 0.5) * 2); //} for( var j = 1; j < data_length; j++ ) { if(inputs[| idx + j].drawJunction(_s, _mx, _my, 1.5)) hover = inputs[| idx + j]; } } } for(var i = 0; i < ds_list_size(outputs); i++) if(outputs[| i].drawJunction(_s, _mx, _my)) hover = outputs[| i]; return hover; } static drawJunctionNames = function(_x, _y, _mx, _my, _s) { if(!active) return; 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; 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); var hov = PANEL_GRAPH._junction_hovering; var ind = -1; if(hov != noone && struct_has(hov, "surface_index")) ind = hov.surface_index; if(ind != -1) { for( var j = 1; j < data_length; j++ ) { if(ind + j >= ds_list_size(inputs)) break; inputs[| ind + j].drawNameBG(_s); } for( var j = 1; j < data_length; j++ ) { if(ind + j >= ds_list_size(inputs)) break; inputs[| ind + j].drawName(_s, _mx, _my); } } else if(show_input_name) { for( var i = 0; i < getInputAmount(); i++ ) { var idx = getInputIndex(i); if(idx == ind) continue; inputs[| idx].drawNameBG(_s); } for( var i = 0; i < getInputAmount(); i++ ) { var idx = getInputIndex(i); if(idx == ind) continue; inputs[| idx].drawName(_s, _mx, _my); } } if(show_output_name) { for(var i = 0; i < ds_list_size(outputs); i++) outputs[| i].drawNameBG(_s); for(var i = 0; i < ds_list_size(outputs); i++) outputs[| i].drawName(_s, _mx, _my); } } static preDraw = function(_x, _y, _s) { var xx = x * _s + _x; var yy = y * _s + _y; var jun; var inamo = input_display_list == -1? ds_list_size(inputs) : array_length(input_display_list); var _in = yy + ui(32) * _s; var hov = PANEL_GRAPH._junction_hovering; var ind = -1; if(hov != noone && struct_has(hov, "surface_index")) ind = hov.surface_index; for( var i = 0; i < getInputAmount(); i++ ) { var idx = getInputIndex(i); jun = ds_list_get(inputs, idx, noone); if(jun == noone || is_undefined(jun)) continue; jun.x = xx; jun.y = _in; if(i >= input_fix_len) { jun.hover_effect = lerp_float(jun.hover_effect, ind == idx, 3); var sp = jun.hover_effect * 24; var sx = xx - sp * _s; var sy = _in; for( var j = 1; j < data_length; j++ ) { var _jun = ds_list_get(inputs, idx + j, noone); _jun.x = sx; _jun.y = sy; sy += sp * _s * _jun.isVisible(); } } _in += 24 * _s * jun.isVisible(); } var outamo = output_display_list == -1? ds_list_size(outputs) : array_length(output_display_list); xx = xx + w * _s; _in = yy + ui(32) * _s; for(var i = 0; i < outamo; i++) { var idx = getOutputJunctionIndex(i); jun = outputs[| idx]; jun.x = xx; jun.y = _in; _in += 24 * _s * jun.isVisible(); } } static onValueFromUpdate = function(index) { if(LOADING || APPENDING) return; if(index + data_length >= ds_list_size(inputs)) createNewSurface(); } static drawOverlay = function(active, _x, _y, _s, _mx, _my, _snx, _sny) { var pad = inputs[| 0].getValue(); var ww = overlay_w; var hh = overlay_h; var x0 = _x + pad[2] * _s; var x1 = _x + (ww - pad[0]) * _s; var y0 = _y + pad[1] * _s; var y1 = _y + (hh - pad[3]) * _s; if(input_dragging > -1) { if(drag_type == NODE_COMPOSE_DRAG.move) { var _dx = (_mx - dragging_mx) / _s; var _dy = (_my - dragging_my) / _s; if(key_mod_press(SHIFT)) { if(abs(_dx) > abs(_dy) + ui(16)) _dy = 0; else if(abs(_dy) > abs(_dx) + ui(16)) _dx = 0; else { _dx = max(_dx, _dy); _dy = _dx; } } var pos_x = value_snap(dragging_sx + _dx, _snx); var pos_y = value_snap(dragging_sy + _dy, _sny); if(key_mod_press(ALT)) { var _surf = current_data[input_dragging - 1]; var _sw = surface_get_width(_surf); var _sh = surface_get_height(_surf); var x0 = pos_x, x1 = pos_x + _sw; var y0 = pos_y, y1 = pos_y + _sh; var xc = (x0 + x1) / 2; var yc = (y0 + y1) / 2; var snap = 4; draw_set_color(COLORS._main_accent); if(abs(x0 - 0) < snap) { pos_x = 0; draw_line_width(_x + _s * 0, 0, _x + _s * 0, WIN_H, 2); } if(abs(y0 - 0) < snap) { pos_y = 0; draw_line_width(0, _y + _s * 0, WIN_W, _y + _s * 0, 2); } if(abs(x1 - ww) < snap) { pos_x = ww - _sw; draw_line_width(_x + _s * ww, 0, _x + _s * ww, WIN_H, 2); } if(abs(y1 - hh) < snap) { pos_y = hh - _sh; draw_line_width(0, _y + _s * hh, WIN_W, _y + _s * hh, 2); } if(abs(xc - ww / 2) < snap) { pos_x = ww / 2 - _sw / 2; draw_line_width(_x + _s * ww / 2, 0, _x + _s * ww / 2, WIN_H, 2); } if(abs(yc - hh / 2) < snap) { pos_y = hh / 2 - _sh / 2; draw_line_width(0, _y + _s * hh / 2, WIN_W, _y + _s * hh / 2, 2); } } if(inputs[| input_dragging].setValue([ pos_x, pos_y ])) UNDO_HOLDING = true; } else if(drag_type == NODE_COMPOSE_DRAG.rotate) { var aa = point_direction(rot_anc_x, rot_anc_y, _mx, _my); var da = angle_difference(dragging_mx, aa); var sa; if(key_mod_press(CTRL)) sa = round((dragging_sx - da) / 15) * 15; else sa = dragging_sx - da; if(inputs[| input_dragging].setValue(sa)) UNDO_HOLDING = true; } else if(drag_type == NODE_COMPOSE_DRAG.scale) { var _surf = inputs[| surf_dragging + 0].getValue(); var _rot = inputs[| surf_dragging + 2].getValue(); var _sw = surface_get_width(_surf); var _sh = surface_get_width(_surf); var _p = point_rotate(_mx - dragging_mx, _my - dragging_my, 0, 0, -_rot); var sca_x = _p[0] / _s / _sw * 2; var sca_y = _p[1] / _s / _sh * 2; if(key_mod_press(SHIFT)) { sca_x = min(sca_x, sca_y); sca_y = min(sca_x, sca_y); } if(inputs[| input_dragging].setValue([ sca_x, sca_y ])) UNDO_HOLDING = true; } if(mouse_release(mb_left)) { input_dragging = -1; UNDO_HOLDING = false; } } var hovering = -1; var hovering_type = 0; var _vis = attributes[? "layer_visible"]; var _sel = attributes[? "layer_selectable"]; var amo = (ds_list_size(inputs) - input_fix_len) / data_length; if(array_length(current_data) < input_fix_len + amo * data_length) return; for(var i = 0; i < amo; i++) { var vis = _vis[| i]; var sel = _sel[| i]; if(!vis) continue; var index = input_fix_len + i * data_length; var _surf = current_data[index + 0]; var _pos = current_data[index + 1]; var _rot = current_data[index + 2]; var _sca = current_data[index + 3]; if(!_surf || is_array(_surf)) continue; var _ww = surface_get_width(_surf); var _hh = surface_get_height(_surf); var _sw = _ww * _sca[0]; var _sh = _hh * _sca[1]; var cx = _pos[0] + _ww / 2; var cy = _pos[1] + _hh / 2; var _d0 = point_rotate(cx - _sw / 2, cy - _sh / 2, cx, cy, _rot); var _d1 = point_rotate(cx - _sw / 2, cy + _sh / 2, cx, cy, _rot); var _d2 = point_rotate(cx + _sw / 2, cy - _sh / 2, cx, cy, _rot); var _d3 = point_rotate(cx + _sw / 2, cy + _sh / 2, cx, cy, _rot); var _rr = point_rotate(cx, cy - _sh / 2 - 1, cx, cy, _rot); _d0[0] = overlay_x(_d0[0], _x, _s); _d0[1] = overlay_y(_d0[1], _y, _s); _d1[0] = overlay_x(_d1[0], _x, _s); _d1[1] = overlay_y(_d1[1], _y, _s); _d2[0] = overlay_x(_d2[0], _x, _s); _d2[1] = overlay_y(_d2[1], _y, _s); _d3[0] = overlay_x(_d3[0], _x, _s); _d3[1] = overlay_y(_d3[1], _y, _s); _rr[0] = overlay_x(_rr[0], _x, _s); _rr[1] = overlay_y(_rr[1], _y, _s); var _borcol = COLORS.node_composite_overlay_border; var _ri = 0; var _si = 0; if(!sel) continue; if(point_in_circle(_mx, _my, _d3[0], _d3[1], 12)) { hovering = index; hovering_type = NODE_COMPOSE_DRAG.scale; _si = 1; } else if(point_in_rectangle_points(_mx, _my, _d0[0], _d0[1], _d1[0], _d1[1], _d2[0], _d2[1], _d3[0], _d3[1])) { hovering = index; hovering_type = NODE_COMPOSE_DRAG.move; } else if(point_in_circle(_mx, _my, _rr[0], _rr[1], 12)) { hovering = index; hovering_type = NODE_COMPOSE_DRAG.rotate; _ri = 1; } draw_sprite_colored(THEME.anchor_rotate, _ri, _rr[0], _rr[1],, _rot); draw_sprite_colored(THEME.anchor_scale, _si, _d3[0], _d3[1],, _rot); draw_set_color(_borcol); draw_line(_d0[0], _d0[1], _d1[0], _d1[1]); draw_line(_d0[0], _d0[1], _d2[0], _d2[1]); draw_line(_d3[0], _d3[1], _d1[0], _d1[1]); draw_line(_d3[0], _d3[1], _d2[0], _d2[1]); } if(hovering != -1) { var _surf = current_data[hovering]; var _pos = current_data[hovering + 1]; var _rot = current_data[hovering + 2]; var _sca = current_data[hovering + 3]; var _ww = surface_get_width(_surf); var _hh = surface_get_height(_surf); var _dx0 = _x + _pos[0] * _s; var _dy0 = _y + _pos[1] * _s; var _dx1 = _dx0 + _ww * _s; var _dy1 = _dy0 + _hh * _s; var _sw = _ww * _sca[0]; var _sh = _hh * _sca[1]; var cx = _pos[0] + _ww / 2; var cy = _pos[1] + _hh / 2; var _d0 = point_rotate(cx - _sw / 2, cy - _sh / 2, cx, cy, _rot); var _d1 = point_rotate(cx - _sw / 2, cy + _sh / 2, cx, cy, _rot); var _d2 = point_rotate(cx + _sw / 2, cy - _sh / 2, cx, cy, _rot); var _d3 = point_rotate(cx + _sw / 2, cy + _sh / 2, cx, cy, _rot); _d0[0] = overlay_x(_d0[0], _x, _s); _d0[1] = overlay_y(_d0[1], _y, _s); _d1[0] = overlay_x(_d1[0], _x, _s); _d1[1] = overlay_y(_d1[1], _y, _s); _d2[0] = overlay_x(_d2[0], _x, _s); _d2[1] = overlay_y(_d2[1], _y, _s); _d3[0] = overlay_x(_d3[0], _x, _s); _d3[1] = overlay_y(_d3[1], _y, _s); if(hovering_type == NODE_COMPOSE_DRAG.move) { draw_set_color(COLORS._main_accent); draw_line_round(_d0[0], _d0[1], _d1[0], _d1[1], 2); draw_line_round(_d0[0], _d0[1], _d2[0], _d2[1], 2); draw_line_round(_d3[0], _d3[1], _d1[0], _d1[1], 2); draw_line_round(_d3[0], _d3[1], _d2[0], _d2[1], 2); if(mouse_press(mb_left, active)) { surf_dragging = hovering; input_dragging = hovering + 1; drag_type = hovering_type; dragging_sx = _pos[0]; dragging_sy = _pos[1]; dragging_mx = _mx; dragging_my = _my; } } else if(hovering_type == NODE_COMPOSE_DRAG.rotate) { //rot if(mouse_press(mb_left, active)) { surf_dragging = hovering; input_dragging = hovering + 2; drag_type = hovering_type; dragging_sx = _rot; rot_anc_x = _dx0 + _ww / 2 * _s; rot_anc_y = _dy0 + _hh / 2 * _s; dragging_mx = point_direction(rot_anc_x, rot_anc_y, _mx, _my); } } else if(hovering_type == NODE_COMPOSE_DRAG.scale) { //sca if(mouse_press(mb_left, active)) { surf_dragging = hovering; input_dragging = hovering + 3; drag_type = hovering_type; dragging_sx = _sca[0]; dragging_sy = _sca[1]; dragging_mx = _dx0 + _ww / 2 * _s; dragging_my = _dy0 + _hh / 2 * _s; } } } if(layer_remove > -1) { deleteLayer(layer_remove); layer_remove = -1; } } static process_data = function(_outSurf, _data, _output_index, _array_index) { if(_output_index == 1) return atlas_data; if(_output_index == 0 && _array_index == 0) atlas_data = []; if(array_length(_data) < 4) return _outSurf; var _pad = _data[0]; var _dim_type = _data[1]; var _dim = _data[2]; var base = _data[3]; var cDep = attrDepth(); var ww = 0, hh = 0; switch(_dim_type) { case COMPOSE_OUTPUT_SCALING.first : inputs[| 2].setVisible(false); ww = surface_get_width(base); hh = surface_get_height(base); break; case COMPOSE_OUTPUT_SCALING.largest : inputs[| 2].setVisible(false); for(var i = input_fix_len; i < array_length(_data) - data_length; i += data_length) { var _s = _data[i]; ww = max(ww, surface_get_width(_s)); hh = max(hh, surface_get_height(_s)); } break; case COMPOSE_OUTPUT_SCALING.constant : inputs[| 2].setVisible(true); ww = _dim[0]; hh = _dim[1]; break; } ww += _pad[0] + _pad[2]; hh += _pad[1] + _pad[3]; overlay_w = ww; overlay_h = hh; if(is_surface(base)) _outSurf = surface_size_to(_outSurf, ww, hh, cDep); for(var i = 0; i < 2; i++) { temp_surface[i] = surface_verify(temp_surface[i], surface_get_width(_outSurf), surface_get_height(_outSurf), cDep); surface_set_target(temp_surface[i]); DRAW_CLEAR surface_reset_target(); } var res_index = 0, bg = 0; var imageAmo = (ds_list_size(inputs) - input_fix_len) / data_length; var _vis = attributes[? "layer_visible"]; surface_set_shader(_outSurf, sh_sample, true, BLEND.alphamulp); for(var i = 0; i < imageAmo; i++) { var vis = _vis[| i]; if(!vis) continue; var startDataIndex = input_fix_len + i * data_length; var _s = _data[startDataIndex + 0]; var _pos = _data[startDataIndex + 1]; var _rot = _data[startDataIndex + 2]; var _sca = _data[startDataIndex + 3]; if(!_s || is_array(_s)) continue; var _ww = surface_get_width(_s); var _hh = surface_get_height(_s); var _sw = _ww * _sca[0]; var _sh = _hh * _sca[1]; var cx = _pos[0] + _ww / 2; var cy = _pos[1] + _hh / 2; var _d0 = point_rotate(cx - _sw / 2, cy - _sh / 2, cx, cy, _rot); shader_set_interpolation(_s); array_push(atlas_data, new SurfaceAtlas(_s, [ _d0[0], _d0[1] ], _rot, [ _sca[0], _sca[1] ])); draw_surface_ext_safe(_s, _d0[0], _d0[1], _sca[0], _sca[1], _rot); } surface_reset_shader(); return _outSurf; } static postDeserialize = function() { var _inputs = load_map[? "inputs"]; for(var i = input_fix_len; i < ds_list_size(_inputs); i += data_length) createNewSurface(); } static attributeSerialize = function() { var att = ds_map_create(); ds_map_add_list(att, "layer_visible", ds_list_clone(attributes[? "layer_visible"])); ds_map_add_list(att, "layer_selectable", ds_list_clone(attributes[? "layer_selectable"])); return att; } static attributeDeserialize = function(attr) { if(ds_map_exists(attr, "layer_visible")) attributes[? "layer_visible"] = ds_list_clone(attr[? "layer_visible"], true); if(ds_map_exists(attr, "layer_selectable")) attributes[? "layer_selectable"] = ds_list_clone(attr[? "layer_selectable"], true); } }