[Armature create] New bone mirror tool.

This commit is contained in:
Tanasart 2024-11-27 14:33:49 +07:00
parent 21839119d9
commit c4aba3af09
17 changed files with 161 additions and 15 deletions

View file

@ -1781,6 +1781,7 @@
{"name":"s_bone_tool_add","order":4,"path":"sprites/s_bone_tool_add/s_bone_tool_add.yy",}, {"name":"s_bone_tool_add","order":4,"path":"sprites/s_bone_tool_add/s_bone_tool_add.yy",},
{"name":"s_bone_tool_detach","order":5,"path":"sprites/s_bone_tool_detach/s_bone_tool_detach.yy",}, {"name":"s_bone_tool_detach","order":5,"path":"sprites/s_bone_tool_detach/s_bone_tool_detach.yy",},
{"name":"s_bone_tool_IK","order":6,"path":"sprites/s_bone_tool_IK/s_bone_tool_IK.yy",}, {"name":"s_bone_tool_IK","order":6,"path":"sprites/s_bone_tool_IK/s_bone_tool_IK.yy",},
{"name":"s_bone_tool_mirror","order":70,"path":"sprites/s_bone_tool_mirror/s_bone_tool_mirror.yy",},
{"name":"s_bone_tool_move","order":7,"path":"sprites/s_bone_tool_move/s_bone_tool_move.yy",}, {"name":"s_bone_tool_move","order":7,"path":"sprites/s_bone_tool_move/s_bone_tool_move.yy",},
{"name":"s_bone_tool_pose","order":69,"path":"sprites/s_bone_tool_pose/s_bone_tool_pose.yy",}, {"name":"s_bone_tool_pose","order":69,"path":"sprites/s_bone_tool_pose/s_bone_tool_pose.yy",},
{"name":"s_bone_tool_remove","order":8,"path":"sprites/s_bone_tool_remove/s_bone_tool_remove.yy",}, {"name":"s_bone_tool_remove","order":8,"path":"sprites/s_bone_tool_remove/s_bone_tool_remove.yy",},

View file

@ -2478,6 +2478,7 @@
{"id":{"name":"s_bone_tool_add","path":"sprites/s_bone_tool_add/s_bone_tool_add.yy",},}, {"id":{"name":"s_bone_tool_add","path":"sprites/s_bone_tool_add/s_bone_tool_add.yy",},},
{"id":{"name":"s_bone_tool_detach","path":"sprites/s_bone_tool_detach/s_bone_tool_detach.yy",},}, {"id":{"name":"s_bone_tool_detach","path":"sprites/s_bone_tool_detach/s_bone_tool_detach.yy",},},
{"id":{"name":"s_bone_tool_IK","path":"sprites/s_bone_tool_IK/s_bone_tool_IK.yy",},}, {"id":{"name":"s_bone_tool_IK","path":"sprites/s_bone_tool_IK/s_bone_tool_IK.yy",},},
{"id":{"name":"s_bone_tool_mirror","path":"sprites/s_bone_tool_mirror/s_bone_tool_mirror.yy",},},
{"id":{"name":"s_bone_tool_move","path":"sprites/s_bone_tool_move/s_bone_tool_move.yy",},}, {"id":{"name":"s_bone_tool_move","path":"sprites/s_bone_tool_move/s_bone_tool_move.yy",},},
{"id":{"name":"s_bone_tool_pose","path":"sprites/s_bone_tool_pose/s_bone_tool_pose.yy",},}, {"id":{"name":"s_bone_tool_pose","path":"sprites/s_bone_tool_pose/s_bone_tool_pose.yy",},},
{"id":{"name":"s_bone_tool_remove","path":"sprites/s_bone_tool_remove/s_bone_tool_remove.yy",},}, {"id":{"name":"s_bone_tool_remove","path":"sprites/s_bone_tool_remove/s_bone_tool_remove.yy",},},

Binary file not shown.

View file

@ -632,7 +632,7 @@ function __Bone(_parent = noone, _distance = 0, _direction = 0, _angle = 0, _len
for( var i = 0, n = array_length(childs); i < n; i++ ) for( var i = 0, n = array_length(childs); i < n; i++ )
childHash += childs[i].getHash() + ","; childHash += childs[i].getHash() + ",";
var h = $"[{ID}] : {parent_anchor}, [{IKlength}, {IKTargetID}], [{apply_scale}, {apply_rotation}], [{childHash}]"; var h = $"{name} [{ID}] : {parent_anchor}, [{IKlength}, {IKTargetID}], [{apply_scale}, {apply_rotation}], [{childHash}]";
return sha1_string_unicode(h); return sha1_string_unicode(h);
} }

