/*
|
* Copyright 2014 Google Inc.
|
*
|
* Use of this source code is governed by a BSD-style license that can be
|
* found in the LICENSE file.
|
*/
|
|
#include "SkCanvas.h"
|
#include "SkData.h"
|
#include "SkImage.h"
|
#include "SkMaskFilter.h"
|
#include "SkMatrix.h"
|
#include "SkPaint.h"
|
#include "SkPath.h"
|
#include "SkPictureRecorder.h"
|
#include "SkSurface.h"
|
|
#include "sk_canvas.h"
|
#include "sk_data.h"
|
#include "sk_image.h"
|
#include "sk_paint.h"
|
#include "sk_path.h"
|
#include "sk_surface.h"
|
#include "sk_types_priv.h"
|
|
const struct {
|
sk_pixelgeometry_t fC;
|
SkPixelGeometry fSK;
|
} gPixelGeometryMap[] = {
|
{ UNKNOWN_SK_PIXELGEOMETRY, kUnknown_SkPixelGeometry },
|
{ RGB_H_SK_PIXELGEOMETRY, kRGB_H_SkPixelGeometry },
|
{ BGR_H_SK_PIXELGEOMETRY, kBGR_H_SkPixelGeometry },
|
{ RGB_V_SK_PIXELGEOMETRY, kRGB_V_SkPixelGeometry },
|
{ BGR_V_SK_PIXELGEOMETRY, kBGR_V_SkPixelGeometry },
|
};
|
|
|
static bool from_c_pixelgeometry(sk_pixelgeometry_t cGeom, SkPixelGeometry* skGeom) {
|
for (size_t i = 0; i < SK_ARRAY_COUNT(gPixelGeometryMap); ++i) {
|
if (gPixelGeometryMap[i].fC == cGeom) {
|
if (skGeom) {
|
*skGeom = gPixelGeometryMap[i].fSK;
|
}
|
return true;
|
}
|
}
|
return false;
|
}
|
|
static void from_c_matrix(const sk_matrix_t* cmatrix, SkMatrix* matrix) {
|
matrix->setAll(cmatrix->mat[0], cmatrix->mat[1], cmatrix->mat[2],
|
cmatrix->mat[3], cmatrix->mat[4], cmatrix->mat[5],
|
cmatrix->mat[6], cmatrix->mat[7], cmatrix->mat[8]);
|
}
|
|
const struct {
|
sk_path_direction_t fC;
|
SkPath::Direction fSk;
|
} gPathDirMap[] = {
|
{ CW_SK_PATH_DIRECTION, SkPath::kCW_Direction },
|
{ CCW_SK_PATH_DIRECTION, SkPath::kCCW_Direction },
|
};
|
|
static bool from_c_path_direction(sk_path_direction_t cdir, SkPath::Direction* dir) {
|
for (size_t i = 0; i < SK_ARRAY_COUNT(gPathDirMap); ++i) {
|
if (gPathDirMap[i].fC == cdir) {
|
if (dir) {
|
*dir = gPathDirMap[i].fSk;
|
}
|
return true;
|
}
|
}
|
return false;
|
}
|
|
static SkData* AsData(const sk_data_t* cdata) {
|
return reinterpret_cast<SkData*>(const_cast<sk_data_t*>(cdata));
|
}
|
|
static sk_data_t* ToData(SkData* data) {
|
return reinterpret_cast<sk_data_t*>(data);
|
}
|
|
static sk_rect_t ToRect(const SkRect& rect) {
|
return reinterpret_cast<const sk_rect_t&>(rect);
|
}
|
|
static const SkRect& AsRect(const sk_rect_t& crect) {
|
return reinterpret_cast<const SkRect&>(crect);
|
}
|
|
static const SkPath& AsPath(const sk_path_t& cpath) {
|
return reinterpret_cast<const SkPath&>(cpath);
|
}
|
|
static SkPath* as_path(sk_path_t* cpath) {
|
return reinterpret_cast<SkPath*>(cpath);
|
}
|
|
static const SkImage* AsImage(const sk_image_t* cimage) {
|
return reinterpret_cast<const SkImage*>(cimage);
|
}
|
|
static sk_image_t* ToImage(SkImage* cimage) {
|
return reinterpret_cast<sk_image_t*>(cimage);
|
}
|
|
static sk_canvas_t* ToCanvas(SkCanvas* canvas) {
|
return reinterpret_cast<sk_canvas_t*>(canvas);
|
}
|
|
static SkCanvas* AsCanvas(sk_canvas_t* ccanvas) {
|
return reinterpret_cast<SkCanvas*>(ccanvas);
|
}
|
|
static SkPictureRecorder* AsPictureRecorder(sk_picture_recorder_t* crec) {
|
return reinterpret_cast<SkPictureRecorder*>(crec);
|
}
|
|
static sk_picture_recorder_t* ToPictureRecorder(SkPictureRecorder* rec) {
|
return reinterpret_cast<sk_picture_recorder_t*>(rec);
|
}
|
|
static const SkPicture* AsPicture(const sk_picture_t* cpic) {
|
return reinterpret_cast<const SkPicture*>(cpic);
|
}
|
|
static SkPicture* AsPicture(sk_picture_t* cpic) {
|
return reinterpret_cast<SkPicture*>(cpic);
|
}
|
|
static sk_picture_t* ToPicture(SkPicture* pic) {
|
return reinterpret_cast<sk_picture_t*>(pic);
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////
|
|
sk_image_t* sk_image_new_raster_copy(const sk_imageinfo_t* cinfo, const void* pixels,
|
size_t rowBytes) {
|
const SkImageInfo* info = reinterpret_cast<const SkImageInfo*>(cinfo);
|
return (sk_image_t*)SkImage::MakeRasterCopy(SkPixmap(*info, pixels, rowBytes)).release();
|
}
|
|
sk_image_t* sk_image_new_from_encoded(const sk_data_t* cdata, const sk_irect_t* subset) {
|
return ToImage(SkImage::MakeFromEncoded(sk_ref_sp(AsData(cdata)),
|
reinterpret_cast<const SkIRect*>(subset)).release());
|
}
|
|
sk_data_t* sk_image_encode(const sk_image_t* cimage) {
|
return ToData(AsImage(cimage)->encodeToData().release());
|
}
|
|
void sk_image_ref(const sk_image_t* cimage) {
|
AsImage(cimage)->ref();
|
}
|
|
void sk_image_unref(const sk_image_t* cimage) {
|
AsImage(cimage)->unref();
|
}
|
|
int sk_image_get_width(const sk_image_t* cimage) {
|
return AsImage(cimage)->width();
|
}
|
|
int sk_image_get_height(const sk_image_t* cimage) {
|
return AsImage(cimage)->height();
|
}
|
|
uint32_t sk_image_get_unique_id(const sk_image_t* cimage) {
|
return AsImage(cimage)->uniqueID();
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////
|
|
sk_path_t* sk_path_new() { return (sk_path_t*)new SkPath; }
|
|
void sk_path_delete(sk_path_t* cpath) { delete as_path(cpath); }
|
|
void sk_path_move_to(sk_path_t* cpath, float x, float y) {
|
as_path(cpath)->moveTo(x, y);
|
}
|
|
void sk_path_line_to(sk_path_t* cpath, float x, float y) {
|
as_path(cpath)->lineTo(x, y);
|
}
|
|
void sk_path_quad_to(sk_path_t* cpath, float x0, float y0, float x1, float y1) {
|
as_path(cpath)->quadTo(x0, y0, x1, y1);
|
}
|
|
void sk_path_conic_to(sk_path_t* cpath, float x0, float y0, float x1, float y1, float w) {
|
as_path(cpath)->conicTo(x0, y0, x1, y1, w);
|
}
|
|
void sk_path_cubic_to(sk_path_t* cpath, float x0, float y0, float x1, float y1, float x2, float y2) {
|
as_path(cpath)->cubicTo(x0, y0, x1, y1, x2, y2);
|
}
|
|
void sk_path_close(sk_path_t* cpath) {
|
as_path(cpath)->close();
|
}
|
|
void sk_path_add_rect(sk_path_t* cpath, const sk_rect_t* crect, sk_path_direction_t cdir) {
|
SkPath::Direction dir;
|
if (!from_c_path_direction(cdir, &dir)) {
|
return;
|
}
|
as_path(cpath)->addRect(AsRect(*crect), dir);
|
}
|
|
void sk_path_add_oval(sk_path_t* cpath, const sk_rect_t* crect, sk_path_direction_t cdir) {
|
SkPath::Direction dir;
|
if (!from_c_path_direction(cdir, &dir)) {
|
return;
|
}
|
as_path(cpath)->addOval(AsRect(*crect), dir);
|
}
|
|
bool sk_path_get_bounds(const sk_path_t* cpath, sk_rect_t* crect) {
|
const SkPath& path = AsPath(*cpath);
|
|
if (path.isEmpty()) {
|
if (crect) {
|
*crect = ToRect(SkRect::MakeEmpty());
|
}
|
return false;
|
}
|
|
if (crect) {
|
*crect = ToRect(path.getBounds());
|
}
|
return true;
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////
|
|
void sk_canvas_save(sk_canvas_t* ccanvas) {
|
AsCanvas(ccanvas)->save();
|
}
|
|
void sk_canvas_save_layer(sk_canvas_t* ccanvas, const sk_rect_t* crect, const sk_paint_t* cpaint) {
|
AsCanvas(ccanvas)->drawRect(AsRect(*crect), AsPaint(*cpaint));
|
}
|
|
void sk_canvas_restore(sk_canvas_t* ccanvas) {
|
AsCanvas(ccanvas)->restore();
|
}
|
|
void sk_canvas_translate(sk_canvas_t* ccanvas, float dx, float dy) {
|
AsCanvas(ccanvas)->translate(dx, dy);
|
}
|
|
void sk_canvas_scale(sk_canvas_t* ccanvas, float sx, float sy) {
|
AsCanvas(ccanvas)->scale(sx, sy);
|
}
|
|
void sk_canvas_rotate_degress(sk_canvas_t* ccanvas, float degrees) {
|
AsCanvas(ccanvas)->rotate(degrees);
|
}
|
|
void sk_canvas_rotate_radians(sk_canvas_t* ccanvas, float radians) {
|
AsCanvas(ccanvas)->rotate(SkRadiansToDegrees(radians));
|
}
|
|
void sk_canvas_skew(sk_canvas_t* ccanvas, float sx, float sy) {
|
AsCanvas(ccanvas)->skew(sx, sy);
|
}
|
|
void sk_canvas_concat(sk_canvas_t* ccanvas, const sk_matrix_t* cmatrix) {
|
SkASSERT(cmatrix);
|
SkMatrix matrix;
|
from_c_matrix(cmatrix, &matrix);
|
AsCanvas(ccanvas)->concat(matrix);
|
}
|
|
void sk_canvas_clip_rect(sk_canvas_t* ccanvas, const sk_rect_t* crect) {
|
AsCanvas(ccanvas)->clipRect(AsRect(*crect));
|
}
|
|
void sk_canvas_clip_path(sk_canvas_t* ccanvas, const sk_path_t* cpath) {
|
AsCanvas(ccanvas)->clipPath(AsPath(*cpath));
|
}
|
|
void sk_canvas_draw_paint(sk_canvas_t* ccanvas, const sk_paint_t* cpaint) {
|
AsCanvas(ccanvas)->drawPaint(AsPaint(*cpaint));
|
}
|
|
void sk_canvas_draw_rect(sk_canvas_t* ccanvas, const sk_rect_t* crect, const sk_paint_t* cpaint) {
|
AsCanvas(ccanvas)->drawRect(AsRect(*crect), AsPaint(*cpaint));
|
}
|
|
void sk_canvas_draw_circle(sk_canvas_t* ccanvas, float cx, float cy, float rad,
|
const sk_paint_t* cpaint) {
|
AsCanvas(ccanvas)->drawCircle(cx, cy, rad, AsPaint(*cpaint));
|
}
|
|
void sk_canvas_draw_oval(sk_canvas_t* ccanvas, const sk_rect_t* crect, const sk_paint_t* cpaint) {
|
AsCanvas(ccanvas)->drawOval(AsRect(*crect), AsPaint(*cpaint));
|
}
|
|
void sk_canvas_draw_path(sk_canvas_t* ccanvas, const sk_path_t* cpath, const sk_paint_t* cpaint) {
|
AsCanvas(ccanvas)->drawPath(AsPath(*cpath), AsPaint(*cpaint));
|
}
|
|
void sk_canvas_draw_image(sk_canvas_t* ccanvas, const sk_image_t* cimage, float x, float y,
|
const sk_paint_t* cpaint) {
|
AsCanvas(ccanvas)->drawImage(AsImage(cimage), x, y, AsPaint(cpaint));
|
}
|
|
void sk_canvas_draw_image_rect(sk_canvas_t* ccanvas, const sk_image_t* cimage,
|
const sk_rect_t* csrcR, const sk_rect_t* cdstR,
|
const sk_paint_t* cpaint) {
|
SkCanvas* canvas = AsCanvas(ccanvas);
|
const SkImage* image = AsImage(cimage);
|
const SkRect& dst = AsRect(*cdstR);
|
const SkPaint* paint = AsPaint(cpaint);
|
|
if (csrcR) {
|
canvas->drawImageRect(image, AsRect(*csrcR), dst, paint);
|
} else {
|
canvas->drawImageRect(image, dst, paint);
|
}
|
}
|
|
void sk_canvas_draw_picture(sk_canvas_t* ccanvas, const sk_picture_t* cpicture,
|
const sk_matrix_t* cmatrix, const sk_paint_t* cpaint) {
|
const SkMatrix* matrixPtr = NULL;
|
SkMatrix matrix;
|
if (cmatrix) {
|
from_c_matrix(cmatrix, &matrix);
|
matrixPtr = &matrix;
|
}
|
AsCanvas(ccanvas)->drawPicture(AsPicture(cpicture), matrixPtr, AsPaint(cpaint));
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////
|
|
sk_surface_t* sk_surface_new_raster(const sk_imageinfo_t* cinfo,
|
const sk_surfaceprops_t* props) {
|
const SkImageInfo* info = reinterpret_cast<const SkImageInfo*>(cinfo);
|
SkPixelGeometry geo = kUnknown_SkPixelGeometry;
|
if (props && !from_c_pixelgeometry(props->pixelGeometry, &geo)) {
|
return NULL;
|
}
|
|
SkSurfaceProps surfProps(0, geo);
|
return (sk_surface_t*)SkSurface::MakeRaster(*info, &surfProps).release();
|
}
|
|
sk_surface_t* sk_surface_new_raster_direct(const sk_imageinfo_t* cinfo, void* pixels,
|
size_t rowBytes,
|
const sk_surfaceprops_t* props) {
|
const SkImageInfo* info = reinterpret_cast<const SkImageInfo*>(cinfo);
|
SkPixelGeometry geo = kUnknown_SkPixelGeometry;
|
if (props && !from_c_pixelgeometry(props->pixelGeometry, &geo)) {
|
return NULL;
|
}
|
|
SkSurfaceProps surfProps(0, geo);
|
return (sk_surface_t*)SkSurface::MakeRasterDirect(*info, pixels, rowBytes, &surfProps).release();
|
}
|
|
void sk_surface_unref(sk_surface_t* csurf) {
|
SkSafeUnref((SkSurface*)csurf);
|
}
|
|
sk_canvas_t* sk_surface_get_canvas(sk_surface_t* csurf) {
|
SkSurface* surf = (SkSurface*)csurf;
|
return (sk_canvas_t*)surf->getCanvas();
|
}
|
|
sk_image_t* sk_surface_new_image_snapshot(sk_surface_t* csurf) {
|
SkSurface* surf = (SkSurface*)csurf;
|
return (sk_image_t*)surf->makeImageSnapshot().release();
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////
|
|
sk_picture_recorder_t* sk_picture_recorder_new() {
|
return ToPictureRecorder(new SkPictureRecorder);
|
}
|
|
void sk_picture_recorder_delete(sk_picture_recorder_t* crec) {
|
delete AsPictureRecorder(crec);
|
}
|
|
sk_canvas_t* sk_picture_recorder_begin_recording(sk_picture_recorder_t* crec,
|
const sk_rect_t* cbounds) {
|
return ToCanvas(AsPictureRecorder(crec)->beginRecording(AsRect(*cbounds)));
|
}
|
|
sk_picture_t* sk_picture_recorder_end_recording(sk_picture_recorder_t* crec) {
|
return ToPicture(AsPictureRecorder(crec)->finishRecordingAsPicture().release());
|
}
|
|
void sk_picture_ref(sk_picture_t* cpic) {
|
SkSafeRef(AsPicture(cpic));
|
}
|
|
void sk_picture_unref(sk_picture_t* cpic) {
|
SkSafeUnref(AsPicture(cpic));
|
}
|
|
uint32_t sk_picture_get_unique_id(sk_picture_t* cpic) {
|
return AsPicture(cpic)->uniqueID();
|
}
|
|
sk_rect_t sk_picture_get_bounds(sk_picture_t* cpic) {
|
return ToRect(AsPicture(cpic)->cullRect());
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////
|
|
sk_data_t* sk_data_new_with_copy(const void* src, size_t length) {
|
return ToData(SkData::MakeWithCopy(src, length).release());
|
}
|
|
sk_data_t* sk_data_new_from_malloc(const void* memory, size_t length) {
|
return ToData(SkData::MakeFromMalloc(memory, length).release());
|
}
|
|
sk_data_t* sk_data_new_subset(const sk_data_t* csrc, size_t offset, size_t length) {
|
return ToData(SkData::MakeSubset(AsData(csrc), offset, length).release());
|
}
|
|
void sk_data_ref(const sk_data_t* cdata) {
|
SkSafeRef(AsData(cdata));
|
}
|
|
void sk_data_unref(const sk_data_t* cdata) {
|
SkSafeUnref(AsData(cdata));
|
}
|
|
size_t sk_data_get_size(const sk_data_t* cdata) {
|
return AsData(cdata)->size();
|
}
|
|
const void* sk_data_get_data(const sk_data_t* cdata) {
|
return AsData(cdata)->data();
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////
|