rig_picker/core/picker.py

608 lines
18 KiB
Python
Raw Normal View History

2022-04-06 10:12:32 +02:00
import bpy
import gpu
from gpu_extras.batch import batch_for_shader
2023-11-09 10:29:56 +01:00
import blf
2022-04-06 10:12:32 +02:00
from mathutils import bvhtree, Vector
from mathutils.geometry import intersect_point_quad_2d, intersect_point_tri_2d, intersect_tri_tri_2d
2023-11-09 10:29:56 +01:00
from ..constants import PICKERS
from .addon_utils import get_operator_from_id
2022-04-06 10:12:32 +02:00
from pathlib import Path
import re
import json
2023-11-09 10:29:56 +01:00
import os
2022-04-06 10:12:32 +02:00
import threading
class Shape:
2023-03-31 14:53:41 +02:00
def __init__(self, picker, points, polygons=None, edges=None, tooltip='', color=None, source_name=''):
2022-04-06 10:12:32 +02:00
self.type = 'display'
self.picker = picker
self.rig = picker.rig
2023-03-31 14:53:41 +02:00
self.source_name = source_name
2022-04-06 10:12:32 +02:00
self.hover = False
self.press = False
2023-11-21 11:40:15 +01:00
self.shader = gpu.shader.from_builtin('UNIFORM_COLOR')
2022-04-06 10:12:32 +02:00
2023-11-21 11:40:15 +01:00
#self.hover_shader = gpu.shader.from_builtin('UNIFORM_COLOR')
2022-04-06 10:12:32 +02:00
#self.hover_shader.uniform_float("color", [1, 1, 1, 0.1])
self.hover_color = [1, 1, 1, 0.1]
self.tooltip = tooltip
self.color = color
self.points = points
self.polygons = polygons or []
self.edges = edges or []
#print(points, self.polygons)
self.p_batch = batch_for_shader(self.shader, 'TRIS', {"pos": self.points}, indices=self.polygons)
self.e_batch = batch_for_shader(self.shader, 'LINES', {"pos": self.points}, indices=self.edges)
#if polygons:
# self.batch = batch_for_shader(self.shader, 'TRIS', {"pos": points}, indices=polygons)
#else:
#pts = []
#for loop in self.edges:
# pts += [self.points[i] for i in loop]
# self.batch = batch_for_shader(self.shader, 'LINES', {"pos": points}, indices=indices)
points_x = [v[0] for v in points]
points_y = [v[1] for v in points]
self.bound = [
(min(points_x), max(points_y)),
(max(points_x), max(points_y)),
(max(points_x), min(points_y)),
(min(points_x), min(points_y))
]
@property
def color(self):
return self._color
@color.setter
def color(self, color=None):
if not color:
color = [0.5, 0.5, 0.5, 1]
elif isinstance(color, (tuple, list)) and len(color) in (3, 4):
if len(color) == 3:
color = [*color, 1]
elif len(color) == 4:
color = list(color)
else:
raise Exception('color must have a len of 3 or 4')
else:
raise Exception(f'color is {type(color)} must be None or (tuple, list)')
#self.shader.uniform_float("color", color)
self._color = color
def draw(self):
self.shader.bind()
self.shader.uniform_float("color", self.color)
if self.polygons:
self.p_batch.draw(self.shader)
if self.edges:
self.e_batch.draw(self.shader)
def move_event(self, location):
if not intersect_point_quad_2d(location, *self.bound):
self.hover = False
return False
for p in self.polygons:
if intersect_point_tri_2d(location, *[self.points[i] for i in p]):
self.hover = True
return True
self.hover = False
return False
def press_event(self, mode='SET'):
self.press = True
def release_event(self, mode='SET'):
self.press = False
class BoneShape(Shape):
2023-03-31 14:53:41 +02:00
def __init__(self, picker, points, polygons, edges, bone, tooltip='', color=None, source_name=''):
2022-04-06 10:12:32 +02:00
super().__init__(picker, points=points, polygons=polygons, edges=edges,
2023-03-31 14:53:41 +02:00
tooltip=tooltip, color=color, source_name=source_name)
2022-04-06 10:12:32 +02:00
self.type = 'bone'
self.bone = bone
self.active_color = [1, 1, 1, 0.1]
2023-11-21 11:40:15 +01:00
self.bone_colors = self.get_bone_colors()
2022-04-06 10:12:32 +02:00
@property
def select(self):
2023-03-31 14:53:41 +02:00
if not self.bone:
return False
2022-04-06 10:12:32 +02:00
return self.bone in (bpy.context.selected_pose_bones or []) #self.bone.bone.select
@property
def active(self):
2023-03-31 14:53:41 +02:00
if not self.bone:
return False
2022-04-06 10:12:32 +02:00
return self.bone == bpy.context.active_pose_bone #self.rig.data.bones.active == self.bone.bone
@property
def hide(self):
2023-03-31 14:53:41 +02:00
if not self.bone:
return False
2022-04-06 10:12:32 +02:00
#return self.bone not in (bpy.context.visible_pose_bones or [])
2023-11-21 11:40:15 +01:00
return self.bone.bone.hide or not any(l.is_visible for l in self.bone.bone.collections)
2022-04-06 10:12:32 +02:00
@property
def bone_color(self):
2023-03-31 14:53:41 +02:00
if not self.bone:
return [0, 0, 0, 1]
2022-04-06 10:12:32 +02:00
if self.select and self.active:
return self.bone_colors['active']
elif self.select:
return self.bone_colors['select']
elif self.hide:
return self.bone_colors['hide']
else:
return self.bone_colors['normal']
2023-11-21 11:40:15 +01:00
def get_bone_colors(self):
theme = bpy.context.preferences.themes['Default']
bone_colors = {
'select': [*theme.view_3d.bone_pose, 1],
'normal': [0.05, 0.05, 0.05, 1],
'active': [*theme.view_3d.bone_pose_active, 1],
'hide': [0.85, 0.85, 0.85, 0.2],
}
if not self.bone:
return bone_colors
if self.bone.color.palette == 'CUSTOM':
bone_color = self.bone
elif self.bone.color.palette == 'DEFAULT':
bone_color = self.bone.bone
normal_color = bone_color.color.custom.normal.copy()
normal_color.s *= 0.75
bone_colors['normal'] = [*normal_color, 1]
bone_colors['select'] = [*bone_color.color.custom.select, 1]
bone_colors['active'] = [*bone_color.color.custom.active, 1]
bone_colors['hide'] = [*normal_color, 0.1]
return bone_colors
2022-04-06 10:12:32 +02:00
def draw(self):
2023-10-03 11:49:21 +02:00
gpu.state.blend_set('ALPHA')
2022-04-06 10:12:32 +02:00
if self.hide:
self.shader.uniform_float("color", (*self.color[:3], 0.4))
self.p_batch.draw(self.shader)
#elif self.select:
# self.shader.uniform_float("color", self.bone_color)
# self.p_batch.draw(self.shader)
else:
super().draw()
2023-03-31 14:53:41 +02:00
# Overlay the fill slightly with the bone color
#self.shader.uniform_float("color", (*self.bone_color[:3], 0.1))
#self.p_batch.draw(self.shader)
2022-04-06 10:12:32 +02:00
if self.select:
color = self.hover_color
if self.select or self.hover:
color = self.hover_color
if self.select and self.hover:
color = self.active_color
self.shader.uniform_float("color", color)
self.p_batch.draw(self.shader)
2023-03-31 14:53:41 +02:00
#Overlay the fill slightly with the bone color
self.shader.uniform_float("color", (*self.bone_colors['normal'][:3], 0.1))
self.p_batch.draw(self.shader)
2022-04-06 10:12:32 +02:00
#self.contour_shader.bind()
#print(self.bone_color)
if self.select or self.active:
2023-10-03 11:49:21 +02:00
gpu.state.line_width_set(2.0)
2022-04-06 10:12:32 +02:00
#if not self.hide:
self.shader.uniform_float("color", self.bone_color)
#for b in self.contour_batches:
self.e_batch.draw(self.shader)
2023-10-03 11:49:21 +02:00
gpu.state.line_width_set(1.0)
gpu.state.blend_set('NONE')
2023-03-31 14:53:41 +02:00
def assign_bone_event(self):
#print('assign_bone_event', self)
scn = bpy.context.scene
rig = scn.rig_picker.rig
source_object = scn.objects.get(self.source_name)
if not source_object:
print(f'Source object {self.source_name} not found')
return
2022-04-06 10:12:32 +02:00
2023-03-31 14:53:41 +02:00
active_bone = rig.data.bones.active
if not active_bone:
print('You need to have an active bone')
return
source_object.rig_picker.name = rig.data.bones.active.name
2022-04-06 10:12:32 +02:00
def release_event(self, mode='SET'):
super().release_event(mode)
2023-03-31 14:53:41 +02:00
if self.hide or not self.bone:
2022-04-06 10:12:32 +02:00
return
select = True
if mode == 'SUBSTRACT':
select = False
2023-03-31 14:53:41 +02:00
self.bone.bone.select = select
2022-04-06 10:12:32 +02:00
if self.hover:
if mode != 'SUBSTRACT':
self.rig.data.bones.active = self.bone.bone
2023-03-31 14:53:41 +02:00
2022-04-06 10:12:32 +02:00
def border_select(self, border, mode='SET'):
'''
if ( not any(intersect_point_quad_2d(b, *self.bound) for b in border) and
not any(intersect_point_quad_2d(b, *border) for b in self.bound) ):
return
'''
2023-03-31 14:53:41 +02:00
if not self.bone:
return
2022-04-06 10:12:32 +02:00
if self.hide:
self.bone.bone.select = False
return
bound_tri1 = self.bound[0], self.bound[1], self.bound[2]
bound_tri2 = self.bound[2], self.bound[3], self.bound[0]
border_tri1 = border[0], border[1], border[2]
border_tri2 = border[2], border[3], border[0]
if (not intersect_tri_tri_2d(*border_tri1, *bound_tri1) and
2022-04-24 08:55:35 +02:00
not intersect_tri_tri_2d(*border_tri1, *bound_tri2) and
not intersect_tri_tri_2d(*border_tri2, *bound_tri1) and
not intersect_tri_tri_2d(*border_tri2, *bound_tri2)):
2022-04-06 10:12:32 +02:00
return
select = True
if mode == 'SUBSTRACT':
select = False
for polygon in self.polygons:
points = [self.points[i] for i in polygon]
if intersect_tri_tri_2d(*border_tri1, *points):
self.bone.bone.select = select
return
if intersect_tri_tri_2d(*border_tri2, *points):
self.bone.bone.select = select
return
'''
for b in border:
if intersect_point_tri_2d(b, *points):
self.bone.bone.select = select
return
for p in points:
if intersect_point_quad_2d(p, *border):
self.bone.bone.select = select
return
'''
class OperatorShape(Shape):
2023-03-31 14:53:41 +02:00
def __init__(self, picker, points, polygons, operator, tooltip='', color=None, source_name=''):
super().__init__(picker, points=points, polygons=polygons, tooltip=tooltip,
color=color, source_name=source_name)
2022-04-06 10:12:32 +02:00
self.type = 'operator'
self.active_color = [1, 1, 1, 0.15]
self.press_color = [0, 0, 0, 0.25]
self.operator = operator
#self.arguments = arguments#{k:eval(v)}
#self.operator = get_operator_from_id(self.operator)
2022-04-24 08:55:35 +02:00
if not tooltip:
self.tooltip = self.operator.replace('bpy.ops.', '').replace("'INVOKE_DEFAULT', ", '')
2022-04-06 10:12:32 +02:00
#self.reg_args = re.compile(r'(\w+)=')
'''
def parse_args(self):
args = self.reg_args.split(self.arguments)[1:]
#print(args, zip(args[::2], args[1::2]))
return {k: eval(v) for k, v in zip(args[::2], args[1::2])}
#return {k:eval(v) for k, v in self.reg_args.split(self.arguments)}
'''
def release_event(self, mode='SET'):
super().release_event(mode)
#args = self.parse_args()
if not self.operator:
return
exec(self.operator)
#f'bpy.ops;{idname}'
#print(self.idname)
#print(self.arguments)
#else:
# self.bone.bone.select = False
def draw(self):
super().draw()
if self.press:
color = self.press_color
elif self.hover:
color = self.hover_color
else:
return
2023-10-03 11:49:21 +02:00
gpu.state.blend_set('ALPHA')
2022-04-06 10:12:32 +02:00
self.shader.uniform_float("color", color)
self.p_batch.draw(self.shader)
2023-10-03 11:49:21 +02:00
gpu.state.blend_set('NONE')
2022-04-06 10:12:32 +02:00
class Picker:
def __init__(self, rig, shapes):
self.region = bpy.context.region
self.rig = rig
self.shapes = []
self.box_select = None
self.hover_shape = None
2023-11-21 11:40:15 +01:00
self.shader = gpu.shader.from_builtin('UNIFORM_COLOR')
2022-04-06 10:12:32 +02:00
self.tooltip_shape = None
self.tooltip_mouse = None
self.tooltip = ''
self.timer = None
self.mouse = None
for s in shapes:
if not s['points']:
continue
if s['type'] in ('CANVAS', 'DISPLAY'):
shape = Shape(
2023-03-31 14:53:41 +02:00
self,
2022-04-06 10:12:32 +02:00
points=s['points'],
polygons=s['polygons'],
edges=s['edges'],
color=s['color']
)
elif s['type'] == 'BONE':
2022-04-09 20:12:40 +02:00
bone = rig.pose.bones.get(s['bone'])
2023-03-31 14:53:41 +02:00
#if not bone:
# print(f'Bone {s["bone"]} not exist')
# continue
2022-04-06 10:12:32 +02:00
shape = BoneShape(
2023-03-31 14:53:41 +02:00
self,
source_name=s['source_name'],
2022-04-06 10:12:32 +02:00
points=s['points'],
polygons=s['polygons'],
edges=s['edges'],
2022-04-09 20:12:40 +02:00
bone=bone,
2022-04-06 10:12:32 +02:00
color=s['color']
)
elif s['type'] == 'OPERATOR':
shape = OperatorShape(
2023-03-31 14:53:41 +02:00
self,
source_name=s['source_name'],
2022-04-06 10:12:32 +02:00
points=s['points'],
polygons=s['polygons'],
operator=s['operator'],
2022-09-06 15:57:48 +02:00
color=s['color'],
tooltip=s['tooltip'],
2022-04-06 10:12:32 +02:00
)
self.shapes.append(shape)
2023-03-31 14:53:41 +02:00
def assign_bone_event(self):
for s in self.shapes:
if s.type=='bone' and s.hover:
s.assign_bone_event()
bpy.ops.rigpicker.save_picker()
2022-04-06 10:12:32 +02:00
def press_event(self, mode='SET'):
for s in self.shapes:
if s.hover:
s.press_event(mode)
else:
s.press = False
def release_event(self, mode='SET'):
if mode == 'SET':
for b in self.rig.pose.bones:
b.bone.select = False
#bpy.ops.pose.select_all(action='DESELECT')
#print('PICKER release event', mode)
#print(f'type={event.type}, value={event.value}, ctrl={event.ctrl}, shift={event.shift}, alt={event.alt}')
for s in self.shapes:
if s.hover:
s.release_event(mode)
s.press = False
2023-11-09 10:29:56 +01:00
#bpy.context.area.tag_redraw()
2022-04-06 10:12:32 +02:00
def tooltip_event(self):
2022-04-24 08:55:35 +02:00
#print('Tooltip Event', self)
2022-04-06 10:12:32 +02:00
2023-03-31 14:53:41 +02:00
#print(self.hover_shape, self.hover_shape.type)
2022-04-24 08:55:35 +02:00
if self.hover_shape and self.hover_shape.type != 'display':
2022-04-06 10:12:32 +02:00
if self.hover_shape.type == 'bone':
self.tooltip = self.hover_shape.bone.name
2022-04-24 08:55:35 +02:00
else:
self.tooltip = self.hover_shape.tooltip
self.tooltip_shape = self.hover_shape
2022-04-06 10:12:32 +02:00
else:
self.tooltip = ''
self.tooltip_shape = None
self.tooltip_mouse = self.mouse
self.timer.cancel()
2023-03-31 14:53:41 +02:00
#print(self.tooltip)
2022-04-06 10:12:32 +02:00
self.region.tag_redraw()
'''
picker.tooltip_event(event='SHOW')
region.tag_redraw()
#context.region.tag_redraw()
picker.tooltip_event(event='HIDE')
bpy.app.timers.register(partial(tooltip, context.region), first_interval=1)
'''
'''
def tooltip_event(self, event):
self.tooltip = ''
if event == 'SHOW':
if self.hover_shape.type == 'bone':
self.tooltip = self.hover_shape.bone.name
#bpy.context.region.tag_redraw()
'''
def border_select(self, border, mode):
border = [bpy.context.region.view2d.region_to_view(*b) for b in border]
if mode == 'SET':
for b in self.rig.pose.bones:
b.bone.select = False
for s in (s for s in self.shapes if s.type=='bone'):
s.border_select(border, mode)
def move_event(self, location):
self.mouse = location
location = self.region.view2d.region_to_view(*location)
2023-11-09 10:29:56 +01:00
self.hover_shape = None
for shape in reversed(self.shapes):
if self.hover_shape:
shape.hover = False
elif shape.move_event(location):
self.hover_shape = shape
2022-04-06 10:12:32 +02:00
#if point_inside_rectangle(self.end, bound):
# over = point_over_shape(self.end,points, edges)
#if bpy.app.timers.is_registered(self.tooltip_event):
#try:
# bpy.app.timers.unregister(self.tooltip_event)
#except:
# pass
if self.tooltip_shape is not self.hover_shape:
self.tooltip = ''
if self.timer:
self.timer.cancel()
self.timer = threading.Timer(0.5, self.tooltip_event)
self.timer.start()
#bpy.app.timers.register(self.tooltip_event, first_interval=1)
def draw(self):
for s in self.shapes:
s.draw()
'''
if self.box_select:
self.box_shader.uniform_float("color", self.box_select_color)
batch = batch_for_shader(self.shader, 'LINE_LOOP', {"pos": []})
for b in self.contour_batches:
b.draw(self.contour_shader)
self.batch.draw(self.shader)
2023-10-03 11:49:21 +02:00
gpu.state.blend_set('NONE')
2022-04-06 10:12:32 +02:00
'''
2023-11-09 10:29:56 +01:00
def get_picker_path(rig, start=None):
picker_path = rig.data.get('rig_picker', {}).get('source')
if not picker_path:
2022-04-06 10:12:32 +02:00
return
2023-11-09 10:29:56 +01:00
picker_path = bpy.path.abspath(picker_path, library=rig.data.library, start=start)
2022-04-12 18:14:56 +02:00
2023-11-09 10:29:56 +01:00
return Path(os.path.abspath(picker_path))
2022-04-12 18:14:56 +02:00
2023-11-09 10:29:56 +01:00
def pack_picker(rig, start=None):
picker_path = get_picker_path(rig, start=start)
if picker_path and picker_path.exists():
if 'rig_picker' not in rig.data.keys():
rig.data['rig_picker'] = {}
2022-04-06 10:12:32 +02:00
2023-11-09 10:29:56 +01:00
rig.data['rig_picker']['picker'] = json.loads(picker_path.read_text())
2022-04-06 10:12:32 +02:00
2023-11-09 10:29:56 +01:00
def unpack_picker(rig):
if 'rig_picker' not in rig.data.keys():
2022-04-06 10:12:32 +02:00
return
2023-11-09 10:29:56 +01:00
if 'picker' in rig.data['rig_picker'].keys():
del rig.data['rig_picker']['picker']