function Node_Path_Smooth(_x, _y, _group = noone) : Node(_x, _y, _group) constructor { name = "Smooth Path"; setDimension(96, 48); newInput(0, nodeValue_Bool("Loop", self, false)) .rejectArray(); newInput(1, nodeValue_Bool("Round anchor", self, false)) .rejectArray(); newInput(2, nodeValue_Float("Smoothness", self, 3)) .setDisplay(VALUE_DISPLAY.slider, { range : [ 1, 5, 0.01 ] } ); newOutput(0, nodeValue_Output("Path data", self, VALUE_TYPE.pathnode, self)); input_display_list = [ ["Path", false], 0, 1, 2, ["Anchors", false], ]; setDynamicInput(1, false); tools = [ new NodeTool( "Anchor add / remove", THEME.path_tools_add ), ]; #region ---- path ---- anchors = []; controls = []; segments = []; lengths = []; lengthAccs = []; lengthTotal = 0; boundary = new BoundingBox(); cached_pos = ds_map_create(); path_preview_surface = noone; #endregion #region ---- editor ---- line_hover = noone; #endregion static resetDisplayList = function() { recordAction(ACTION_TYPE.var_modify, self, [ array_clone(input_display_list), "input_display_list" ]); input_display_list = [ ["Path", false], 0, 1, 2, ["Anchors", false], ]; for( var i = input_fix_len, n = array_length(inputs); i < n; i++ ) { array_push(input_display_list, i); inputs[i].name = $"Anchor {i - input_fix_len}"; } } static createNewInput = function(_x = 0, _y = 0) { var index = array_length(inputs); newInput(index, nodeValue_Vec2("Anchor", self, [ _x, _y ])); recordAction(ACTION_TYPE.array_insert, inputs, [ inputs[index], index, "add path anchor point" ]); resetDisplayList(); return inputs[index]; } static drawOverlay = function(hover, active, _x, _y, _s, _mx, _my, _snx, _sny) { var sample = PREFERENCES.path_resolution; var ansize = array_length(inputs) - input_fix_len; var loop = getInputData(0); var rond = getInputData(1); var _line_hover = -1; var _anchor_hover = -1; if(!array_empty(anchors)) { draw_set_color(COLORS._main_accent); for( var i = 0, n = array_length(segments); i < n; i++ ) { // draw path var _seg = segments[i]; var _ox = 0, _oy = 0, _nx = 0, _ny = 0, p = 0; for( var j = 0, m = array_length(_seg); j < m; j += 2 ) { _nx = _x + _seg[j + 0] * _s; _ny = _y + _seg[j + 1] * _s; if(j) { if((key_mod_press(CTRL) || isUsingTool(0)) && distance_to_line(_mx, _my, _ox, _oy, _nx, _ny) < 4) _line_hover = i; draw_line_width(_ox, _oy, _nx, _ny, 1 + 2 * (line_hover == i)); } _ox = _nx; _oy = _ny; } } var _act = active && !isUsingTool(0); for(var i = input_fix_len; i < array_length(inputs); i++) { var a = inputs[i].drawOverlay(hover, _act, _x, _y, _s, _mx, _my, _snx, _sny); _act &= !a; if(a) _anchor_hover = i; } } line_hover = _line_hover; if(key_mod_press(CTRL) || isUsingTool(0)) { // anchor edit draw_sprite_ui_uniform(_anchor_hover == -1? THEME.cursor_path_add : THEME.cursor_path_remove, 0, _mx + 16, _my + 16); if(mouse_press(mb_left, active)) { if(_anchor_hover == -1) { var anc = createNewInput(value_snap((_mx - _x) / _s, _snx), value_snap((_my - _y) / _s, _sny)); UNDO_HOLDING = true; if(_line_hover != -1) { array_remove(inputs, anc); array_insert(inputs, input_fix_len + _line_hover + 1, anc); } } else { // print($"{array_length(inputs)}: {_anchor_hover}"); recordAction(ACTION_TYPE.array_delete, inputs, [ inputs[_anchor_hover], _anchor_hover, "remove path anchor point" ]); array_delete(inputs, _anchor_hover, 1); resetDisplayList(); } RENDER_ALL } } } static updateLength = function() { var loop = getInputData(0); segments = []; lengths = []; lengthAccs = []; lengthTotal = 0; boundary = new BoundingBox(); var sample = PREFERENCES.path_resolution; var ansize = array_length(inputs) - input_fix_len; if(ansize < 2) return; var con = loop? ansize : ansize - 1; for(var i = 0; i < con; i++) { var _a0 = anchors[ (i + 0) % ansize]; var _a1 = anchors[ (i + 1) % ansize]; var _c0 = controls[(i + 0) % ansize]; var _c1 = controls[(i + 1) % ansize]; var l = 0, _ox = 0, _oy = 0, _nx = 0, _ny = 0, p = 0; var sg = array_create(sample * 2); for(var j = 0; j < sample; j++) { if(_c0[2] == 0 && _c0[3] == 0 && _c1[0] == 0 && _c1[1] == 0) { _nx = lerp(_a0[0], _a1[0], j / sample); _ny = lerp(_a0[1], _a1[1], j / sample); } else { _nx = eval_bezier_x(j / sample, _a0[0], _a0[1], _a1[0], _a1[1], _a0[0] + _c0[2], _a0[1] + _c0[3], _a1[0] + _c1[0], _a1[1] + _c1[1]); _ny = eval_bezier_y(j / sample, _a0[0], _a0[1], _a1[0], _a1[1], _a0[0] + _c0[2], _a0[1] + _c0[3], _a1[0] + _c1[0], _a1[1] + _c1[1]); } sg[j * 2 + 0] = _nx; sg[j * 2 + 1] = _ny; boundary.addPoint(_nx, _ny); if(j) l += point_distance(_nx, _ny, _ox, _oy); _ox = _nx; _oy = _ny; } segments[i] = sg; lengths[i] = l; lengthTotal += l; lengthAccs[i] = lengthTotal; } // Surface generate var pad = min(8, abs(boundary.maxx - boundary.minx) * 0.1, abs(boundary.maxy - boundary.miny) * 0.1); var minx = boundary.minx - pad, miny = boundary.miny - pad; var maxx = boundary.maxx + pad, maxy = boundary.maxy + pad; var rngx = maxx - minx, rngy = maxy - miny; var prev_s = 128; var _surf = surface_create(prev_s, prev_s); _surf = surface_verify(_surf, prev_s, prev_s); surface_set_target(_surf); DRAW_CLEAR var ox, oy, nx, ny; draw_set_color(c_white); for (var i = 0, n = array_length(segments); i < n; i++) { var segment = segments[i]; for (var j = 0, m = array_length(segment); j < m; j += 2) { nx = (segment[j + 0] - minx) / rngx * prev_s; ny = (segment[j + 1] - miny) / rngy * prev_s; if(j) draw_line_round(ox, oy, nx, ny, 4); ox = nx; oy = ny; } } draw_set_color(COLORS._main_accent); for (var i = 0, n = array_length(anchors); i < n; i++) { var _a0 = anchors[i]; draw_circle((_a0[0] - minx) / rngx * prev_s, (_a0[1] - miny) / rngy * prev_s, 8, false); } surface_reset_target(); path_preview_surface = surface_verify(path_preview_surface, prev_s, prev_s); surface_set_shader(path_preview_surface, sh_FXAA); shader_set_f("dimension", prev_s, prev_s); shader_set_f("cornerDis", 0.5); shader_set_f("mixAmo", 1); draw_surface_safe(_surf); surface_reset_shader(); surface_free(_surf); } 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, _ind = 0, out = undefined) { if(out == undefined) out = new __vec2(); else { out.x = 0; out.y = 0; } if(array_empty(lengths)) return out; var _cKey = _dist; if(ds_map_exists(cached_pos, _cKey)) { var _p = cached_pos[? _cKey]; out.x = _p.x; out.y = _p.y; return out; } var loop = getInputData(1); if(loop) _dist = safe_mod(_dist, lengthTotal, MOD_NEG.wrap); var ansize = array_length(inputs) - input_fix_len; if(ansize == 0) return out; for(var i = 0; i < ansize; i++) { var _a0 = anchors[ (i + 0) % ansize]; var _a1 = anchors[ (i + 1) % ansize]; var _c0 = controls[(i + 0) % ansize]; var _c1 = controls[(i + 1) % ansize]; if(_dist > lengths[i]) { _dist -= lengths[i]; continue; } var _t = _dist / lengths[i]; if(_c0[2] == 0 && _c0[3] == 0 && _c1[0] == 0 && _c1[1] == 0) { out.x = lerp(_a0[0], _a1[0], _t); out.y = lerp(_a0[1], _a1[1], _t); } else { out.x = eval_bezier_x(_t, _a0[0], _a0[1], _a1[0], _a1[1], _a0[0] + _c0[2], _a0[1] + _c0[3], _a1[0] + _c1[0], _a1[1] + _c1[1]); out.y = eval_bezier_y(_t, _a0[0], _a0[1], _a1[0], _a1[1], _a0[0] + _c0[2], _a0[1] + _c0[3], _a1[0] + _c1[0], _a1[1] + _c1[1]); } cached_pos[? _cKey] = out.clone(); return out; } return out; } static getPointRatio = function(_rat, _ind = 0, out = undefined) { var pix = frac(_rat) * lengthTotal; return getPointDistance(pix, _ind, out); } static update = function(frame = CURRENT_FRAME) { ds_map_clear(cached_pos); var loop = getInputData(0); var rond = getInputData(1); var smot = getInputData(2); var _a = []; for(var i = input_fix_len; i < array_length(inputs); i++) { var _anc = array_clone(getInputData(i)); if(rond) { _anc[0] = round(_anc[0]); _anc[1] = round(_anc[1]); } array_push(_a, _anc); } var amo = array_length(_a); anchors = _a; controls = array_create(amo); if(amo == 2) { controls = [ [ 0, 0, 0, 0 ], [ 0, 0, 0, 0 ], ]; } else { for( var i = 0, n = amo; i < n; i++ ) { var _a0 = array_safe_get_fast(anchors, (i - 1 + amo) % n, [ 0, 0 ]); var _a1 = array_safe_get_fast(anchors, (i + amo) % n, [ 0, 0 ]); var _a2 = array_safe_get_fast(anchors, (i + 1 + amo) % n, [ 0, 0 ]); var _dr = point_direction(_a0[0], _a0[1], _a2[0], _a2[1]); var _ds0 = point_distance(_a1[0], _a1[1], _a0[0], _a0[1]) / smot; var _ds2 = point_distance(_a1[0], _a1[1], _a2[0], _a2[1]) / smot; controls[i] = [ -lengthdir_x(_ds0, _dr), -lengthdir_y(_ds0, _dr), lengthdir_x(_ds2, _dr), lengthdir_y(_ds2, _dr) ]; } if(!loop && amo) { controls[0] = [ 0, 0, 0, 0 ]; controls[amo - 1] = [ 0, 0, 0, 0 ]; } } updateLength(); } static onDrawNode = function(xx, yy, _mx, _my, _s, _hover, _focus) { var bbox = drawGetBbox(xx, yy, _s); if(array_empty(segments)) { draw_sprite_fit(s_node_path, 0, bbox.xc, bbox.yc, bbox.w, bbox.h); } else { gpu_set_tex_filter(true); draw_surface_bbox(path_preview_surface, bbox); gpu_set_tex_filter(false); } } }