/*
Copyright (C) 2009 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, see .
*/
#include "common.h"
#include
#include "red_window.h"
#include "red_cairo_canvas.h"
#include "utils.h"
#include "debug.h"
#include "region.h"
#include "red_pixmap_cairo.h"
CCanvas::CCanvas(PixmapCache& pixmap_cache, PaletteCache& palette_cache,
GlzDecoderWindow &glz_decoder_window)
: Canvas (pixmap_cache, palette_cache, glz_decoder_window)
, _canvas (NULL)
, _pixmap (0)
{
}
CCanvas::~CCanvas()
{
destroy();
}
void CCanvas::destroy()
{
if (_canvas) {
cairo_t *cairo = canvas_get_cairo(_canvas);
canvas_destroy(_canvas);
cairo_destroy(cairo);
_canvas = NULL;
}
destroy_pixmap();
}
void CCanvas::clear()
{
if (_canvas) {
canvas_clear(_canvas);
}
}
void CCanvas::destroy_pixmap()
{
delete _pixmap;
_pixmap = NULL;
}
void CCanvas::create_pixmap(int width, int height, RedWindow *win)
{
_pixmap = new RedPixmapCairo(width, height, RedPixmap::RGB32, true, NULL, win);
}
void CCanvas::copy_pixels(const QRegion& region, RedDrawable& dest_dc)
{
for (int i = 0; i < (int)region.num_rects; i++) {
SpiceRect* r = ®ion.rects[i];
dest_dc.copy_pixels(*_pixmap, r->left, r->top, *r);
}
}
void CCanvas::copy_pixels(const QRegion& region, RedDrawable* dest_dc, const PixmapHeader* pixmap)
{
copy_pixels(region, *dest_dc);
}
void CCanvas::set_mode(int width, int height, int depth, RedWindow *win)
{
cairo_surface_t *cairo_surface;
cairo_t *cairo;
destroy();
create_pixmap(width, height, win);
cairo_surface = cairo_image_surface_create_for_data(_pixmap->get_data(), CAIRO_FORMAT_RGB24,
width, height, _pixmap->get_stride());
if (cairo_surface_status(cairo_surface) != CAIRO_STATUS_SUCCESS) {
THROW("create surface failed, %s",
cairo_status_to_string(cairo_surface_status(cairo_surface)));
}
cairo = cairo_create(cairo_surface);
cairo_surface_destroy(cairo_surface);
if (cairo_status(cairo) != CAIRO_STATUS_SUCCESS) {
THROW("create cairo failed, %s", cairo_status_to_string(cairo_status(cairo)));
}
if (!(_canvas = canvas_create(cairo, depth,
&pixmap_cache(), bits_cache_put, bits_cache_get,
&palette_cache(), palette_cache_put, palette_cache_get,
palette_cache_release,
&glz_decoder(),
glz_decode))) {
THROW("create canvas failed");
}
}
void CCanvas::set_access_params(unsigned long base, unsigned long max)
{
canvas_set_access_params(_canvas, base, max);
}
void CCanvas::draw_fill(SpiceRect *bbox, SpiceClip *clip, SpiceFill *fill)
{
canvas_draw_fill(_canvas, bbox, clip, fill);
}
void CCanvas::draw_text(SpiceRect *bbox, SpiceClip *clip, SpiceText *text)
{
canvas_draw_text(_canvas, bbox, clip, text);
}
void CCanvas::draw_opaque(SpiceRect *bbox, SpiceClip *clip, SpiceOpaque *opaque)
{
canvas_draw_opaque(_canvas, bbox, clip, opaque);
}
void CCanvas::draw_copy(SpiceRect *bbox, SpiceClip *clip, SpiceCopy *copy)
{
canvas_draw_copy(_canvas, bbox, clip, copy);
}
void CCanvas::draw_transparent(SpiceRect *bbox, SpiceClip *clip, SpiceTransparent* transparent)
{
canvas_draw_transparent(_canvas, bbox, clip, transparent);
}
void CCanvas::draw_alpha_blend(SpiceRect *bbox, SpiceClip *clip, SpiceAlphaBlnd* alpha_blend)
{
canvas_draw_alpha_blend(_canvas, bbox, clip, alpha_blend);
}
void CCanvas::copy_bits(SpiceRect *bbox, SpiceClip *clip, SpicePoint *src_pos)
{
canvas_copy_bits(_canvas, bbox, clip, src_pos);
}
void CCanvas::draw_blend(SpiceRect *bbox, SpiceClip *clip, SpiceBlend *blend)
{
canvas_draw_blend(_canvas, bbox, clip, blend);
}
void CCanvas::draw_blackness(SpiceRect *bbox, SpiceClip *clip, SpiceBlackness *blackness)
{
canvas_draw_blackness(_canvas, bbox, clip, blackness);
}
void CCanvas::draw_whiteness(SpiceRect *bbox, SpiceClip *clip, SpiceWhiteness *whiteness)
{
canvas_draw_whiteness(_canvas, bbox, clip, whiteness);
}
void CCanvas::draw_invers(SpiceRect *bbox, SpiceClip *clip, SpiceInvers *invers)
{
canvas_draw_invers(_canvas, bbox, clip, invers);
}
void CCanvas::draw_rop3(SpiceRect *bbox, SpiceClip *clip, SpiceRop3 *rop3)
{
canvas_draw_rop3(_canvas, bbox, clip, rop3);
}
void CCanvas::draw_stroke(SpiceRect *bbox, SpiceClip *clip, SpiceStroke *stroke)
{
canvas_draw_stroke(_canvas, bbox, clip, stroke);
}
#ifdef WIN32
void CCanvas::put_image(HDC dc, const PixmapHeader& image, const SpiceRect& dest, const QRegion* clip)
{
canvas_put_image(_canvas, dc, &dest, image.data, image.width, image.height, image.stride,
clip);
}
#else
void CCanvas::put_image(const PixmapHeader& image, const SpiceRect& dest, const QRegion* clip)
{
canvas_put_image(_canvas, &dest, image.data, image.width, image.height, image.stride,
clip);
}
#endif
CanvasType CCanvas::get_pixmap_type()
{
return CANVAS_TYPE_CAIRO;
}