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 #####
|
|
|
|
|
|
|
|
|
2019-02-08 15:44:02 +01:00
|
|
|
import bpy
|
2019-09-30 13:35:50 +02:00
|
|
|
|
2020-09-15 12:31:46 +02:00
|
|
|
from . import operators
|
2020-09-21 16:47:49 +02:00
|
|
|
from .utils import get_preferences, get_expanded_icon, get_folder_size
|
2020-07-10 16:50:09 +02:00
|
|
|
from replication.constants import (ADDED, ERROR, FETCHED,
|
2020-02-07 17:56:58 +01:00
|
|
|
MODIFIED, RP_COMMON, UP,
|
|
|
|
STATE_ACTIVE, STATE_AUTH,
|
|
|
|
STATE_CONFIG, STATE_SYNCING,
|
|
|
|
STATE_INITIAL, STATE_SRV_SYNC,
|
2020-03-20 11:46:43 +01:00
|
|
|
STATE_WAITING, STATE_QUITTING,
|
2020-06-16 00:02:02 +02:00
|
|
|
STATE_LOBBY,
|
2020-03-20 11:46:43 +01:00
|
|
|
STATE_LAUNCHING_SERVICES)
|
2020-09-18 23:25:01 +02:00
|
|
|
from replication import __version__
|
2019-04-24 17:42:23 +02:00
|
|
|
|
2019-09-13 16:46:26 +02:00
|
|
|
ICONS_PROP_STATES = ['TRIA_DOWN', # ADDED
|
2019-08-28 18:58:18 +02:00
|
|
|
'TRIA_UP', # COMMITED
|
|
|
|
'KEYTYPE_KEYFRAME_VEC', # PUSHED
|
|
|
|
'TRIA_DOWN', # FETCHED
|
|
|
|
'FILE_REFRESH', # UP
|
|
|
|
'TRIA_UP'] # CHANGED
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2020-06-16 16:40:00 +02:00
|
|
|
|
|
|
|
def printProgressBar(iteration, total, prefix='', suffix='', decimals=1, length=100, fill='█', fill_empty=' '):
|
2020-02-09 00:41:00 +01:00
|
|
|
"""
|
|
|
|
Call in a loop to create terminal progress bar
|
|
|
|
@params:
|
|
|
|
iteration - Required : current iteration (Int)
|
|
|
|
total - Required : total iterations (Int)
|
|
|
|
prefix - Optional : prefix string (Str)
|
|
|
|
suffix - Optional : suffix string (Str)
|
|
|
|
decimals - Optional : positive number of decimals in percent complete (Int)
|
|
|
|
length - Optional : character length of bar (Int)
|
|
|
|
fill - Optional : bar fill character (Str)
|
2020-02-09 23:37:02 +01:00
|
|
|
From here:
|
|
|
|
https://gist.github.com/greenstick/b23e475d2bfdc3a82e34eaa1f6781ee4
|
2020-02-09 00:41:00 +01:00
|
|
|
"""
|
2020-07-07 15:50:05 +02:00
|
|
|
if total == 0:
|
|
|
|
return ""
|
2020-02-09 00:41:00 +01:00
|
|
|
filledLength = int(length * iteration // total)
|
|
|
|
bar = fill * filledLength + fill_empty * (length - filledLength)
|
2020-04-22 17:04:14 +02:00
|
|
|
return f"{prefix} |{bar}| {iteration}/{total}{suffix}"
|
2020-02-09 00:41:00 +01:00
|
|
|
|
2020-06-16 16:40:00 +02:00
|
|
|
|
2020-02-07 17:56:58 +01:00
|
|
|
def get_state_str(state):
|
2020-03-20 11:46:43 +01:00
|
|
|
state_str = 'UNKNOWN'
|
2020-02-07 17:56:58 +01:00
|
|
|
if state == STATE_WAITING:
|
2020-02-09 00:41:00 +01:00
|
|
|
state_str = 'WARMING UP DATA'
|
2020-02-07 17:56:58 +01:00
|
|
|
elif state == STATE_SYNCING:
|
2020-06-17 12:10:16 +02:00
|
|
|
state_str = 'FETCHING'
|
2020-02-07 17:56:58 +01:00
|
|
|
elif state == STATE_AUTH:
|
|
|
|
state_str = 'AUTHENTIFICATION'
|
|
|
|
elif state == STATE_CONFIG:
|
|
|
|
state_str = 'CONFIGURATION'
|
|
|
|
elif state == STATE_ACTIVE:
|
2020-02-09 00:41:00 +01:00
|
|
|
state_str = 'ONLINE'
|
2020-02-07 17:56:58 +01:00
|
|
|
elif state == STATE_SRV_SYNC:
|
2020-06-17 12:10:16 +02:00
|
|
|
state_str = 'PUSHING'
|
2020-02-19 16:22:06 +01:00
|
|
|
elif state == STATE_INITIAL:
|
|
|
|
state_str = 'INIT'
|
2020-02-20 13:17:28 +01:00
|
|
|
elif state == STATE_QUITTING:
|
2020-06-17 12:10:16 +02:00
|
|
|
state_str = 'QUITTING'
|
2020-03-20 11:46:43 +01:00
|
|
|
elif state == STATE_LAUNCHING_SERVICES:
|
|
|
|
state_str = 'LAUNCHING SERVICES'
|
2020-06-16 00:02:02 +02:00
|
|
|
elif state == STATE_LOBBY:
|
|
|
|
state_str = 'LOBBY'
|
2020-03-20 11:46:43 +01:00
|
|
|
|
2020-02-07 17:56:58 +01:00
|
|
|
return state_str
|
2019-08-22 15:35:21 +02:00
|
|
|
|
2020-06-16 16:40:00 +02:00
|
|
|
|
2019-05-15 14:52:45 +02:00
|
|
|
class SESSION_PT_settings(bpy.types.Panel):
|
2019-07-02 16:43:30 +02:00
|
|
|
"""Settings panel"""
|
2019-06-13 18:09:16 +02:00
|
|
|
bl_idname = "MULTIUSER_SETTINGS_PT_panel"
|
2020-07-24 14:55:14 +02:00
|
|
|
bl_label = " "
|
2019-06-13 18:09:16 +02:00
|
|
|
bl_space_type = 'VIEW_3D'
|
|
|
|
bl_region_type = 'UI'
|
|
|
|
bl_category = "Multiuser"
|
2019-02-08 15:44:02 +01:00
|
|
|
|
2020-06-17 12:10:16 +02:00
|
|
|
def draw_header(self, context):
|
|
|
|
layout = self.layout
|
|
|
|
if operators.client and operators.client.state['STATE'] != STATE_INITIAL:
|
|
|
|
cli_state = operators.client.state
|
|
|
|
state = operators.client.state.get('STATE')
|
|
|
|
connection_icon = "KEYTYPE_MOVING_HOLD_VEC"
|
|
|
|
|
|
|
|
if state == STATE_ACTIVE:
|
|
|
|
connection_icon = 'PROP_ON'
|
|
|
|
else:
|
|
|
|
connection_icon = 'PROP_CON'
|
|
|
|
|
|
|
|
layout.label(text=f"Session - {get_state_str(cli_state['STATE'])}", icon=connection_icon)
|
|
|
|
else:
|
2020-09-18 23:25:01 +02:00
|
|
|
layout.label(text=f"Session - v{__version__}",icon="PROP_OFF")
|
2020-06-17 12:10:16 +02:00
|
|
|
|
2019-02-08 15:44:02 +01:00
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
2019-08-22 15:35:21 +02:00
|
|
|
layout.use_property_split = True
|
2019-08-09 16:47:15 +02:00
|
|
|
row = layout.row()
|
2020-06-17 22:11:20 +02:00
|
|
|
runtime_settings = context.window_manager.session
|
2020-09-15 12:31:46 +02:00
|
|
|
settings = get_preferences()
|
2019-03-15 16:50:59 +01:00
|
|
|
|
2019-08-23 12:28:57 +02:00
|
|
|
if hasattr(context.window_manager, 'session'):
|
2019-08-06 11:34:39 +02:00
|
|
|
# STATE INITIAL
|
2019-08-23 12:28:57 +02:00
|
|
|
if not operators.client \
|
2020-02-07 17:56:58 +01:00
|
|
|
or (operators.client and operators.client.state['STATE'] == STATE_INITIAL):
|
2019-08-09 15:20:49 +02:00
|
|
|
pass
|
2019-04-08 17:01:02 +02:00
|
|
|
else:
|
2020-02-07 17:56:58 +01:00
|
|
|
cli_state = operators.client.state
|
2020-06-16 16:40:00 +02:00
|
|
|
|
2020-06-17 12:10:16 +02:00
|
|
|
|
2020-02-09 00:41:00 +01:00
|
|
|
row = layout.row()
|
2020-06-16 16:40:00 +02:00
|
|
|
|
2020-02-20 13:17:28 +01:00
|
|
|
current_state = cli_state['STATE']
|
|
|
|
|
2019-08-23 12:28:57 +02:00
|
|
|
# STATE ACTIVE
|
2020-07-24 16:02:19 +02:00
|
|
|
if current_state in [STATE_ACTIVE]:
|
2019-04-08 17:01:02 +02:00
|
|
|
row.operator("session.stop", icon='QUIT', text="Exit")
|
2019-07-01 18:04:35 +02:00
|
|
|
row = layout.row()
|
2020-06-17 22:11:20 +02:00
|
|
|
if runtime_settings.is_host:
|
|
|
|
row = row.box()
|
2020-07-24 14:55:14 +02:00
|
|
|
row.label(text=f"LAN: {runtime_settings.internet_ip}", icon='INFO')
|
2020-06-17 22:11:20 +02:00
|
|
|
row = layout.row()
|
2020-07-24 16:02:19 +02:00
|
|
|
if current_state == STATE_LOBBY:
|
|
|
|
row = row.box()
|
|
|
|
row.label(text=f"Waiting the session to start", icon='INFO')
|
|
|
|
row = layout.row()
|
|
|
|
row.operator("session.stop", icon='QUIT', text="Exit")
|
2020-02-20 13:17:28 +01:00
|
|
|
# CONNECTION STATE
|
2020-06-16 16:40:00 +02:00
|
|
|
elif current_state in [STATE_SRV_SYNC,
|
|
|
|
STATE_SYNCING,
|
|
|
|
STATE_AUTH,
|
|
|
|
STATE_CONFIG,
|
|
|
|
STATE_WAITING]:
|
|
|
|
|
|
|
|
if cli_state['STATE'] in [STATE_SYNCING, STATE_SRV_SYNC, STATE_WAITING]:
|
2020-02-20 13:17:28 +01:00
|
|
|
box = row.box()
|
2020-02-09 00:41:00 +01:00
|
|
|
box.label(text=printProgressBar(
|
|
|
|
cli_state['CURRENT'],
|
|
|
|
cli_state['TOTAL'],
|
|
|
|
length=16
|
|
|
|
))
|
|
|
|
|
2019-04-22 15:01:09 +02:00
|
|
|
row = layout.row()
|
|
|
|
row.operator("session.stop", icon='QUIT', text="CANCEL")
|
2020-02-20 13:17:28 +01:00
|
|
|
elif current_state == STATE_QUITTING:
|
|
|
|
row = layout.row()
|
|
|
|
box = row.box()
|
2020-02-20 18:36:53 +01:00
|
|
|
|
|
|
|
num_online_services = 0
|
|
|
|
for name, state in operators.client.services_state.items():
|
|
|
|
if state == STATE_ACTIVE:
|
|
|
|
num_online_services += 1
|
|
|
|
|
2020-06-16 16:40:00 +02:00
|
|
|
total_online_services = len(
|
|
|
|
operators.client.services_state)
|
2020-02-20 18:36:53 +01:00
|
|
|
|
|
|
|
box.label(text=printProgressBar(
|
2020-06-16 16:40:00 +02:00
|
|
|
total_online_services-num_online_services,
|
|
|
|
total_online_services,
|
|
|
|
length=16
|
|
|
|
))
|
2019-05-03 11:32:14 +02:00
|
|
|
|
2020-06-10 18:43:21 +02:00
|
|
|
|
2019-08-09 15:20:49 +02:00
|
|
|
class SESSION_PT_settings_network(bpy.types.Panel):
|
|
|
|
bl_idname = "MULTIUSER_SETTINGS_NETWORK_PT_panel"
|
|
|
|
bl_label = "Network"
|
|
|
|
bl_space_type = 'VIEW_3D'
|
|
|
|
bl_region_type = 'UI'
|
|
|
|
bl_parent_id = 'MULTIUSER_SETTINGS_PT_panel'
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2019-08-09 15:20:49 +02:00
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
2019-08-23 12:28:57 +02:00
|
|
|
return not operators.client \
|
2020-02-07 17:56:58 +01:00
|
|
|
or (operators.client and operators.client.state['STATE'] == 0)
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2020-06-16 16:40:00 +02:00
|
|
|
def draw_header(self, context):
|
|
|
|
self.layout.label(text="", icon='URL')
|
|
|
|
|
2019-08-09 15:20:49 +02:00
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
2020-06-16 16:40:00 +02:00
|
|
|
|
2020-02-28 17:34:30 +01:00
|
|
|
runtime_settings = context.window_manager.session
|
2020-09-15 12:31:46 +02:00
|
|
|
settings = get_preferences()
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2019-08-09 15:20:49 +02:00
|
|
|
# USER SETTINGS
|
|
|
|
row = layout.row()
|
2020-02-28 17:34:30 +01:00
|
|
|
row.prop(runtime_settings, "session_mode", expand=True)
|
2019-08-09 15:20:49 +02:00
|
|
|
row = layout.row()
|
|
|
|
|
2020-02-19 18:07:25 +01:00
|
|
|
box = row.box()
|
2020-06-16 16:40:00 +02:00
|
|
|
|
2020-02-28 17:34:30 +01:00
|
|
|
if runtime_settings.session_mode == 'HOST':
|
2020-06-16 18:25:49 +02:00
|
|
|
row = box.row()
|
|
|
|
row.label(text="Port:")
|
|
|
|
row.prop(settings, "port", text="")
|
|
|
|
row = box.row()
|
2020-06-17 19:21:57 +02:00
|
|
|
row.label(text="Start from:")
|
|
|
|
row.prop(settings, "init_method", text="")
|
|
|
|
row = box.row()
|
2020-06-11 18:35:52 +02:00
|
|
|
row.label(text="Admin password:")
|
2020-06-04 18:38:03 +02:00
|
|
|
row.prop(runtime_settings, "password", text="")
|
2019-08-09 15:20:49 +02:00
|
|
|
row = box.row()
|
|
|
|
row.operator("session.start", text="HOST").host = True
|
|
|
|
else:
|
2020-06-16 18:25:49 +02:00
|
|
|
row = box.row()
|
|
|
|
row.prop(settings, "ip", text="IP")
|
|
|
|
row = box.row()
|
|
|
|
row.label(text="Port:")
|
|
|
|
row.prop(settings, "port", text="")
|
2020-06-17 19:21:57 +02:00
|
|
|
|
2020-06-16 18:25:49 +02:00
|
|
|
row = box.row()
|
2020-06-16 16:40:00 +02:00
|
|
|
row.prop(runtime_settings, "admin", text='Connect as admin', icon='DISCLOSURE_TRI_DOWN' if runtime_settings.admin
|
|
|
|
else 'DISCLOSURE_TRI_RIGHT')
|
2020-06-04 18:38:03 +02:00
|
|
|
if runtime_settings.admin:
|
|
|
|
row = box.row()
|
|
|
|
row.label(text="Password:")
|
|
|
|
row.prop(runtime_settings, "password", text="")
|
2019-08-09 15:20:49 +02:00
|
|
|
row = box.row()
|
|
|
|
row.operator("session.start", text="CONNECT").host = False
|
|
|
|
|
|
|
|
|
|
|
|
class SESSION_PT_settings_user(bpy.types.Panel):
|
|
|
|
bl_idname = "MULTIUSER_SETTINGS_USER_PT_panel"
|
2020-06-16 16:40:00 +02:00
|
|
|
bl_label = "User info"
|
2019-08-09 15:20:49 +02:00
|
|
|
bl_space_type = 'VIEW_3D'
|
|
|
|
bl_region_type = 'UI'
|
|
|
|
bl_parent_id = 'MULTIUSER_SETTINGS_PT_panel'
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2019-08-09 15:20:49 +02:00
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
2019-08-23 12:28:57 +02:00
|
|
|
return not operators.client \
|
2020-02-07 17:56:58 +01:00
|
|
|
or (operators.client and operators.client.state['STATE'] == 0)
|
2020-06-16 16:40:00 +02:00
|
|
|
|
|
|
|
def draw_header(self, context):
|
|
|
|
self.layout.label(text="", icon='USER')
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2019-08-09 15:20:49 +02:00
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
|
|
|
|
2020-02-28 17:34:30 +01:00
|
|
|
runtime_settings = context.window_manager.session
|
2020-09-15 12:31:46 +02:00
|
|
|
settings = get_preferences()
|
2020-06-16 16:40:00 +02:00
|
|
|
|
2019-08-09 15:20:49 +02:00
|
|
|
row = layout.row()
|
|
|
|
# USER SETTINGS
|
2019-09-24 18:37:36 +02:00
|
|
|
row.prop(settings, "username", text="name")
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2019-08-09 15:20:49 +02:00
|
|
|
row = layout.row()
|
2019-08-23 12:28:57 +02:00
|
|
|
row.prop(settings, "client_color", text="color")
|
2019-08-09 15:20:49 +02:00
|
|
|
row = layout.row()
|
|
|
|
|
2019-03-14 12:09:33 +01:00
|
|
|
|
2020-06-17 12:10:16 +02:00
|
|
|
class SESSION_PT_advanced_settings(bpy.types.Panel):
|
2019-08-28 18:58:18 +02:00
|
|
|
bl_idname = "MULTIUSER_SETTINGS_REPLICATION_PT_panel"
|
2019-09-16 17:24:48 +02:00
|
|
|
bl_label = "Advanced"
|
2019-08-28 18:58:18 +02:00
|
|
|
bl_space_type = 'VIEW_3D'
|
|
|
|
bl_region_type = 'UI'
|
|
|
|
bl_parent_id = 'MULTIUSER_SETTINGS_PT_panel'
|
2019-09-24 14:42:59 +02:00
|
|
|
bl_options = {'DEFAULT_CLOSED'}
|
2019-08-28 18:58:18 +02:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
|
|
|
return not operators.client \
|
2020-02-07 17:56:58 +01:00
|
|
|
or (operators.client and operators.client.state['STATE'] == 0)
|
2019-08-28 18:58:18 +02:00
|
|
|
|
2020-06-16 16:40:00 +02:00
|
|
|
def draw_header(self, context):
|
2020-06-17 12:10:16 +02:00
|
|
|
self.layout.label(text="", icon='PREFERENCES')
|
2020-06-16 16:40:00 +02:00
|
|
|
|
2019-08-28 18:58:18 +02:00
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
|
|
|
|
2020-02-28 17:34:30 +01:00
|
|
|
runtime_settings = context.window_manager.session
|
2020-09-15 12:31:46 +02:00
|
|
|
settings = get_preferences()
|
2020-02-28 17:34:30 +01:00
|
|
|
|
2020-06-17 12:10:16 +02:00
|
|
|
|
|
|
|
net_section = layout.row().box()
|
2020-09-15 12:31:46 +02:00
|
|
|
net_section.prop(
|
|
|
|
settings,
|
|
|
|
"sidebar_advanced_net_expanded",
|
|
|
|
text="Network",
|
|
|
|
icon=get_expanded_icon(settings.sidebar_advanced_net_expanded),
|
|
|
|
emboss=False)
|
|
|
|
|
|
|
|
if settings.sidebar_advanced_net_expanded:
|
|
|
|
net_section_row = net_section.row()
|
|
|
|
net_section_row.label(text="IPC Port:")
|
|
|
|
net_section_row.prop(settings, "ipc_port", text="")
|
|
|
|
net_section_row = net_section.row()
|
|
|
|
net_section_row.label(text="Timeout (ms):")
|
|
|
|
net_section_row.prop(settings, "connection_timeout", text="")
|
2020-06-17 12:10:16 +02:00
|
|
|
|
|
|
|
replication_section = layout.row().box()
|
2020-09-15 12:31:46 +02:00
|
|
|
replication_section.prop(
|
|
|
|
settings,
|
|
|
|
"sidebar_advanced_rep_expanded",
|
|
|
|
text="Replication",
|
|
|
|
icon=get_expanded_icon(settings.sidebar_advanced_rep_expanded),
|
|
|
|
emboss=False)
|
|
|
|
|
|
|
|
if settings.sidebar_advanced_rep_expanded:
|
|
|
|
replication_section_row = replication_section.row()
|
|
|
|
|
|
|
|
replication_section_row.label(text="Sync flags", icon='COLLECTION_NEW')
|
|
|
|
replication_section_row = replication_section.row()
|
|
|
|
replication_section_row.prop(settings.sync_flags, "sync_render_settings")
|
2020-09-08 23:09:42 +02:00
|
|
|
replication_section_row = replication_section.row()
|
2020-09-15 12:31:46 +02:00
|
|
|
|
|
|
|
replication_section_row.prop(settings, "enable_editmode_updates")
|
|
|
|
replication_section_row = replication_section.row()
|
|
|
|
if settings.enable_editmode_updates:
|
|
|
|
warning = replication_section_row.box()
|
|
|
|
warning.label(text="Don't use this with heavy meshes !", icon='ERROR')
|
|
|
|
replication_section_row = replication_section.row()
|
|
|
|
|
|
|
|
replication_section_row.label(text="Update method", icon='RECOVER_LAST')
|
|
|
|
replication_section_row = replication_section.row()
|
|
|
|
replication_section_row.prop(settings, "update_method", expand=True)
|
|
|
|
replication_section_row = replication_section.row()
|
|
|
|
replication_timers = replication_section_row.box()
|
|
|
|
replication_timers.label(text="Replication timers", icon='TIME')
|
|
|
|
if settings.update_method == "DEFAULT":
|
|
|
|
replication_timers = replication_timers.row()
|
|
|
|
# Replication frequencies
|
|
|
|
flow = replication_timers.grid_flow(
|
|
|
|
row_major=True, columns=0, even_columns=True, even_rows=False, align=True)
|
2020-09-03 15:59:19 +02:00
|
|
|
line = flow.row(align=True)
|
2020-09-15 12:31:46 +02:00
|
|
|
line.label(text=" ")
|
2020-09-03 15:59:19 +02:00
|
|
|
line.separator()
|
2020-09-15 12:31:46 +02:00
|
|
|
line.label(text="refresh (sec)")
|
|
|
|
line.label(text="apply (sec)")
|
|
|
|
|
|
|
|
for item in settings.supported_datablocks:
|
|
|
|
line = flow.row(align=True)
|
|
|
|
line.prop(item, "auto_push", text="", icon=item.icon)
|
|
|
|
line.separator()
|
|
|
|
line.prop(item, "bl_delay_refresh", text="")
|
|
|
|
line.prop(item, "bl_delay_apply", text="")
|
|
|
|
else:
|
|
|
|
replication_timers = replication_timers.row()
|
|
|
|
replication_timers.label(text="Update rate (ms):")
|
|
|
|
replication_timers.prop(settings, "depsgraph_update_rate", text="")
|
|
|
|
|
2020-09-21 12:12:19 +02:00
|
|
|
cache_section = layout.row().box()
|
|
|
|
cache_section.prop(
|
|
|
|
settings,
|
|
|
|
"sidebar_advanced_cache_expanded",
|
|
|
|
text="Cache",
|
|
|
|
icon=get_expanded_icon(settings.sidebar_advanced_cache_expanded),
|
|
|
|
emboss=False)
|
|
|
|
if settings.sidebar_advanced_cache_expanded:
|
|
|
|
cache_section_row = cache_section.row()
|
|
|
|
cache_section_row.label(text="Cache directory:")
|
|
|
|
cache_section_row = cache_section.row()
|
|
|
|
cache_section_row.prop(settings, "cache_directory", text="")
|
|
|
|
cache_section_row = cache_section.row()
|
|
|
|
cache_section_row.label(text="Clear memory filecache:")
|
|
|
|
cache_section_row.prop(settings, "clear_memory_filecache", text="")
|
2020-09-21 16:47:49 +02:00
|
|
|
cache_section_row = cache_section.row()
|
|
|
|
cache_section_row.operator('session.clear_cache', text=f"Clear cache ({get_folder_size(settings.cache_directory)})")
|
2020-09-15 12:40:51 +02:00
|
|
|
log_section = layout.row().box()
|
|
|
|
log_section.prop(
|
2020-09-15 12:31:46 +02:00
|
|
|
settings,
|
|
|
|
"sidebar_advanced_log_expanded",
|
|
|
|
text="Logging",
|
|
|
|
icon=get_expanded_icon(settings.sidebar_advanced_log_expanded),
|
|
|
|
emboss=False)
|
|
|
|
|
|
|
|
if settings.sidebar_advanced_log_expanded:
|
2020-09-15 12:40:51 +02:00
|
|
|
log_section_row = log_section.row()
|
|
|
|
log_section_row.label(text="Log level:")
|
2020-09-15 12:31:46 +02:00
|
|
|
log_section_row.prop(settings, 'logging_level', text="")
|
2019-05-15 14:52:45 +02:00
|
|
|
class SESSION_PT_user(bpy.types.Panel):
|
2019-06-13 18:09:16 +02:00
|
|
|
bl_idname = "MULTIUSER_USER_PT_panel"
|
2020-01-17 18:15:37 +01:00
|
|
|
bl_label = "Online users"
|
2019-06-13 18:09:16 +02:00
|
|
|
bl_space_type = 'VIEW_3D'
|
|
|
|
bl_region_type = 'UI'
|
2019-08-09 15:20:49 +02:00
|
|
|
bl_parent_id = 'MULTIUSER_SETTINGS_PT_panel'
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2019-03-13 17:02:53 +01:00
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
2020-06-16 15:19:38 +02:00
|
|
|
return operators.client and operators.client.state['STATE'] in [STATE_ACTIVE, STATE_LOBBY]
|
2019-03-13 17:02:53 +01:00
|
|
|
|
2020-06-16 16:40:00 +02:00
|
|
|
def draw_header(self, context):
|
|
|
|
self.layout.label(text="", icon='USER')
|
|
|
|
|
2019-03-13 17:02:53 +01:00
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
2020-01-17 18:15:37 +01:00
|
|
|
online_users = context.window_manager.online_users
|
|
|
|
selected_user = context.window_manager.user_index
|
2020-09-15 12:31:46 +02:00
|
|
|
settings = get_preferences()
|
2020-06-16 16:40:00 +02:00
|
|
|
active_user = online_users[selected_user] if len(
|
|
|
|
online_users)-1 >= selected_user else 0
|
2020-04-03 14:59:33 +02:00
|
|
|
runtime_settings = context.window_manager.session
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2019-03-13 17:02:53 +01:00
|
|
|
# Create a simple row.
|
2020-01-17 18:15:37 +01:00
|
|
|
row = layout.row()
|
|
|
|
box = row.box()
|
2020-06-11 15:25:58 +02:00
|
|
|
split = box.split(factor=0.35)
|
2020-01-17 18:15:37 +01:00
|
|
|
split.label(text="user")
|
2020-03-05 16:19:13 +01:00
|
|
|
split = split.split(factor=0.5)
|
2020-06-16 18:50:08 +02:00
|
|
|
split.label(text="location")
|
2020-01-17 18:15:37 +01:00
|
|
|
split.label(text="frame")
|
|
|
|
split.label(text="ping")
|
2019-08-09 16:47:15 +02:00
|
|
|
|
2020-01-17 18:15:37 +01:00
|
|
|
row = layout.row()
|
2020-06-16 16:40:00 +02:00
|
|
|
layout.template_list("SESSION_UL_users", "", context.window_manager,
|
|
|
|
"online_users", context.window_manager, "user_index")
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2020-01-17 18:15:37 +01:00
|
|
|
if active_user != 0 and active_user.username != settings.username:
|
|
|
|
row = layout.row()
|
|
|
|
user_operations = row.split()
|
2020-07-24 16:02:19 +02:00
|
|
|
if operators.client.state['STATE'] == STATE_ACTIVE:
|
|
|
|
|
|
|
|
user_operations.alert = context.window_manager.session.time_snap_running
|
|
|
|
user_operations.operator(
|
|
|
|
"session.snapview",
|
|
|
|
text="",
|
|
|
|
icon='VIEW_CAMERA').target_client = active_user.username
|
|
|
|
|
|
|
|
user_operations.alert = context.window_manager.session.user_snap_running
|
|
|
|
user_operations.operator(
|
|
|
|
"session.snaptime",
|
|
|
|
text="",
|
|
|
|
icon='TIME').target_client = active_user.username
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2020-06-10 18:43:21 +02:00
|
|
|
if operators.client.online_users[settings.username]['admin']:
|
2020-04-03 14:59:33 +02:00
|
|
|
user_operations.operator(
|
|
|
|
"session.kick",
|
|
|
|
text="",
|
|
|
|
icon='CANCEL').user = active_user.username
|
|
|
|
|
2019-04-17 13:39:36 +02:00
|
|
|
|
2020-01-17 18:15:37 +01:00
|
|
|
class SESSION_UL_users(bpy.types.UIList):
|
|
|
|
def draw_item(self, context, layout, data, item, icon, active_data, active_propname, index, flt_flag):
|
2020-01-19 23:10:22 +01:00
|
|
|
session = operators.client
|
2020-09-15 12:31:46 +02:00
|
|
|
settings = get_preferences()
|
2020-01-17 18:15:37 +01:00
|
|
|
is_local_user = item.username == settings.username
|
2020-01-19 23:10:22 +01:00
|
|
|
ping = '-'
|
2020-01-22 16:17:48 +01:00
|
|
|
frame_current = '-'
|
2020-03-05 16:19:13 +01:00
|
|
|
scene_current = '-'
|
2020-06-17 22:11:20 +02:00
|
|
|
status_icon = 'BLANK1'
|
2020-01-19 23:10:22 +01:00
|
|
|
if session:
|
2020-01-22 16:17:48 +01:00
|
|
|
user = session.online_users.get(item.username)
|
|
|
|
if user:
|
|
|
|
ping = str(user['latency'])
|
|
|
|
metadata = user.get('metadata')
|
2020-02-07 17:08:36 +01:00
|
|
|
if metadata and 'frame_current' in metadata:
|
2020-08-28 15:27:46 +02:00
|
|
|
frame_current = str(metadata.get('frame_current','-'))
|
|
|
|
scene_current = metadata.get('scene_current','-')
|
2020-06-09 23:56:20 +02:00
|
|
|
if user['admin']:
|
|
|
|
status_icon = 'FAKE_USER_ON'
|
2020-06-11 15:25:58 +02:00
|
|
|
split = layout.split(factor=0.35)
|
2020-06-09 23:56:20 +02:00
|
|
|
split.label(text=item.username, icon=status_icon)
|
2020-03-05 16:19:13 +01:00
|
|
|
split = split.split(factor=0.5)
|
|
|
|
split.label(text=scene_current)
|
2020-01-22 16:17:48 +01:00
|
|
|
split.label(text=frame_current)
|
2020-01-19 23:10:22 +01:00
|
|
|
split.label(text=ping)
|
2020-03-05 16:19:13 +01:00
|
|
|
|
2019-07-01 15:59:51 +02:00
|
|
|
|
2019-09-24 14:42:59 +02:00
|
|
|
class SESSION_PT_presence(bpy.types.Panel):
|
|
|
|
bl_idname = "MULTIUSER_MODULE_PT_panel"
|
|
|
|
bl_label = "Presence overlay"
|
|
|
|
bl_space_type = 'VIEW_3D'
|
|
|
|
bl_region_type = 'UI'
|
|
|
|
bl_parent_id = 'MULTIUSER_SETTINGS_PT_panel'
|
2019-09-24 18:37:36 +02:00
|
|
|
bl_options = {'DEFAULT_CLOSED'}
|
2019-09-24 14:42:59 +02:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
2020-02-08 23:23:00 +01:00
|
|
|
return not operators.client \
|
|
|
|
or (operators.client and operators.client.state['STATE'] in [STATE_INITIAL, STATE_ACTIVE])
|
2019-09-24 14:42:59 +02:00
|
|
|
|
|
|
|
def draw_header(self, context):
|
2020-06-16 16:40:00 +02:00
|
|
|
self.layout.prop(context.window_manager.session,
|
|
|
|
"enable_presence", text="",icon='OVERLAY')
|
2019-09-24 14:42:59 +02:00
|
|
|
|
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
|
|
|
|
|
|
|
settings = context.window_manager.session
|
|
|
|
layout.active = settings.enable_presence
|
|
|
|
col = layout.column()
|
2020-06-16 16:40:00 +02:00
|
|
|
col.prop(settings, "presence_show_selected")
|
|
|
|
col.prop(settings, "presence_show_user")
|
2020-03-05 17:20:04 +01:00
|
|
|
row = layout.column()
|
2020-06-16 16:40:00 +02:00
|
|
|
row.active = settings.presence_show_user
|
|
|
|
row.prop(settings, "presence_show_far_user")
|
|
|
|
|
2019-09-24 14:42:59 +02:00
|
|
|
|
2020-02-19 16:22:06 +01:00
|
|
|
class SESSION_PT_services(bpy.types.Panel):
|
|
|
|
bl_idname = "MULTIUSER_SERVICE_PT_panel"
|
|
|
|
bl_label = "Services"
|
|
|
|
bl_space_type = 'VIEW_3D'
|
|
|
|
bl_region_type = 'UI'
|
|
|
|
bl_parent_id = 'MULTIUSER_SETTINGS_PT_panel'
|
2020-02-20 14:17:50 +01:00
|
|
|
bl_options = {'DEFAULT_CLOSED'}
|
2020-02-19 16:22:06 +01:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
|
|
|
return operators.client and operators.client.state['STATE'] == 2
|
|
|
|
|
2020-06-17 22:11:20 +02:00
|
|
|
def draw_header(self, context):
|
|
|
|
self.layout.label(text="", icon='FILE_CACHE')
|
|
|
|
|
2020-02-19 16:22:06 +01:00
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
|
|
|
online_users = context.window_manager.online_users
|
|
|
|
selected_user = context.window_manager.user_index
|
|
|
|
settings = context.window_manager.session
|
2020-06-16 16:40:00 +02:00
|
|
|
active_user = online_users[selected_user] if len(online_users)-1 >= selected_user else 0
|
2020-06-16 18:04:27 +02:00
|
|
|
|
2020-02-19 16:22:06 +01:00
|
|
|
# Create a simple row.
|
|
|
|
for name, state in operators.client.services_state.items():
|
2020-02-19 18:07:25 +01:00
|
|
|
row = layout.row()
|
2020-02-19 16:22:06 +01:00
|
|
|
row.label(text=name)
|
2020-06-16 16:40:00 +02:00
|
|
|
row.label(text=get_state_str(state))
|
2020-02-19 16:22:06 +01:00
|
|
|
|
2019-09-24 14:42:59 +02:00
|
|
|
|
2019-08-23 12:28:57 +02:00
|
|
|
def draw_property(context, parent, property_uuid, level=0):
|
2020-09-15 12:31:46 +02:00
|
|
|
settings = get_preferences()
|
2020-02-28 17:34:30 +01:00
|
|
|
runtime_settings = context.window_manager.session
|
2019-08-26 17:27:12 +02:00
|
|
|
item = operators.client.get(uuid=property_uuid)
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2020-01-22 14:33:34 +01:00
|
|
|
if item.state == ERROR:
|
2019-08-22 15:35:21 +02:00
|
|
|
return
|
|
|
|
|
2019-08-23 12:28:57 +02:00
|
|
|
area_msg = parent.row(align=True)
|
2019-08-22 15:35:21 +02:00
|
|
|
if level > 0:
|
|
|
|
for i in range(level):
|
|
|
|
area_msg.label(text="")
|
|
|
|
line = area_msg.box()
|
|
|
|
|
2019-10-09 14:09:11 +02:00
|
|
|
name = item.data['name'] if item.data else item.uuid
|
2019-08-22 15:35:21 +02:00
|
|
|
|
2019-08-23 12:28:57 +02:00
|
|
|
detail_item_box = line.row(align=True)
|
2019-08-22 15:35:21 +02:00
|
|
|
|
2019-08-28 18:58:18 +02:00
|
|
|
detail_item_box.label(text="",
|
2020-02-28 17:34:30 +01:00
|
|
|
icon=settings.supported_datablocks[item.str_type].icon)
|
2020-04-22 17:04:14 +02:00
|
|
|
detail_item_box.label(text=f"{name}")
|
2019-08-28 16:19:32 +02:00
|
|
|
|
|
|
|
# Operations
|
|
|
|
|
2020-06-04 18:38:03 +02:00
|
|
|
have_right_to_modify = item.owner == settings.username or \
|
2020-06-16 16:40:00 +02:00
|
|
|
item.owner == RP_COMMON
|
|
|
|
|
2019-09-27 17:16:02 +02:00
|
|
|
if have_right_to_modify:
|
|
|
|
detail_item_box.operator(
|
|
|
|
"session.commit",
|
|
|
|
text="",
|
|
|
|
icon='TRIA_UP').target = item.uuid
|
|
|
|
detail_item_box.separator()
|
2020-06-16 16:40:00 +02:00
|
|
|
|
2019-08-28 16:19:32 +02:00
|
|
|
if item.state in [FETCHED, UP]:
|
2019-08-23 12:28:57 +02:00
|
|
|
detail_item_box.operator(
|
|
|
|
"session.apply",
|
|
|
|
text="",
|
2019-08-28 18:58:18 +02:00
|
|
|
icon=ICONS_PROP_STATES[item.state]).target = item.uuid
|
2019-09-13 16:46:26 +02:00
|
|
|
elif item.state in [MODIFIED, ADDED]:
|
2019-08-28 13:13:32 +02:00
|
|
|
detail_item_box.operator(
|
|
|
|
"session.commit",
|
|
|
|
text="",
|
2019-08-28 18:58:18 +02:00
|
|
|
icon=ICONS_PROP_STATES[item.state]).target = item.uuid
|
2019-08-22 15:35:21 +02:00
|
|
|
else:
|
2019-08-28 18:58:18 +02:00
|
|
|
detail_item_box.label(text="", icon=ICONS_PROP_STATES[item.state])
|
2019-08-22 15:35:21 +02:00
|
|
|
|
|
|
|
right_icon = "DECORATE_UNLOCKED"
|
2019-09-24 14:42:59 +02:00
|
|
|
if not have_right_to_modify:
|
2019-08-23 12:28:57 +02:00
|
|
|
right_icon = "DECORATE_LOCKED"
|
|
|
|
|
2019-08-28 16:19:32 +02:00
|
|
|
if have_right_to_modify:
|
|
|
|
ro = detail_item_box.operator(
|
|
|
|
"session.right", text="", icon=right_icon)
|
|
|
|
ro.key = property_uuid
|
2019-08-22 15:35:21 +02:00
|
|
|
|
2019-08-28 16:19:32 +02:00
|
|
|
detail_item_box.operator(
|
|
|
|
"session.remove_prop", text="", icon="X").property_path = property_uuid
|
|
|
|
else:
|
2019-08-28 18:58:18 +02:00
|
|
|
detail_item_box.label(text="", icon="DECORATE_LOCKED")
|
|
|
|
|
2019-08-22 15:35:21 +02:00
|
|
|
|
2020-06-16 15:19:38 +02:00
|
|
|
class SESSION_PT_repository(bpy.types.Panel):
|
2019-06-13 18:09:16 +02:00
|
|
|
bl_idname = "MULTIUSER_PROPERTIES_PT_panel"
|
2020-06-10 18:43:21 +02:00
|
|
|
bl_label = "Repository"
|
2019-06-13 18:09:16 +02:00
|
|
|
bl_space_type = 'VIEW_3D'
|
|
|
|
bl_region_type = 'UI'
|
2020-04-14 17:22:28 +02:00
|
|
|
bl_parent_id = 'MULTIUSER_SETTINGS_PT_panel'
|
2019-03-13 17:02:53 +01:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
2020-07-24 16:02:19 +02:00
|
|
|
session = operators.client
|
2020-09-15 12:31:46 +02:00
|
|
|
settings = get_preferences()
|
2020-07-24 16:02:19 +02:00
|
|
|
admin = False
|
|
|
|
|
|
|
|
if session and hasattr(session,'online_users'):
|
|
|
|
usr = session.online_users.get(settings.username)
|
|
|
|
if usr:
|
|
|
|
admin = usr['admin']
|
2020-06-10 18:43:21 +02:00
|
|
|
return hasattr(context.window_manager, 'session') and \
|
2020-06-16 16:40:00 +02:00
|
|
|
operators.client and \
|
2020-07-24 16:02:19 +02:00
|
|
|
(operators.client.state['STATE'] == STATE_ACTIVE or \
|
|
|
|
operators.client.state['STATE'] == STATE_LOBBY and admin)
|
2019-03-13 17:02:53 +01:00
|
|
|
|
2019-07-01 18:04:35 +02:00
|
|
|
def draw_header(self, context):
|
|
|
|
self.layout.label(text="", icon='OUTLINER_OB_GROUP_INSTANCE')
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2019-03-13 17:02:53 +01:00
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2020-06-10 18:43:21 +02:00
|
|
|
# Filters
|
2020-09-15 12:31:46 +02:00
|
|
|
settings = get_preferences()
|
2020-06-10 18:43:21 +02:00
|
|
|
runtime_settings = context.window_manager.session
|
2020-06-16 15:19:38 +02:00
|
|
|
|
|
|
|
session = operators.client
|
|
|
|
usr = session.online_users.get(settings.username)
|
2020-06-16 16:40:00 +02:00
|
|
|
|
|
|
|
row = layout.row()
|
|
|
|
|
2020-06-16 15:19:38 +02:00
|
|
|
if session.state['STATE'] == STATE_ACTIVE:
|
2019-08-28 18:58:18 +02:00
|
|
|
flow = layout.grid_flow(
|
2019-09-16 11:16:46 +02:00
|
|
|
row_major=True,
|
|
|
|
columns=0,
|
|
|
|
even_columns=True,
|
|
|
|
even_rows=False,
|
|
|
|
align=True)
|
2019-08-28 18:58:18 +02:00
|
|
|
|
2020-02-28 17:34:30 +01:00
|
|
|
for item in settings.supported_datablocks:
|
2019-09-13 17:00:15 +02:00
|
|
|
col = flow.column(align=True)
|
2019-09-16 11:16:46 +02:00
|
|
|
col.prop(item, "use_as_filter", text="", icon=item.icon)
|
2019-04-01 16:14:21 +02:00
|
|
|
|
2019-09-18 23:10:36 +02:00
|
|
|
row = layout.row(align=True)
|
2020-02-28 17:34:30 +01:00
|
|
|
row.prop(runtime_settings, "filter_owned", text="Show only owned")
|
2019-09-18 23:10:36 +02:00
|
|
|
|
2019-04-11 14:39:31 +02:00
|
|
|
row = layout.row(align=True)
|
2019-09-18 23:10:36 +02:00
|
|
|
|
|
|
|
# Properties
|
2020-02-28 17:34:30 +01:00
|
|
|
types_filter = [t.type_name for t in settings.supported_datablocks
|
2019-09-16 11:16:46 +02:00
|
|
|
if t.use_as_filter]
|
2019-09-13 17:00:15 +02:00
|
|
|
|
2019-09-18 23:10:36 +02:00
|
|
|
key_to_filter = operators.client.list(
|
2020-02-28 17:34:30 +01:00
|
|
|
filter_owner=settings.username) if runtime_settings.filter_owned else operators.client.list()
|
2019-09-18 23:10:36 +02:00
|
|
|
|
|
|
|
client_keys = [key for key in key_to_filter
|
2020-06-16 16:40:00 +02:00
|
|
|
if operators.client.get(uuid=key).str_type
|
|
|
|
in types_filter]
|
2019-08-09 16:47:15 +02:00
|
|
|
|
2020-06-10 18:43:21 +02:00
|
|
|
if client_keys:
|
2019-08-08 17:17:58 +02:00
|
|
|
col = layout.column(align=True)
|
2019-08-09 16:47:15 +02:00
|
|
|
for key in client_keys:
|
2019-08-23 12:28:57 +02:00
|
|
|
draw_property(context, col, key)
|
|
|
|
|
2019-04-11 14:39:31 +02:00
|
|
|
else:
|
2019-09-13 17:00:15 +02:00
|
|
|
row.label(text="Empty")
|
2019-04-11 14:39:31 +02:00
|
|
|
|
2020-06-16 15:19:38 +02:00
|
|
|
elif session.state['STATE'] == STATE_LOBBY and usr and usr['admin']:
|
2020-06-10 18:43:21 +02:00
|
|
|
row.operator("session.init", icon='TOOL_SETTINGS', text="Init")
|
|
|
|
else:
|
2020-06-17 12:10:16 +02:00
|
|
|
row.label(text="Waiting to start")
|
2019-02-08 15:44:02 +01:00
|
|
|
|
2020-06-16 16:40:00 +02:00
|
|
|
|
2019-02-08 15:44:02 +01:00
|
|
|
classes = (
|
2020-01-17 18:15:37 +01:00
|
|
|
SESSION_UL_users,
|
2019-05-15 14:52:45 +02:00
|
|
|
SESSION_PT_settings,
|
2019-08-09 15:20:49 +02:00
|
|
|
SESSION_PT_settings_user,
|
|
|
|
SESSION_PT_settings_network,
|
2019-09-24 18:37:36 +02:00
|
|
|
SESSION_PT_presence,
|
2020-06-17 12:10:16 +02:00
|
|
|
SESSION_PT_advanced_settings,
|
2019-05-15 14:52:45 +02:00
|
|
|
SESSION_PT_user,
|
2020-04-14 17:22:28 +02:00
|
|
|
SESSION_PT_services,
|
2020-06-16 15:19:38 +02:00
|
|
|
SESSION_PT_repository,
|
2020-06-16 16:40:00 +02:00
|
|
|
|
2019-02-08 15:44:02 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
register, unregister = bpy.utils.register_classes_factory(classes)
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
2019-03-14 12:09:33 +01:00
|
|
|
register()
|