/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 4 -*- * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "gfxQuartzSurface.h" #include "gfxContext.h" #include "gfxImageSurface.h" #include "cairo-quartz.h" void gfxQuartzSurface::MakeInvalid() { mSize = gfxIntSize(-1, -1); } gfxQuartzSurface::gfxQuartzSurface(const gfxSize& desiredSize, gfxImageFormat format) : mCGContext(nullptr), mSize(desiredSize) { gfxIntSize size((unsigned int) floor(desiredSize.width), (unsigned int) floor(desiredSize.height)); if (!CheckSurfaceSize(size)) MakeInvalid(); unsigned int width = static_cast(mSize.width); unsigned int height = static_cast(mSize.height); cairo_surface_t *surf = cairo_quartz_surface_create ((cairo_format_t) format, width, height); mCGContext = cairo_quartz_surface_get_cg_context (surf); CGContextRetain(mCGContext); Init(surf); if (mSurfaceValid) { RecordMemoryUsed(mSize.height * 4 + sizeof(gfxQuartzSurface)); } } gfxQuartzSurface::gfxQuartzSurface(CGContextRef context, const gfxSize& desiredSize) : mCGContext(context), mSize(desiredSize) { gfxIntSize size((unsigned int) floor(desiredSize.width), (unsigned int) floor(desiredSize.height)); if (!CheckSurfaceSize(size)) MakeInvalid(); unsigned int width = static_cast(mSize.width); unsigned int height = static_cast(mSize.height); cairo_surface_t *surf = cairo_quartz_surface_create_for_cg_context(context, width, height); CGContextRetain(mCGContext); Init(surf); if (mSurfaceValid) { RecordMemoryUsed(mSize.height * 4 + sizeof(gfxQuartzSurface)); } } gfxQuartzSurface::gfxQuartzSurface(CGContextRef context, const gfxIntSize& size) : mCGContext(context), mSize(size) { if (!CheckSurfaceSize(size)) MakeInvalid(); unsigned int width = static_cast(mSize.width); unsigned int height = static_cast(mSize.height); cairo_surface_t *surf = cairo_quartz_surface_create_for_cg_context(context, width, height); CGContextRetain(mCGContext); Init(surf); if (mSurfaceValid) { RecordMemoryUsed(mSize.height * 4 + sizeof(gfxQuartzSurface)); } } gfxQuartzSurface::gfxQuartzSurface(cairo_surface_t *csurf, const gfxIntSize& aSize) : mSize(aSize) { mCGContext = cairo_quartz_surface_get_cg_context (csurf); CGContextRetain (mCGContext); Init(csurf, true); } gfxQuartzSurface::gfxQuartzSurface(unsigned char *data, const gfxSize& desiredSize, long stride, gfxImageFormat format) : mCGContext(nullptr), mSize(desiredSize) { gfxIntSize size((unsigned int) floor(desiredSize.width), (unsigned int) floor(desiredSize.height)); if (!CheckSurfaceSize(size)) MakeInvalid(); unsigned int width = static_cast(mSize.width); unsigned int height = static_cast(mSize.height); cairo_surface_t *surf = cairo_quartz_surface_create_for_data (data, (cairo_format_t) format, width, height, stride); mCGContext = cairo_quartz_surface_get_cg_context (surf); CGContextRetain(mCGContext); Init(surf); if (mSurfaceValid) { RecordMemoryUsed(mSize.height * stride + sizeof(gfxQuartzSurface)); } } gfxQuartzSurface::gfxQuartzSurface(unsigned char *data, const gfxIntSize& aSize, long stride, gfxImageFormat format) : mCGContext(nullptr), mSize(aSize.width, aSize.height) { if (!CheckSurfaceSize(aSize)) MakeInvalid(); cairo_surface_t *surf = cairo_quartz_surface_create_for_data (data, (cairo_format_t) format, aSize.width, aSize.height, stride); mCGContext = cairo_quartz_surface_get_cg_context (surf); CGContextRetain(mCGContext); Init(surf); if (mSurfaceValid) { RecordMemoryUsed(mSize.height * stride + sizeof(gfxQuartzSurface)); } } already_AddRefed gfxQuartzSurface::CreateSimilarSurface(gfxContentType aType, const gfxIntSize& aSize) { cairo_surface_t *surface = cairo_quartz_surface_create_cg_layer(mSurface, (cairo_content_t)aType, aSize.width, aSize.height); if (cairo_surface_status(surface)) { cairo_surface_destroy(surface); return nullptr; } nsRefPtr result = Wrap(surface, aSize); cairo_surface_destroy(surface); return result.forget(); } CGContextRef gfxQuartzSurface::GetCGContextWithClip(gfxContext *ctx) { return cairo_quartz_get_cg_context_with_clip(ctx->GetCairo()); } already_AddRefed gfxQuartzSurface::GetAsImageSurface() { cairo_surface_t *surface = cairo_quartz_surface_get_image(mSurface); if (!surface || cairo_surface_status(surface)) return nullptr; nsRefPtr img = Wrap(surface); // cairo_quartz_surface_get_image returns a referenced image, and thebes // shares the refcounts of Cairo surfaces. However, Wrap also adds a // reference to the image. We need to remove one of these references // explicitly so we don't leak. img->Release(); img->SetOpaqueRect(GetOpaqueRect()); return img.forget().downcast(); } gfxQuartzSurface::~gfxQuartzSurface() { CGContextRelease(mCGContext); }