mirror of
https://github.com/Ttanasart-pt/Pixel-Composer.git
synced 2025-01-01 01:46:32 +01:00
923 lines
27 KiB
Text
923 lines
27 KiB
Text
function Node_Armature(_x, _y, _group = noone) : Node(_x, _y, _group) constructor {
|
|
name = "Armature Create";
|
|
setDimension(96, 96);
|
|
draw_padding = 8;
|
|
|
|
bone_constrain_adding = noone;
|
|
function addConstain(_c) {
|
|
if(bone_constrain_adding == noone) return;
|
|
array_push(bones.constrains, new __Bone_Constrain(bones).build(_c, bone_constrain_adding.ID));
|
|
triggerRender();
|
|
}
|
|
|
|
bone_constrain_menu = [
|
|
new MenuItem("Copy Position", function() /*=>*/ { addConstain("Copy Position") }, [ s_bone_constrain, 0, 1, c_white ]),
|
|
new MenuItem("Copy Rotation", function() /*=>*/ { addConstain("Copy Rotation") }, [ s_bone_constrain, 1, 1, c_white ]),
|
|
new MenuItem("Copy Scale", function() /*=>*/ { addConstain("Copy Scale") }, [ s_bone_constrain, 2, 1, c_white ]),
|
|
-1,
|
|
new MenuItem("Look At", function() /*=>*/ { addConstain("Look At") }, [ s_bone_constrain, 3, 1, c_white ]),
|
|
new MenuItem("Move To", function() /*=>*/ { addConstain("Move To") }, [ s_bone_constrain, 4, 1, c_white ]),
|
|
new MenuItem("Stretch To", function() /*=>*/ { addConstain("Stretch To") }, [ s_bone_constrain, 5, 1, c_white ]),
|
|
-1,
|
|
new MenuItem("Limit Rotation", function() /*=>*/ { addConstain("Limit Rotation") }, [ s_bone_constrain, 6, 1, c_white ]),
|
|
];
|
|
|
|
bone_array = [];
|
|
bone_renderer = new Inspector_Custom_Renderer(function(_x, _y, _w, _m, _hover, _focus) {
|
|
var _b = bones;
|
|
if(_b == noone) return 0;
|
|
var amo = _b.childCount();
|
|
var _hh = ui(28);
|
|
var bh = ui(16) + amo * _hh;
|
|
var ty = _y;
|
|
|
|
draw_sprite_stretched_ext(THEME.ui_panel_bg, 1, _x, ty, _w, bh, COLORS.node_composite_bg_blend, 1);
|
|
ty += ui(8);
|
|
|
|
var hovering = noone;
|
|
var _bst = ds_stack_create();
|
|
ds_stack_push(_bst, [ _b, _x, _w ]);
|
|
|
|
var bone_remove = noone;
|
|
|
|
while(!ds_stack_empty(_bst)) {
|
|
var _st = ds_stack_pop(_bst);
|
|
var bone = _st[0];
|
|
var __x = _st[1];
|
|
var __w = _st[2];
|
|
|
|
for( var i = 0, n = array_length(bone.childs); i < n; i++ )
|
|
ds_stack_push(_bst, [ bone.childs[i], __x + 16, __w - 16 ]);
|
|
|
|
if(bone.is_main) continue;
|
|
|
|
if(bone.parent_anchor)
|
|
draw_sprite_ui(THEME.bone, 1, __x + 12, ty + 14,,,, COLORS._main_icon);
|
|
|
|
else if(bone.IKlength)
|
|
draw_sprite_ui(THEME.bone, 2, __x + 12, ty + 14,,,, COLORS._main_icon);
|
|
|
|
else {
|
|
if(_hover && point_in_circle(_m[0], _m[1], __x + 12, ty + 12, 12)) {
|
|
draw_sprite_ui(THEME.bone, 0, __x + 12, ty + 14,,,, COLORS._main_icon_light);
|
|
if(mouse_press(mb_left, _focus))
|
|
bone_dragging = bone;
|
|
} else
|
|
draw_sprite_ui(THEME.bone, 0, __x + 12, ty + 14,,,, COLORS._main_icon);
|
|
}
|
|
|
|
if(point_in_rectangle(_m[0], _m[1], __x + 24, ty + 3, __x + __w, ty + _hh - 3))
|
|
anchor_selecting = [ bone, 2 ];
|
|
|
|
var bx = __x + __w - 24;
|
|
var by = ty + _hh / 2;
|
|
|
|
if(point_in_circle(_m[0], _m[1], bx, by, 16)) {
|
|
draw_sprite_ui_uniform(THEME.icon_delete, 3, bx, by, 1, COLORS._main_value_negative);
|
|
|
|
if(mouse_press(mb_left, _focus))
|
|
bone_remove = bone;
|
|
} else
|
|
draw_sprite_ui_uniform(THEME.icon_delete, 3, bx, by, 1, COLORS._main_icon);
|
|
|
|
draw_set_font(f_p2);
|
|
var ww = string_width(bone.name);
|
|
|
|
bone.tb_name.setFocusHover(_focus, _hover);
|
|
bone.tb_name.draw(__x + 24, ty + 3, ww + 16, _hh - 6, bone.name, _m);
|
|
|
|
var _x0 = bx - 24;
|
|
var _y0 = by;
|
|
var cc = bone.apply_scale? COLORS._main_icon : COLORS._main_value_negative;
|
|
if(_hover && point_in_circle(_m[0], _m[1], _x0, _y0, 10)) {
|
|
TOOLTIP = "Apply Scale";
|
|
draw_sprite_ui(THEME.bone, 3, _x0, _y0,,,, cc, 0.75);
|
|
|
|
if(mouse_press(mb_left, _focus))
|
|
bone.apply_scale = !bone.apply_scale;
|
|
} else
|
|
draw_sprite_ui(THEME.bone, 3, _x0, _y0,,,, cc, 0.5);
|
|
|
|
_x0 -= 20;
|
|
var cc = bone.apply_rotation? COLORS._main_icon : COLORS._main_value_negative;
|
|
if(_hover && point_in_circle(_m[0], _m[1], _x0, _y0, 10)) {
|
|
TOOLTIP = "Apply Rotation";
|
|
draw_sprite_ui(THEME.bone, 4, _x0, _y0,,,, cc, 0.75);
|
|
|
|
if(mouse_press(mb_left, _focus))
|
|
bone.apply_rotation = !bone.apply_rotation;
|
|
} else
|
|
draw_sprite_ui(THEME.bone, 4, _x0, _y0,,,, cc, 0.5);
|
|
|
|
_x0 -= 20;
|
|
var cc = COLORS._main_icon;
|
|
if(_hover && point_in_circle(_m[0], _m[1], _x0, _y0, 10)) {
|
|
TOOLTIP = "Add Constrains";
|
|
draw_sprite_ui(THEME.bone, 5, _x0, _y0,,,, cc, .75);
|
|
|
|
if(mouse_press(mb_left, _focus)) {
|
|
bone_constrain_adding = bone;
|
|
menuCall("bone_constrain_add", bone_constrain_menu);
|
|
}
|
|
} else
|
|
draw_sprite_ui(THEME.bone, 5, _x0, _y0,,,, cc, .5);
|
|
|
|
ty += _hh;
|
|
|
|
if(!ds_stack_empty(_bst)) {
|
|
draw_set_color(COLORS.node_composite_separator);
|
|
draw_line(_x + 16, ty, _x + _w - 16, ty);
|
|
}
|
|
}
|
|
|
|
ds_stack_destroy(_bst);
|
|
|
|
if(bone_dragging && mouse_release(mb_left))
|
|
bone_dragging = noone;
|
|
|
|
if(bone_remove != noone) {
|
|
var _par = bone_remove.parent;
|
|
recordAction(ACTION_TYPE.struct_modify, bones, bones.serialize());
|
|
array_remove(_par.childs, bone_remove);
|
|
|
|
for( var i = 0, n = array_length(bone_remove.childs); i < n; i++ ) {
|
|
var _ch = bone_remove.childs[i];
|
|
_par.addChild(_ch);
|
|
|
|
_ch.parent_anchor = bone_remove.parent_anchor;
|
|
}
|
|
}
|
|
|
|
return bh;
|
|
});
|
|
|
|
constrains_h = 0;
|
|
constrain_renderer = new Inspector_Custom_Renderer(function(_x, _y, _w, _m, _hover, _focus) {
|
|
var _b = bones;
|
|
var hh = 0;
|
|
if(_b == noone) return 0;
|
|
|
|
var ty = _y;
|
|
var constrains = _b.constrains;
|
|
|
|
constrains_h = 0;
|
|
var _del = -1;
|
|
|
|
var _drawParam = {
|
|
rx: constrain_renderer.rx,
|
|
ry: constrain_renderer.ry,
|
|
|
|
selecting: anchor_selecting,
|
|
bone_array: bone_array,
|
|
}
|
|
|
|
for( var i = 0, n = array_length(constrains); i < n; i++ ) {
|
|
var _con = constrains[i];
|
|
|
|
draw_sprite_stretched_ext(THEME.ui_panel_bg, 1, _x, ty, _w, _con.draw_height + ui(32), COLORS.node_composite_bg_blend, 1);
|
|
draw_sprite_ui(s_bone_constrain, _con.sindex, _x + ui(4 + 16), ty + ui(16), 1, 1, 0, c_white, 1);
|
|
|
|
draw_set_text(f_p2, fa_left, fa_top, COLORS._main_text_sub);
|
|
draw_text_add(_x + ui(4 + 32), ty + ui(6), _con.name);
|
|
|
|
var bx = _x + _w - ui(16);
|
|
var by = ty + ui(16);
|
|
|
|
if(point_in_circle(_m[0], _m[1], bx, by, 16)) {
|
|
draw_sprite_ui_uniform(THEME.icon_delete, 3, bx, by, 1, COLORS._main_value_negative);
|
|
|
|
if(mouse_press(mb_left, _focus)) _del = i;
|
|
} else
|
|
draw_sprite_ui_uniform(THEME.icon_delete, 3, bx, by, 1, COLORS._main_icon);
|
|
|
|
_con.node = self;
|
|
var _h = _con.draw_inspector(_x, ty + ui(32), _w, _m, _hover, _focus, _drawParam);
|
|
|
|
_con.draw_height = _h;
|
|
constrains_h += _h + ui(32 + 4);
|
|
ty += _h + ui(32 + 4);
|
|
}
|
|
|
|
if(_del != -1) array_delete(constrains, _del, 1);
|
|
hh += constrains_h;
|
|
return hh;
|
|
});
|
|
|
|
input_display_list = [
|
|
["Bones", false], bone_renderer,
|
|
["Constrains", false], constrain_renderer,
|
|
];
|
|
|
|
static createBone = function(parent, distance, direction) {
|
|
recordAction(ACTION_TYPE.struct_modify, bones, bones.serialize());
|
|
|
|
var bone = new __Bone(parent, distance, direction, 0, 0, self);
|
|
parent.addChild(bone);
|
|
|
|
if(parent == bones)
|
|
bone.parent_anchor = false;
|
|
|
|
bone.distance = distance;
|
|
bone.direction = direction;
|
|
|
|
return bone;
|
|
}
|
|
|
|
newOutput(0, nodeValue_Output("Armature", self, VALUE_TYPE.armature, noone));
|
|
|
|
#region ++++ attributes ++++
|
|
bones = new __Bone(,,,,, self);
|
|
bones.name = "Main";
|
|
bones.is_main = true;
|
|
bones.node = self;
|
|
|
|
attributes.display_name = true;
|
|
attributes.display_bone = 0;
|
|
|
|
array_push(attributeEditors, "Display");
|
|
array_push(attributeEditors, ["Display name", function() /*=>*/ {return attributes.display_name}, new checkBox(function() /*=>*/ { attributes.display_name = !attributes.display_name; })]);
|
|
array_push(attributeEditors, ["Display bone", function() /*=>*/ {return attributes.display_bone}, new scrollBox(["Octahedral", "Stick"], function(ind) /*=>*/ { attributes.display_bone = ind; })]);
|
|
#endregion
|
|
|
|
tools = [
|
|
new NodeTool( "Transform", THEME.bone_tool_move ),
|
|
new NodeTool( [ "Add bones", "Mirror bones"], [ THEME.bone_tool_add, THEME.bone_tool_mirror ]),
|
|
new NodeTool( "Remove bones", THEME.bone_tool_remove ),
|
|
new NodeTool( "Detach bones", THEME.bone_tool_detach ),
|
|
new NodeTool( "IK", THEME.bone_tool_IK ),
|
|
];
|
|
|
|
anchor_selecting = noone;
|
|
builder_bone = noone;
|
|
builder_type = 0;
|
|
builder_sv = 0;
|
|
builder_sx = 0;
|
|
builder_sy = 0;
|
|
builder_mx = 0;
|
|
builder_my = 0;
|
|
|
|
bone_dragging = noone;
|
|
ik_dragging = noone;
|
|
|
|
moving = false;
|
|
scaling = false;
|
|
|
|
bone_arr = [];
|
|
bone_point_maps = [];
|
|
bone_point_mape = [];
|
|
|
|
bone_bbox = undefined;
|
|
bone_transform_bbox = -1;
|
|
bone_transform_type = -1;
|
|
|
|
mirroring_bone = noone;
|
|
|
|
static drawOverlay = function(hover, active, _x, _y, _s, _mx, _my, _snx, _sny, _panel) {
|
|
var mx = (_mx - _x) / _s;
|
|
var my = (_my - _y) / _s;
|
|
|
|
var smx = value_snap(mx, _snx);
|
|
var smy = value_snap(my, _sny);
|
|
|
|
var _b = bones;
|
|
|
|
if(builder_bone != noone) {
|
|
gpu_set_texfilter(true);
|
|
|
|
anchor_selecting = _b.draw(attributes, false, _x, _y, _s, _mx, _my, anchor_selecting, builder_bone);
|
|
_b.drawControl(attributes);
|
|
|
|
var dir = point_direction(builder_sx, builder_sy, smx, smy);
|
|
var dis = point_distance(builder_sx, builder_sy, smx, smy);
|
|
|
|
if(builder_type == 2) {
|
|
var an = point_direction(builder_sx, builder_sy, mx, my);
|
|
builder_bone.angle += angle_difference(an, builder_sv);
|
|
builder_sv = an;
|
|
|
|
var _rx = _x + _s * builder_sx;
|
|
var _ry = _y + _s * builder_sy;
|
|
draw_sprite_ext(s_bone_rotate, 0, _rx, _ry, 1, 1, builder_bone.angle, COLORS._main_value_positive, 1);
|
|
|
|
} else if(builder_type == 3) {
|
|
|
|
builder_bone.length = point_distance(builder_sx, builder_sy, mx, my) / builder_sv;
|
|
|
|
orig = builder_bone.getPoint(0.8);
|
|
var _rx = _x + _s * orig.x;
|
|
var _ry = _y + _s * orig.y;
|
|
draw_sprite_ext(s_bone_scale, 0, _rx, _ry, 1, 1, builder_bone.angle, COLORS._main_value_positive, 1);
|
|
|
|
} else if(key_mod_press(ALT)) {
|
|
if(builder_type == 0) {
|
|
var bo = builder_bone.getTail();
|
|
|
|
builder_bone.direction = dir;
|
|
builder_bone.distance = dis;
|
|
|
|
var bn = builder_bone.getHead();
|
|
|
|
builder_bone.angle = point_direction(bn.x, bn.y, bo.x, bo.y);
|
|
builder_bone.length = point_distance( bn.x, bn.y, bo.x, bo.y);
|
|
|
|
} else if(builder_type == 1) {
|
|
var chs = [];
|
|
for( var i = 0, n = array_length(builder_bone.childs); i < n; i++ ) {
|
|
var ch = builder_bone.childs[i];
|
|
chs[i] = ch.getTail();
|
|
}
|
|
|
|
builder_bone.angle = dir;
|
|
builder_bone.length = dis;
|
|
|
|
for( var i = 0, n = array_length(builder_bone.childs); i < n; i++ ) {
|
|
var ch = builder_bone.childs[i];
|
|
var c0 = ch.getHead();
|
|
|
|
ch.angle = point_direction(c0.x, c0.y, chs[i].x, chs[i].y);
|
|
ch.length = point_distance( c0.x, c0.y, chs[i].x, chs[i].y);
|
|
}
|
|
}
|
|
} else {
|
|
if(builder_type == 0) {
|
|
builder_bone.direction = dir;
|
|
builder_bone.distance = dis;
|
|
|
|
orig = builder_bone.getHead();
|
|
var _rx = _x + _s * orig.x;
|
|
var _ry = _y + _s * orig.y;
|
|
var cc = COLORS._main_value_positive;
|
|
|
|
if(isUsingTool("Detach bones") && builder_bone.parent) { // re-attach
|
|
var par_anc = builder_bone.parent.getTail();
|
|
var pardist = point_distance(orig.x, orig.y, par_anc.x, par_anc.y) * _s;
|
|
var inRange = pardist < 8 && !builder_bone.parent.is_main;
|
|
|
|
if(inRange) {
|
|
cc = COLORS._main_accent;
|
|
if(mouse_release(mb_left))
|
|
builder_bone.parent_anchor = true;
|
|
}
|
|
}
|
|
|
|
draw_sprite_ext(s_bone_move, 0, _rx, _ry, 1, 1, 0, cc, 1);
|
|
|
|
} else if(builder_type == 1) {
|
|
builder_bone.angle = dir;
|
|
builder_bone.length = dis;
|
|
|
|
orig = builder_bone.getTail();
|
|
var _rx = _x + _s * orig.x;
|
|
var _ry = _y + _s * orig.y;
|
|
draw_sprite_ext(s_bone_move, 0, _rx, _ry, 1, 1, 0, COLORS._main_value_positive, 1);
|
|
}
|
|
}
|
|
|
|
if(mouse_release(mb_left)) {
|
|
builder_bone = noone;
|
|
UNDO_HOLDING = false;
|
|
}
|
|
|
|
gpu_set_texfilter(false);
|
|
triggerRender();
|
|
|
|
} else if(ik_dragging != noone) {
|
|
anchor_selecting = _b.draw(attributes, active * 0b100, _x, _y, _s, _mx, _my, anchor_selecting, ik_dragging);
|
|
_b.drawControl(attributes);
|
|
|
|
var _bone = ik_dragging;
|
|
var p1 = ik_dragging.getTail();
|
|
var p1x = _x + p1.x * _s;
|
|
var p1y = _y + p1.y * _s;
|
|
|
|
if(anchor_selecting != noone && anchor_selecting[1] == 2) {
|
|
var anc = anchor_selecting[0];
|
|
var bne = anc;
|
|
var _reach = false;
|
|
var _blen = 0;
|
|
|
|
while(_bone != noone) {
|
|
_blen++;
|
|
_bone = _bone.parent;
|
|
|
|
if(_bone == anc.parent) {
|
|
_reach = true;
|
|
bne = anc;
|
|
anc = anc.parent;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(_reach) {
|
|
var p0 = anc.getHead();
|
|
var p0t = bne.getHead();
|
|
var _px = _x + p0t.x * _s;
|
|
var _py = _y + p0t.y * _s;
|
|
draw_line_dashed(_px, _py, p1x, p1y, 2, 8);
|
|
draw_sprite_ui(THEME.preview_bone_IK, 1, _px, _py, 1, 1, 0, COLORS._main_accent, 1);
|
|
|
|
if(mouse_release(mb_left)) {
|
|
var _len = point_distance(p0.x, p0.y, p1.x, p1.y);
|
|
var _ang = point_direction(p0.x, p0.y, p1.x, p1.y);
|
|
|
|
recordAction(ACTION_TYPE.struct_modify, bones, bones.serialize());
|
|
|
|
var IKbone = new __Bone(anc, _len, _ang, ik_dragging.angle + 90, 0, self);
|
|
anc.addChild(IKbone);
|
|
|
|
IKbone.direction = _ang;
|
|
IKbone.distance = _len;
|
|
IKbone.IKlength = _blen;
|
|
IKbone.IKTargetID = ik_dragging.ID;
|
|
|
|
IKbone.name = "IK handle";
|
|
IKbone.parent_anchor = false;
|
|
|
|
bones.setPosition();
|
|
}
|
|
}
|
|
|
|
draw_sprite_ui(THEME.preview_bone_IK, 0, p1x, p1y, 1, 1, 0, _reach? COLORS._main_value_positive : COLORS._main_accent, 1);
|
|
|
|
} else
|
|
draw_sprite_ui(THEME.preview_bone_IK, 0, p1x, p1y, 1, 1, 0, COLORS._main_accent, 0.5);
|
|
|
|
if(mouse_release(mb_left)) {
|
|
ik_dragging = noone;
|
|
UNDO_HOLDING = false;
|
|
bones.setPosition();
|
|
}
|
|
|
|
triggerRender();
|
|
|
|
}
|
|
|
|
if(isUsingTool("Transform")) {
|
|
_b.draw(attributes, false, _x, _y, _s, _mx, _my);
|
|
|
|
var bbox = _b.bbox();
|
|
if(!is_array(bbox)) return;
|
|
|
|
var x0 = _x + bbox[0] * _s;
|
|
var y0 = _y + bbox[1] * _s;
|
|
|
|
var x1 = _x + bbox[2] * _s;
|
|
var y1 = _y + bbox[3] * _s;
|
|
|
|
if(bone_transform_type >= 0) {
|
|
var dx = mx - builder_mx;
|
|
var dy = my - builder_my;
|
|
|
|
var ox0 = bone_transform_bbox[0], oy0 = bone_transform_bbox[1];
|
|
var ox1 = bone_transform_bbox[2], oy1 = bone_transform_bbox[3];
|
|
var ow = ox1 - ox0;
|
|
var oh = oy1 - oy0;
|
|
|
|
var nx0 = ox0, ny0 = oy0;
|
|
var nx1 = ox1, ny1 = oy1;
|
|
|
|
switch(bone_transform_type) { // transform
|
|
case 0 :
|
|
nx0 = ox0 + dx;
|
|
ny0 = oy0 + dy;
|
|
break;
|
|
|
|
case 1 :
|
|
nx1 = ox1 + dx;
|
|
ny0 = oy0 + dy;
|
|
break;
|
|
|
|
case 2 :
|
|
nx0 = ox0 + dx;
|
|
ny1 = oy1 + dy;
|
|
break;
|
|
|
|
case 3 :
|
|
nx1 = ox1 + dx;
|
|
ny1 = oy1 + dy;
|
|
break;
|
|
|
|
case 4 :
|
|
nx0 = ox0 + dx;
|
|
ny0 = oy0 + dy;
|
|
nx1 = ox1 + dx;
|
|
ny1 = oy1 + dy;
|
|
break;
|
|
}
|
|
|
|
var nw = nx1 - nx0;
|
|
var nh = ny1 - ny0;
|
|
|
|
for( var i = 0, n = array_length(bone_point_maps); i < n; i++ ) {
|
|
var _bmap = bone_point_maps[i];
|
|
var _emap = bone_point_mape[i];
|
|
|
|
_emap[0].x = (_bmap[0].x - ox0) / ow * nw + nx0;
|
|
_emap[0].y = (_bmap[0].y - oy0) / oh * nh + ny0;
|
|
|
|
_emap[1].x = (_bmap[1].x - ox0) / ow * nw + nx0;
|
|
_emap[1].y = (_bmap[1].y - oy0) / oh * nh + ny0;
|
|
}
|
|
|
|
for( var i = 0, n = array_length(bone_arr); i < n; i++ ) {
|
|
var _bone = bone_arr[i];
|
|
var _emap = bone_point_mape[i];
|
|
|
|
var _h = _emap[0];
|
|
var _t = _emap[1];
|
|
|
|
if(!_bone.parent_anchor) {
|
|
var _ox = 0;
|
|
var _oy = 0;
|
|
|
|
if(_bone.parent) {
|
|
var _ph = _bone.parent.getHead();
|
|
_ox = _ph.x;
|
|
_oy = _ph.y;
|
|
}
|
|
|
|
_bone.distance = point_distance( _ox, _oy, _h.x, _h.y );
|
|
_bone.direction = point_direction( _ox, _oy, _h.x, _h.y );
|
|
}
|
|
|
|
_bone.length = point_distance(_h.x, _h.y, _t.x, _t.y);
|
|
_bone.angle = point_direction(_h.x, _h.y, _t.x, _t.y);
|
|
|
|
_bone.__setPosition();
|
|
}
|
|
|
|
if(mouse_release(mb_left)) {
|
|
bone_transform_type = -1;
|
|
UNDO_HOLDING = false;
|
|
}
|
|
|
|
triggerRender();
|
|
|
|
draw_set_color(COLORS._main_accent);
|
|
draw_rectangle(x0, y0, x1, y1, true);
|
|
|
|
} else {
|
|
var _hov_type = -1;
|
|
var _r = 8;
|
|
|
|
if(point_in_circle(_mx, _my, x0, y0, _r + 2)) _hov_type = 0;
|
|
else if(point_in_circle(_mx, _my, x1, y0, _r + 2)) _hov_type = 1;
|
|
else if(point_in_circle(_mx, _my, x0, y1, _r + 2)) _hov_type = 2;
|
|
else if(point_in_circle(_mx, _my, x1, y1, _r + 2)) _hov_type = 3;
|
|
else if(point_in_rectangle(_mx, _my, x0, y0, x1, y1)) _hov_type = 4;
|
|
|
|
draw_set_color(_hov_type == 4? COLORS._main_accent : c_white);
|
|
draw_set_alpha(0.5);
|
|
draw_rectangle(x0, y0, x1, y1, true);
|
|
draw_set_alpha(1);
|
|
|
|
draw_anchor(_hov_type == 0, x0, y0, _r);
|
|
draw_anchor(_hov_type == 1, x1, y0, _r);
|
|
draw_anchor(_hov_type == 2, x0, y1, _r);
|
|
draw_anchor(_hov_type == 3, x1, y1, _r);
|
|
|
|
if(_hov_type >= -1 && mouse_press(mb_left, active)) {
|
|
bone_arr = _b.toArray();
|
|
for( var i = 0, n = array_length(bone_arr); i < n; i++ ) {
|
|
var _h = bone_arr[i].getHead();
|
|
var _t = bone_arr[i].getTail();
|
|
|
|
bone_point_maps[i] = [ _h, _t ];
|
|
bone_point_mape[i] = [ _h.clone(), _t.clone() ];
|
|
}
|
|
|
|
bone_transform_type = _hov_type;
|
|
bone_transform_bbox = _b.bbox();
|
|
builder_mx = mx;
|
|
builder_my = my;
|
|
|
|
recordAction(ACTION_TYPE.struct_modify, bones, bones.serialize());
|
|
}
|
|
}
|
|
|
|
} else if(isUsingTool("Add bones")) { // builder
|
|
if(builder_bone == noone) {
|
|
anchor_selecting = _b.draw(attributes, active * 0b111, _x, _y, _s, _mx, _my, anchor_selecting);
|
|
_b.drawControl(attributes);
|
|
}
|
|
|
|
if(mouse_press(mb_left, active)) {
|
|
|
|
if(anchor_selecting == noone) {
|
|
builder_bone = createBone(bones, point_distance(0, 0, smx, smy), point_direction(0, 0, smx, smy));
|
|
builder_type = 1;
|
|
|
|
builder_sx = smx;
|
|
builder_sy = smy;
|
|
UNDO_HOLDING = true;
|
|
bones.setPosition();
|
|
|
|
} else if(anchor_selecting[1] == 1) {
|
|
builder_bone = createBone(anchor_selecting[0], 0, 0);
|
|
builder_type = 1;
|
|
|
|
builder_sx = smx;
|
|
builder_sy = smy;
|
|
UNDO_HOLDING = true;
|
|
bones.setPosition();
|
|
|
|
} else if(anchor_selecting[1] == 2) {
|
|
var _pr = anchor_selecting[0];
|
|
recordAction(ACTION_TYPE.struct_modify, bones, bones.serialize());
|
|
|
|
var _md = new __Bone(noone, 0, 0, _pr.angle, _pr.length / 2, self);
|
|
_pr.length = _md.length;
|
|
|
|
for( var i = 0, n = array_length(_pr.childs); i < n; i++ )
|
|
_md.addChild(_pr.childs[i]);
|
|
|
|
_pr.childs = [];
|
|
_pr.addChild(_md);
|
|
|
|
UNDO_HOLDING = true;
|
|
triggerRender();
|
|
}
|
|
}
|
|
|
|
if(anchor_selecting == noone)
|
|
draw_sprite_ext(THEME.bone_tool_add, 1, _mx + 16, _my + 16, 1, 1, 0, c_white, 1);
|
|
|
|
else if(anchor_selecting[1] == 1) {
|
|
draw_sprite_ext(THEME.bone_tool_add, 0, _mx + 24, _my + 24, 1, 1, 0, c_white, 1);
|
|
draw_sprite_ext(THEME.bone_tool_add, 1, _mx + 24, _my + 24, 1, 1, 0, c_white, 1);
|
|
|
|
} else if(anchor_selecting[1] == 2)
|
|
draw_sprite_ext(THEME.bone_tool_add, 0, _mx + 24, _my + 24, 1, 1, 0, c_white, 1);
|
|
|
|
} else if(isUsingTool("Remove bones")) { //remover
|
|
anchor_selecting = _b.draw(attributes, active * 0b100, _x, _y, _s, _mx, _my, anchor_selecting);
|
|
|
|
if(anchor_selecting != noone && anchor_selecting[1] == 2 && anchor_selecting[0].parent != noone && mouse_press(mb_left, active)) {
|
|
var _bone = anchor_selecting[0];
|
|
var _par = _bone.parent;
|
|
|
|
array_remove(_par.childs, _bone);
|
|
|
|
for( var i = 0, n = array_length(_bone.childs); i < n; i++ ) {
|
|
var _ch = _bone.childs[i];
|
|
_par.addChild(_ch);
|
|
|
|
_ch.parent_anchor = _bone.parent_anchor;
|
|
}
|
|
|
|
triggerRender();
|
|
}
|
|
|
|
if(anchor_selecting != noone)
|
|
draw_sprite_ext(THEME.bone_tool_remove, 1, _mx + 24, _my + 24, 1, 1, 0, c_white, 1);
|
|
|
|
} else if(isUsingTool("Detach bones")) { //detach
|
|
if(builder_bone == noone)
|
|
anchor_selecting = _b.draw(attributes, active * 0b100, _x, _y, _s, _mx, _my, anchor_selecting);
|
|
|
|
if(anchor_selecting != noone && anchor_selecting[1] == 2 && mouse_press(mb_left, active)) {
|
|
var detach_bone = anchor_selecting[0];
|
|
|
|
var par = detach_bone.parent;
|
|
if(detach_bone.parent_anchor) {
|
|
detach_bone.distance = par.length;
|
|
detach_bone.direction = par.angle;
|
|
}
|
|
detach_bone.parent_anchor = false;
|
|
|
|
builder_bone = detach_bone;
|
|
builder_type = 0;
|
|
|
|
var head = par.getHead();
|
|
builder_sx = head.x;
|
|
builder_sy = head.y;
|
|
builder_mx = mx;
|
|
builder_my = my;
|
|
UNDO_HOLDING = true;
|
|
}
|
|
|
|
} else if(isUsingTool("IK")) { //IK
|
|
if(ik_dragging == noone)
|
|
anchor_selecting = _b.draw(attributes, active * 0b100, _x, _y, _s, _mx, _my, anchor_selecting);
|
|
|
|
if(anchor_selecting != noone && anchor_selecting[1] == 2 && mouse_press(mb_left, active))
|
|
ik_dragging = anchor_selecting[0];
|
|
|
|
} else if(isUsingTool("Mirror bones")) {
|
|
if(builder_bone == noone)
|
|
anchor_selecting = _b.draw(attributes, active * 0b111, _x, _y, _s, _mx, _my, anchor_selecting);
|
|
|
|
if(mirroring_bone != noone) {
|
|
var _maxis = anchor_selecting == noone || anchor_selecting[0] == mirroring_bone? mirroring_bone.parent.angle : anchor_selecting[0].angle;
|
|
var _ori = mirroring_bone.getHead();
|
|
|
|
var _ox = _x + _ori.x * _s;
|
|
var _oy = _y + _ori.y * _s;
|
|
|
|
var _x0 = _ox + lengthdir_x(64, _maxis);
|
|
var _y0 = _oy + lengthdir_y(64, _maxis);
|
|
var _x1 = _ox - lengthdir_x(64, _maxis);
|
|
var _y1 = _oy - lengthdir_y(64, _maxis);
|
|
|
|
BLEND_ADD
|
|
draw_set_color(COLORS._main_accent);
|
|
draw_line_dashed(_x0, _y0, _x1, _y1, 3);
|
|
BLEND_NORMAL
|
|
|
|
if(mouse_release(mb_left)) {
|
|
var _mirrored = mirroring_bone.clone();
|
|
var _mirArr = _mirrored.toArray();
|
|
|
|
for( var i = 0, n = array_length(_mirArr); i < n; i++ ) {
|
|
var _bm = _mirArr[i];
|
|
_bm.ID = UUID_generate();
|
|
|
|
if(string_pos(" L", _bm.name)) _bm.name = string_replace(_bm.name, " L", " R");
|
|
else if(string_pos(" R", _bm.name)) _bm.name = string_replace(_bm.name, " R", " L");
|
|
|
|
_bm.angle = _maxis + angle_difference(_maxis, _bm.angle);
|
|
}
|
|
|
|
mirroring_bone.parent.addChild(_mirrored);
|
|
mirroring_bone = noone;
|
|
|
|
triggerRender();
|
|
}
|
|
|
|
} else if(anchor_selecting != noone) {
|
|
var _bne = anchor_selecting[0];
|
|
var _typ = anchor_selecting[1];
|
|
|
|
if(_bne.parent && mouse_press(mb_left, active))
|
|
mirroring_bone = _bne;
|
|
}
|
|
|
|
} else { // move tools
|
|
if(builder_bone == noone) {
|
|
anchor_selecting = _b.draw(attributes, active * 0b111, _x, _y, _s, _mx, _my, anchor_selecting);
|
|
_b.drawControl(attributes);
|
|
}
|
|
|
|
if(anchor_selecting != noone) {
|
|
var _bne = anchor_selecting[0];
|
|
var _typ = anchor_selecting[1];
|
|
|
|
if(_bne.IKlength) _typ = 0;
|
|
|
|
gpu_set_texfilter(true);
|
|
|
|
if(_typ == 0) { // free move
|
|
var orig = _bne.getHead();
|
|
draw_sprite_ext(s_bone_move, 0, _x + _s * orig.x, _y + _s * orig.y, 1, 1, 0, COLORS._main_accent, 1);
|
|
|
|
} else if(_typ == 1) { // bone move
|
|
var orig = _bne.getTail();
|
|
draw_sprite_ext(s_bone_move, 0, _x + _s * orig.x, _y + _s * orig.y, 1, 1, 0, COLORS._main_accent, 1);
|
|
|
|
} else if(_typ == 2) { // bone rotate
|
|
var orig = _bne.getHead();
|
|
var _rx = _x + _s * orig.x;
|
|
var _ry = _y + _s * orig.y;
|
|
|
|
var orig = _bne.getPoint(0.8);
|
|
var _sx = _x + _s * orig.x;
|
|
var _sy = _y + _s * orig.y;
|
|
|
|
if(point_in_circle(_mx, _my, _sx, _sy, 12)) {
|
|
draw_sprite_ext(s_bone_scale, 0, _sx, _sy, 1, 1, _bne.angle, COLORS._main_accent, 1);
|
|
draw_sprite_ext(s_bone_rotate, 0, _rx, _ry, 1, 1, _bne.angle, c_white, 1);
|
|
_typ = 3;
|
|
|
|
} else {
|
|
draw_sprite_ext(s_bone_scale, 0, _sx, _sy, 1, 1, _bne.angle, c_white, 1);
|
|
draw_sprite_ext(s_bone_rotate, 0, _rx, _ry, 1, 1, _bne.angle, COLORS._main_accent, 1);
|
|
_typ = 2;
|
|
}
|
|
}
|
|
|
|
gpu_set_texfilter(false);
|
|
|
|
if(mouse_press(mb_left, active)) {
|
|
builder_bone = _bne;
|
|
builder_type = _typ;
|
|
|
|
recordAction(ACTION_TYPE.struct_modify, bones, bones.serialize());
|
|
|
|
if(_typ == 0) {
|
|
var orig = _bne.parent.getHead();
|
|
builder_sx = orig.x;
|
|
builder_sy = orig.y;
|
|
|
|
} else if(_typ == 1) {
|
|
var orig = _bne.getHead();
|
|
builder_sx = orig.x;
|
|
builder_sy = orig.y;
|
|
|
|
} else if(_typ == 2) {
|
|
var orig = _bne.getHead();
|
|
builder_sv = point_direction(orig.x, orig.y, mx, my);
|
|
builder_sx = orig.x;
|
|
builder_sy = orig.y;
|
|
builder_mx = mx;
|
|
builder_my = my;
|
|
|
|
} else if(_typ == 3) {
|
|
var orig = _bne.getHead();
|
|
builder_sv = point_distance(orig.x, orig.y, mx, my) / _bne.length;
|
|
builder_sx = orig.x;
|
|
builder_sy = orig.y;
|
|
}
|
|
|
|
UNDO_HOLDING = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static step = function() {}
|
|
|
|
static update = function(frame = CURRENT_FRAME) {
|
|
array_foreach(bones.constrains, function(c) /*=>*/ { c.bone = bones; c.init(); });
|
|
|
|
bones.resetPose()
|
|
.setPosition();
|
|
bone_bbox = bones.bbox();
|
|
|
|
outputs[0].setValue(bones);
|
|
bone_array = bones.toArray();
|
|
}
|
|
|
|
////- Draw
|
|
|
|
static getPreviewBoundingBox = function() {
|
|
var minx = 9999999;
|
|
var miny = 9999999;
|
|
var maxx = -9999999;
|
|
var maxy = -9999999;
|
|
|
|
var _b = bones;
|
|
var _bst = ds_stack_create();
|
|
ds_stack_push(_bst, _b);
|
|
|
|
while(!ds_stack_empty(_bst)) {
|
|
var __b = ds_stack_pop(_bst);
|
|
|
|
for( var i = 0, n = array_length(__b.childs); i < n; i++ ) {
|
|
var p0 = __b.childs[i].getHead();
|
|
var p1 = __b.childs[i].getTail();
|
|
|
|
minx = min(minx, p0.x); miny = min(miny, p0.y);
|
|
maxx = max(maxx, p0.x); maxy = max(maxy, p0.y);
|
|
|
|
minx = min(minx, p1.x); miny = min(miny, p1.y);
|
|
maxx = max(maxx, p1.x); maxy = max(maxy, p1.y);
|
|
|
|
ds_stack_push(_bst, __b.childs[i]);
|
|
}
|
|
}
|
|
|
|
ds_stack_destroy(_bst);
|
|
|
|
if(minx == 9999999) return noone;
|
|
if(abs(maxx - minx) < 1) maxx = minx + 1;
|
|
if(abs(maxy - miny) < 1) maxy = miny + 1;
|
|
|
|
return BBOX().fromPoints(minx, miny, maxx, maxy);
|
|
}
|
|
|
|
static onDrawNode = function(xx, yy, _mx, _my, _s, _hover, _focus) {
|
|
var bbox = drawGetBbox(xx, yy, _s);
|
|
|
|
var _ss = _s * .5;
|
|
gpu_set_tex_filter(1);
|
|
draw_sprite_ext(s_node_armature_create, 0, bbox.x0 + 24 * _ss, bbox.y1 - 24 * _ss, _ss, _ss, 0, c_white, 0.5);
|
|
gpu_set_tex_filter(0);
|
|
|
|
bones.drawThumbnail(_s, bbox, bone_bbox);
|
|
}
|
|
|
|
////- Serialize
|
|
|
|
static doSerialize = function(_map) {
|
|
_map.bones = bones.serialize();
|
|
}
|
|
|
|
static postDeserialize = function() {
|
|
if(struct_has(attributes, "bones")) struct_remove(attributes, "bones");
|
|
if(!struct_has(load_map, "bones")) return;
|
|
|
|
bones = new __Bone(,,,,, self);
|
|
bones.deserialize(load_map.bones, self);
|
|
bones.connect();
|
|
|
|
bone_array = bones.toArray();
|
|
}
|
|
|
|
////- Actions
|
|
|
|
static boneSelector = function(fn) {
|
|
__bone_fn = fn;
|
|
menuCall("", array_map(bone_array, function(b) /*=>*/ {return new MenuItem(b.name, __bone_fn, [ THEME.bone, 1, 1 ]).setParam({ bone: b })}) );
|
|
}
|
|
}
|
|
|