357 lines
14 KiB
Python
357 lines
14 KiB
Python
|
import bpy
|
|||
|
from pathlib import Path
|
|||
|
from .. import fn
|
|||
|
from .. constants import PREFIX
|
|||
|
from bpy.types import UIList, PropertyGroup, Menu
|
|||
|
from bpy.props import (PointerProperty,
|
|||
|
IntProperty,
|
|||
|
BoolProperty,
|
|||
|
StringProperty,
|
|||
|
EnumProperty,
|
|||
|
FloatProperty
|
|||
|
)
|
|||
|
|
|||
|
class BPM_UL_bg_list(UIList):
|
|||
|
# order_by_distance : BoolProperty(default=True)
|
|||
|
|
|||
|
show_items : EnumProperty(
|
|||
|
name="Show Items", description="Filter items to show, GP objects, Background or all",
|
|||
|
default='all', options={'HIDDEN'},
|
|||
|
items=(
|
|||
|
('all', 'All', 'Show Background and Gp object', '', 0),
|
|||
|
('obj', 'Gp Objects', 'Show only Gp object', '', 1),
|
|||
|
('bg', 'Backgrounds', 'Show only backgrounds', '', 2),
|
|||
|
))
|
|||
|
#(key, label, descr, id[, icon])
|
|||
|
|
|||
|
def draw_item(self, context, layout, data, item, icon, active_data, active_propname):
|
|||
|
# draw_item must handle the three layout types... Usually 'DEFAULT' and 'COMPACT' can share the same code.
|
|||
|
# layout.alignment = 'CENTER'
|
|||
|
|
|||
|
## TODO: Find a better solution to get the collection
|
|||
|
## Get Collection from plane name -> problem: prefix "BG_" and suffix
|
|||
|
# if not item.plane.get('is_background_holder'):
|
|||
|
if not item.plane.name.startswith(PREFIX): # (can check if has a parent using PREFIX)
|
|||
|
gp_ob = item.plane
|
|||
|
layout.prop(gp_ob, 'hide_viewport', text='', emboss=False, icon='HIDE_OFF')
|
|||
|
|
|||
|
icon_col = layout.row()
|
|||
|
icon_col.label(text='', icon='OUTLINER_OB_GREASEPENCIL') # BLANK1
|
|||
|
icon_col.ui_units_x = context.scene.bg_props.ui_list_scale # 1.6
|
|||
|
|
|||
|
row = layout.row()
|
|||
|
row.label(text=gp_ob.name)
|
|||
|
|
|||
|
if gp_ob.data.users > 1:
|
|||
|
row.template_ID(item, "data")
|
|||
|
else:
|
|||
|
row.label(text='', icon='BLANK1')
|
|||
|
|
|||
|
row.label(text='', icon='BLANK1')# <- add selection
|
|||
|
|
|||
|
return
|
|||
|
|
|||
|
layercol = context.view_layer.layer_collection.children['Background'].children.get(fn.clean_image_name(item.plane.name[len(PREFIX):]))
|
|||
|
if not layercol:
|
|||
|
layout.label(text=f'{item.plane.name} (problem with name)', icon='ERROR')
|
|||
|
return
|
|||
|
|
|||
|
# icon = 'HIDE_ON' if layercol.exclude else 'HIDE_OFF'
|
|||
|
layout.prop(layercol, 'exclude', text='', emboss=False, icon='HIDE_OFF')
|
|||
|
|
|||
|
if not item.plane.children:
|
|||
|
layout.label(text=f'{item.plane.name} (No children)', icon='ERROR')
|
|||
|
|
|||
|
## Image preview
|
|||
|
image = fn.get_image(item.plane.children[0])
|
|||
|
# layout.label(icon_value=image.preview_ensure().icon_id)
|
|||
|
# layout.template_icon(icon_value=image.preview_ensure().icon_id)
|
|||
|
|
|||
|
icon_col = layout.row()
|
|||
|
icon_col.template_icon(icon_value=image.preview_ensure().icon_id, scale=1.0)
|
|||
|
icon_col.ui_units_x = context.scene.bg_props.ui_list_scale # 1.6
|
|||
|
|
|||
|
## Name
|
|||
|
row = layout.row()
|
|||
|
row.enabled = not layercol.exclude
|
|||
|
|
|||
|
# row.label(text=item.plane.name) # <- Object has BG_ prefix, trim or use collection name
|
|||
|
row.label(text=layercol.name)
|
|||
|
|
|||
|
if context.scene.bg_props.show_distance:# and not layercol.exclude:
|
|||
|
row = layout.row()
|
|||
|
# row.enabled = not layercol.exclude
|
|||
|
row.prop(item.plane, '["distance"]', text='')
|
|||
|
# layout.prop(item.plane, 'location', index=2, text='') # , emboss=False
|
|||
|
|
|||
|
ob = context.object
|
|||
|
if ob and ob.parent == item.plane:
|
|||
|
layout.label(text='', icon='DECORATE_LINKED') # select from prop group
|
|||
|
else:
|
|||
|
layout.label(text='', icon='BLANK1') # select from prop group
|
|||
|
|
|||
|
if not layercol.exclude:
|
|||
|
icon = 'LAYER_ACTIVE' if item.plane.select_get() else 'LAYER_USED'
|
|||
|
layout.prop(item, 'select', icon=icon, text='', emboss=False) # select from prop group
|
|||
|
else:
|
|||
|
layout.label(text='', icon='BLANK1') # select from prop group
|
|||
|
|
|||
|
|
|||
|
|
|||
|
## note
|
|||
|
# You should always start your row layout by a label (icon + text), or a non-embossed text field,
|
|||
|
# this will also make the row easily selectable in the list! The later also enables ctrl-click rename.
|
|||
|
# We use icon_value of label, as our given icon is an integer value, not an enum ID.
|
|||
|
# Note "data" names should never be translated!
|
|||
|
|
|||
|
def draw_filter(self, context, layout):
|
|||
|
row = layout.row()
|
|||
|
|
|||
|
subrow = row.row(align=True)
|
|||
|
subrow.prop(self, "filter_name", text="") # Only show items matching this name (use ‘*’ as wildcard)
|
|||
|
|
|||
|
# invert result
|
|||
|
# icon = 'ZOOM_OUT' if self.use_filter_invert else 'ZOOM_IN'
|
|||
|
# subrow.prop(self, "use_filter_invert", text="", icon=icon)
|
|||
|
|
|||
|
# sort by name : ALPHA SORTING NOT WORKING, MUST CHANGE IN filter_items
|
|||
|
# subrow.prop(self, "use_filter_sort_alpha", text="", icon='SORTALPHA') # buit-in sort
|
|||
|
|
|||
|
subrow.prop(self, "show_items", text="") # type enum filter # icon='DOWNARROW_HLT'
|
|||
|
|
|||
|
# reverse order
|
|||
|
icon = 'SORT_DESC' if self.use_filter_sort_reverse else 'SORT_ASC'
|
|||
|
subrow.prop(self, "use_filter_sort_reverse", text="", icon=icon) # built-in reverse
|
|||
|
|
|||
|
def filter_items(self, context, data, propname):
|
|||
|
helpers = bpy.types.UI_UL_list
|
|||
|
items = getattr(data, propname)
|
|||
|
|
|||
|
filtered = []
|
|||
|
|
|||
|
# ordered = [items[:].index(i) for i in sorted(items[:], key=lambda o: o.plane.location.z)]
|
|||
|
items = [(items[:].index(i), i) for i in items[:]]
|
|||
|
|
|||
|
# needed ?
|
|||
|
filtered = [self.bitflag_filter_item] * len(items)
|
|||
|
|
|||
|
## Filter out out items thata
|
|||
|
for i, item in items:
|
|||
|
## GP/BG Type filter
|
|||
|
if self.show_items != 'all':
|
|||
|
if self.show_items != item.type:
|
|||
|
filtered[i] &= ~self.bitflag_filter_item
|
|||
|
## Search filter
|
|||
|
if item.plane and self.filter_name.lower() not in item.plane.name.lower():
|
|||
|
filtered[i] &= ~self.bitflag_filter_item
|
|||
|
|
|||
|
|
|||
|
# if self.order_by_distance:
|
|||
|
cam = context.scene.objects.get('bg_cam') or context.scene.camera
|
|||
|
if not cam:
|
|||
|
return filtered, ordered
|
|||
|
|
|||
|
## Real Distance from bg_cam or active camera
|
|||
|
ordered = helpers.sort_items_helper(items, lambda o: (o[1].plane.matrix_world.to_translation() - cam.matrix_world.to_translation()).length)
|
|||
|
## By distance attribute (only Backgrounds have distance attr)
|
|||
|
# ordered = helpers.sort_items_helper(items, lambda o: o[1].plane.get('distance'))
|
|||
|
|
|||
|
return filtered, ordered
|
|||
|
|
|||
|
'''
|
|||
|
class BPM_UL_bg_list(UIList):
|
|||
|
# order_by_distance : BoolProperty(default=True)
|
|||
|
|
|||
|
def draw_item(self, context, layout, data, item, icon, active_data, active_propname):
|
|||
|
# draw_item must handle the three layout types... Usually 'DEFAULT' and 'COMPACT' can share the same code.
|
|||
|
# layout.alignment = 'CENTER'
|
|||
|
|
|||
|
## TODO: Find a better solution to get the collection
|
|||
|
## Get Collection from plane name -> problem: prefix "BG_" and suffix
|
|||
|
|
|||
|
layercol = context.view_layer.layer_collection.children['Background'].children.get(fn.clean_image_name(item.plane.name[len(PREFIX):]))
|
|||
|
if not layercol:
|
|||
|
layout.label(text=f'{item.plane.name} (problem with name)', icon='ERROR')
|
|||
|
return
|
|||
|
|
|||
|
# icon = 'HIDE_ON' if layercol.exclude else 'HIDE_OFF'
|
|||
|
layout.prop(layercol, 'exclude', text='', emboss=False, icon='HIDE_OFF')
|
|||
|
|
|||
|
if not item.plane.children:
|
|||
|
layout.label(text=f'{item.plane.name} (No children)', icon='ERROR')
|
|||
|
|
|||
|
## Image preview
|
|||
|
image = fn.get_image(item.plane.children[0])
|
|||
|
# layout.label(icon_value=image.preview_ensure().icon_id)
|
|||
|
# layout.template_icon(icon_value=image.preview_ensure().icon_id)
|
|||
|
|
|||
|
icon_col = layout.row()
|
|||
|
icon_col.template_icon(icon_value=image.preview_ensure().icon_id, scale=1.0)
|
|||
|
icon_col.ui_units_x = context.scene.bg_props.ui_list_scale # 1.6
|
|||
|
|
|||
|
|
|||
|
## Name
|
|||
|
row = layout.row()
|
|||
|
row.enabled = not layercol.exclude
|
|||
|
|
|||
|
# row.label(text=item.plane.name) # <- Object has BG_ prefix, trim or use collection name
|
|||
|
row.label(text=layercol.name)
|
|||
|
|
|||
|
ob = context.object
|
|||
|
if ob and ob.parent == item.plane:
|
|||
|
layout.label(text='', icon='DECORATE_LINKED') # select from prop group
|
|||
|
else:
|
|||
|
layout.label(text='', icon='BLANK1') # select from prop group
|
|||
|
|
|||
|
if not layercol.exclude:
|
|||
|
icon = 'LAYER_ACTIVE' if item.plane.select_get() else 'LAYER_USED'
|
|||
|
layout.prop(item, 'select', icon=icon, text='', emboss=False) # select from prop group
|
|||
|
else:
|
|||
|
layout.label(text='', icon='BLANK1') # select from prop group
|
|||
|
|
|||
|
if context.scene.bg_props.show_distance:# and not layercol.exclude:
|
|||
|
row = layout.row()
|
|||
|
# row.enabled = not layercol.exclude
|
|||
|
row.prop(item.plane, '["distance"]', text='')
|
|||
|
# layout.prop(item.plane, 'location', index=2, text='') # , emboss=False
|
|||
|
|
|||
|
|
|||
|
## note
|
|||
|
# You should always start your row layout by a label (icon + text), or a non-embossed text field,
|
|||
|
# this will also make the row easily selectable in the list! The later also enables ctrl-click rename.
|
|||
|
# We use icon_value of label, as our given icon is an integer value, not an enum ID.
|
|||
|
# Note "data" names should never be translated!
|
|||
|
|
|||
|
|
|||
|
def filter_items(self, context, data, propname):
|
|||
|
helpers = bpy.types.UI_UL_list
|
|||
|
items = getattr(data, propname)
|
|||
|
|
|||
|
filtered = []
|
|||
|
|
|||
|
# ordered = [items[:].index(i) for i in sorted(items[:], key=lambda o: o.plane.location.z)]
|
|||
|
items = [(items[:].index(i), i) for i in items[:]]
|
|||
|
|
|||
|
# needed ?
|
|||
|
filtered = [self.bitflag_filter_item] * len(items)
|
|||
|
|
|||
|
for i, item in items:
|
|||
|
if item.plane and self.filter_name.lower() not in item.plane.name.lower():
|
|||
|
filtered[i] &= ~self.bitflag_filter_item
|
|||
|
#else:
|
|||
|
|
|||
|
|
|||
|
# if self.order_by_distance:
|
|||
|
|
|||
|
ordered = helpers.sort_items_helper(items, lambda o: o[1].plane.get('distance'))
|
|||
|
|
|||
|
return filtered, ordered
|
|||
|
|
|||
|
# def draw_filter(self, context, layout):
|
|||
|
# """UI code for the filtering/sorting/search area."""
|
|||
|
# col = layout.column(align=True)
|
|||
|
# row = col.row(align=True)
|
|||
|
|
|||
|
# # row.prop(self, 'order_by_distance', text='', icon='DRIVER_DISTANCE')
|
|||
|
# # row.prop(self, 'use_filter_invert', text='', icon='ARROW_LEFTRIGHT')
|
|||
|
'''
|
|||
|
|
|||
|
### --- updates
|
|||
|
|
|||
|
def get_plane_targets(context):
|
|||
|
selection = [i.plane for i in context.scene.bg_props.planes if i.plane.select_get()]
|
|||
|
if selection:
|
|||
|
return selection
|
|||
|
## active (even not selected)
|
|||
|
return [context.scene.bg_props.planes[context.scene.bg_props.index].plane]
|
|||
|
|
|||
|
def update_opacity(self, context):
|
|||
|
pool = get_plane_targets(context)
|
|||
|
for ob in pool:
|
|||
|
if not ob or not ob.children:
|
|||
|
continue
|
|||
|
fn.set_opacity(ob.children[0], opacity=context.scene.bg_props.opacity)
|
|||
|
|
|||
|
def update_on_index_change(self, context):
|
|||
|
# print('index change:', context.scene.bg_props.index)
|
|||
|
props = context.scene.bg_props
|
|||
|
planes_list = props.planes
|
|||
|
item = planes_list[props.index]
|
|||
|
if not item.plane:
|
|||
|
## remove slot !
|
|||
|
print(f'bg_props.planes: No plane/object at index {props.index}, removing item')
|
|||
|
planes_list.remove(props.index)
|
|||
|
return
|
|||
|
|
|||
|
if item.type == 'bg':
|
|||
|
plane = item.plane
|
|||
|
if not plane.children:
|
|||
|
return
|
|||
|
opacity = fn.get_opacity(plane.children[0])
|
|||
|
if not opacity:
|
|||
|
return
|
|||
|
props['opacity'] = opacity
|
|||
|
|
|||
|
elif item.type == 'obj':
|
|||
|
fn.gp_transfer_mode(item.plane, context)
|
|||
|
|
|||
|
|
|||
|
def update_select(self, context):
|
|||
|
# print('index change:', context.scene.bg_props.index)
|
|||
|
plane = self.plane
|
|||
|
plane.select_set(not plane.select_get())
|
|||
|
context.scene.bg_props['index'] = context.scene.bg_props.planes[:].index(self)
|
|||
|
|
|||
|
class BPM_more_option(Menu):
|
|||
|
bl_idname = "BPM_MT_more_option"
|
|||
|
bl_label = "Options"
|
|||
|
|
|||
|
def draw(self, context):
|
|||
|
layout = self.layout
|
|||
|
layout.operator("bpm.reload_list", text="Refresh", icon="FILE_REFRESH")
|
|||
|
layout.operator("bpm.open_bg_folder", icon="FILE_FOLDER")
|
|||
|
layout.operator("bpm.import_bg_images", text="Import Planes", icon="IMPORT")
|
|||
|
layout.operator("bpm.convert_planes", text="Convert Planes", icon="OUTLINER_OB_IMAGE")
|
|||
|
layout.separator()
|
|||
|
layout.prop(context.scene.bg_props, 'show_distance', icon='DRIVER_DISTANCE')
|
|||
|
layout.prop(context.scene.bg_props, 'ui_list_scale')
|
|||
|
|
|||
|
# Create sniptool property group
|
|||
|
class BPM_bg_list_prop(PropertyGroup):
|
|||
|
plane : PointerProperty(type=bpy.types.Object)
|
|||
|
select: BoolProperty(update=update_select) # use and update to set the plane selection
|
|||
|
type: StringProperty(default='bg')
|
|||
|
# is_bg: BoolProperty(default=True)
|
|||
|
|
|||
|
|
|||
|
class BPM_bg_settings(PropertyGroup):
|
|||
|
index : IntProperty(update=update_on_index_change)
|
|||
|
planes : bpy.props.CollectionProperty(type=BPM_bg_list_prop)
|
|||
|
show_distance : BoolProperty(name='Show Distance', default=False)
|
|||
|
opacity : FloatProperty(name='Opacity', default=1.0, min=0.0, max=1.0, update=update_opacity)
|
|||
|
move_hided : BoolProperty(name='Move hided', default=True)
|
|||
|
ui_list_scale : FloatProperty(name='UI Item Y Scale', default=1.6, min=0.6, soft_min=1.0, max=2.0)
|
|||
|
# distance : FloatProperty(name='Distance', default=0.0, update=update_distance)
|
|||
|
|
|||
|
|
|||
|
### --- REGISTER ---
|
|||
|
|
|||
|
classes=(
|
|||
|
# prop and UIlist
|
|||
|
BPM_bg_list_prop,
|
|||
|
BPM_bg_settings,
|
|||
|
BPM_UL_bg_list,
|
|||
|
BPM_more_option,
|
|||
|
)
|
|||
|
|
|||
|
|
|||
|
def register():
|
|||
|
for cls in classes:
|
|||
|
bpy.utils.register_class(cls)
|
|||
|
bpy.types.Scene.bg_props = bpy.props.PointerProperty(type=BPM_bg_settings)
|
|||
|
|
|||
|
def unregister():
|
|||
|
for cls in reversed(classes):
|
|||
|
bpy.utils.unregister_class(cls)
|
|||
|
del bpy.types.Scene.bg_props
|