2024-01-23 11:01:19 +01:00
|
|
|
function Node_Path_Smooth(_x, _y, _group = noone) : Node(_x, _y, _group) constructor {
|
|
|
|
name = "Smooth Path";
|
2025-01-01 02:12:36 +01:00
|
|
|
setDimension(96, 48);
|
2024-01-23 11:01:19 +01:00
|
|
|
|
2024-08-18 09:13:41 +02:00
|
|
|
newInput(0, nodeValue_Bool("Loop", self, false))
|
2024-01-23 11:01:19 +01:00
|
|
|
.rejectArray();
|
|
|
|
|
2024-08-18 09:13:41 +02:00
|
|
|
newInput(1, nodeValue_Bool("Round anchor", self, false))
|
2024-01-23 11:01:19 +01:00
|
|
|
.rejectArray();
|
|
|
|
|
2024-08-18 09:13:41 +02:00
|
|
|
newInput(2, nodeValue_Float("Smoothness", self, 3))
|
2024-01-23 11:01:19 +01:00
|
|
|
.setDisplay(VALUE_DISPLAY.slider, { range : [ 1, 5, 0.01 ] } );
|
|
|
|
|
2024-09-04 03:57:11 +02:00
|
|
|
newOutput(0, nodeValue_Output("Path data", self, VALUE_TYPE.pathnode, self));
|
2024-01-23 11:01:19 +01:00
|
|
|
|
|
|
|
input_display_list = [
|
|
|
|
["Path", false], 0, 1, 2,
|
|
|
|
["Anchors", false],
|
|
|
|
];
|
|
|
|
|
2024-05-23 10:59:39 +02:00
|
|
|
setDynamicInput(1, false);
|
2024-01-23 11:01:19 +01:00
|
|
|
|
|
|
|
tools = [
|
|
|
|
new NodeTool( "Anchor add / remove", THEME.path_tools_add ),
|
|
|
|
];
|
|
|
|
|
|
|
|
#region ---- path ----
|
|
|
|
anchors = [];
|
|
|
|
controls = [];
|
|
|
|
segments = [];
|
|
|
|
lengths = [];
|
|
|
|
lengthAccs = [];
|
|
|
|
lengthTotal = 0;
|
|
|
|
boundary = new BoundingBox();
|
|
|
|
|
2024-09-06 09:42:03 +02:00
|
|
|
cached_pos = ds_map_create();
|
|
|
|
path_preview_surface = noone;
|
2024-01-23 11:01:19 +01:00
|
|
|
#endregion
|
|
|
|
|
|
|
|
#region ---- editor ----
|
2024-06-16 05:27:57 +02:00
|
|
|
line_hover = noone;
|
2024-01-23 11:01:19 +01:00
|
|
|
#endregion
|
|
|
|
|
2024-09-05 09:25:06 +02:00
|
|
|
static resetDisplayList = function() {
|
2024-01-23 11:01:19 +01:00
|
|
|
recordAction(ACTION_TYPE.var_modify, self, [ array_clone(input_display_list), "input_display_list" ]);
|
|
|
|
|
|
|
|
input_display_list = [
|
|
|
|
["Path", false], 0, 1, 2,
|
|
|
|
["Anchors", false],
|
|
|
|
];
|
|
|
|
|
2024-08-08 06:57:51 +02:00
|
|
|
for( var i = input_fix_len, n = array_length(inputs); i < n; i++ ) {
|
2024-01-23 11:01:19 +01:00
|
|
|
array_push(input_display_list, i);
|
2024-08-08 06:57:51 +02:00
|
|
|
inputs[i].name = $"Anchor {i - input_fix_len}";
|
2024-01-23 11:01:19 +01:00
|
|
|
}
|
2024-09-05 09:25:06 +02:00
|
|
|
}
|
2024-01-23 11:01:19 +01:00
|
|
|
|
2024-09-05 09:25:06 +02:00
|
|
|
static createNewInput = function(_x = 0, _y = 0) {
|
2024-08-08 06:57:51 +02:00
|
|
|
var index = array_length(inputs);
|
2024-01-23 11:01:19 +01:00
|
|
|
|
2024-08-18 06:16:20 +02:00
|
|
|
newInput(index, nodeValue_Vec2("Anchor", self, [ _x, _y ]));
|
2024-01-23 11:01:19 +01:00
|
|
|
|
2024-09-03 11:49:51 +02:00
|
|
|
recordAction(ACTION_TYPE.array_insert, inputs, [ inputs[index], index, "add path anchor point" ]);
|
2024-01-23 11:01:19 +01:00
|
|
|
resetDisplayList();
|
|
|
|
|
2024-08-08 06:57:51 +02:00
|
|
|
return inputs[index];
|
2024-09-05 09:25:06 +02:00
|
|
|
}
|
2024-01-23 11:01:19 +01:00
|
|
|
|
2024-09-05 09:25:06 +02:00
|
|
|
static drawOverlay = function(hover, active, _x, _y, _s, _mx, _my, _snx, _sny) {
|
2024-01-23 11:01:19 +01:00
|
|
|
var sample = PREFERENCES.path_resolution;
|
2024-08-08 06:57:51 +02:00
|
|
|
var ansize = array_length(inputs) - input_fix_len;
|
2024-01-23 11:01:19 +01:00
|
|
|
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);
|
|
|
|
|
2024-09-05 09:25:06 +02:00
|
|
|
for( var i = 0, n = array_length(segments); i < n; i++ ) { // draw path
|
2024-01-23 11:01:19 +01:00
|
|
|
var _seg = segments[i];
|
|
|
|
var _ox = 0, _oy = 0, _nx = 0, _ny = 0, p = 0;
|
|
|
|
|
2024-05-28 04:57:00 +02:00
|
|
|
for( var j = 0, m = array_length(_seg); j < m; j += 2 ) {
|
|
|
|
_nx = _x + _seg[j + 0] * _s;
|
|
|
|
_ny = _y + _seg[j + 1] * _s;
|
2024-01-23 11:01:19 +01:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
2024-09-05 09:25:06 +02:00
|
|
|
}
|
2024-01-23 11:01:19 +01:00
|
|
|
|
2024-09-05 09:25:06 +02:00
|
|
|
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;
|
|
|
|
}
|
2024-01-23 11:01:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
line_hover = _line_hover;
|
|
|
|
|
2024-09-05 09:25:06 +02:00
|
|
|
if(key_mod_press(CTRL) || isUsingTool(0)) { // anchor edit
|
2024-01-23 11:01:19 +01:00
|
|
|
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) {
|
2024-08-08 06:57:51 +02:00
|
|
|
array_remove(inputs, anc);
|
|
|
|
array_insert(inputs, input_fix_len + _line_hover + 1, anc);
|
2024-01-23 11:01:19 +01:00
|
|
|
}
|
|
|
|
} else {
|
2024-09-05 09:25:06 +02:00
|
|
|
// 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);
|
2024-01-23 11:01:19 +01:00
|
|
|
resetDisplayList();
|
|
|
|
}
|
|
|
|
|
|
|
|
RENDER_ALL
|
|
|
|
}
|
|
|
|
}
|
2024-09-05 09:25:06 +02:00
|
|
|
}
|
2024-01-23 11:01:19 +01:00
|
|
|
|
2024-09-05 09:25:06 +02:00
|
|
|
static updateLength = function() {
|
2024-01-23 11:01:19 +01:00
|
|
|
var loop = getInputData(0);
|
|
|
|
|
|
|
|
segments = [];
|
|
|
|
lengths = [];
|
|
|
|
lengthAccs = [];
|
|
|
|
lengthTotal = 0;
|
|
|
|
boundary = new BoundingBox();
|
|
|
|
|
|
|
|
var sample = PREFERENCES.path_resolution;
|
2024-08-08 06:57:51 +02:00
|
|
|
var ansize = array_length(inputs) - input_fix_len;
|
2024-01-23 11:01:19 +01:00
|
|
|
if(ansize < 2) return;
|
|
|
|
|
|
|
|
var con = loop? ansize : ansize - 1;
|
|
|
|
|
|
|
|
for(var i = 0; i < con; i++) {
|
2024-01-23 14:40:03 +01:00
|
|
|
var _a0 = anchors[ (i + 0) % ansize];
|
|
|
|
var _a1 = anchors[ (i + 1) % ansize];
|
2024-01-23 11:01:19 +01:00
|
|
|
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;
|
2024-05-28 04:57:00 +02:00
|
|
|
var sg = array_create(sample * 2);
|
2024-01-23 11:01:19 +01:00
|
|
|
|
|
|
|
for(var j = 0; j < sample; j++) {
|
2024-05-28 04:57:00 +02:00
|
|
|
|
|
|
|
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;
|
2024-01-23 11:01:19 +01:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
2024-09-06 09:42:03 +02:00
|
|
|
|
|
|
|
// 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);
|
2024-09-05 09:25:06 +02:00
|
|
|
}
|
2024-01-23 11:01:19 +01:00
|
|
|
|
|
|
|
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; }
|
|
|
|
|
2024-09-05 09:25:06 +02:00
|
|
|
static getPointDistance = function(_dist, _ind = 0, out = undefined) {
|
2024-01-23 11:01:19 +01:00
|
|
|
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);
|
|
|
|
|
2024-08-08 06:57:51 +02:00
|
|
|
var ansize = array_length(inputs) - input_fix_len;
|
2024-01-23 11:01:19 +01:00
|
|
|
if(ansize == 0) return out;
|
|
|
|
|
|
|
|
for(var i = 0; i < ansize; i++) {
|
2024-01-23 14:40:03 +01:00
|
|
|
var _a0 = anchors[ (i + 0) % ansize];
|
|
|
|
var _a1 = anchors[ (i + 1) % ansize];
|
|
|
|
var _c0 = controls[(i + 0) % ansize];
|
|
|
|
var _c1 = controls[(i + 1) % ansize];
|
2024-01-23 11:01:19 +01:00
|
|
|
|
|
|
|
if(_dist > lengths[i]) {
|
|
|
|
_dist -= lengths[i];
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
var _t = _dist / lengths[i];
|
2024-05-28 04:57:00 +02:00
|
|
|
|
|
|
|
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]);
|
|
|
|
}
|
2024-01-23 11:01:19 +01:00
|
|
|
|
|
|
|
cached_pos[? _cKey] = out.clone();
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
return out;
|
2024-09-05 09:25:06 +02:00
|
|
|
}
|
2024-01-23 11:01:19 +01:00
|
|
|
|
2024-09-05 09:25:06 +02:00
|
|
|
static getPointRatio = function(_rat, _ind = 0, out = undefined) {
|
2024-01-23 11:01:19 +01:00
|
|
|
var pix = frac(_rat) * lengthTotal;
|
|
|
|
return getPointDistance(pix, _ind, out);
|
2024-09-05 09:25:06 +02:00
|
|
|
}
|
2024-01-23 11:01:19 +01:00
|
|
|
|
2024-09-05 09:25:06 +02:00
|
|
|
static update = function(frame = CURRENT_FRAME) {
|
2024-01-23 11:01:19 +01:00
|
|
|
ds_map_clear(cached_pos);
|
|
|
|
|
|
|
|
var loop = getInputData(0);
|
|
|
|
var rond = getInputData(1);
|
|
|
|
var smot = getInputData(2);
|
|
|
|
|
|
|
|
var _a = [];
|
2024-08-08 06:57:51 +02:00
|
|
|
for(var i = input_fix_len; i < array_length(inputs); i++) {
|
2024-01-23 11:01:19 +01:00
|
|
|
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++ ) {
|
2024-06-16 05:27:57 +02:00
|
|
|
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 ]);
|
2024-05-23 10:59:39 +02:00
|
|
|
|
2024-01-23 11:01:19 +01:00
|
|
|
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();
|
2024-09-05 09:25:06 +02:00
|
|
|
}
|
2024-01-23 11:01:19 +01:00
|
|
|
|
2024-09-05 09:25:06 +02:00
|
|
|
static onDrawNode = function(xx, yy, _mx, _my, _s, _hover, _focus) {
|
2024-01-23 11:01:19 +01:00
|
|
|
var bbox = drawGetBbox(xx, yy, _s);
|
2024-09-06 09:42:03 +02:00
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
2024-09-05 09:25:06 +02:00
|
|
|
}
|
2024-09-06 09:42:03 +02:00
|
|
|
|
2024-01-23 11:01:19 +01:00
|
|
|
}
|