Newer
Older
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
#====================== BEGIN GPL LICENSE BLOCK ======================
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software Foundation,
# Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
#
#======================= END GPL LICENSE BLOCK ========================
import bpy
from bpy.props import *
import rigify
from rigify.utils import get_rig_type
from rigify import generate
from rna_prop_ui import rna_idprop_ui_prop_get
class DATA_PT_rigify_buttons(bpy.types.Panel):
bl_label = "Rigify Buttons"
bl_space_type = 'PROPERTIES'
bl_region_type = 'WINDOW'
bl_context = "data"
#bl_options = {'DEFAULT_OPEN'}
@classmethod
def poll(cls, context):
if not context.armature:
return False
#obj = context.object
#if obj:
# return (obj.mode in ('POSE', 'OBJECT', 'EDIT'))
#return False
return True
def draw(self, context):
C = context
layout = self.layout
obj = context.object
Campbell Barton
committed
id_store = C.window_manager
if obj.mode in ('POSE', 'OBJECT'):
row = layout.row()
row.operator("pose.rigify_generate", text="Generate")
elif obj.mode == 'EDIT':
# Build types list
Campbell Barton
committed
collection_name = str(id_store.rigify_collection).replace(" ", "")
Campbell Barton
committed
for i in range(0, len(id_store.rigify_types)):
id_store.rigify_types.remove(0)
for r in rigify.rig_list:
collection = r.split('.')[0]
if collection_name == "All":
Campbell Barton
committed
a = id_store.rigify_types.add()
a.name = r
elif r.startswith(collection_name + '.'):
Campbell Barton
committed
a = id_store.rigify_types.add()
a.name = r
elif collection_name == "None" and len(r.split('.')) == 1:
Campbell Barton
committed
a = id_store.rigify_types.add()
a.name = r
## Rig collection field
#row = layout.row()
Campbell Barton
committed
#row.prop(id_store, 'rigify_collection', text="Category")
# Rig type list
row = layout.row()
Campbell Barton
committed
row.template_list(id_store, "rigify_types", id_store, 'rigify_active_type')
row = layout.row()
op = row.operator("armature.metarig_sample_add", text="Add sample")
Campbell Barton
committed
op.metarig_type = id_store.rigify_types[id_store.rigify_active_type].name
Nathan Vegdahl
committed
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
class DATA_PT_rigify_layer_names(bpy.types.Panel):
bl_label = "Rigify Layer Names"
bl_space_type = 'PROPERTIES'
bl_region_type = 'WINDOW'
bl_context = "data"
bl_options = {'DEFAULT_CLOSED'}
@classmethod
def poll(cls, context):
if not context.armature:
return False
#obj = context.object
#if obj:
# return (obj.mode in ('POSE', 'OBJECT', 'EDIT'))
#return False
return True
def draw(self, context):
C = context
layout = self.layout
obj = context.object
if len(obj.data.rigify_props) < 1:
obj.data.rigify_props.add()
for i in range(28):
if (i % 16) == 0:
col = layout.column()
if i == 0:
col.label(text="Top Row:")
else:
col.label(text="Bottom Row:")
if (i % 8) == 0:
col = layout.column(align=True)
row = col.row()
row.prop(obj.data, "layers", index=i, text="", toggle=True)
row.prop(obj.data.rigify_props[0], "layer_name_%s" % str(i+1).rjust(2, "0"), text="Layer %d" % (i + 1))
class BONE_PT_rigify_buttons(bpy.types.Panel):
bl_label = "Rigify Type"
bl_space_type = 'PROPERTIES'
bl_region_type = 'WINDOW'
bl_context = "bone"
#bl_options = {'DEFAULT_OPEN'}
@classmethod
def poll(cls, context):
if not context.armature or not context.active_pose_bone:
return False
obj = context.object
if obj:
return (obj.mode in ('POSE'))
return False
def draw(self, context):
C = context
Campbell Barton
committed
id_store = C.window_manager
Campbell Barton
committed
collection_name = str(id_store.rigify_collection).replace(" ", "")
rig_name = str(context.active_pose_bone.rigify_type).replace(" ", "")
layout = self.layout
# Build types list
Campbell Barton
committed
for i in range(0, len(id_store.rigify_types)):
id_store.rigify_types.remove(0)
for r in rigify.rig_list:
collection = r.split('.')[0]
if collection_name == "All":
Campbell Barton
committed
a = id_store.rigify_types.add()
a.name = r
elif r.startswith(collection_name + '.'):
Campbell Barton
committed
a = id_store.rigify_types.add()
a.name = r
elif collection_name == "None" and len(r.split('.')) == 1:
Campbell Barton
committed
a = id_store.rigify_types.add()
a.name = r
# Rig type field
row = layout.row()
Campbell Barton
committed
row.prop_search(bone, "rigify_type", id_store, "rigify_types", text="Rig type:")
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
# Rig type parameters / Rig type non-exist alert
if rig_name != "":
if len(bone.rigify_parameters) < 1:
bone.rigify_parameters.add()
try:
rig = get_rig_type(rig_name)
rig.Rig
except (ImportError, AttributeError):
row = layout.row()
box = row.box()
box.label(text="ALERT: type \"%s\" does not exist!" % rig_name)
else:
try:
rig.Rig.parameters_ui
except AttributeError:
pass
else:
col = layout.column()
col.label(text="Options:")
box = layout.box()
rig.Rig.parameters_ui(box, C.active_object, bone.name)
#class INFO_MT_armature_metarig_add(bpy.types.Menu):
# bl_idname = "INFO_MT_armature_metarig_add"
# bl_label = "Meta-Rig"
# def draw(self, context):
#import rigify
#layout = self.layout
#layout.operator_context = 'INVOKE_REGION_WIN'
#for submodule_type in rigify.get_submodule_types():
# text = bpy.path.display_name(submodule_type)
# layout.operator("pose.metarig_sample_add", text=text, icon='OUTLINER_OB_ARMATURE').metarig_type = submodule_type
def rigify_report_exception(operator, exception):
import traceback
import sys
import os
# find the module name where the error happened
# hint, this is the metarig type!
exceptionType, exceptionValue, exceptionTraceback = sys.exc_info()
fn = traceback.extract_tb(exceptionTraceback)[-1][0]
fn = os.path.basename(fn)
fn = os.path.splitext(fn)[0]
message = []
if fn.startswith("__"):
message.append("Incorrect armature...")
else:
message.append("Incorrect armature for type '%s'" % fn)
message.append(exception.message)
message.reverse() # XXX - stupid! menu's are upside down!
operator.report(set(['INFO']), '\n'.join(message))
class Generate(bpy.types.Operator):
'''Generates a rig from the active metarig armature'''
bl_idname = "pose.rigify_generate"
bl_label = "Rigify Generate Rig"
import imp
imp.reload(generate)
use_global_undo = context.user_preferences.edit.use_global_undo
context.user_preferences.edit.use_global_undo = False
try:
generate.generate_rig(context, context.object)
except rigify.utils.MetarigError as rig_exception:
rigify_report_exception(self, rig_exception)
finally:
context.user_preferences.edit.use_global_undo = use_global_undo
return {'FINISHED'}
class Sample(bpy.types.Operator):
'''Create a sample metarig to be modified before generating the final rig.'''
bl_idname = "armature.metarig_sample_add"
bl_label = "Add a sample metarig for a rig type"
metarig_type = StringProperty(name="Type", description="Name of the rig type to generate a sample of", maxlen=128, default="")
def execute(self, context):
if context.mode == 'EDIT_ARMATURE' and self.metarig_type != "":
use_global_undo = context.user_preferences.edit.use_global_undo
context.user_preferences.edit.use_global_undo = False
try:
rig = get_rig_type(self.metarig_type).Rig
create_sample = rig.create_sample
except (ImportError, AttributeError):
print("Rigify: rig type has no sample.")
else:
create_sample(context.active_object)
finally:
context.user_preferences.edit.use_global_undo = use_global_undo
bpy.ops.object.mode_set(mode='EDIT')
return {'FINISHED'}
#menu_func = (lambda self, context: self.layout.menu("INFO_MT_armature_metarig_add", icon='OUTLINER_OB_ARMATURE'))
#from bl_ui import space_info # ensure the menu is loaded first
Nathan Vegdahl
committed
bpy.utils.register_class(DATA_PT_rigify_layer_names)
bpy.utils.register_class(DATA_PT_rigify_buttons)
bpy.utils.register_class(BONE_PT_rigify_buttons)
bpy.utils.register_class(Generate)
bpy.utils.register_class(Sample)
#space_info.INFO_MT_armature_add.append(ui.menu_func)
def unregister():
Nathan Vegdahl
committed
bpy.utils.unregister_class(DATA_PT_rigify_layer_names)
bpy.utils.unregister_class(DATA_PT_rigify_buttons)
bpy.utils.unregister_class(BONE_PT_rigify_buttons)
bpy.utils.unregister_class(Generate)
bpy.utils.unregister_class(Sample)