multi-user/multi_user/ui.py

672 lines
24 KiB
Python
Raw Normal View History

2020-03-20 21:56:50 +08: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 bpy
2019-09-30 19:35:50 +08:00
from . import operators
2020-09-21 22:47:49 +08:00
from .utils import get_preferences, get_expanded_icon, get_folder_size
2020-07-10 22:50:09 +08:00
from replication.constants import (ADDED, ERROR, FETCHED,
2020-02-08 00:56:58 +08:00
MODIFIED, RP_COMMON, UP,
STATE_ACTIVE, STATE_AUTH,
STATE_CONFIG, STATE_SYNCING,
STATE_INITIAL, STATE_SRV_SYNC,
STATE_WAITING, STATE_QUITTING,
2020-06-16 06:02:02 +08:00
STATE_LOBBY,
STATE_LAUNCHING_SERVICES)
from replication import __version__
2019-04-24 23:42:23 +08:00
2019-09-13 22:46:26 +08:00
ICONS_PROP_STATES = ['TRIA_DOWN', # ADDED
'TRIA_UP', # COMMITED
'KEYTYPE_KEYFRAME_VEC', # PUSHED
'TRIA_DOWN', # FETCHED
'FILE_REFRESH', # UP
'TRIA_UP'] # CHANGED
2019-08-23 18:28:57 +08:00
2020-06-16 22:40:00 +08:00
def printProgressBar(iteration, total, prefix='', suffix='', decimals=1, length=100, fill='', fill_empty=' '):
2020-02-09 07:41:00 +08: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-10 06:37:02 +08:00
From here:
https://gist.github.com/greenstick/b23e475d2bfdc3a82e34eaa1f6781ee4
2020-02-09 07:41:00 +08:00
"""
if total == 0:
return ""
2020-02-09 07:41:00 +08:00
filledLength = int(length * iteration // total)
bar = fill * filledLength + fill_empty * (length - filledLength)
2020-04-22 23:04:14 +08:00
return f"{prefix} |{bar}| {iteration}/{total}{suffix}"
2020-02-09 07:41:00 +08:00
2020-06-16 22:40:00 +08:00
2020-02-08 00:56:58 +08:00
def get_state_str(state):
state_str = 'UNKNOWN'
2020-02-08 00:56:58 +08:00
if state == STATE_WAITING:
2020-02-09 07:41:00 +08:00
state_str = 'WARMING UP DATA'
2020-02-08 00:56:58 +08:00
elif state == STATE_SYNCING:
2020-06-17 18:10:16 +08:00
state_str = 'FETCHING'
2020-02-08 00:56:58 +08:00
elif state == STATE_AUTH:
state_str = 'AUTHENTIFICATION'
elif state == STATE_CONFIG:
state_str = 'CONFIGURATION'
elif state == STATE_ACTIVE:
2020-02-09 07:41:00 +08:00
state_str = 'ONLINE'
2020-02-08 00:56:58 +08:00
elif state == STATE_SRV_SYNC:
2020-06-17 18:10:16 +08:00
state_str = 'PUSHING'
elif state == STATE_INITIAL:
state_str = 'INIT'
2020-02-20 20:17:28 +08:00
elif state == STATE_QUITTING:
2020-06-17 18:10:16 +08:00
state_str = 'QUITTING'
elif state == STATE_LAUNCHING_SERVICES:
state_str = 'LAUNCHING SERVICES'
2020-06-16 06:02:02 +08:00
elif state == STATE_LOBBY:
state_str = 'LOBBY'
2020-02-08 00:56:58 +08:00
return state_str
2020-06-16 22:40:00 +08:00
class SESSION_PT_settings(bpy.types.Panel):
2019-07-02 22:43:30 +08:00
"""Settings panel"""
bl_idname = "MULTIUSER_SETTINGS_PT_panel"
2020-07-24 20:55:14 +08:00
bl_label = " "
bl_space_type = 'VIEW_3D'
bl_region_type = 'UI'
bl_category = "Multiuser"
2020-06-17 18:10:16 +08: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:
layout.label(text=f"Session - v{__version__}",icon="PROP_OFF")
2020-06-17 18:10:16 +08:00
def draw(self, context):
layout = self.layout
layout.use_property_split = True
2019-08-09 22:47:15 +08:00
row = layout.row()
2020-06-18 04:11:20 +08:00
runtime_settings = context.window_manager.session
settings = get_preferences()
2019-03-15 23:50:59 +08:00
2019-08-23 18:28:57 +08:00
if hasattr(context.window_manager, 'session'):
2019-08-06 17:34:39 +08:00
# STATE INITIAL
2019-08-23 18:28:57 +08:00
if not operators.client \
2020-02-08 00:56:58 +08:00
or (operators.client and operators.client.state['STATE'] == STATE_INITIAL):
2019-08-09 21:20:49 +08:00
pass
2019-04-08 23:01:02 +08:00
else:
2020-09-25 17:23:36 +08:00
cli_state = operators.client.state
2020-02-09 07:41:00 +08:00
row = layout.row()
2020-06-16 22:40:00 +08:00
2020-02-20 20:17:28 +08:00
current_state = cli_state['STATE']
2020-09-25 17:23:36 +08:00
info_msg = None
if current_state in [STATE_ACTIVE]:
row = row.split(factor=0.3)
row.prop(settings.sync_flags, "sync_render_settings",text="",icon_only=True, icon='SCENE')
row.prop(settings.sync_flags, "sync_during_editmode", text="",icon_only=True, icon='EDITMODE_HLT')
row.prop(settings.sync_flags, "sync_active_camera", text="",icon_only=True, icon='OBJECT_DATAMODE')
row= layout.row()
2020-09-25 17:23:36 +08:00
if current_state in [STATE_ACTIVE] and runtime_settings.is_host:
info_msg = f"LAN: {runtime_settings.internet_ip}"
2020-07-24 22:02:19 +08:00
if current_state == STATE_LOBBY:
2020-09-25 17:23:36 +08:00
info_msg = "Waiting the session to start."
if info_msg:
info_box = row.box()
info_box.row().label(text=info_msg,icon='INFO')
# Progress bar
if current_state in [STATE_SYNCING, STATE_SRV_SYNC, STATE_WAITING]:
info_box = row.box()
info_box.row().label(text=printProgressBar(
cli_state['CURRENT'],
cli_state['TOTAL'],
2020-06-16 22:40:00 +08:00
length=16
))
2019-05-03 17:32:14 +08:00
2020-09-25 17:23:36 +08:00
layout.row().operator("session.stop", icon='QUIT', text="Exit")
2020-06-11 00:43:21 +08:00
2019-08-09 21:20:49 +08: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 18:28:57 +08:00
2019-08-09 21:20:49 +08:00
@classmethod
def poll(cls, context):
2019-08-23 18:28:57 +08:00
return not operators.client \
2020-02-08 00:56:58 +08:00
or (operators.client and operators.client.state['STATE'] == 0)
2019-08-23 18:28:57 +08:00
2020-06-16 22:40:00 +08:00
def draw_header(self, context):
self.layout.label(text="", icon='URL')
2019-08-09 21:20:49 +08:00
def draw(self, context):
layout = self.layout
2020-06-16 22:40:00 +08:00
runtime_settings = context.window_manager.session
settings = get_preferences()
2019-08-23 18:28:57 +08:00
2019-08-09 21:20:49 +08:00
# USER SETTINGS
row = layout.row()
row.prop(runtime_settings, "session_mode", expand=True)
2019-08-09 21:20:49 +08:00
row = layout.row()
2020-02-20 01:07:25 +08:00
box = row.box()
2020-06-16 22:40:00 +08:00
if runtime_settings.session_mode == 'HOST':
row = box.row()
row.label(text="Port:")
row.prop(settings, "port", text="")
row = box.row()
row.label(text="Start from:")
row.prop(settings, "init_method", text="")
row = box.row()
2020-06-12 00:35:52 +08:00
row.label(text="Admin password:")
row.prop(runtime_settings, "password", text="")
2019-08-09 21:20:49 +08:00
row = box.row()
row.operator("session.start", text="HOST").host = True
else:
row = box.row()
row.prop(settings, "ip", text="IP")
row = box.row()
row.label(text="Port:")
row.prop(settings, "port", text="")
row = box.row()
2020-06-16 22:40:00 +08:00
row.prop(runtime_settings, "admin", text='Connect as admin', icon='DISCLOSURE_TRI_DOWN' if runtime_settings.admin
else 'DISCLOSURE_TRI_RIGHT')
if runtime_settings.admin:
row = box.row()
row.label(text="Password:")
row.prop(runtime_settings, "password", text="")
2019-08-09 21:20:49 +08: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 22:40:00 +08:00
bl_label = "User info"
2019-08-09 21:20:49 +08:00
bl_space_type = 'VIEW_3D'
bl_region_type = 'UI'
bl_parent_id = 'MULTIUSER_SETTINGS_PT_panel'
2019-08-23 18:28:57 +08:00
2019-08-09 21:20:49 +08:00
@classmethod
def poll(cls, context):
2019-08-23 18:28:57 +08:00
return not operators.client \
2020-02-08 00:56:58 +08:00
or (operators.client and operators.client.state['STATE'] == 0)
2020-06-16 22:40:00 +08:00
def draw_header(self, context):
self.layout.label(text="", icon='USER')
2019-08-23 18:28:57 +08:00
2019-08-09 21:20:49 +08:00
def draw(self, context):
layout = self.layout
runtime_settings = context.window_manager.session
settings = get_preferences()
2020-06-16 22:40:00 +08:00
2019-08-09 21:20:49 +08:00
row = layout.row()
# USER SETTINGS
2019-09-25 00:37:36 +08:00
row.prop(settings, "username", text="name")
2019-08-23 18:28:57 +08:00
2019-08-09 21:20:49 +08:00
row = layout.row()
2019-08-23 18:28:57 +08:00
row.prop(settings, "client_color", text="color")
2019-08-09 21:20:49 +08:00
row = layout.row()
2019-03-14 19:09:33 +08:00
2020-06-17 18:10:16 +08:00
class SESSION_PT_advanced_settings(bpy.types.Panel):
bl_idname = "MULTIUSER_SETTINGS_REPLICATION_PT_panel"
bl_label = "Advanced"
bl_space_type = 'VIEW_3D'
bl_region_type = 'UI'
bl_parent_id = 'MULTIUSER_SETTINGS_PT_panel'
2019-09-24 20:42:59 +08:00
bl_options = {'DEFAULT_CLOSED'}
@classmethod
def poll(cls, context):
return not operators.client \
2020-02-08 00:56:58 +08:00
or (operators.client and operators.client.state['STATE'] == 0)
2020-06-16 22:40:00 +08:00
def draw_header(self, context):
2020-06-17 18:10:16 +08:00
self.layout.label(text="", icon='PREFERENCES')
2020-06-16 22:40:00 +08:00
def draw(self, context):
layout = self.layout
runtime_settings = context.window_manager.session
settings = get_preferences()
2020-06-17 18:10:16 +08:00
net_section = layout.row().box()
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 18:10:16 +08:00
replication_section = layout.row().box()
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-09 05:09:42 +08:00
replication_section_row = replication_section.row()
2020-09-25 17:33:35 +08:00
replication_section_row.prop(settings.sync_flags, "sync_active_camera")
replication_section_row = replication_section.row()
2020-09-25 17:23:36 +08:00
replication_section_row.prop(settings.sync_flags, "sync_during_editmode")
replication_section_row = replication_section.row()
2020-09-25 17:23:36 +08:00
if settings.sync_flags.sync_during_editmode:
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)
line = flow.row(align=True)
line.label(text=" ")
line.separator()
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="")
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 22:47:49 +08: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 18:40:51 +08:00
log_section = layout.row().box()
log_section.prop(
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 18:40:51 +08:00
log_section_row = log_section.row()
log_section_row.label(text="Log level:")
log_section_row.prop(settings, 'logging_level', text="")
class SESSION_PT_user(bpy.types.Panel):
bl_idname = "MULTIUSER_USER_PT_panel"
bl_label = "Online users"
bl_space_type = 'VIEW_3D'
bl_region_type = 'UI'
2019-08-09 21:20:49 +08:00
bl_parent_id = 'MULTIUSER_SETTINGS_PT_panel'
2019-08-23 18:28:57 +08:00
2019-03-14 00:02:53 +08:00
@classmethod
def poll(cls, context):
return operators.client and operators.client.state['STATE'] in [STATE_ACTIVE, STATE_LOBBY]
2019-03-14 00:02:53 +08:00
2020-06-16 22:40:00 +08:00
def draw_header(self, context):
self.layout.label(text="", icon='USER')
2019-03-14 00:02:53 +08:00
def draw(self, context):
layout = self.layout
online_users = context.window_manager.online_users
selected_user = context.window_manager.user_index
settings = get_preferences()
2020-06-16 22:40:00 +08:00
active_user = online_users[selected_user] if len(
online_users)-1 >= selected_user else 0
2020-04-03 20:59:33 +08:00
runtime_settings = context.window_manager.session
2019-08-23 18:28:57 +08:00
2019-03-14 00:02:53 +08:00
# Create a simple row.
row = layout.row()
box = row.box()
split = box.split(factor=0.35)
split.label(text="user")
split = split.split(factor=0.5)
2020-06-17 00:50:08 +08:00
split.label(text="location")
split.label(text="frame")
split.label(text="ping")
2019-08-09 22:47:15 +08:00
row = layout.row()
2020-06-16 22:40:00 +08:00
layout.template_list("SESSION_UL_users", "", context.window_manager,
"online_users", context.window_manager, "user_index")
2019-08-23 18:28:57 +08:00
if active_user != 0 and active_user.username != settings.username:
row = layout.row()
user_operations = row.split()
2020-07-24 22:02:19 +08: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 18:28:57 +08:00
2020-06-11 00:43:21 +08:00
if operators.client.online_users[settings.username]['admin']:
2020-04-03 20:59:33 +08:00
user_operations.operator(
"session.kick",
text="",
icon='CANCEL').user = active_user.username
2019-04-17 19:39:36 +08: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-20 06:10:22 +08:00
session = operators.client
settings = get_preferences()
is_local_user = item.username == settings.username
2020-01-20 06:10:22 +08:00
ping = '-'
frame_current = '-'
scene_current = '-'
2020-06-18 04:11:20 +08:00
status_icon = 'BLANK1'
2020-01-20 06:10:22 +08:00
if session:
user = session.online_users.get(item.username)
if user:
ping = str(user['latency'])
metadata = user.get('metadata')
if metadata and 'frame_current' in metadata:
2020-08-28 21:27:46 +08:00
frame_current = str(metadata.get('frame_current','-'))
scene_current = metadata.get('scene_current','-')
if user['admin']:
status_icon = 'FAKE_USER_ON'
split = layout.split(factor=0.35)
split.label(text=item.username, icon=status_icon)
split = split.split(factor=0.5)
split.label(text=scene_current)
split.label(text=frame_current)
2020-01-20 06:10:22 +08:00
split.label(text=ping)
2019-07-01 21:59:51 +08:00
2019-09-24 20:42:59 +08: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-25 00:37:36 +08:00
bl_options = {'DEFAULT_CLOSED'}
2019-09-24 20:42:59 +08:00
@classmethod
def poll(cls, context):
return not operators.client \
or (operators.client and operators.client.state['STATE'] in [STATE_INITIAL, STATE_ACTIVE])
2019-09-24 20:42:59 +08:00
def draw_header(self, context):
2020-06-16 22:40:00 +08:00
self.layout.prop(context.window_manager.session,
"enable_presence", text="",icon='OVERLAY')
2019-09-24 20:42:59 +08:00
def draw(self, context):
layout = self.layout
settings = context.window_manager.session
layout.active = settings.enable_presence
col = layout.column()
2020-06-16 22:40:00 +08:00
col.prop(settings, "presence_show_selected")
col.prop(settings, "presence_show_user")
row = layout.column()
2020-06-16 22:40:00 +08:00
row.active = settings.presence_show_user
row.prop(settings, "presence_show_far_user")
2019-08-23 18:28:57 +08:00
def draw_property(context, parent, property_uuid, level=0):
settings = get_preferences()
runtime_settings = context.window_manager.session
item = operators.client.get(uuid=property_uuid)
2019-08-23 18:28:57 +08:00
2020-01-22 21:33:34 +08:00
if item.state == ERROR:
return
2019-08-23 18:28:57 +08:00
area_msg = parent.row(align=True)
if level > 0:
for i in range(level):
area_msg.label(text="")
line = area_msg.box()
2019-10-09 20:09:11 +08:00
name = item.data['name'] if item.data else item.uuid
2019-08-23 18:28:57 +08:00
detail_item_box = line.row(align=True)
detail_item_box.label(text="",
icon=settings.supported_datablocks[item.str_type].icon)
2020-04-22 23:04:14 +08:00
detail_item_box.label(text=f"{name}")
2019-08-28 22:19:32 +08:00
# Operations
have_right_to_modify = item.owner == settings.username or \
2020-06-16 22:40:00 +08:00
item.owner == RP_COMMON
2019-09-27 23:16:02 +08: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 22:40:00 +08:00
2019-08-28 22:19:32 +08:00
if item.state in [FETCHED, UP]:
2019-08-23 18:28:57 +08:00
detail_item_box.operator(
"session.apply",
text="",
icon=ICONS_PROP_STATES[item.state]).target = item.uuid
2019-09-13 22:46:26 +08:00
elif item.state in [MODIFIED, ADDED]:
2019-08-28 19:13:32 +08:00
detail_item_box.operator(
"session.commit",
text="",
icon=ICONS_PROP_STATES[item.state]).target = item.uuid
else:
detail_item_box.label(text="", icon=ICONS_PROP_STATES[item.state])
right_icon = "DECORATE_UNLOCKED"
2019-09-24 20:42:59 +08:00
if not have_right_to_modify:
2019-08-23 18:28:57 +08:00
right_icon = "DECORATE_LOCKED"
2019-08-28 22:19:32 +08:00
if have_right_to_modify:
ro = detail_item_box.operator(
"session.right", text="", icon=right_icon)
ro.key = property_uuid
2019-08-28 22:19:32 +08:00
detail_item_box.operator(
"session.remove_prop", text="", icon="X").property_path = property_uuid
else:
detail_item_box.label(text="", icon="DECORATE_LOCKED")
class SESSION_PT_repository(bpy.types.Panel):
bl_idname = "MULTIUSER_PROPERTIES_PT_panel"
2020-06-11 00:43:21 +08:00
bl_label = "Repository"
bl_space_type = 'VIEW_3D'
bl_region_type = 'UI'
2020-04-14 23:22:28 +08:00
bl_parent_id = 'MULTIUSER_SETTINGS_PT_panel'
2019-03-14 00:02:53 +08:00
@classmethod
def poll(cls, context):
2020-07-24 22:02:19 +08:00
session = operators.client
settings = get_preferences()
2020-07-24 22:02:19 +08:00
admin = False
if session and hasattr(session,'online_users'):
usr = session.online_users.get(settings.username)
if usr:
admin = usr['admin']
2020-06-11 00:43:21 +08:00
return hasattr(context.window_manager, 'session') and \
2020-06-16 22:40:00 +08:00
operators.client and \
2020-07-24 22:02:19 +08:00
(operators.client.state['STATE'] == STATE_ACTIVE or \
operators.client.state['STATE'] == STATE_LOBBY and admin)
2019-03-14 00:02:53 +08:00
2019-07-02 00:04:35 +08:00
def draw_header(self, context):
self.layout.label(text="", icon='OUTLINER_OB_GROUP_INSTANCE')
2019-08-23 18:28:57 +08:00
2019-03-14 00:02:53 +08:00
def draw(self, context):
layout = self.layout
2019-08-23 18:28:57 +08:00
2020-06-11 00:43:21 +08:00
# Filters
settings = get_preferences()
2020-06-11 00:43:21 +08:00
runtime_settings = context.window_manager.session
session = operators.client
usr = session.online_users.get(settings.username)
2020-06-16 22:40:00 +08:00
row = layout.row()
if session.state['STATE'] == STATE_ACTIVE:
flow = layout.grid_flow(
row_major=True,
columns=0,
even_columns=True,
even_rows=False,
align=True)
for item in settings.supported_datablocks:
2019-09-13 23:00:15 +08:00
col = flow.column(align=True)
col.prop(item, "use_as_filter", text="", icon=item.icon)
2019-04-01 22:14:21 +08:00
2019-09-19 05:10:36 +08:00
row = layout.row(align=True)
row.prop(runtime_settings, "filter_owned", text="Show only owned")
2019-09-19 05:10:36 +08:00
2019-04-11 20:39:31 +08:00
row = layout.row(align=True)
2019-09-19 05:10:36 +08:00
# Properties
types_filter = [t.type_name for t in settings.supported_datablocks
if t.use_as_filter]
2019-09-13 23:00:15 +08:00
2019-09-19 05:10:36 +08:00
key_to_filter = operators.client.list(
filter_owner=settings.username) if runtime_settings.filter_owned else operators.client.list()
2019-09-19 05:10:36 +08:00
client_keys = [key for key in key_to_filter
2020-06-16 22:40:00 +08:00
if operators.client.get(uuid=key).str_type
in types_filter]
2019-08-09 22:47:15 +08:00
2020-06-11 00:43:21 +08:00
if client_keys:
col = layout.column(align=True)
2019-08-09 22:47:15 +08:00
for key in client_keys:
2019-08-23 18:28:57 +08:00
draw_property(context, col, key)
2019-04-11 20:39:31 +08:00
else:
2019-09-13 23:00:15 +08:00
row.label(text="Empty")
2019-04-11 20:39:31 +08:00
elif session.state['STATE'] == STATE_LOBBY and usr and usr['admin']:
2020-06-11 00:43:21 +08:00
row.operator("session.init", icon='TOOL_SETTINGS', text="Init")
else:
2020-06-17 18:10:16 +08:00
row.label(text="Waiting to start")
class VIEW3D_PT_overlay_session(bpy.types.Panel):
bl_space_type = 'VIEW_3D'
bl_region_type = 'HEADER'
bl_parent_id = 'VIEW3D_PT_overlay'
bl_label = "Multi-user"
@classmethod
def poll(cls, context):
return True
def draw(self, context):
layout = self.layout
view = context.space_data
overlay = view.overlay
display_all = overlay.show_overlays
col = layout.column()
col.active = display_all
row = col.row(align=True)
settings = context.window_manager.session
layout.active = settings.enable_presence
col = layout.column()
col.prop(settings, "presence_show_selected")
col.prop(settings, "presence_show_user")
row = layout.column()
row.active = settings.presence_show_user
row.prop(settings, "presence_show_far_user")
2020-06-16 22:40:00 +08:00
classes = (
SESSION_UL_users,
SESSION_PT_settings,
2019-08-09 21:20:49 +08:00
SESSION_PT_settings_user,
SESSION_PT_settings_network,
2019-09-25 00:37:36 +08:00
SESSION_PT_presence,
2020-06-17 18:10:16 +08:00
SESSION_PT_advanced_settings,
SESSION_PT_user,
SESSION_PT_repository,
VIEW3D_PT_overlay_session,
)
register, unregister = bpy.utils.register_classes_factory(classes)
if __name__ == "__main__":
2019-03-14 19:09:33 +08:00
register()