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-08-28 12:57:09 +02:00
|
|
|
import logging
|
2020-12-22 16:04:50 +01:00
|
|
|
import sys
|
2021-03-11 15:45:48 +01:00
|
|
|
import traceback
|
2019-09-30 13:35:50 +02:00
|
|
|
import bpy
|
2020-12-22 16:04:50 +01:00
|
|
|
from replication.constants import (FETCHED, RP_COMMON, STATE_ACTIVE,
|
|
|
|
STATE_INITIAL, STATE_LOBBY, STATE_QUITTING,
|
|
|
|
STATE_SRV_SYNC, STATE_SYNCING, UP)
|
2021-01-11 19:54:57 +01:00
|
|
|
from replication.exception import NonAuthorizedOperationError, ContextError
|
2021-03-09 14:07:59 +01:00
|
|
|
from replication.interface import session
|
2021-05-17 11:12:18 +02:00
|
|
|
from replication import porcelain
|
2019-09-30 13:35:50 +02:00
|
|
|
|
2020-12-22 16:04:50 +01:00
|
|
|
from . import operators, utils
|
2021-06-24 16:01:14 +02:00
|
|
|
from .presence import (UserFrustumWidget, UserNameWidget, UserModeWidget, UserSelectionWidget,
|
2020-12-22 16:04:50 +01:00
|
|
|
generate_user_camera, get_view_matrix, refresh_3d_view,
|
|
|
|
refresh_sidebar_view, renderer)
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2021-06-18 14:34:11 +02:00
|
|
|
from . import shared_data
|
|
|
|
|
2020-12-22 16:04:50 +01:00
|
|
|
this = sys.modules[__name__]
|
2020-10-05 22:34:43 +02:00
|
|
|
|
2020-12-22 16:04:50 +01:00
|
|
|
# Registered timers
|
|
|
|
this.registry = dict()
|
2020-12-09 14:49:26 +01:00
|
|
|
|
|
|
|
def is_annotating(context: bpy.types.Context):
|
2020-11-18 19:13:22 +01:00
|
|
|
""" Check if the annotate mode is enabled
|
|
|
|
"""
|
2021-07-06 15:39:19 +02:00
|
|
|
active_tool = bpy.context.workspace.tools.from_space_view3d_mode('OBJECT', create=False)
|
|
|
|
return (active_tool and active_tool.idname == 'builtin.annotate')
|
2020-12-09 14:49:26 +01:00
|
|
|
|
2019-08-13 18:00:54 +02:00
|
|
|
|
2020-12-22 16:04:50 +01:00
|
|
|
class Timer(object):
|
2019-08-13 18:00:54 +02:00
|
|
|
"""Timer binder interface for blender
|
|
|
|
|
|
|
|
Run a bpy.app.Timer in the background looping at the given rate
|
|
|
|
"""
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2020-12-22 16:04:50 +01:00
|
|
|
def __init__(self, timeout=10, id=None):
|
|
|
|
self._timeout = timeout
|
2020-12-09 18:34:56 +01:00
|
|
|
self.is_running = False
|
2020-12-22 16:04:50 +01:00
|
|
|
self.id = id if id else self.__class__.__name__
|
2019-08-13 18:00:54 +02:00
|
|
|
|
|
|
|
def register(self):
|
|
|
|
"""Register the timer into the blender timer system
|
|
|
|
"""
|
2020-10-05 22:34:43 +02:00
|
|
|
|
2020-12-09 18:34:56 +01:00
|
|
|
if not self.is_running:
|
2020-12-22 16:04:50 +01:00
|
|
|
this.registry[self.id] = self
|
2020-10-01 10:58:30 +02:00
|
|
|
bpy.app.timers.register(self.main)
|
2020-12-09 18:34:56 +01:00
|
|
|
self.is_running = True
|
2020-10-01 10:58:30 +02:00
|
|
|
logging.debug(f"Register {self.__class__.__name__}")
|
|
|
|
else:
|
2020-10-05 22:34:43 +02:00
|
|
|
logging.debug(
|
|
|
|
f"Timer {self.__class__.__name__} already registered")
|
2019-09-27 14:50:00 +02:00
|
|
|
|
|
|
|
def main(self):
|
2020-12-09 18:34:56 +01:00
|
|
|
try:
|
|
|
|
self.execute()
|
|
|
|
except Exception as e:
|
|
|
|
logging.error(e)
|
|
|
|
self.unregister()
|
2021-05-09 17:42:56 +02:00
|
|
|
traceback.print_exc()
|
2021-03-04 14:22:54 +01:00
|
|
|
session.disconnect(reason=f"Error during timer {self.id} execution")
|
2020-12-09 18:34:56 +01:00
|
|
|
else:
|
|
|
|
if self.is_running:
|
|
|
|
return self._timeout
|
2019-09-27 14:50:00 +02:00
|
|
|
|
2019-08-13 18:00:54 +02:00
|
|
|
def execute(self):
|
|
|
|
"""Main timer loop
|
|
|
|
"""
|
2019-09-27 14:50:00 +02:00
|
|
|
raise NotImplementedError
|
2019-08-13 18:00:54 +02:00
|
|
|
|
|
|
|
def unregister(self):
|
|
|
|
"""Unnegister the timer of the blender timer system
|
|
|
|
"""
|
2019-09-27 14:50:00 +02:00
|
|
|
if bpy.app.timers.is_registered(self.main):
|
2020-12-22 16:04:50 +01:00
|
|
|
logging.info(f"Unregistering {self.id}")
|
2019-09-27 14:50:00 +02:00
|
|
|
bpy.app.timers.unregister(self.main)
|
2021-06-18 14:34:11 +02:00
|
|
|
|
2020-12-22 16:04:50 +01:00
|
|
|
del this.registry[self.id]
|
2020-12-09 18:34:56 +01:00
|
|
|
self.is_running = False
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2020-12-22 16:04:50 +01:00
|
|
|
class SessionBackupTimer(Timer):
|
|
|
|
def __init__(self, timeout=10, filepath=None):
|
2020-12-10 15:50:43 +01:00
|
|
|
self._filepath = filepath
|
2020-12-22 16:04:50 +01:00
|
|
|
super().__init__(timeout)
|
2020-12-10 15:50:43 +01:00
|
|
|
|
|
|
|
|
|
|
|
def execute(self):
|
2021-06-02 09:35:55 +02:00
|
|
|
session.repository.dumps(self._filepath)
|
2019-09-30 13:35:50 +02:00
|
|
|
|
2021-03-04 14:22:54 +01:00
|
|
|
class SessionListenTimer(Timer):
|
|
|
|
def execute(self):
|
|
|
|
session.listen()
|
|
|
|
|
2019-08-13 18:00:54 +02:00
|
|
|
class ApplyTimer(Timer):
|
|
|
|
def execute(self):
|
2021-03-04 14:22:54 +01:00
|
|
|
if session and session.state == STATE_ACTIVE:
|
2021-06-04 14:02:09 +02:00
|
|
|
for node in session.repository.graph.keys():
|
|
|
|
node_ref = session.repository.graph.get(node)
|
2019-08-13 18:00:54 +02:00
|
|
|
|
|
|
|
if node_ref.state == FETCHED:
|
2019-10-03 18:30:46 +02:00
|
|
|
try:
|
2021-06-18 14:34:11 +02:00
|
|
|
shared_data.session.applied_updates.append(node)
|
2021-05-17 11:12:18 +02:00
|
|
|
porcelain.apply(session.repository, node)
|
2019-10-14 13:08:31 +02:00
|
|
|
except Exception as e:
|
2021-03-31 15:38:35 +02:00
|
|
|
logging.error(f"Fail to apply {node_ref.uuid}")
|
2021-03-11 15:45:48 +01:00
|
|
|
traceback.print_exc()
|
2020-12-09 14:49:26 +01:00
|
|
|
else:
|
2021-05-18 23:14:09 +02:00
|
|
|
impl = session.repository.rdp.get_implementation(node_ref.instance)
|
|
|
|
if impl.bl_reload_parent:
|
2021-06-04 14:02:09 +02:00
|
|
|
for parent in session.repository.graph.get_parents(node):
|
2021-02-12 10:49:04 +01:00
|
|
|
logging.debug("Refresh parent {node}")
|
2021-05-17 11:12:18 +02:00
|
|
|
porcelain.apply(session.repository,
|
2021-03-14 20:58:25 +01:00
|
|
|
parent.uuid,
|
|
|
|
force=True)
|
2021-06-22 11:36:51 +02:00
|
|
|
if hasattr(impl, 'bl_reload_child') and impl.bl_reload_child:
|
|
|
|
for dep in node_ref.dependencies:
|
|
|
|
porcelain.apply(session.repository,
|
|
|
|
dep,
|
|
|
|
force=True)
|
2019-08-13 18:00:54 +02:00
|
|
|
|
2021-01-13 15:49:07 +01:00
|
|
|
|
2021-07-06 16:06:14 +02:00
|
|
|
class AnnotationUpdates(Timer):
|
|
|
|
def __init__(self, timeout=1):
|
|
|
|
self._annotating = False
|
|
|
|
self._settings = utils.get_preferences()
|
|
|
|
|
|
|
|
super().__init__(timeout)
|
|
|
|
|
|
|
|
def execute(self):
|
|
|
|
if session and session.state == STATE_ACTIVE:
|
|
|
|
ctx = bpy.context
|
|
|
|
annotation_gp = ctx.scene.grease_pencil
|
|
|
|
|
|
|
|
if annotation_gp and not annotation_gp.uuid:
|
|
|
|
ctx.scene.update_tag()
|
|
|
|
|
|
|
|
# if an annotation exist and is tracked
|
|
|
|
if annotation_gp and annotation_gp.uuid:
|
|
|
|
registered_gp = session.repository.graph.get(annotation_gp.uuid)
|
|
|
|
if is_annotating(bpy.context):
|
|
|
|
# try to get the right on it
|
|
|
|
if registered_gp.owner == RP_COMMON:
|
|
|
|
self._annotating = True
|
|
|
|
logging.debug(
|
|
|
|
"Getting the right on the annotation GP")
|
|
|
|
porcelain.lock(session.repository,
|
2021-07-14 12:38:30 +02:00
|
|
|
[registered_gp.uuid],
|
2021-07-06 16:06:14 +02:00
|
|
|
ignore_warnings=True,
|
|
|
|
affect_dependencies=False)
|
|
|
|
|
|
|
|
if registered_gp.owner == self._settings.username:
|
|
|
|
porcelain.commit(session.repository, annotation_gp.uuid)
|
|
|
|
porcelain.push(session.repository, 'origin', annotation_gp.uuid)
|
|
|
|
|
|
|
|
elif self._annotating:
|
|
|
|
porcelain.unlock(session.repository,
|
2021-07-14 12:38:30 +02:00
|
|
|
[registered_gp.uuid],
|
2021-07-06 16:06:14 +02:00
|
|
|
ignore_warnings=True,
|
|
|
|
affect_dependencies=False)
|
2021-07-14 12:38:30 +02:00
|
|
|
self._annotating = False
|
2021-07-06 16:06:14 +02:00
|
|
|
|
2019-09-18 23:55:30 +02:00
|
|
|
class DynamicRightSelectTimer(Timer):
|
2020-12-22 16:04:50 +01:00
|
|
|
def __init__(self, timeout=.1):
|
|
|
|
super().__init__(timeout)
|
2021-07-12 12:06:45 +02:00
|
|
|
self._last_selection = set()
|
2019-11-22 18:32:39 +01:00
|
|
|
self._user = None
|
2019-09-18 23:55:30 +02:00
|
|
|
|
|
|
|
def execute(self):
|
2020-03-02 11:09:45 +01:00
|
|
|
settings = utils.get_preferences()
|
2019-11-22 18:32:39 +01:00
|
|
|
|
2021-03-04 14:22:54 +01:00
|
|
|
if session and session.state == STATE_ACTIVE:
|
2019-11-22 18:32:39 +01:00
|
|
|
# Find user
|
|
|
|
if self._user is None:
|
2020-01-22 11:20:04 +01:00
|
|
|
self._user = session.online_users.get(settings.username)
|
2019-11-22 18:32:39 +01:00
|
|
|
|
|
|
|
if self._user:
|
2021-07-12 12:06:45 +02:00
|
|
|
current_selection = set(utils.get_selected_objects(
|
2020-02-25 16:35:26 +01:00
|
|
|
bpy.context.scene,
|
|
|
|
bpy.data.window_managers['WinMan'].windows[0].view_layer
|
2021-07-12 12:06:45 +02:00
|
|
|
))
|
2019-11-22 18:32:39 +01:00
|
|
|
if current_selection != self._last_selection:
|
2021-07-12 12:06:45 +02:00
|
|
|
to_lock = list(current_selection.difference(self._last_selection))
|
|
|
|
to_release = list(self._last_selection.difference(current_selection))
|
|
|
|
instances_to_lock = list()
|
|
|
|
|
|
|
|
for node_id in to_lock:
|
|
|
|
node = session.repository.graph.get(node_id)
|
2021-10-21 12:19:46 +02:00
|
|
|
if node and hasattr(node,'data'):
|
|
|
|
instance_mode = node.data.get('instance_type')
|
|
|
|
if instance_mode and instance_mode == 'COLLECTION':
|
|
|
|
to_lock.remove(node_id)
|
|
|
|
instances_to_lock.append(node_id)
|
2021-07-12 12:06:45 +02:00
|
|
|
if instances_to_lock:
|
|
|
|
try:
|
|
|
|
porcelain.lock(session.repository,
|
|
|
|
instances_to_lock,
|
|
|
|
ignore_warnings=True,
|
|
|
|
affect_dependencies=False)
|
|
|
|
except NonAuthorizedOperationError as e:
|
|
|
|
logging.warning(e)
|
|
|
|
|
|
|
|
if to_release:
|
|
|
|
try:
|
|
|
|
porcelain.unlock(session.repository,
|
|
|
|
to_release,
|
|
|
|
ignore_warnings=True,
|
|
|
|
affect_dependencies=True)
|
|
|
|
except NonAuthorizedOperationError as e:
|
|
|
|
logging.warning(e)
|
|
|
|
if to_lock:
|
|
|
|
try:
|
|
|
|
porcelain.lock(session.repository,
|
|
|
|
to_lock,
|
|
|
|
ignore_warnings=True,
|
|
|
|
affect_dependencies=True)
|
|
|
|
except NonAuthorizedOperationError as e:
|
|
|
|
logging.warning(e)
|
2020-06-16 18:04:27 +02:00
|
|
|
|
|
|
|
self._last_selection = current_selection
|
|
|
|
|
|
|
|
user_metadata = {
|
|
|
|
'selected_objects': current_selection
|
|
|
|
}
|
|
|
|
|
2021-06-02 12:59:53 +02:00
|
|
|
porcelain.update_user_metadata(session.repository, user_metadata)
|
2020-06-16 18:04:27 +02:00
|
|
|
logging.debug("Update selection")
|
|
|
|
|
|
|
|
# Fix deselection until right managment refactoring (with Roles concepts)
|
2020-12-09 18:34:56 +01:00
|
|
|
if len(current_selection) == 0 :
|
2021-06-04 14:02:09 +02:00
|
|
|
owned_keys = [k for k, v in session.repository.graph.items() if v.owner==settings.username]
|
2021-07-12 12:06:45 +02:00
|
|
|
if owned_keys:
|
2020-10-14 00:36:59 +02:00
|
|
|
try:
|
2021-06-02 11:31:23 +02:00
|
|
|
porcelain.unlock(session.repository,
|
2021-07-12 12:06:45 +02:00
|
|
|
owned_keys,
|
2021-06-02 11:31:23 +02:00
|
|
|
ignore_warnings=True,
|
2021-06-03 15:03:09 +02:00
|
|
|
affect_dependencies=True)
|
2021-07-12 12:06:45 +02:00
|
|
|
except NonAuthorizedOperationError as e:
|
|
|
|
logging.warning(e)
|
2019-08-23 12:28:57 +02:00
|
|
|
|
2021-07-12 12:06:45 +02:00
|
|
|
# Objects selectability
|
2020-10-12 18:56:42 +02:00
|
|
|
for obj in bpy.data.objects:
|
|
|
|
object_uuid = getattr(obj, 'uuid', None)
|
|
|
|
if object_uuid:
|
2021-06-02 10:22:37 +02:00
|
|
|
is_selectable = not session.repository.is_node_readonly(object_uuid)
|
2020-10-12 18:56:42 +02:00
|
|
|
if obj.hide_select != is_selectable:
|
|
|
|
obj.hide_select = is_selectable
|
2021-06-18 14:34:11 +02:00
|
|
|
shared_data.session.applied_updates.append(object_uuid)
|
2019-10-14 13:08:31 +02:00
|
|
|
|
2020-12-09 14:49:26 +01:00
|
|
|
|
2019-10-03 13:23:59 +02:00
|
|
|
class ClientUpdate(Timer):
|
2020-12-22 16:04:50 +01:00
|
|
|
def __init__(self, timeout=.1):
|
|
|
|
super().__init__(timeout)
|
2020-02-26 12:03:48 +01:00
|
|
|
self.handle_quit = False
|
2020-04-06 14:47:03 +02:00
|
|
|
self.users_metadata = {}
|
2019-08-13 18:00:54 +02:00
|
|
|
|
|
|
|
def execute(self):
|
2020-03-02 11:09:45 +01:00
|
|
|
settings = utils.get_preferences()
|
2020-01-22 18:37:46 +01:00
|
|
|
|
2020-06-16 17:15:32 +02:00
|
|
|
if session and renderer:
|
2021-03-04 14:22:54 +01:00
|
|
|
if session.state in [STATE_ACTIVE, STATE_LOBBY]:
|
2020-10-02 00:05:33 +02:00
|
|
|
local_user = session.online_users.get(
|
2020-09-17 22:47:11 +02:00
|
|
|
settings.username)
|
2020-06-16 17:15:32 +02:00
|
|
|
|
2020-04-14 18:56:20 +02:00
|
|
|
if not local_user:
|
|
|
|
return
|
|
|
|
else:
|
2020-10-02 00:05:33 +02:00
|
|
|
for username, user_data in session.online_users.items():
|
2020-04-14 18:56:20 +02:00
|
|
|
if username != settings.username:
|
2020-09-17 22:47:11 +02:00
|
|
|
cached_user_data = self.users_metadata.get(
|
|
|
|
username)
|
2020-10-02 00:05:33 +02:00
|
|
|
new_user_data = session.online_users[username]['metadata']
|
2020-06-16 17:15:32 +02:00
|
|
|
|
2020-04-14 18:56:20 +02:00
|
|
|
if cached_user_data is None:
|
|
|
|
self.users_metadata[username] = user_data['metadata']
|
|
|
|
elif 'view_matrix' in cached_user_data and 'view_matrix' in new_user_data and cached_user_data['view_matrix'] != new_user_data['view_matrix']:
|
2020-10-05 22:34:43 +02:00
|
|
|
refresh_3d_view()
|
2020-04-14 18:56:20 +02:00
|
|
|
self.users_metadata[username] = user_data['metadata']
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
self.users_metadata[username] = user_data['metadata']
|
|
|
|
|
|
|
|
local_user_metadata = local_user.get('metadata')
|
|
|
|
scene_current = bpy.context.scene.name
|
2020-09-17 22:47:11 +02:00
|
|
|
local_user = session.online_users.get(settings.username)
|
2020-10-05 22:34:43 +02:00
|
|
|
current_view_corners = generate_user_camera()
|
2020-06-16 17:15:32 +02:00
|
|
|
|
2020-04-14 18:56:20 +02:00
|
|
|
# Init client metadata
|
|
|
|
if not local_user_metadata or 'color' not in local_user_metadata.keys():
|
|
|
|
metadata = {
|
2020-10-05 22:34:43 +02:00
|
|
|
'view_corners': get_view_matrix(),
|
|
|
|
'view_matrix': get_view_matrix(),
|
2020-04-14 18:56:20 +02:00
|
|
|
'color': (settings.client_color.r,
|
2020-09-17 22:47:11 +02:00
|
|
|
settings.client_color.g,
|
|
|
|
settings.client_color.b,
|
|
|
|
1),
|
2020-06-16 17:15:32 +02:00
|
|
|
'frame_current': bpy.context.scene.frame_current,
|
2021-06-24 16:01:14 +02:00
|
|
|
'scene_current': scene_current,
|
|
|
|
'mode_current': bpy.context.mode
|
2020-04-14 18:56:20 +02:00
|
|
|
}
|
2021-06-02 12:59:53 +02:00
|
|
|
porcelain.update_user_metadata(session.repository, metadata)
|
2020-04-14 18:56:20 +02:00
|
|
|
|
|
|
|
# Update client representation
|
|
|
|
# Update client current scene
|
|
|
|
elif scene_current != local_user_metadata['scene_current']:
|
|
|
|
local_user_metadata['scene_current'] = scene_current
|
2021-06-02 12:59:53 +02:00
|
|
|
porcelain.update_user_metadata(session.repository, local_user_metadata)
|
2020-06-16 17:15:32 +02:00
|
|
|
elif 'view_corners' in local_user_metadata and current_view_corners != local_user_metadata['view_corners']:
|
2020-04-14 18:56:20 +02:00
|
|
|
local_user_metadata['view_corners'] = current_view_corners
|
2020-10-05 22:34:43 +02:00
|
|
|
local_user_metadata['view_matrix'] = get_view_matrix(
|
2020-09-17 22:47:11 +02:00
|
|
|
)
|
2021-06-02 12:59:53 +02:00
|
|
|
porcelain.update_user_metadata(session.repository, local_user_metadata)
|
2021-06-24 16:01:14 +02:00
|
|
|
elif bpy.context.mode != local_user_metadata['mode_current']:
|
|
|
|
local_user_metadata['mode_current'] = bpy.context.mode
|
|
|
|
porcelain.update_user_metadata(session.repository, local_user_metadata)
|
2020-04-14 18:56:20 +02:00
|
|
|
|
2020-09-17 22:47:11 +02:00
|
|
|
|
2020-07-24 14:56:20 +02:00
|
|
|
class SessionStatusUpdate(Timer):
|
2020-12-22 16:04:50 +01:00
|
|
|
def __init__(self, timeout=1):
|
|
|
|
super().__init__(timeout)
|
2020-06-16 17:15:32 +02:00
|
|
|
|
2020-07-24 14:56:20 +02:00
|
|
|
def execute(self):
|
2020-10-05 22:34:43 +02:00
|
|
|
refresh_sidebar_view()
|
2020-06-16 17:15:32 +02:00
|
|
|
|
2020-09-17 22:47:11 +02:00
|
|
|
|
2020-07-24 14:56:20 +02:00
|
|
|
class SessionUserSync(Timer):
|
2020-12-22 16:04:50 +01:00
|
|
|
def __init__(self, timeout=1):
|
|
|
|
super().__init__(timeout)
|
2020-10-05 22:34:43 +02:00
|
|
|
self.settings = utils.get_preferences()
|
2020-06-16 17:15:32 +02:00
|
|
|
|
2020-07-17 16:33:39 +02:00
|
|
|
def execute(self):
|
2020-07-24 14:56:20 +02:00
|
|
|
if session and renderer:
|
|
|
|
# sync online users
|
2020-10-02 00:05:33 +02:00
|
|
|
session_users = session.online_users
|
2020-07-24 14:56:20 +02:00
|
|
|
ui_users = bpy.context.window_manager.online_users
|
|
|
|
|
|
|
|
for index, user in enumerate(ui_users):
|
2020-10-06 16:10:10 +02:00
|
|
|
if user.username not in session_users.keys() and \
|
|
|
|
user.username != self.settings.username:
|
|
|
|
renderer.remove_widget(f"{user.username}_cam")
|
|
|
|
renderer.remove_widget(f"{user.username}_select")
|
|
|
|
renderer.remove_widget(f"{user.username}_name")
|
2021-06-29 17:10:59 +02:00
|
|
|
renderer.remove_widget(f"{user.username}_mode")
|
2020-07-24 14:56:20 +02:00
|
|
|
ui_users.remove(index)
|
|
|
|
break
|
|
|
|
|
|
|
|
for user in session_users:
|
|
|
|
if user not in ui_users:
|
|
|
|
new_key = ui_users.add()
|
|
|
|
new_key.name = user
|
2020-09-17 22:47:11 +02:00
|
|
|
new_key.username = user
|
2021-07-01 11:58:52 +02:00
|
|
|
if user != self.settings.username:
|
|
|
|
renderer.add_widget(
|
|
|
|
f"{user}_cam", UserFrustumWidget(user))
|
|
|
|
renderer.add_widget(
|
|
|
|
f"{user}_select", UserSelectionWidget(user))
|
|
|
|
renderer.add_widget(
|
|
|
|
f"{user}_name", UserNameWidget(user))
|
|
|
|
renderer.add_widget(
|
|
|
|
f"{user}_mode", UserModeWidget(user))
|
2020-10-01 10:58:30 +02:00
|
|
|
|
|
|
|
|
2020-10-02 12:11:53 +02:00
|
|
|
class MainThreadExecutor(Timer):
|
2020-12-22 16:04:50 +01:00
|
|
|
def __init__(self, timeout=1, execution_queue=None):
|
|
|
|
super().__init__(timeout)
|
2020-10-01 10:58:30 +02:00
|
|
|
self.execution_queue = execution_queue
|
2020-10-05 22:34:43 +02:00
|
|
|
|
2020-10-01 10:58:30 +02:00
|
|
|
def execute(self):
|
|
|
|
while not self.execution_queue.empty():
|
2020-11-26 11:37:51 +01:00
|
|
|
function, kwargs = self.execution_queue.get()
|
2020-10-01 10:58:30 +02:00
|
|
|
logging.debug(f"Executing {function.__name__}")
|
2020-12-22 16:04:50 +01:00
|
|
|
function(**kwargs)
|