4e402b3972
The 'cursor-mode', which currently is limited to RecordMonitor(), allows the user to either do screen casts where the cursor is hidden, embedded in the framebuffer, or sent as PipeWire stream metadata. The latter allows the user to get cursor updates sent, including the cursor sprite, without requiring a stage paint each frame. Currently this is done by using the cursor sprite texture, and either reading directly from, or drawing to an offscreen framebuffer which is read from instead, in case the texture is scaled. https://gitlab.gnome.org/GNOME/mutter/merge_requests/357
248 lines
7.6 KiB
C
248 lines
7.6 KiB
C
/*
|
|
* Copyright (C) 2018 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-window-stream-src.h"
|
|
|
|
#include "backends/meta-backend-private.h"
|
|
#include "backends/meta-screen-cast-window.h"
|
|
#include "backends/meta-screen-cast-window-stream.h"
|
|
#include "compositor/meta-window-actor-private.h"
|
|
|
|
struct _MetaScreenCastWindowStreamSrc
|
|
{
|
|
MetaScreenCastStreamSrc parent;
|
|
|
|
MetaWindowActor *window_actor;
|
|
|
|
unsigned long actor_painted_handler_id;
|
|
unsigned long actor_destroyed_handler_id;
|
|
};
|
|
|
|
G_DEFINE_TYPE (MetaScreenCastWindowStreamSrc,
|
|
meta_screen_cast_window_stream_src,
|
|
META_TYPE_SCREEN_CAST_STREAM_SRC)
|
|
|
|
static MetaScreenCastWindowStream *
|
|
get_window_stream (MetaScreenCastWindowStreamSrc *window_src)
|
|
{
|
|
MetaScreenCastStreamSrc *src;
|
|
MetaScreenCastStream *stream;
|
|
|
|
src = META_SCREEN_CAST_STREAM_SRC (window_src);
|
|
stream = meta_screen_cast_stream_src_get_stream (src);
|
|
|
|
return META_SCREEN_CAST_WINDOW_STREAM (stream);
|
|
}
|
|
|
|
static MetaWindow *
|
|
get_window (MetaScreenCastWindowStreamSrc *window_src)
|
|
{
|
|
MetaScreenCastWindowStream *window_stream;
|
|
|
|
window_stream = get_window_stream (window_src);
|
|
|
|
return meta_screen_cast_window_stream_get_window (window_stream);
|
|
}
|
|
|
|
static int
|
|
get_stream_width (MetaScreenCastWindowStreamSrc *window_src)
|
|
{
|
|
MetaScreenCastWindowStream *window_stream;
|
|
|
|
window_stream = get_window_stream (window_src);
|
|
|
|
return meta_screen_cast_window_stream_get_width (window_stream);
|
|
}
|
|
|
|
static int
|
|
get_stream_height (MetaScreenCastWindowStreamSrc *window_src)
|
|
{
|
|
MetaScreenCastWindowStream *window_stream;
|
|
|
|
window_stream = get_window_stream (window_src);
|
|
|
|
return meta_screen_cast_window_stream_get_height (window_stream);
|
|
}
|
|
|
|
static gboolean
|
|
capture_into (MetaScreenCastWindowStreamSrc *window_src,
|
|
uint8_t *data)
|
|
{
|
|
MetaRectangle stream_rect;
|
|
MetaScreenCastWindow *screen_cast_window;
|
|
|
|
stream_rect.x = 0;
|
|
stream_rect.y = 0;
|
|
stream_rect.width = get_stream_width (window_src);
|
|
stream_rect.height = get_stream_height (window_src);
|
|
|
|
screen_cast_window = META_SCREEN_CAST_WINDOW (window_src->window_actor);
|
|
meta_screen_cast_window_capture_into (screen_cast_window, &stream_rect, data);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
meta_screen_cast_window_stream_src_get_specs (MetaScreenCastStreamSrc *src,
|
|
int *width,
|
|
int *height,
|
|
float *frame_rate)
|
|
{
|
|
MetaScreenCastWindowStreamSrc *window_src =
|
|
META_SCREEN_CAST_WINDOW_STREAM_SRC (src);
|
|
|
|
*width = get_stream_width (window_src);
|
|
*height = get_stream_height (window_src);
|
|
*frame_rate = 60.0f;
|
|
}
|
|
|
|
static gboolean
|
|
meta_screen_cast_window_stream_src_get_videocrop (MetaScreenCastStreamSrc *src,
|
|
MetaRectangle *crop_rect)
|
|
{
|
|
MetaScreenCastWindowStreamSrc *window_src =
|
|
META_SCREEN_CAST_WINDOW_STREAM_SRC (src);
|
|
MetaScreenCastWindow *screen_cast_window;
|
|
MetaRectangle stream_rect;
|
|
|
|
screen_cast_window = META_SCREEN_CAST_WINDOW (window_src->window_actor);
|
|
meta_screen_cast_window_get_frame_bounds (screen_cast_window, crop_rect);
|
|
|
|
stream_rect.x = 0;
|
|
stream_rect.y = 0;
|
|
stream_rect.width = get_stream_width (window_src);
|
|
stream_rect.height = get_stream_height (window_src);
|
|
|
|
meta_rectangle_intersect (crop_rect, &stream_rect, crop_rect);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
meta_screen_cast_window_stream_src_stop (MetaScreenCastWindowStreamSrc *window_src)
|
|
|
|
{
|
|
if (!window_src->window_actor)
|
|
return;
|
|
|
|
if (window_src->actor_painted_handler_id)
|
|
g_signal_handler_disconnect (window_src->window_actor,
|
|
window_src->actor_painted_handler_id);
|
|
window_src->actor_painted_handler_id = 0;
|
|
|
|
if (window_src->actor_destroyed_handler_id)
|
|
g_signal_handler_disconnect (window_src->window_actor,
|
|
window_src->actor_destroyed_handler_id);
|
|
window_src->actor_destroyed_handler_id = 0;
|
|
}
|
|
|
|
static void
|
|
window_actor_painted (MetaWindowActor *actor,
|
|
MetaScreenCastWindowStreamSrc *window_src)
|
|
{
|
|
MetaScreenCastStreamSrc *src = META_SCREEN_CAST_STREAM_SRC (window_src);
|
|
|
|
meta_screen_cast_stream_src_maybe_record_frame (src);
|
|
}
|
|
|
|
static void
|
|
window_actor_destroyed (MetaWindowActor *actor,
|
|
MetaScreenCastWindowStreamSrc *window_src)
|
|
{
|
|
meta_screen_cast_window_stream_src_stop (window_src);
|
|
window_src->window_actor = NULL;
|
|
}
|
|
|
|
static void
|
|
meta_screen_cast_window_stream_src_enable (MetaScreenCastStreamSrc *src)
|
|
{
|
|
MetaScreenCastWindowStreamSrc *window_src =
|
|
META_SCREEN_CAST_WINDOW_STREAM_SRC (src);
|
|
MetaWindowActor *window_actor;
|
|
|
|
window_actor = meta_window_actor_from_window (get_window (window_src));
|
|
if (!window_actor)
|
|
return;
|
|
|
|
window_src->window_actor = window_actor;
|
|
|
|
window_src->actor_painted_handler_id =
|
|
g_signal_connect_after (window_src->window_actor,
|
|
"paint",
|
|
G_CALLBACK (window_actor_painted),
|
|
window_src);
|
|
|
|
window_src->actor_destroyed_handler_id =
|
|
g_signal_connect (window_src->window_actor,
|
|
"destroy",
|
|
G_CALLBACK (window_actor_destroyed),
|
|
window_src);
|
|
}
|
|
|
|
static void
|
|
meta_screen_cast_window_stream_src_disable (MetaScreenCastStreamSrc *src)
|
|
{
|
|
MetaScreenCastWindowStreamSrc *window_src =
|
|
META_SCREEN_CAST_WINDOW_STREAM_SRC (src);
|
|
|
|
meta_screen_cast_window_stream_src_stop (window_src);
|
|
}
|
|
|
|
static gboolean
|
|
meta_screen_cast_window_stream_src_record_frame (MetaScreenCastStreamSrc *src,
|
|
uint8_t *data)
|
|
{
|
|
MetaScreenCastWindowStreamSrc *window_src =
|
|
META_SCREEN_CAST_WINDOW_STREAM_SRC (src);
|
|
|
|
capture_into (window_src, data);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
MetaScreenCastWindowStreamSrc *
|
|
meta_screen_cast_window_stream_src_new (MetaScreenCastWindowStream *window_stream,
|
|
GError **error)
|
|
{
|
|
return g_initable_new (META_TYPE_SCREEN_CAST_WINDOW_STREAM_SRC, NULL, error,
|
|
"stream", window_stream,
|
|
NULL);
|
|
}
|
|
|
|
static void
|
|
meta_screen_cast_window_stream_src_init (MetaScreenCastWindowStreamSrc *window_src)
|
|
{
|
|
}
|
|
|
|
static void
|
|
meta_screen_cast_window_stream_src_class_init (MetaScreenCastWindowStreamSrcClass *klass)
|
|
{
|
|
MetaScreenCastStreamSrcClass *src_class =
|
|
META_SCREEN_CAST_STREAM_SRC_CLASS (klass);
|
|
|
|
src_class->get_specs = meta_screen_cast_window_stream_src_get_specs;
|
|
src_class->enable = meta_screen_cast_window_stream_src_enable;
|
|
src_class->disable = meta_screen_cast_window_stream_src_disable;
|
|
src_class->record_frame = meta_screen_cast_window_stream_src_record_frame;
|
|
src_class->get_videocrop = meta_screen_cast_window_stream_src_get_videocrop;
|
|
}
|