mirror of
https://github.com/Ttanasart-pt/Pixel-Composer.git
synced 2025-01-25 20:38:07 +01:00
352 lines
No EOL
11 KiB
Text
352 lines
No EOL
11 KiB
Text
function Node_Image_Sheet(_x, _y, _group = -1) : Node(_x, _y, _group) constructor {
|
|
name = "Splice Spritesheet";
|
|
|
|
surf_array = [];
|
|
|
|
inputs[| 0] = nodeValue(0, "Surface in", self, JUNCTION_CONNECT.input, VALUE_TYPE.surface, 0).rejectArray();
|
|
|
|
inputs[| 1] = nodeValue(1, "Sprite size", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, [ 32, 32 ])
|
|
.setDisplay(VALUE_DISPLAY.vector);
|
|
|
|
inputs[| 2] = nodeValue(2, "Row", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 1); //unused
|
|
inputs[| 3] = nodeValue(3, "Amount", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, [ 1, 1 ])
|
|
.setDisplay(VALUE_DISPLAY.vector);
|
|
|
|
inputs[| 4] = nodeValue(4, "Offset", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, [ 0, 0 ])
|
|
.setDisplay(VALUE_DISPLAY.vector);
|
|
|
|
inputs[| 5] = nodeValue(5, "Spacing", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, [ 0, 0 ])
|
|
.setDisplay(VALUE_DISPLAY.vector);
|
|
|
|
inputs[| 6] = nodeValue(6, "Padding", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, [0, 0, 0, 0])
|
|
.setDisplay(VALUE_DISPLAY.padding);
|
|
|
|
inputs[| 7] = nodeValue(7, "Output", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 1)
|
|
.setDisplay(VALUE_DISPLAY.enum_scroll, [ "Animation", "Array"]);
|
|
|
|
inputs[| 8] = nodeValue(8, "Animation speed", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, 1);
|
|
|
|
inputs[| 9] = nodeValue(9, "Orientation", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 0)
|
|
.setDisplay(VALUE_DISPLAY.enum_scroll, [ "Horizontal", "Vertical"]);
|
|
|
|
inputs[| 10] = nodeValue(10, "Auto fill", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 0)
|
|
.setDisplay(VALUE_DISPLAY.button, [ function() {
|
|
var _sur = inputs[| 0].getValue();
|
|
if(!is_surface(_sur) || _sur == DEF_SURFACE) return;
|
|
var ww = surface_get_width(_sur);
|
|
var hh = surface_get_height(_sur);
|
|
|
|
var _size = inputs[| 1].getValue();
|
|
var _offs = inputs[| 4].getValue();
|
|
var _spac = inputs[| 5].getValue();
|
|
var _orie = inputs[| 9].getValue();
|
|
|
|
var sh_w = _size[0] + _spac[0];
|
|
var sh_h = _size[1] + _spac[1];
|
|
|
|
var fill_w = floor((ww - _offs[0]) / sh_w);
|
|
var fill_h = floor((hh - _offs[1]) / sh_h);
|
|
|
|
if(_orie == 0)
|
|
inputs[| 3].setValue([ fill_w, fill_h ]);
|
|
else
|
|
inputs[| 3].setValue([ fill_h, fill_w ]);
|
|
|
|
inspectorUpdate();
|
|
}, "Auto fill"] );
|
|
|
|
inputs[| 11] = nodeValue(11, "Sync animation", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 0)
|
|
.setDisplay(VALUE_DISPLAY.button, [ function() {
|
|
var _amo = inputs[| 3].getValue();
|
|
ANIMATOR.frames_total = max(1, _amo[0] * _amo[1]);
|
|
}, "Sync frames"] );
|
|
|
|
inputs[| 12] = nodeValue(12, "Filter empty output", self, JUNCTION_CONNECT.input, VALUE_TYPE.boolean, false);
|
|
|
|
input_display_list = [
|
|
["Sprite", false], 0, 1, 6, 10,
|
|
["Sheet", false], 3, 9, 4, 5,
|
|
["Output", false], 7, 8, 12, 11
|
|
];
|
|
|
|
outputs[| 0] = nodeValue(0, "Surface out", self, JUNCTION_CONNECT.output, VALUE_TYPE.surface, PIXEL_SURFACE);
|
|
|
|
drag_type = 0;
|
|
drag_sx = 0;
|
|
drag_sy = 0;
|
|
drag_mx = 0;
|
|
drag_my = 0;
|
|
curr_off = [0, 0];
|
|
curr_dim = [0, 0];
|
|
curr_amo = [0, 0];
|
|
|
|
sprite_valid = [];
|
|
|
|
static getPreviewValue = function() { return inputs[| 0]; }
|
|
|
|
function getSpritePosition(index) {
|
|
var _dim = curr_dim;
|
|
var _col = curr_amo[0];
|
|
var _off = curr_off;
|
|
var _spa = inputs[| 5].getValue();
|
|
var _ori = inputs[| 9].getValue();
|
|
|
|
var _irow = floor(index / _col);
|
|
var _icol = safe_mod(index, _col);
|
|
|
|
var _x, _y;
|
|
|
|
var _x = _off[0] + _icol * (_dim[0] + _spa[0]);
|
|
var _y = _off[1] + _irow * (_dim[1] + _spa[1]);
|
|
|
|
if(_ori == 0)
|
|
return [_x, _y];
|
|
else
|
|
return [_y, _x];
|
|
}
|
|
|
|
static drawOverlay = function(active, _x, _y, _s, _mx, _my, _snx, _sny) {
|
|
var _inSurf = inputs[| 0].getValue();
|
|
if(!is_surface(_inSurf)) return;
|
|
|
|
var _out = inputs[| 7].getValue();
|
|
var _spc = inputs[| 5].getValue();
|
|
if(drag_type == 0) {
|
|
curr_dim = inputs[| 1].getValue();
|
|
curr_amo = inputs[| 3].getValue();
|
|
curr_off = inputs[| 4].getValue();
|
|
}
|
|
|
|
var _amo = curr_amo[0] * curr_amo[1];
|
|
|
|
for(var i = _amo - 1; i >= 0; i--) {
|
|
if(!array_safe_get(sprite_valid, i, true))
|
|
continue;
|
|
|
|
var _f = getSpritePosition(i);
|
|
var _fx0 = _x + _f[0] * _s;
|
|
var _fy0 = _y + _f[1] * _s;
|
|
var _fx1 = _fx0 + curr_dim[0] * _s;
|
|
var _fy1 = _fy0 + curr_dim[1] * _s;
|
|
|
|
draw_set_color(COLORS._main_accent);
|
|
draw_set_alpha(i == 0? 1 : 0.75);
|
|
draw_rectangle(_fx0, _fy0, _fx1 - 1, _fy1 - 1, true);
|
|
draw_set_alpha(1);
|
|
|
|
//draw_set_text(f_p1, fa_left, fa_top);
|
|
//draw_text(_fx0 + 2, _fy0 + 2, string(i));
|
|
}
|
|
|
|
var __ax = curr_off[0];
|
|
var __ay = curr_off[1];
|
|
var __aw = curr_dim[0];
|
|
var __ah = curr_dim[1];
|
|
|
|
var _ax = __ax * _s + _x;
|
|
var _ay = __ay * _s + _y;
|
|
var _aw = __aw * _s;
|
|
var _ah = __ah * _s;
|
|
|
|
var _bw = curr_amo[0] * (curr_dim[0] + _spc[0]) - _spc[0]; _bw *= _s;
|
|
var _bh = curr_amo[1] * (curr_dim[1] + _spc[1]) - _spc[1]; _bh *= _s;
|
|
|
|
draw_sprite_ui_uniform(THEME.anchor, 0, _ax, _ay);
|
|
draw_sprite_ui_uniform(THEME.anchor_selector, 0, _ax + _aw, _ay + _ah);
|
|
draw_sprite_ui_uniform(THEME.anchor_arrow, 0, _ax + _bw + _s * 4, _ay + _bh / 2);
|
|
draw_sprite_ui_uniform(THEME.anchor_arrow, 0, _ax + _bw / 2, _ay + _bh + _s * 4,,,, -90);
|
|
|
|
if(active) {
|
|
if(point_in_circle(_mx, _my, _ax + _aw, _ay + _ah, 8))
|
|
draw_sprite_ui_uniform(THEME.anchor_selector, 1, _ax + _aw, _ay + _ah);
|
|
else if(point_in_rectangle(_mx, _my, _ax - _aw, _ay - _ah, _ax + _aw, _ay + _ah))
|
|
draw_sprite_ui_uniform(THEME.anchor, 0, _ax, _ay, 1.25, c_white);
|
|
else if(point_in_circle(_mx, _my, _ax + _bw + _s * 4, _ay + _bh / 2, 8))
|
|
draw_sprite_ui_uniform(THEME.anchor_arrow, 1, _ax + _bw + _s * 4, _ay + _bh / 2);
|
|
else if(point_in_circle(_mx, _my, _ax + _bw / 2, _ay + _bh + _s * 4, 8))
|
|
draw_sprite_ui_uniform(THEME.anchor_arrow, 1, _ax + _bw / 2, _ay + _bh + _s * 4,,,, -90);
|
|
}
|
|
|
|
#region area
|
|
var __dim = inputs[| 1].getValue();
|
|
var __amo = inputs[| 3].getValue();
|
|
var __off = inputs[| 4].getValue();
|
|
|
|
var _ax = __off[0] * _s + _x;
|
|
var _ay = __off[1] * _s + _y;
|
|
var _aw = __dim[0] * _s;
|
|
var _ah = __dim[1] * _s;
|
|
|
|
//draw_set_color(COLORS._main_accent);
|
|
//draw_rectangle(_ax - _aw, _ay - _ah, _ax + _aw, _ay + _ah, true);
|
|
|
|
if(drag_type == 1) {
|
|
var _xx = value_snap(round(drag_sx + (_mx - drag_mx) / _s), _snx);
|
|
var _yy = value_snap(round(drag_sy + (_my - drag_my) / _s), _sny);
|
|
|
|
var off = [_xx, _yy];
|
|
curr_off = off;
|
|
|
|
if(mouse_release(mb_left)) {
|
|
drag_type = 0;
|
|
inputs[| 4].setValue(off);
|
|
}
|
|
} else if(drag_type == 2) {
|
|
var _dx = value_snap(round(abs((_mx - drag_mx) / _s)), _snx);
|
|
var _dy = value_snap(round(abs((_my - drag_my) / _s)), _sny);
|
|
|
|
var dim = [_dx, _dy];
|
|
curr_dim = dim;
|
|
|
|
if(keyboard_check(vk_shift)) {
|
|
dim[0] = max(_dx, _dy);
|
|
dim[1] = max(_dx, _dy);
|
|
}
|
|
|
|
if(mouse_release(mb_left)) {
|
|
drag_type = 0;
|
|
inputs[| 1].setValue(dim);
|
|
}
|
|
} else if(drag_type == 3) {
|
|
var _col = floor((abs(_mx - drag_mx) / _s - _spc[0]) / (__dim[0] + _spc[0]));
|
|
curr_amo[0] = _col;
|
|
|
|
if(mouse_release(mb_left)) {
|
|
drag_type = 0;
|
|
inputs[| 3].setValue(curr_amo);
|
|
}
|
|
} else if(drag_type == 4) {
|
|
var _row = floor((abs(_my - drag_my) / _s - _spc[1]) / (__dim[1] + _spc[1]));
|
|
curr_amo[1] = _row;
|
|
|
|
if(mouse_release(mb_left)) {
|
|
drag_type = 0;
|
|
inputs[| 3].setValue(curr_amo);
|
|
}
|
|
}
|
|
|
|
if(mouse_press(mb_left, active)) {
|
|
if(point_in_circle(_mx, _my, _ax + _aw, _ay + _ah, 8)) { // drag size
|
|
drag_type = 2;
|
|
drag_mx = _ax;
|
|
drag_my = _ay;
|
|
} else if(point_in_rectangle(_mx, _my, _ax - _aw, _ay - _ah, _ax + _aw, _ay + _ah)) { // drag position
|
|
drag_type = 1;
|
|
drag_sx = __off[0];
|
|
drag_sy = __off[1];
|
|
drag_mx = _mx;
|
|
drag_my = _my;
|
|
} else if(point_in_circle(_mx, _my, _ax + _bw + _s * 4, _ay + _bh / 2, 8)) { // drag col
|
|
drag_type = 3;
|
|
drag_mx = _ax;
|
|
drag_my = _ay;
|
|
} else if(point_in_circle(_mx, _my, _ax + _bw / 2, _ay + _bh + _s * 4, 8)) { // drag row
|
|
drag_type = 4;
|
|
drag_mx = _ax;
|
|
drag_my = _ay;
|
|
}
|
|
}
|
|
#endregion
|
|
}
|
|
|
|
static step = function() {
|
|
var _out = inputs[| 7].getValue();
|
|
inputs[| 11].setVisible(!_out);
|
|
inputs[| 8].setVisible(!_out);
|
|
inputs[| 12].setVisible( _out);
|
|
}
|
|
|
|
static inspectorUpdate = function() {
|
|
var _inSurf = inputs[| 0].getValue();
|
|
if(!is_surface(_inSurf)) return;
|
|
|
|
var _outSurf = outputs[| 0].getValue();
|
|
var _out = inputs[| 7].getValue();
|
|
|
|
var _dim = inputs[| 1].getValue();
|
|
var _amo = inputs[| 3].getValue();
|
|
var _off = inputs[| 4].getValue();
|
|
var _total = _amo[0] * _amo[1];
|
|
var _pad = inputs[| 6].getValue();
|
|
|
|
var ww = _dim[0] + _pad[0] + _pad[2];
|
|
var hh = _dim[1] + _pad[1] + _pad[3];
|
|
|
|
var _filt = inputs[| 12].getValue();
|
|
|
|
curr_dim = _dim;
|
|
curr_amo = is_array(_amo)? _amo : [1, 1];
|
|
curr_off = _off;
|
|
|
|
var filSize = 4;
|
|
var _empS = surface_create(filSize, filSize);
|
|
var _buff = buffer_create(filSize * filSize * 4, buffer_fixed, 2);
|
|
|
|
surf_array = [];
|
|
for( var i = 0; i < array_length(surf_array); i++ ) {
|
|
if(is_surface(surf_array[i]))
|
|
surface_free(surf_array[i]);
|
|
}
|
|
|
|
for(var i = 0; i < _total; i++) {
|
|
var _s = surface_create_valid(ww, hh);
|
|
var _spr_pos = getSpritePosition(i);
|
|
|
|
surface_set_target(_s);
|
|
draw_clear_alpha(c_black, 0);
|
|
BLEND_OVERRIDE
|
|
draw_surface_part(_inSurf, _spr_pos[0], _spr_pos[1], _dim[0], _dim[1], _pad[2], _pad[1]);
|
|
BLEND_NORMAL
|
|
surface_reset_target();
|
|
|
|
if(_filt) {
|
|
gpu_set_tex_filter(true);
|
|
surface_set_target(_empS);
|
|
draw_clear_alpha(0, 0);
|
|
BLEND_OVERRIDE
|
|
draw_surface_stretched(_s, 0, 0, filSize, filSize);
|
|
BLEND_NORMAL
|
|
surface_reset_target();
|
|
gpu_set_tex_filter(false);
|
|
|
|
buffer_get_surface(_buff, _empS, 0);
|
|
buffer_seek(_buff, buffer_seek_start, 0);
|
|
var empty = true;
|
|
var c0 = buffer_read(_buff, buffer_u32) & ~(0b11111111 << 24);
|
|
repeat(filSize * filSize - 1) {
|
|
if(buffer_read(_buff, buffer_u32) & ~(0b11111111 << 24) != c0) {
|
|
empty = false;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(!empty)
|
|
array_push(surf_array, _s);
|
|
sprite_valid[i] = !empty;
|
|
} else {
|
|
array_push(surf_array, _s);
|
|
sprite_valid[i] = true;
|
|
}
|
|
}
|
|
|
|
if(_out == 1)
|
|
outputs[| 0].setValue(surf_array);
|
|
|
|
buffer_delete(_buff);
|
|
surface_free(_empS);
|
|
}
|
|
|
|
static update = function() {
|
|
var _out = inputs[| 7].getValue();
|
|
if(_out == 1) {
|
|
outputs[| 0].setValue(surf_array);
|
|
update_on_frame = false;
|
|
return;
|
|
}
|
|
|
|
var _spd = inputs[| 8].getValue();
|
|
update_on_frame = true;
|
|
var ind = safe_mod(ANIMATOR.current_frame * _spd, array_length(surf_array));
|
|
outputs[| 0].setValue(surf_array[ind]);
|
|
}
|
|
} |