• Home
  • History
  • Annotate
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1  /*
2   * Copyright 2014 Google Inc.
3   *
4   * Use of this source code is governed by a BSD-style license that can be
5   * found in the LICENSE file.
6   */
7  
8  #include "SkCanvas.h"
9  #include "SkData.h"
10  #include "SkImage.h"
11  #include "SkMaskFilter.h"
12  #include "SkMatrix.h"
13  #include "SkPaint.h"
14  #include "SkPath.h"
15  #include "SkPictureRecorder.h"
16  #include "SkSurface.h"
17  
18  #include "sk_canvas.h"
19  #include "sk_data.h"
20  #include "sk_image.h"
21  #include "sk_paint.h"
22  #include "sk_path.h"
23  #include "sk_surface.h"
24  #include "sk_types_priv.h"
25  
26  const struct {
27      sk_pixelgeometry_t fC;
28      SkPixelGeometry    fSK;
29  } gPixelGeometryMap[] = {
30      { UNKNOWN_SK_PIXELGEOMETRY, kUnknown_SkPixelGeometry },
31      { RGB_H_SK_PIXELGEOMETRY,   kRGB_H_SkPixelGeometry   },
32      { BGR_H_SK_PIXELGEOMETRY,   kBGR_H_SkPixelGeometry   },
33      { RGB_V_SK_PIXELGEOMETRY,   kRGB_V_SkPixelGeometry   },
34      { BGR_V_SK_PIXELGEOMETRY,   kBGR_V_SkPixelGeometry   },
35  };
36  
37  
from_c_pixelgeometry(sk_pixelgeometry_t cGeom,SkPixelGeometry * skGeom)38  static bool from_c_pixelgeometry(sk_pixelgeometry_t cGeom, SkPixelGeometry* skGeom) {
39      for (size_t i = 0; i < SK_ARRAY_COUNT(gPixelGeometryMap); ++i) {
40          if (gPixelGeometryMap[i].fC == cGeom) {
41              if (skGeom) {
42                  *skGeom = gPixelGeometryMap[i].fSK;
43              }
44              return true;
45          }
46      }
47      return false;
48  }
49  
from_c_matrix(const sk_matrix_t * cmatrix,SkMatrix * matrix)50  static void from_c_matrix(const sk_matrix_t* cmatrix, SkMatrix* matrix) {
51      matrix->setAll(cmatrix->mat[0], cmatrix->mat[1], cmatrix->mat[2],
52                     cmatrix->mat[3], cmatrix->mat[4], cmatrix->mat[5],
53                     cmatrix->mat[6], cmatrix->mat[7], cmatrix->mat[8]);
54  }
55  
56  const struct {
57      sk_path_direction_t fC;
58      SkPath::Direction   fSk;
59  } gPathDirMap[] = {
60      { CW_SK_PATH_DIRECTION,  SkPath::kCW_Direction },
61      { CCW_SK_PATH_DIRECTION, SkPath::kCCW_Direction },
62  };
63  
from_c_path_direction(sk_path_direction_t cdir,SkPath::Direction * dir)64  static bool from_c_path_direction(sk_path_direction_t cdir, SkPath::Direction* dir) {
65      for (size_t i = 0; i < SK_ARRAY_COUNT(gPathDirMap); ++i) {
66          if (gPathDirMap[i].fC == cdir) {
67              if (dir) {
68                  *dir = gPathDirMap[i].fSk;
69              }
70              return true;
71          }
72      }
73      return false;
74  }
75  
AsData(const sk_data_t * cdata)76  static SkData* AsData(const sk_data_t* cdata) {
77      return reinterpret_cast<SkData*>(const_cast<sk_data_t*>(cdata));
78  }
79  
ToData(SkData * data)80  static sk_data_t* ToData(SkData* data) {
81      return reinterpret_cast<sk_data_t*>(data);
82  }
83  
ToRect(const SkRect & rect)84  static sk_rect_t ToRect(const SkRect& rect) {
85      return reinterpret_cast<const sk_rect_t&>(rect);
86  }
87  
AsRect(const sk_rect_t & crect)88  static const SkRect& AsRect(const sk_rect_t& crect) {
89      return reinterpret_cast<const SkRect&>(crect);
90  }
91  
AsPath(const sk_path_t & cpath)92  static const SkPath& AsPath(const sk_path_t& cpath) {
93      return reinterpret_cast<const SkPath&>(cpath);
94  }
95  
as_path(sk_path_t * cpath)96  static SkPath* as_path(sk_path_t* cpath) {
97      return reinterpret_cast<SkPath*>(cpath);
98  }
99  
AsImage(const sk_image_t * cimage)100  static const SkImage* AsImage(const sk_image_t* cimage) {
101      return reinterpret_cast<const SkImage*>(cimage);
102  }
103  
ToImage(SkImage * cimage)104  static sk_image_t* ToImage(SkImage* cimage) {
105      return reinterpret_cast<sk_image_t*>(cimage);
106  }
107  
ToCanvas(SkCanvas * canvas)108  static sk_canvas_t* ToCanvas(SkCanvas* canvas) {
109      return reinterpret_cast<sk_canvas_t*>(canvas);
110  }
111  
AsCanvas(sk_canvas_t * ccanvas)112  static SkCanvas* AsCanvas(sk_canvas_t* ccanvas) {
113      return reinterpret_cast<SkCanvas*>(ccanvas);
114  }
115  
AsPictureRecorder(sk_picture_recorder_t * crec)116  static SkPictureRecorder* AsPictureRecorder(sk_picture_recorder_t* crec) {
117      return reinterpret_cast<SkPictureRecorder*>(crec);
118  }
119  
ToPictureRecorder(SkPictureRecorder * rec)120  static sk_picture_recorder_t* ToPictureRecorder(SkPictureRecorder* rec) {
121      return reinterpret_cast<sk_picture_recorder_t*>(rec);
122  }
123  
AsPicture(const sk_picture_t * cpic)124  static const SkPicture* AsPicture(const sk_picture_t* cpic) {
125      return reinterpret_cast<const SkPicture*>(cpic);
126  }
127  
AsPicture(sk_picture_t * cpic)128  static SkPicture* AsPicture(sk_picture_t* cpic) {
129      return reinterpret_cast<SkPicture*>(cpic);
130  }
131  
ToPicture(SkPicture * pic)132  static sk_picture_t* ToPicture(SkPicture* pic) {
133      return reinterpret_cast<sk_picture_t*>(pic);
134  }
135  
136  ///////////////////////////////////////////////////////////////////////////////////////////
137  
sk_image_new_raster_copy(const sk_imageinfo_t * cinfo,const void * pixels,size_t rowBytes)138  sk_image_t* sk_image_new_raster_copy(const sk_imageinfo_t* cinfo, const void* pixels,
139                                       size_t rowBytes) {
140      const SkImageInfo* info = reinterpret_cast<const SkImageInfo*>(cinfo);
141      return (sk_image_t*)SkImage::MakeRasterCopy(SkPixmap(*info, pixels, rowBytes)).release();
142  }
143  
sk_image_new_from_encoded(const sk_data_t * cdata,const sk_irect_t * subset)144  sk_image_t* sk_image_new_from_encoded(const sk_data_t* cdata, const sk_irect_t* subset) {
145      return ToImage(SkImage::MakeFromEncoded(sk_ref_sp(AsData(cdata)),
146                                             reinterpret_cast<const SkIRect*>(subset)).release());
147  }
148  
sk_image_encode(const sk_image_t * cimage)149  sk_data_t* sk_image_encode(const sk_image_t* cimage) {
150      return ToData(AsImage(cimage)->encodeToData().release());
151  }
152  
sk_image_ref(const sk_image_t * cimage)153  void sk_image_ref(const sk_image_t* cimage) {
154      AsImage(cimage)->ref();
155  }
156  
sk_image_unref(const sk_image_t * cimage)157  void sk_image_unref(const sk_image_t* cimage) {
158      AsImage(cimage)->unref();
159  }
160  
sk_image_get_width(const sk_image_t * cimage)161  int sk_image_get_width(const sk_image_t* cimage) {
162      return AsImage(cimage)->width();
163  }
164  
sk_image_get_height(const sk_image_t * cimage)165  int sk_image_get_height(const sk_image_t* cimage) {
166      return AsImage(cimage)->height();
167  }
168  
sk_image_get_unique_id(const sk_image_t * cimage)169  uint32_t sk_image_get_unique_id(const sk_image_t* cimage) {
170      return AsImage(cimage)->uniqueID();
171  }
172  
173  ///////////////////////////////////////////////////////////////////////////////////////////
174  
sk_path_new()175  sk_path_t* sk_path_new() { return (sk_path_t*)new SkPath; }
176  
sk_path_delete(sk_path_t * cpath)177  void sk_path_delete(sk_path_t* cpath) { delete as_path(cpath); }
178  
sk_path_move_to(sk_path_t * cpath,float x,float y)179  void sk_path_move_to(sk_path_t* cpath, float x, float y) {
180      as_path(cpath)->moveTo(x, y);
181  }
182  
sk_path_line_to(sk_path_t * cpath,float x,float y)183  void sk_path_line_to(sk_path_t* cpath, float x, float y) {
184      as_path(cpath)->lineTo(x, y);
185  }
186  
sk_path_quad_to(sk_path_t * cpath,float x0,float y0,float x1,float y1)187  void sk_path_quad_to(sk_path_t* cpath, float x0, float y0, float x1, float y1) {
188      as_path(cpath)->quadTo(x0, y0, x1, y1);
189  }
190  
sk_path_conic_to(sk_path_t * cpath,float x0,float y0,float x1,float y1,float w)191  void sk_path_conic_to(sk_path_t* cpath, float x0, float y0, float x1, float y1, float w) {
192      as_path(cpath)->conicTo(x0, y0, x1, y1, w);
193  }
194  
sk_path_cubic_to(sk_path_t * cpath,float x0,float y0,float x1,float y1,float x2,float y2)195  void sk_path_cubic_to(sk_path_t* cpath, float x0, float y0, float x1, float y1, float x2, float y2) {
196      as_path(cpath)->cubicTo(x0, y0, x1, y1, x2, y2);
197  }
198  
sk_path_close(sk_path_t * cpath)199  void sk_path_close(sk_path_t* cpath) {
200      as_path(cpath)->close();
201  }
202  
sk_path_add_rect(sk_path_t * cpath,const sk_rect_t * crect,sk_path_direction_t cdir)203  void sk_path_add_rect(sk_path_t* cpath, const sk_rect_t* crect, sk_path_direction_t cdir) {
204      SkPath::Direction dir;
205      if (!from_c_path_direction(cdir, &dir)) {
206          return;
207      }
208      as_path(cpath)->addRect(AsRect(*crect), dir);
209  }
210  
sk_path_add_oval(sk_path_t * cpath,const sk_rect_t * crect,sk_path_direction_t cdir)211  void sk_path_add_oval(sk_path_t* cpath, const sk_rect_t* crect, sk_path_direction_t cdir) {
212      SkPath::Direction dir;
213      if (!from_c_path_direction(cdir, &dir)) {
214          return;
215      }
216      as_path(cpath)->addOval(AsRect(*crect), dir);
217  }
218  
sk_path_get_bounds(const sk_path_t * cpath,sk_rect_t * crect)219  bool sk_path_get_bounds(const sk_path_t* cpath, sk_rect_t* crect) {
220      const SkPath& path = AsPath(*cpath);
221  
222      if (path.isEmpty()) {
223          if (crect) {
224              *crect = ToRect(SkRect::MakeEmpty());
225          }
226          return false;
227      }
228  
229      if (crect) {
230          *crect = ToRect(path.getBounds());
231      }
232      return true;
233  }
234  
235  ///////////////////////////////////////////////////////////////////////////////////////////
236  
sk_canvas_save(sk_canvas_t * ccanvas)237  void sk_canvas_save(sk_canvas_t* ccanvas) {
238      AsCanvas(ccanvas)->save();
239  }
240  
sk_canvas_save_layer(sk_canvas_t * ccanvas,const sk_rect_t * crect,const sk_paint_t * cpaint)241  void sk_canvas_save_layer(sk_canvas_t* ccanvas, const sk_rect_t* crect, const sk_paint_t* cpaint) {
242      AsCanvas(ccanvas)->drawRect(AsRect(*crect), AsPaint(*cpaint));
243  }
244  
sk_canvas_restore(sk_canvas_t * ccanvas)245  void sk_canvas_restore(sk_canvas_t* ccanvas) {
246      AsCanvas(ccanvas)->restore();
247  }
248  
sk_canvas_translate(sk_canvas_t * ccanvas,float dx,float dy)249  void sk_canvas_translate(sk_canvas_t* ccanvas, float dx, float dy) {
250      AsCanvas(ccanvas)->translate(dx, dy);
251  }
252  
sk_canvas_scale(sk_canvas_t * ccanvas,float sx,float sy)253  void sk_canvas_scale(sk_canvas_t* ccanvas, float sx, float sy) {
254      AsCanvas(ccanvas)->scale(sx, sy);
255  }
256  
sk_canvas_rotate_degress(sk_canvas_t * ccanvas,float degrees)257  void sk_canvas_rotate_degress(sk_canvas_t* ccanvas, float degrees) {
258      AsCanvas(ccanvas)->rotate(degrees);
259  }
260  
sk_canvas_rotate_radians(sk_canvas_t * ccanvas,float radians)261  void sk_canvas_rotate_radians(sk_canvas_t* ccanvas, float radians) {
262      AsCanvas(ccanvas)->rotate(SkRadiansToDegrees(radians));
263  }
264  
sk_canvas_skew(sk_canvas_t * ccanvas,float sx,float sy)265  void sk_canvas_skew(sk_canvas_t* ccanvas, float sx, float sy) {
266      AsCanvas(ccanvas)->skew(sx, sy);
267  }
268  
sk_canvas_concat(sk_canvas_t * ccanvas,const sk_matrix_t * cmatrix)269  void sk_canvas_concat(sk_canvas_t* ccanvas, const sk_matrix_t* cmatrix) {
270      SkASSERT(cmatrix);
271      SkMatrix matrix;
272      from_c_matrix(cmatrix, &matrix);
273      AsCanvas(ccanvas)->concat(matrix);
274  }
275  
sk_canvas_clip_rect(sk_canvas_t * ccanvas,const sk_rect_t * crect)276  void sk_canvas_clip_rect(sk_canvas_t* ccanvas, const sk_rect_t* crect) {
277      AsCanvas(ccanvas)->clipRect(AsRect(*crect));
278  }
279  
sk_canvas_clip_path(sk_canvas_t * ccanvas,const sk_path_t * cpath)280  void sk_canvas_clip_path(sk_canvas_t* ccanvas, const sk_path_t* cpath) {
281      AsCanvas(ccanvas)->clipPath(AsPath(*cpath));
282  }
283  
sk_canvas_draw_paint(sk_canvas_t * ccanvas,const sk_paint_t * cpaint)284  void sk_canvas_draw_paint(sk_canvas_t* ccanvas, const sk_paint_t* cpaint) {
285      AsCanvas(ccanvas)->drawPaint(AsPaint(*cpaint));
286  }
287  
sk_canvas_draw_rect(sk_canvas_t * ccanvas,const sk_rect_t * crect,const sk_paint_t * cpaint)288  void sk_canvas_draw_rect(sk_canvas_t* ccanvas, const sk_rect_t* crect, const sk_paint_t* cpaint) {
289      AsCanvas(ccanvas)->drawRect(AsRect(*crect), AsPaint(*cpaint));
290  }
291  
sk_canvas_draw_circle(sk_canvas_t * ccanvas,float cx,float cy,float rad,const sk_paint_t * cpaint)292  void sk_canvas_draw_circle(sk_canvas_t* ccanvas, float cx, float cy, float rad,
293                             const sk_paint_t* cpaint) {
294      AsCanvas(ccanvas)->drawCircle(cx, cy, rad, AsPaint(*cpaint));
295  }
296  
sk_canvas_draw_oval(sk_canvas_t * ccanvas,const sk_rect_t * crect,const sk_paint_t * cpaint)297  void sk_canvas_draw_oval(sk_canvas_t* ccanvas, const sk_rect_t* crect, const sk_paint_t* cpaint) {
298      AsCanvas(ccanvas)->drawOval(AsRect(*crect), AsPaint(*cpaint));
299  }
300  
sk_canvas_draw_path(sk_canvas_t * ccanvas,const sk_path_t * cpath,const sk_paint_t * cpaint)301  void sk_canvas_draw_path(sk_canvas_t* ccanvas, const sk_path_t* cpath, const sk_paint_t* cpaint) {
302      AsCanvas(ccanvas)->drawPath(AsPath(*cpath), AsPaint(*cpaint));
303  }
304  
sk_canvas_draw_image(sk_canvas_t * ccanvas,const sk_image_t * cimage,float x,float y,const sk_paint_t * cpaint)305  void sk_canvas_draw_image(sk_canvas_t* ccanvas, const sk_image_t* cimage, float x, float y,
306                            const sk_paint_t* cpaint) {
307      AsCanvas(ccanvas)->drawImage(AsImage(cimage), x, y, AsPaint(cpaint));
308  }
309  
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)310  void sk_canvas_draw_image_rect(sk_canvas_t* ccanvas, const sk_image_t* cimage,
311                                 const sk_rect_t* csrcR, const sk_rect_t* cdstR,
312                                 const sk_paint_t* cpaint) {
313      SkCanvas* canvas = AsCanvas(ccanvas);
314      const SkImage* image = AsImage(cimage);
315      const SkRect& dst = AsRect(*cdstR);
316      const SkPaint* paint = AsPaint(cpaint);
317  
318      if (csrcR) {
319          canvas->drawImageRect(image, AsRect(*csrcR), dst, paint);
320      } else {
321          canvas->drawImageRect(image, dst, paint);
322      }
323  }
324  
sk_canvas_draw_picture(sk_canvas_t * ccanvas,const sk_picture_t * cpicture,const sk_matrix_t * cmatrix,const sk_paint_t * cpaint)325  void sk_canvas_draw_picture(sk_canvas_t* ccanvas, const sk_picture_t* cpicture,
326                              const sk_matrix_t* cmatrix, const sk_paint_t* cpaint) {
327      const SkMatrix* matrixPtr = NULL;
328      SkMatrix matrix;
329      if (cmatrix) {
330          from_c_matrix(cmatrix, &matrix);
331          matrixPtr = &matrix;
332      }
333      AsCanvas(ccanvas)->drawPicture(AsPicture(cpicture), matrixPtr, AsPaint(cpaint));
334  }
335  
336  ///////////////////////////////////////////////////////////////////////////////////////////
337  
sk_surface_new_raster(const sk_imageinfo_t * cinfo,const sk_surfaceprops_t * props)338  sk_surface_t* sk_surface_new_raster(const sk_imageinfo_t* cinfo,
339                                      const sk_surfaceprops_t* props) {
340      const SkImageInfo* info = reinterpret_cast<const SkImageInfo*>(cinfo);
341      SkPixelGeometry geo = kUnknown_SkPixelGeometry;
342      if (props && !from_c_pixelgeometry(props->pixelGeometry, &geo)) {
343          return NULL;
344      }
345  
346      SkSurfaceProps surfProps(0, geo);
347      return (sk_surface_t*)SkSurface::MakeRaster(*info, &surfProps).release();
348  }
349  
sk_surface_new_raster_direct(const sk_imageinfo_t * cinfo,void * pixels,size_t rowBytes,const sk_surfaceprops_t * props)350  sk_surface_t* sk_surface_new_raster_direct(const sk_imageinfo_t* cinfo, void* pixels,
351                                             size_t rowBytes,
352                                             const sk_surfaceprops_t* props) {
353      const SkImageInfo* info = reinterpret_cast<const SkImageInfo*>(cinfo);
354      SkPixelGeometry geo = kUnknown_SkPixelGeometry;
355      if (props && !from_c_pixelgeometry(props->pixelGeometry, &geo)) {
356          return NULL;
357      }
358  
359      SkSurfaceProps surfProps(0, geo);
360      return (sk_surface_t*)SkSurface::MakeRasterDirect(*info, pixels, rowBytes, &surfProps).release();
361  }
362  
sk_surface_unref(sk_surface_t * csurf)363  void sk_surface_unref(sk_surface_t* csurf) {
364      SkSafeUnref((SkSurface*)csurf);
365  }
366  
sk_surface_get_canvas(sk_surface_t * csurf)367  sk_canvas_t* sk_surface_get_canvas(sk_surface_t* csurf) {
368      SkSurface* surf = (SkSurface*)csurf;
369      return (sk_canvas_t*)surf->getCanvas();
370  }
371  
sk_surface_new_image_snapshot(sk_surface_t * csurf)372  sk_image_t* sk_surface_new_image_snapshot(sk_surface_t* csurf) {
373      SkSurface* surf = (SkSurface*)csurf;
374      return (sk_image_t*)surf->makeImageSnapshot().release();
375  }
376  
377  ///////////////////////////////////////////////////////////////////////////////////////////
378  
sk_picture_recorder_new()379  sk_picture_recorder_t* sk_picture_recorder_new() {
380      return ToPictureRecorder(new SkPictureRecorder);
381  }
382  
sk_picture_recorder_delete(sk_picture_recorder_t * crec)383  void sk_picture_recorder_delete(sk_picture_recorder_t* crec) {
384      delete AsPictureRecorder(crec);
385  }
386  
sk_picture_recorder_begin_recording(sk_picture_recorder_t * crec,const sk_rect_t * cbounds)387  sk_canvas_t* sk_picture_recorder_begin_recording(sk_picture_recorder_t* crec,
388                                                   const sk_rect_t* cbounds) {
389      return ToCanvas(AsPictureRecorder(crec)->beginRecording(AsRect(*cbounds)));
390  }
391  
sk_picture_recorder_end_recording(sk_picture_recorder_t * crec)392  sk_picture_t* sk_picture_recorder_end_recording(sk_picture_recorder_t* crec) {
393      return ToPicture(AsPictureRecorder(crec)->finishRecordingAsPicture().release());
394  }
395  
sk_picture_ref(sk_picture_t * cpic)396  void sk_picture_ref(sk_picture_t* cpic) {
397      SkSafeRef(AsPicture(cpic));
398  }
399  
sk_picture_unref(sk_picture_t * cpic)400  void sk_picture_unref(sk_picture_t* cpic) {
401      SkSafeUnref(AsPicture(cpic));
402  }
403  
sk_picture_get_unique_id(sk_picture_t * cpic)404  uint32_t sk_picture_get_unique_id(sk_picture_t* cpic) {
405      return AsPicture(cpic)->uniqueID();
406  }
407  
sk_picture_get_bounds(sk_picture_t * cpic)408  sk_rect_t sk_picture_get_bounds(sk_picture_t* cpic) {
409      return ToRect(AsPicture(cpic)->cullRect());
410  }
411  
412  ///////////////////////////////////////////////////////////////////////////////////////////
413  
sk_data_new_with_copy(const void * src,size_t length)414  sk_data_t* sk_data_new_with_copy(const void* src, size_t length) {
415      return ToData(SkData::MakeWithCopy(src, length).release());
416  }
417  
sk_data_new_from_malloc(const void * memory,size_t length)418  sk_data_t* sk_data_new_from_malloc(const void* memory, size_t length) {
419      return ToData(SkData::MakeFromMalloc(memory, length).release());
420  }
421  
sk_data_new_subset(const sk_data_t * csrc,size_t offset,size_t length)422  sk_data_t* sk_data_new_subset(const sk_data_t* csrc, size_t offset, size_t length) {
423      return ToData(SkData::MakeSubset(AsData(csrc), offset, length).release());
424  }
425  
sk_data_ref(const sk_data_t * cdata)426  void sk_data_ref(const sk_data_t* cdata) {
427      SkSafeRef(AsData(cdata));
428  }
429  
sk_data_unref(const sk_data_t * cdata)430  void sk_data_unref(const sk_data_t* cdata) {
431      SkSafeUnref(AsData(cdata));
432  }
433  
sk_data_get_size(const sk_data_t * cdata)434  size_t sk_data_get_size(const sk_data_t* cdata) {
435      return AsData(cdata)->size();
436  }
437  
sk_data_get_data(const sk_data_t * cdata)438  const void* sk_data_get_data(const sk_data_t* cdata) {
439      return AsData(cdata)->data();
440  }
441  
442  ///////////////////////////////////////////////////////////////////////////////////////////
443