-
Notifications
You must be signed in to change notification settings - Fork 40
/
object_convert_to_armature.py
137 lines (116 loc) · 4.97 KB
/
object_convert_to_armature.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
# coding: utf-8
# This script is Free software. Please share and reuse.
# ♡2010-2019 Adam Dominec <[email protected]>
bl_info = {
"name": "Convert Mesh to Armature",
"author": "Addam Dominec",
"version": (1, 0),
"blender": (2, 80, 0),
"location": "Object > Convert to Armature",
"warning": "",
"description": "Generate an armature with a single bone controlling each face",
"category": "Object",
"wiki_url": "http://wiki.blender.org/index.php/Extensions:2.6/Py/Scripts/Object/Convert_to_Armature",
}
import bpy
import bmesh
from itertools import repeat
from math import asin, pi
def add_bone(name, armature, head, tail):
bo = armature.edit_bones.new(name)
bo.head = head
bo.tail = tail
return bo
def main(context):
mesh_object = context.object
if context.mode == 'EDIT':
bm = bmesh.from_edit_mesh(mesh_object.data)
else:
bm = bmesh.new()
bm.from_mesh(mesh_object.data)
bm.verts.ensure_lookup_table()
#Create an armature
armature = bpy.data.armatures.new(f"{mesh_object.name} Armature")
armature_object = bpy.data.objects.new(armature.name, object_data=armature)
armature_object.matrix_local = mesh_object.matrix_local
context.scene.collection.objects.link(armature_object)
modifier = mesh_object.modifiers.new("Folding Armature", 'ARMATURE')
modifier.use_bone_envelopes = False
modifier.use_vertex_groups = True
modifier.object = armature_object
visited_faces = set()
loops = 0
#Generate the bones
angles = dict() #bone -> folding angle
queue = [(None, bm.faces.active, None, None)] #Edge, face and a bone connecting these two and parent in the tree
context.view_layer.objects.active = armature_object
recall_mode = context.mode
bpy.ops.object.mode_set(mode='EDIT')
while queue:
edge, face, parent_bone, parent_face = queue.pop()
if face in visited_faces:
loops += 1 #We went to the same face from two different directions
continue
visited_faces.add(face)
vgroup = mesh_object.vertex_groups.new(name=f"Face {face.index}") #Create a vertex group for this face
vgroup.add([v.index for v in face.verts], 1, 'ADD')
tail = face.calc_center_median_weighted()
if edge:
a, b = [v.co for v in edge.verts]
edge_vector = b - a
head = (a + b) / 2
tail -= (tail - head).project(edge_vector)
else: #root bone
tail = vertex_avg(face.verts)
head = tail.copy()
head.z -= 1
bone = add_bone(vgroup.name, armature, head, tail)
bone.align_roll(face.normal)
bone.parent = parent_bone
if edge: #all except the root bone
angles[bone.name] = -edge.calc_face_angle_signed(0)
for loop in face.loops:
if loop.edge is not edge:
queue += [(loop.edge, l.face, bone, face) for l in loop.link_loops]
#Set transform locks (must be done in object mode)
bpy.ops.object.mode_set(mode='OBJECT')
pose_bone = armature_object.pose.bones[0]
pose_bone.lock_scale[0:3] = True, True, True
for pose_bone in armature_object.pose.bones[1:]:
pose_bone.lock_rotation[0:3] = False, True, True
pose_bone.lock_location[0:3] = True, True, True
pose_bone.lock_scale[0:3] = True, True, True
pose_bone.lock_ik_x, pose_bone.lock_ik_y, pose_bone.lock_ik_z = False, True, True
pose_bone.rotation_mode = 'XYZ'
pose_bone.rotation_euler.x = -angles[pose_bone.name]
context.view_layer.objects.active = mesh_object
bpy.ops.object.mode_set(mode=recall_mode)
if loops:
raise ValueError(loops)
class OBJECT_OT_convert_to_armature(bpy.types.Operator):
'''Generate an armature with a single bone controlling each face. The mesh must be a tree-like structure for this to make sense. Active face is used for main bone.'''
bl_idname = "object.convert_to_armature"
bl_label = "Convert to Armature"
bl_description = "Generate an armature from the active mesh"
@classmethod
def poll(cls, context):
return context.active_object.type == 'MESH'
def execute(self, context):
try:
main(context)
except ValueError as E:
if isinstance(E.args[0], int):
self.report({'ERROR', 'ERROR_INVALID_INPUT'}, "There is a loop of connected faces. Use Export Paper Model add-on and EdgeSplit modifier to eliminate them.\n" "Otherwise, the armature may be unusable.")
else:
raise
return {'FINISHED'}
def menu_func(self, context):
self.layout.operator(OBJECT_OT_convert_to_armature.bl_idname, text="Convert to Armature")
def register():
bpy.utils.register_class(OBJECT_OT_convert_to_armature)
bpy.types.VIEW3D_MT_object.append(menu_func)
def unregister():
bpy.types.VIEW3D_MT_object.remove(menu_func)
bpy.utils.unregister_class(OBJECT_OT_convert_to_armature)
if __name__ == "__main__":
register()