multi-user/multi_user/timers.py

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