mirror of
https://github.com/Ttanasart-pt/Pixel-Composer.git
synced 2025-01-13 15:56:35 +01:00
508 lines
No EOL
16 KiB
Text
508 lines
No EOL
16 KiB
Text
function Node_Path_Shape(_x, _y, _group = noone) : Node(_x, _y, _group) constructor {
|
|
name = "Path Shape";
|
|
setDimension(96, 48);
|
|
|
|
newInput(0, nodeValue_Vec2("Position", self, [ .5, .5 ]))
|
|
.setUnitRef(function() /*=>*/ {return DEF_SURF}, VALUE_UNIT.reference);
|
|
|
|
newInput(1, nodeValue_Vec2("Half Size", self, [ .5, .5 ]))
|
|
.setUnitRef(function() /*=>*/ {return DEF_SURF}, VALUE_UNIT.reference);
|
|
|
|
newInput(2, nodeValue_Rotation("Rotation", self, 0));
|
|
|
|
shapeScroll = [
|
|
new scrollItem("Rectangle", s_node_shape_type, 0),
|
|
new scrollItem("Trapezoid", s_node_shape_type, 2),
|
|
new scrollItem("Parallelogram", s_node_shape_type, 3),
|
|
-1,
|
|
new scrollItem("Ellipse", s_node_shape_type, 5),
|
|
new scrollItem("Arc", s_node_shape_type, 6),
|
|
new scrollItem("Squircle", s_node_shape_type, 11),
|
|
-1,
|
|
new scrollItem("Polygon", s_node_shape_type, 12),
|
|
new scrollItem("Star", s_node_shape_type, 13),
|
|
-1,
|
|
new scrollItem("Line", s_node_shape_type, 16),
|
|
new scrollItem("Curve", s_shape_curve, 0),
|
|
];
|
|
newInput(3, nodeValue_Enum_Scroll("Shape", self, 0, { data: shapeScroll, horizontal: true, text_pad: ui(16) }));
|
|
|
|
newInput(4, nodeValue_Float("Skew", self, .5))
|
|
.setDisplay(VALUE_DISPLAY.slider, { range: [ -1, 1, 0.01] });
|
|
|
|
newInput(5, nodeValue_Rotation_Range("Angle Range", self, [ 0, 90 ]));
|
|
|
|
newInput(6, nodeValue_Float("Factor", self, 4));
|
|
|
|
newInput(7, nodeValue_Int("Sides", self, 4));
|
|
|
|
newInput(8, nodeValue_Float("Inner Radius", self, .5));
|
|
|
|
newInput(9, nodeValue_Corner("Corner Radius", self, [ 0, 0, 0, 0 ]));
|
|
|
|
newOutput(0, nodeValue_Output("Path data", self, VALUE_TYPE.pathnode, self));
|
|
|
|
input_display_list = [
|
|
["Transform", false], 0, 2, 1,
|
|
["Shape", false], 3, 4, 5, 6, 7, 8, 9,
|
|
];
|
|
|
|
points = [];
|
|
lengths = [];
|
|
lengthAccs = [];
|
|
lengthTotal = 0;
|
|
boundary = new BoundingBox();
|
|
cached_pos = ds_map_create();
|
|
|
|
loop = true;
|
|
shape = 0;
|
|
posx = 0; posy = 0;
|
|
scax = 1; scay = 1;
|
|
rot = 0;
|
|
|
|
pa1 = 0;
|
|
pa2x = 0; pa2y = 0;
|
|
pa3 = 0;
|
|
|
|
corners = [ 0, 0, 0, 0 ];
|
|
|
|
preview_surf = noone;
|
|
|
|
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 getPointRatio = function(_rat, _ind = 0, out = undefined) {
|
|
out ??= new __vec2();
|
|
_rat = frac(_rat);
|
|
|
|
switch(shapeScroll[shape].name) {
|
|
case "Ellipse" :
|
|
var a = 360 * _rat;
|
|
out.x = posx + lengthdir_x(scax, a);
|
|
out.y = posy + lengthdir_y(scay, a);
|
|
break;
|
|
|
|
case "Arc" :
|
|
var a = lerp_float_angle(pa2x, pa2y, _rat);
|
|
out.x = posx + lengthdir_x(scax, a);
|
|
out.y = posy + lengthdir_y(scay, a);
|
|
break;
|
|
|
|
case "Squircle" :
|
|
var a = 360 * _rat;
|
|
var r = 1 / power(power(dcos(a % 90), pa3) + power(dsin(a % 90), pa3), 1 / pa3);
|
|
|
|
out.x = posx + lengthdir_x(scax * r, a);
|
|
out.y = posy + lengthdir_y(scay * r, a);
|
|
break;
|
|
|
|
case "Trapezoid" :
|
|
case "Parallelogram" :
|
|
case "Rectangle" :
|
|
case "Polygon" :
|
|
case "Star" :
|
|
case "Line" :
|
|
case "Curve" :
|
|
return getPointDistance(_rat * lengthTotal, _ind, out);
|
|
|
|
}
|
|
|
|
point_vec2_rotate(out, posx, posy, rot);
|
|
return out;
|
|
}
|
|
|
|
static getPointDistance = function(_dist, _ind = 0, out = undefined) {
|
|
out ??= new __vec2();
|
|
_dist = safe_mod(_dist, lengthTotal);
|
|
|
|
var _d = _dist, l;
|
|
var np = array_length(points);
|
|
|
|
for( var i = 0, n = array_length(lengths); i < n; i++ ) {
|
|
l = lengths[i];
|
|
if(_d > l) { _d -= l; continue; }
|
|
|
|
var p0 = points[(i + 0) % np];
|
|
var p1 = points[(i + 1) % np];
|
|
|
|
out.x = lerp(p0[0], p1[0], _d / l);
|
|
out.y = lerp(p0[1], p1[1], _d / l);
|
|
break;
|
|
}
|
|
|
|
return out;
|
|
}
|
|
|
|
static drawOverlay = function(hover, active, _x, _y, _s, _mx, _my, _snx, _sny) {
|
|
if(array_empty(points)) return;
|
|
|
|
var ox = _x + points[0][0] * _s, x0 = ox;
|
|
var oy = _y + points[0][1] * _s, y0 = oy;
|
|
var nx, ny;
|
|
|
|
draw_set_color(COLORS._main_accent);
|
|
for( var i = 1, n = array_length(points); i < n; i++ ) {
|
|
nx = _x + points[i][0] * _s;
|
|
ny = _y + points[i][1] * _s;
|
|
|
|
draw_line(ox, oy, nx, ny);
|
|
|
|
ox = nx;
|
|
oy = ny;
|
|
}
|
|
|
|
if(loop) draw_line(ox, oy, x0, y0);
|
|
|
|
var _px = _x + posx * _s;
|
|
var _py = _y + posy * _s;
|
|
|
|
var h = inputs[0].drawOverlay(hover, active, _x, _y, _s, _mx, _my, _snx, _sny); hover &= !h;
|
|
var h = inputs[2].drawOverlay(hover, active, _px, _py, _s, _mx, _my, _snx, _sny); hover &= !h;
|
|
var h = inputs[1].drawOverlay(hover, active, _px, _py, _s, _mx, _my, _snx, _sny); hover &= !h;
|
|
|
|
}
|
|
|
|
static update = function(frame = CURRENT_FRAME) {
|
|
var _pos = inputs[0].getValue();
|
|
var _rot = inputs[2].getValue();
|
|
var _sca = inputs[1].getValue();
|
|
var _pa1 = inputs[4].getValue();
|
|
var _aran = inputs[5].getValue();
|
|
var _pa3 = inputs[6].getValue();
|
|
var _sid = inputs[7].getValue();
|
|
var _inn = inputs[8].getValue();
|
|
var _c = inputs[9].getValue();
|
|
|
|
shape = inputs[3].getValue();
|
|
posx = _pos[0];
|
|
posy = _pos[1];
|
|
rot = _rot;
|
|
scax = _sca[0];
|
|
scay = _sca[1];
|
|
|
|
pa1 = _pa1;
|
|
pa2x = _aran[0];
|
|
pa2y = _aran[1];
|
|
pa3 = _pa3;
|
|
|
|
corners = _c;
|
|
|
|
var ox, oy, nx, ny, x0, y0;
|
|
|
|
inputs[4].setVisible(false);
|
|
inputs[5].setVisible(false);
|
|
inputs[6].setVisible(false);
|
|
inputs[7].setVisible(false);
|
|
inputs[8].setVisible(false);
|
|
inputs[9].setVisible(false);
|
|
|
|
switch(shapeScroll[shape].name) {
|
|
case "Rectangle" :
|
|
loop = true;
|
|
inputs[9].setVisible(true);
|
|
|
|
var x0 = posx - scax;
|
|
var y0 = posy - scay;
|
|
|
|
var x1 = posx + scax;
|
|
var y1 = posy + scay;
|
|
|
|
var p = [
|
|
[ x0, y1 ],
|
|
[ x0, y0 ],
|
|
[ x1, y0 ],
|
|
|
|
[ x0, y0 ],
|
|
[ x1, y0 ],
|
|
[ x1, y1 ],
|
|
|
|
[ x1, y0 ],
|
|
[ x1, y1 ],
|
|
[ x0, y1 ],
|
|
|
|
[ x1, y1 ],
|
|
[ x0, y1 ],
|
|
[ x0, y0 ],
|
|
];
|
|
|
|
var ar = array_create(4);
|
|
for( var i = 0; i < 4; i++ ) {
|
|
ar[i] = _c[i]? get_corner_radius(p[i * 3 + 0][0], p[i * 3 + 0][1],
|
|
p[i * 3 + 1][0], p[i * 3 + 1][1],
|
|
p[i * 3 + 2][0], p[i * 3 + 2][1], _c[i], 64, 0) : [ p[i * 3 + 1] ];
|
|
}
|
|
|
|
points = array_merge( ar[0], ar[1], ar[2], ar[3] );
|
|
break;
|
|
|
|
case "Trapezoid" :
|
|
loop = true;
|
|
inputs[4].setVisible(true);
|
|
inputs[9].setVisible(true);
|
|
|
|
var x0 = posx - scax * saturate(1 - _pa1);
|
|
var y0 = posy - scay;
|
|
|
|
var x1 = posx + scax * saturate(1 - _pa1);
|
|
var y1 = posy - scay;
|
|
|
|
var x2 = posx + scax * saturate(1 + _pa1);
|
|
var y2 = posy + scay;
|
|
|
|
var x3 = posx - scax * saturate(1 + _pa1);
|
|
var y3 = posy + scay;
|
|
|
|
var p = [
|
|
[ x3, y3 ],
|
|
[ x0, y0 ],
|
|
[ x1, y1 ],
|
|
|
|
[ x0, y0 ],
|
|
[ x1, y1 ],
|
|
[ x2, y2 ],
|
|
|
|
[ x1, y1 ],
|
|
[ x2, y2 ],
|
|
[ x3, y3 ],
|
|
|
|
[ x2, y3 ],
|
|
[ x3, y3 ],
|
|
[ x0, y0 ],
|
|
];
|
|
|
|
var ar = array_create(4);
|
|
for( var i = 0; i < 4; i++ ) {
|
|
ar[i] = _c[i]? get_corner_radius(p[i * 3 + 0][0], p[i * 3 + 0][1],
|
|
p[i * 3 + 1][0], p[i * 3 + 1][1],
|
|
p[i * 3 + 2][0], p[i * 3 + 2][1], _c[i], 64, 0) : [ p[i * 3 + 1] ];
|
|
}
|
|
|
|
points = array_merge( ar[0], ar[1], ar[2], ar[3] );
|
|
break;
|
|
|
|
case "Parallelogram" :
|
|
loop = true;
|
|
inputs[4].setVisible(true);
|
|
inputs[9].setVisible(true);
|
|
|
|
var x0 = posx - scax * saturate(1 - _pa1);
|
|
var y0 = posy - scay;
|
|
|
|
var x1 = posx + scax * saturate(1 + _pa1);
|
|
var y1 = posy - scay;
|
|
|
|
var x2 = posx + scax * saturate(1 - _pa1);
|
|
var y2 = posy + scay;
|
|
|
|
var x3 = posx - scax * saturate(1 + _pa1);
|
|
var y3 = posy + scay;
|
|
|
|
var p = [
|
|
[ x3, y3 ],
|
|
[ x0, y0 ],
|
|
[ x1, y1 ],
|
|
|
|
[ x0, y0 ],
|
|
[ x1, y1 ],
|
|
[ x2, y2 ],
|
|
|
|
[ x1, y1 ],
|
|
[ x2, y2 ],
|
|
[ x3, y3 ],
|
|
|
|
[ x2, y3 ],
|
|
[ x3, y3 ],
|
|
[ x0, y0 ],
|
|
];
|
|
|
|
var ar = array_create(4);
|
|
for( var i = 0; i < 4; i++ ) {
|
|
ar[i] = _c[i]? get_corner_radius(p[i * 3 + 0][0], p[i * 3 + 0][1],
|
|
p[i * 3 + 1][0], p[i * 3 + 1][1],
|
|
p[i * 3 + 2][0], p[i * 3 + 2][1], _c[i], 64, 0) : [ p[i * 3 + 1] ];
|
|
}
|
|
|
|
points = array_merge( ar[0], ar[1], ar[2], ar[3] );
|
|
break;
|
|
|
|
case "Ellipse" :
|
|
loop = true;
|
|
var _st = 64;
|
|
var _as = 360 / _st;
|
|
points = array_create(_st);
|
|
|
|
for( var i = 0; i < _st; i++ ) {
|
|
nx = posx + lengthdir_x(scax, _as * i);
|
|
ny = posy + lengthdir_y(scay, _as * i);
|
|
points[i] = [ nx, ny ];
|
|
}
|
|
break;
|
|
|
|
case "Arc" :
|
|
loop = false;
|
|
inputs[5].setVisible(true);
|
|
|
|
var _st = 64;
|
|
var _as = 1 / (_st - 1);
|
|
points = array_create(_st);
|
|
|
|
for( var i = 0; i < _st; i++ ) {
|
|
var a = lerp_float_angle(_aran[0], _aran[1], i * _as);
|
|
nx = posx + lengthdir_x(scax, a);
|
|
ny = posy + lengthdir_y(scay, a);
|
|
points[i] = [ nx, ny ];
|
|
}
|
|
break;
|
|
|
|
case "Squircle" :
|
|
loop = true;
|
|
inputs[6].setVisible(true);
|
|
|
|
var _st = 64;
|
|
var _as = 360 / _st;
|
|
points = array_create(_st);
|
|
|
|
for( var i = 0; i < _st; i++ ) {
|
|
var a = _as * i;
|
|
var r = 1 / power(power(dcos(a % 90), pa3) + power(dsin(a % 90), pa3), 1 / pa3);
|
|
|
|
nx = posx + lengthdir_x(scax * r, a);
|
|
ny = posy + lengthdir_y(scay * r, a);
|
|
points[i] = [ nx, ny ];
|
|
}
|
|
break;
|
|
|
|
case "Polygon" :
|
|
loop = true;
|
|
inputs[7].setVisible(true);
|
|
|
|
var _st = _sid;
|
|
var _as = 360 / _st;
|
|
points = array_create(_st);
|
|
|
|
for( var i = 0; i < _st; i++ ) {
|
|
nx = posx + lengthdir_x(scax, _as * i);
|
|
ny = posy + lengthdir_y(scay, _as * i);
|
|
points[i] = [ nx, ny ];
|
|
}
|
|
break;
|
|
|
|
case "Star" :
|
|
loop = true;
|
|
inputs[7].setVisible(true);
|
|
inputs[8].setVisible(true);
|
|
|
|
var _st = _sid;
|
|
var _as = 360 / _st;
|
|
points = array_create(_st * 2);
|
|
|
|
for( var i = 0; i < _st; i++ ) {
|
|
nx = posx + lengthdir_x(scax, _as * i);
|
|
ny = posy + lengthdir_y(scay, _as * i);
|
|
points[i * 2 + 0] = [ nx, ny ];
|
|
|
|
nx = posx + lengthdir_x(scax * _inn, _as * i + _as / 2);
|
|
ny = posy + lengthdir_y(scay * _inn, _as * i + _as / 2);
|
|
points[i * 2 + 1] = [ nx, ny ];
|
|
}
|
|
break;
|
|
|
|
case "Line":
|
|
loop = false;
|
|
points = [
|
|
[ posx - scax, posy ],
|
|
[ posx + scax, posy ],
|
|
];
|
|
break;
|
|
|
|
case "Curve":
|
|
loop = false;
|
|
inputs[6].setVisible(true);
|
|
|
|
var _st = 64;
|
|
var _as = 180 / (_st - 1);
|
|
var _x0 = posx - scax;
|
|
var _x1 = posx + scax;
|
|
var _yy = posy;
|
|
|
|
points = array_create(_st);
|
|
for( var i = 0; i < _st; i++ ) {
|
|
points[i] = [
|
|
lerp(_x0, _x1, i / (_st - 1)),
|
|
_yy + dsin(i * _as) * pa3,
|
|
]
|
|
}
|
|
break;
|
|
}
|
|
|
|
array_map_ext(points, function(p) /*=>*/ {return point_rotate(p[0], p[1], posx, posy, rot, p)});
|
|
|
|
var n = array_length(points);
|
|
lengths = array_create(n + loop);
|
|
|
|
if(n) {
|
|
for( var i = 0; i < n; i++ ) {
|
|
nx = points[i][0];
|
|
ny = points[i][1];
|
|
|
|
if(i) lengths[i - 1] = point_distance(ox, oy, nx, ny);
|
|
else { x0 = nx; y0 = ny; }
|
|
|
|
ox = nx;
|
|
oy = ny;
|
|
}
|
|
|
|
if(loop) lengths[n - 1] = point_distance(ox, oy, x0, y0);
|
|
}
|
|
|
|
var _len = array_length(lengths);
|
|
lengthTotal = 0;
|
|
lengthAccs = array_create(_len);
|
|
|
|
for( var i = 0; i < _len; i++ ) {
|
|
lengthTotal += lengths[i];
|
|
lengthAccs[i] = lengthTotal;
|
|
}
|
|
|
|
boundary = new BoundingBox(posx - scax, posy - scay, posx + scax, posy + scay);
|
|
|
|
preview_surf = surface_verify(preview_surf, 128, 128);
|
|
surface_set_target(preview_surf);
|
|
DRAW_CLEAR
|
|
|
|
var ox, x0;
|
|
var oy, y0;
|
|
var nx, ny;
|
|
var xx = posx - scax;
|
|
var yy = posy - scay;
|
|
var ww = scax * 2;
|
|
var hh = scay * 2;
|
|
draw_set_color(COLORS._main_accent);
|
|
|
|
if(array_length(points)) {
|
|
for( var i = 0, n = array_length(points); i < n; i++ ) {
|
|
nx = 4 + (points[i][0] - xx) / ww * 120;
|
|
ny = 4 + (points[i][1] - yy) / hh * 120;
|
|
|
|
if(i) draw_line_width(ox, oy, nx, ny, 8);
|
|
else { x0 = nx; y0 = ny; }
|
|
|
|
ox = nx;
|
|
oy = ny;
|
|
}
|
|
|
|
if(loop) draw_line_width(ox, oy, x0, y0, 8);
|
|
}
|
|
surface_reset_target();
|
|
}
|
|
|
|
static onDrawNode = function(xx, yy, _mx, _my, _s, _hover, _focus) {
|
|
var bbox = drawGetBbox(xx, yy, _s);
|
|
draw_surface_bbox(preview_surf, bbox);
|
|
}
|
|
|
|
static getPreviewBoundingBox = function() { return BBOX().fromBoundingBox(boundary); }
|
|
} |