multi-user/multi_user/presence.py

355 lines
11 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 #####
2019-08-14 14:25:20 +02:00
import copy
import logging
2019-08-13 22:09:36 +02:00
import math
2019-09-30 13:35:50 +02:00
import bgl
import blf
import bpy
import gpu
import mathutils
2019-04-01 12:28:13 +02:00
from bpy_extras import view3d_utils
from gpu_extras.batch import batch_for_shader
from . import utils
2019-04-18 15:05:48 +02:00
renderer = None
2019-08-23 12:28:57 +02:00
2019-09-26 22:30:22 +02:00
2019-04-01 12:28:13 +02:00
def view3d_find():
2020-04-14 18:56:20 +02:00
""" Find the first 'VIEW_3D' windows found in areas
:return: tuple(Area, Region, RegionView3D)
"""
2019-04-01 12:28:13 +02:00
for area in bpy.data.window_managers[0].windows[0].screen.areas:
if area.type == 'VIEW_3D':
v3d = area.spaces[0]
rv3d = v3d.region_3d
for region in area.regions:
if region.type == 'WINDOW':
return area, region, rv3d
return None, None, None
2019-09-26 22:30:22 +02:00
def refresh_3d_view():
2020-04-14 18:56:20 +02:00
""" Refresh the viewport
"""
area, region, rv3d = view3d_find()
if area and region and rv3d:
area.tag_redraw()
2019-04-01 12:28:13 +02:00
2020-04-14 18:56:20 +02:00
def refresh_sidebar_view():
""" Refresh the blender sidebar
"""
area, region, rv3d = view3d_find()
area.regions[3].tag_redraw()
2019-09-26 22:30:22 +02:00
2019-04-01 12:28:13 +02:00
def get_target(region, rv3d, coord):
target = [0, 0, 0]
if coord and region and rv3d:
view_vector = view3d_utils.region_2d_to_vector_3d(region, rv3d, coord)
ray_origin = view3d_utils.region_2d_to_origin_3d(region, rv3d, coord)
target = ray_origin + view_vector
return [target.x, target.y, target.z]
2019-04-26 17:18:55 +02:00
def get_target_far(region, rv3d, coord, distance):
target = [0, 0, 0]
if coord and region and rv3d:
view_vector = view3d_utils.region_2d_to_vector_3d(region, rv3d, coord)
ray_origin = view3d_utils.region_2d_to_origin_3d(region, rv3d, coord)
target = ray_origin + view_vector * distance
return [target.x, target.y, target.z]
def get_default_bbox(obj, radius):
coords = [
(-radius, -radius, -radius), (+radius, -radius, -radius),
(-radius, +radius, -radius), (+radius, +radius, -radius),
(-radius, -radius, +radius), (+radius, -radius, +radius),
(-radius, +radius, +radius), (+radius, +radius, +radius)]
base = obj.matrix_world
bbox_corners = [base @ mathutils.Vector(corner) for corner in coords]
return [(point.x, point.y, point.z)
for point in bbox_corners]
2019-04-01 12:28:13 +02:00
2020-01-21 23:24:44 +01:00
def get_view_corners():
2019-04-01 12:28:13 +02:00
area, region, rv3d = view3d_find()
v1 = [0, 0, 0]
v2 = [0, 0, 0]
v3 = [0, 0, 0]
v4 = [0, 0, 0]
v5 = [0, 0, 0]
v6 = [0, 0, 0]
v7 = [0, 0, 0]
2019-04-01 12:28:13 +02:00
if area and region and rv3d:
width = region.width
height = region.height
v1 = get_target(region, rv3d, (0, 0))
v3 = get_target(region, rv3d, (0, height))
v2 = get_target(region, rv3d, (width, height))
v4 = get_target(region, rv3d, (width, 0))
v5 = get_target(region, rv3d, (width/2, height/2))
2019-08-14 14:25:20 +02:00
v6 = list(rv3d.view_location)
v7 = get_target_far(region, rv3d, (width/2, height/2), -.8)
2019-08-23 12:28:57 +02:00
coords = [v1, v2, v3, v4, v5, v6, v7]
2019-04-01 12:28:13 +02:00
return coords
def get_client_2d(coords):
area, region, rv3d = view3d_find()
if area and region and rv3d:
return view3d_utils.location_3d_to_region_2d(region, rv3d, coords)
else:
2019-04-18 15:05:48 +02:00
return (0, 0)
2019-04-01 12:28:13 +02:00
def get_bb_coords_from_obj(object, parent=None):
base = object.matrix_world if parent is None else parent.matrix_world
bbox_corners = [base @ mathutils.Vector(
corner) for corner in object.bound_box]
return [(point.x, point.y, point.z)
for point in bbox_corners]
2019-08-23 12:28:57 +02:00
2020-01-21 23:24:44 +01:00
def get_view_matrix():
area, region, rv3d = view3d_find()
2020-04-01 11:34:24 +02:00
if area and region and rv3d:
return [list(v) for v in rv3d.view_matrix]
2020-01-21 23:24:44 +01:00
2019-09-24 14:42:59 +02:00
def update_presence(self, context):
2019-09-26 22:30:22 +02:00
global renderer
2019-11-04 16:42:18 +01:00
if 'renderer' in globals() and hasattr(renderer, 'run'):
if self.enable_presence:
renderer.run()
else:
renderer.stop()
2019-09-26 22:30:22 +02:00
2019-04-01 12:28:13 +02:00
2019-09-24 14:42:59 +02:00
def update_overlay_settings(self, context):
2019-09-26 22:30:22 +02:00
global renderer
if renderer and not self.presence_show_selected:
renderer.flush_selection()
if renderer and not self.presence_show_user:
renderer.flush_users()
2019-04-01 12:28:13 +02:00
2019-09-24 14:42:59 +02:00
class DrawFactory(object):
def __init__(self):
2019-04-17 11:49:27 +02:00
self.d3d_items = {}
self.d2d_items = {}
2019-04-01 12:28:13 +02:00
self.draw3d_handle = None
self.draw2d_handle = None
self.draw_event = None
self.coords = None
self.active_object = None
def run(self):
self.register_handlers()
2019-08-23 12:28:57 +02:00
def stop(self):
2020-01-22 18:37:46 +01:00
self.flush_users()
self.flush_selection()
self.unregister_handlers()
2019-08-23 12:28:57 +02:00
2020-01-22 18:37:46 +01:00
refresh_3d_view()
def register_handlers(self):
2019-04-01 12:28:13 +02:00
self.draw3d_handle = bpy.types.SpaceView3D.draw_handler_add(
self.draw3d_callback, (), 'WINDOW', 'POST_VIEW')
self.draw2d_handle = bpy.types.SpaceView3D.draw_handler_add(
self.draw2d_callback, (), 'WINDOW', 'POST_PIXEL')
def unregister_handlers(self):
2019-04-01 12:28:13 +02:00
if self.draw2d_handle:
bpy.types.SpaceView3D.draw_handler_remove(
self.draw2d_handle, "WINDOW")
self.draw2d_handle = None
if self.draw3d_handle:
bpy.types.SpaceView3D.draw_handler_remove(
self.draw3d_handle, "WINDOW")
self.draw3d_handle = None
2019-04-15 17:08:05 +02:00
self.d3d_items.clear()
self.d2d_items.clear()
def flush_selection(self, user=None):
key_to_remove = []
2020-04-22 17:04:14 +02:00
select_key = f"{user}_select" if user else "select"
for k in self.d3d_items.keys():
2019-10-02 14:02:30 +02:00
if select_key in k:
key_to_remove.append(k)
2019-09-26 22:30:22 +02:00
2019-09-24 14:42:59 +02:00
for k in key_to_remove:
del self.d3d_items[k]
2019-09-24 14:42:59 +02:00
def flush_users(self):
key_to_remove = []
for k in self.d3d_items.keys():
if "select" not in k:
key_to_remove.append(k)
2019-09-26 22:30:22 +02:00
for k in key_to_remove:
del self.d3d_items[k]
2019-09-26 22:30:22 +02:00
2019-09-24 14:42:59 +02:00
self.d2d_items.clear()
2020-01-22 14:33:34 +01:00
def draw_client_selection(self, client_id, client_color, client_selection):
local_user = utils.get_preferences().username
2019-09-26 22:30:22 +02:00
2020-01-22 14:33:34 +01:00
if local_user != client_id:
self.flush_selection(client_id)
2019-10-02 14:02:30 +02:00
for select_ob in client_selection:
2020-04-22 17:04:14 +02:00
drawable_key = f"{client_id}_select_{select_ob}"
ob = utils.find_from_attr("uuid", select_ob, bpy.data.objects)
2019-10-08 21:48:45 +02:00
if not ob:
return
if ob.type == 'EMPTY':
# TODO: Child case
# Collection instance case
indices = (
(0, 1), (1, 2), (2, 3), (0, 3),
(4, 5), (5, 6), (6, 7), (4, 7),
(0, 4), (1, 5), (2, 6), (3, 7))
if ob.instance_collection:
for obj in ob.instance_collection.objects:
if obj.type == 'MESH':
self.append_3d_item(
drawable_key,
client_color,
get_bb_coords_from_obj(obj, parent=ob),
indices)
if ob.type in ['MESH','META']:
indices = (
(0, 1), (1, 2), (2, 3), (0, 3),
(4, 5), (5, 6), (6, 7), (4, 7),
(0, 4), (1, 5), (2, 6), (3, 7))
self.append_3d_item(
drawable_key,
client_color,
get_bb_coords_from_obj(ob),
indices)
else:
indices = (
(0, 1), (0, 2), (1, 3), (2, 3),
(4, 5), (4, 6), (5, 7), (6, 7),
(0, 4), (1, 5), (2, 6), (3, 7))
self.append_3d_item(
drawable_key,
client_color,
get_default_bbox(ob, ob.scale.x),
indices)
def append_3d_item(self,key,color, coords, indices):
shader = gpu.shader.from_builtin('3D_UNIFORM_COLOR')
color = color
batch = batch_for_shader(
shader, 'LINES', {"pos": coords}, indices=indices)
self.d3d_items[key] = (shader, batch, color)
2019-09-26 22:30:22 +02:00
2020-01-22 14:33:34 +01:00
def draw_client_camera(self, client_id, client_location, client_color):
2019-08-14 14:25:20 +02:00
if client_location:
local_user = utils.get_preferences().username
2019-04-17 16:15:21 +02:00
2020-01-22 14:33:34 +01:00
if local_user != client_id:
2019-09-30 16:12:19 +02:00
try:
indices = (
(1, 3), (2, 1), (3, 0),
(2, 0), (4, 5), (1, 6),
(2, 6), (3, 6), (0, 6)
)
2019-04-01 12:28:13 +02:00
2019-09-30 16:12:19 +02:00
shader = gpu.shader.from_builtin('3D_UNIFORM_COLOR')
position = [tuple(coord) for coord in client_location]
color = client_color
2019-04-01 12:28:13 +02:00
2019-09-30 16:12:19 +02:00
batch = batch_for_shader(
shader, 'LINES', {"pos": position}, indices=indices)
2019-04-01 12:28:13 +02:00
2020-01-22 14:33:34 +01:00
self.d3d_items[client_id] = (shader, batch, color)
self.d2d_items[client_id] = (position[1], client_id, color)
2019-08-14 14:25:20 +02:00
2019-09-30 16:12:19 +02:00
except Exception as e:
2020-04-22 17:04:14 +02:00
logging.error(f"Draw client exception: {e}")
2019-04-01 12:28:13 +02:00
def draw3d_callback(self):
2019-08-14 14:25:20 +02:00
bgl.glLineWidth(1.5)
bgl.glEnable(bgl.GL_DEPTH_TEST)
bgl.glEnable(bgl.GL_BLEND)
bgl.glEnable(bgl.GL_LINE_SMOOTH)
2019-04-15 17:08:05 +02:00
try:
2019-04-17 11:49:27 +02:00
for shader, batch, color in self.d3d_items.values():
2019-04-15 17:08:05 +02:00
shader.bind()
shader.uniform_float("color", color)
batch.draw(shader)
2019-08-23 12:28:57 +02:00
except Exception:
2020-04-22 17:04:14 +02:00
logging.error("3D Exception")
2019-04-01 12:28:13 +02:00
def draw2d_callback(self):
2019-04-17 11:49:27 +02:00
for position, font, color in self.d2d_items.values():
2019-04-01 12:28:13 +02:00
try:
2019-04-15 17:08:05 +02:00
coords = get_client_2d(position)
2019-04-01 12:28:13 +02:00
2019-04-17 11:49:27 +02:00
if coords:
blf.position(0, coords[0], coords[1]+10, 0)
2019-08-14 14:25:20 +02:00
blf.size(0, 16, 72)
2019-04-17 11:49:27 +02:00
blf.color(0, color[0], color[1], color[2], color[3])
blf.draw(0, font)
2019-04-01 12:28:13 +02:00
2019-08-23 12:28:57 +02:00
except Exception:
2020-04-22 17:04:14 +02:00
logging.error("2D EXCEPTION")
2019-08-23 12:28:57 +02:00
def register():
global renderer
renderer = DrawFactory()
def unregister():
global renderer
renderer.unregister_handlers()
2019-08-23 12:28:57 +02:00
del renderer