canvas: Use the window-scaling-factor setting

ClutterCanvas is a ClutterContent interface implementation; this means
that it can be created and modified regardless of whether it is
associated to a specific actor or a stage. For this reason, we cannot
walk the hierarchy and get the window scaling factor for high DPI
density displays out of the ClutterStage when we create the Cairo
surface that we will use to draw the canvas contents on.

We can use ClutterSettings:window-scaling-factor instead, since it's
what each ClutterStage will use anyway.

This will get slightly more complicated when we support per-output
window scaling factors (like on Wayland), but that will require changes
in the entire settings architecture anyway.

https://bugzilla.gnome.org/show_bug.cgi?id=705915
This commit is contained in:
Emmanuele Bassi 2014-01-16 12:24:57 +00:00
parent ed0633468f
commit c1d6194d24

View File

@ -62,10 +62,12 @@
#include "clutter-cairo.h" #include "clutter-cairo.h"
#include "clutter-color.h" #include "clutter-color.h"
#include "clutter-content-private.h" #include "clutter-content-private.h"
#include "clutter-debug.h"
#include "clutter-marshal.h" #include "clutter-marshal.h"
#include "clutter-paint-node.h" #include "clutter-paint-node.h"
#include "clutter-paint-nodes.h" #include "clutter-paint-nodes.h"
#include "clutter-private.h" #include "clutter-private.h"
#include "clutter-settings.h"
struct _ClutterCanvasPrivate struct _ClutterCanvasPrivate
{ {
@ -353,23 +355,37 @@ static void
clutter_canvas_emit_draw (ClutterCanvas *self) clutter_canvas_emit_draw (ClutterCanvas *self)
{ {
ClutterCanvasPrivate *priv = self->priv; ClutterCanvasPrivate *priv = self->priv;
int real_width, real_height;
cairo_surface_t *surface; cairo_surface_t *surface;
gboolean mapped_buffer; gboolean mapped_buffer;
unsigned char *data; unsigned char *data;
CoglBuffer *buffer; CoglBuffer *buffer;
int window_scale = 1;
gboolean res; gboolean res;
cairo_t *cr; cairo_t *cr;
g_assert (priv->width > 0 && priv->width > 0); g_assert (priv->width > 0 && priv->width > 0);
g_object_get (clutter_settings_get_default (),
"window-scaling-factor", &window_scale,
NULL);
real_width = priv->width * window_scale;
real_height = priv->height * window_scale;
CLUTTER_NOTE (MISC, "Creating Cairo surface with size %d x %d (real: %d x %d, scale: %d)",
priv->width, priv->height,
real_width, real_height,
window_scale);
if (priv->buffer == NULL) if (priv->buffer == NULL)
{ {
CoglContext *ctx; CoglContext *ctx;
ctx = clutter_backend_get_cogl_context (clutter_get_default_backend ()); ctx = clutter_backend_get_cogl_context (clutter_get_default_backend ());
priv->buffer = cogl_bitmap_new_with_size (ctx, priv->buffer = cogl_bitmap_new_with_size (ctx,
priv->width, real_width,
priv->height, real_height,
CLUTTER_CAIRO_FORMAT_ARGB32); CLUTTER_CAIRO_FORMAT_ARGB32);
} }
@ -389,20 +405,24 @@ clutter_canvas_emit_draw (ClutterCanvas *self)
surface = cairo_image_surface_create_for_data (data, surface = cairo_image_surface_create_for_data (data,
CAIRO_FORMAT_ARGB32, CAIRO_FORMAT_ARGB32,
priv->width, real_width,
priv->height, real_height,
bitmap_stride); bitmap_stride);
mapped_buffer = TRUE; mapped_buffer = TRUE;
} }
else else
{ {
surface = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, surface = cairo_image_surface_create (CAIRO_FORMAT_ARGB32,
priv->width, real_width,
priv->height); real_height);
mapped_buffer = FALSE; mapped_buffer = FALSE;
} }
#ifdef HAVE_CAIRO_SURFACE_SET_DEVICE_SCALE
cairo_surface_set_device_scale (surface, window_scale, window_scale);
#endif
self->priv->cr = cr = cairo_create (surface); self->priv->cr = cr = cairo_create (surface);
g_signal_emit (self, canvas_signals[DRAW], 0, g_signal_emit (self, canvas_signals[DRAW], 0,