809 lines
25 KiB
Python
809 lines
25 KiB
Python
# SPDX-License-Identifier: GPL-3.0-or-later
|
||
|
||
## Standalone based on SPA code
|
||
## Code from create_gp_texture_ref.py (Copyright (C) 2023, The SPA Studios. All rights reserved.)
|
||
## adapted to create as a single 'GP as image' and fit in camera with driver
|
||
|
||
import bpy
|
||
import os
|
||
import re
|
||
import time
|
||
from pathlib import Path
|
||
from mathutils import Vector, geometry
|
||
from pprint import pprint as pp
|
||
from .constants import PREFIX, BGCOL
|
||
|
||
def get_addon_prefs():
|
||
return bpy.context.preferences.addons[__package__].preferences
|
||
|
||
def norm_str(string, separator='_', format=str.lower, padding=0):
|
||
import unicodedata
|
||
string = str(string)
|
||
string = string.replace('_', ' ')
|
||
string = string.replace('-', ' ')
|
||
string = re.sub('[ ]+', ' ', string)
|
||
string = re.sub('[ ]+\/[ ]+', '/', string)
|
||
string = string.strip()
|
||
|
||
if format:
|
||
string = format(string)
|
||
|
||
# Padd rightest number
|
||
string = re.sub(r'(\d+)(?!.*\d)', lambda x : x.group(1).zfill(padding), string)
|
||
|
||
string = string.replace(' ', separator)
|
||
string = unicodedata.normalize('NFKD', string).encode('ASCII', 'ignore').decode("utf-8")
|
||
|
||
return string
|
||
|
||
def clean_image_name(name):
|
||
'''Strip blender numbering then strip extension'''
|
||
name = re.sub(r'\.\d{3}$', '', name.strip())
|
||
return os.path.splitext(name)[0].strip()
|
||
|
||
def get_image_infos_from_object(ob):
|
||
'''Get image filepath of a background whatever the type
|
||
return image_data, transparency'''
|
||
|
||
transparency = None
|
||
if ob.type == 'EMPTY':
|
||
image_data = ob.data
|
||
transparency = ob.color[3]
|
||
|
||
elif ob.type == 'MESH':
|
||
texture_plane_ng = bpy.data.node_groups.get('texture_plane')
|
||
if not texture_plane_ng:
|
||
print(f'{ob.name} : No texture plane nodegroup found !')
|
||
return
|
||
|
||
mat = ob.active_material
|
||
node = next((n for n in mat.node_tree.nodes
|
||
if n.type=='GROUP' and n.node_tree is texture_plane_ng), None)
|
||
if not node:
|
||
print(f'{ob.name} : No texture plane nodegroup in material {mat.name}')
|
||
return
|
||
|
||
## Get input texture
|
||
tex_node = next(n for n in mat.node_tree.nodes
|
||
if n.type=='TEX_IMAGE')
|
||
|
||
if not tex_node:
|
||
print(f'{ob.name} : No image texture node in material {mat.name}')
|
||
return
|
||
|
||
image_data = tex_node.image
|
||
transparency = node.inputs['Transparency'].default_value
|
||
|
||
elif ob.type == 'GPENCIL':
|
||
image_data = ob.data.materials[0].grease_pencil.fill_image
|
||
transparency = ob.data.layers[0].opacity
|
||
|
||
if not image_data:
|
||
return
|
||
|
||
return (image_data, transparency)
|
||
|
||
def get_image(ob):
|
||
'''Get image data from image object'''
|
||
infos = get_image_infos_from_object(ob)
|
||
if infos is None:
|
||
return
|
||
img, _opacity = infos
|
||
return img
|
||
|
||
def get_opacity(ob):
|
||
'''Get opacity from image object'''
|
||
infos = get_image_infos_from_object(ob)
|
||
if infos is None:
|
||
return
|
||
_img, opacity = infos
|
||
return opacity
|
||
|
||
def set_opacity(ob, opacity):
|
||
'''Set opacity of BG image object whatever the type'''
|
||
if ob.type == 'EMPTY':
|
||
ob.color[3] = opacity
|
||
|
||
elif ob.type == 'MESH':
|
||
texture_plane_ng = bpy.data.node_groups.get('texture_plane')
|
||
if texture_plane_ng:
|
||
node = next(n for n in ob.active_material.node_tree.nodes
|
||
if n.type == 'GROUP' and n.node_tree is texture_plane_ng)
|
||
node.inputs['Transparency'].default_value = opacity
|
||
|
||
elif ob.type == 'GPENCIL':
|
||
ob.data.layers[0].opacity = opacity
|
||
|
||
def create_plane_holder(img, name=None, parent=None, link_in_col=None):
|
||
'''
|
||
img :: blender image or path to an image
|
||
prefix :: name holder with image name with given prefix
|
||
parent :: object to parent to (only if passed)
|
||
link_in_col :: collection to link after creation
|
||
|
||
Return created object mesh without face
|
||
None if creation failed
|
||
'''
|
||
## get image if a path is passed in
|
||
if not isinstance(img, bpy.types.Image):
|
||
try:
|
||
# load from path (str)
|
||
img = bpy.data.images.load(str(img), check_existing=True)
|
||
except:
|
||
return
|
||
|
||
# name = img.name.split('.')[0]
|
||
name = name or clean_image_name(img.name)
|
||
obj_name = name # PREFIX + name # <- always set prefix ?
|
||
x,y = img.size[:]
|
||
|
||
# create and place object plane
|
||
# find height from width (as 1)
|
||
|
||
if y < x:
|
||
h = y/x
|
||
w = 1
|
||
else:
|
||
h = 1
|
||
w = x/y
|
||
# size/2 to get center image (half width | height)
|
||
h = h/2
|
||
w = w/2
|
||
|
||
verts = [
|
||
(-w, h, 0),#2
|
||
(w, h, 0),#3
|
||
(w, -h, 0),#4
|
||
(-w, -h, 0),#1
|
||
]
|
||
|
||
mesh = bpy.data.meshes.new(obj_name)
|
||
mesh.from_pydata(verts, [(3,2), (2,1), (1,0), (0,3)], [])
|
||
holder = bpy.data.objects.new(obj_name, mesh)
|
||
holder.lock_location = (True, True, False)
|
||
holder.hide_render = True
|
||
|
||
if link_in_col is not None:
|
||
link_in_col.objects.link(holder)
|
||
# bpy.context.scene.collection.objects.link(holder)
|
||
|
||
if parent:
|
||
holder.parent = parent
|
||
return holder
|
||
|
||
## Remove func is no intermediate collection created
|
||
def get_col(name, parent=None, create=True):
|
||
parent = parent or bpy.context.scene.collection
|
||
col = bpy.data.collections.get(name)
|
||
if not col and create:
|
||
col = bpy.data.collections.new(name)
|
||
|
||
if col not in parent.children[:]:
|
||
parent.children.link(col)
|
||
|
||
return col
|
||
|
||
def create_cam(name, type='ORTHO', size=6, focale=50):
|
||
'''Create an orthographic camera with given scale
|
||
size :: othographic scale of the camera
|
||
Return camera object
|
||
'''
|
||
|
||
cam_data = bpy.data.cameras.new(name)
|
||
cam_data.type = type
|
||
cam_data.clip_start = 0.02
|
||
cam_data.clip_end = 2000
|
||
cam_data.ortho_scale = size
|
||
cam_data.lens = focale
|
||
cam = bpy.data.objects.new(name, cam_data)
|
||
cam.rotation_euler.x = 1.5707963705062866
|
||
cam.location = (0, -8, 0)
|
||
bpy.context.scene.collection.objects.link(cam)
|
||
return cam
|
||
|
||
def link_nodegroup(filepath, group_name, link=True, keep_existing=False):
|
||
ng = bpy.data.node_groups.get(group_name)
|
||
if ng:
|
||
if keep_existing:
|
||
return ng
|
||
## risk deleting nodegroup from other object !
|
||
# else:
|
||
# bpy.data.node_groups.remove(ng)
|
||
|
||
lib_path = bpy.path.relpath(str(filepath))
|
||
with bpy.data.libraries.load(lib_path, link=link) as (data_from, data_to):
|
||
data_to.node_groups = [n for n in data_from.node_groups if n.startswith(group_name)]
|
||
|
||
group = bpy.data.node_groups.get(group_name)
|
||
if group:
|
||
return group
|
||
|
||
def create_plane_driver(plane, cam, distance=None):
|
||
|
||
# Multiple planes spacing
|
||
plane.lock_location = (True, True, False)
|
||
plane.lock_rotation = (True,)*3
|
||
plane.lock_scale = (True,)*3
|
||
|
||
plane['scale'] = 1.0
|
||
plane['distance'] = distance if distance is not None else 9.0
|
||
|
||
# DRIVERS
|
||
## LOC X AND Y (shift) ##
|
||
for axis in range(2):
|
||
driver = plane.driver_add('scale', axis)
|
||
|
||
# Driver type
|
||
driver.driver.type = 'SCRIPTED'
|
||
|
||
# Variable DISTANCE
|
||
var = driver.driver.variables.new()
|
||
var.name = "distance"
|
||
#-# use a propertie to drive distance to camera
|
||
var.type = 'SINGLE_PROP'
|
||
var.targets[0].id = plane
|
||
var.targets[0].data_path = '["distance"]'
|
||
#-# Can use directly object location (but give refresh problem)
|
||
# var.type = 'TRANSFORMS'
|
||
# var.targets[0].id = plane
|
||
# var.targets[0].transform_type = 'LOC_Z'
|
||
# var.targets[0].transform_space = 'LOCAL_SPACE'
|
||
|
||
# Variable FOV
|
||
var = driver.driver.variables.new()
|
||
var.name = "FOV"
|
||
var.type = 'SINGLE_PROP'
|
||
var.targets[0].id_type = "OBJECT"
|
||
var.targets[0].id = cam
|
||
var.targets[0].data_path = 'data.angle'
|
||
|
||
# Variable scale
|
||
var = driver.driver.variables.new()
|
||
var.name = "scale"
|
||
var.type = 'SINGLE_PROP'
|
||
var.targets[0].id = plane
|
||
var.targets[0].data_path = '["scale"]'
|
||
|
||
# Expression
|
||
# -distance*tan(FOV/2)
|
||
driver.driver.expression = \
|
||
"tan(FOV/2) * distance * scale * 2"
|
||
# "tan(FOV/2) * -distance * scale * 2" # inverse dist if based on locZ
|
||
|
||
# Driver for location
|
||
dist_drv = plane.driver_add('location', 2)
|
||
dist_drv.driver.type = 'SCRIPTED' #'AVERAGE' #need to be inverted
|
||
var = dist_drv.driver.variables.new()
|
||
var.name = "distance"
|
||
var.type = 'SINGLE_PROP'
|
||
var.targets[0].id = plane
|
||
var.targets[0].data_path = '["distance"]'
|
||
dist_drv.driver.expression = '-distance'
|
||
|
||
return driver, dist_drv
|
||
|
||
def set_collection(ob, collection, unlink=True) :
|
||
''' link an object in a collection and create it if necessary
|
||
if unlink object is removed from other collections
|
||
|
||
return collection (get/created)
|
||
'''
|
||
|
||
scn = bpy.context.scene
|
||
col = None
|
||
visible = False
|
||
linked = False
|
||
|
||
# check if collection exist or create it
|
||
for c in bpy.data.collections :
|
||
if c.name == collection : col = c
|
||
if not col :
|
||
col = bpy.data.collections.new(name=collection)
|
||
|
||
# link the collection to the scene's collection if necessary
|
||
# for c in scn.collection.children :
|
||
# if c.name == col.name : visible = True
|
||
# if not visible : scn.collection.children.link(col)
|
||
|
||
# check if the object is already in the collection and link it if necessary
|
||
for o in col.objects :
|
||
if o == ob : linked = True
|
||
if not linked : col.objects.link(ob)
|
||
|
||
# remove object from scene's collection
|
||
for o in scn.collection.objects :
|
||
if o == ob : scn.collection.objects.unlink(ob)
|
||
|
||
# if unlink flag we remove the object from other collections
|
||
if unlink :
|
||
for c in ob.users_collection :
|
||
if c.name != collection : c.objects.unlink(ob)
|
||
|
||
return col
|
||
|
||
def placeholder_name(name='', context=None) -> str:
|
||
|
||
# def increment(match):
|
||
# return str(int(match.group(1))+1).zfill(len(match.group(1)))
|
||
|
||
context = context or bpy.context
|
||
name = name.strip()
|
||
if not name:
|
||
# Create a default name (fing last 3 number, before blender increment if exists, default increment)
|
||
numbers = [int(match.group(1)) for o in context.scene.objects\
|
||
if (match := re.search(r'^drawing.*_(\d+)(?:\.\d{3})?$', o.name, flags=re.I))\
|
||
# and o.type == 'GPENCIL'
|
||
]
|
||
if numbers:
|
||
numbers.sort()
|
||
name = f'drawing_{numbers[-1] + 1:03d}'
|
||
else:
|
||
name = 'drawing_001'
|
||
|
||
# elif context.scene.objects.get(name):
|
||
# name = re.sub(r'(\d+)(?!.*\d)', increment, name) # find rightmost number
|
||
|
||
return name
|
||
|
||
def reset_gp_uv(ob):
|
||
uvs = [(0.5, 0.5), (-0.5, 0.5), (-0.5, -0.5), (0.5, -0.5)]
|
||
try:
|
||
for p, uv in zip(ob.data.layers[0].frames[0].strokes[0].points, uvs):
|
||
p.uv_fill = uv
|
||
except:
|
||
print('Could not set Gp points UV')
|
||
|
||
## Get empty coords
|
||
def get_ref_object_space_coord(o):
|
||
size = o.empty_display_size
|
||
x,y = o.empty_image_offset
|
||
img = o.data
|
||
|
||
res_x, res_y = img.size
|
||
scaling = 1 / max(res_x, res_y)
|
||
|
||
# 3----2
|
||
# | |
|
||
# 0----1
|
||
|
||
corners = [
|
||
Vector((0,0)),
|
||
Vector((res_x, 0)),
|
||
Vector((0, res_y)),
|
||
Vector((res_x, res_y)),
|
||
]
|
||
|
||
obj_space_corners = []
|
||
for co in corners:
|
||
nco_x = ((co.x + (x * res_x)) * size) * scaling
|
||
nco_y = ((co.y + (y * res_y)) * size) * scaling
|
||
obj_space_corners.append(Vector((nco_x, nco_y, 0)))
|
||
return obj_space_corners
|
||
|
||
|
||
## Generate GP object
|
||
def create_gpencil_reference(
|
||
gpd: bpy.types.GreasePencil,
|
||
gpf: bpy.types.GPencilFrame,
|
||
image: bpy.types.Image,
|
||
) -> bpy.types.GPencilStroke:
|
||
"""
|
||
Add a rectangular stroke textured with `image` to the given grease pencil fame.
|
||
:param gpd: The grease pencil data.
|
||
:param gpf: The grease pencil frame.
|
||
:param image: The image to use as texture.
|
||
:return: The created grease pencil stroke.
|
||
"""
|
||
name = clean_image_name(image.name) + '_texgp'
|
||
|
||
# Create new material
|
||
mat = bpy.data.materials.new(f".ref/{name}")
|
||
bpy.data.materials.create_gpencil_data(mat)
|
||
gpd.materials.append(mat)
|
||
idx = gpd.materials.find(mat.name)
|
||
|
||
# Setup material settings
|
||
mat.grease_pencil.show_stroke = False
|
||
mat.grease_pencil.show_fill = True
|
||
mat.grease_pencil.fill_image = image
|
||
mat.grease_pencil.fill_style = "TEXTURE"
|
||
mat.grease_pencil.mix_factor = 0.0
|
||
mat.grease_pencil.texture_offset = (0.0, 0.0)
|
||
mat.grease_pencil.texture_angle = 0.0
|
||
mat.grease_pencil.texture_scale = (1.0, 1.0)
|
||
mat.grease_pencil.texture_clamp = True
|
||
|
||
# Create the stroke
|
||
gps_new = gpf.strokes.new()
|
||
gps_new.points.add(4, pressure=0, strength=0)
|
||
|
||
## This will make sure that the uv's always remain the same
|
||
# gps_new.use_automatic_uvs = False # <<- /!\ exists only in SPA core changes
|
||
|
||
gps_new.use_cyclic = True
|
||
gps_new.material_index = idx
|
||
|
||
x,y = image.size[:]
|
||
|
||
# create and place object plane
|
||
# find height from width (as 1)
|
||
if y < x:
|
||
h = y/x
|
||
w = 1
|
||
else:
|
||
h = 1
|
||
w = x/y
|
||
# size/2 to get center image (half width | height)
|
||
h = h/2
|
||
w = w/2
|
||
|
||
coords = [
|
||
(w, h, 0),
|
||
(-w, h, 0),
|
||
(-w, -h, 0),
|
||
(w, -h, 0),
|
||
]
|
||
|
||
uvs = [(0.5, 0.5), (-0.5, 0.5), (-0.5, -0.5), (0.5, -0.5)]
|
||
|
||
for i, (co, uv) in enumerate(zip(coords, uvs)):
|
||
gps_new.points[i].co = co
|
||
gps_new.points[i].uv_fill = uv
|
||
|
||
return gps_new
|
||
|
||
def import_image_as_gp_reference(
|
||
context: bpy.types.Context,
|
||
image,
|
||
pack_image: bool = False,
|
||
):
|
||
"""
|
||
Import image from `image` as a textured rectangle in the given
|
||
grease pencil object.
|
||
:param context: The active context.
|
||
:param image: The image filepath or image datablock
|
||
:param pack_image: Whether to pack the image into the Blender file.
|
||
"""
|
||
|
||
if not image:
|
||
return
|
||
|
||
scene = context.scene
|
||
if not isinstance(image, bpy.types.Image):
|
||
image = bpy.data.images.load(str(image), check_existing=True)
|
||
|
||
## Create grease pencil object
|
||
gpd = bpy.data.grease_pencils.new(image.name)
|
||
ob = bpy.data.objects.new(image.name, gpd)
|
||
gpl = gpd.layers.new(image.name)
|
||
gpf = gpl.frames.new(0) # TDOO: test negative
|
||
|
||
if pack_image:
|
||
image.pack()
|
||
|
||
gps: bpy.types.GPencilStroke = create_gpencil_reference(
|
||
gpd,
|
||
gpf,
|
||
image,
|
||
)
|
||
|
||
gps.select = True # Not needed
|
||
|
||
cam = scene.camera # should be bg_cam
|
||
|
||
# Create the driver and parent
|
||
ob.use_grease_pencil_lights = False
|
||
ob['is_background'] = True
|
||
print(f'Created {ob.name} GP background object')
|
||
return ob
|
||
|
||
def gp_transfer_mode(ob, context=None):
|
||
context= context or bpy.context
|
||
if ob.type != 'GPENCIL' or context.object is ob:
|
||
return
|
||
|
||
prev_mode = context.mode
|
||
possible_gp_mods = ('OBJECT',
|
||
'EDIT_GPENCIL', 'SCULPT_GPENCIL', 'PAINT_GPENCIL',
|
||
'WEIGHT_GPENCIL', 'VERTEX_GPENCIL')
|
||
|
||
if prev_mode not in possible_gp_mods:
|
||
prev_mode = None
|
||
|
||
mode_swap = False
|
||
|
||
## TODO optional: Option to stop mode sync ?
|
||
## Set in same mode as previous object
|
||
if context.scene.tool_settings.lock_object_mode:
|
||
if context.mode != 'OBJECT':
|
||
mode_swap = True
|
||
bpy.ops.object.mode_set(mode='OBJECT')
|
||
|
||
# set active
|
||
context.view_layer.objects.active = ob
|
||
|
||
## keep same mode accross objects
|
||
if mode_swap and prev_mode is not None:
|
||
bpy.ops.object.mode_set(mode=prev_mode)
|
||
|
||
else:
|
||
## keep same mode accross objects
|
||
context.view_layer.objects.active = ob
|
||
if context.mode != prev_mode is not None:
|
||
bpy.ops.object.mode_set(mode=prev_mode)
|
||
|
||
for o in [o for o in context.scene.objects if o.type == 'GPENCIL']:
|
||
o.select_set(o == ob) # select only active (when not in object mode)
|
||
|
||
## generate tex plane
|
||
# def create_image_plane(coords, name):
|
||
# '''Create an a mesh plane with a defaut UVmap from passed coordinate
|
||
# object and mesh get passed name
|
||
|
||
# return plane object
|
||
# '''
|
||
# fac = [(0, 1, 3, 2)]
|
||
# me = bpy.data.meshes.new(name)
|
||
# me.from_pydata(coords, [], fac)
|
||
# plane = bpy.data.objects.new(name, me)
|
||
# # col = bpy.context.collection
|
||
# # col.objects.link(plane)
|
||
|
||
# me.uv_layers.new(name='UVMap')
|
||
# return plane
|
||
|
||
def create_material(name, img, node_group, keep_existing=False):
|
||
if keep_existing:
|
||
## Reuse if exist and seem clean
|
||
m = bpy.data.materials.get(name)
|
||
if m:
|
||
if name in m.name and m.use_nodes:
|
||
valid_image = next((n for n in m.node_tree.nodes if n.type == 'TEX_IMAGE' and n.image == img), None)
|
||
valid_nodegroup = next((n for n in m.node_tree.nodes if n.type == 'GROUP' and n.node_tree), None)
|
||
if valid_image and valid_nodegroup:
|
||
# Replace node_tree if different
|
||
print('node_group: ', node_group)
|
||
print('valid_nodegroup.node_tree: ', valid_nodegroup.node_tree)
|
||
if node_group != valid_nodegroup.node_tree:
|
||
valid_nodegroup = node_group
|
||
return m
|
||
|
||
# create mat
|
||
mat = bpy.data.materials.new(name)
|
||
mat.blend_method = 'BLEND' # 'CLIP' (clip is able to respect space)
|
||
|
||
mat.use_nodes = True
|
||
|
||
node_tree = mat.node_tree
|
||
nodes = node_tree.nodes
|
||
links = node_tree.links
|
||
|
||
# clear default BSDG
|
||
principled = nodes.get('Principled BSDF')
|
||
if principled: nodes.remove(principled)
|
||
mat_output = nodes.get('Material Output')
|
||
|
||
group = nodes.new("ShaderNodeGroup")
|
||
group.node_tree = node_group
|
||
group.location = (0, 276)
|
||
|
||
img_tex = nodes.new('ShaderNodeTexImage')
|
||
img_tex.image = img
|
||
img_tex.interpolation = 'Linear' # 'Closest','Cubic','Smart'
|
||
img_tex.extension = 'CLIP' # or EXTEND
|
||
img_tex.location = (-400, 218)
|
||
|
||
# links.new(img_tex.outputs['Color'], group.inputs['Color'])
|
||
# links.new(img_tex.outputs['Alpha'], group.inputs['Alpha'])
|
||
links.new(img_tex.outputs[0], group.inputs[0])
|
||
links.new(img_tex.outputs[1], group.inputs[1])
|
||
links.new(group.outputs[0], mat_output.inputs[0])
|
||
|
||
return mat
|
||
|
||
|
||
def create_image_plane(img, node_group, parent=None):
|
||
'''
|
||
img :: blender image or path to an image
|
||
parent :: object to parent to (only if passed)
|
||
|
||
Return created object or None if creation failed
|
||
'''
|
||
|
||
## get image if a path is passed in
|
||
if not isinstance(img, bpy.types.Image):
|
||
try:
|
||
img = bpy.data.images.load(str(img), check_existing=True)
|
||
except:
|
||
return
|
||
|
||
name = img.name.split('.')[0]
|
||
obj_name = f'{name}_texplane'
|
||
x,y = img.size[:]
|
||
|
||
# create and place object plane
|
||
# find height from width (as 1)
|
||
if y < x:
|
||
h = y/x
|
||
w = 1
|
||
else:
|
||
h = 1
|
||
w = x/y
|
||
# size/2 to get center image (half width | height)
|
||
h = h/2
|
||
w = w/2
|
||
|
||
verts = [
|
||
(-w, h, 0),#2
|
||
(w, h, 0),#3
|
||
(w, -h, 0),#4
|
||
(-w, -h, 0),#1
|
||
]
|
||
|
||
faces = [(3, 2, 1, 0)]
|
||
mesh = bpy.data.meshes.new(obj_name)
|
||
mesh.from_pydata(verts, [], faces)
|
||
plane = bpy.data.objects.new(obj_name, mesh)
|
||
|
||
# setup material (link nodegroup)
|
||
|
||
plane.data.uv_layers.new()
|
||
|
||
# Create and assign material
|
||
|
||
mat = create_material(name, img, node_group)
|
||
plane.data.materials.append(mat)
|
||
bpy.context.scene.collection.objects.link(plane)
|
||
# full lock (or just kill selectability...)
|
||
plane.lock_location = [True]*3
|
||
plane.hide_select = True
|
||
|
||
if parent:
|
||
plane.parent = parent
|
||
|
||
return plane
|
||
|
||
def create_empty_image(image):
|
||
if not isinstance(image, bpy.types.Image):
|
||
try:
|
||
image = bpy.data.images.load(str(image), check_existing=True)
|
||
except:
|
||
return
|
||
|
||
# ob = bpy.data.objects.new(f'{image_name}_texempty', None)
|
||
ob = bpy.data.objects.new(image.name, None)
|
||
# ob.hide_viewport = (cam_type == 'ORTHO') # hide if in persp mode
|
||
|
||
# load image on the empty
|
||
ob.empty_display_type = 'IMAGE'
|
||
ob.data = image
|
||
# ob.empty_display_size = bg_cam.data.ortho_scale # set display size
|
||
ob.use_empty_image_alpha = True
|
||
ob.color[3] = 0.5 # transparency
|
||
return ob
|
||
|
||
## Scan background objects and reload
|
||
|
||
def scan_backgrounds(context=None, scene=None, all_bg=False):
|
||
'''return a list (sorted by negative locZ)
|
||
of all holder object (parent of texempty and texplane)
|
||
'''
|
||
|
||
context = context or bpy.context
|
||
scene = scene or context.scene
|
||
|
||
oblist = []
|
||
if all_bg:
|
||
# oblist = [o for o in scene.objects if o.get('is_background_holder')]
|
||
oblist = [o for o in scene.objects if o.name.startswith(PREFIX) and o.type == 'MESH']
|
||
else:
|
||
col = bpy.data.collections.get(BGCOL)
|
||
if not col:
|
||
return oblist
|
||
# oblist = [o for o in col.all_objects if o.get('is_background_holder')]
|
||
oblist = [o for o in col.all_objects if o.name.startswith(PREFIX) and o.type == 'MESH']
|
||
|
||
oblist.sort(key=lambda x : x.location.z, reverse=True)
|
||
return oblist
|
||
|
||
def add_loc_z_driver(plane):
|
||
# Create properties
|
||
# plane['distance'] = abs(plane.location.z)
|
||
plane['distance'] = -plane.location.z
|
||
# Driver for location
|
||
dist_drv = plane.driver_add('location', 2)
|
||
dist_drv.driver.type = 'SCRIPTED' #'AVERAGE' #need to be inverted
|
||
var = dist_drv.driver.variables.new()
|
||
var.name = "distance"
|
||
var.type = 'SINGLE_PROP'
|
||
var.targets[0].id = plane
|
||
var.targets[0].data_path = '["distance"]'
|
||
dist_drv.driver.expression = '-distance'
|
||
|
||
def reload_bg_list(scene=None):
|
||
'''List holder plane object and add them to bg collection'''
|
||
start = time.time()
|
||
scn = scene or bpy.context.scene
|
||
uilist = scn.bg_props.planes
|
||
# current_index = scn.bg_props.index
|
||
|
||
oblist = scan_backgrounds()
|
||
# print('reload_bg_list > oblist:')
|
||
# pp(oblist)
|
||
warning = []
|
||
if not oblist:
|
||
return f'No BG found, need to be in collection "{BGCOL}" !'
|
||
|
||
## Clean
|
||
for ob in oblist:
|
||
bg_viewlayer = bpy.context.view_layer.layer_collection.children.get(BGCOL)
|
||
if not bg_viewlayer:
|
||
return "No 'Background' collection in scene master collection !"
|
||
|
||
ob_viewcol = bg_viewlayer.children.get(ob.name)
|
||
if not ob_viewcol:
|
||
warning.append(f'{ob.name} not found in "Background" collection')
|
||
continue
|
||
|
||
# if vl_col is hided or object is hided
|
||
# visible = any(o for o in ob.children if o.visible_get()) # ob.visible_get()
|
||
# for o in ob.children: o.hide_set(False)
|
||
|
||
ob_viewcol.hide_viewport = False
|
||
ob_viewcol.collection.hide_viewport = False
|
||
# ob_viewcol.exclude = not visible
|
||
|
||
## Driver check (don't erase if exists, else will delete persp mode drivers...)
|
||
if not ob.get('distance'):
|
||
if ob.animation_data:
|
||
for dr in ob.animation_data.drivers:
|
||
if dr.driver.variables.get('distance'):
|
||
# dist = ob.get('distance')
|
||
ob.animation_data.drivers.remove(dr)
|
||
add_loc_z_driver(ob)
|
||
|
||
uilist.clear()
|
||
|
||
## Populate the list
|
||
for ob in oblist: # populate list
|
||
item = uilist.add()
|
||
item.plane = ob
|
||
item.type = "bg"
|
||
scn.bg_props.index = len(uilist) - 1 # id (len of list in the ad loop)
|
||
if ob.children and ob.children[0].type == 'GPENCIL':
|
||
reset_gp_uv(ob.children[0]) # Force reset UV
|
||
|
||
## Add Grease pencil objects
|
||
gp_list = [o for o in bpy.context.scene.objects if o.type == 'GPENCIL' \
|
||
and not o.get('is_background') and o not in oblist]
|
||
for ob in gp_list:
|
||
item = uilist.add()
|
||
item.plane = ob
|
||
item.type = "obj"
|
||
scn.bg_props.index = len(uilist) - 1 # id (len of list in the ad loop)
|
||
|
||
print(f"{len(uilist)} loaded in {time.time()-start:.2f}")
|
||
return warning
|
||
|
||
|
||
def coord_distance_from_cam_straight(coord=None, context=None, camera=None):
|
||
context = context or bpy.context
|
||
coord = coord or context.scene.cursor.location
|
||
camera = camera or context.scene.camera
|
||
if not camera:
|
||
return
|
||
|
||
view_mat = camera.matrix_world
|
||
view_point = view_mat @ Vector((0, 0, -1000))
|
||
co = geometry.intersect_line_plane(view_mat.translation, view_point, coord, view_point)
|
||
if co is None:
|
||
return None
|
||
return (co - view_mat.translation).length
|
||
|
||
def coord_distance_from_cam(coord, context=None, camera=None):
|
||
context = context or bpy.context
|
||
coord = coord or context.scene.cursor.location
|
||
camera = camera or context.scene.camera
|
||
if not camera:
|
||
return
|
||
return (coord - camera.matrix_world.to_translation()).length |