Pixel-Composer/scripts/node_armature_pose/node_armature_pose.gml

466 lines
13 KiB
Text
Raw Normal View History

function Node_Armature_Pose(_x, _y, _group = noone) : Node(_x, _y, _group) constructor {
name = "Armature Pose";
2024-11-29 06:18:28 +01:00
setDimension(96, 96);
draw_padding = 8;
2023-06-25 20:12:17 +02:00
2024-08-18 09:13:41 +02:00
newInput(0, nodeValue_Armature("Armature", self, noone))
2023-06-23 15:39:24 +02:00
.setVisible(true, true);
input_display_list = [ 0 ]
2023-06-21 20:36:53 +02:00
2024-09-04 03:57:11 +02:00
newOutput(0, nodeValue_Output("Armature", self, VALUE_TYPE.armature, noone));
2024-11-29 06:18:28 +01:00
boneHash = "";
bonePose = noone;
bone_bbox = undefined;
boneArray = [];
boneMap = {};
2023-06-21 20:36:53 +02:00
attributes.display_name = true;
2023-07-05 15:09:52 +02:00
attributes.display_bone = 0;
array_push(attributeEditors, "Display");
2024-11-26 11:28:04 +01:00
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; })]);
2023-06-21 20:36:53 +02:00
2024-05-23 10:59:39 +02:00
static createNewInput = function(bone = noone) {
2024-08-08 06:57:51 +02:00
var index = array_length(inputs);
2023-06-21 20:36:53 +02:00
2024-08-20 10:15:53 +02:00
newInput(index, nodeValue(bone != noone? bone.name : "bone", self, CONNECT_TYPE.input, VALUE_TYPE.float, [ 0, 0, 0, 1 ] ))
2023-06-21 20:36:53 +02:00
.setDisplay(VALUE_DISPLAY.transform);
2024-11-23 12:08:44 +01:00
inputs[index].attributes.bone_id = bone != noone? bone.ID : noone;
2023-06-21 20:36:53 +02:00
2024-11-12 01:41:42 +01:00
if(bone != noone) boneMap[$ bone.ID] = inputs[index];
2023-06-21 20:36:53 +02:00
array_push(input_display_list, index);
2023-06-23 15:39:24 +02:00
2024-08-08 06:57:51 +02:00
return inputs[index];
2024-11-26 11:28:04 +01:00
}
setDynamicInput(1, false);
2023-06-21 20:36:53 +02:00
2024-08-06 14:04:41 +02:00
static setBone = function() {
2024-11-27 05:32:43 +01:00
// print("Setting dem bones...");
var _b = getInputData(0);
2023-06-21 20:36:53 +02:00
if(_b == noone) return;
2024-11-27 05:32:43 +01:00
bonePose = _b.clone().connect();
2024-11-29 06:18:28 +01:00
boneArray = bonePose.toArray();
2024-08-08 06:57:51 +02:00
var _inputs = [ inputs[0] ];
2024-11-29 06:18:28 +01:00
var _input_display_list = array_clone(input_display_list_raw, 1);
2023-06-21 20:36:53 +02:00
2024-11-29 06:18:28 +01:00
for( var i = 0, n = array_length(boneArray); i < n; i++ ) {
var bone = boneArray[i];
2024-08-08 06:57:51 +02:00
var _idx = array_length(_inputs);
2024-11-26 11:28:04 +01:00
var _inp;
2023-06-23 15:39:24 +02:00
array_push(_input_display_list, _idx);
2024-11-12 01:41:42 +01:00
if(struct_exists(boneMap, bone.ID)) {
2024-11-26 11:28:04 +01:00
_inp = boneMap[$ bone.ID];
2023-06-21 20:36:53 +02:00
_inp.index = _idx;
2024-11-26 11:28:04 +01:00
} else
_inp = createNewInput(bone);
array_push(_inputs, _inp);
2023-06-21 20:36:53 +02:00
}
inputs = _inputs;
input_display_list = _input_display_list;
2024-08-06 14:04:41 +02:00
}
2023-06-21 20:36:53 +02:00
tools = [];
2023-06-21 20:36:53 +02:00
anchor_selecting = noone;
posing_bone = noone;
posing_input = 0;
posing_type = 0;
pose_child_lock = false;
2024-11-27 05:32:43 +01:00
posing_sx = 0;
posing_sy = 0;
posing_mx = 0;
posing_my = 0;
////- Preview
2023-06-21 20:36:53 +02:00
2024-08-06 14:04:41 +02:00
static drawOverlay = function(hover, active, _x, _y, _s, _mx, _my, _snx, _sny) {
2024-11-27 05:32:43 +01:00
var _b = inputs[0].getValue();
2024-11-29 06:18:28 +01:00
if(_b == noone || bonePose == noone) return;
var _hov = noone;
2024-11-29 06:18:28 +01:00
var _bhov = anchor_selecting;
2023-06-21 20:36:53 +02:00
2024-11-29 06:18:28 +01:00
for( var i = 0, n = array_length(boneArray); i < n; i++ ) {
var _bne = boneArray[i];
var _sel = false;
var cc = c_white;
2024-11-29 06:18:28 +01:00
if(struct_has(boneMap, _bne.ID)) {
var _inp = boneMap[$ _bne.ID];
_sel = _inp.value_from == noone || is(_inp.value_from.node, Node_Vector4);
if(_bhov == noone && PANEL_INSPECTOR.prop_hover == _inp)
_bhov = [ _bne, 2 ];
}
var hh = _bne.drawBone(attributes, _sel * active * 0b111, _x, _y, _s, _mx, _my, _bhov, posing_bone, cc, .5 + .5 * _sel);
2024-11-29 06:18:28 +01:00
if(hh != noone && (_hov == noone || _bne.IKlength)) _hov = hh;
}
anchor_selecting = _hov;
bonePose.setControl(_x, _y, _s);
2024-11-27 05:32:43 +01:00
bonePose.drawControl(attributes);
2024-11-29 06:18:28 +01:00
if(anchor_selecting != noone) {
var _bne = anchor_selecting[0];
if(struct_has(boneMap, _bne.ID)) {
var _inp = boneMap[$ _bne.ID];
_inp.editWidget.temp_hovering = true;
}
}
2023-06-21 20:36:53 +02:00
var mx = (_mx - _x) / _s;
var my = (_my - _y) / _s;
2023-08-01 19:21:51 +02:00
var smx = value_snap(mx, _snx);
var smy = value_snap(my, _sny);
2024-08-06 14:04:41 +02:00
if(posing_bone) {
2024-11-27 05:32:43 +01:00
gpu_set_texfilter(true);
var val = array_clone(posing_input.getValue());
if(posing_type == 0) { //move
var ang = posing_bone.pose_local_rotate;
2023-08-01 19:21:51 +02:00
var pp = point_rotate(smx - posing_mx, smy - posing_my, 0, 0, -ang);
var bx = posing_sx + pp[0];
var by = posing_sy + pp[1];
2023-06-21 20:36:53 +02:00
val[TRANSFORM.pos_x] = bx;
val[TRANSFORM.pos_y] = by;
2024-11-27 05:32:43 +01:00
orig = posing_bone.getHead();
var _rx = _x + _s * orig.x;
var _ry = _y + _s * orig.y;
draw_sprite_ext(THEME.bone_move, 0, _rx, _ry, 1, 1, 0, COLORS._main_value_positive, 1);
2023-06-21 20:36:53 +02:00
2024-11-27 05:32:43 +01:00
} else if(posing_type == 1) { //free move
var _direction = point_direction(posing_sx, posing_sy, smx, smy);
var _distance = point_distance(posing_sx, posing_sy, smx, smy);
2023-06-21 20:36:53 +02:00
var _orot = val[TRANSFORM.rot];
var _osca = val[TRANSFORM.sca_x];
var _nrot = _direction - (posing_bone.angle + posing_bone.pose_local_rotate);
var _nsca = _distance / (posing_bone.length * posing_bone.pose_local_scale);
val[TRANSFORM.rot] = _nrot;
val[TRANSFORM.sca_x] = _nsca;
if(pose_child_lock) {
var _dr = angle_difference(_nrot, _orot);
var _ds = _nsca / _osca;
for( var i = 0, n = array_length(posing_bone.childs); i < n; i++ ) {
var _child = posing_bone.childs[i];
var _input = boneMap[$ _child.ID];
var _val = array_clone(_input.getValue());
if(_child.apply_rotation) _val[TRANSFORM.rot] -= _dr;
if(_child.apply_scale) _val[TRANSFORM.sca_x] /= _ds;
_input.setValue(_val);
}
}
2023-06-21 20:36:53 +02:00
2024-11-27 05:32:43 +01:00
orig = posing_bone.getTail();
var _rx = _x + _s * orig.x;
var _ry = _y + _s * orig.y;
draw_sprite_ext(THEME.bone_move, 0, _rx, _ry, 1, 1, 0, COLORS._main_value_positive, 1);
2024-11-27 05:32:43 +01:00
2023-06-23 15:39:24 +02:00
} else if(posing_type == 2) { //rotate
var ori = posing_bone.getHead();
2023-06-23 15:39:24 +02:00
var ang = point_direction(ori.x, ori.y, mx, my);
var rot = angle_difference(ang, posing_sy);
posing_sy = ang;
2023-06-23 15:39:24 +02:00
posing_sx += rot;
2023-06-21 20:36:53 +02:00
2023-06-23 15:39:24 +02:00
val[TRANSFORM.rot] = posing_sx;
2023-08-08 18:45:00 +02:00
if(pose_child_lock && rot != 0)
for( var i = 0, n = array_length(posing_bone.childs); i < n; i++ ) {
var _child = posing_bone.childs[i];
if(!_child.apply_rotation) continue;
var _input = boneMap[$ _child.ID];
var _val = array_clone(_input.getValue());
_val[TRANSFORM.rot] -= rot;
_input.setValue(_val);
}
2024-11-27 05:32:43 +01:00
orig = posing_bone.getHead();
var _rx = _x + _s * orig.x;
var _ry = _y + _s * orig.y;
draw_sprite_ext(THEME.bone_rotate, 0, _rx, _ry, 1, 1, posing_bone.pose_angle, COLORS._main_value_positive, 1);
2024-11-27 05:32:43 +01:00
} else if(posing_type == 3) { //scale
var ps = val[TRANSFORM.sca_x];
var ss = point_distance(posing_mx, posing_my, smx, smy) / posing_sx;
var ds = ss / ps;
2024-11-27 05:32:43 +01:00
val[TRANSFORM.sca_x] = ss;
if(pose_child_lock && ds != 1 && ds != 0)
for( var i = 0, n = array_length(posing_bone.childs); i < n; i++ ) {
var _child = posing_bone.childs[i];
if(!_child.apply_scale) continue;
var _input = boneMap[$ _child.ID];
var _val = array_clone(_input.getValue());
_val[TRANSFORM.sca_x] /= ds;
_input.setValue(_val);
}
2024-11-27 05:32:43 +01:00
orig = posing_bone.getPoint(0.8);
var _rx = _x + _s * orig.x;
var _ry = _y + _s * orig.y;
draw_sprite_ext(THEME.bone_scale, 0, _rx, _ry, 1, 1, posing_bone.pose_angle, COLORS._main_value_positive, 1);
2024-11-27 05:32:43 +01:00
}
gpu_set_texfilter(false);
2024-11-27 09:03:12 +01:00
if(posing_input.value_from == noone) {
if(posing_input.setValue(val)) UNDO_HOLDING = true;
} else if(is(posing_input.value_from.node, Node_Vector4)) {
var _nod = posing_input.value_from.node;
for( var i = 0; i < 4; i++ )
if(_nod.inputs[i].setValue(val[i])) UNDO_HOLDING = true;
}
2023-06-21 20:36:53 +02:00
if(mouse_release(mb_left)) {
posing_bone = noone;
2023-06-23 15:39:24 +02:00
posing_type = noone;
2023-06-21 20:36:53 +02:00
UNDO_HOLDING = false;
}
2024-11-27 05:32:43 +01:00
} else if(anchor_selecting != noone) {
var _bne = anchor_selecting[0];
var _typ = anchor_selecting[1];
var _lck = key_mod_press(ALT);
2024-11-27 05:32:43 +01:00
if(_bne.IKlength) _typ = 0;
gpu_set_texfilter(true);
if(_typ == 0) { // free move
var orig = _bne.getHead();
draw_sprite_ext(THEME.bone_move, 0, _x + _s * orig.x, _y + _s * orig.y, 1, 1, 0, COLORS._main_accent, 1);
2024-11-27 05:32:43 +01:00
} else if(_typ == 1) { // bone move
var orig = _bne.getTail();
draw_sprite_ext(THEME.bone_move, 0, _x + _s * orig.x, _y + _s * orig.y, 1, 1, 0, COLORS._main_accent, 1);
2023-06-21 20:36:53 +02:00
2024-11-27 05:32:43 +01:00
} else if(_typ == 2) { // bone rotate
var orig = _bne.getHead();
var _rx = _x + _s * orig.x;
var _ry = _y + _s * orig.y;
2023-06-21 20:36:53 +02:00
2024-11-27 05:32:43 +01:00
var orig = _bne.getPoint(0.8);
var _sx = _x + _s * orig.x;
var _sy = _y + _s * orig.y;
2023-06-21 20:36:53 +02:00
2024-11-27 05:32:43 +01:00
if(point_in_circle(_mx, _my, _sx, _sy, 12)) {
draw_sprite_ext(THEME.bone_scale, 0, _sx, _sy, 1, 1, _bne.pose_angle, COLORS._main_accent, 1);
draw_sprite_ext(THEME.bone_rotate, 0, _rx, _ry, 1, 1, _bne.pose_angle, c_white, 1);
2024-11-27 05:32:43 +01:00
_typ = 3;
} else {
draw_sprite_ext(THEME.bone_scale, 0, _sx, _sy, 1, 1, _bne.pose_angle, c_white, 1);
draw_sprite_ext(THEME.bone_rotate, 0, _rx, _ry, 1, 1, _bne.pose_angle, COLORS._main_accent, 1);
2024-11-27 05:32:43 +01:00
_typ = 2;
}
}
if(_lck) {
for( var i = 0, n = array_length(_bne.childs); i < n; i++ ) {
var _ch = _bne.childs[i];
var _bc = _ch.getPoint(0.5);
var _cx = _x + _s * _bc.x;
var _cy = _y + _s * _bc.y;
BLEND_SUBTRACT
draw_set_color(c_white);
draw_circle(_cx, _cy, 16, false);
BLEND_NORMAL
draw_sprite_ext(THEME.lock, 0, _cx, _cy, 1, 1, 0, COLORS._main_accent, 1);
}
}
2024-11-27 05:32:43 +01:00
gpu_set_texfilter(false);
2024-11-27 05:32:43 +01:00
if(mouse_press(mb_left, active)) {
posing_bone = _bne;
posing_type = _typ;
pose_child_lock = _lck;
2023-06-21 20:36:53 +02:00
2024-11-27 05:32:43 +01:00
if(!struct_exists(boneMap, _bne.ID)) setBone();
posing_input = boneMap[$ _bne.ID];
2023-06-21 20:36:53 +02:00
2024-11-27 05:32:43 +01:00
if(_typ == 0) { // move
var val = posing_input.getValue();
var _p = anchor_selecting[2];
posing_sx = val[TRANSFORM.pos_x];
posing_sy = val[TRANSFORM.pos_y];
posing_mx = _p.x;
posing_my = _p.y;
} else if(_typ == 1) { // free move
var orig = _bne.getHead();
posing_sx = orig.x;
posing_sy = orig.y;
} else if(_typ == 2) { // rotate
var orig = _bne.getHead();
var val = posing_input.getValue();
posing_sx = val[TRANSFORM.rot];
posing_sy = point_direction(orig.x, orig.y, mx, my);
posing_mx = mx;
posing_my = my;
} else if(_typ == 3) { // scale
var orig = _bne.getHead();
var _sca = point_distance(orig.x, orig.y, mx, my) / _bne.pose_length;
posing_sx = _bne.length * _bne.pose_local_scale * _sca;
posing_mx = orig.x;
posing_my = orig.y;
}
2023-06-21 20:36:53 +02:00
}
2024-08-06 14:04:41 +02:00
}
}
2023-06-21 20:36:53 +02:00
////- Update
2024-11-27 05:32:43 +01:00
static update = function(frame = CURRENT_FRAME) {
var _b = getInputData(0);
if(_b == noone) { boneHash = ""; return; }
2023-06-21 20:36:53 +02:00
2024-11-27 05:32:43 +01:00
var _h = _b.getHash();
if(boneHash != _h) { boneHash = _h; setBone(); }
2023-06-21 20:36:53 +02:00
2024-11-27 05:32:43 +01:00
bonePose.resetPose()
.setPosition();
2024-12-13 13:23:06 +01:00
bonePose.constrains = _b.constrains;
2024-11-27 05:32:43 +01:00
var _bArr = _b.toArray();
var bArr = bonePose.toArray();
2023-06-21 20:36:53 +02:00
2024-11-27 05:32:43 +01:00
for( var i = 0, n = array_length(bArr); i < n; i++ ) {
var _bone = _bArr[i];
var bone = bArr[i];
var _id = bone.ID;
2023-06-21 20:36:53 +02:00
2024-11-27 05:32:43 +01:00
if(!struct_exists(boneMap, _id)) continue;
var _inp = boneMap[$ _id];
_inp.updateName(bone.name);
2023-06-21 20:36:53 +02:00
2024-11-27 05:32:43 +01:00
var _trn = _inp.getValue();
bone.angle = _bone.angle;
bone.length = _bone.length;
bone.direction = _bone.direction;
bone.distance = _bone.distance;
bone.pose_posit = [ _trn[TRANSFORM.pos_x], _trn[TRANSFORM.pos_y] ];
bone.pose_rotate = _trn[TRANSFORM.rot];
bone.pose_scale = _trn[TRANSFORM.sca_x];
2023-06-21 20:36:53 +02:00
}
2024-11-27 05:32:43 +01:00
bonePose.setPose();
2024-11-29 06:18:28 +01:00
bone_bbox = bonePose.bbox();
2023-06-21 20:36:53 +02:00
2024-11-27 05:32:43 +01:00
outputs[0].setValue(bonePose);
2024-08-06 14:04:41 +02:00
}
////- Draw
2024-08-06 14:04:41 +02:00
static getPreviewBoundingBox = function() {
2023-07-10 20:14:10 +02:00
var minx = 9999999;
var miny = 9999999;
var maxx = -9999999;
var maxy = -9999999;
2024-08-08 06:57:51 +02:00
var _b = outputs[0].getValue();
2023-08-01 19:21:51 +02:00
if(_b == noone) return BBOX().fromPoints(0, 0, 1, 1);
2023-07-10 20:14:10 +02:00
var _bst = ds_stack_create();
ds_stack_push(_bst, _b);
while(!ds_stack_empty(_bst)) {
var __b = ds_stack_pop(_bst);
2023-07-25 20:12:40 +02:00
for( var i = 0, n = array_length(__b.childs); i < n; i++ ) {
var p0 = __b.childs[i].getHead();
var p1 = __b.childs[i].getTail();
2023-07-10 20:14:10 +02:00
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;
2023-07-10 20:14:10 +02:00
return BBOX().fromPoints(minx, miny, maxx, maxy);
2024-08-06 14:04:41 +02:00
}
2023-07-10 20:14:10 +02:00
static postApplyDeserialize = function() {
2024-08-08 06:57:51 +02:00
for( var i = input_fix_len; i < array_length(inputs); i += data_length ) {
var inp = inputs[i];
2024-11-23 12:08:44 +01:00
var idx = LOADING_VERSION < 1_18_04_0? struct_try_get(inp.display_data, "bone_id", 0) :
struct_try_get(inp.attributes, "bone_id", 0);
2023-06-21 20:36:53 +02:00
2024-11-12 01:41:42 +01:00
boneMap[$ idx] = inp;
2024-11-23 12:08:44 +01:00
inp.attributes.bone_id = idx;
2023-06-21 20:36:53 +02:00
}
2023-06-25 20:12:17 +02:00
setBone();
2024-08-06 14:04:41 +02:00
}
2023-06-25 20:12:17 +02:00
2024-08-06 14:04:41 +02:00
static onDrawNode = function(xx, yy, _mx, _my, _s, _hover, _focus) {
2023-06-25 20:12:17 +02:00
var bbox = drawGetBbox(xx, yy, _s);
2024-11-29 06:18:28 +01:00
if(bonePose != noone) {
var _ss = _s * .5;
gpu_set_tex_filter(1);
draw_sprite_ext(s_node_armature_pose, 0, bbox.x0 + 24 * _ss, bbox.y1 - 24 * _ss, _ss, _ss, 0, c_white, 0.5);
gpu_set_tex_filter(0);
bonePose.drawThumbnail(_s, bbox, bone_bbox);
} else {
gpu_set_tex_filter(1);
draw_sprite_fit(s_node_armature_pose, 0, bbox.xc, bbox.yc, bbox.w, bbox.h);
gpu_set_tex_filter(0);
}
2024-08-06 14:04:41 +02:00
}
}