fixing bugs

master
ChristopheSeux 2023-01-18 16:02:42 +01:00
parent 9d339585f5
commit c673d03a6a
25 changed files with 228 additions and 149 deletions

View File

@ -1,8 +1,8 @@
bl_info = { bl_info = {
"name": "Bone Widget", "name": "Bone Widget",
"author": "Christophe SEUX", "author": "Christophe SEUX",
"version": (2, 0, 0), "version": (2, 0, 1),
"blender": (2, 92, 0), "blender": (3, 4, 1),
"description": "Create custom shapes for bone controller", "description": "Create custom shapes for bone controller",
"warning": "", "warning": "",
"wiki_url": "", "wiki_url": "",
@ -12,22 +12,37 @@ bl_info = {
import sys import sys
if "bpy" in locals(): if "bpy" in locals():
import importlib as imp import importlib as imp
imp.reload(context) imp.reload(context)
sys.modules.update({"bone_widget.ctx": context.BW_context()})
imp.reload(shape_utils)
imp.reload(transform_utils)
imp.reload(icon_utils)
imp.reload(properties) imp.reload(properties)
imp.reload(operators) imp.reload(operators)
imp.reload(ui) imp.reload(ui)
else: else:
from . import context from . import context
sys.modules.update({"bone_widget.ctx": context.BW_context()})
from . import operators from . import operators
from . import ui from . import ui
from . import properties from . import properties
from . import shape_utils
from . import transform_utils
from . import icon_utils
import bpy import bpy
import sys
#sys.modules.update({"bone_widget.ctx": context.BW_ctx()}) #sys.modules.update({"bone_widget.ctx": context.BW_ctx()})
from bone_widget import ctx
def register(): def register():
@ -38,6 +53,8 @@ def register():
#bpy.types.Scene.bone_widget = bpy.props.PointerProperty(type=BoneWidgetSettings) #bpy.types.Scene.bone_widget = bpy.props.PointerProperty(type=BoneWidgetSettings)
#get_widgets(DefaultFolder, DefaultShapes) #get_widgets(DefaultFolder, DefaultShapes)
#get_widgets(CustomFolder, CustomShapes) #get_widgets(CustomFolder, CustomShapes)
from bone_widget import ctx
for f in ctx.folders: for f in ctx.folders:
f.load_widgets() f.load_widgets()

View File

@ -6,7 +6,7 @@ import os
from bone_widget.shape_utils import get_bone from bone_widget.shape_utils import get_bone
class BW_ctx: class BW_context:
def __init__(self): def __init__(self):
self.module_dir = Path(__file__).parent self.module_dir = Path(__file__).parent
@ -48,11 +48,15 @@ class BW_ctx:
@property @property
def show_prefs_op(self): def show_prefs_op(self):
ops = bpy.context.window_manager.operators ops = bpy.context.window_manager.operators
op_idname = self.get_op_id('show_preferences') id_names = [
self.get_op_id('show_preferences'),
self.get_op_id('add_folder'),
self.get_op_id('remove_folder')
]
#transforms = self.prefs.transforms #transforms = self.prefs.transforms
if ops and ops[-1].bl_idname == op_idname: if ops and ops[-1].bl_idname in id_names:
return ops[-1] return True
@property @property
def category(self): def category(self):
@ -70,6 +74,9 @@ class BW_ctx:
@property @property
def widget_col(self): def widget_col(self):
col_name = self.prefs.collection col_name = self.prefs.collection
col_name = col_name.format(ob=bpy.context.object)
col = bpy.data.collections.get(col_name) col = bpy.data.collections.get(col_name)
if not col: if not col:
col = bpy.data.collections.new(col_name) col = bpy.data.collections.new(col_name)
@ -278,6 +285,6 @@ class BW_ctx:
# ops_data = getattr(bpy.ops, ctx.id_name) # ops_data = getattr(bpy.ops, ctx.id_name)
# getattr(ops_data, op)() # getattr(ops_data, op)()
sys.modules.update({"bone_widget.ctx": BW_ctx()})

View File

@ -16,6 +16,8 @@ def bounds_2d_co(coords, width, height):
co_2d = [space_2d(region, rv3d, co) for co in coords] co_2d = [space_2d(region, rv3d, co) for co in coords]
print("co_2d", co_2d)
x_value = sorted(co_2d, key=lambda x:x[0]) x_value = sorted(co_2d, key=lambda x:x[0])
y_value = sorted(co_2d, key=lambda x:x[1]) y_value = sorted(co_2d, key=lambda x:x[1])
@ -54,6 +56,9 @@ def render_widget(shape, filepath, width=32, height=32) :
bm = bmesh.new() bm = bmesh.new()
bm.from_object(shape, dg) bm.from_object(shape, dg)
width *= 2
height *= 2
coords = [shape.matrix_world @ v.co for v in bm.verts] coords = [shape.matrix_world @ v.co for v in bm.verts]
coords = bounds_2d_co(coords, width, height) coords = bounds_2d_co(coords, width, height)
@ -66,45 +71,53 @@ def render_widget(shape, filepath, width=32, height=32) :
batch = batch_for_shader(shader, 'LINES', {"pos": coords}, indices=indices) batch = batch_for_shader(shader, 'LINES', {"pos": coords}, indices=indices)
with offscreen.bind(): with offscreen.bind():
bgl.glClearColor(0.0, 0.0, 0.0, 0.0) fb = gpu.state.active_framebuffer_get()
bgl.glClear(bgl.GL_COLOR_BUFFER_BIT) fb.clear(color=(0.0, 0.0, 0.0, 0.0))
with gpu.matrix.push_pop(): with gpu.matrix.push_pop():
# reset matrices -> use normalized device coordinates [-1, 1] # reset matrices -> use normalized device coordinates [-1, 1]
gpu.matrix.load_matrix(Matrix.Identity(4)) gpu.matrix.load_matrix(Matrix.Identity(4))
gpu.matrix.load_projection_matrix(Matrix.Identity(4)) gpu.matrix.load_projection_matrix(Matrix.Identity(4))
bgl.glLineWidth(4)
bgl.glEnable( bgl.GL_LINE_SMOOTH )
bgl.glEnable(bgl.GL_BLEND)
shader.bind() shader.bind()
shader.uniform_float("color", (0, 0, 0, 0.1)) gpu.state.line_width_set(4*2)
#bgl.glEnable(bgl.GL_BLEND)
#bgl.glEnable( bgl.GL_LINE_SMOOTH )
shader.uniform_float("color", (0, 0, 0, 0.2))
batch.draw(shader) batch.draw(shader)
bgl.glLineWidth(2) gpu.state.line_width_set(2*2)
shader.uniform_float("color", (0.85, 0.85, 0.85, 1)) shader.uniform_float("color", (0.85, 0.85, 0.85, 1))
batch.draw(shader) batch.draw(shader)
buffer = bgl.Buffer(bgl.GL_BYTE, width * height * 4) buffer = fb.read_color(0, 0, width, height, 4, 0, 'FLOAT')
bgl.glReadBuffer(bgl.GL_BACK) buffer.dimensions = width * height * 4
bgl.glReadPixels(0, 0, width, height, bgl.GL_RGBA, bgl.GL_UNSIGNED_BYTE, buffer) #buffer = bgl.Buffer(bgl.GL_BYTE, width * height * 4)
#bgl.glReadBuffer(bgl.GL_BACK)
#bgl.glReadPixels(0, 0, width, height, bgl.GL_RGBA, bgl.GL_UNSIGNED_BYTE, buffer)
offscreen.free() offscreen.free()
#icon_name = '.' + shape.name + '_icon.png' #icon_name = '.' + shape.name + '_icon.png'
icon_name = shape.name + '_icon.png' icon_name = f'.{shape.name}_icon.png'
image = bpy.data.images.get(icon_name) image = bpy.data.images.get(icon_name)
if image: if image:
bpy.data.images.remove(image) bpy.data.images.remove(image)
image = bpy.data.images.new(icon_name, width, height) image = bpy.data.images.new(icon_name, width, height, alpha=True, float_buffer=True)
#image_data = np.asarray(buffer, dtype=np.uint8) #image_data = np.asarray(buffer, dtype=np.uint8)
#image.pixels.foreach_set(image_data.flatten()/255) #image.pixels.foreach_set(image_data.flatten()/255)
#image_data = np.asarray(buffer, dtype=np.uint8) #image_data = np.asarray(buffer, dtype=np.uint8)
image.pixels = [v / 255 for v in buffer] image.pixels.foreach_set(buffer)
image.save_render(str(filepath)) image.scale(int(width/2), int(height/2))
image.filepath_raw = str(filepath)
image.save()
bpy.data.images.remove(image)

View File

@ -8,7 +8,7 @@ from bone_widget import ctx
from .transform_utils import transform_matrix, apply_mat_to_verts, get_bone_size_factor, \ from .transform_utils import transform_matrix, apply_mat_to_verts, get_bone_size_factor, \
get_bone_matrix get_bone_matrix
from .icon_utils import render_widget from .icon_utils import render_widget
from .shape_utils import get_clean_shape, symmetrize_bone_shape, link_to_col, get_bone from .shape_utils import get_clean_shape, symmetrize_bone_shape, link_to_col, get_bone, custom_shape_matrix
import re import re
from pathlib import Path from pathlib import Path
@ -39,7 +39,6 @@ class BW_OT_copy_widgets(Operator) :
else: else:
bones = ctx.bones bones = ctx.bones
for b in bones: for b in bones:
if b.custom_shape: if b.custom_shape:
s = b.custom_shape.copy() s = b.custom_shape.copy()
@ -222,14 +221,14 @@ class BW_OT_load_default_color(Operator) :
ob = context.object ob = context.object
colors = { colors = {
'root': [0, 0, 0], 'Black': [0, 0, 0],
'spine chest hips': [1, 1, 0], 'Yellow': [1, 1, 0],
'.R': [0, 0.035, 0.95], 'Red': [0, 0.035, 0.95],
'.L': [1, 0, 0], 'Blue': [1, 0, 0],
'ik.R': [1, 0.1, 0.85], 'Pink': [1, 0.1, 0.85],
'ik.L': [0.67, 0, 0.87], 'Purple': [0.67, 0, 0.87],
'tweak.L': [0.75, 0.65, 0], 'Brown': [0.75, 0.65, 0],
'tweak.R': [1, 0.6, 0], 'Orange': [1, 0.6, 0],
} }
for k, v in colors.items(): for k, v in colors.items():
@ -244,9 +243,9 @@ class BW_OT_load_default_color(Operator) :
bone_group.colors.select = self.select_color bone_group.colors.select = self.select_color
bone_group.colors.active = self.active_color bone_group.colors.active = self.active_color
return {'FINISHED'} return {'FINISHED'}
class BW_OT_copy_bone_groups(Operator): class BW_OT_copy_bone_groups(Operator):
bl_idname = 'bonewidget.copy_bone_groups' bl_idname = 'bonewidget.copy_bone_groups'
bl_label = "Copy Bone Groups" bl_label = "Copy Bone Groups"
@ -272,10 +271,9 @@ class BW_OT_copy_bone_groups(Operator) :
blend_file.write_text(json.dumps(bone_groups)) blend_file.write_text(json.dumps(bone_groups))
return {'FINISHED'} return {'FINISHED'}
class BW_OT_paste_bone_groups(Operator): class BW_OT_paste_bone_groups(Operator):
bl_idname = 'bonewidget.paste_bone_groups' bl_idname = 'bonewidget.paste_bone_groups'
bl_label = "Paste Bone Groups" bl_label = "Paste Bone Groups"
@ -384,6 +382,7 @@ class BW_OT_rename_folder(Operator):
class BW_OT_show_preferences(Operator): class BW_OT_show_preferences(Operator):
"""Display the preferences to the tab panel"""
bl_idname = 'bonewidget.show_preferences' bl_idname = 'bonewidget.show_preferences'
bl_label = "Show Preferences" bl_label = "Show Preferences"
bl_options = {'REGISTER'} bl_options = {'REGISTER'}
@ -394,6 +393,7 @@ class BW_OT_show_preferences(Operator):
class BW_OT_transform_widget(Operator): class BW_OT_transform_widget(Operator):
"""Transform the Rotation Location or Scale of the selected shapes"""
bl_idname = 'bonewidget.transform_widget' bl_idname = 'bonewidget.transform_widget'
bl_label = "Transfom" bl_label = "Transfom"
bl_options = {'REGISTER', 'UNDO'} bl_options = {'REGISTER', 'UNDO'}
@ -410,12 +410,17 @@ class BW_OT_transform_widget(Operator):
class BW_OT_create_widget(Operator): class BW_OT_create_widget(Operator):
"""Create the widget shape and assign it to the bone"""
bl_idname = 'bonewidget.create_widget' bl_idname = 'bonewidget.create_widget'
bl_label = "Create Widget" bl_label = "Create Widget"
bl_options = {'REGISTER', 'UNDO'} bl_options = {'REGISTER', 'UNDO'}
symmetrize: BoolProperty(name='Symmetrize', default=True) symmetrize: BoolProperty(name='Symmetrize', default=True)
@classmethod
def poll(cls, context):
return ctx.bone and ctx.active_widget
def invoke(self, context, event): def invoke(self, context, event):
self.symmetrize = ctx.prefs.symmetrize self.symmetrize = ctx.prefs.symmetrize
return self.execute(context) return self.execute(context)
@ -429,7 +434,7 @@ class BW_OT_create_widget(Operator):
shape = data_src.objects[0] shape = data_src.objects[0]
for bone in ctx.selected_bones: for bone in ctx.selected_bones:
shape_copy = shape.copy() #shape_copy = shape.copy()
if bpy.app.version_string < '3.0.0': if bpy.app.version_string < '3.0.0':
bone.custom_shape_scale = 1.0 bone.custom_shape_scale = 1.0
@ -443,7 +448,7 @@ class BW_OT_create_widget(Operator):
#copy_shape = shape.copy() #copy_shape = shape.copy()
#copy_shape.data = copy_shape.data.copy() #copy_shape.data = copy_shape.data.copy()
bone.custom_shape = get_clean_shape(bone, shape_copy, col=ctx.widget_col, prefix=ctx.prefs.prefix, separate=False) bone.custom_shape = get_clean_shape(bone, shape, col=ctx.widget_col, prefix=ctx.prefs.prefix)
if self.symmetrize: if self.symmetrize:
symmetrize_bone_shape(bone, prefix=ctx.prefs.prefix) symmetrize_bone_shape(bone, prefix=ctx.prefs.prefix)
@ -648,9 +653,19 @@ class BW_OT_add_widget(Operator):
widget_path.parent.mkdir(exist_ok=True, parents=True) widget_path.parent.mkdir(exist_ok=True, parents=True)
bpy.data.libraries.write(str(widget_path), {shape}) bpy.data.libraries.write(str(widget_path), {shape})
render_widget(shape, icon_path) # Copy the shape to apply the bone matrix to the mesh
widget = folder.add_widget(name) shape_copy = shape.copy()
shape_copy.data = shape_copy.data.copy()
shape_copy.matrix_world = get_bone_matrix(bone)
mat = custom_shape_matrix(bone)
shape_copy.data.transform(mat)
render_widget(shape_copy, icon_path)
folder.add_widget(name)
bpy.data.objects.remove(shape_copy)
bpy.context.area.tag_redraw() bpy.context.area.tag_redraw()

View File

@ -19,7 +19,7 @@ from .shape_utils import symmetrize_bone_shape, get_flipped_bone
def undo_redo(self, context): def undo_redo(self, context):
wm = context.window_manager wm = context.window_manager
op_names = ('transform_widget','create_widget', 'match_transform', 'clean_widget') op_names = ('transform_widget','create_widget', 'match_transform', 'clean_widget')
op_ids = [ctx.get_op_id(o) for o in op_ids] op_ids = [ctx.get_op_id(o) for o in op_names]
if (ops and ops[-1].bl_idname in op_ids): if (ops and ops[-1].bl_idname in op_ids):
bpy.ops.wm.undo() bpy.ops.wm.undo()
@ -63,12 +63,15 @@ def rename_widget(self, context):
class BW_PG_widget(PropertyGroup): class BW_PG_widget(PropertyGroup):
name: StringProperty(update=rename_widget) name: StringProperty(update=rename_widget)
source_bone: StringProperty()
#path: StringProperty(subtype='FILE_PATH') #path: StringProperty(subtype='FILE_PATH')
def refresh(self, context):
bpy.ops.bonewidget.refresh_folders()
class BW_PG_folder(PropertyGroup): class BW_PG_folder(PropertyGroup):
icons = bpy.utils.previews.new() icons = bpy.utils.previews.new()
path : StringProperty(subtype='FILE_PATH', default='Default') path: StringProperty(subtype='FILE_PATH', default='Default', update=refresh)
expand: BoolProperty() expand: BoolProperty()
widgets: CollectionProperty(type=BW_PG_widget) widgets: CollectionProperty(type=BW_PG_widget)
widget_index: IntProperty() widget_index: IntProperty()
@ -82,28 +85,33 @@ class BW_PG_folder(PropertyGroup):
return Path(self.path).name return Path(self.path).name
def load_icon(self, widget): def load_icon(self, widget):
icon = self.get_widget_icon(widget)
icon_name = self.get_icon_name(widget)
icon = self.icons.get(icon_name)
if icon: if icon:
icon.reload() icon.reload()
else: else:
icon_path = self.get_icon_path(widget) icon_path = self.get_icon_path(widget)
self.icons.load(widget.name, str(icon_path), 'IMAGE', True) self.icons.load(icon_name, str(icon_path), 'IMAGE', True)
def get_icon_name(self, widget):
return f'{Path(self.path).stem}_{widget.name}'
def get_widget_icon(self, widget): def get_widget_icon(self, widget):
return self.icons.get(widget.name) return self.icons.get(self.get_icon_name(widget))
def add_widget(self, name): def add_widget(self, name):
name = self.get_widget_display_name(name) name = self.get_widget_display_name(name)
w = self.widgets.get(name) w = self.widgets.get(name)
if w: if w:
self.load_icon(w) self.load_icon(w)
self.active_widget = w
return return
w = self.widgets.add() w = self.widgets.add()
w.name = name w.name = name
self.active_widget = w self.active_widget = w
self.load_icon(w) self.load_icon(w)
return w return w
@ -198,8 +206,8 @@ class BW_prefs(AddonPreferences):
folders: CollectionProperty(type=BW_PG_folder) folders: CollectionProperty(type=BW_PG_folder)
folder_index: IntProperty() folder_index: IntProperty()
folder_enum: EnumProperty(name='Folders', items=lambda s,c: ctx.folder_items) folder_enum: EnumProperty(name='Folders', items=lambda s,c: ctx.folder_items)
collection : StringProperty(name='Collection', default='Widget') collection: StringProperty(name='Collection', default='Widgets', description='Name of the widget collection use {ob.name} to include the name of the rig')
prefix : StringProperty(name='Prefix', default='WGT-') prefix: StringProperty(name='Prefix', default='WGT-', description='Prefix for the shape object and data name')
category: StringProperty(name='Tab', default='Rigging', update=lambda s,c: update_tab()) category: StringProperty(name='Tab', default='Rigging', update=lambda s,c: update_tab())
symmetrize: BoolProperty(name='Symmetrize', default=True, update=undo_redo) symmetrize: BoolProperty(name='Symmetrize', default=True, update=undo_redo)
@ -210,7 +218,7 @@ class BW_prefs(AddonPreferences):
show_transforms: BoolProperty(default=False) show_transforms: BoolProperty(default=False)
transforms: PointerProperty(type=BW_PG_transforms) transforms: PointerProperty(type=BW_PG_transforms)
grid_view : BoolProperty(name='Grid View', default=True) #grid_view: BoolProperty(name='Grid View', default=True)
#use_custom_colors: BoolProperty(name='Custom Colors', default=False, update=set_default_colors) #use_custom_colors: BoolProperty(name='Custom Colors', default=False, update=set_default_colors)
@ -232,6 +240,8 @@ def register():
for cls in classes: for cls in classes:
bpy.utils.register_class(cls) bpy.utils.register_class(cls)
#bpy.types.Armature.widget_collection = PointerProperty(type=bpy.types.Collection)
#update_folder_items() #update_folder_items()
@ -240,6 +250,7 @@ def unregister():
#for f in ctx.folders: #for f in ctx.folders:
# bpy.utils.previews.remove(f.icons) # bpy.utils.previews.remove(f.icons)
#del bpy.types.Armature.widget_collection
for cls in classes: for cls in classes:
bpy.utils.unregister_class(cls) bpy.utils.unregister_class(cls)

View File

@ -1,6 +1,7 @@
import bpy import bpy
from .transform_utils import get_bone_matrix, transform_matrix from .transform_utils import get_bone_matrix, transform_matrix
from mathutils import Matrix
#from bone_widget import ctx #from bone_widget import ctx
@ -73,26 +74,42 @@ def link_to_col(shape, col):
col.objects.link(shape) col.objects.link(shape)
def custom_shape_matrix(bone):
loc = bone.custom_shape_translation.copy()
rot = bone.custom_shape_rotation_euler.copy()
scale = bone.custom_shape_scale_xyz.copy()
if bone.use_custom_shape_bone_size:
loc /= bone.bone.length
else:
scale /= bone.bone.length
return Matrix.LocRotScale(loc, rot, scale)
def get_clean_shape(bone, shape, separate=True, rename=True, def get_clean_shape(bone, shape, separate=True, rename=True,
col=None, match=True, prefix='', apply_transforms=True): col=None, match=True, prefix='', apply_transforms=True):
old_shape = shape #bone.custom_shape = None
old_bone = get_bone(old_shape)
bone.custom_shape = None #Remove bone shape if no other bone are using it
if bone.custom_shape:
old_bones = get_bones(bone.custom_shape)
old_bones.remove(bone)
print('old_bones', old_bones)
if old_bones:
rename_shape(bone.custom_shape, old_bones[0], prefix=prefix)
elif bone.custom_shape != shape:
bpy.data.objects.remove(bone.custom_shape)
if separate: if separate:
if old_bone:
rename_shape(old_shape, old_bone, prefix=prefix)
else:
bpy.data.objects.remove(old_shape)
shape = shape.copy() shape = shape.copy()
shape.data = shape.data.copy() shape.data = shape.data.copy()
bone.custom_shape = shape bone.custom_shape = shape
if match:
shape.matrix_world = get_bone_matrix(bone)
if apply_transforms: if apply_transforms:
if bpy.app.version_string < '3.0.0': if bpy.app.version_string < '3.0.0':
@ -108,23 +125,19 @@ col=None, match=True, prefix='', apply_transforms=True):
bone.custom_shape_scale = 1 bone.custom_shape_scale = 1
#mirror_bone.custom_shape_scale = #mirror_bone.custom_shape_scale =
else: else:
loc = bone.custom_shape_translation
rot = bone.custom_shape_rotation_euler
scale = bone.custom_shape_scale_xyz
if not bone.use_custom_shape_bone_size:
scale /= bone.bone.length
mat = transform_matrix(loc=loc, rot=rot, scale=scale)
mat = custom_shape_matrix(bone)
shape.data.transform(mat) shape.data.transform(mat)
bone.custom_shape_translation = 0, 0, 0 bone.custom_shape_translation = (0, 0, 0)
bone.custom_shape_rotation_euler = 0, 0, 0 bone.custom_shape_rotation_euler = (0, 0, 0)
bone.custom_shape_scale_xyz = 1, 1, 1 bone.custom_shape_scale_xyz = (1, 1, 1)
bone.use_custom_shape_bone_size = True bone.use_custom_shape_bone_size = True
if match:
shape.matrix_world = get_bone_matrix(bone)
if rename: if rename:
rename_shape(shape, bone, prefix=prefix) rename_shape(shape, bone, prefix=prefix)
@ -134,10 +147,14 @@ col=None, match=True, prefix='', apply_transforms=True):
return shape return shape
def get_bone(shape): def get_bones(shape):
armatures = [o for o in bpy.context.scene.objects if o.type == 'ARMATURE'] armatures = [o for o in bpy.context.scene.objects if o.type == 'ARMATURE']
return next((b for a in armatures for b in a.pose.bones if b.custom_shape == shape), None) return [b for a in armatures for b in a.pose.bones if b.custom_shape == shape]
def get_bone(shape):
bones = get_bones(shape)
if bones:
return bones[0]
def symmetrize_bone_shape(bone, prefix=None): def symmetrize_bone_shape(bone, prefix=None):
active_side = get_side(bone.name, 'LEFT') active_side = get_side(bone.name, 'LEFT')

45
ui.py
View File

@ -59,24 +59,26 @@ def add_color_row(layout, data=None, name='', index=0):
''' '''
def draw_prefs(layout): def draw_prefs(layout):
layout = layout.column(align=False)
add_row(layout, 'category', name='Tab', icon='COPY_ID') box = layout.box()
add_row(layout, 'collection', name='Collection', icon='OUTLINER_COLLECTION') col = box.column(align=False)
add_row(layout, 'prefix', name='Prefix', icon='SYNTAX_OFF')
layout.separator() add_row(col, 'category', name='Tab', icon='COPY_ID')
add_row(col, 'collection', name='Collection', icon='OUTLINER_COLLECTION')
add_row(col, 'prefix', name='Prefix', icon='SYNTAX_OFF')
add_row(layout, 'path', data=ctx.prefs.default_folder, name='Folders', col.separator()
add_row(col, 'path', data=ctx.prefs.default_folder, name='Folders',
icon='ADD', operator='bonewidget.add_folder') icon='ADD', operator='bonewidget.add_folder')
for i, f in enumerate(ctx.prefs.folders): for i, f in enumerate(ctx.prefs.folders):
add_row(layout, 'path', icon='REMOVE', add_row(col, 'path', icon='REMOVE',
operator='bonewidget.remove_folder', data=f, properties={'index': i}) operator='bonewidget.remove_folder', data=f, properties={'index': i})
layout.separator() col.separator()
split = layout.split(factor=0.33, align=True) split = col.split(factor=0.33, align=True)
split.alignment= 'RIGHT' split.alignment= 'RIGHT'
split.label(text='Auto:') split.label(text='Auto:')
@ -88,6 +90,10 @@ def draw_prefs(layout):
add_bool_row(col, 'match_transform', name='Match Transform', icon='TRANSFORM_ORIGINS') add_bool_row(col, 'match_transform', name='Match Transform', icon='TRANSFORM_ORIGINS')
add_bool_row(col, 'rename', name='Rename', icon='SYNTAX_OFF') add_bool_row(col, 'rename', name='Rename', icon='SYNTAX_OFF')
#bpy.ops.wm.save_userpref()
prefs_unsaved = bpy.context.preferences.is_dirty
layout.operator('wm.save_userpref', text="Save Preferences" + (" *" if prefs_unsaved else ""))
#col.prop(ctx.prefs, 'symmetrize', text='Symetrize', icon='MOD_MIRROR') #col.prop(ctx.prefs, 'symmetrize', text='Symetrize', icon='MOD_MIRROR')
#col.prop(ctx.prefs, 'separate', text='Separate', icon='UNLINKED') #col.prop(ctx.prefs, 'separate', text='Separate', icon='UNLINKED')
#col.prop(ctx.prefs, 'match_transform', text='Match Transform', icon='TRANSFORM_ORIGINS') #col.prop(ctx.prefs, 'match_transform', text='Match Transform', icon='TRANSFORM_ORIGINS')
@ -115,17 +121,14 @@ def draw_prefs(layout):
class BW_UL_widget(UIList): class BW_UL_widget(UIList):
def draw_item(self, context, layout, data, item, icon, active_data, active_propname, index, flt_flag): def draw_item(self, context, layout, data, item, icon, active_data, active_propname, index, flt_flag):
icon = data.icons.get(item.name) icon = data.get_widget_icon(item)
icon_id = icon.icon_id if icon else -1 icon_id = icon.icon_id if icon else -1
#layout.scale_x = 1.5 #layout.scale_x = 1.5
row = layout.row(align=True)
row.use_property_split = False
if self.layout_type in {'DEFAULT', 'COMPACT'}: row.prop(item, 'name', text="", emboss=False, icon_value=icon_id)
layout.prop(item, 'name', text="", emboss=False, icon_value=icon_id)
elif self.layout_type in {'GRID'}:
layout.alignment = 'CENTER'
layout.label(text="", icon_value=icon_id)
class BW_MT_folder(Menu): class BW_MT_folder(Menu):
@ -142,7 +145,7 @@ class BW_MT_folder(Menu):
layout.operator('bonewidget.copy_widgets', icon='COPYDOWN') layout.operator('bonewidget.copy_widgets', icon='COPYDOWN')
layout.operator('bonewidget.paste_widgets', icon='PASTEDOWN') layout.operator('bonewidget.paste_widgets', icon='PASTEDOWN')
layout.prop(ctx.prefs, 'grid_view') #layout.prop(ctx.prefs, 'grid_view')
class BW_PT_transforms(Panel): class BW_PT_transforms(Panel):
@ -231,7 +234,7 @@ class BW_PT_main(Panel):
if folder: if folder:
widget_col.template_list('BW_UL_widget', 'BW_widgets', folder, 'widgets', folder, 'widget_index', widget_col.template_list('BW_UL_widget', 'BW_widgets', folder, 'widgets', folder, 'widget_index',
rows=5, columns=self.get_nb_col(context), type='GRID' if ctx.prefs.grid_view else 'DEFAULT') rows=5, columns=self.get_nb_col(context), type='DEFAULT')
#widget_col.template_list('UI_UL_list', 'BW_widgets', folder, 'widgets', folder, 'widget_index', rows=4) #widget_col.template_list('UI_UL_list', 'BW_widgets', folder, 'widgets', folder, 'widget_index', rows=4)
#layout.prop(self, 'folder_enum') #layout.prop(self, 'folder_enum')
@ -267,13 +270,9 @@ class BW_PT_main(Panel):
#opt_col = layout.column() #opt_col = layout.column()
elif ctx.show_prefs_op: elif ctx.show_prefs_op:
layout.separator() #layout.separator()
draw_prefs(layout) draw_prefs(layout)
#bpy.ops.wm.save_userpref()
prefs_unsaved = context.preferences.is_dirty
layout.operator('wm.save_userpref', text="Save Preferences" + (" *" if prefs_unsaved else ""))
classes = ( classes = (
BW_UL_widget, BW_UL_widget,

BIN
widgets/Custom/hips.blend Normal file

Binary file not shown.

BIN
widgets/Custom/hips.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.6 KiB

BIN
widgets/Custom/root.blend Normal file

Binary file not shown.

BIN
widgets/Custom/root.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.5 KiB

BIN
widgets/Custome/root.blend Normal file

Binary file not shown.

BIN
widgets/Custome/root.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.3 KiB

Binary file not shown.

BIN
widgets/Default/bone_l.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 624 B

BIN
widgets/Default/chest.blend Normal file

Binary file not shown.

BIN
widgets/Default/chest.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.8 KiB

Binary file not shown.

BIN
widgets/Default/pelvis.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.7 KiB

Binary file not shown.

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.5 KiB

After

Width:  |  Height:  |  Size: 723 B

Binary file not shown.

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.3 KiB

BIN
widgets/Default/torso.blend Normal file

Binary file not shown.

BIN
widgets/Default/torso.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.3 KiB