/* cairo - a vector graphics library with display and print output * * Copyright © 2002 University of Southern California * Copyright © 2005 Red Hat, Inc. * * This library is free software; you can redistribute it and/or * modify it either under the terms of the GNU Lesser General Public * License version 2.1 as published by the Free Software Foundation * (the "LGPL") or, at your option, under the terms of the Mozilla * Public License Version 1.1 (the "MPL"). If you do not alter this * notice, a recipient may use your version of this file under either * the MPL or the LGPL. * * You should have received a copy of the LGPL along with this library * in the file COPYING-LGPL-2.1; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * You should have received a copy of the MPL along with this library * in the file COPYING-MPL-1.1 * * The contents of this file are subject to the Mozilla Public License * Version 1.1 (the "License"); you may not use this file except in * compliance with the License. You may obtain a copy of the License at * http://www.mozilla.org/MPL/ * * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY * OF ANY KIND, either express or implied. See the LGPL or the MPL for * the specific language governing rights and limitations. * * The Original Code is the cairo graphics library. * * The Initial Developer of the Original Code is University of Southern * California. * * Contributor(s): * Carl D. Worth * Kristian Høgsberg */ #include "cairoint.h" #include "cairo-clip-private.h" static cairo_clip_path_t * _cairo_clip_path_reference (cairo_clip_path_t *clip_path); static void _cairo_clip_path_destroy (cairo_clip_path_t *clip_path); void _cairo_clip_init (cairo_clip_t *clip, cairo_surface_t *target) { if (target) clip->mode = _cairo_surface_get_clip_mode (target); else clip->mode = CAIRO_CLIP_MODE_MASK; clip->surface = NULL; clip->surface_rect.x = 0; clip->surface_rect.y = 0; clip->surface_rect.width = 0; clip->surface_rect.height = 0; clip->serial = 0; pixman_region_init (&clip->region); clip->has_region = FALSE; clip->path = NULL; } cairo_status_t _cairo_clip_init_copy (cairo_clip_t *clip, cairo_clip_t *other) { clip->mode = other->mode; clip->surface = cairo_surface_reference (other->surface); clip->surface_rect = other->surface_rect; clip->serial = other->serial; pixman_region_init (&clip->region); if (other->has_region) { if (pixman_region_copy (&clip->region, &other->region) != PIXMAN_REGION_STATUS_SUCCESS) { pixman_region_fini (&clip->region); cairo_surface_destroy (clip->surface); return CAIRO_STATUS_NO_MEMORY; } clip->has_region = TRUE; } else { clip->has_region = FALSE; } clip->path = _cairo_clip_path_reference (other->path); return CAIRO_STATUS_SUCCESS; } void _cairo_clip_reset (cairo_clip_t *clip) { /* destroy any existing clip-region artifacts */ cairo_surface_destroy (clip->surface); clip->surface = NULL; clip->serial = 0; if (clip->has_region) { /* pixman_region_fini just releases the resources used but * doesn't bother with leaving the region in a valid state. * So pixman_region_init has to be called afterwards. */ pixman_region_fini (&clip->region); pixman_region_init (&clip->region); clip->has_region = FALSE; } _cairo_clip_path_destroy (clip->path); clip->path = NULL; } static cairo_status_t _cairo_clip_path_intersect_to_rectangle (cairo_clip_path_t *clip_path, cairo_rectangle_int16_t *rectangle) { while (clip_path) { cairo_status_t status; cairo_traps_t traps; cairo_box_t extents; cairo_rectangle_int16_t extents_rect; _cairo_traps_init (&traps); status = _cairo_path_fixed_fill_to_traps (&clip_path->path, clip_path->fill_rule, clip_path->tolerance, &traps); if (status) { _cairo_traps_fini (&traps); return status; } _cairo_traps_extents (&traps, &extents); _cairo_box_round_to_rectangle (&extents, &extents_rect); _cairo_rectangle_intersect (rectangle, &extents_rect); _cairo_traps_fini (&traps); clip_path = clip_path->prev; } return CAIRO_STATUS_SUCCESS; } cairo_status_t _cairo_clip_intersect_to_rectangle (cairo_clip_t *clip, cairo_rectangle_int16_t *rectangle) { if (!clip) return CAIRO_STATUS_SUCCESS; if (clip->path) { cairo_status_t status; status = _cairo_clip_path_intersect_to_rectangle (clip->path, rectangle); if (status) return status; } if (clip->has_region) { cairo_status_t status = CAIRO_STATUS_SUCCESS; pixman_region16_t intersection; pixman_region_init_rect (&intersection, rectangle->x, rectangle->y, rectangle->width, rectangle->height); if (PIXMAN_REGION_STATUS_SUCCESS != pixman_region_intersect (&intersection, &clip->region, &intersection)) { status = CAIRO_STATUS_NO_MEMORY; } else { _cairo_region_extents_rectangle (&intersection, rectangle); } pixman_region_fini (&intersection); if (status) return status; } if (clip->surface) _cairo_rectangle_intersect (rectangle, &clip->surface_rect); return CAIRO_STATUS_SUCCESS; } cairo_status_t _cairo_clip_intersect_to_region (cairo_clip_t *clip, pixman_region16_t *region) { pixman_region_status_t pixman_status; if (!clip) return CAIRO_STATUS_SUCCESS; if (clip->path) { /* Intersect clip path into region. */ } if (clip->has_region) { pixman_status = pixman_region_intersect (region, &clip->region, region); if (pixman_status != PIXMAN_REGION_STATUS_SUCCESS) return CAIRO_STATUS_NO_MEMORY; } if (clip->surface) { cairo_status_t status = CAIRO_STATUS_SUCCESS; pixman_region16_t clip_rect; pixman_region_init_rect (&clip_rect, clip->surface_rect.x, clip->surface_rect.y, clip->surface_rect.width, clip->surface_rect.height); if (PIXMAN_REGION_STATUS_SUCCESS != pixman_region_intersect (region, &clip_rect, region)) status = CAIRO_STATUS_NO_MEMORY; pixman_region_fini (&clip_rect); if (status) return status; } return CAIRO_STATUS_SUCCESS; } /* Combines the region of clip->surface given by extents in * device backend coordinates into the given temporary surface, * which has its origin at dst_x, dst_y in backend coordinates */ cairo_status_t _cairo_clip_combine_to_surface (cairo_clip_t *clip, cairo_operator_t op, cairo_surface_t *dst, int dst_x, int dst_y, const cairo_rectangle_int16_t *extents) { cairo_pattern_union_t pattern; cairo_status_t status; _cairo_pattern_init_for_surface (&pattern.surface, clip->surface); status = _cairo_surface_composite (op, &pattern.base, NULL, dst, extents->x - clip->surface_rect.x, extents->y - clip->surface_rect.y, 0, 0, extents->x - dst_x, extents->y - dst_y, extents->width, extents->height); _cairo_pattern_fini (&pattern.base); return status; } static cairo_status_t _cairo_clip_intersect_path (cairo_clip_t *clip, cairo_path_fixed_t *path, cairo_fill_rule_t fill_rule, double tolerance, cairo_antialias_t antialias) { cairo_clip_path_t *clip_path; cairo_status_t status; if (clip->mode != CAIRO_CLIP_MODE_PATH) return CAIRO_INT_STATUS_UNSUPPORTED; clip_path = malloc (sizeof (cairo_clip_path_t)); if (clip_path == NULL) return CAIRO_STATUS_NO_MEMORY; status = _cairo_path_fixed_init_copy (&clip_path->path, path); if (status) { free (clip_path); return status; } clip_path->ref_count = 1; clip_path->fill_rule = fill_rule; clip_path->tolerance = tolerance; clip_path->antialias = antialias; clip_path->prev = clip->path; clip->path = clip_path; return CAIRO_STATUS_SUCCESS; } static cairo_clip_path_t * _cairo_clip_path_reference (cairo_clip_path_t *clip_path) { if (clip_path == NULL) return NULL; clip_path->ref_count++; return clip_path; } static void _cairo_clip_path_destroy (cairo_clip_path_t *clip_path) { if (clip_path == NULL) return; clip_path->ref_count--; if (clip_path->ref_count) return; _cairo_path_fixed_fini (&clip_path->path); _cairo_clip_path_destroy (clip_path->prev); free (clip_path); } static cairo_int_status_t _cairo_clip_intersect_region (cairo_clip_t *clip, cairo_traps_t *traps, cairo_surface_t *target) { pixman_region16_t region; cairo_int_status_t status; if (clip->mode != CAIRO_CLIP_MODE_REGION) return CAIRO_INT_STATUS_UNSUPPORTED; status = _cairo_traps_extract_region (traps, ®ion); if (status) return status; status = CAIRO_STATUS_SUCCESS; if (!clip->has_region) { if (pixman_region_copy (&clip->region, ®ion) == PIXMAN_REGION_STATUS_SUCCESS) clip->has_region = TRUE; } else { pixman_region16_t intersection; pixman_region_init (&intersection); if (PIXMAN_REGION_STATUS_SUCCESS != pixman_region_intersect (&intersection, &clip->region, ®ion) || PIXMAN_REGION_STATUS_SUCCESS != pixman_region_copy (&clip->region, &intersection)) status = CAIRO_STATUS_NO_MEMORY; pixman_region_fini (&intersection); } clip->serial = _cairo_surface_allocate_clip_serial (target); pixman_region_fini (®ion); return status; } static cairo_status_t _cairo_clip_intersect_mask (cairo_clip_t *clip, cairo_traps_t *traps, cairo_antialias_t antialias, cairo_surface_t *target) { cairo_pattern_union_t pattern; cairo_box_t extents; cairo_rectangle_int16_t surface_rect, target_rect; cairo_surface_t *surface; cairo_status_t status; /* Represent the clip as a mask surface. We create a new surface * the size of the intersection of the old mask surface and the * extents of the new clip path. */ _cairo_traps_extents (traps, &extents); _cairo_box_round_to_rectangle (&extents, &surface_rect); if (clip->surface != NULL) _cairo_rectangle_intersect (&surface_rect, &clip->surface_rect); /* Intersect with the target surface rectangle so we don't use * more memory and time than we need to. */ status = _cairo_surface_get_extents (target, &target_rect); if (!status) _cairo_rectangle_intersect (&surface_rect, &target_rect); surface = _cairo_surface_create_similar_solid (target, CAIRO_CONTENT_ALPHA, surface_rect.width, surface_rect.height, CAIRO_COLOR_WHITE); if (surface->status) return CAIRO_STATUS_NO_MEMORY; /* Render the new clipping path into the new mask surface. */ _cairo_traps_translate (traps, -surface_rect.x, -surface_rect.y); _cairo_pattern_init_solid (&pattern.solid, CAIRO_COLOR_WHITE); status = _cairo_surface_composite_trapezoids (CAIRO_OPERATOR_IN, &pattern.base, surface, antialias, 0, 0, 0, 0, surface_rect.width, surface_rect.height, traps->traps, traps->num_traps); _cairo_pattern_fini (&pattern.base); if (status) { cairo_surface_destroy (surface); return status; } /* If there was a clip surface already, combine it with the new * mask surface using the IN operator, so we get the intersection * of the old and new clipping paths. */ if (clip->surface != NULL) { _cairo_pattern_init_for_surface (&pattern.surface, clip->surface); status = _cairo_surface_composite (CAIRO_OPERATOR_IN, &pattern.base, NULL, surface, surface_rect.x - clip->surface_rect.x, surface_rect.y - clip->surface_rect.y, 0, 0, 0, 0, surface_rect.width, surface_rect.height); _cairo_pattern_fini (&pattern.base); if (status) { cairo_surface_destroy (surface); return status; } cairo_surface_destroy (clip->surface); } clip->surface = surface; clip->surface_rect = surface_rect; clip->serial = _cairo_surface_allocate_clip_serial (target); return status; } cairo_status_t _cairo_clip_clip (cairo_clip_t *clip, cairo_path_fixed_t *path, cairo_fill_rule_t fill_rule, double tolerance, cairo_antialias_t antialias, cairo_surface_t *target) { cairo_status_t status; cairo_traps_t traps; status = _cairo_clip_intersect_path (clip, path, fill_rule, tolerance, antialias); if (status == CAIRO_STATUS_SUCCESS) clip->serial = _cairo_surface_allocate_clip_serial (target); if (status != CAIRO_INT_STATUS_UNSUPPORTED) return status; _cairo_traps_init (&traps); status = _cairo_path_fixed_fill_to_traps (path, fill_rule, tolerance, &traps); if (status) goto bail; status = _cairo_clip_intersect_region (clip, &traps, target); if (status != CAIRO_INT_STATUS_UNSUPPORTED) goto bail; status = _cairo_clip_intersect_mask (clip, &traps, antialias, target); bail: _cairo_traps_fini (&traps); return status; } void _cairo_clip_translate (cairo_clip_t *clip, cairo_fixed_t tx, cairo_fixed_t ty) { if (clip->has_region) { pixman_region_translate (&clip->region, _cairo_fixed_integer_part (tx), _cairo_fixed_integer_part (ty)); } if (clip->surface) { clip->surface_rect.x += _cairo_fixed_integer_part (tx); clip->surface_rect.y += _cairo_fixed_integer_part (ty); } if (clip->path) { cairo_clip_path_t *clip_path = clip->path; cairo_matrix_t matrix; cairo_matrix_init_translate (&matrix, _cairo_fixed_to_double (tx), _cairo_fixed_to_double (ty)); while (clip_path) { _cairo_path_fixed_device_transform (&clip_path->path, &matrix); clip_path = clip_path->prev; } } } static void _cairo_clip_path_reapply_clip_path (cairo_clip_t *clip, cairo_clip_path_t *clip_path) { if (clip_path->prev) _cairo_clip_path_reapply_clip_path (clip, clip_path->prev); _cairo_clip_intersect_path (clip, &clip_path->path, clip_path->fill_rule, clip_path->tolerance, clip_path->antialias); } cairo_status_t _cairo_clip_init_deep_copy (cairo_clip_t *clip, cairo_clip_t *other, cairo_surface_t *target) { _cairo_clip_init (clip, target); if (other->mode != clip->mode) { /* We should reapply the original clip path in this case, and let * whatever the right handling is happen */ } else { if (other->has_region) { if (pixman_region_copy (&clip->region, &other->region) != PIXMAN_REGION_STATUS_SUCCESS) goto BAIL; clip->has_region = TRUE; } if (other->surface) { if (_cairo_surface_clone_similar (target, other->surface, other->surface_rect.x, other->surface_rect.y, other->surface_rect.width, other->surface_rect.height, &clip->surface) != CAIRO_STATUS_SUCCESS) goto BAIL; clip->surface_rect = other->surface_rect; } if (other->path) { _cairo_clip_path_reapply_clip_path (clip, other->path); } } return CAIRO_STATUS_SUCCESS; BAIL: if (clip->has_region) pixman_region_fini (&clip->region); if (clip->surface) cairo_surface_destroy (clip->surface); return CAIRO_STATUS_NO_MEMORY; } const cairo_rectangle_list_t _cairo_rectangles_nil = { CAIRO_STATUS_NO_MEMORY, NULL, 0 }; static const cairo_rectangle_list_t _cairo_rectangles_not_representable = { CAIRO_STATUS_CLIP_NOT_REPRESENTABLE, NULL, 0 }; static cairo_bool_t _cairo_clip_rect_to_user (cairo_gstate_t *gstate, double x, double y, double width, double height, cairo_rectangle_t *rectangle) { double x2 = x + width; double y2 = y + height; cairo_bool_t is_tight; _cairo_gstate_backend_to_user_rectangle (gstate, &x, &y, &x2, &y2, &is_tight); rectangle->x = x; rectangle->y = y; rectangle->width = x2 - x; rectangle->height = y2 - y; return is_tight; } cairo_private cairo_rectangle_list_t* _cairo_clip_copy_rectangle_list (cairo_clip_t *clip, cairo_gstate_t *gstate) { cairo_rectangle_list_t *list; cairo_rectangle_t *rectangles; int n_boxes; if (clip->path || clip->surface) return (cairo_rectangle_list_t*) &_cairo_rectangles_not_representable; n_boxes = clip->has_region ? pixman_region_num_rects (&clip->region) : 1; rectangles = malloc (sizeof (cairo_rectangle_t)*n_boxes); if (rectangles == NULL) return (cairo_rectangle_list_t*) &_cairo_rectangles_nil; if (clip->has_region) { pixman_box16_t *boxes; int i; boxes = pixman_region_rects (&clip->region); for (i = 0; i < n_boxes; ++i) { if (!_cairo_clip_rect_to_user(gstate, boxes[i].x1, boxes[i].y1, boxes[i].x2 - boxes[i].x1, boxes[i].y2 - boxes[i].y1, &rectangles[i])) { free (rectangles); return (cairo_rectangle_list_t*) &_cairo_rectangles_not_representable; } } } else { cairo_rectangle_int16_t extents; if (_cairo_surface_get_extents (_cairo_gstate_get_target (gstate), &extents) != CAIRO_STATUS_SUCCESS || !_cairo_clip_rect_to_user(gstate, extents.x, extents.y, extents.width, extents.height, rectangles)) { free (rectangles); return (cairo_rectangle_list_t*) &_cairo_rectangles_not_representable; } } list = malloc (sizeof (cairo_rectangle_list_t)); if (list == NULL) { free (rectangles); return (cairo_rectangle_list_t*) &_cairo_rectangles_nil; } list->status = CAIRO_STATUS_SUCCESS; list->rectangles = rectangles; list->num_rectangles = n_boxes; return list; } /** * cairo_rectangle_list_destroy: * @rectangle_list: a rectangle list, as obtained from cairo_copy_clip_rectangles() * * Unconditionally frees @rectangle_list and all associated * references. After this call, the @rectangle_list pointer must not * be dereferenced. * * Since: 1.4 **/ void cairo_rectangle_list_destroy (cairo_rectangle_list_t *rectangle_list) { if (rectangle_list == NULL || rectangle_list == &_cairo_rectangles_nil || rectangle_list == &_cairo_rectangles_not_representable) return; free (rectangle_list->rectangles); free (rectangle_list); }