function Node_Path(_x, _y, _group = noone) : Node(_x, _y, _group) constructor { name = "Path"; previewable = false; w = 96; inputs[| 0] = nodeValue("Path progress", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, 0, "Sample position from path.") .setDisplay(VALUE_DISPLAY.slider, [0, 1, 0.01]); inputs[| 1] = nodeValue("Loop", self, JUNCTION_CONNECT.input, VALUE_TYPE.boolean, false) .rejectArray(); inputs[| 2] = nodeValue("Progress mode", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 0) .setDisplay(VALUE_DISPLAY.enum_scroll, ["Entire line", "Segment"]) .rejectArray(); inputs[| 3] = nodeValue("Round anchor", self, JUNCTION_CONNECT.input, VALUE_TYPE.boolean, false) .rejectArray(); input_display_list = [ ["Path", false], 0, 2, 1, 3, ["Anchors", false], ]; input_fix_len = ds_list_size(inputs); input_display_list_len = array_length(input_display_list); function createAnchor(_x, _y, _dxx = 0, _dxy = 0, _dyx = 0, _dyy = 0) { var index = ds_list_size(inputs); inputs[| index] = nodeValue("Anchor", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, [ _x, _y, _dxx, _dxy, _dyx, _dyy ]) .setDisplay(VALUE_DISPLAY.vector); recordAction(ACTION_TYPE.var_modify, self, [ array_clone(input_display_list), "input_display_list" ]); recordAction(ACTION_TYPE.list_insert, inputs, [ inputs[| index], index, "add path anchor point" ]); array_push(input_display_list, index); return inputs[| index]; } outputs[| 0] = nodeValue("Position out", self, JUNCTION_CONNECT.output, VALUE_TYPE.float, [ 0, 0 ]) .setDisplay(VALUE_DISPLAY.vector); outputs[| 1] = nodeValue("Path data", self, JUNCTION_CONNECT.output, VALUE_TYPE.pathnode, self); outputs[| 2] = nodeValue("Anchors", self, JUNCTION_CONNECT.output, VALUE_TYPE.float, []); tool_pathDrawer = new NodeTool( "Draw path", THEME.path_tools_draw ) .addSetting("Smoothness", VALUE_TYPE.float, function(val) { tool_pathDrawer.attribute.thres = val; }, "thres", 4) .addSetting("Replace", VALUE_TYPE.boolean, function() { tool_pathDrawer.attribute.create = !tool_pathDrawer.attribute.create; }, "create", true); tools = [ new NodeTool( "Transform", THEME.path_tools_transform ), new NodeTool( "Anchor add / remove (+ Shift)", THEME.path_tools_add ), new NodeTool( "Edit Control point", THEME.path_tools_anchor ), tool_pathDrawer, new NodeTool( "Rectangle path", THEME.path_tools_rectangle ), new NodeTool( "Circle path", THEME.path_tools_circle ), ]; anchors = []; lengths = []; lengthAccs = []; boundary = []; lengthTotal = 0; drag_point = -1; drag_points = []; drag_type = 0; drag_point_mx = 0; drag_point_my = 0; drag_point_sx = 0; drag_point_sy = 0; transform_type = 0; transform_minx = 0; transform_miny = 0; transform_maxx = 0; transform_maxy = 0; transform_mx = 0; transform_my = 0; static onValueUpdate = function(index = 0) { if(index == 2) { var type = inputs[| 2].getValue(); if(type == 0) inputs[| 0].setDisplay(VALUE_DISPLAY.slider, [0, 1, 0.01]); else if(type == 1) inputs[| 0].setDisplay(VALUE_DISPLAY._default); } } static drawOverlay = function(active, _x, _y, _s, _mx, _my, _snx, _sny) { var sample = PREF_MAP[? "path_resolution"]; var loop = inputs[| 1].getValue(); var ansize = ds_list_size(inputs) - input_fix_len; var _edited = false; if(transform_type > 0) { var dx = _mx - transform_mx; var dy = _my - transform_my; var _transform_minx = transform_minx; var _transform_miny = transform_miny; var _transform_maxx = transform_maxx; var _transform_maxy = transform_maxy; if(transform_type == 1) { transform_minx += dx / _s; transform_miny += dy / _s; } else if(transform_type == 2) { transform_maxx += dx / _s; transform_miny += dy / _s; } else if(transform_type == 3) { transform_minx += dx / _s; transform_maxy += dy / _s; } else if(transform_type == 4) { transform_maxx += dx / _s; transform_maxy += dy / _s; } if(transform_type == 5) { for( var i = input_fix_len; i < ds_list_size(inputs); i++ ) { var p = inputs[| i].getValue(); p[0] += dx / _s; p[1] += dy / _s; if(inputs[| i].setValue(p)) _edited = true; } } else { for( var i = input_fix_len; i < ds_list_size(inputs); i++ ) { var p = inputs[| i].getValue(); p[0] = transform_minx + (p[0] - _transform_minx) / (_transform_maxx - _transform_minx) * (transform_maxx - transform_minx); p[1] = transform_miny + (p[1] - _transform_miny) / (_transform_maxy - _transform_miny) * (transform_maxy - transform_miny); if(inputs[| i].setValue(p)) _edited = true; } } if(_edited) UNDO_HOLDING = true; transform_mx = _mx; transform_my = _my; if(mouse_release(mb_left)) { transform_type = 0; UPDATE |= RENDER_TYPE.full; UNDO_HOLDING = false; } } else if(drag_point > -1) { var dx = value_snap(drag_point_sx + (_mx - drag_point_mx) / _s, _snx); var dy = value_snap(drag_point_sy + (_my - drag_point_my) / _s, _sny); if(drag_type < 2) { var inp = inputs[| input_fix_len + drag_point]; var anc = inp.getValue(); if(drag_type == 0) { //drag anchor point anc[0] = dx; anc[1] = dy; if(key_mod_press(CTRL)) { anc[0] = round(anc[0]); anc[1] = round(anc[1]); } } else if(drag_type == 1) { //drag control 1 anc[2] = dx - anc[0]; anc[3] = dy - anc[1]; if(!key_mod_press(SHIFT)) { anc[4] = -anc[2]; anc[5] = -anc[3]; } if(key_mod_press(CTRL)) { anc[2] = round(anc[2]); anc[3] = round(anc[3]); if(key_mod_press(SHIFT)) { anc[4] = round(anc[4]); anc[5] = round(anc[5]); } } } else if(drag_type == -1) { //drag control 2 anc[4] = dx - anc[0]; anc[5] = dy - anc[1]; if(!key_mod_press(SHIFT)) { anc[2] = -anc[4]; anc[3] = -anc[5]; } if(key_mod_press(CTRL)) { anc[2] = round(anc[2]); anc[3] = round(anc[3]); if(!key_mod_press(SHIFT)) { anc[4] = round(anc[4]); anc[5] = round(anc[5]); } } } if(inp.setValue(anc)) _edited = true; } else if(drag_type == 2) { //pen tools var ox, oy, nx, ny; var pxx = (_mx - _x) / _s; var pxy = (_my - _y) / _s; draw_set_color(COLORS._main_accent); for( var i = 0; i < array_length(drag_points); i++ ) { var _p = drag_points[i]; nx = _x + _p[0] * _s; ny = _y + _p[1] * _s; if(i) draw_line(ox, oy, nx, ny); ox = nx; oy = ny; } if(point_distance(drag_point_mx, drag_point_my, pxx, pxy) > 4 / _s) { array_push(drag_points, [ pxx, pxy ]); drag_point_mx = pxx; drag_point_my = pxy; } if(mouse_release(mb_left)) { var amo = array_length(drag_points); var _p = 0; var points = []; var thres = tool_pathDrawer.attribute.thres; var replace = tool_pathDrawer.attribute.create; var asize = ds_list_size(inputs) - input_fix_len; for( var i = 0; i < amo; i++ ) { var pT = drag_points[i]; if(i == 0 || i == amo - 1) { array_push(points, i); continue; } var maxT = 0; var pF = drag_points[_p]; for( var j = _p; j < i; j++ ) { var pP = drag_points[j]; maxT = max(maxT, distance_to_line(pP[0], pP[1], pF[0], pF[1], pT[0], pT[1])); } if(maxT >= thres) { array_push(points, i); _p = i; } } var amo = array_length(points); if(!replace) amo = min(amo, asize); var i = 0; var anc = []; for( i = 0; i < amo; i++ ) { var ind = replace? i : clamp(i / amo * array_length(points), 0, array_length(points) - 1); var _ind = points[ind]; var _p = drag_points[_ind]; var dxx = 0; var dxy = 0; var dyx = 0; var dyy = 0; if(i > 0 && i < amo - 1) { var _p0 = drag_points[points[i - 1]]; var _p1 = drag_points[points[i + 1]]; var d0 = point_direction(_p0[0], _p0[1], _p[0], _p[1]); var d1 = point_direction(_p[0], _p[1], _p1[0], _p1[1]); var dd = d0 + angle_difference(d1, d0) / 2; var ds0 = point_distance(_p0[0], _p0[1], _p[0], _p[1]); var ds1 = point_distance(_p[0], _p[1], _p1[0], _p1[1]); dxx = lengthdir_x(ds0 / 3, dd + 180); dxy = lengthdir_y(ds0 / 3, dd + 180); dyx = lengthdir_x(ds1 / 3, dd); dyy = lengthdir_y(ds1 / 3, dd); } anc = [_p[0], _p[1], dxx, dxy, dyx, dyy]; if(input_fix_len + i >= ds_list_size(inputs)) createAnchor(_p[0], _p[1], dxx, dxy, dyx, dyy); else inputs[| input_fix_len + i].setValue(anc); } if(!replace) { for(; i < asize; i++ ) inputs[| input_fix_len + i].setValue(anc); } } } else if(drag_type == 3) { var minx = min((_mx - _x) / _s, (drag_point_mx - _x) / _s); var maxx = max((_mx - _x) / _s, (drag_point_mx - _x) / _s); var miny = min((_my - _y) / _s, (drag_point_my - _y) / _s); var maxy = max((_my - _y) / _s, (drag_point_my - _y) / _s); minx = value_snap(minx, _snx); maxx = value_snap(maxx, _snx); miny = value_snap(miny, _sny); maxy = value_snap(maxy, _sny); if(key_mod_press(SHIFT)) { var n = max(maxx - minx, maxy - miny); maxx = minx + n; maxy = miny + n; } var a = []; for( var i = 0; i < 4; i++ ) a[i] = inputs[| input_fix_len + i].getValue(); a[0][0] = minx; a[0][1] = miny; a[1][0] = maxx; a[1][1] = miny; a[2][0] = maxx; a[2][1] = maxy; a[3][0] = minx; a[3][1] = maxy; for( var i = 0; i < 4; i++ ) { if(inputs[| input_fix_len + i].setValue(a[i])) _edited = true; } } else if(drag_type == 4) { var minx = min((_mx - _x) / _s, (drag_point_mx - _x) / _s); var maxx = max((_mx - _x) / _s, (drag_point_mx - _x) / _s); var miny = min((_my - _y) / _s, (drag_point_my - _y) / _s); var maxy = max((_my - _y) / _s, (drag_point_my - _y) / _s); minx = value_snap(minx, _snx); maxx = value_snap(maxx, _snx); miny = value_snap(miny, _sny); maxy = value_snap(maxy, _sny); if(key_mod_press(SHIFT)) { var n = max(maxx - minx, maxy - miny); maxx = minx + n; maxy = miny + n; } var a = []; for( var i = 0; i < 4; i++ ) a[i] = inputs[| input_fix_len + i].getValue(); a[0][0] = (minx + maxx) / 2; a[0][1] = miny; a[0][2] = -(maxx - minx) * 0.27614; a[0][3] = 0; a[0][4] = (maxx - minx) * 0.27614; a[0][5] = 0; a[1][0] = maxx; a[1][1] = (miny + maxy) / 2; a[1][2] = 0; a[1][3] = -(maxy - miny) * 0.27614; a[1][4] = 0; a[1][5] = (maxy - miny) * 0.27614; a[2][0] = (minx + maxx) / 2; a[2][1] = maxy; a[2][2] = (maxx - minx) * 0.27614; a[2][3] = 0; a[2][4] = -(maxx - minx) * 0.27614; a[2][5] = 0; a[3][0] = minx; a[3][1] = (miny + maxy) / 2; a[3][2] = 0; a[3][3] = (maxy - miny) * 0.27614; a[3][4] = 0; a[3][5] = -(maxy - miny) * 0.27614; for( var i = 0; i < 4; i++ ) { if(inputs[| input_fix_len + i].setValue(a[i])) _edited = true; } } if(_edited) UNDO_HOLDING = true; if(mouse_release(mb_left)) { drag_point = -1; UPDATE |= RENDER_TYPE.full; UNDO_HOLDING = false; } } #region draw control anchor var line_hover = -1; var points = []; var _a0, _a1; var minx = 99999, miny = 99999; var maxx = 0 , maxy = 0; for(var i = loop? 0 : 1; i < ansize; i++) { if(i) { _a0 = inputs[| input_fix_len + i - 1].getValue(); _a1 = inputs[| input_fix_len + i].getValue(); } else { _a0 = inputs[| input_fix_len + ansize - 1].getValue(); _a1 = inputs[| input_fix_len + 0].getValue(); } var _ox = 0, _oy = 0, _nx = 0, _ny = 0, p = 0, pnt = []; for(var j = 0; j < sample; j++) { if(array_length(_a0) < 6) continue; p = eval_bezier(j / (sample - 1), _a0[0], _a0[1], _a1[0], _a1[1], _a0[0] + _a0[4], _a0[1] + _a0[5], _a1[0] + _a1[2], _a1[1] + _a1[3]); _nx = _x + p[0] * _s; _ny = _y + p[1] * _s; minx = min(minx, _nx); miny = min(miny, _ny); maxx = max(maxx, _nx); maxy = max(maxy, _ny); array_push(pnt, [ _nx, _ny ]); if(j && (key_mod_press(CTRL) || isUsingTool(1)) && distance_to_line(_mx, _my, _ox, _oy, _nx, _ny) < 4) line_hover = i; _ox = _nx; _oy = _ny; } array_push(points, pnt); } #endregion draw_set_color(isUsingTool(0)? c_white : COLORS._main_accent); var ind = 0; for(var i = loop? 0 : 1; i < ansize; i++) { for(var j = 0; j < sample; j++) { _nx = points[ind][j][0]; _ny = points[ind][j][1]; if(j) draw_line_width(_ox, _oy, _nx, _ny, 1 + 2 * (line_hover == i)); _ox = _nx; _oy = _ny; } ind++; } var anchor_hover = -1; var hover_type = 0; if(!isUsingTool(0)) for(var i = 0; i < ansize; i++) { var _a = inputs[| input_fix_len + i].getValue(); var xx = _x + _a[0] * _s; var yy = _y + _a[1] * _s; var cont = false; var _ax0 = 0, _ay0 = 0; var _ax1 = 0, _ay1 = 0; if(array_length(_a) < 6) continue; if(_a[2] != 0 || _a[3] != 0 || _a[4] != 0 || _a[5] != 0) { _ax0 = _x + (_a[0] + _a[2]) * _s; _ay0 = _y + (_a[1] + _a[3]) * _s; _ax1 = _x + (_a[0] + _a[4]) * _s; _ay1 = _y + (_a[1] + _a[5]) * _s; cont = true; draw_set_color(COLORS.node_path_overlay_control_line); draw_line(_ax0, _ay0, xx, yy); draw_line(_ax1, _ay1, xx, yy); draw_sprite_colored(THEME.anchor_selector, 2, _ax0, _ay0); draw_sprite_colored(THEME.anchor_selector, 2, _ax1, _ay1); } draw_sprite_colored(THEME.anchor_selector, 0, xx, yy); if(drag_point == i) { draw_sprite_colored(THEME.anchor_selector, 1, xx, yy); } else if(point_in_circle(_mx, _my, xx, yy, 8)) { draw_sprite_colored(THEME.anchor_selector, 1, xx, yy); anchor_hover = i; hover_type = 0; } else if(cont && point_in_circle(_mx, _my, _ax0, _ay0, 8)) { draw_sprite_colored(THEME.anchor_selector, 0, _ax0, _ay0); anchor_hover = i; hover_type = 1; } else if(cont && point_in_circle(_mx, _my, _ax1, _ay1, 8)) { draw_sprite_colored(THEME.anchor_selector, 0, _ax1, _ay1); anchor_hover = i; hover_type = -1; } } if(isUsingTool(0)) { //transform tools var hov = 0; if(point_in_circle(_mx, _my, minx, miny, 8)) hov = 1; else if(point_in_circle(_mx, _my, maxx, miny, 8)) hov = 2; else if(point_in_circle(_mx, _my, minx, maxy, 8)) hov = 3; else if(point_in_circle(_mx, _my, maxx, maxy, 8)) hov = 4; else if(point_in_rectangle(_mx, _my, minx, miny, maxx, maxy)) hov = 5; draw_set_color(COLORS._main_accent); draw_rectangle_border(minx, miny, maxx, maxy, 1 + (hov == 5)); draw_sprite_colored(THEME.anchor_selector, hov == 1, minx, miny); draw_sprite_colored(THEME.anchor_selector, hov == 2, maxx, miny); draw_sprite_colored(THEME.anchor_selector, hov == 3, minx, maxy); draw_sprite_colored(THEME.anchor_selector, hov == 4, maxx, maxy); if(hov && mouse_press(mb_left, active)) { transform_type = hov; transform_minx = (minx - _x) / _s; transform_maxx = (maxx - _x) / _s; transform_miny = (miny - _y) / _s; transform_maxy = (maxy - _y) / _s; transform_mx = _mx; transform_my = _my; } } else if(isUsingTool(3)) { //pen tools draw_sprite_ui_uniform(THEME.path_tools_draw, 0, _mx + 16, _my + 16); if(mouse_press(mb_left, active)) { var replace = tool_pathDrawer.attribute.create; if(replace) { while(ds_list_size(inputs) > input_fix_len) ds_list_delete(inputs, input_fix_len); array_resize(input_display_list, input_display_list_len); } drag_point = 0; drag_type = 2; drag_points = [ [ (_mx - _x) / _s, (_my - _y) / _s ] ]; drag_point_mx = (_mx - _x) / _s; drag_point_my = (_my - _y) / _s; } } else if(isUsingTool(4) || isUsingTool(5)) { //shape tools draw_sprite_ui_uniform(THEME.cursor_path_add, 0, _mx + 16, _my + 16); if(mouse_press(mb_left, active)) { while(ds_list_size(inputs) > input_fix_len) ds_list_delete(inputs, input_fix_len); array_resize(input_display_list, input_display_list_len); drag_point = 0; drag_type = isUsingTool(4)? 3 : 4; drag_point_mx = _mx; drag_point_my = _my; inputs[| 1].setValue(true); repeat(4) createAnchor(value_snap((_mx - _x) / _s, _snx), value_snap((_my - _y) / _s, _sny)); } } else if(anchor_hover != -1) { //no tool, dragging existing point var _a = inputs[| input_fix_len + anchor_hover].getValue(); if(isUsingTool(2)) { draw_sprite_ui_uniform(THEME.cursor_path_anchor, 0, _mx + 16, _my + 16); if(mouse_press(mb_left, active)) { if(_a[2] != 0 || _a[3] != 0 || _a[4] != 0 || _a[5] != 0) { _a[2] = 0; _a[3] = 0; _a[4] = 0; _a[5] = 0; inputs[| input_fix_len + anchor_hover].setValue(_a); } else { _a[2] = -8; _a[3] = 0; _a[4] = 8; _a[5] = 0; drag_point = anchor_hover; drag_type = 1; drag_point_mx = _mx; drag_point_my = _my; drag_point_sx = _a[0]; drag_point_sy = _a[1]; } } } else if(hover_type == 0 && key_mod_press(SHIFT)) { //remove draw_sprite_ui_uniform(THEME.cursor_path_remove, 0, _mx + 16, _my + 16); if(mouse_press(mb_left, active)) { recordAction(ACTION_TYPE.var_modify, self, [ array_clone(input_display_list), "input_display_list" ]); recordAction(ACTION_TYPE.list_delete, inputs, [ inputs[| input_fix_len + anchor_hover], input_fix_len + anchor_hover, "remove path anchor point" ]); ds_list_delete(inputs, input_fix_len + anchor_hover); array_remove(input_display_list, input_fix_len + anchor_hover); doUpdate(); } } else { draw_sprite_ui_uniform(THEME.cursor_path_move, 0, _mx + 16, _my + 16); if(mouse_press(mb_left, active)) { drag_point = anchor_hover; drag_type = hover_type; drag_point_mx = _mx; drag_point_my = _my; drag_point_sx = _a[0]; drag_point_sy = _a[1]; if(hover_type == 1) { drag_point_sx = _a[0] + _a[2]; drag_point_sy = _a[1] + _a[3]; } else if(hover_type == -1) { drag_point_sx = _a[0] + _a[4]; drag_point_sy = _a[1] + _a[5]; } } } } else if(key_mod_press(CTRL) || isUsingTool(1)) { //anchor edit draw_sprite_ui_uniform(THEME.cursor_path_add, 0, _mx + 16, _my + 16); if(mouse_press(mb_left, active)) { var anc = createAnchor(value_snap((_mx - _x) / _s, _snx), value_snap((_my - _y) / _s, _sny)); UNDO_HOLDING = true; if(line_hover == -1) { drag_point = ds_list_size(inputs) - input_fix_len - 1; } else { ds_list_remove(inputs, anc); ds_list_insert(inputs, input_fix_len + line_hover, anc); drag_point = line_hover; } drag_type = -1; drag_point_mx = _mx; drag_point_my = _my; drag_point_sx = (_mx - _x) / _s; drag_point_sy = (_my - _y) / _s; UPDATE |= RENDER_TYPE.full; } } } static updateLength = function() { boundary = new BoundingBox(); lengthTotal = 0; var loop = inputs[| 1].getValue(); var rond = inputs[| 3].getValue(); var ansize = ds_list_size(inputs) - input_fix_len; if(ansize < 2) { lengths = []; anchors = []; return; } var sample = PREF_MAP[? "path_resolution"]; var con = loop? ansize : ansize - 1; lengths = []; lengthAccs = []; array_resize(anchors, ansize); for(var i = 0; i < con; i++) { var index_0 = input_fix_len + i; var index_1 = input_fix_len + i + 1; if(index_1 >= ds_list_size(inputs)) index_1 = input_fix_len; var _a0 = inputs[| index_0].getValue(); var _a1 = inputs[| index_1].getValue(); anchors[i + 0] = _a0; anchors[i + 1] = _a1; if(rond) { _a0[0] = round(_a0[0]); _a0[1] = round(_a0[1]); _a1[0] = round(_a1[0]); _a1[1] = round(_a1[1]); } var l = 0, _ox = 0, _oy = 0, _nx = 0, _ny = 0, p = 0; for(var j = 0; j < sample; j++) { p = eval_bezier(j / sample, _a0[0], _a0[1], _a1[0], _a1[1], _a0[0] + _a0[4], _a0[1] + _a0[5], _a1[0] + _a1[2], _a1[1] + _a1[3]); _nx = p[0]; _ny = p[1]; boundary.addPoint(_nx, _ny); if(j) l += point_distance(_nx, _ny, _ox, _oy); _ox = _nx; _oy = _ny; } lengths[i] = l; lengthTotal += l; lengthAccs[i] = lengthTotal; } } static getLineCount = function() { return 1; } static getSegmentCount = function() { return array_length(lengths); } static getBoundary = function() { return boundary; } static getLength = function() { return lengthTotal; } static getAccuLength = function() { return lengthAccs; } static getPointDistance = function(_dist) { var loop = inputs[| 1].getValue(); var rond = inputs[| 3].getValue(); var ansize = array_length(lengths); var amo = ds_list_size(inputs) - input_fix_len; if(ansize == 0) return new Point(); var _a0, _a1; for(var i = 0; i < ansize; i++) { _a0 = anchors[safe_mod(i + 0, amo)]; _a1 = anchors[safe_mod(i + 1, amo)]; if(rond) { _a0[0] = round(_a0[0]); _a0[1] = round(_a0[1]); _a1[0] = round(_a1[0]); _a1[1] = round(_a1[1]); } if(_dist > lengths[i]) { _dist -= lengths[i]; continue; } var _t = _dist / lengths[i]; if(!is_array(_a0) || !is_array(_a1)) return new Point(); var _p = eval_bezier(_t, _a0[0], _a0[1], _a1[0], _a1[1], _a0[0] + _a0[4], _a0[1] + _a0[5], _a1[0] + _a1[2], _a1[1] + _a1[3]); return new Point(_p); } return new Point(); } static getPointRatio = function(_rat) { var pix = frac(_rat) * lengthTotal; return getPointDistance(pix); } static getPointSegment = function(_rat) { var loop = inputs[| 1].getValue(); var rond = inputs[| 3].getValue(); var ansize = array_length(lengths); var amo = ds_list_size(inputs) - input_fix_len; if(amo < 1) return new Point(0, 0); if(_rat < 0) { var _p0 = inputs[| input_fix_len].getValue(); if(rond) return new Point(round(_p0[0]), round(_p0[1])); return new Point(_p0[0], _p0[1]); } _rat = safe_mod(_rat, ansize); var _i0 = clamp(floor(_rat), 0, amo - 1); var _t = frac(_rat); var _i1 = _i0 + 1; if(_i1 >= amo) { if(!loop) { var _p1 = inputs[| ds_list_size(inputs) - 1].getValue() if(rond) return new Point(round(_p1[0]), round(_p1[1])); return new Point(_p1[0], _p1[1]); } _i1 = 0; } var _a0 = inputs[| input_fix_len + _i0].getValue(); var _a1 = inputs[| input_fix_len + _i1].getValue(); if(rond) { _a0[0] = round(_a0[0]); _a0[1] = round(_a0[1]); _a1[0] = round(_a1[0]); _a1[1] = round(_a1[1]); } var p = eval_bezier(_t, _a0[0], _a0[1], _a1[0], _a1[1], _a0[0] + _a0[4], _a0[1] + _a0[5], _a1[0] + _a1[2], _a1[1] + _a1[3]); return new Point(p[0], p[1]); } static update = function(frame = ANIMATOR.current_frame) { updateLength(); var _rat = inputs[| 0].getValue(); var _typ = inputs[| 2].getValue(); var anchors = []; for(var i = input_fix_len; i < ds_list_size(inputs); i++) array_push(anchors, inputs[| i].getValue()); outputs[| 2].setValue(anchors); if(is_array(_rat)) { var _out = array_create(array_length(_rat)); for( var i = 0; i < array_length(_rat); i++ ) { if(_typ == 0) _out[i] = getPointRatio(_rat[i]); else if(_typ == 1) _out[i] = getPointSegment(_rat[i]); } outputs[| 0].setValue(_out); } else { var _out = [0, 0]; if(_typ == 0) _out = getPointRatio(_rat); else if(_typ == 1) _out = getPointSegment(_rat); outputs[| 0].setValue(_out.toArray()); } } static onDrawNode = function(xx, yy, _mx, _my, _s, _hover, _focus) { var bbox = drawGetBbox(xx, yy, _s); draw_sprite_fit(THEME.node_draw_path, 0, bbox.xc, bbox.yc, bbox.w, bbox.h); } static postDeserialize = function() { var _inputs = load_map[? "inputs"]; if(LOADING_VERSION < 1380 && !CLONING) ds_list_insert(_inputs, 3, noone); for(var i = input_fix_len; i < ds_list_size(_inputs); i++) createAnchor(0, 0); } }