enum OUTPUT_SCALING { same_as_input, constant, relative, scale } function Node_Transform(_x, _y, _group = noone) : Node_Processor(_x, _y, _group) constructor { name = "Transform"; inputs[| 0] = nodeValue("Surface in", self, JUNCTION_CONNECT.input, VALUE_TYPE.surface, noone); inputs[| 1] = nodeValue("Output dimension", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, [1, 1]) .setDisplay(VALUE_DISPLAY.vector) .setVisible(false); inputs[| 2] = nodeValue("Position", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, [ 0.5, 0.5 ]) .setDisplay(VALUE_DISPLAY.vector) .setUnitRef(function(index) { return getDimension(index); }, VALUE_UNIT.reference); inputs[| 3] = nodeValue("Anchor", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, [ 0.5, 0.5 ]) .setDisplay(VALUE_DISPLAY.vector, { #region side_button : new buttonAnchor(function(ind) { switch(ind) { case 0 : inputs[| 3].setValue([ 0.0, 0.0 ]); break; case 1 : inputs[| 3].setValue([ 0.5, 0.0 ]); break; case 2 : inputs[| 3].setValue([ 1.0, 0.0 ]); break; case 3 : inputs[| 3].setValue([ 0.0, 0.5 ]); break; case 4 : inputs[| 3].setValue([ 0.5, 0.5 ]); break; case 5 : inputs[| 3].setValue([ 1.0, 0.5 ]); break; case 6 : inputs[| 3].setValue([ 0.0, 1.0 ]); break; case 7 : inputs[| 3].setValue([ 0.5, 1.0 ]); break; case 8 : inputs[| 3].setValue([ 1.0, 1.0 ]); break; } }) }); #endregion inputs[| 4] = nodeValue("Relative anchor", self, JUNCTION_CONNECT.input, VALUE_TYPE.boolean, true); inputs[| 5] = nodeValue("Rotation", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 0) .setDisplay(VALUE_DISPLAY.rotation); inputs[| 6] = nodeValue("Scale", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, [ 1, 1 ]) .setDisplay(VALUE_DISPLAY.vector); inputs[| 7] = nodeValue("Render Mode", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 0) .setDisplay(VALUE_DISPLAY.enum_button, [ "Normal", "Tile", "Wrap" ]); inputs[| 8] = nodeValue("Rotate by velocity", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, 0, "Make the surface rotates to follow its movement.") .setDisplay(VALUE_DISPLAY.slider); inputs[| 9] = nodeValue("Output dimension type", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, OUTPUT_SCALING.same_as_input) .setDisplay(VALUE_DISPLAY.enum_scroll, [ "Same as input", "Constant", "Relative to input", "Transformed" ]); inputs[| 10] = nodeValue("Round position", self, JUNCTION_CONNECT.input, VALUE_TYPE.boolean, false, "Round position to the nearest integer value to avoid jittering."); inputs[| 11] = nodeValue("Active", self, JUNCTION_CONNECT.input, VALUE_TYPE.boolean, true); active_index = 11; inputs[| 12] = nodeValue("Echo", self, JUNCTION_CONNECT.input, VALUE_TYPE.boolean, false); inputs[| 13] = nodeValue("Echo amount", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 8); inputs[| 14] = nodeValue("Opacity", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, 1) .setDisplay(VALUE_DISPLAY.slider); input_display_list = [ 11, 0, ["Output", true], 9, 1, 7, ["Position", false], 2, 10, ["Rotation", false], 3, 5, 8, ["Scale", false], 6, ["Render", false], 14, ["Echo", true, 12], 13, ]; outputs[| 0] = nodeValue("Surface out", self, JUNCTION_CONNECT.output, VALUE_TYPE.surface, noone); attribute_surface_depth(); attribute_interpolation(); vel = 0; prev_pos = [ 0, 0 ]; prev_data = noone; static getDimension = function(arr = 0) { #region var _surf = getSingleValue(0, arr); var _out_type = getSingleValue(9, arr); var _out = getSingleValue(1, arr); var _rotate = getSingleValue(5, arr); var _scale = getSingleValue(6, arr); var ww, hh; var sw = surface_get_width_safe(_surf); var sh = surface_get_height_safe(_surf); switch(_out_type) { case OUTPUT_SCALING.same_as_input : ww = sw; hh = sh; break; case OUTPUT_SCALING.relative : ww = sw * _out[0]; hh = sh * _out[1]; break; case OUTPUT_SCALING.constant : ww = _out[0]; hh = _out[1]; break; case OUTPUT_SCALING.scale : ww = sw * _scale[0]; hh = sh * _scale[1]; var p0 = point_rotate( 0, 0, ww / 2, hh / 2, _rotate); var p1 = point_rotate(ww, 0, ww / 2, hh / 2, _rotate); var p2 = point_rotate( 0, hh, ww / 2, hh / 2, _rotate); var p3 = point_rotate(ww, hh, ww / 2, hh / 2, _rotate); var minx = min(p0[0], p1[0], p2[0], p3[0]); var maxx = max(p0[0], p1[0], p2[0], p3[0]); var miny = min(p0[1], p1[1], p2[1], p3[1]); var maxy = max(p0[1], p1[1], p2[1], p3[1]); ww = maxx - minx; hh = maxy - miny; break; } return [ ww, hh ]; } #endregion static centerAnchor = function() { #region var _surf = getInputData(0); var _out_type = getInputData(9); var _out = getInputData(1); var _sca = getInputData(6); if(is_array(_surf)) { if(array_length(_surf) == 0) return; _surf = _surf[preview_index]; } inputs[| 3].setValue([ 0.5, 0.5 ]); inputs[| 2].setValue([ surface_get_width_safe(_surf) / 2, surface_get_height_safe(_surf) / 2 ]); } #endregion static step = function() { #region var pos = getSingleValue(2); var anc = getSingleValue(3); var _b = inputs[| 3].editWidget.side_button; var _a = anc[0] * 2 + anc[1] * 20; switch(_a) { case 0 : _b.index = 0; break; case 1 : _b.index = 1; break; case 2 : _b.index = 2; break; case 10 : _b.index = 3; break; case 11 : _b.index = 4; break; case 12 : _b.index = 5; break; case 20 : _b.index = 6; break; case 21 : _b.index = 7; break; case 22 : _b.index = 8; break; default : _b.index = -1; } if(!PROJECT.animator.frame_progress) return; if(IS_FIRST_FRAME) { vel = 0; prev_pos[0] = pos[0]; prev_pos[1] = pos[1]; } else { vel = point_direction(prev_pos[0], prev_pos[1], pos[0], pos[1]); prev_pos[0] = pos[0]; prev_pos[1] = pos[1]; } } #endregion static processData_prebatch = function() { shader_preset_interpolation(); } static processData_postbatch = function() { shader_postset_interpolation(); } static processData = function(_outSurf, _data, _output_index, _array_index) { #region var ins = _data[0]; var out_type = _data[9]; var out = _data[1]; var pos = [ _data[2][0], _data[2][1] ]; var pos_exact = _data[10]; var anc = [ _data[3][0], _data[3][1] ]; var rot_vel = vel * _data[8]; var rot = _data[5] + rot_vel; var sca = _data[6]; var mode = _data[7]; var echo = _data[12]; var echo_amo = _data[13]; var alp = _data[14]; var cDep = attrDepth(); var ww = surface_get_width_safe(ins); var hh = surface_get_height_safe(ins); var _ww = ww; var _hh = hh; if(_ww <= 1 && _hh <= 1) return _outSurf; switch(out_type) { #region output dimension case OUTPUT_SCALING.same_as_input : inputs[| 1].setVisible(false); break; case OUTPUT_SCALING.constant : inputs[| 1].setVisible(true); _ww = out[0]; _hh = out[1]; break; case OUTPUT_SCALING.relative : inputs[| 1].setVisible(true); _ww = ww * out[0]; _hh = hh * out[1]; break; case OUTPUT_SCALING.scale : inputs[| 1].setVisible(false); _ww = ww * sca[0]; _hh = hh * sca[1]; var p0 = point_rotate( 0, 0, _ww / 2, _hh / 2, rot); var p1 = point_rotate(_ww, 0, _ww / 2, _hh / 2, rot); var p2 = point_rotate( 0, _hh, _ww / 2, _hh / 2, rot); var p3 = point_rotate(_ww, _hh, _ww / 2, _hh / 2, rot); var minx = min(p0[0], p1[0], p2[0], p3[0]); var maxx = max(p0[0], p1[0], p2[0], p3[0]); var miny = min(p0[1], p1[1], p2[1], p3[1]); var maxy = max(p0[1], p1[1], p2[1], p3[1]); _ww = maxx - minx; _hh = maxy - miny; break; } #endregion if(_ww <= 0 || _hh <= 0) return; _outSurf = surface_verify(_outSurf, _ww, _hh, cDep); anc[0] *= ww * sca[0]; anc[1] *= hh * sca[1]; pos[0] -= anc[0]; pos[1] -= anc[1]; pos = point_rotate(pos[0], pos[1], pos[0] + anc[0], pos[1] + anc[1], rot); var draw_x, draw_y; draw_x = pos[0]; draw_y = pos[1]; if(pos_exact) { draw_x = round(draw_x); draw_y = round(draw_y); } if(mode == 1) { #region // Tile surface_set_shader(_outSurf); shader_set_interpolation(ins); draw_surface_tiled_ext_safe(ins, draw_x, draw_y, sca[0], sca[1], rot, c_white, alp); surface_reset_shader(); #endregion } else { #region // Normal or wrap surface_set_shader(_outSurf); shader_set_interpolation(ins); if(echo && CURRENT_FRAME && prev_data != noone) { var _pre = prev_data[_array_index]; for( var i = 0; i <= echo_amo; i++ ) { var rat = i / echo_amo; var _px = lerp(_pre[0][0], pos[0], rat); var _py = lerp(_pre[0][1], pos[1], rat); var _rt = lerp(_pre[1], rot, rat); var _sx = lerp(_pre[2][0], sca[0], rat); var _sy = lerp(_pre[2][1], sca[1], rat); var _ps = point_rotate(_px, _py, _px + anc[0], _py + anc[1], rot); var _dx = _ps[0]; var _dy = _ps[1]; if(pos_exact) { _dx = round(_dx); _dy = round(_dy); } draw_surface_ext_safe(ins, _dx, _dy, _sx, _sy, _rt, c_white, alp); } } else draw_surface_ext_safe(ins, draw_x, draw_y, sca[0], sca[1], rot, c_white, alp); if(mode == 2) { draw_surface_ext_safe(ins, draw_x - _ww, draw_y - _hh, sca[0], sca[1], rot, c_white, alp); draw_surface_ext_safe(ins, draw_x, draw_y - _hh, sca[0], sca[1], rot, c_white, alp); draw_surface_ext_safe(ins, draw_x + _ww, draw_y - _hh, sca[0], sca[1], rot, c_white, alp); draw_surface_ext_safe(ins, draw_x - _ww, draw_y, sca[0], sca[1], rot, c_white, alp); draw_surface_ext_safe(ins, draw_x + _ww, draw_y, sca[0], sca[1], rot, c_white, alp); draw_surface_ext_safe(ins, draw_x - _ww, draw_y + _hh, sca[0], sca[1], rot, c_white, alp); draw_surface_ext_safe(ins, draw_x, draw_y + _hh, sca[0], sca[1], rot, c_white, alp); draw_surface_ext_safe(ins, draw_x + _ww, draw_y + _hh, sca[0], sca[1], rot, c_white, alp); } surface_reset_shader(); #endregion } prev_data[_array_index] = [ [ pos[0], pos[1] ], rot, [ sca[0], sca[1] ], ]; return _outSurf; } #endregion overlay_dragging = 0; corner_dragging = 0; overlay_drag_mx = 0; overlay_drag_my = 0; overlay_drag_sx = 0; overlay_drag_sy = 0; overlay_drag_px = 0; overlay_drag_py = 0; overlay_drag_ma = 0; overlay_drag_sa = 0; static drawOverlay = function(hover, active, _x, _y, _s, _mx, _my, _snx, _sny) { #region PROCESSOR_OVERLAY_CHECK var _surf = getInputData(0); if(is_array(_surf)) { if(array_length(_surf) == 0) return; _surf = _surf[preview_index]; } var _surf_out = outputs[| 0].getValue(); if(is_array(_surf_out)) { if(array_length(_surf_out) == 0) return; _surf_out = _surf_out[preview_index]; } var __pos = getSingleValue(2); var pos = [ __pos[0], __pos[1] ]; var _pos = [ __pos[0], __pos[1] ]; var __anc = getSingleValue(3); var anc = [ __anc[0], __anc[1] ]; var _anc = [ __anc[0], __anc[1] ]; var rot = getSingleValue(5); var sca = getSingleValue(6); var srw = surface_get_width_safe(_surf); var srh = surface_get_height_safe(_surf); var ow = surface_get_width_safe(_surf_out); var oh = surface_get_height_safe(_surf_out); var ww = srw * sca[0]; var hh = srh * sca[1]; anc[0] *= ww; anc[1] *= hh; pos[0] -= anc[0]; pos[1] -= anc[1]; #region bounding box var bx0 = _x + pos[0] * _s; var by0 = _y + pos[1] * _s; var bx1 = _x + (pos[0] + ww) * _s; var by1 = _y + (pos[1] + hh) * _s; var bx2 = _x + (pos[0] + ww) * _s + 18; var by2 = _y + (pos[1] + hh) * _s + 18; var bax = _x + (pos[0] + anc[0]) * _s; var bay = _y + (pos[1] + anc[1]) * _s; var tl = point_rotate(bx0, by0, bax, bay, rot); var tr = point_rotate(bx1, by0, bax, bay, rot); var bl = point_rotate(bx0, by1, bax, bay, rot); var br = point_rotate(bx1, by1, bax, bay, rot); var sz = point_rotate(bx2, by2, bax, bay, rot); var rth = point_rotate((bx0 + bx1) / 2, by0 - 16, bax, bay, rot); var a_index = 0; var r_index = 0; var tl_index = 0; var tr_index = 0; var bl_index = 0; var br_index = 0; var sz_index = 0; draw_set_color(COLORS._main_accent); draw_line(tl[0], tl[1], tr[0], tr[1]); draw_line(tl[0], tl[1], bl[0], bl[1]); draw_line(tr[0], tr[1], br[0], br[1]); draw_line(bl[0], bl[1], br[0], br[1]); if(point_in_circle(_mx, _my, bax, bay, 8)) a_index = 1; else if(point_in_circle(_mx, _my, rth[0], rth[1], 8)) r_index = 1; else if(point_in_circle(_mx, _my, tl[0], tl[1], 8)) tl_index = 1; else if(point_in_circle(_mx, _my, tr[0], tr[1], 8)) tr_index = 1; else if(point_in_circle(_mx, _my, bl[0], bl[1], 8)) bl_index = 1; else if(point_in_circle(_mx, _my, br[0], br[1], 8)) br_index = 1; else if(point_in_circle(_mx, _my, sz[0], sz[1], 8)) sz_index = 1; draw_sprite_colored(THEME.anchor, a_index, bax, bay); draw_sprite_colored(THEME.anchor_selector, tl_index, tl[0], tl[1]); draw_sprite_colored(THEME.anchor_selector, tr_index, tr[0], tr[1]); draw_sprite_colored(THEME.anchor_selector, bl_index, bl[0], bl[1]); draw_sprite_colored(THEME.anchor_selector, br_index, br[0], br[1]); draw_sprite_colored(THEME.anchor_scale, sz_index, sz[0], sz[1], 1, rot); draw_sprite_colored(THEME.anchor_rotate, r_index, rth[0], rth[1], 1, rot); #endregion if(overlay_dragging && overlay_dragging < 3) { //Transform var px = _mx - overlay_drag_mx; var py = _my - overlay_drag_my; var pos_x, pos_y; if(key_mod_press(SHIFT)) { var ang = round(point_direction(overlay_drag_mx, overlay_drag_my, _mx, _my) / 45) * 45; var dist = point_distance(overlay_drag_mx, overlay_drag_my, _mx, _my) / _s; pos_x = overlay_drag_sx + lengthdir_x(dist, ang); pos_y = overlay_drag_sy + lengthdir_y(dist, ang); } else { pos_x = overlay_drag_sx + px / _s; pos_y = overlay_drag_sy + py / _s; } pos_x = value_snap(pos_x, _snx); pos_y = value_snap(pos_y, _sny); if(overlay_dragging == 1) { //Move if(inputs[| 2].setValue([ pos_x, pos_y ])) UNDO_HOLDING = true; } else if(overlay_dragging == 2) { //Move anchor var nanx = pos_x / ww; var nany = pos_y / hh; if(key_mod_press(ALT)) { var modi = false; modi |= inputs[| 3].setValue([ nanx, nany ]); modi |= inputs[| 2].setValue([ overlay_drag_px + pos_x, overlay_drag_py + pos_y ]); if(modi) UNDO_HOLDING = true; } else { if(inputs[| 3].setValue([ nanx, nany ])) UNDO_HOLDING = true; } } if(mouse_release(mb_left)) { overlay_dragging = 0; UNDO_HOLDING = false; } } else if(overlay_dragging == 3) { //Angle var aa = point_direction(bax, bay, _mx, _my); var da = angle_difference(overlay_drag_ma, aa); var sa; if(key_mod_press(CTRL)) sa = round((overlay_drag_sa - da) / 15) * 15; else sa = overlay_drag_sa - da; if(inputs[| 5].setValue(sa)) UNDO_HOLDING = true; if(mouse_release(mb_left)) { overlay_dragging = 0; UNDO_HOLDING = false; } } else if(overlay_dragging == 4) { //Scale var ol_x = (overlay_drag_mx - _x) / _s; var ol_y = (overlay_drag_my - _y) / _s; var ml_x = (_mx - _x) / _s; var ml_y = (_my - _y) / _s; var os_x = value_snap(ol_x, _snx); var os_y = value_snap(ol_y, _sny); var ms_x = value_snap(ml_x, _snx); var ms_y = value_snap(ml_y, _sny); var _p = point_rotate(ms_x - os_x, ms_y - os_y, 0, 0, -rot); var _sw = _p[0] / srw; var _sh = _p[1] / srh; var sw, sh; if(corner_dragging == 0) { sw = -_sw / _anc[0]; sh = -_sh / _anc[1]; } else if(corner_dragging == 1) { sw = _sw / (1 - _anc[0]); sh = -_sh / _anc[1]; } else if(corner_dragging == 2) { sw = -_sw / _anc[0]; sh = _sh / (1 - _anc[1]); } else if(corner_dragging == 3) { sw = _sw / (1 - _anc[0]); sh = _sh / (1 - _anc[1]); } else if(corner_dragging == 4) { sw = _sw / (1 - _anc[0]); sh = _sh / (1 - _anc[1]); } var _sw = overlay_drag_sx + sw; var _sh = overlay_drag_sy + sh; if(key_mod_press(SHIFT)) { _sw = max(_sw, _sh); _sh = _sw; } if(inputs[| 6].setValue([ _sw, _sh ])) UNDO_HOLDING = true; if(mouse_release(mb_left)) { overlay_dragging = 0; UNDO_HOLDING = false; } } if(overlay_dragging == 0 && mouse_press(mb_left, active)) { if(point_in_circle(_mx, _my, bax, bay, 8)) { overlay_dragging = 2; overlay_drag_mx = _mx; overlay_drag_my = _my; overlay_drag_sx = anc[0]; overlay_drag_sy = anc[1]; overlay_drag_px = pos[0]; overlay_drag_py = pos[1]; } else if(point_in_circle(_mx, _my, tl[0], tl[1], 8) || point_in_circle(_mx, _my, tr[0], tr[1], 8) || point_in_circle(_mx, _my, bl[0], bl[1], 8) || point_in_circle(_mx, _my, br[0], br[1], 8) || point_in_circle(_mx, _my, sz[0], sz[1], 8)) { overlay_dragging = 4; if(point_in_circle(_mx, _my, tl[0], tl[1], 8)) corner_dragging = 0; else if(point_in_circle(_mx, _my, tr[0], tr[1], 8)) corner_dragging = 1; else if(point_in_circle(_mx, _my, bl[0], bl[1], 8)) corner_dragging = 2; else if(point_in_circle(_mx, _my, br[0], br[1], 8)) corner_dragging = 3; else if(point_in_circle(_mx, _my, sz[0], sz[1], 8)) corner_dragging = 4; overlay_drag_mx = _mx; overlay_drag_my = _my; overlay_drag_sx = sca[0]; overlay_drag_sy = sca[1]; } else if(point_in_circle(_mx, _my, rth[0], rth[1], 8)) { overlay_dragging = 3; overlay_drag_ma = point_direction(bax, bay, _mx, _my); overlay_drag_sa = rot; } else if(point_in_triangle(_mx, _my, tl[0], tl[1], tr[0], tr[1], bl[0], bl[1]) || point_in_triangle(_mx, _my, tr[0], tr[1], bl[0], bl[1], br[0], br[1])) { overlay_dragging = 1; overlay_drag_mx = _mx; overlay_drag_my = _my; overlay_drag_sx = _pos[0]; overlay_drag_sy = _pos[1]; } } #region path if(inputs[| 2].is_anim && inputs[| 2].isLeaf() && !inputs[| 2].sep_axis) { var posInp = inputs[| 2]; var allPos = posInp.animator.values; var ox, oy, nx, ny; draw_set_color(COLORS._main_accent); for( var i = 0; i < ds_list_size(allPos); i++ ) { var pos = allPos[| i].value; var _pos = [ pos[0], pos[1] ]; if(posInp.unit.mode == VALUE_UNIT.reference) { _pos[0] *= ow; _pos[1] *= oh; } nx = _x + _pos[0] * _s; ny = _y + _pos[1] * _s; draw_set_alpha(1); draw_circle_prec(nx, ny, 4, false); if(i) { draw_set_alpha(0.5); draw_line_dashed(ox, oy, nx, ny); } ox = nx; oy = ny; } draw_set_alpha(1); } #endregion } #endregion }