2019-05-02 17:58:37 +02:00
|
|
|
import logging
|
2019-05-01 18:51:51 +02:00
|
|
|
import sys
|
2019-04-19 15:46:54 +02:00
|
|
|
from uuid import uuid4
|
2019-05-03 16:20:40 +02:00
|
|
|
import json
|
2019-08-08 15:00:07 +02:00
|
|
|
import os
|
2019-08-08 15:35:43 +02:00
|
|
|
import string
|
|
|
|
import random
|
2019-05-02 17:58:37 +02:00
|
|
|
|
|
|
|
import bpy
|
|
|
|
import mathutils
|
|
|
|
|
2019-08-08 15:35:43 +02:00
|
|
|
from . import presence, environment
|
2019-05-02 17:58:37 +02:00
|
|
|
from .libs import dump_anything
|
2019-04-10 17:01:21 +02:00
|
|
|
|
2019-05-15 14:52:45 +02:00
|
|
|
# TODO: replace hardcoded values...
|
2019-08-08 15:00:07 +02:00
|
|
|
BPY_TYPES = {'Image': 'images', 'Texture': 'textures', 'Material': 'materials', 'GreasePencil': 'grease_pencils', 'Curve': 'curves', 'Collection': 'collections', 'Mesh': 'meshes', 'Object': 'objects',
|
|
|
|
'Scene': 'scenes', 'Light': 'lights', 'SunLight': 'lights', 'SpotLight': 'lights', 'AreaLight': 'lights', 'PointLight': 'lights', 'Camera': 'cameras', 'Action': 'actions', 'Armature': 'armatures'}
|
2019-04-11 14:39:31 +02:00
|
|
|
|
2019-04-19 15:46:54 +02:00
|
|
|
logger = logging.getLogger(__name__)
|
2019-08-08 15:00:07 +02:00
|
|
|
logger.setLevel(logging.DEBUG)
|
|
|
|
|
2019-04-17 15:48:20 +02:00
|
|
|
# UTILITY FUNCTIONS
|
2019-08-08 15:35:43 +02:00
|
|
|
def random_string_digits(stringLength=6):
|
|
|
|
"""Generate a random string of letters and digits """
|
|
|
|
lettersAndDigits = string.ascii_letters + string.digits
|
|
|
|
return ''.join(random.choice(lettersAndDigits) for i in range(stringLength))
|
|
|
|
|
|
|
|
def clean_scene():
|
|
|
|
for datablock in BPY_TYPES:
|
|
|
|
datablock_ref = getattr(bpy.data, BPY_TYPES[datablock])
|
|
|
|
for item in datablock_ref:
|
|
|
|
try:
|
|
|
|
datablock_ref.remove(item)
|
|
|
|
# Catch last scene remove
|
|
|
|
except RuntimeError:
|
|
|
|
pass
|
2019-08-08 15:00:07 +02:00
|
|
|
|
|
|
|
|
2019-05-03 18:52:49 +02:00
|
|
|
def revers(d):
|
|
|
|
l = []
|
|
|
|
for i in d:
|
|
|
|
l.append(i)
|
2019-05-08 17:06:52 +02:00
|
|
|
|
2019-05-03 18:52:49 +02:00
|
|
|
return l[::-1]
|
2019-05-02 17:58:37 +02:00
|
|
|
|
2019-05-08 17:06:52 +02:00
|
|
|
|
2019-05-03 14:55:18 +02:00
|
|
|
def get_armature_edition_context(armature):
|
2019-05-08 17:06:52 +02:00
|
|
|
|
2019-05-03 16:20:40 +02:00
|
|
|
override = {}
|
2019-05-03 14:55:18 +02:00
|
|
|
# Set correct area
|
2019-05-08 17:06:52 +02:00
|
|
|
for area in bpy.data.window_managers[0].windows[0].screen.areas:
|
2019-05-03 14:55:18 +02:00
|
|
|
if area.type == 'VIEW_3D':
|
2019-05-03 16:20:40 +02:00
|
|
|
override = bpy.context.copy()
|
2019-05-03 14:55:18 +02:00
|
|
|
override['area'] = area
|
|
|
|
break
|
|
|
|
|
|
|
|
# Set correct armature settings
|
2019-05-08 17:06:52 +02:00
|
|
|
override['window'] = bpy.data.window_managers[0].windows[0]
|
2019-05-03 16:20:40 +02:00
|
|
|
override['screen'] = bpy.data.window_managers[0].windows[0].screen
|
2019-05-03 14:55:18 +02:00
|
|
|
override['mode'] = 'EDIT_ARMATURE'
|
|
|
|
override['active_object'] = armature
|
|
|
|
override['selected_objects'] = [armature]
|
|
|
|
|
|
|
|
for o in bpy.data.objects:
|
|
|
|
if o.data == armature:
|
2019-05-03 16:20:40 +02:00
|
|
|
override['edit_object'] = o
|
2019-05-08 17:06:52 +02:00
|
|
|
|
2019-05-03 14:55:18 +02:00
|
|
|
break
|
|
|
|
|
|
|
|
return override
|
2019-04-18 15:05:48 +02:00
|
|
|
|
2019-05-08 17:06:52 +02:00
|
|
|
|
2019-04-17 16:15:21 +02:00
|
|
|
def get_selected_objects(scene):
|
2019-04-17 15:48:20 +02:00
|
|
|
selected_objects = []
|
2019-04-17 16:15:21 +02:00
|
|
|
for obj in scene.objects:
|
2019-04-17 15:48:20 +02:00
|
|
|
if obj.select_get():
|
|
|
|
selected_objects.append(obj.name)
|
2019-04-10 18:01:55 +02:00
|
|
|
|
2019-04-17 15:48:20 +02:00
|
|
|
return selected_objects
|
2019-04-18 15:05:48 +02:00
|
|
|
|
2019-08-08 15:00:07 +02:00
|
|
|
def load_dict(src_dict, target):
|
|
|
|
try:
|
|
|
|
for item in src_dict:
|
|
|
|
# attr =
|
|
|
|
setattr(target, item, src_dict[item])
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
logger.error(e)
|
|
|
|
pass
|
|
|
|
|
2019-04-10 18:01:55 +02:00
|
|
|
def resolve_bpy_path(path):
|
|
|
|
"""
|
|
|
|
Get bpy property value from path
|
|
|
|
"""
|
|
|
|
item = None
|
2019-04-10 17:01:21 +02:00
|
|
|
|
2019-04-10 18:01:55 +02:00
|
|
|
try:
|
|
|
|
path = path.split('/')
|
2019-08-08 15:00:07 +02:00
|
|
|
item = getattr(bpy.data, BPY_TYPES[path[0]])[path[1]]
|
2019-04-10 17:01:21 +02:00
|
|
|
|
2019-04-10 18:01:55 +02:00
|
|
|
except:
|
|
|
|
pass
|
2019-04-10 17:01:21 +02:00
|
|
|
|
2019-04-10 18:01:55 +02:00
|
|
|
return item
|
2019-04-10 17:01:21 +02:00
|
|
|
|
2019-04-17 15:48:20 +02:00
|
|
|
|
2019-04-18 15:05:48 +02:00
|
|
|
def load_client(client=None, data=None):
|
2019-04-17 16:15:21 +02:00
|
|
|
C = bpy.context
|
|
|
|
D = bpy.data
|
2019-05-15 14:52:45 +02:00
|
|
|
net_settings = C.window_manager.session
|
2019-05-02 17:52:32 +02:00
|
|
|
|
2019-04-17 15:01:15 +02:00
|
|
|
if client and data:
|
2019-07-01 18:04:35 +02:00
|
|
|
if net_settings.enable_presence:
|
2019-05-02 17:52:32 +02:00
|
|
|
draw.renderer.draw_client(data)
|
|
|
|
draw.renderer.draw_client_selected_objects(data)
|
2019-04-18 15:05:48 +02:00
|
|
|
|
2019-08-08 15:00:07 +02:00
|
|
|
|
2019-05-03 14:55:18 +02:00
|
|
|
def load_armature(target=None, data=None, create=False):
|
2019-05-03 16:20:40 +02:00
|
|
|
file = "cache_{}.json".format(data['name'])
|
2019-05-03 18:52:49 +02:00
|
|
|
context = bpy.context
|
2019-05-03 16:20:40 +02:00
|
|
|
|
2019-05-03 18:52:49 +02:00
|
|
|
if not target:
|
2019-05-08 17:06:52 +02:00
|
|
|
target = bpy.data.armatures.new(data['name'])
|
|
|
|
|
2019-05-03 16:20:40 +02:00
|
|
|
dump_anything.load(target, data)
|
2019-05-03 14:55:18 +02:00
|
|
|
|
2019-05-03 16:20:40 +02:00
|
|
|
with open(file, 'w') as fp:
|
|
|
|
json.dump(data, fp)
|
2019-05-08 17:06:52 +02:00
|
|
|
fp.close()
|
|
|
|
|
2019-05-03 16:20:40 +02:00
|
|
|
target.id = data['id']
|
|
|
|
else:
|
2019-05-03 18:52:49 +02:00
|
|
|
# Construct a correct execution context
|
2019-05-03 16:20:40 +02:00
|
|
|
file = "cache_{}.json".format(target.name)
|
|
|
|
|
|
|
|
with open(file, 'r') as fp:
|
|
|
|
data = json.load(fp)
|
2019-05-03 14:55:18 +02:00
|
|
|
|
2019-05-03 16:20:40 +02:00
|
|
|
if data:
|
2019-05-03 18:52:49 +02:00
|
|
|
ob = None
|
|
|
|
for o in bpy.data.objects:
|
|
|
|
if o.data == target:
|
|
|
|
ob = o
|
|
|
|
if ob:
|
|
|
|
bpy.context.view_layer.objects.active = ob
|
|
|
|
bpy.ops.object.mode_set(mode='EDIT', toggle=False)
|
|
|
|
for eb in data['edit_bones']:
|
|
|
|
if eb in target.edit_bones.keys():
|
|
|
|
# Update the bone
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
# Add new edit bone and load it
|
|
|
|
|
|
|
|
target_new_b = target.edit_bones.new[eb]
|
|
|
|
dump_anything.load(target_new_b, data['bones'][eb])
|
|
|
|
|
2019-06-10 18:26:44 +02:00
|
|
|
logger.debug(eb)
|
2019-05-03 18:52:49 +02:00
|
|
|
|
|
|
|
bpy.ops.object.mode_set(mode='OBJECT', toggle=False)
|
|
|
|
fp.close()
|
|
|
|
import os
|
|
|
|
os.remove(file)
|
2019-05-03 14:55:18 +02:00
|
|
|
|
2019-04-10 18:01:55 +02:00
|
|
|
def dump_datablock(datablock, depth):
|
|
|
|
if datablock:
|
|
|
|
dumper = dump_anything.Dumper()
|
|
|
|
dumper.type_subset = dumper.match_subset_all
|
|
|
|
dumper.depth = depth
|
|
|
|
|
|
|
|
datablock_type = datablock.bl_rna.name
|
|
|
|
key = "{}/{}".format(datablock_type, datablock.name)
|
|
|
|
data = dumper.dump(datablock)
|
|
|
|
|
|
|
|
return data
|
|
|
|
|
2019-05-14 11:00:38 +02:00
|
|
|
|
2019-08-08 15:00:07 +02:00
|
|
|
def dump_datablock_attibutes(datablock=None, attributes=[], depth=1, dickt=None):
|
2019-04-10 18:01:55 +02:00
|
|
|
if datablock:
|
|
|
|
dumper = dump_anything.Dumper()
|
|
|
|
dumper.type_subset = dumper.match_subset_all
|
|
|
|
dumper.depth = depth
|
|
|
|
|
2019-05-13 11:49:46 +02:00
|
|
|
datablock_type = datablock.bl_rna.name
|
2019-04-10 18:01:55 +02:00
|
|
|
key = "{}/{}".format(datablock_type, datablock.name)
|
|
|
|
|
|
|
|
data = {}
|
2019-05-08 17:06:52 +02:00
|
|
|
|
2019-05-03 18:52:49 +02:00
|
|
|
if dickt:
|
|
|
|
data = dickt
|
2019-04-10 18:01:55 +02:00
|
|
|
for attr in attributes:
|
|
|
|
try:
|
|
|
|
data[attr] = dumper.dump(getattr(datablock, attr))
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
return data
|
|
|
|
|
2019-04-17 16:15:21 +02:00
|
|
|
|
2019-08-08 15:00:07 +02:00
|
|
|
|
|
|
|
|
2019-04-17 15:48:20 +02:00
|
|
|
def init_client(key=None):
|
|
|
|
client_dict = {}
|
2019-04-18 15:05:48 +02:00
|
|
|
|
2019-04-17 15:48:20 +02:00
|
|
|
C = bpy.context
|
2019-05-15 14:52:45 +02:00
|
|
|
Net = C.window_manager.session
|
2019-04-19 15:46:54 +02:00
|
|
|
client_dict['uuid'] = str(uuid4())
|
2019-04-18 15:05:48 +02:00
|
|
|
client_dict['location'] = [[0, 0, 0], [0, 0, 0], [0, 0, 0], [0, 0, 0]]
|
|
|
|
client_dict['color'] = [Net.client_color.r,
|
|
|
|
Net.client_color.g, Net.client_color.b, 1]
|
2019-04-17 15:48:20 +02:00
|
|
|
|
|
|
|
client_dict['active_objects'] = get_selected_objects(C.view_layer)
|
2019-04-18 11:34:16 +02:00
|
|
|
|
2019-04-18 15:05:48 +02:00
|
|
|
return client_dict
|