forked from AuroraMiddleware/gtk
276 lines
7.8 KiB
C
276 lines
7.8 KiB
C
/* GDK - The GIMP Drawing Kit
|
|
*
|
|
* gdkcairocontext.c: Cairo wrappers
|
|
*
|
|
* Copyright © 2018 Benjamin Otte
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "gdkcairocontext.h"
|
|
|
|
#include "gdkcairocontextprivate.h"
|
|
|
|
#include "gdkcairo.h"
|
|
#include "gdkinternals.h"
|
|
|
|
/**
|
|
* SECTION:gdkcairocontext
|
|
* @Title: GdkCairoContext
|
|
* @Short_description: Cairo draw context
|
|
*
|
|
* #GdkCairoContext is an object representing the platform-specific
|
|
* draw context.
|
|
*
|
|
* #GdkCairoContexts are created for a #GdkDisplay using
|
|
* gdk_surface_create_cairo_context(), and the context can then be used
|
|
* to draw on that #GdkSurface.
|
|
*/
|
|
|
|
/**
|
|
* GdkCairoContext:
|
|
*
|
|
* The GdkCairoContext struct contains only private fields and should not
|
|
* be accessed directly.
|
|
*/
|
|
|
|
typedef struct _GdkCairoContextPrivate GdkCairoContextPrivate;
|
|
|
|
struct _GdkCairoContextPrivate {
|
|
gpointer unused;
|
|
};
|
|
|
|
G_DEFINE_ABSTRACT_TYPE_WITH_CODE (GdkCairoContext, gdk_cairo_context, GDK_TYPE_DRAW_CONTEXT,
|
|
G_ADD_PRIVATE (GdkCairoContext))
|
|
|
|
static cairo_surface_t *
|
|
gdk_surface_ref_impl_surface (GdkSurface *surface)
|
|
{
|
|
return GDK_SURFACE_IMPL_GET_CLASS (surface->impl)->ref_cairo_surface (surface);
|
|
}
|
|
|
|
static cairo_content_t
|
|
gdk_surface_get_content (GdkSurface *surface)
|
|
{
|
|
cairo_surface_t *cairo_surface;
|
|
cairo_content_t content;
|
|
|
|
g_return_val_if_fail (GDK_IS_SURFACE (surface), 0);
|
|
|
|
cairo_surface = gdk_surface_ref_impl_surface (surface);
|
|
content = cairo_surface_get_content (cairo_surface);
|
|
cairo_surface_destroy (cairo_surface);
|
|
|
|
return content;
|
|
}
|
|
|
|
static cairo_t *
|
|
gdk_cairo_context_default_cairo_create (GdkCairoContext *self)
|
|
{
|
|
GdkSurface *surface;
|
|
cairo_region_t *region;
|
|
cairo_surface_t *cairo_surface;
|
|
cairo_t *cr;
|
|
|
|
g_return_val_if_fail (GDK_IS_CAIRO_CONTEXT (self), NULL);
|
|
|
|
surface = gdk_draw_context_get_surface (GDK_DRAW_CONTEXT (self));
|
|
if (surface->drawing_context == NULL ||
|
|
gdk_drawing_context_get_paint_context (surface->drawing_context) != GDK_DRAW_CONTEXT (self))
|
|
return NULL;
|
|
|
|
cairo_surface = _gdk_surface_ref_cairo_surface (surface);
|
|
cr = cairo_create (cairo_surface);
|
|
|
|
region = gdk_surface_get_current_paint_region (surface);
|
|
gdk_cairo_region (cr, region);
|
|
cairo_clip (cr);
|
|
|
|
cairo_region_destroy (region);
|
|
cairo_surface_destroy (cairo_surface);
|
|
|
|
return cr;
|
|
}
|
|
|
|
static void
|
|
gdk_surface_clear_backing_region (GdkSurface *surface)
|
|
{
|
|
cairo_t *cr;
|
|
|
|
if (GDK_SURFACE_DESTROYED (surface))
|
|
return;
|
|
|
|
cr = cairo_create (surface->current_paint.surface);
|
|
|
|
cairo_set_operator (cr, CAIRO_OPERATOR_CLEAR);
|
|
gdk_cairo_region (cr, surface->current_paint.region);
|
|
cairo_fill (cr);
|
|
|
|
cairo_destroy (cr);
|
|
}
|
|
|
|
static void
|
|
gdk_surface_free_current_paint (GdkSurface *surface)
|
|
{
|
|
cairo_surface_destroy (surface->current_paint.surface);
|
|
surface->current_paint.surface = NULL;
|
|
|
|
cairo_region_destroy (surface->current_paint.region);
|
|
surface->current_paint.region = NULL;
|
|
|
|
surface->current_paint.surface_needs_composite = FALSE;
|
|
}
|
|
|
|
static void
|
|
gdk_cairo_context_begin_frame (GdkDrawContext *draw_context,
|
|
cairo_region_t *region)
|
|
{
|
|
GdkRectangle clip_box;
|
|
GdkSurface *surface;
|
|
GdkSurfaceImplClass *impl_class;
|
|
double sx, sy;
|
|
gboolean needs_surface;
|
|
cairo_content_t surface_content;
|
|
|
|
surface = gdk_draw_context_get_surface (draw_context);
|
|
if (surface->current_paint.surface != NULL)
|
|
{
|
|
g_warning ("A paint operation on the surface is alredy in progress. "
|
|
"This is not allowed.");
|
|
return;
|
|
}
|
|
|
|
impl_class = GDK_SURFACE_IMPL_GET_CLASS (surface->impl);
|
|
|
|
needs_surface = TRUE;
|
|
if (impl_class->begin_paint)
|
|
needs_surface = impl_class->begin_paint (surface);
|
|
|
|
surface->current_paint.region = cairo_region_copy (region);
|
|
cairo_region_get_extents (surface->current_paint.region, &clip_box);
|
|
|
|
surface_content = gdk_surface_get_content (surface);
|
|
|
|
if (needs_surface)
|
|
{
|
|
surface->current_paint.surface = gdk_surface_create_similar_surface (surface,
|
|
surface_content,
|
|
MAX (clip_box.width, 1),
|
|
MAX (clip_box.height, 1));
|
|
sx = sy = 1;
|
|
cairo_surface_get_device_scale (surface->current_paint.surface, &sx, &sy);
|
|
cairo_surface_set_device_offset (surface->current_paint.surface, -clip_box.x*sx, -clip_box.y*sy);
|
|
|
|
surface->current_paint.surface_needs_composite = TRUE;
|
|
}
|
|
else
|
|
{
|
|
surface->current_paint.surface = gdk_surface_ref_impl_surface (surface);
|
|
surface->current_paint.surface_needs_composite = FALSE;
|
|
}
|
|
|
|
if (!cairo_region_is_empty (surface->current_paint.region))
|
|
gdk_surface_clear_backing_region (surface);
|
|
}
|
|
|
|
static void
|
|
gdk_cairo_context_end_frame (GdkDrawContext *draw_context,
|
|
cairo_region_t *painted,
|
|
cairo_region_t *damage)
|
|
{
|
|
GdkSurfaceImplClass *impl_class;
|
|
GdkSurface *surface;
|
|
cairo_t *cr;
|
|
|
|
surface = gdk_draw_context_get_surface (draw_context);
|
|
if (surface->current_paint.surface == NULL)
|
|
{
|
|
g_warning (G_STRLOC": no preceding call to gdk_surface_begin_draw_frame(), see documentation");
|
|
return;
|
|
}
|
|
|
|
impl_class = GDK_SURFACE_IMPL_GET_CLASS (surface->impl);
|
|
|
|
if (impl_class->end_paint)
|
|
impl_class->end_paint (surface);
|
|
|
|
if (surface->current_paint.surface_needs_composite)
|
|
{
|
|
cairo_surface_t *cairo_surface;
|
|
|
|
cairo_surface = gdk_surface_ref_impl_surface (surface);
|
|
cr = cairo_create (cairo_surface);
|
|
|
|
cairo_set_source_surface (cr, surface->current_paint.surface, 0, 0);
|
|
gdk_cairo_region (cr, surface->current_paint.region);
|
|
cairo_clip (cr);
|
|
|
|
cairo_set_operator (cr, CAIRO_OPERATOR_SOURCE);
|
|
cairo_paint (cr);
|
|
|
|
cairo_destroy (cr);
|
|
|
|
cairo_surface_flush (cairo_surface);
|
|
cairo_surface_destroy (cairo_surface);
|
|
}
|
|
|
|
gdk_surface_free_current_paint (surface);
|
|
}
|
|
|
|
static void
|
|
gdk_cairo_context_surface_resized (GdkDrawContext *draw_context)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gdk_cairo_context_class_init (GdkCairoContextClass *klass)
|
|
{
|
|
GdkDrawContextClass *draw_context_class = GDK_DRAW_CONTEXT_CLASS (klass);
|
|
|
|
draw_context_class->begin_frame = gdk_cairo_context_begin_frame;
|
|
draw_context_class->end_frame = gdk_cairo_context_end_frame;
|
|
draw_context_class->surface_resized = gdk_cairo_context_surface_resized;
|
|
|
|
klass->cairo_create = gdk_cairo_context_default_cairo_create;
|
|
}
|
|
|
|
static void
|
|
gdk_cairo_context_init (GdkCairoContext *self)
|
|
{
|
|
}
|
|
|
|
/**
|
|
* gdk_cairo_context_cairo_create:
|
|
* @context: a #GdkCairoContext that is currently drawing
|
|
*
|
|
* Retrieves a Cairo context to be used to draw on the #GdkSurface
|
|
* of @context. A call to gdk_surface_begin_draw_frame() with this
|
|
* @context must have been done or this function will return %NULL.
|
|
*
|
|
* The returned context is guaranteed to be valid until
|
|
* gdk_surface_end_draw_frame() is called.
|
|
*
|
|
* Returns: (transfer full) (nullable): a Cairo context to be used
|
|
* to draw the contents of the #GdkSurface. %NULL is returned
|
|
* when @contet is not drawing.
|
|
*/
|
|
cairo_t *
|
|
gdk_cairo_context_cairo_create (GdkCairoContext *self)
|
|
{
|
|
return GDK_CAIRO_CONTEXT_GET_CLASS (self)->cairo_create (self);
|
|
}
|
|
|