2017-06-21 06:23:44 +00:00
|
|
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2017 Red Hat Inc.
|
|
|
|
*
|
|
|
|
* 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 2 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, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
|
|
* 02111-1307, USA.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "backends/meta-screen-cast-monitor-stream-src.h"
|
|
|
|
|
2018-12-14 16:47:57 +00:00
|
|
|
#include <spa/buffer/meta.h>
|
|
|
|
|
2017-06-21 06:23:44 +00:00
|
|
|
#include "backends/meta-backend-private.h"
|
2018-12-14 16:47:57 +00:00
|
|
|
#include "backends/meta-cursor-tracker-private.h"
|
2017-06-21 06:23:44 +00:00
|
|
|
#include "backends/meta-logical-monitor.h"
|
|
|
|
#include "backends/meta-monitor.h"
|
2018-07-10 08:36:24 +00:00
|
|
|
#include "backends/meta-screen-cast-monitor-stream.h"
|
2018-12-14 16:47:57 +00:00
|
|
|
#include "backends/meta-screen-cast-session.h"
|
2017-06-21 06:23:44 +00:00
|
|
|
#include "clutter/clutter.h"
|
|
|
|
#include "clutter/clutter-mutter.h"
|
2018-12-14 16:47:57 +00:00
|
|
|
#include "core/boxes-private.h"
|
2017-06-21 06:23:44 +00:00
|
|
|
|
|
|
|
struct _MetaScreenCastMonitorStreamSrc
|
|
|
|
{
|
|
|
|
MetaScreenCastStreamSrc parent;
|
|
|
|
|
2019-01-03 15:51:08 +00:00
|
|
|
gboolean cursor_bitmap_invalid;
|
|
|
|
|
2018-12-12 14:29:21 +00:00
|
|
|
gulong actors_painted_handler_id;
|
2018-12-14 16:47:57 +00:00
|
|
|
gulong paint_handler_id;
|
|
|
|
gulong cursor_moved_handler_id;
|
|
|
|
gulong cursor_changed_handler_id;
|
2017-06-21 06:23:44 +00:00
|
|
|
};
|
|
|
|
|
2018-12-14 16:47:57 +00:00
|
|
|
static void
|
|
|
|
hw_cursor_inhibitor_iface_init (MetaHwCursorInhibitorInterface *iface);
|
|
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (MetaScreenCastMonitorStreamSrc,
|
|
|
|
meta_screen_cast_monitor_stream_src,
|
|
|
|
META_TYPE_SCREEN_CAST_STREAM_SRC,
|
|
|
|
G_IMPLEMENT_INTERFACE (META_TYPE_HW_CURSOR_INHIBITOR,
|
|
|
|
hw_cursor_inhibitor_iface_init))
|
2017-06-21 06:23:44 +00:00
|
|
|
|
|
|
|
static ClutterStage *
|
|
|
|
get_stage (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaScreenCastStreamSrc *src;
|
|
|
|
MetaScreenCastStream *stream;
|
|
|
|
MetaScreenCastMonitorStream *monitor_stream;
|
|
|
|
|
|
|
|
src = META_SCREEN_CAST_STREAM_SRC (monitor_src);
|
|
|
|
stream = meta_screen_cast_stream_src_get_stream (src);
|
|
|
|
monitor_stream = META_SCREEN_CAST_MONITOR_STREAM (stream);
|
|
|
|
|
|
|
|
return meta_screen_cast_monitor_stream_get_stage (monitor_stream);
|
|
|
|
}
|
|
|
|
|
|
|
|
static MetaMonitor *
|
|
|
|
get_monitor (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaScreenCastStreamSrc *src;
|
|
|
|
MetaScreenCastStream *stream;
|
|
|
|
MetaScreenCastMonitorStream *monitor_stream;
|
|
|
|
|
|
|
|
src = META_SCREEN_CAST_STREAM_SRC (monitor_src);
|
|
|
|
stream = meta_screen_cast_stream_src_get_stream (src);
|
|
|
|
monitor_stream = META_SCREEN_CAST_MONITOR_STREAM (stream);
|
|
|
|
|
|
|
|
return meta_screen_cast_monitor_stream_get_monitor (monitor_stream);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_screen_cast_monitor_stream_src_get_specs (MetaScreenCastStreamSrc *src,
|
|
|
|
int *width,
|
|
|
|
int *height,
|
|
|
|
float *frame_rate)
|
|
|
|
{
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (src);
|
|
|
|
MetaMonitor *monitor;
|
|
|
|
MetaLogicalMonitor *logical_monitor;
|
|
|
|
float scale;
|
|
|
|
MetaMonitorMode *mode;
|
|
|
|
|
|
|
|
monitor = get_monitor (monitor_src);
|
|
|
|
logical_monitor = meta_monitor_get_logical_monitor (monitor);
|
|
|
|
mode = meta_monitor_get_current_mode (monitor);
|
|
|
|
|
2019-01-25 21:06:39 +00:00
|
|
|
if (meta_is_stage_views_scaled ())
|
|
|
|
scale = logical_monitor->scale;
|
|
|
|
else
|
|
|
|
scale = 1.0;
|
|
|
|
|
2017-06-21 06:23:44 +00:00
|
|
|
*width = (int) roundf (logical_monitor->rect.width * scale);
|
|
|
|
*height = (int) roundf (logical_monitor->rect.height * scale);
|
|
|
|
*frame_rate = meta_monitor_mode_get_refresh_rate (mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
stage_painted (ClutterActor *actor,
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaScreenCastStreamSrc *src = META_SCREEN_CAST_STREAM_SRC (monitor_src);
|
|
|
|
|
|
|
|
meta_screen_cast_stream_src_maybe_record_frame (src);
|
|
|
|
}
|
|
|
|
|
2018-12-14 16:47:57 +00:00
|
|
|
static MetaBackend *
|
|
|
|
get_backend (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaScreenCastStreamSrc *src = META_SCREEN_CAST_STREAM_SRC (monitor_src);
|
|
|
|
MetaScreenCastStream *stream = meta_screen_cast_stream_src_get_stream (src);
|
|
|
|
MetaScreenCastSession *session = meta_screen_cast_stream_get_session (stream);
|
|
|
|
MetaScreenCast *screen_cast =
|
|
|
|
meta_screen_cast_session_get_screen_cast (session);
|
|
|
|
|
|
|
|
return meta_screen_cast_get_backend (screen_cast);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
is_cursor_in_stream (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaBackend *backend = get_backend (monitor_src);
|
|
|
|
MetaCursorRenderer *cursor_renderer =
|
|
|
|
meta_backend_get_cursor_renderer (backend);
|
|
|
|
MetaMonitor *monitor;
|
|
|
|
MetaLogicalMonitor *logical_monitor;
|
|
|
|
MetaRectangle logical_monitor_layout;
|
|
|
|
ClutterRect logical_monitor_rect;
|
|
|
|
MetaCursorSprite *cursor_sprite;
|
|
|
|
|
|
|
|
monitor = get_monitor (monitor_src);
|
|
|
|
logical_monitor = meta_monitor_get_logical_monitor (monitor);
|
|
|
|
logical_monitor_layout = meta_logical_monitor_get_layout (logical_monitor);
|
|
|
|
logical_monitor_rect =
|
|
|
|
meta_rectangle_to_clutter_rect (&logical_monitor_layout);
|
|
|
|
|
|
|
|
cursor_sprite = meta_cursor_renderer_get_cursor (cursor_renderer);
|
|
|
|
if (cursor_sprite)
|
|
|
|
{
|
|
|
|
ClutterRect cursor_rect;
|
|
|
|
|
|
|
|
cursor_rect = meta_cursor_renderer_calculate_rect (cursor_renderer,
|
|
|
|
cursor_sprite);
|
|
|
|
return clutter_rect_intersection (&cursor_rect,
|
|
|
|
&logical_monitor_rect,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ClutterPoint cursor_position;
|
|
|
|
|
|
|
|
cursor_position = meta_cursor_renderer_get_position (cursor_renderer);
|
|
|
|
return clutter_rect_contains_point (&logical_monitor_rect,
|
|
|
|
&cursor_position);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
sync_cursor_state (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaScreenCastStreamSrc *src = META_SCREEN_CAST_STREAM_SRC (monitor_src);
|
|
|
|
ClutterStage *stage = get_stage (monitor_src);
|
|
|
|
|
|
|
|
if (!is_cursor_in_stream (monitor_src))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (clutter_stage_is_redraw_queued (stage))
|
|
|
|
return;
|
|
|
|
|
|
|
|
meta_screen_cast_stream_src_maybe_record_frame (src);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
cursor_moved (MetaCursorTracker *cursor_tracker,
|
|
|
|
float x,
|
|
|
|
float y,
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
sync_cursor_state (monitor_src);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
cursor_changed (MetaCursorTracker *cursor_tracker,
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
2019-01-03 15:51:08 +00:00
|
|
|
monitor_src->cursor_bitmap_invalid = TRUE;
|
2018-12-14 16:47:57 +00:00
|
|
|
sync_cursor_state (monitor_src);
|
|
|
|
}
|
|
|
|
|
|
|
|
static MetaCursorRenderer *
|
|
|
|
get_cursor_renderer (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaScreenCastStreamSrc *src = META_SCREEN_CAST_STREAM_SRC (monitor_src);
|
|
|
|
MetaScreenCastStream *stream = meta_screen_cast_stream_src_get_stream (src);
|
|
|
|
MetaScreenCastSession *session = meta_screen_cast_stream_get_session (stream);
|
|
|
|
MetaScreenCast *screen_cast =
|
|
|
|
meta_screen_cast_session_get_screen_cast (session);
|
|
|
|
MetaBackend *backend = meta_screen_cast_get_backend (screen_cast);
|
|
|
|
|
|
|
|
return meta_backend_get_cursor_renderer (backend);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
inhibit_hw_cursor (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaCursorRenderer *cursor_renderer;
|
|
|
|
MetaHwCursorInhibitor *inhibitor;
|
|
|
|
|
|
|
|
cursor_renderer = get_cursor_renderer (monitor_src);
|
|
|
|
inhibitor = META_HW_CURSOR_INHIBITOR (monitor_src);
|
|
|
|
meta_cursor_renderer_add_hw_cursor_inhibitor (cursor_renderer, inhibitor);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
uninhibit_hw_cursor (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
|
|
|
MetaCursorRenderer *cursor_renderer;
|
|
|
|
MetaHwCursorInhibitor *inhibitor;
|
|
|
|
|
|
|
|
cursor_renderer = get_cursor_renderer (monitor_src);
|
|
|
|
inhibitor = META_HW_CURSOR_INHIBITOR (monitor_src);
|
|
|
|
meta_cursor_renderer_remove_hw_cursor_inhibitor (cursor_renderer, inhibitor);
|
|
|
|
}
|
|
|
|
|
2017-06-21 06:23:44 +00:00
|
|
|
static void
|
2017-06-26 04:55:48 +00:00
|
|
|
meta_screen_cast_monitor_stream_src_enable (MetaScreenCastStreamSrc *src)
|
2017-06-21 06:23:44 +00:00
|
|
|
{
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
2017-06-26 04:55:48 +00:00
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (src);
|
2018-12-14 16:47:57 +00:00
|
|
|
MetaBackend *backend = get_backend (monitor_src);
|
|
|
|
MetaCursorTracker *cursor_tracker = meta_backend_get_cursor_tracker (backend);
|
2017-06-21 06:23:44 +00:00
|
|
|
ClutterStage *stage;
|
2018-12-14 16:47:57 +00:00
|
|
|
MetaScreenCastStream *stream;
|
2017-06-21 06:23:44 +00:00
|
|
|
|
2018-12-14 16:47:57 +00:00
|
|
|
stream = meta_screen_cast_stream_src_get_stream (src);
|
2017-06-21 06:23:44 +00:00
|
|
|
stage = get_stage (monitor_src);
|
2018-12-14 16:47:57 +00:00
|
|
|
|
|
|
|
switch (meta_screen_cast_stream_get_cursor_mode (stream))
|
|
|
|
{
|
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_METADATA:
|
|
|
|
monitor_src->cursor_moved_handler_id =
|
|
|
|
g_signal_connect_after (cursor_tracker, "cursor-moved",
|
|
|
|
G_CALLBACK (cursor_moved),
|
|
|
|
monitor_src);
|
|
|
|
monitor_src->cursor_changed_handler_id =
|
|
|
|
g_signal_connect_after (cursor_tracker, "cursor-changed",
|
|
|
|
G_CALLBACK (cursor_changed),
|
|
|
|
monitor_src);
|
|
|
|
/* Intentional fall-through */
|
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_HIDDEN:
|
|
|
|
monitor_src->actors_painted_handler_id =
|
|
|
|
g_signal_connect (stage, "actors-painted",
|
|
|
|
G_CALLBACK (stage_painted),
|
|
|
|
monitor_src);
|
|
|
|
break;
|
|
|
|
case META_SCREEN_CAST_CURSOR_MODE_EMBEDDED:
|
|
|
|
inhibit_hw_cursor (monitor_src);
|
|
|
|
monitor_src->paint_handler_id =
|
|
|
|
g_signal_connect_after (stage, "paint",
|
|
|
|
G_CALLBACK (stage_painted),
|
|
|
|
monitor_src);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-06-21 06:23:44 +00:00
|
|
|
clutter_actor_queue_redraw (CLUTTER_ACTOR (stage));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2017-06-26 04:55:48 +00:00
|
|
|
meta_screen_cast_monitor_stream_src_disable (MetaScreenCastStreamSrc *src)
|
2017-06-21 06:23:44 +00:00
|
|
|
{
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
2017-06-26 04:55:48 +00:00
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (src);
|
2018-12-14 16:47:57 +00:00
|
|
|
MetaBackend *backend = get_backend (monitor_src);
|
|
|
|
MetaCursorTracker *cursor_tracker = meta_backend_get_cursor_tracker (backend);
|
2017-06-21 06:23:44 +00:00
|
|
|
ClutterStage *stage;
|
|
|
|
|
|
|
|
stage = get_stage (monitor_src);
|
2018-12-14 16:47:57 +00:00
|
|
|
|
|
|
|
if (monitor_src->actors_painted_handler_id)
|
|
|
|
{
|
|
|
|
g_signal_handler_disconnect (stage,
|
|
|
|
monitor_src->actors_painted_handler_id);
|
|
|
|
monitor_src->actors_painted_handler_id = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (monitor_src->paint_handler_id)
|
|
|
|
{
|
|
|
|
g_signal_handler_disconnect (stage,
|
|
|
|
monitor_src->paint_handler_id);
|
|
|
|
monitor_src->paint_handler_id = 0;
|
|
|
|
uninhibit_hw_cursor (monitor_src);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (monitor_src->cursor_moved_handler_id)
|
|
|
|
{
|
|
|
|
g_signal_handler_disconnect (cursor_tracker,
|
|
|
|
monitor_src->cursor_moved_handler_id);
|
|
|
|
monitor_src->cursor_moved_handler_id = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (monitor_src->cursor_changed_handler_id)
|
|
|
|
{
|
|
|
|
g_signal_handler_disconnect (cursor_tracker,
|
|
|
|
monitor_src->cursor_changed_handler_id);
|
|
|
|
monitor_src->cursor_changed_handler_id = 0;
|
|
|
|
}
|
2017-06-26 04:55:48 +00:00
|
|
|
}
|
|
|
|
|
2018-12-14 16:47:57 +00:00
|
|
|
static gboolean
|
2017-06-26 04:55:48 +00:00
|
|
|
meta_screen_cast_monitor_stream_src_record_frame (MetaScreenCastStreamSrc *src,
|
|
|
|
uint8_t *data)
|
|
|
|
{
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (src);
|
|
|
|
ClutterStage *stage;
|
|
|
|
MetaMonitor *monitor;
|
|
|
|
MetaLogicalMonitor *logical_monitor;
|
|
|
|
|
|
|
|
stage = get_stage (monitor_src);
|
2018-12-14 16:47:57 +00:00
|
|
|
if (!clutter_stage_is_redraw_queued (stage))
|
|
|
|
return FALSE;
|
|
|
|
|
2017-06-26 04:55:48 +00:00
|
|
|
monitor = get_monitor (monitor_src);
|
|
|
|
logical_monitor = meta_monitor_get_logical_monitor (monitor);
|
|
|
|
clutter_stage_capture_into (stage, FALSE, &logical_monitor->rect, data);
|
2018-12-14 16:47:57 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
draw_cursor_sprite_via_offscreen (MetaScreenCastMonitorStreamSrc *monitor_src,
|
|
|
|
CoglTexture *cursor_texture,
|
|
|
|
int bitmap_width,
|
|
|
|
int bitmap_height,
|
|
|
|
uint32_t *bitmap_data,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
MetaBackend *backend = get_backend (monitor_src);
|
|
|
|
ClutterBackend *clutter_backend = meta_backend_get_clutter_backend (backend);
|
|
|
|
CoglContext *cogl_context =
|
|
|
|
clutter_backend_get_cogl_context (clutter_backend);
|
|
|
|
CoglTexture2D *bitmap_texture;
|
|
|
|
CoglOffscreen *offscreen;
|
|
|
|
CoglFramebuffer *fb;
|
|
|
|
CoglPipeline *pipeline;
|
|
|
|
CoglColor clear_color;
|
|
|
|
|
|
|
|
bitmap_texture = cogl_texture_2d_new_with_size (cogl_context,
|
|
|
|
bitmap_width, bitmap_height);
|
|
|
|
cogl_primitive_texture_set_auto_mipmap (COGL_PRIMITIVE_TEXTURE (bitmap_texture),
|
|
|
|
FALSE);
|
|
|
|
if (!cogl_texture_allocate (COGL_TEXTURE (bitmap_texture), error))
|
|
|
|
{
|
|
|
|
cogl_object_unref (bitmap_texture);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
offscreen = cogl_offscreen_new_with_texture (COGL_TEXTURE (bitmap_texture));
|
|
|
|
fb = COGL_FRAMEBUFFER (offscreen);
|
|
|
|
cogl_object_unref (bitmap_texture);
|
|
|
|
if (!cogl_framebuffer_allocate (fb, error))
|
|
|
|
{
|
|
|
|
cogl_object_unref (fb);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
pipeline = cogl_pipeline_new (cogl_context);
|
|
|
|
cogl_pipeline_set_layer_texture (pipeline, 0, cursor_texture);
|
|
|
|
cogl_pipeline_set_layer_filters (pipeline, 0,
|
|
|
|
COGL_PIPELINE_FILTER_LINEAR,
|
|
|
|
COGL_PIPELINE_FILTER_LINEAR);
|
|
|
|
cogl_color_init_from_4ub (&clear_color, 0, 0, 0, 0);
|
|
|
|
cogl_framebuffer_clear (fb, COGL_BUFFER_BIT_COLOR, &clear_color);
|
|
|
|
cogl_framebuffer_draw_rectangle (fb, pipeline,
|
|
|
|
-1, 1, 1, -1);
|
|
|
|
cogl_object_unref (pipeline);
|
|
|
|
|
|
|
|
cogl_framebuffer_read_pixels (fb,
|
|
|
|
0, 0,
|
|
|
|
bitmap_width, bitmap_height,
|
|
|
|
COGL_PIXEL_FORMAT_RGBA_8888_PRE,
|
|
|
|
(uint8_t *) bitmap_data);
|
|
|
|
cogl_object_unref (fb);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_screen_cast_monitor_stream_src_set_cursor_metadata (MetaScreenCastStreamSrc *src,
|
|
|
|
struct spa_meta_cursor *spa_meta_cursor)
|
|
|
|
{
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (src);
|
|
|
|
MetaBackend *backend = get_backend (monitor_src);
|
|
|
|
MetaCursorRenderer *cursor_renderer =
|
|
|
|
meta_backend_get_cursor_renderer (backend);
|
|
|
|
MetaRenderer *renderer = meta_backend_get_renderer (backend);
|
|
|
|
MetaSpaType *spa_type = meta_screen_cast_stream_src_get_spa_type (src);
|
|
|
|
GError *error = NULL;
|
|
|
|
MetaCursorSprite *cursor_sprite;
|
|
|
|
CoglTexture *cursor_texture;
|
|
|
|
MetaMonitor *monitor;
|
|
|
|
MetaLogicalMonitor *logical_monitor;
|
|
|
|
MetaRectangle logical_monitor_layout;
|
|
|
|
ClutterRect logical_monitor_rect;
|
|
|
|
MetaRendererView *view;
|
|
|
|
float view_scale;
|
|
|
|
ClutterPoint cursor_position;
|
|
|
|
struct spa_meta_bitmap *spa_meta_bitmap;
|
|
|
|
|
|
|
|
cursor_sprite = meta_cursor_renderer_get_cursor (cursor_renderer);
|
|
|
|
if (cursor_sprite)
|
|
|
|
cursor_texture = meta_cursor_sprite_get_cogl_texture (cursor_sprite);
|
|
|
|
else
|
|
|
|
cursor_texture = NULL;
|
|
|
|
|
|
|
|
if (!is_cursor_in_stream (monitor_src))
|
|
|
|
{
|
|
|
|
spa_meta_cursor->id = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
monitor = get_monitor (monitor_src);
|
|
|
|
logical_monitor = meta_monitor_get_logical_monitor (monitor);
|
|
|
|
logical_monitor_layout = meta_logical_monitor_get_layout (logical_monitor);
|
|
|
|
logical_monitor_rect =
|
|
|
|
meta_rectangle_to_clutter_rect (&logical_monitor_layout);
|
|
|
|
|
|
|
|
view = meta_renderer_get_view_from_logical_monitor (renderer,
|
|
|
|
logical_monitor);
|
|
|
|
if (view)
|
|
|
|
view_scale = clutter_stage_view_get_scale (CLUTTER_STAGE_VIEW (view));
|
|
|
|
else
|
|
|
|
view_scale = 1.0;
|
|
|
|
|
|
|
|
cursor_position = meta_cursor_renderer_get_position (cursor_renderer);
|
|
|
|
cursor_position.x -= logical_monitor_rect.origin.x;
|
|
|
|
cursor_position.y -= logical_monitor_rect.origin.y;
|
|
|
|
cursor_position.x *= view_scale;
|
|
|
|
cursor_position.y *= view_scale;
|
|
|
|
|
|
|
|
spa_meta_cursor->id = 1;
|
|
|
|
spa_meta_cursor->position.x = (int32_t) roundf (cursor_position.x);
|
|
|
|
spa_meta_cursor->position.y = (int32_t) roundf (cursor_position.y);
|
2019-01-03 15:51:08 +00:00
|
|
|
|
|
|
|
if (!monitor_src->cursor_bitmap_invalid)
|
|
|
|
{
|
|
|
|
spa_meta_cursor->hotspot.x = 0;
|
|
|
|
spa_meta_cursor->hotspot.y = 0;
|
|
|
|
spa_meta_cursor->bitmap_offset = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
monitor_src->cursor_bitmap_invalid = FALSE;
|
|
|
|
|
2018-12-14 16:47:57 +00:00
|
|
|
spa_meta_cursor->bitmap_offset = sizeof (struct spa_meta_cursor);
|
|
|
|
|
|
|
|
spa_meta_bitmap = SPA_MEMBER (spa_meta_cursor,
|
|
|
|
spa_meta_cursor->bitmap_offset,
|
|
|
|
struct spa_meta_bitmap);
|
|
|
|
spa_meta_bitmap->format = spa_type->video_format.RGBA;
|
|
|
|
spa_meta_bitmap->offset = sizeof (struct spa_meta_bitmap);
|
|
|
|
|
|
|
|
if (cursor_texture)
|
|
|
|
{
|
|
|
|
float cursor_scale;
|
|
|
|
float bitmap_scale;
|
|
|
|
int hotspot_x, hotspot_y;
|
|
|
|
int texture_width, texture_height;
|
|
|
|
int bitmap_width, bitmap_height;
|
|
|
|
uint32_t *bitmap_data;
|
|
|
|
|
|
|
|
cursor_scale = meta_cursor_sprite_get_texture_scale (cursor_sprite);
|
|
|
|
bitmap_scale = view_scale * cursor_scale;
|
|
|
|
|
|
|
|
meta_cursor_sprite_get_hotspot (cursor_sprite, &hotspot_x, &hotspot_y);
|
|
|
|
spa_meta_cursor->hotspot.x = (int32_t) roundf (hotspot_x * bitmap_scale);
|
|
|
|
spa_meta_cursor->hotspot.y = (int32_t) roundf (hotspot_y * bitmap_scale);
|
|
|
|
|
|
|
|
texture_width = cogl_texture_get_width (cursor_texture);
|
|
|
|
texture_height = cogl_texture_get_height (cursor_texture);
|
|
|
|
bitmap_width = texture_width * bitmap_scale;
|
|
|
|
bitmap_height = texture_height * bitmap_scale;
|
|
|
|
|
|
|
|
spa_meta_bitmap->size.width = bitmap_width;
|
|
|
|
spa_meta_bitmap->size.height = bitmap_height;
|
|
|
|
spa_meta_bitmap->stride = bitmap_width * 4;
|
|
|
|
|
|
|
|
bitmap_data = SPA_MEMBER (spa_meta_bitmap,
|
|
|
|
spa_meta_bitmap->offset,
|
|
|
|
uint32_t);
|
|
|
|
|
|
|
|
if (texture_width == bitmap_width &&
|
|
|
|
texture_height == bitmap_height)
|
|
|
|
{
|
|
|
|
cogl_texture_get_data (cursor_texture,
|
|
|
|
COGL_PIXEL_FORMAT_RGBA_8888_PRE,
|
|
|
|
texture_width * 4,
|
|
|
|
(uint8_t *) bitmap_data);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!draw_cursor_sprite_via_offscreen (monitor_src,
|
|
|
|
cursor_texture,
|
|
|
|
bitmap_width,
|
|
|
|
bitmap_height,
|
|
|
|
bitmap_data,
|
|
|
|
&error))
|
|
|
|
{
|
|
|
|
g_warning ("Failed to draw cursor via offscreen: %s",
|
|
|
|
error->message);
|
|
|
|
g_error_free (error);
|
|
|
|
spa_meta_cursor->id = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
spa_meta_cursor->hotspot.x = 0;
|
|
|
|
spa_meta_cursor->hotspot.y = 0;
|
|
|
|
|
|
|
|
*spa_meta_bitmap = (struct spa_meta_bitmap) { 0 };
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
meta_screen_cast_monitor_stream_src_is_cursor_sprite_inhibited (MetaHwCursorInhibitor *inhibitor,
|
|
|
|
MetaCursorSprite *cursor_sprite)
|
|
|
|
{
|
|
|
|
MetaScreenCastMonitorStreamSrc *monitor_src =
|
|
|
|
META_SCREEN_CAST_MONITOR_STREAM_SRC (inhibitor);
|
|
|
|
|
|
|
|
return is_cursor_in_stream (monitor_src);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
hw_cursor_inhibitor_iface_init (MetaHwCursorInhibitorInterface *iface)
|
|
|
|
{
|
|
|
|
iface->is_cursor_sprite_inhibited =
|
|
|
|
meta_screen_cast_monitor_stream_src_is_cursor_sprite_inhibited;
|
2017-06-26 04:55:48 +00:00
|
|
|
}
|
2017-06-21 06:23:44 +00:00
|
|
|
|
2017-06-26 04:55:48 +00:00
|
|
|
MetaScreenCastMonitorStreamSrc *
|
|
|
|
meta_screen_cast_monitor_stream_src_new (MetaScreenCastMonitorStream *monitor_stream,
|
|
|
|
GError **error)
|
|
|
|
{
|
2017-06-28 07:17:32 +00:00
|
|
|
return g_initable_new (META_TYPE_SCREEN_CAST_MONITOR_STREAM_SRC, NULL, error,
|
2017-06-26 04:55:48 +00:00
|
|
|
"stream", monitor_stream,
|
|
|
|
NULL);
|
2017-06-21 06:23:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_screen_cast_monitor_stream_src_init (MetaScreenCastMonitorStreamSrc *monitor_src)
|
|
|
|
{
|
2019-01-03 15:51:08 +00:00
|
|
|
monitor_src->cursor_bitmap_invalid = TRUE;
|
2017-06-21 06:23:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_screen_cast_monitor_stream_src_class_init (MetaScreenCastMonitorStreamSrcClass *klass)
|
|
|
|
{
|
|
|
|
MetaScreenCastStreamSrcClass *src_class =
|
|
|
|
META_SCREEN_CAST_STREAM_SRC_CLASS (klass);
|
|
|
|
|
|
|
|
src_class->get_specs = meta_screen_cast_monitor_stream_src_get_specs;
|
2017-06-26 04:55:48 +00:00
|
|
|
src_class->enable = meta_screen_cast_monitor_stream_src_enable;
|
|
|
|
src_class->disable = meta_screen_cast_monitor_stream_src_disable;
|
2017-06-21 06:23:44 +00:00
|
|
|
src_class->record_frame = meta_screen_cast_monitor_stream_src_record_frame;
|
2018-12-14 16:47:57 +00:00
|
|
|
src_class->set_cursor_metadata =
|
|
|
|
meta_screen_cast_monitor_stream_src_set_cursor_metadata;
|
2017-06-21 06:23:44 +00:00
|
|
|
}
|