2019-02-07 17:41:18 +01:00
|
|
|
bl_info = {
|
2019-07-01 15:39:01 +02:00
|
|
|
"name": "Multi-User",
|
2019-10-22 16:37:23 +02:00
|
|
|
"author": "Swann Martinez",
|
2019-05-21 16:53:55 +02:00
|
|
|
"description": "",
|
|
|
|
"blender": (2, 80, 0),
|
|
|
|
"location": "",
|
2019-09-18 17:54:02 +02:00
|
|
|
"warning": "Unstable addon, use it at your own risks",
|
2019-05-21 16:53:55 +02:00
|
|
|
"category": "Collaboration"
|
2019-02-07 17:41:18 +01:00
|
|
|
}
|
|
|
|
|
2019-03-14 16:44:18 +01:00
|
|
|
|
2019-07-01 18:04:35 +02:00
|
|
|
import logging
|
2019-08-23 12:28:57 +02:00
|
|
|
import os
|
2019-07-01 15:39:01 +02:00
|
|
|
import random
|
|
|
|
import sys
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2019-07-01 15:39:01 +02:00
|
|
|
import bpy
|
2019-08-22 10:58:54 +02:00
|
|
|
from bpy.app.handlers import persistent
|
2019-07-01 15:39:01 +02:00
|
|
|
|
2019-09-24 18:37:36 +02:00
|
|
|
from . import environment, utils, presence
|
2019-09-24 14:42:59 +02:00
|
|
|
from .libs.replication.replication.constants import RP_COMMON
|
|
|
|
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2019-09-18 17:54:02 +02:00
|
|
|
# TODO: remove dependency as soon as replication will be installed as a module
|
2019-07-01 15:39:01 +02:00
|
|
|
DEPENDENCIES = {
|
2019-08-26 23:14:30 +02:00
|
|
|
("zmq","zmq"),
|
2019-09-17 18:25:06 +02:00
|
|
|
("msgpack","msgpack"),
|
2019-08-27 00:50:38 +02:00
|
|
|
("yaml","pyyaml"),
|
|
|
|
("jsondiff","jsondiff")
|
2019-07-01 15:39:01 +02:00
|
|
|
}
|
2019-05-22 18:19:11 +02:00
|
|
|
|
2019-07-02 16:43:30 +02:00
|
|
|
|
2019-07-01 18:04:35 +02:00
|
|
|
logger = logging.getLogger(__name__)
|
2020-02-07 00:48:34 +01:00
|
|
|
logger.setLevel(logging.WARNING)
|
2019-05-15 14:52:45 +02:00
|
|
|
|
2019-08-29 14:52:14 +02:00
|
|
|
#TODO: refactor config
|
2019-05-15 14:52:45 +02:00
|
|
|
# UTILITY FUNCTIONS
|
2019-08-28 18:58:18 +02:00
|
|
|
def generate_supported_types():
|
|
|
|
stype_dict = {'supported_types':{}}
|
|
|
|
for type in bl_types.types_to_register():
|
2020-01-13 18:51:31 +01:00
|
|
|
type_module = getattr(bl_types, type)
|
|
|
|
type_impl_name = "Bl{}".format(type.split('_')[1].capitalize())
|
|
|
|
type_module_class = getattr(type_module, type_impl_name)
|
|
|
|
|
2019-08-28 18:58:18 +02:00
|
|
|
props = {}
|
2020-01-13 18:51:31 +01:00
|
|
|
props['bl_delay_refresh']=type_module_class.bl_delay_refresh
|
|
|
|
props['bl_delay_apply']=type_module_class.bl_delay_apply
|
2019-08-28 18:58:18 +02:00
|
|
|
props['use_as_filter'] = False
|
2020-01-13 18:51:31 +01:00
|
|
|
props['icon'] = type_module_class.bl_icon
|
|
|
|
props['auto_push']=type_module_class.bl_automatic_push
|
|
|
|
props['bl_name']=type_module_class.bl_id
|
2019-09-18 17:54:02 +02:00
|
|
|
|
2020-01-13 18:51:31 +01:00
|
|
|
stype_dict['supported_types'][type_impl_name] = props
|
2019-08-28 18:58:18 +02:00
|
|
|
|
|
|
|
return stype_dict
|
|
|
|
|
2019-09-18 17:54:02 +02:00
|
|
|
|
2019-05-15 14:52:45 +02:00
|
|
|
def client_list_callback(scene, context):
|
2019-08-14 15:57:07 +02:00
|
|
|
from . import operators
|
2019-05-22 17:17:09 +02:00
|
|
|
|
2019-09-24 14:42:59 +02:00
|
|
|
items = [(RP_COMMON, RP_COMMON, "")]
|
2019-05-15 14:52:45 +02:00
|
|
|
|
2019-05-21 16:53:55 +02:00
|
|
|
username = bpy.context.window_manager.session.username
|
2019-08-14 15:57:07 +02:00
|
|
|
cli = operators.client
|
|
|
|
if cli:
|
2020-01-22 14:33:34 +01:00
|
|
|
client_ids = cli.online_users.keys()
|
|
|
|
for id in client_ids:
|
|
|
|
name_desc = id
|
|
|
|
if id == username:
|
2019-07-11 15:45:56 +02:00
|
|
|
name_desc += " (self)"
|
2019-05-15 14:52:45 +02:00
|
|
|
|
2020-01-22 14:33:34 +01:00
|
|
|
items.append((id, name_desc, ""))
|
2019-05-15 14:52:45 +02:00
|
|
|
|
|
|
|
return items
|
|
|
|
|
|
|
|
|
|
|
|
def randomColor():
|
|
|
|
r = random.random()
|
|
|
|
v = random.random()
|
|
|
|
b = random.random()
|
|
|
|
return [r, v, b]
|
|
|
|
|
2019-07-02 15:55:57 +02:00
|
|
|
class ReplicatedDatablock(bpy.types.PropertyGroup):
|
|
|
|
'''name = StringProperty() '''
|
2019-07-02 16:43:30 +02:00
|
|
|
type_name: bpy.props.StringProperty()
|
2019-09-16 17:24:48 +02:00
|
|
|
bl_name: bpy.props.StringProperty()
|
2019-08-28 18:58:18 +02:00
|
|
|
bl_delay_refresh: bpy.props.FloatProperty()
|
|
|
|
bl_delay_apply: bpy.props.FloatProperty()
|
2019-09-13 17:00:15 +02:00
|
|
|
use_as_filter: bpy.props.BoolProperty(default=True)
|
2019-09-16 13:43:43 +02:00
|
|
|
auto_push: bpy.props.BoolProperty(default=True)
|
2019-08-28 18:58:18 +02:00
|
|
|
icon: bpy.props.StringProperty()
|
2019-07-03 15:32:00 +02:00
|
|
|
|
2020-01-17 18:15:37 +01:00
|
|
|
class SessionUser(bpy.types.PropertyGroup):
|
|
|
|
"""Session User
|
|
|
|
|
|
|
|
Blender user information property
|
|
|
|
"""
|
|
|
|
username: bpy.props.StringProperty(name="username")
|
|
|
|
current_frame: bpy.props.IntProperty(name="current_frame")
|
|
|
|
|
|
|
|
|
2019-05-21 16:53:55 +02:00
|
|
|
class SessionProps(bpy.types.PropertyGroup):
|
2019-05-15 18:37:14 +02:00
|
|
|
username: bpy.props.StringProperty(
|
2019-05-15 14:52:45 +02:00
|
|
|
name="Username",
|
2019-09-24 13:26:51 +02:00
|
|
|
default="user_{}".format(utils.random_string_digits())
|
2019-05-21 16:53:55 +02:00
|
|
|
)
|
2019-05-15 18:37:14 +02:00
|
|
|
ip: bpy.props.StringProperty(
|
2019-05-15 14:52:45 +02:00
|
|
|
name="ip",
|
|
|
|
description='Distant host ip',
|
2019-09-24 13:26:51 +02:00
|
|
|
default="127.0.0.1"
|
2019-07-01 18:04:35 +02:00
|
|
|
)
|
2019-08-13 18:00:54 +02:00
|
|
|
user_uuid: bpy.props.StringProperty(
|
|
|
|
name="user_uuid",
|
|
|
|
default="None"
|
|
|
|
)
|
2019-05-15 18:37:14 +02:00
|
|
|
port: bpy.props.IntProperty(
|
2019-05-15 14:52:45 +02:00
|
|
|
name="port",
|
|
|
|
description='Distant host port',
|
2019-09-24 13:26:51 +02:00
|
|
|
default=5555
|
2019-07-01 18:04:35 +02:00
|
|
|
)
|
2020-02-19 18:07:25 +01:00
|
|
|
ipc_port: bpy.props.IntProperty(
|
|
|
|
name="ipc_port",
|
2020-02-06 18:54:46 +01:00
|
|
|
description='internal ttl port(only usefull for multiple local instances)',
|
|
|
|
default=5561
|
|
|
|
)
|
2019-09-24 13:26:51 +02:00
|
|
|
is_admin: bpy.props.BoolProperty(
|
|
|
|
name="is_admin",
|
|
|
|
default=False
|
|
|
|
)
|
2019-07-01 18:14:48 +02:00
|
|
|
start_empty: bpy.props.BoolProperty(
|
|
|
|
name="start_empty",
|
2019-09-24 13:26:51 +02:00
|
|
|
default=True
|
|
|
|
)
|
2019-05-15 18:37:14 +02:00
|
|
|
session_mode: bpy.props.EnumProperty(
|
2019-05-15 14:52:45 +02:00
|
|
|
name='session_mode',
|
|
|
|
description='session mode',
|
|
|
|
items={
|
|
|
|
('HOST', 'hosting', 'host a session'),
|
|
|
|
('CONNECT', 'connexion', 'connect to a session')},
|
|
|
|
default='HOST')
|
2019-09-24 18:37:36 +02:00
|
|
|
right_strategy: bpy.props.EnumProperty(
|
|
|
|
name='right_strategy',
|
|
|
|
description='right strategy',
|
|
|
|
items={
|
|
|
|
('STRICT', 'strict', 'strict right repartition'),
|
|
|
|
('COMMON', 'common', 'relaxed right repartition')},
|
|
|
|
default='COMMON')
|
2019-05-15 18:37:14 +02:00
|
|
|
client_color: bpy.props.FloatVectorProperty(
|
2019-05-15 14:52:45 +02:00
|
|
|
name="client_instance_color",
|
|
|
|
subtype='COLOR',
|
2019-09-24 13:26:51 +02:00
|
|
|
default=randomColor())
|
2019-05-15 18:37:14 +02:00
|
|
|
clients: bpy.props.EnumProperty(
|
2019-05-15 14:52:45 +02:00
|
|
|
name="clients",
|
|
|
|
description="client enum",
|
2019-08-09 10:58:47 +02:00
|
|
|
items=client_list_callback)
|
2019-07-01 18:04:35 +02:00
|
|
|
enable_presence: bpy.props.BoolProperty(
|
2019-09-19 13:02:39 +02:00
|
|
|
name="Presence overlay",
|
2019-05-15 14:52:45 +02:00
|
|
|
description='Enable overlay drawing module',
|
2019-09-24 14:42:59 +02:00
|
|
|
default=True,
|
|
|
|
update=presence.update_presence
|
|
|
|
)
|
|
|
|
presence_show_selected: bpy.props.BoolProperty(
|
|
|
|
name="Show selected objects",
|
|
|
|
description='Enable selection overlay ',
|
|
|
|
default=True,
|
|
|
|
update=presence.update_overlay_settings
|
2019-09-24 18:37:36 +02:00
|
|
|
)
|
2019-09-24 14:42:59 +02:00
|
|
|
presence_show_user: bpy.props.BoolProperty(
|
|
|
|
name="Show users",
|
|
|
|
description='Enable user overlay ',
|
|
|
|
default=True,
|
|
|
|
update=presence.update_overlay_settings
|
2019-07-01 18:04:35 +02:00
|
|
|
)
|
2019-07-02 15:55:57 +02:00
|
|
|
supported_datablock: bpy.props.CollectionProperty(
|
2019-07-02 17:44:59 +02:00
|
|
|
type=ReplicatedDatablock,
|
2019-07-02 15:55:57 +02:00
|
|
|
)
|
2019-08-28 18:58:18 +02:00
|
|
|
session_filter: bpy.props.CollectionProperty(
|
|
|
|
type=ReplicatedDatablock,
|
|
|
|
)
|
2019-09-18 23:10:36 +02:00
|
|
|
filter_owned: bpy.props.BoolProperty(
|
|
|
|
name="filter_owned",
|
|
|
|
description='Show only owned datablocks',
|
|
|
|
default=True
|
|
|
|
)
|
2020-01-22 16:17:48 +01:00
|
|
|
user_snap_running: bpy.props.BoolProperty(
|
|
|
|
default=False
|
|
|
|
)
|
|
|
|
time_snap_running: bpy.props.BoolProperty(
|
|
|
|
default=False
|
|
|
|
)
|
2019-07-01 18:04:35 +02:00
|
|
|
|
|
|
|
def load(self):
|
|
|
|
config = environment.load_config()
|
2019-08-28 18:58:18 +02:00
|
|
|
if "username" in config.keys():
|
2019-07-01 18:04:35 +02:00
|
|
|
self.username = config["username"]
|
|
|
|
self.ip = config["ip"]
|
|
|
|
self.port = config["port"]
|
|
|
|
self.start_empty = config["start_empty"]
|
|
|
|
self.enable_presence = config["enable_presence"]
|
2019-07-02 16:43:30 +02:00
|
|
|
self.client_color = config["client_color"]
|
2019-07-01 18:04:35 +02:00
|
|
|
else:
|
2019-07-02 15:55:57 +02:00
|
|
|
logger.error("Fail to read user config")
|
|
|
|
|
2019-07-02 16:43:30 +02:00
|
|
|
if len(self.supported_datablock)>0:
|
|
|
|
self.supported_datablock.clear()
|
2019-08-28 18:58:18 +02:00
|
|
|
if "supported_types" not in config:
|
|
|
|
config = generate_supported_types()
|
|
|
|
for datablock in config["supported_types"].keys():
|
2019-07-02 15:55:57 +02:00
|
|
|
rep_value = self.supported_datablock.add()
|
2019-07-03 15:03:37 +02:00
|
|
|
rep_value.name = datablock
|
2019-07-02 16:43:30 +02:00
|
|
|
rep_value.type_name = datablock
|
2019-09-24 13:26:51 +02:00
|
|
|
|
|
|
|
config_block = config["supported_types"][datablock]
|
|
|
|
rep_value.bl_delay_refresh = config_block['bl_delay_refresh']
|
|
|
|
rep_value.bl_delay_apply = config_block['bl_delay_apply']
|
|
|
|
rep_value.icon = config_block['icon']
|
|
|
|
rep_value.auto_push = config_block['auto_push']
|
|
|
|
rep_value.bl_name = config_block['bl_name']
|
2019-09-16 13:43:43 +02:00
|
|
|
|
2019-07-02 17:44:59 +02:00
|
|
|
def save(self,context):
|
|
|
|
config = environment.load_config()
|
|
|
|
|
2019-09-30 18:41:04 +02:00
|
|
|
if "supported_types" not in config:
|
|
|
|
config = generate_supported_types()
|
|
|
|
|
2019-07-02 17:44:59 +02:00
|
|
|
config["username"] = self.username
|
|
|
|
config["ip"] = self.ip
|
|
|
|
config["port"] = self.port
|
|
|
|
config["start_empty"] = self.start_empty
|
|
|
|
config["enable_presence"] = self.enable_presence
|
|
|
|
config["client_color"] = [self.client_color.r,self.client_color.g,self.client_color.b]
|
2019-07-02 15:55:57 +02:00
|
|
|
|
2019-09-30 18:41:04 +02:00
|
|
|
|
2019-07-02 17:44:59 +02:00
|
|
|
for bloc in self.supported_datablock:
|
2019-09-24 13:26:51 +02:00
|
|
|
config_block = config["supported_types"][bloc.type_name]
|
|
|
|
config_block['bl_delay_refresh'] = bloc.bl_delay_refresh
|
|
|
|
config_block['bl_delay_apply'] = bloc.bl_delay_apply
|
|
|
|
config_block['use_as_filter'] = bloc.use_as_filter
|
|
|
|
config_block['icon'] = bloc.icon
|
|
|
|
config_block['auto_push'] = bloc.auto_push
|
|
|
|
config_block['bl_name'] = bloc.bl_name
|
2019-07-02 17:44:59 +02:00
|
|
|
environment.save_config(config)
|
2019-07-02 15:55:57 +02:00
|
|
|
|
2019-05-15 14:52:45 +02:00
|
|
|
|
2019-07-02 16:43:30 +02:00
|
|
|
classes = (
|
2020-01-17 18:15:37 +01:00
|
|
|
SessionUser,
|
2019-07-02 15:55:57 +02:00
|
|
|
ReplicatedDatablock,
|
2019-08-08 15:35:43 +02:00
|
|
|
SessionProps,
|
2019-07-02 15:55:57 +02:00
|
|
|
|
2019-07-02 16:43:30 +02:00
|
|
|
)
|
2019-05-15 14:52:45 +02:00
|
|
|
|
2020-02-03 19:04:08 +01:00
|
|
|
libs = os.path.dirname(os.path.abspath(__file__))+"\\libs\\replication\\replication"
|
2019-05-21 16:53:55 +02:00
|
|
|
|
2019-07-08 22:39:11 +02:00
|
|
|
@persistent
|
|
|
|
def load_handler(dummy):
|
|
|
|
import bpy
|
|
|
|
bpy.context.window_manager.session.load()
|
|
|
|
|
2019-05-23 16:49:32 +02:00
|
|
|
def register():
|
2019-08-05 16:48:46 +02:00
|
|
|
if libs not in sys.path:
|
|
|
|
sys.path.append(libs)
|
|
|
|
|
2019-07-01 18:04:35 +02:00
|
|
|
environment.setup(DEPENDENCIES,bpy.app.binary_path_python)
|
2019-05-15 18:37:14 +02:00
|
|
|
|
2019-09-27 14:50:00 +02:00
|
|
|
from . import presence
|
2019-05-15 18:37:14 +02:00
|
|
|
from . import operators
|
|
|
|
from . import ui
|
|
|
|
|
2019-05-15 14:52:45 +02:00
|
|
|
for cls in classes:
|
|
|
|
bpy.utils.register_class(cls)
|
|
|
|
|
|
|
|
bpy.types.WindowManager.session = bpy.props.PointerProperty(
|
2019-05-21 16:53:55 +02:00
|
|
|
type=SessionProps)
|
2019-08-08 18:12:13 +02:00
|
|
|
bpy.types.ID.uuid = bpy.props.StringProperty(default="")
|
2020-01-17 18:15:37 +01:00
|
|
|
bpy.types.WindowManager.online_users = bpy.props.CollectionProperty(
|
|
|
|
type=SessionUser
|
|
|
|
)
|
|
|
|
bpy.types.WindowManager.user_index = bpy.props.IntProperty()
|
2019-07-01 18:04:35 +02:00
|
|
|
bpy.context.window_manager.session.load()
|
2019-09-24 13:26:51 +02:00
|
|
|
|
2019-09-27 14:50:00 +02:00
|
|
|
presence.register()
|
2019-04-10 17:01:21 +02:00
|
|
|
operators.register()
|
|
|
|
ui.register()
|
2019-08-29 14:52:14 +02:00
|
|
|
bpy.app.handlers.load_post.append(load_handler)
|
2019-02-07 17:41:18 +01:00
|
|
|
|
|
|
|
def unregister():
|
2019-09-27 14:50:00 +02:00
|
|
|
from . import presence
|
2019-05-15 18:37:14 +02:00
|
|
|
from . import operators
|
|
|
|
from . import ui
|
2019-05-21 16:53:55 +02:00
|
|
|
|
2019-09-27 14:50:00 +02:00
|
|
|
presence.unregister()
|
2019-04-10 17:01:21 +02:00
|
|
|
ui.unregister()
|
|
|
|
operators.unregister()
|
|
|
|
|
2019-05-15 14:52:45 +02:00
|
|
|
del bpy.types.WindowManager.session
|
|
|
|
|
2019-07-02 16:43:30 +02:00
|
|
|
for cls in reversed(classes):
|
2019-05-15 14:52:45 +02:00
|
|
|
bpy.utils.unregister_class(cls)
|