2020-06-09 00:49:55 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2009 Sander Dijkhuis
|
|
|
|
* Copyright 2014 Red Hat, Inc.
|
|
|
|
* Copyright 2020 Endless Foundation.
|
|
|
|
*
|
|
|
|
* 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, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* Portions adapted from gnome-shell/src/shell-global.c
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* SECTION:meta-background-content
|
|
|
|
* @title: MetaBackgroundContent
|
|
|
|
* @short_description: ClutterContent for painting the root window background
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The overall model drawing model of this content is that we have one
|
|
|
|
* texture, or two interpolated textures, possibly with alpha or
|
|
|
|
* margins that let the underlying background show through, blended
|
|
|
|
* over a solid color or a gradient. The result of that combination
|
|
|
|
* can then be affected by a "vignette" that darkens the background
|
|
|
|
* away from a central point (or as a no-GLSL fallback, simply darkens
|
|
|
|
* the background) and by overall opacity.
|
|
|
|
*
|
|
|
|
* As of GNOME 3.14, GNOME is only using a fraction of this when the
|
|
|
|
* user sets the background through the control center - what can be
|
|
|
|
* set is:
|
|
|
|
*
|
|
|
|
* A single image without a border
|
|
|
|
* An animation of images without a border that blend together,
|
|
|
|
* with the blend changing every 4-5 minutes
|
|
|
|
* A solid color with a repeated noise texture blended over it
|
|
|
|
*
|
|
|
|
* This all is pretty easy to do in a fragment shader, except when:
|
|
|
|
*
|
|
|
|
* A) We don't have GLSL - in this case, the operation of
|
|
|
|
* interpolating the two textures and blending the result over the
|
|
|
|
* background can't be expressed with Cogl's fixed-function layer
|
|
|
|
* combining (which is confined to what GL's texture environment
|
|
|
|
* combining can do) So we can only handle the above directly if
|
|
|
|
* there are no margins or alpha.
|
|
|
|
*
|
|
|
|
* B) The image textures are sliced. Texture size limits on older
|
|
|
|
* hardware (pre-965 intel hardware, r300, etc.) is often 2048,
|
|
|
|
* and it would be common to use a texture larger than this for a
|
|
|
|
* background and expect it to be scaled down. Cogl can compensate
|
|
|
|
* for this by breaking the texture up into multiple textures, but
|
|
|
|
* can't multitexture with sliced textures. So we can only handle
|
|
|
|
* the above if there's a single texture.
|
|
|
|
*
|
|
|
|
* However, even when we *can* represent everything in a single pass,
|
|
|
|
* it's not necessarily efficient. If we want to draw a 1024x768
|
|
|
|
* background, it's pretty inefficient to bilinearly texture from
|
|
|
|
* two 2560x1440 images and mix that. So the drawing model we take
|
|
|
|
* here is that MetaBackground generates a single texture (which
|
|
|
|
* might be a 1x1 texture for a solid color, or a 1x2 texture for a
|
|
|
|
* gradient, or a repeated texture for wallpaper, or a pre-rendered
|
|
|
|
* texture the size of the screen), and we draw with that, possibly
|
|
|
|
* adding the vignette and opacity.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "compositor/meta-background-content-private.h"
|
|
|
|
|
2021-02-08 08:33:55 +00:00
|
|
|
#include "backends/meta-backend-private.h"
|
2020-06-09 00:49:55 +00:00
|
|
|
#include "clutter/clutter.h"
|
|
|
|
#include "compositor/clutter-utils.h"
|
|
|
|
#include "compositor/cogl-utils.h"
|
|
|
|
#include "compositor/meta-background-private.h"
|
|
|
|
#include "compositor/meta-cullable.h"
|
|
|
|
#include "meta/display.h"
|
|
|
|
|
|
|
|
typedef enum
|
|
|
|
{
|
|
|
|
CHANGED_BACKGROUND = 1 << 0,
|
|
|
|
CHANGED_EFFECTS = 1 << 2,
|
|
|
|
CHANGED_VIGNETTE_PARAMETERS = 1 << 3,
|
|
|
|
CHANGED_GRADIENT_PARAMETERS = 1 << 4,
|
2021-02-08 08:33:55 +00:00
|
|
|
CHANGED_ROUNDED_CLIP_PARAMETERS = 1 << 5,
|
2020-06-09 00:49:55 +00:00
|
|
|
CHANGED_ALL = 0xFFFF
|
|
|
|
} ChangedFlags;
|
|
|
|
|
|
|
|
#define GRADIENT_VERTEX_SHADER_DECLARATIONS \
|
|
|
|
"uniform vec2 scale;\n" \
|
|
|
|
"varying vec2 position;\n" \
|
|
|
|
|
|
|
|
#define GRADIENT_VERTEX_SHADER_CODE \
|
|
|
|
"position = cogl_tex_coord0_in.xy * scale;\n" \
|
|
|
|
|
|
|
|
#define GRADIENT_FRAGMENT_SHADER_DECLARATIONS \
|
|
|
|
"uniform float gradient_height_perc;\n" \
|
|
|
|
"uniform float gradient_max_darkness;\n" \
|
|
|
|
"varying vec2 position;\n" \
|
|
|
|
|
|
|
|
#define GRADIENT_FRAGMENT_SHADER_CODE \
|
|
|
|
"float min_brightness = 1.0 - gradient_max_darkness;\n" \
|
|
|
|
"float gradient_y_pos = min(position.y, gradient_height_perc) / gradient_height_perc;\n" \
|
|
|
|
"float pixel_brightness = (1.0 - min_brightness) * gradient_y_pos + min_brightness;\n" \
|
|
|
|
"cogl_color_out.rgb = cogl_color_out.rgb * pixel_brightness;\n" \
|
|
|
|
|
|
|
|
#define VIGNETTE_VERTEX_SHADER_DECLARATIONS \
|
|
|
|
"uniform vec2 scale;\n" \
|
|
|
|
"uniform vec2 offset;\n" \
|
|
|
|
"varying vec2 position;\n" \
|
|
|
|
|
|
|
|
#define VIGNETTE_VERTEX_SHADER_CODE \
|
|
|
|
"position = cogl_tex_coord0_in.xy * scale + offset;\n" \
|
|
|
|
|
|
|
|
#define VIGNETTE_SQRT_2 "1.4142"
|
|
|
|
|
|
|
|
#define VIGNETTE_FRAGMENT_SHADER_DECLARATIONS \
|
|
|
|
"uniform float vignette_sharpness;\n" \
|
|
|
|
"varying vec2 position;\n" \
|
|
|
|
"float rand(vec2 p) { return fract(sin(dot(p, vec2(12.9898, 78.233))) * 43758.5453123); }\n" \
|
|
|
|
|
|
|
|
#define VIGNETTE_FRAGMENT_SHADER_CODE \
|
|
|
|
"float t = " VIGNETTE_SQRT_2 " * length(position);\n" \
|
|
|
|
"t = min(t, 1.0);\n" \
|
|
|
|
"float pixel_brightness = 1.0 - t * vignette_sharpness;\n" \
|
|
|
|
"cogl_color_out.rgb = cogl_color_out.rgb * pixel_brightness;\n" \
|
|
|
|
"cogl_color_out.rgb += (rand(position) - 0.5) / 255.0;\n" \
|
|
|
|
|
2021-02-08 08:33:55 +00:00
|
|
|
#define ROUNDED_CLIP_FRAGMENT_SHADER_DECLARATIONS \
|
2021-05-10 09:18:55 +00:00
|
|
|
"uniform vec4 bounds; // x, y: top left; z, w: bottom right \n"\
|
|
|
|
"uniform float clip_radius; \n"\
|
2021-02-08 08:33:55 +00:00
|
|
|
"uniform vec2 pixel_step; \n"\
|
|
|
|
" \n"\
|
|
|
|
"float \n"\
|
2021-05-10 09:18:55 +00:00
|
|
|
"rounded_rect_coverage (vec2 p) \n"\
|
2021-02-08 08:33:55 +00:00
|
|
|
"{ \n"\
|
2021-05-10 09:18:55 +00:00
|
|
|
" float center_left = bounds.x + clip_radius; \n"\
|
|
|
|
" float center_right = bounds.z - clip_radius; \n"\
|
|
|
|
" float center_x; \n"\
|
2021-02-08 08:33:55 +00:00
|
|
|
" \n"\
|
2021-05-10 09:18:55 +00:00
|
|
|
" if (p.x < center_left) \n"\
|
|
|
|
" center_x = center_left; \n"\
|
|
|
|
" else if (p.x > center_right) \n"\
|
|
|
|
" center_x = center_right; \n"\
|
|
|
|
" else \n"\
|
|
|
|
" return 1.0; // The vast majority of pixels exit early here \n"\
|
2021-02-08 08:33:55 +00:00
|
|
|
" \n"\
|
2021-05-10 09:18:55 +00:00
|
|
|
" float center_top = bounds.y + clip_radius; \n"\
|
|
|
|
" float center_bottom = bounds.w - clip_radius; \n"\
|
|
|
|
" float center_y; \n"\
|
2021-02-08 08:33:55 +00:00
|
|
|
" \n"\
|
2021-05-10 09:18:55 +00:00
|
|
|
" if (p.y < center_top) \n"\
|
|
|
|
" center_y = center_top; \n"\
|
|
|
|
" else if (p.y > center_bottom) \n"\
|
|
|
|
" center_y = center_bottom; \n"\
|
|
|
|
" else \n"\
|
2021-03-11 21:57:35 +00:00
|
|
|
" return 1.0; \n"\
|
|
|
|
" \n"\
|
2021-05-10 09:18:55 +00:00
|
|
|
" vec2 delta = p - vec2 (center_x, center_y); \n"\
|
|
|
|
" float dist_squared = dot (delta, delta); \n"\
|
2021-02-08 08:33:55 +00:00
|
|
|
" \n"\
|
2021-05-10 09:18:55 +00:00
|
|
|
" // Fully outside the circle \n"\
|
2021-11-22 10:23:08 +00:00
|
|
|
" float outer_radius = clip_radius + 0.5; \n"\
|
|
|
|
" if (dist_squared >= (outer_radius * outer_radius)) \n"\
|
2021-05-10 09:18:55 +00:00
|
|
|
" return 0.0; \n"\
|
2021-02-08 08:33:55 +00:00
|
|
|
" \n"\
|
2021-05-10 09:18:55 +00:00
|
|
|
" // Fully inside the circle \n"\
|
2021-11-22 10:23:08 +00:00
|
|
|
" float inner_radius = clip_radius - 0.5; \n"\
|
2021-05-10 09:18:55 +00:00
|
|
|
" if (dist_squared <= (inner_radius * inner_radius)) \n"\
|
|
|
|
" return 1.0; \n"\
|
2021-02-08 08:33:55 +00:00
|
|
|
" \n"\
|
2021-05-10 09:18:55 +00:00
|
|
|
" // Only pixels on the edge of the curve need expensive antialiasing \n"\
|
2021-11-22 10:23:08 +00:00
|
|
|
" return outer_radius - sqrt (dist_squared); \n"\
|
2021-02-08 08:33:55 +00:00
|
|
|
"} \n"
|
|
|
|
|
|
|
|
#define ROUNDED_CLIP_FRAGMENT_SHADER_CODE \
|
|
|
|
"vec2 texture_coord; \n"\
|
|
|
|
" \n"\
|
|
|
|
"texture_coord = cogl_tex_coord0_in.xy / pixel_step; \n"\
|
|
|
|
" \n"\
|
2021-05-10 09:18:55 +00:00
|
|
|
"cogl_color_out *= rounded_rect_coverage (texture_coord); \n"
|
2021-02-08 08:33:55 +00:00
|
|
|
|
2020-06-09 00:49:55 +00:00
|
|
|
typedef struct _MetaBackgroundLayer MetaBackgroundLayer;
|
|
|
|
|
|
|
|
typedef enum
|
|
|
|
{
|
|
|
|
PIPELINE_VIGNETTE = (1 << 0),
|
|
|
|
PIPELINE_BLEND = (1 << 1),
|
|
|
|
PIPELINE_GRADIENT = (1 << 2),
|
2021-02-08 08:33:55 +00:00
|
|
|
PIPELINE_ROUNDED_CLIP = (1 << 3),
|
background-content: Fix pipeline cache size
The cache had the size 9, which was "big enough" in the past, but when
more ways pipelines could be constructed, the size was not enough. The
need to increase the cache size was hard to spot though, since adding
pipeline flag didn't give any hints about the cache being directly tied
to these flag values.
So, when enough flag bits were set when attempting to retrieve and put a
pipeline in the cache, it'd instead overwrite some arbitrary stack
memory, which would sooner or later result in a memory corruption
induced crash. Valgrind could not detect this particular memory
corruption, as it messed up stack memory, not e.g. freed heap memory, so
it instead got confused and thought plain stack values were unreadable.
Fix these two issues by making the cache size the combination of all
pipeline flags + 1, so that we can safely put any flag combination in
the cache.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1747>
2021-02-24 15:39:42 +00:00
|
|
|
|
|
|
|
PIPELINE_ALL = (PIPELINE_VIGNETTE |
|
|
|
|
PIPELINE_BLEND |
|
|
|
|
PIPELINE_GRADIENT |
|
|
|
|
PIPELINE_ROUNDED_CLIP)
|
2020-06-09 00:49:55 +00:00
|
|
|
} PipelineFlags;
|
|
|
|
|
|
|
|
struct _MetaBackgroundContent
|
|
|
|
{
|
|
|
|
GObject parent;
|
|
|
|
|
|
|
|
MetaDisplay *display;
|
|
|
|
int monitor;
|
|
|
|
|
|
|
|
MetaBackground *background;
|
|
|
|
|
|
|
|
gboolean gradient;
|
|
|
|
double gradient_max_darkness;
|
|
|
|
int gradient_height;
|
|
|
|
|
|
|
|
gboolean vignette;
|
|
|
|
double vignette_brightness;
|
|
|
|
double vignette_sharpness;
|
|
|
|
|
2021-02-08 08:33:55 +00:00
|
|
|
gboolean has_rounded_clip;
|
|
|
|
float rounded_clip_radius;
|
|
|
|
gboolean rounded_clip_bounds_set;
|
|
|
|
graphene_rect_t rounded_clip_bounds;
|
|
|
|
|
2020-06-09 00:49:55 +00:00
|
|
|
ChangedFlags changed;
|
|
|
|
CoglPipeline *pipeline;
|
|
|
|
PipelineFlags pipeline_flags;
|
|
|
|
cairo_rectangle_int_t texture_area;
|
2020-07-01 10:09:14 +00:00
|
|
|
int texture_width, texture_height;
|
2020-06-09 00:49:55 +00:00
|
|
|
|
|
|
|
cairo_region_t *clip_region;
|
|
|
|
cairo_region_t *unobscured_region;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void clutter_content_iface_init (ClutterContentInterface *iface);
|
|
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (MetaBackgroundContent,
|
|
|
|
meta_background_content,
|
|
|
|
G_TYPE_OBJECT,
|
|
|
|
G_IMPLEMENT_INTERFACE (CLUTTER_TYPE_CONTENT,
|
|
|
|
clutter_content_iface_init));
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
PROP_0,
|
|
|
|
PROP_META_DISPLAY,
|
|
|
|
PROP_MONITOR,
|
|
|
|
PROP_BACKGROUND,
|
|
|
|
PROP_GRADIENT,
|
|
|
|
PROP_GRADIENT_HEIGHT,
|
|
|
|
PROP_GRADIENT_MAX_DARKNESS,
|
|
|
|
PROP_VIGNETTE,
|
|
|
|
PROP_VIGNETTE_SHARPNESS,
|
|
|
|
PROP_VIGNETTE_BRIGHTNESS,
|
2021-02-08 08:33:55 +00:00
|
|
|
PROP_ROUNDED_CLIP_RADIUS,
|
2020-06-09 00:49:55 +00:00
|
|
|
N_PROPS,
|
|
|
|
};
|
|
|
|
|
|
|
|
static GParamSpec *properties[N_PROPS] = { NULL, };
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_clip_region (MetaBackgroundContent *self,
|
|
|
|
cairo_region_t *clip_region)
|
|
|
|
{
|
|
|
|
g_clear_pointer (&self->clip_region, cairo_region_destroy);
|
|
|
|
if (clip_region)
|
|
|
|
{
|
|
|
|
if (cairo_region_is_empty (clip_region))
|
|
|
|
self->clip_region = cairo_region_reference (clip_region);
|
|
|
|
else
|
|
|
|
self->clip_region = cairo_region_copy (clip_region);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_unobscured_region (MetaBackgroundContent *self,
|
|
|
|
cairo_region_t *unobscured_region)
|
|
|
|
{
|
|
|
|
g_clear_pointer (&self->unobscured_region, cairo_region_destroy);
|
|
|
|
if (unobscured_region)
|
|
|
|
{
|
|
|
|
if (cairo_region_is_empty (unobscured_region))
|
|
|
|
self->unobscured_region = cairo_region_reference (unobscured_region);
|
|
|
|
else
|
|
|
|
self->unobscured_region = cairo_region_copy (unobscured_region);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
invalidate_pipeline (MetaBackgroundContent *self,
|
|
|
|
ChangedFlags changed)
|
|
|
|
{
|
|
|
|
self->changed |= changed;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_background_changed (MetaBackground *background,
|
|
|
|
MetaBackgroundContent *self)
|
|
|
|
{
|
|
|
|
invalidate_pipeline (self, CHANGED_BACKGROUND);
|
|
|
|
clutter_content_invalidate (CLUTTER_CONTENT (self));
|
|
|
|
}
|
|
|
|
|
|
|
|
static CoglPipeline *
|
|
|
|
make_pipeline (PipelineFlags pipeline_flags)
|
|
|
|
{
|
background-content: Fix pipeline cache size
The cache had the size 9, which was "big enough" in the past, but when
more ways pipelines could be constructed, the size was not enough. The
need to increase the cache size was hard to spot though, since adding
pipeline flag didn't give any hints about the cache being directly tied
to these flag values.
So, when enough flag bits were set when attempting to retrieve and put a
pipeline in the cache, it'd instead overwrite some arbitrary stack
memory, which would sooner or later result in a memory corruption
induced crash. Valgrind could not detect this particular memory
corruption, as it messed up stack memory, not e.g. freed heap memory, so
it instead got confused and thought plain stack values were unreadable.
Fix these two issues by making the cache size the combination of all
pipeline flags + 1, so that we can safely put any flag combination in
the cache.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1747>
2021-02-24 15:39:42 +00:00
|
|
|
static CoglPipeline *templates[PIPELINE_ALL + 1];
|
2020-06-09 00:49:55 +00:00
|
|
|
CoglPipeline **templatep;
|
|
|
|
|
background-content: Fix pipeline cache size
The cache had the size 9, which was "big enough" in the past, but when
more ways pipelines could be constructed, the size was not enough. The
need to increase the cache size was hard to spot though, since adding
pipeline flag didn't give any hints about the cache being directly tied
to these flag values.
So, when enough flag bits were set when attempting to retrieve and put a
pipeline in the cache, it'd instead overwrite some arbitrary stack
memory, which would sooner or later result in a memory corruption
induced crash. Valgrind could not detect this particular memory
corruption, as it messed up stack memory, not e.g. freed heap memory, so
it instead got confused and thought plain stack values were unreadable.
Fix these two issues by making the cache size the combination of all
pipeline flags + 1, so that we can safely put any flag combination in
the cache.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1747>
2021-02-24 15:39:42 +00:00
|
|
|
g_assert (pipeline_flags < G_N_ELEMENTS (templates));
|
|
|
|
|
2020-06-09 00:49:55 +00:00
|
|
|
templatep = &templates[pipeline_flags];
|
|
|
|
if (*templatep == NULL)
|
|
|
|
{
|
|
|
|
/* Cogl automatically caches pipelines with no eviction policy,
|
|
|
|
* so we need to prevent identical pipelines from getting cached
|
|
|
|
* separately, by reusing the same shader snippets.
|
|
|
|
*/
|
|
|
|
*templatep = COGL_PIPELINE (meta_create_texture_pipeline (NULL));
|
|
|
|
|
|
|
|
if ((pipeline_flags & PIPELINE_VIGNETTE) != 0)
|
|
|
|
{
|
|
|
|
static CoglSnippet *vignette_vertex_snippet;
|
|
|
|
static CoglSnippet *vignette_fragment_snippet;
|
|
|
|
|
|
|
|
if (!vignette_vertex_snippet)
|
|
|
|
vignette_vertex_snippet = cogl_snippet_new (COGL_SNIPPET_HOOK_VERTEX,
|
|
|
|
VIGNETTE_VERTEX_SHADER_DECLARATIONS,
|
|
|
|
VIGNETTE_VERTEX_SHADER_CODE);
|
|
|
|
|
|
|
|
cogl_pipeline_add_snippet (*templatep, vignette_vertex_snippet);
|
|
|
|
|
|
|
|
if (!vignette_fragment_snippet)
|
|
|
|
vignette_fragment_snippet = cogl_snippet_new (COGL_SNIPPET_HOOK_FRAGMENT,
|
|
|
|
VIGNETTE_FRAGMENT_SHADER_DECLARATIONS,
|
|
|
|
VIGNETTE_FRAGMENT_SHADER_CODE);
|
|
|
|
|
|
|
|
cogl_pipeline_add_snippet (*templatep, vignette_fragment_snippet);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((pipeline_flags & PIPELINE_GRADIENT) != 0)
|
|
|
|
{
|
|
|
|
static CoglSnippet *gradient_vertex_snippet;
|
|
|
|
static CoglSnippet *gradient_fragment_snippet;
|
|
|
|
|
|
|
|
if (!gradient_vertex_snippet)
|
|
|
|
gradient_vertex_snippet = cogl_snippet_new (COGL_SNIPPET_HOOK_VERTEX,
|
|
|
|
GRADIENT_VERTEX_SHADER_DECLARATIONS,
|
|
|
|
GRADIENT_VERTEX_SHADER_CODE);
|
|
|
|
|
|
|
|
cogl_pipeline_add_snippet (*templatep, gradient_vertex_snippet);
|
|
|
|
|
|
|
|
if (!gradient_fragment_snippet)
|
|
|
|
gradient_fragment_snippet = cogl_snippet_new (COGL_SNIPPET_HOOK_FRAGMENT,
|
|
|
|
GRADIENT_FRAGMENT_SHADER_DECLARATIONS,
|
|
|
|
GRADIENT_FRAGMENT_SHADER_CODE);
|
|
|
|
|
|
|
|
cogl_pipeline_add_snippet (*templatep, gradient_fragment_snippet);
|
|
|
|
}
|
|
|
|
|
2021-02-08 08:33:55 +00:00
|
|
|
if ((pipeline_flags & PIPELINE_ROUNDED_CLIP) != 0)
|
|
|
|
{
|
|
|
|
static CoglSnippet *rounded_clip_fragment_snippet;
|
|
|
|
|
|
|
|
if (!rounded_clip_fragment_snippet)
|
|
|
|
{
|
|
|
|
rounded_clip_fragment_snippet =
|
|
|
|
cogl_snippet_new (COGL_SNIPPET_HOOK_FRAGMENT,
|
|
|
|
ROUNDED_CLIP_FRAGMENT_SHADER_DECLARATIONS,
|
|
|
|
ROUNDED_CLIP_FRAGMENT_SHADER_CODE);
|
|
|
|
}
|
|
|
|
|
|
|
|
cogl_pipeline_add_snippet (*templatep, rounded_clip_fragment_snippet);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-06-09 00:49:55 +00:00
|
|
|
if ((pipeline_flags & PIPELINE_BLEND) == 0)
|
|
|
|
cogl_pipeline_set_blend (*templatep, "RGBA = ADD (SRC_COLOR, 0)", NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
return cogl_pipeline_copy (*templatep);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
setup_pipeline (MetaBackgroundContent *self,
|
|
|
|
ClutterActor *actor,
|
|
|
|
ClutterPaintContext *paint_context,
|
|
|
|
cairo_rectangle_int_t *actor_pixel_rect)
|
|
|
|
{
|
|
|
|
PipelineFlags pipeline_flags = 0;
|
|
|
|
guint8 opacity;
|
|
|
|
float color_component;
|
|
|
|
CoglFramebuffer *fb;
|
2020-07-01 08:43:50 +00:00
|
|
|
CoglPipelineFilter min_filter, mag_filter;
|
2020-06-09 00:49:55 +00:00
|
|
|
|
|
|
|
opacity = clutter_actor_get_paint_opacity (actor);
|
|
|
|
if (opacity < 255)
|
|
|
|
pipeline_flags |= PIPELINE_BLEND;
|
2021-09-17 21:01:37 +00:00
|
|
|
if (self->vignette)
|
2020-06-09 00:49:55 +00:00
|
|
|
pipeline_flags |= PIPELINE_VIGNETTE;
|
2021-09-17 21:01:37 +00:00
|
|
|
if (self->gradient)
|
2020-06-09 00:49:55 +00:00
|
|
|
pipeline_flags |= PIPELINE_GRADIENT;
|
2021-09-17 21:01:37 +00:00
|
|
|
if (self->has_rounded_clip)
|
2021-02-08 08:33:55 +00:00
|
|
|
pipeline_flags |= PIPELINE_ROUNDED_CLIP | PIPELINE_BLEND;
|
2020-06-09 00:49:55 +00:00
|
|
|
|
|
|
|
if (pipeline_flags != self->pipeline_flags)
|
|
|
|
g_clear_pointer (&self->pipeline, cogl_object_unref);
|
|
|
|
|
|
|
|
if (self->pipeline == NULL)
|
|
|
|
{
|
|
|
|
self->pipeline_flags = pipeline_flags;
|
|
|
|
self->pipeline = make_pipeline (pipeline_flags);
|
|
|
|
self->changed = CHANGED_ALL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (self->changed & CHANGED_BACKGROUND)
|
|
|
|
{
|
|
|
|
CoglPipelineWrapMode wrap_mode;
|
|
|
|
CoglTexture *texture = meta_background_get_texture (self->background,
|
|
|
|
self->monitor,
|
|
|
|
&self->texture_area,
|
|
|
|
&wrap_mode);
|
2020-07-01 10:09:14 +00:00
|
|
|
|
|
|
|
if (texture)
|
|
|
|
{
|
|
|
|
self->texture_width = cogl_texture_get_width (texture);
|
|
|
|
self->texture_height = cogl_texture_get_height (texture);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
self->texture_width = 0;
|
|
|
|
self->texture_height = 0;
|
|
|
|
}
|
2020-06-09 00:49:55 +00:00
|
|
|
|
|
|
|
cogl_pipeline_set_layer_texture (self->pipeline, 0, texture);
|
|
|
|
cogl_pipeline_set_layer_wrap_mode (self->pipeline, 0, wrap_mode);
|
|
|
|
|
|
|
|
self->changed &= ~CHANGED_BACKGROUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (self->changed & CHANGED_VIGNETTE_PARAMETERS)
|
|
|
|
{
|
|
|
|
cogl_pipeline_set_uniform_1f (self->pipeline,
|
|
|
|
cogl_pipeline_get_uniform_location (self->pipeline,
|
|
|
|
"vignette_sharpness"),
|
|
|
|
self->vignette_sharpness);
|
|
|
|
|
|
|
|
self->changed &= ~CHANGED_VIGNETTE_PARAMETERS;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (self->changed & CHANGED_GRADIENT_PARAMETERS)
|
|
|
|
{
|
|
|
|
MetaRectangle monitor_geometry;
|
|
|
|
float gradient_height_perc;
|
|
|
|
|
|
|
|
meta_display_get_monitor_geometry (self->display,
|
|
|
|
self->monitor, &monitor_geometry);
|
|
|
|
gradient_height_perc = MAX (0.0001, self->gradient_height / (float)monitor_geometry.height);
|
|
|
|
cogl_pipeline_set_uniform_1f (self->pipeline,
|
|
|
|
cogl_pipeline_get_uniform_location (self->pipeline,
|
|
|
|
"gradient_height_perc"),
|
|
|
|
gradient_height_perc);
|
|
|
|
cogl_pipeline_set_uniform_1f (self->pipeline,
|
|
|
|
cogl_pipeline_get_uniform_location (self->pipeline,
|
|
|
|
"gradient_max_darkness"),
|
|
|
|
self->gradient_max_darkness);
|
|
|
|
|
|
|
|
self->changed &= ~CHANGED_GRADIENT_PARAMETERS;
|
|
|
|
}
|
|
|
|
|
2021-02-08 08:33:55 +00:00
|
|
|
if (self->changed & CHANGED_ROUNDED_CLIP_PARAMETERS)
|
|
|
|
{
|
|
|
|
float monitor_scale;
|
|
|
|
float bounds_x1, bounds_x2, bounds_y1, bounds_y2;
|
|
|
|
float clip_radius;
|
|
|
|
|
|
|
|
monitor_scale = meta_is_stage_views_scaled ()
|
|
|
|
? meta_display_get_monitor_scale (self->display, self->monitor)
|
|
|
|
: 1.0;
|
|
|
|
|
|
|
|
if (self->rounded_clip_bounds_set)
|
|
|
|
{
|
|
|
|
bounds_x1 = self->rounded_clip_bounds.origin.x;
|
|
|
|
bounds_x2 = self->rounded_clip_bounds.origin.x + self->rounded_clip_bounds.size.width;
|
|
|
|
bounds_y1 = self->rounded_clip_bounds.origin.y;
|
|
|
|
bounds_y2 = self->rounded_clip_bounds.origin.y + self->rounded_clip_bounds.size.height;
|
|
|
|
|
|
|
|
bounds_x1 *= monitor_scale;
|
|
|
|
bounds_x2 *= monitor_scale;
|
|
|
|
bounds_y1 *= monitor_scale;
|
|
|
|
bounds_y2 *= monitor_scale;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
bounds_x1 = 0.0;
|
|
|
|
bounds_x2 = self->texture_width;
|
|
|
|
bounds_y1 = 0.0;
|
|
|
|
bounds_y2 = self->texture_height;
|
|
|
|
}
|
|
|
|
|
|
|
|
clip_radius = self->rounded_clip_radius * monitor_scale;
|
|
|
|
|
|
|
|
float bounds[] = {
|
|
|
|
bounds_x1,
|
|
|
|
bounds_y1,
|
|
|
|
bounds_x2,
|
|
|
|
bounds_y2,
|
|
|
|
};
|
|
|
|
|
|
|
|
int bounds_uniform_location;
|
2021-05-10 09:18:55 +00:00
|
|
|
int clip_radius_uniform_location;
|
2021-02-08 08:33:55 +00:00
|
|
|
|
|
|
|
bounds_uniform_location =
|
|
|
|
cogl_pipeline_get_uniform_location (self->pipeline, "bounds");
|
2021-05-10 09:18:55 +00:00
|
|
|
clip_radius_uniform_location =
|
|
|
|
cogl_pipeline_get_uniform_location (self->pipeline, "clip_radius");
|
2021-02-08 08:33:55 +00:00
|
|
|
|
|
|
|
cogl_pipeline_set_uniform_float (self->pipeline,
|
|
|
|
bounds_uniform_location,
|
|
|
|
4, 1,
|
|
|
|
bounds);
|
|
|
|
|
2021-05-10 09:18:55 +00:00
|
|
|
cogl_pipeline_set_uniform_1f (self->pipeline,
|
|
|
|
clip_radius_uniform_location,
|
|
|
|
clip_radius);
|
2021-02-08 08:33:55 +00:00
|
|
|
|
|
|
|
self->changed &= ~CHANGED_ROUNDED_CLIP_PARAMETERS;
|
|
|
|
}
|
|
|
|
|
2020-06-09 00:49:55 +00:00
|
|
|
if (self->vignette)
|
2021-09-17 21:01:37 +00:00
|
|
|
color_component = self->vignette_brightness * opacity / 255.;
|
2020-06-09 00:49:55 +00:00
|
|
|
else
|
2021-09-17 21:01:37 +00:00
|
|
|
color_component = opacity / 255.;
|
2020-06-09 00:49:55 +00:00
|
|
|
|
|
|
|
cogl_pipeline_set_color4f (self->pipeline,
|
|
|
|
color_component,
|
|
|
|
color_component,
|
|
|
|
color_component,
|
|
|
|
opacity / 255.);
|
|
|
|
|
|
|
|
fb = clutter_paint_context_get_framebuffer (paint_context);
|
2020-07-01 10:09:14 +00:00
|
|
|
if (meta_actor_painting_untransformed (fb,
|
2020-06-09 00:49:55 +00:00
|
|
|
actor_pixel_rect->width,
|
|
|
|
actor_pixel_rect->height,
|
2020-07-01 10:09:14 +00:00
|
|
|
self->texture_width,
|
|
|
|
self->texture_height,
|
2021-06-01 15:50:07 +00:00
|
|
|
NULL))
|
2020-07-01 08:43:50 +00:00
|
|
|
{
|
|
|
|
min_filter = COGL_PIPELINE_FILTER_NEAREST;
|
|
|
|
mag_filter = COGL_PIPELINE_FILTER_NEAREST;
|
|
|
|
}
|
2020-06-09 00:49:55 +00:00
|
|
|
else
|
2020-07-01 08:43:50 +00:00
|
|
|
{
|
|
|
|
min_filter = COGL_PIPELINE_FILTER_LINEAR_MIPMAP_NEAREST;
|
|
|
|
mag_filter = COGL_PIPELINE_FILTER_LINEAR;
|
|
|
|
}
|
2020-06-09 00:49:55 +00:00
|
|
|
|
2020-07-01 08:43:50 +00:00
|
|
|
cogl_pipeline_set_layer_filters (self->pipeline, 0, min_filter, mag_filter);
|
2020-06-09 00:49:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_glsl_parameters (MetaBackgroundContent *self,
|
|
|
|
cairo_rectangle_int_t *actor_pixel_rect)
|
|
|
|
{
|
2021-02-08 08:33:55 +00:00
|
|
|
float monitor_scale;
|
2020-06-09 00:49:55 +00:00
|
|
|
float scale[2];
|
|
|
|
float offset[2];
|
2021-02-08 08:33:55 +00:00
|
|
|
int pixel_step_uniform_location;
|
|
|
|
|
|
|
|
monitor_scale = meta_is_stage_views_scaled ()
|
|
|
|
? meta_display_get_monitor_scale (self->display, self->monitor)
|
|
|
|
: 1.0;
|
|
|
|
|
|
|
|
float pixel_step[] = {
|
|
|
|
1.f / (self->texture_area.width * monitor_scale),
|
|
|
|
1.f / (self->texture_area.height * monitor_scale),
|
|
|
|
};
|
|
|
|
|
|
|
|
pixel_step_uniform_location =
|
|
|
|
cogl_pipeline_get_uniform_location (self->pipeline,
|
|
|
|
"pixel_step");
|
2020-06-09 00:49:55 +00:00
|
|
|
|
|
|
|
/* Compute a scale and offset for transforming texture coordinates to the
|
|
|
|
* coordinate system from [-0.5 to 0.5] across the area of the actor
|
|
|
|
*/
|
|
|
|
scale[0] = self->texture_area.width / (float)actor_pixel_rect->width;
|
|
|
|
scale[1] = self->texture_area.height / (float)actor_pixel_rect->height;
|
|
|
|
offset[0] = self->texture_area.x / (float)actor_pixel_rect->width - 0.5;
|
|
|
|
offset[1] = self->texture_area.y / (float)actor_pixel_rect->height - 0.5;
|
|
|
|
|
|
|
|
cogl_pipeline_set_uniform_float (self->pipeline,
|
|
|
|
cogl_pipeline_get_uniform_location (self->pipeline,
|
|
|
|
"scale"),
|
|
|
|
2, 1, scale);
|
|
|
|
|
|
|
|
cogl_pipeline_set_uniform_float (self->pipeline,
|
|
|
|
cogl_pipeline_get_uniform_location (self->pipeline,
|
|
|
|
"offset"),
|
|
|
|
2, 1, offset);
|
2021-02-08 08:33:55 +00:00
|
|
|
|
|
|
|
cogl_pipeline_set_uniform_float (self->pipeline,
|
|
|
|
pixel_step_uniform_location,
|
|
|
|
2, 1,
|
|
|
|
pixel_step);
|
2020-06-09 00:49:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2020-06-09 16:02:50 +00:00
|
|
|
paint_clipped_rectangle (MetaBackgroundContent *self,
|
|
|
|
ClutterPaintNode *node,
|
2020-06-09 16:52:14 +00:00
|
|
|
ClutterActorBox *actor_box,
|
2020-06-09 16:02:50 +00:00
|
|
|
cairo_rectangle_int_t *rect)
|
2020-06-09 00:49:55 +00:00
|
|
|
{
|
|
|
|
g_autoptr (ClutterPaintNode) pipeline_node = NULL;
|
2020-06-09 16:52:14 +00:00
|
|
|
float h_scale, v_scale;
|
2020-06-09 00:49:55 +00:00
|
|
|
float x1, y1, x2, y2;
|
|
|
|
float tx1, ty1, tx2, ty2;
|
|
|
|
|
2020-06-09 16:52:14 +00:00
|
|
|
h_scale = self->texture_area.width / clutter_actor_box_get_width (actor_box);
|
|
|
|
v_scale = self->texture_area.height / clutter_actor_box_get_height (actor_box);
|
|
|
|
|
2020-06-09 00:49:55 +00:00
|
|
|
x1 = rect->x;
|
|
|
|
y1 = rect->y;
|
|
|
|
x2 = rect->x + rect->width;
|
|
|
|
y2 = rect->y + rect->height;
|
|
|
|
|
2020-06-09 16:52:14 +00:00
|
|
|
tx1 = (x1 * h_scale - self->texture_area.x) / (float)self->texture_area.width;
|
|
|
|
ty1 = (y1 * v_scale - self->texture_area.y) / (float)self->texture_area.height;
|
|
|
|
tx2 = (x2 * h_scale - self->texture_area.x) / (float)self->texture_area.width;
|
|
|
|
ty2 = (y2 * v_scale - self->texture_area.y) / (float)self->texture_area.height;
|
2020-06-09 00:49:55 +00:00
|
|
|
|
2020-06-09 16:02:50 +00:00
|
|
|
pipeline_node = clutter_pipeline_node_new (self->pipeline);
|
2020-06-09 00:49:55 +00:00
|
|
|
clutter_paint_node_set_name (pipeline_node, "MetaBackgroundContent (Slice)");
|
|
|
|
clutter_paint_node_add_texture_rectangle (pipeline_node,
|
|
|
|
&(ClutterActorBox) {
|
|
|
|
.x1 = x1,
|
|
|
|
.y1 = y1,
|
|
|
|
.x2 = x2,
|
|
|
|
.y2 = y2,
|
|
|
|
},
|
|
|
|
tx1, ty1,
|
|
|
|
tx2, ty2);
|
|
|
|
|
|
|
|
clutter_paint_node_add_child (node, pipeline_node);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_background_content_paint_content (ClutterContent *content,
|
|
|
|
ClutterActor *actor,
|
|
|
|
ClutterPaintNode *node,
|
|
|
|
ClutterPaintContext *paint_context)
|
|
|
|
{
|
|
|
|
MetaBackgroundContent *self = META_BACKGROUND_CONTENT (content);
|
|
|
|
ClutterActorBox actor_box;
|
2020-07-09 08:52:07 +00:00
|
|
|
cairo_rectangle_int_t rect_within_actor;
|
|
|
|
cairo_rectangle_int_t rect_within_stage;
|
2020-06-09 00:49:55 +00:00
|
|
|
cairo_region_t *region;
|
|
|
|
int i, n_rects;
|
2020-07-09 08:52:07 +00:00
|
|
|
float transformed_x, transformed_y, transformed_width, transformed_height;
|
|
|
|
gboolean untransformed;
|
2020-06-09 00:49:55 +00:00
|
|
|
|
|
|
|
if ((self->clip_region && cairo_region_is_empty (self->clip_region)))
|
|
|
|
return;
|
|
|
|
|
2020-07-09 08:52:07 +00:00
|
|
|
clutter_actor_get_content_box (actor, &actor_box);
|
|
|
|
rect_within_actor.x = actor_box.x1;
|
|
|
|
rect_within_actor.y = actor_box.y1;
|
|
|
|
rect_within_actor.width = actor_box.x2 - actor_box.x1;
|
|
|
|
rect_within_actor.height = actor_box.y2 - actor_box.y1;
|
|
|
|
|
2020-10-20 09:49:26 +00:00
|
|
|
if (clutter_actor_is_in_clone_paint (actor))
|
|
|
|
{
|
|
|
|
untransformed = FALSE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
clutter_actor_get_transformed_position (actor,
|
|
|
|
&transformed_x,
|
|
|
|
&transformed_y);
|
|
|
|
rect_within_stage.x = floorf (transformed_x);
|
|
|
|
rect_within_stage.y = floorf (transformed_y);
|
|
|
|
|
|
|
|
clutter_actor_get_transformed_size (actor,
|
|
|
|
&transformed_width,
|
|
|
|
&transformed_height);
|
2020-10-27 08:54:50 +00:00
|
|
|
rect_within_stage.width = ceilf (transformed_width);
|
|
|
|
rect_within_stage.height = ceilf (transformed_height);
|
2020-10-20 09:49:26 +00:00
|
|
|
|
|
|
|
untransformed =
|
|
|
|
rect_within_actor.x == rect_within_stage.x &&
|
|
|
|
rect_within_actor.y == rect_within_stage.y &&
|
|
|
|
rect_within_actor.width == rect_within_stage.width &&
|
|
|
|
rect_within_actor.height == rect_within_stage.height;
|
|
|
|
}
|
2020-07-09 08:52:07 +00:00
|
|
|
|
|
|
|
if (untransformed) /* actor and stage space are the same */
|
2020-06-09 00:49:55 +00:00
|
|
|
{
|
2020-07-09 08:52:07 +00:00
|
|
|
if (self->clip_region)
|
|
|
|
{
|
|
|
|
region = cairo_region_copy (self->clip_region);
|
|
|
|
cairo_region_intersect_rectangle (region, &rect_within_stage);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-07-09 09:48:32 +00:00
|
|
|
const cairo_region_t *redraw_clip;
|
|
|
|
|
|
|
|
redraw_clip = clutter_paint_context_get_redraw_clip (paint_context);
|
|
|
|
if (redraw_clip)
|
|
|
|
{
|
|
|
|
region = cairo_region_copy (redraw_clip);
|
|
|
|
cairo_region_intersect_rectangle (region, &rect_within_stage);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
region = cairo_region_create_rectangle (&rect_within_stage);
|
|
|
|
}
|
2020-07-09 08:52:07 +00:00
|
|
|
}
|
2020-06-09 00:49:55 +00:00
|
|
|
}
|
2020-07-09 08:52:07 +00:00
|
|
|
else /* actor and stage space are different but we need actor space */
|
2020-06-09 00:49:55 +00:00
|
|
|
{
|
2020-10-01 10:01:46 +00:00
|
|
|
if (self->clip_region)
|
|
|
|
{
|
|
|
|
region = cairo_region_copy (self->clip_region);
|
|
|
|
cairo_region_intersect_rectangle (region, &rect_within_actor);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
region = cairo_region_create_rectangle (&rect_within_actor);
|
|
|
|
}
|
2020-06-09 00:49:55 +00:00
|
|
|
}
|
|
|
|
|
2020-10-01 10:01:46 +00:00
|
|
|
if (self->unobscured_region)
|
|
|
|
cairo_region_intersect (region, self->unobscured_region);
|
|
|
|
|
2020-07-09 08:52:07 +00:00
|
|
|
/* region is now in actor space */
|
2020-06-09 00:49:55 +00:00
|
|
|
if (cairo_region_is_empty (region))
|
|
|
|
{
|
|
|
|
cairo_region_destroy (region);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-07-09 08:52:07 +00:00
|
|
|
setup_pipeline (self, actor, paint_context, &rect_within_actor);
|
|
|
|
set_glsl_parameters (self, &rect_within_actor);
|
2020-07-09 08:19:29 +00:00
|
|
|
|
|
|
|
/* Limit to how many separate rectangles we'll draw; beyond this just
|
|
|
|
* fall back and draw the whole thing */
|
|
|
|
#define MAX_RECTS 64
|
|
|
|
|
2020-06-09 00:49:55 +00:00
|
|
|
n_rects = cairo_region_num_rectangles (region);
|
|
|
|
if (n_rects <= MAX_RECTS)
|
|
|
|
{
|
|
|
|
for (i = 0; i < n_rects; i++)
|
|
|
|
{
|
|
|
|
cairo_rectangle_int_t rect;
|
|
|
|
cairo_region_get_rectangle (region, i, &rect);
|
2020-06-09 16:52:14 +00:00
|
|
|
paint_clipped_rectangle (self, node, &actor_box, &rect);
|
2020-06-09 00:49:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
cairo_rectangle_int_t rect;
|
|
|
|
cairo_region_get_extents (region, &rect);
|
2020-06-09 16:52:14 +00:00
|
|
|
paint_clipped_rectangle (self, node, &actor_box, &rect);
|
2020-06-09 00:49:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
cairo_region_destroy (region);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
meta_background_content_get_preferred_size (ClutterContent *content,
|
|
|
|
float *width,
|
|
|
|
float *height)
|
|
|
|
|
|
|
|
{
|
|
|
|
MetaBackgroundContent *background_content = META_BACKGROUND_CONTENT (content);
|
|
|
|
MetaRectangle monitor_geometry;
|
|
|
|
|
|
|
|
meta_display_get_monitor_geometry (background_content->display,
|
|
|
|
background_content->monitor,
|
|
|
|
&monitor_geometry);
|
|
|
|
|
|
|
|
if (width)
|
|
|
|
*width = monitor_geometry.width;
|
|
|
|
|
|
|
|
if (height)
|
|
|
|
*height = monitor_geometry.height;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
clutter_content_iface_init (ClutterContentInterface *iface)
|
|
|
|
{
|
|
|
|
iface->paint_content = meta_background_content_paint_content;
|
|
|
|
iface->get_preferred_size = meta_background_content_get_preferred_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_monitor (MetaBackgroundContent *self,
|
|
|
|
int monitor)
|
|
|
|
{
|
|
|
|
MetaRectangle old_monitor_geometry;
|
|
|
|
MetaRectangle new_monitor_geometry;
|
|
|
|
MetaDisplay *display = self->display;
|
|
|
|
|
|
|
|
if(self->monitor == monitor)
|
|
|
|
return;
|
|
|
|
|
|
|
|
meta_display_get_monitor_geometry (display, self->monitor, &old_monitor_geometry);
|
|
|
|
meta_display_get_monitor_geometry (display, monitor, &new_monitor_geometry);
|
|
|
|
if(old_monitor_geometry.height != new_monitor_geometry.height)
|
|
|
|
invalidate_pipeline (self, CHANGED_GRADIENT_PARAMETERS);
|
|
|
|
|
|
|
|
self->monitor = monitor;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_background_content_dispose (GObject *object)
|
|
|
|
{
|
|
|
|
MetaBackgroundContent *self = META_BACKGROUND_CONTENT (object);
|
|
|
|
|
|
|
|
set_clip_region (self, NULL);
|
|
|
|
set_unobscured_region (self, NULL);
|
|
|
|
meta_background_content_set_background (self, NULL);
|
|
|
|
|
|
|
|
g_clear_pointer (&self->pipeline, cogl_object_unref);
|
|
|
|
|
|
|
|
G_OBJECT_CLASS (meta_background_content_parent_class)->dispose (object);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_background_content_set_property (GObject *object,
|
|
|
|
guint prop_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
MetaBackgroundContent *self = META_BACKGROUND_CONTENT (object);
|
|
|
|
|
|
|
|
switch (prop_id)
|
|
|
|
{
|
|
|
|
case PROP_META_DISPLAY:
|
|
|
|
self->display = g_value_get_object (value);
|
|
|
|
break;
|
|
|
|
case PROP_MONITOR:
|
|
|
|
set_monitor (self, g_value_get_int (value));
|
|
|
|
break;
|
|
|
|
case PROP_BACKGROUND:
|
|
|
|
meta_background_content_set_background (self, g_value_get_object (value));
|
|
|
|
break;
|
|
|
|
case PROP_GRADIENT:
|
|
|
|
meta_background_content_set_gradient (self,
|
|
|
|
g_value_get_boolean (value),
|
|
|
|
self->gradient_height,
|
|
|
|
self->gradient_max_darkness);
|
|
|
|
break;
|
|
|
|
case PROP_GRADIENT_HEIGHT:
|
|
|
|
meta_background_content_set_gradient (self,
|
|
|
|
self->gradient,
|
|
|
|
g_value_get_int (value),
|
|
|
|
self->gradient_max_darkness);
|
|
|
|
break;
|
|
|
|
case PROP_GRADIENT_MAX_DARKNESS:
|
|
|
|
meta_background_content_set_gradient (self,
|
|
|
|
self->gradient,
|
|
|
|
self->gradient_height,
|
|
|
|
g_value_get_double (value));
|
|
|
|
break;
|
|
|
|
case PROP_VIGNETTE:
|
|
|
|
meta_background_content_set_vignette (self,
|
|
|
|
g_value_get_boolean (value),
|
|
|
|
self->vignette_brightness,
|
|
|
|
self->vignette_sharpness);
|
|
|
|
break;
|
|
|
|
case PROP_VIGNETTE_SHARPNESS:
|
|
|
|
meta_background_content_set_vignette (self,
|
|
|
|
self->vignette,
|
|
|
|
self->vignette_brightness,
|
|
|
|
g_value_get_double (value));
|
|
|
|
break;
|
|
|
|
case PROP_VIGNETTE_BRIGHTNESS:
|
|
|
|
meta_background_content_set_vignette (self,
|
|
|
|
self->vignette,
|
|
|
|
g_value_get_double (value),
|
|
|
|
self->vignette_sharpness);
|
|
|
|
break;
|
2021-02-08 08:33:55 +00:00
|
|
|
case PROP_ROUNDED_CLIP_RADIUS:
|
|
|
|
meta_background_content_set_rounded_clip_radius (self,
|
|
|
|
g_value_get_float (value));
|
|
|
|
break;
|
2020-06-09 00:49:55 +00:00
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_background_content_get_property (GObject *object,
|
|
|
|
guint prop_id,
|
|
|
|
GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
MetaBackgroundContent *self = META_BACKGROUND_CONTENT (object);
|
|
|
|
|
|
|
|
switch (prop_id)
|
|
|
|
{
|
|
|
|
case PROP_META_DISPLAY:
|
|
|
|
g_value_set_object (value, self->display);
|
|
|
|
break;
|
|
|
|
case PROP_MONITOR:
|
|
|
|
g_value_set_int (value, self->monitor);
|
|
|
|
break;
|
|
|
|
case PROP_BACKGROUND:
|
|
|
|
g_value_set_object (value, self->background);
|
|
|
|
break;
|
|
|
|
case PROP_GRADIENT:
|
|
|
|
g_value_set_boolean (value, self->gradient);
|
|
|
|
break;
|
|
|
|
case PROP_GRADIENT_HEIGHT:
|
|
|
|
g_value_set_int (value, self->gradient_height);
|
|
|
|
break;
|
|
|
|
case PROP_GRADIENT_MAX_DARKNESS:
|
|
|
|
g_value_set_double (value, self->gradient_max_darkness);
|
|
|
|
break;
|
|
|
|
case PROP_VIGNETTE:
|
|
|
|
g_value_set_boolean (value, self->vignette);
|
|
|
|
break;
|
|
|
|
case PROP_VIGNETTE_BRIGHTNESS:
|
|
|
|
g_value_set_double (value, self->vignette_brightness);
|
|
|
|
break;
|
|
|
|
case PROP_VIGNETTE_SHARPNESS:
|
|
|
|
g_value_set_double (value, self->vignette_sharpness);
|
|
|
|
break;
|
2021-02-08 08:33:55 +00:00
|
|
|
case PROP_ROUNDED_CLIP_RADIUS:
|
|
|
|
g_value_set_float (value, self->rounded_clip_radius);
|
|
|
|
break;
|
2020-06-09 00:49:55 +00:00
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_background_content_class_init (MetaBackgroundContentClass *klass)
|
|
|
|
{
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
|
|
|
object_class->dispose = meta_background_content_dispose;
|
|
|
|
object_class->set_property = meta_background_content_set_property;
|
|
|
|
object_class->get_property = meta_background_content_get_property;
|
|
|
|
|
|
|
|
properties[PROP_META_DISPLAY] =
|
|
|
|
g_param_spec_object ("meta-display",
|
|
|
|
"MetaDisplay",
|
|
|
|
"MetaDisplay",
|
|
|
|
META_TYPE_DISPLAY,
|
|
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY);
|
|
|
|
|
|
|
|
properties[PROP_MONITOR] =
|
|
|
|
g_param_spec_int ("monitor",
|
|
|
|
"monitor",
|
|
|
|
"monitor",
|
|
|
|
0, G_MAXINT, 0,
|
|
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY);
|
|
|
|
|
|
|
|
properties[PROP_BACKGROUND] =
|
|
|
|
g_param_spec_object ("background",
|
|
|
|
"Background",
|
|
|
|
"MetaBackground object holding background parameters",
|
|
|
|
META_TYPE_BACKGROUND,
|
|
|
|
G_PARAM_READWRITE);
|
|
|
|
|
|
|
|
properties[PROP_GRADIENT] =
|
|
|
|
g_param_spec_boolean ("gradient",
|
|
|
|
"Gradient",
|
|
|
|
"Whether gradient effect is enabled",
|
|
|
|
FALSE,
|
|
|
|
G_PARAM_READWRITE);
|
|
|
|
|
|
|
|
properties[PROP_GRADIENT_HEIGHT] =
|
|
|
|
g_param_spec_int ("gradient-height",
|
|
|
|
"Gradient Height",
|
|
|
|
"Height of gradient effect",
|
|
|
|
0, G_MAXINT, 0,
|
|
|
|
G_PARAM_READWRITE);
|
|
|
|
|
|
|
|
properties[PROP_GRADIENT_MAX_DARKNESS] =
|
|
|
|
g_param_spec_double ("gradient-max-darkness",
|
|
|
|
"Gradient Max Darkness",
|
|
|
|
"How dark is the gradient initially",
|
|
|
|
0.0, 1.0, 0.0,
|
|
|
|
G_PARAM_READWRITE);
|
|
|
|
|
|
|
|
properties[PROP_VIGNETTE] =
|
|
|
|
g_param_spec_boolean ("vignette",
|
|
|
|
"Vignette",
|
|
|
|
"Whether vignette effect is enabled",
|
|
|
|
FALSE,
|
|
|
|
G_PARAM_READWRITE);
|
|
|
|
|
|
|
|
properties[PROP_VIGNETTE_BRIGHTNESS] =
|
|
|
|
g_param_spec_double ("brightness",
|
|
|
|
"Vignette Brightness",
|
|
|
|
"Brightness of vignette effect",
|
|
|
|
0.0, 1.0, 1.0,
|
|
|
|
G_PARAM_READWRITE);
|
|
|
|
|
|
|
|
properties[PROP_VIGNETTE_SHARPNESS] =
|
|
|
|
g_param_spec_double ("vignette-sharpness",
|
|
|
|
"Vignette Sharpness",
|
|
|
|
"Sharpness of vignette effect",
|
|
|
|
0.0, G_MAXDOUBLE, 0.0,
|
|
|
|
G_PARAM_READWRITE);
|
|
|
|
|
2021-02-08 08:33:55 +00:00
|
|
|
properties[PROP_ROUNDED_CLIP_RADIUS] =
|
|
|
|
g_param_spec_float ("rounded-clip-radius",
|
|
|
|
"Rounded clip radius",
|
|
|
|
"Rounded clip radius",
|
|
|
|
0.0, G_MAXFLOAT, 0.0,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_STATIC_STRINGS |
|
|
|
|
G_PARAM_EXPLICIT_NOTIFY);
|
|
|
|
|
2020-06-09 00:49:55 +00:00
|
|
|
g_object_class_install_properties (object_class, N_PROPS, properties);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_background_content_init (MetaBackgroundContent *self)
|
|
|
|
{
|
|
|
|
self->gradient = FALSE;
|
|
|
|
self->gradient_height = 0;
|
|
|
|
self->gradient_max_darkness = 0.0;
|
|
|
|
|
|
|
|
self->vignette = FALSE;
|
|
|
|
self->vignette_brightness = 1.0;
|
|
|
|
self->vignette_sharpness = 0.0;
|
2021-02-08 08:33:55 +00:00
|
|
|
|
|
|
|
self->has_rounded_clip = FALSE;
|
|
|
|
self->rounded_clip_radius = 0.0;
|
2020-06-09 00:49:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* meta_background_content_new:
|
|
|
|
* @monitor: Index of the monitor for which to draw the background
|
|
|
|
*
|
|
|
|
* Creates a new actor to draw the background for the given monitor.
|
|
|
|
*
|
2020-06-10 08:40:45 +00:00
|
|
|
* Return value: (transfer full): the newly created background actor
|
2020-06-09 00:49:55 +00:00
|
|
|
*/
|
|
|
|
ClutterContent *
|
|
|
|
meta_background_content_new (MetaDisplay *display,
|
|
|
|
int monitor)
|
|
|
|
{
|
|
|
|
return g_object_new (META_TYPE_BACKGROUND_CONTENT,
|
|
|
|
"meta-display", display,
|
|
|
|
"monitor", monitor,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_background_content_set_background (MetaBackgroundContent *self,
|
|
|
|
MetaBackground *background)
|
|
|
|
{
|
|
|
|
g_return_if_fail (META_IS_BACKGROUND_CONTENT (self));
|
|
|
|
g_return_if_fail (background == NULL || META_IS_BACKGROUND (background));
|
|
|
|
|
|
|
|
if (background == self->background)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (self->background)
|
|
|
|
{
|
|
|
|
g_signal_handlers_disconnect_by_func (self->background,
|
|
|
|
(gpointer)on_background_changed,
|
|
|
|
self);
|
|
|
|
g_clear_object (&self->background);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (background)
|
|
|
|
{
|
|
|
|
self->background = g_object_ref (background);
|
|
|
|
g_signal_connect (self->background, "changed",
|
|
|
|
G_CALLBACK (on_background_changed), self);
|
|
|
|
}
|
|
|
|
|
|
|
|
invalidate_pipeline (self, CHANGED_BACKGROUND);
|
|
|
|
clutter_content_invalidate (CLUTTER_CONTENT (self));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_background_content_set_gradient (MetaBackgroundContent *self,
|
|
|
|
gboolean enabled,
|
|
|
|
int height,
|
|
|
|
double max_darkness)
|
|
|
|
{
|
|
|
|
gboolean changed = FALSE;
|
|
|
|
|
|
|
|
g_return_if_fail (META_IS_BACKGROUND_CONTENT (self));
|
|
|
|
g_return_if_fail (height >= 0);
|
|
|
|
g_return_if_fail (max_darkness >= 0. && max_darkness <= 1.);
|
|
|
|
|
|
|
|
enabled = enabled != FALSE && height != 0;
|
|
|
|
|
|
|
|
if (enabled != self->gradient)
|
|
|
|
{
|
|
|
|
self->gradient = enabled;
|
|
|
|
invalidate_pipeline (self, CHANGED_EFFECTS);
|
|
|
|
changed = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (height != self->gradient_height || max_darkness != self->gradient_max_darkness)
|
|
|
|
{
|
|
|
|
self->gradient_height = height;
|
|
|
|
self->gradient_max_darkness = max_darkness;
|
|
|
|
invalidate_pipeline (self, CHANGED_GRADIENT_PARAMETERS);
|
|
|
|
changed = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (changed)
|
|
|
|
clutter_content_invalidate (CLUTTER_CONTENT (self));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_background_content_set_vignette (MetaBackgroundContent *self,
|
|
|
|
gboolean enabled,
|
|
|
|
double brightness,
|
|
|
|
double sharpness)
|
|
|
|
{
|
|
|
|
gboolean changed = FALSE;
|
|
|
|
|
|
|
|
g_return_if_fail (META_IS_BACKGROUND_CONTENT (self));
|
|
|
|
g_return_if_fail (brightness >= 0. && brightness <= 1.);
|
|
|
|
g_return_if_fail (sharpness >= 0.);
|
|
|
|
|
|
|
|
enabled = enabled != FALSE;
|
|
|
|
|
|
|
|
if (enabled != self->vignette)
|
|
|
|
{
|
|
|
|
self->vignette = enabled;
|
|
|
|
invalidate_pipeline (self, CHANGED_EFFECTS);
|
|
|
|
changed = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (brightness != self->vignette_brightness || sharpness != self->vignette_sharpness)
|
|
|
|
{
|
|
|
|
self->vignette_brightness = brightness;
|
|
|
|
self->vignette_sharpness = sharpness;
|
|
|
|
invalidate_pipeline (self, CHANGED_VIGNETTE_PARAMETERS);
|
|
|
|
changed = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (changed)
|
|
|
|
clutter_content_invalidate (CLUTTER_CONTENT (self));
|
|
|
|
}
|
|
|
|
|
2021-02-08 08:33:55 +00:00
|
|
|
void
|
|
|
|
meta_background_content_set_rounded_clip_radius (MetaBackgroundContent *self,
|
|
|
|
float radius)
|
|
|
|
{
|
|
|
|
gboolean enabled;
|
|
|
|
gboolean changed = FALSE;
|
|
|
|
|
|
|
|
g_return_if_fail (META_IS_BACKGROUND_CONTENT (self));
|
|
|
|
g_return_if_fail (radius >= 0.0);
|
|
|
|
|
|
|
|
enabled = radius > 0.0;
|
|
|
|
|
|
|
|
if (enabled != self->has_rounded_clip)
|
|
|
|
{
|
|
|
|
self->has_rounded_clip = enabled;
|
|
|
|
invalidate_pipeline (self, CHANGED_EFFECTS);
|
|
|
|
changed = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!G_APPROX_VALUE (radius, self->rounded_clip_radius, FLT_EPSILON))
|
|
|
|
{
|
|
|
|
self->rounded_clip_radius = radius;
|
|
|
|
invalidate_pipeline (self, CHANGED_ROUNDED_CLIP_PARAMETERS);
|
|
|
|
changed = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (changed)
|
|
|
|
{
|
|
|
|
clutter_content_invalidate (CLUTTER_CONTENT (self));
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_ROUNDED_CLIP_RADIUS]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* meta_background_content_set_rounded_clip_bounds:
|
|
|
|
* @self: The #MetaBackgroundContent
|
|
|
|
* @bounds: (allow-none): The new bounding clip rectangle, or %NULL
|
|
|
|
*
|
|
|
|
* Sets the bounding clip rectangle of the #MetaBackgroundContent that's used
|
|
|
|
* when a rounded clip set via meta_background_content_set_rounded_clip_radius()
|
|
|
|
* is in effect, set it to %NULL to use no bounding clip, rounding the edges
|
|
|
|
* of the full texture.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
meta_background_content_set_rounded_clip_bounds (MetaBackgroundContent *self,
|
|
|
|
const graphene_rect_t *bounds)
|
|
|
|
{
|
|
|
|
g_return_if_fail (META_IS_BACKGROUND_CONTENT (self));
|
|
|
|
|
|
|
|
if (bounds == NULL)
|
|
|
|
{
|
|
|
|
if (!self->rounded_clip_bounds_set)
|
|
|
|
return;
|
|
|
|
|
|
|
|
self->rounded_clip_bounds_set = FALSE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (self->rounded_clip_bounds_set &&
|
|
|
|
graphene_rect_equal (&self->rounded_clip_bounds, bounds))
|
|
|
|
return;
|
|
|
|
|
|
|
|
self->rounded_clip_bounds_set = TRUE;
|
|
|
|
graphene_rect_init_from_rect (&self->rounded_clip_bounds, bounds);
|
|
|
|
}
|
|
|
|
|
|
|
|
invalidate_pipeline (self, CHANGED_ROUNDED_CLIP_PARAMETERS);
|
|
|
|
clutter_content_invalidate (CLUTTER_CONTENT (self));
|
|
|
|
}
|
|
|
|
|
2020-06-09 00:49:55 +00:00
|
|
|
cairo_region_t *
|
|
|
|
meta_background_content_get_clip_region (MetaBackgroundContent *self)
|
|
|
|
{
|
|
|
|
return self->clip_region;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_background_content_cull_out (MetaBackgroundContent *self,
|
|
|
|
cairo_region_t *unobscured_region,
|
|
|
|
cairo_region_t *clip_region)
|
|
|
|
{
|
|
|
|
set_unobscured_region (self, unobscured_region);
|
|
|
|
set_clip_region (self, clip_region);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_background_content_reset_culling (MetaBackgroundContent *self)
|
|
|
|
{
|
|
|
|
set_unobscured_region (self, NULL);
|
|
|
|
set_clip_region (self, NULL);
|
|
|
|
}
|