multi-user/multi_user/preferences.py

626 lines
20 KiB
Python
Raw Normal View History

2020-03-20 14:56:50 +01:00
# ##### BEGIN GPL LICENSE BLOCK #####
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
#
# ##### END GPL LICENSE BLOCK #####
import random
import logging
import bpy
import string
2020-06-24 16:58:44 +02:00
import re
2020-09-21 16:47:49 +02:00
import os
from pathlib import Path
from . import bl_types, environment, addon_updater_ops, presence, ui
from .utils import get_preferences, get_expanded_icon
2020-07-10 16:50:09 +02:00
from replication.constants import RP_COMMON
from replication.interface import session
2020-11-06 22:33:33 +01:00
# From https://stackoverflow.com/a/106223
IP_REGEX = re.compile("^(([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\.){3}([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])$")
HOSTNAME_REGEX = re.compile("^(([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9\-]*[a-zA-Z0-9])\.)*([A-Za-z0-9]|[A-Za-z0-9][A-Za-z0-9\-]*[A-Za-z0-9])$")
2020-04-14 17:22:28 +02:00
2021-06-11 16:57:02 +02:00
DEFAULT_PRESETS = {
"localhost" : {
"server_ip": "localhost",
"server_port": 5555,
"server_password": "admin"
},
"public session" : {
"server_ip": "51.75.71.183",
"server_port": 5555,
2021-06-14 15:17:07 +02:00
"server_password": ""
2021-06-11 16:57:02 +02:00
},
}
def randomColor():
"""Generate a random color """
r = random.random()
v = random.random()
b = random.random()
return [r, v, b]
def random_string_digits(stringLength=6):
"""Generate a random string of letters and digits"""
lettersAndDigits = string.ascii_letters + string.digits
return ''.join(random.choices(lettersAndDigits, k=stringLength))
2020-04-14 17:22:28 +02:00
def update_panel_category(self, context):
ui.unregister()
ui.SESSION_PT_settings.bl_category = self.panel_category
ui.register()
2020-06-24 16:58:44 +02:00
def update_ip(self, context):
2020-11-06 22:33:33 +01:00
ip = IP_REGEX.search(self.ip)
dns = HOSTNAME_REGEX.search(self.ip)
2020-06-24 16:58:44 +02:00
if ip:
self['ip'] = ip.group()
2020-11-06 22:33:33 +01:00
elif dns:
self['ip'] = dns.group()
2020-06-24 16:58:44 +02:00
else:
logging.error("Wrong IP format")
self['ip'] = "127.0.0.1"
2021-06-11 12:13:23 +02:00
def update_server_preset_interface(self, context):
self.server_name = self.server_preset.get(self.server_preset_interface).name
self.ip = self.server_preset.get(self.server_preset_interface).server_ip
self.port = self.server_preset.get(self.server_preset_interface).server_port
2021-06-11 12:13:23 +02:00
self.password = self.server_preset.get(self.server_preset_interface).server_password
2020-09-21 16:47:49 +02:00
def update_directory(self, context):
new_dir = Path(self.cache_directory)
if new_dir.exists() and any(Path(self.cache_directory).iterdir()):
logging.error("The folder is not empty, choose another one.")
self['cache_directory'] = environment.DEFAULT_CACHE_DIR
elif not new_dir.exists():
logging.info("Target cache folder doesn't exist, creating it.")
os.makedirs(self.cache_directory, exist_ok=True)
def set_log_level(self, value):
logging.getLogger().setLevel(value)
def get_log_level(self):
return logging.getLogger().level
class ReplicatedDatablock(bpy.types.PropertyGroup):
type_name: bpy.props.StringProperty()
bl_name: bpy.props.StringProperty()
use_as_filter: bpy.props.BoolProperty(default=True)
auto_push: bpy.props.BoolProperty(default=True)
icon: bpy.props.StringProperty()
class ServerPreset(bpy.types.PropertyGroup):
server_ip: bpy.props.StringProperty()
server_port: bpy.props.IntProperty(default=5555)
server_password: bpy.props.StringProperty(default="admin", subtype = "PASSWORD")
2020-03-11 22:42:09 +01:00
2020-09-25 11:33:35 +02:00
def set_sync_render_settings(self, value):
self['sync_render_settings'] = value
if session and bpy.context.scene.uuid and value:
2020-10-15 12:11:28 +02:00
bpy.ops.session.apply('INVOKE_DEFAULT',
target=bpy.context.scene.uuid,
reset_dependencies=False)
2020-09-25 11:23:36 +02:00
2020-09-25 11:33:35 +02:00
def set_sync_active_camera(self, value):
self['sync_active_camera'] = value
if session and bpy.context.scene.uuid and value:
2020-10-15 12:11:28 +02:00
bpy.ops.session.apply('INVOKE_DEFAULT',
target=bpy.context.scene.uuid,
reset_dependencies=False)
2020-09-25 11:23:36 +02:00
2020-09-25 11:33:35 +02:00
2020-04-02 18:42:41 +02:00
class ReplicationFlags(bpy.types.PropertyGroup):
2020-09-25 11:33:35 +02:00
def get_sync_render_settings(self):
return self.get('sync_render_settings', True)
2020-09-25 11:23:36 +02:00
2020-09-25 11:33:35 +02:00
def get_sync_active_camera(self):
return self.get('sync_active_camera', True)
2020-04-02 18:42:41 +02:00
sync_render_settings: bpy.props.BoolProperty(
name="Synchronize render settings",
description="Synchronize render settings (eevee and cycles only)",
default=False,
2020-09-25 11:33:35 +02:00
set=set_sync_render_settings,
get=get_sync_render_settings
)
2020-09-25 11:23:36 +02:00
sync_during_editmode: bpy.props.BoolProperty(
name="Edit mode updates",
description="Enable objects update in edit mode (! Impact performances !)",
default=False
)
2020-09-25 11:33:35 +02:00
sync_active_camera: bpy.props.BoolProperty(
name="Synchronize active camera",
description="Synchronize the active camera",
default=True,
get=get_sync_active_camera,
set=set_sync_active_camera
)
2020-04-02 18:42:41 +02:00
class SessionPrefs(bpy.types.AddonPreferences):
2020-03-11 22:42:09 +01:00
bl_idname = __package__
ip: bpy.props.StringProperty(
name="ip",
description='Distant host ip',
2021-06-11 16:57:02 +02:00
default="localhost",
2020-06-24 16:58:44 +02:00
update=update_ip)
username: bpy.props.StringProperty(
name="Username",
default=f"user_{random_string_digits()}"
)
client_color: bpy.props.FloatVectorProperty(
name="client_instance_color",
subtype='COLOR',
default=randomColor())
port: bpy.props.IntProperty(
name="port",
description='Distant host port',
default=5555
2020-03-11 22:42:09 +01:00
)
server_name: bpy.props.StringProperty(
name="server_name",
2021-06-11 12:13:23 +02:00
description="Custom name of the server",
2021-06-11 16:57:02 +02:00
default='localhost',
)
2021-06-11 12:18:51 +02:00
password: bpy.props.StringProperty(
name="password",
default=random_string_digits(),
description='Session password',
subtype='PASSWORD'
)
2020-04-02 18:42:41 +02:00
sync_flags: bpy.props.PointerProperty(
type=ReplicationFlags
)
supported_datablocks: bpy.props.CollectionProperty(
type=ReplicatedDatablock,
2020-03-11 22:42:09 +01:00
)
init_method: bpy.props.EnumProperty(
name='init_method',
description='Init repo',
items={
('EMPTY', 'an empty scene', 'start empty'),
('BLEND', 'current scenes', 'use current scenes')},
default='BLEND')
cache_directory: bpy.props.StringProperty(
name="cache directory",
subtype="DIR_PATH",
2020-09-21 16:47:49 +02:00
default=environment.DEFAULT_CACHE_DIR,
update=update_directory)
2020-04-08 11:15:29 +02:00
connection_timeout: bpy.props.IntProperty(
name='connection timeout',
description='connection timeout before disconnection',
default=5000
2020-04-08 11:15:29 +02:00
)
# Replication update settings
depsgraph_update_rate: bpy.props.FloatProperty(
name='depsgraph update rate (s)',
description='Dependency graph uppdate rate (s)',
default=1
)
clear_memory_filecache: bpy.props.BoolProperty(
name="Clear memory filecache",
description="Remove filecache from memory",
default=False
)
# for UI
2020-03-11 22:42:09 +01:00
category: bpy.props.EnumProperty(
name="Category",
description="Preferences Category",
items=[
('CONFIG', "Configuration", "Configuration of this add-on"),
2020-03-11 22:42:09 +01:00
('UPDATE', "Update", "Update this add-on"),
],
default='CONFIG'
2020-03-11 22:42:09 +01:00
)
2020-04-22 17:04:14 +02:00
logging_level: bpy.props.EnumProperty(
name="Log level",
description="Log verbosity level",
items=[
('ERROR', "error", "show only errors", logging.ERROR),
('WARNING', "warning", "only show warnings and errors", logging.WARNING),
('INFO', "info", "default level", logging.INFO),
('DEBUG', "debug", "show all logs", logging.DEBUG),
2020-04-22 17:04:14 +02:00
],
default='INFO',
set=set_log_level,
get=get_log_level
2020-04-22 17:04:14 +02:00
)
presence_hud_scale: bpy.props.FloatProperty(
name="Text scale",
description="Adjust the session widget text scale",
min=7,
max=90,
default=25,
)
presence_hud_hpos: bpy.props.FloatProperty(
name="Horizontal position",
description="Adjust the session widget horizontal position",
min=1,
max=90,
default=1,
step=1,
subtype='PERCENTAGE',
)
presence_hud_vpos: bpy.props.FloatProperty(
name="Vertical position",
description="Adjust the session widget vertical position",
min=1,
max=94,
default=1,
step=1,
subtype='PERCENTAGE',
)
2021-06-30 15:34:03 +02:00
presence_mode_distance: bpy.props.FloatProperty(
name="Distance mode visibilty",
description="Adjust the distance visibilty of user's mode",
min=0.1,
max=1000,
default=100,
)
conf_session_identity_expanded: bpy.props.BoolProperty(
name="Identity",
description="Identity",
default=True
)
conf_session_net_expanded: bpy.props.BoolProperty(
name="Net",
description="net",
default=True
)
2020-03-04 12:51:56 +01:00
conf_session_hosting_expanded: bpy.props.BoolProperty(
name="Rights",
description="Rights",
default=False
)
conf_session_cache_expanded: bpy.props.BoolProperty(
name="Cache",
description="cache",
2020-03-04 12:51:56 +01:00
default=False
)
2020-04-14 17:22:28 +02:00
conf_session_ui_expanded: bpy.props.BoolProperty(
name="Interface",
description="Interface",
default=False
)
sidebar_advanced_rep_expanded: bpy.props.BoolProperty(
name="sidebar_advanced_rep_expanded",
description="sidebar_advanced_rep_expanded",
default=False
)
sidebar_advanced_log_expanded: bpy.props.BoolProperty(
name="sidebar_advanced_log_expanded",
description="sidebar_advanced_log_expanded",
default=False
)
sidebar_advanced_net_expanded: bpy.props.BoolProperty(
name="sidebar_advanced_net_expanded",
description="sidebar_advanced_net_expanded",
default=False
)
sidebar_advanced_cache_expanded: bpy.props.BoolProperty(
name="sidebar_advanced_cache_expanded",
description="sidebar_advanced_cache_expanded",
default=False
)
2020-03-11 22:42:09 +01:00
auto_check_update: bpy.props.BoolProperty(
name="Auto-check for Update",
description="If enabled, auto-check for updates using an interval",
default=False,
)
updater_intrval_months: bpy.props.IntProperty(
name='Months',
description="Number of months between checking for updates",
default=0,
min=0
)
updater_intrval_days: bpy.props.IntProperty(
name='Days',
description="Number of days between checking for updates",
default=7,
min=0,
max=31
)
updater_intrval_hours: bpy.props.IntProperty(
name='Hours',
description="Number of hours between checking for updates",
default=0,
min=0,
max=23
)
updater_intrval_minutes: bpy.props.IntProperty(
name='Minutes',
description="Number of minutes between checking for updates",
default=0,
min=0,
max=59
)
# Server preset
def server_list_callback(scene, context):
settings = get_preferences()
enum = []
for i in settings.server_preset:
enum.append((i.name, i.name, ""))
return enum
server_preset: bpy.props.CollectionProperty(
name="server preset",
type=ServerPreset,
)
server_preset_interface: bpy.props.EnumProperty(
name="servers",
description="servers enum",
items=server_list_callback,
2021-06-11 12:13:23 +02:00
update=update_server_preset_interface,
)
2020-04-14 17:22:28 +02:00
# Custom panel
panel_category: bpy.props.StringProperty(
description="Choose a name for the category of the panel",
default="Multiuser",
update=update_panel_category)
def draw(self, context):
layout = self.layout
2020-03-11 22:42:09 +01:00
layout.row().prop(self, "category", expand=True)
2020-04-14 17:22:28 +02:00
2020-03-11 22:42:09 +01:00
if self.category == 'CONFIG':
grid = layout.column()
2020-04-14 17:22:28 +02:00
2020-03-11 22:42:09 +01:00
# USER INFORMATIONS
box = grid.box()
box.prop(
self, "conf_session_identity_expanded", text="User information",
icon=get_expanded_icon(self.conf_session_identity_expanded),
emboss=False)
2020-03-11 22:42:09 +01:00
if self.conf_session_identity_expanded:
box.row().prop(self, "username", text="name")
box.row().prop(self, "client_color", text="color")
# NETWORK SETTINGS
box = grid.box()
box.prop(
self, "conf_session_net_expanded", text="Networking",
icon=get_expanded_icon(self.conf_session_net_expanded),
emboss=False)
2020-03-11 22:42:09 +01:00
if self.conf_session_net_expanded:
box.row().prop(self, "ip", text="Address")
row = box.row()
row.label(text="Port:")
row.prop(self, "port", text="")
2020-03-11 22:42:09 +01:00
row = box.row()
row.label(text="Init the session from:")
row.prop(self, "init_method", text="")
2020-03-11 22:42:09 +01:00
# HOST SETTINGS
box = grid.box()
box.prop(
self, "conf_session_hosting_expanded", text="Hosting",
icon=get_expanded_icon(self.conf_session_hosting_expanded),
emboss=False)
2020-03-11 22:42:09 +01:00
if self.conf_session_hosting_expanded:
row = box.row()
row.label(text="Init the session from:")
row.prop(self, "init_method", text="")
2020-04-14 17:22:28 +02:00
2020-03-11 22:42:09 +01:00
# CACHE SETTINGS
box = grid.box()
box.prop(
self, "conf_session_cache_expanded", text="Cache",
icon=get_expanded_icon(self.conf_session_cache_expanded),
emboss=False)
2020-03-11 22:42:09 +01:00
if self.conf_session_cache_expanded:
box.row().prop(self, "cache_directory", text="Cache directory")
box.row().prop(self, "clear_memory_filecache", text="Clear memory filecache")
2020-03-11 22:42:09 +01:00
2020-04-14 17:22:28 +02:00
# INTERFACE SETTINGS
box = grid.box()
box.prop(
self, "conf_session_ui_expanded", text="Interface",
icon=get_expanded_icon(self.conf_session_ui_expanded),
2020-04-14 17:22:28 +02:00
emboss=False)
if self.conf_session_ui_expanded:
box.row().prop(self, "panel_category", text="Panel category", expand=True)
row = box.row()
row.label(text="Session widget:")
col = box.column(align=True)
col.prop(self, "presence_hud_scale", expand=True)
col.prop(self, "presence_hud_hpos", expand=True)
col.prop(self, "presence_hud_vpos", expand=True)
2020-04-14 17:22:28 +02:00
2021-06-30 15:34:03 +02:00
col.prop(self, "presence_mode_distance", expand=True)
2020-03-11 22:42:09 +01:00
if self.category == 'UPDATE':
from . import addon_updater_ops
addon_updater_ops.update_settings_ui(self, context)
def generate_supported_types(self):
self.supported_datablocks.clear()
bpy_protocol = bl_types.get_data_translation_protocol()
# init the factory with supported types
for dcc_type_id, impl in bpy_protocol.implementations.items():
new_db = self.supported_datablocks.add()
new_db.name = dcc_type_id
new_db.type_name = dcc_type_id
new_db.use_as_filter = True
new_db.icon = impl.bl_icon
new_db.bl_name = impl.bl_id
2021-06-11 12:13:23 +02:00
# custom at launch server preset
2021-06-11 16:57:02 +02:00
def generate_default_presets(self):
for preset_name, preset_data in DEFAULT_PRESETS.items():
2021-06-14 15:17:07 +02:00
existing_preset = self.server_preset.get(preset_name)
if existing_preset :
continue
2021-06-11 16:57:02 +02:00
new_server = self.server_preset.add()
new_server.name = preset_name
new_server.server_ip = preset_data.get('server_ip')
new_server.server_port = preset_data.get('server_port')
new_server.server_password = preset_data.get('server_password',None)
2021-06-11 12:13:23 +02:00
def client_list_callback(scene, context):
from . import operators
2020-04-14 17:22:28 +02:00
items = [(RP_COMMON, RP_COMMON, "")]
username = get_preferences().username
if session:
client_ids = session.online_users.keys()
for id in client_ids:
2020-04-14 17:22:28 +02:00
name_desc = id
if id == username:
name_desc += " (self)"
2020-04-14 17:22:28 +02:00
items.append((id, name_desc, ""))
return items
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")
class SessionProps(bpy.types.PropertyGroup):
session_mode: bpy.props.EnumProperty(
name='session_mode',
description='session mode',
items={
2020-06-11 15:00:51 +02:00
('HOST', 'HOST', 'host a session'),
('CONNECT', 'JOIN', 'connect to a session')},
default='CONNECT')
clients: bpy.props.EnumProperty(
name="clients",
description="client enum",
items=client_list_callback)
enable_presence: bpy.props.BoolProperty(
name="Presence overlay",
description='Enable overlay drawing module',
default=True,
2020-04-14 17:22:28 +02:00
)
presence_show_selected: bpy.props.BoolProperty(
name="Show selected objects",
description='Enable selection overlay ',
default=True,
)
presence_show_user: bpy.props.BoolProperty(
name="Show users",
description='Enable user overlay ',
default=True,
2020-04-14 17:22:28 +02:00
)
2021-06-29 17:10:59 +02:00
presence_show_mode: bpy.props.BoolProperty(
name="Show users current mode",
description='Enable user mode overlay ',
2021-06-30 15:34:03 +02:00
default=False,
2021-06-29 17:10:59 +02:00
)
presence_show_far_user: bpy.props.BoolProperty(
name="Show users on different scenes",
description="Show user on different scenes",
default=False,
2020-04-14 17:22:28 +02:00
)
2020-10-05 23:38:52 +02:00
presence_show_session_status: bpy.props.BoolProperty(
name="Show session status ",
description="Show session status on the viewport",
default=True,
)
filter_owned: bpy.props.BoolProperty(
name="filter_owned",
description='Show only owned datablocks',
default=True
)
2021-05-19 15:59:36 +02:00
filter_name: bpy.props.StringProperty(
name="filter_name",
default="",
description='Node name filter',
)
admin: bpy.props.BoolProperty(
name="admin",
description='Connect as admin',
default=False
)
2020-06-17 22:11:20 +02:00
internet_ip: bpy.props.StringProperty(
name="internet ip",
default="no found",
description='Internet interface ip',
)
user_snap_running: bpy.props.BoolProperty(
default=False
)
time_snap_running: bpy.props.BoolProperty(
default=False
)
2020-06-17 22:11:20 +02:00
is_host: bpy.props.BoolProperty(
default=False
)
2020-04-14 17:22:28 +02:00
classes = (
SessionUser,
SessionProps,
2020-04-02 18:42:41 +02:00
ReplicationFlags,
ReplicatedDatablock,
ServerPreset,
SessionPrefs,
)
2020-03-11 22:42:09 +01:00
def register():
from bpy.utils import register_class
for cls in classes:
register_class(cls)
prefs = bpy.context.preferences.addons[__package__].preferences
if len(prefs.supported_datablocks) == 0:
2020-04-22 17:04:14 +02:00
logging.debug('Generating bl_types preferences')
prefs.generate_supported_types()
2021-06-11 12:13:23 +02:00
# at launch server presets
2021-06-11 16:57:02 +02:00
prefs.generate_default_presets()
2021-06-11 12:13:23 +02:00
2020-03-11 22:42:09 +01:00
def unregister():
from bpy.utils import unregister_class
for cls in reversed(classes):
2020-03-11 22:42:09 +01:00
unregister_class(cls)