View file

@ -157,11 +157,12 @@ function Node_Armature(_x, _y, _group = noone) : Node(_x, _y, _group) constructo
#endregion #endregion
tools = [ tools = [
new NodeTool( "Transform", THEME.bone_tool_move ), new NodeTool( "Transform", THEME.bone_tool_move ),
new NodeTool( "Add bones", THEME.bone_tool_add ), new NodeTool( "Add bones", THEME.bone_tool_add ),
new NodeTool( "Remove bones", THEME.bone_tool_remove ), new NodeTool( "Remove bones", THEME.bone_tool_remove ),
new NodeTool( "Detach bones", THEME.bone_tool_detach ), new NodeTool( "Detach bones", THEME.bone_tool_detach ),
new NodeTool( "IK", THEME.bone_tool_IK ), new NodeTool( "IK", THEME.bone_tool_IK ),
new NodeTool( "Mirror bones", THEME.bone_tool_mirror ),
]; ];
anchor_selecting = noone; anchor_selecting = noone;
@ -186,6 +187,8 @@ function Node_Armature(_x, _y, _group = noone) : Node(_x, _y, _group) constructo
bone_transform_bbox = -1; bone_transform_bbox = -1;
bone_transform_type = -1; bone_transform_type = -1;
mirroring_bone = noone;
static drawOverlay = function(hover, active, _x, _y, _s, _mx, _my, _snx, _sny, _panel) { static drawOverlay = function(hover, active, _x, _y, _s, _mx, _my, _snx, _sny, _panel) {
var mx = (_mx - _x) / _s; var mx = (_mx - _x) / _s;
var my = (_my - _y) / _s; var my = (_my - _y) / _s;
@ -611,8 +614,56 @@ function Node_Armature(_x, _y, _group = noone) : Node(_x, _y, _group) constructo
if(ik_dragging == noone) if(ik_dragging == noone)
anchor_selecting = _b.draw(attributes, active * 0b100, _x, _y, _s, _mx, _my, anchor_selecting); 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)) { if(anchor_selecting != noone && anchor_selecting[1] == 2 && mouse_press(mb_left, active))
ik_dragging = anchor_selecting[0]; 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 } else { // move tools

View file

@ -5,9 +5,7 @@ function Node_Armature_Pose(_x, _y, _group = noone) : Node(_x, _y, _group) const
newInput(0, nodeValue_Armature("Armature", self, noone)) newInput(0, nodeValue_Armature("Armature", self, noone))
.setVisible(true, true); .setVisible(true, true);
input_display_list = [ 0, input_display_list = [ 0 ]
["Bones", false]
]
newOutput(0, nodeValue_Output("Armature", self, VALUE_TYPE.armature, noone)); newOutput(0, nodeValue_Output("Armature", self, VALUE_TYPE.armature, noone));

View file

@ -303,6 +303,7 @@ function Theme() constructor {
bone_tool_pose = s_bone_tool_pose; bone_tool_pose = s_bone_tool_pose;
bone_tool_remove = s_bone_tool_remove; bone_tool_remove = s_bone_tool_remove;
bone_tool_scale = s_bone_tool_scale; bone_tool_scale = s_bone_tool_scale;
bone_tool_mirror = s_bone_tool_mirror;
canvas_draw_layer = s_canvas_draw_layer; canvas_draw_layer = s_canvas_draw_layer;
canvas_iso_angle = s_canvas_iso_angle; canvas_iso_angle = s_canvas_iso_angle;
canvas_fill_type = s_canvas_fill_type; canvas_fill_type = s_canvas_fill_type;

Binary file not shown.

Before

Width:  |  Height:  |  Size: 532 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 539 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 539 B

View file

@ -2,24 +2,24 @@
"$GMSprite":"", "$GMSprite":"",
"%Name":"s_bone_scale", "%Name":"s_bone_scale",
"bboxMode":0, "bboxMode":0,
"bbox_bottom":32, "bbox_bottom":34,
"bbox_left":8, "bbox_left":14,
"bbox_right":42, "bbox_right":35,
"bbox_top":15, "bbox_top":13,
"collisionKind":1, "collisionKind":1,
"collisionTolerance":0, "collisionTolerance":0,
"DynamicTexturePage":false, "DynamicTexturePage":false,
"edgeFiltering":false, "edgeFiltering":false,
"For3D":false, "For3D":false,
"frames":[ "frames":[
{"$GMSpriteFrame":"","%Name":"2de46b2b-d64b-4497-a785-81906d872c31","name":"2de46b2b-d64b-4497-a785-81906d872c31","resourceType":"GMSpriteFrame","resourceVersion":"2.0",}, {"$GMSpriteFrame":"","%Name":"b3bac891-34ed-4f4b-b531-d215fc41ce83","name":"b3bac891-34ed-4f4b-b531-d215fc41ce83","resourceType":"GMSpriteFrame","resourceVersion":"2.0",},
], ],
"gridX":0, "gridX":0,
"gridY":0, "gridY":0,
"height":48, "height":48,
"HTile":false, "HTile":false,
"layers":[ "layers":[
{"$GMImageLayer":"","%Name":"dfa57967-60b1-4c24-ae6c-4f613947bdcd","blendMode":0,"displayName":"default","isLocked":false,"name":"dfa57967-60b1-4c24-ae6c-4f613947bdcd","opacity":100.0,"resourceType":"GMImageLayer","resourceVersion":"2.0","visible":true,}, {"$GMImageLayer":"","%Name":"6f7f34f2-0a28-4e6d-836a-9cbdbaddcb5c","blendMode":0,"displayName":"default","isLocked":false,"name":"6f7f34f2-0a28-4e6d-836a-9cbdbaddcb5c","opacity":100.0,"resourceType":"GMImageLayer","resourceVersion":"2.0","visible":true,},
], ],
"name":"s_bone_scale", "name":"s_bone_scale",
"nineSlice":null, "nineSlice":null,
@ -69,8 +69,8 @@
"tracks":[ "tracks":[
{"$GMSpriteFramesTrack":"","builtinName":0,"events":[],"inheritsTrackColour":true,"interpolation":1,"isCreationTrack":false,"keyframes":{"$KeyframeStore<SpriteFrameKeyframe>":"","Keyframes":[ {"$GMSpriteFramesTrack":"","builtinName":0,"events":[],"inheritsTrackColour":true,"interpolation":1,"isCreationTrack":false,"keyframes":{"$KeyframeStore<SpriteFrameKeyframe>":"","Keyframes":[
{"$Keyframe<SpriteFrameKeyframe>":"","Channels":{ {"$Keyframe<SpriteFrameKeyframe>":"","Channels":{
"0":{"$SpriteFrameKeyframe":"","Id":{"name":"2de46b2b-d64b-4497-a785-81906d872c31","path":"sprites/s_bone_scale/s_bone_scale.yy",},"resourceType":"SpriteFrameKeyframe","resourceVersion":"2.0",}, "0":{"$SpriteFrameKeyframe":"","Id":{"name":"b3bac891-34ed-4f4b-b531-d215fc41ce83","path":"sprites/s_bone_scale/s_bone_scale.yy",},"resourceType":"SpriteFrameKeyframe","resourceVersion":"2.0",},
},"Disabled":false,"id":"47cbaea7-265a-4656-960f-a01215b92edc","IsCreationKey":false,"Key":0.0,"Length":1.0,"resourceType":"Keyframe<SpriteFrameKeyframe>","resourceVersion":"2.0","Stretch":false,}, },"Disabled":false,"id":"5f0e680f-0ddb-4862-8014-1b45a95fdb51","IsCreationKey":false,"Key":0.0,"Length":1.0,"resourceType":"Keyframe<SpriteFrameKeyframe>","resourceVersion":"2.0","Stretch":false,},
],"resourceType":"KeyframeStore<SpriteFrameKeyframe>","resourceVersion":"2.0",},"modifiers":[],"name":"frames","resourceType":"GMSpriteFramesTrack","resourceVersion":"2.0","spriteId":null,"trackColour":0,"tracks":[],"traits":0,}, ],"resourceType":"KeyframeStore<SpriteFrameKeyframe>","resourceVersion":"2.0",},"modifiers":[],"name":"frames","resourceType":"GMSpriteFramesTrack","resourceVersion":"2.0","spriteId":null,"trackColour":0,"tracks":[],"traits":0,},
], ],
"visibleRange":null, "visibleRange":null,

Binary file not shown.

After

Width:  |  Height:  |  Size: 591 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 486 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 591 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 486 B

View file

@ -0,0 +1,94 @@
{
"$GMSprite":"",
"%Name":"s_bone_tool_mirror",
"bboxMode":0,
"bbox_bottom":16,
"bbox_left":1,
"bbox_right":16,
"bbox_top":1,
"collisionKind":1,
"collisionTolerance":0,
"DynamicTexturePage":false,
"edgeFiltering":false,
"For3D":false,
"frames":[
{"$GMSpriteFrame":"","%Name":"3f583b24-d2b0-4fb1-8600-42570017307f","name":"3f583b24-d2b0-4fb1-8600-42570017307f","resourceType":"GMSpriteFrame","resourceVersion":"2.0",},
{"$GMSpriteFrame":"","%Name":"b30f665b-391e-4e70-bbe8-b0ec4e0a7e6a","name":"b30f665b-391e-4e70-bbe8-b0ec4e0a7e6a","resourceType":"GMSpriteFrame","resourceVersion":"2.0",},
],
"gridX":0,
"gridY":0,
"height":18,
"HTile":false,
"layers":[
{"$GMImageLayer":"","%Name":"eb06fcff-2331-4ae5-aa77-c96cd2fe6d0d","blendMode":0,"displayName":"default","isLocked":false,"name":"eb06fcff-2331-4ae5-aa77-c96cd2fe6d0d","opacity":100.0,"resourceType":"GMImageLayer","resourceVersion":"2.0","visible":true,},
],
"name":"s_bone_tool_mirror",
"nineSlice":null,
"origin":4,
"parent":{
"name":"tool",
"path":"folders/theme/tool.yy",
},
"preMultiplyAlpha":false,
"resourceType":"GMSprite",
"resourceVersion":"2.0",
"sequence":{
"$GMSequence":"",
"%Name":"s_bone_tool_mirror",
"autoRecord":true,
"backdropHeight":768,
"backdropImageOpacity":0.5,
"backdropImagePath":"",
"backdropWidth":1366,
"backdropXOffset":0.0,
"backdropYOffset":0.0,
"events":{
"$KeyframeStore<MessageEventKeyframe>":"",
"Keyframes":[],
"resourceType":"KeyframeStore<MessageEventKeyframe>",
"resourceVersion":"2.0",
},
"eventStubScript":null,
"eventToFunction":{},
"length":2.0,
"lockOrigin":false,
"moments":{
"$KeyframeStore<MomentsEventKeyframe>":"",
"Keyframes":[],
"resourceType":"KeyframeStore<MomentsEventKeyframe>",
"resourceVersion":"2.0",
},
"name":"s_bone_tool_mirror",
"playback":1,
"playbackSpeed":30.0,
"playbackSpeedType":0,
"resourceType":"GMSequence",
"resourceVersion":"2.0",
"showBackdrop":true,
"showBackdropImage":false,
"timeUnits":1,
"tracks":[
{"$GMSpriteFramesTrack":"","builtinName":0,"events":[],"inheritsTrackColour":true,"interpolation":1,"isCreationTrack":false,"keyframes":{"$KeyframeStore<SpriteFrameKeyframe>":"","Keyframes":[
{"$Keyframe<SpriteFrameKeyframe>":"","Channels":{
"0":{"$SpriteFrameKeyframe":"","Id":{"name":"3f583b24-d2b0-4fb1-8600-42570017307f","path":"sprites/s_bone_tool_mirror/s_bone_tool_mirror.yy",},"resourceType":"SpriteFrameKeyframe","resourceVersion":"2.0",},
},"Disabled":false,"id":"7ab06af3-5187-46e9-b2e6-a6ff295168fa","IsCreationKey":false,"Key":0.0,"Length":1.0,"resourceType":"Keyframe<SpriteFrameKeyframe>","resourceVersion":"2.0","Stretch":false,},
{"$Keyframe<SpriteFrameKeyframe>":"","Channels":{
"0":{"$SpriteFrameKeyframe":"","Id":{"name":"b30f665b-391e-4e70-bbe8-b0ec4e0a7e6a","path":"sprites/s_bone_tool_mirror/s_bone_tool_mirror.yy",},"resourceType":"SpriteFrameKeyframe","resourceVersion":"2.0",},
},"Disabled":false,"id":"7549b53c-5dc2-431d-90bc-a95597afdd7a","IsCreationKey":false,"Key":1.0,"Length":1.0,"resourceType":"Keyframe<SpriteFrameKeyframe>","resourceVersion":"2.0","Stretch":false,},
],"resourceType":"KeyframeStore<SpriteFrameKeyframe>","resourceVersion":"2.0",},"modifiers":[],"name":"frames","resourceType":"GMSpriteFramesTrack","resourceVersion":"2.0","spriteId":null,"trackColour":0,"tracks":[],"traits":0,},
],
"visibleRange":null,
"volume":1.0,
"xorigin":9,
"yorigin":9,
},
"swatchColours":null,
"swfPrecision":0.5,
"textureGroupId":{
"name":"Default",
"path":"texturegroups/Default",
},
"type":0,
"VTile":false,
"width":18,
}