| /* | 
 |  * Copyright 2011 Google Inc. | 
 |  * | 
 |  * Use of this source code is governed by a BSD-style license that can be | 
 |  * found in the LICENSE file. | 
 |  */ | 
 |  | 
 | #include "gm.h" | 
 |  | 
 | class SkJSCanvas { | 
 | public: | 
 |     SkJSCanvas(SkCanvas* target); | 
 |     ~SkJSCanvas(); | 
 |  | 
 |     void save(); | 
 |     void restore(); | 
 |  | 
 |     double lineWidth; | 
 |     void setLineWidth(double); | 
 |  | 
 |     void beginPath(); | 
 |     void moveTo(double x, double y); | 
 |     void lineTo(double x, double y); | 
 |     void closePath(); | 
 |  | 
 |     void fill(); | 
 |     void stroke(); | 
 |  | 
 |     void fillText(const char text[], double x, double y); | 
 |  | 
 | private: | 
 |     SkCanvas*   fTarget; | 
 |     SkPaint     fFillPaint; | 
 |     SkPaint     fStrokePaint; | 
 |     SkPath      fPath; | 
 | }; | 
 |  | 
 | SkJSCanvas::SkJSCanvas(SkCanvas* target) : fTarget(target) { | 
 |     fFillPaint.setAntiAlias(true); | 
 |     fStrokePaint.setAntiAlias(true); | 
 |     fStrokePaint.setStyle(SkPaint::kStroke_Style); | 
 |     fStrokePaint.setStrokeWidth(SK_Scalar1); | 
 | } | 
 |  | 
 | SkJSCanvas::~SkJSCanvas() {} | 
 |  | 
 | void SkJSCanvas::save() { fTarget->save(); } | 
 | void SkJSCanvas::restore() { fTarget->restore(); } | 
 |  | 
 | void SkJSCanvas::beginPath() { fPath.reset(); } | 
 | void SkJSCanvas::moveTo(double x, double y) { | 
 |     fPath.moveTo(SkDoubleToScalar(x), SkDoubleToScalar(y)); | 
 | } | 
 |  | 
 | void SkJSCanvas::lineTo(double x, double y) { | 
 |     fPath.lineTo(SkDoubleToScalar(x), SkDoubleToScalar(y)); | 
 | } | 
 |  | 
 | void SkJSCanvas::closePath() { fPath.close(); } | 
 |  | 
 | void SkJSCanvas::fill() { | 
 |     fTarget->drawPath(fPath, fFillPaint); | 
 | } | 
 |  | 
 | void SkJSCanvas::stroke() { | 
 |     fStrokePaint.setStrokeWidth(SkDoubleToScalar(lineWidth)); | 
 |     fTarget->drawPath(fPath, fStrokePaint); | 
 | } | 
 |  | 
 | void SkJSCanvas::fillText(const char text[], double x, double y) { | 
 |     fTarget->drawText(text, strlen(text), | 
 |                       SkDoubleToScalar(x), SkDoubleToScalar(y), fFillPaint); | 
 | } | 
 |  | 
 | /////////////////////////////////////////////////////////////////////////////// | 
 |  | 
 | static void dump(const SkPath& path) { | 
 |     const SkRect& r = path.getBounds(); | 
 |     SkDebugf("isEmpty %d, bounds [%g %g %g %g]\n", path.isEmpty(), | 
 |              r.fLeft, r.fTop, r.fRight, r.fBottom); | 
 | } | 
 |  | 
 | static void test_stroke(SkCanvas* canvas) { | 
 |     if (true) { | 
 |         SkPath path; | 
 |         dump(path); | 
 |         path.reset(); path.moveTo(0, 0); | 
 |         dump(path); | 
 |         path.reset(); path.moveTo(100, 100); | 
 |         dump(path); | 
 |         path.reset(); path.moveTo(0, 0); path.moveTo(100, 100); | 
 |         dump(path); | 
 |         path.reset(); path.moveTo(0, 0); path.lineTo(100, 100); | 
 |         dump(path); | 
 |         path.reset(); path.moveTo(0, 0); path.lineTo(100, 100); path.moveTo(200, 200); | 
 |         dump(path); | 
 |     } | 
 |  | 
 | #if 0 | 
 |     // TEST 1 - The rectangle as it's expected to look | 
 |     var canvas = document.createElement('canvas'); | 
 |     document.body.appendChild(canvas); | 
 |     var ctx = canvas.getContext("2d"); | 
 | #else | 
 |     SkJSCanvas ctx(canvas); | 
 | #endif | 
 |  | 
 |     ctx.save(); | 
 |     ctx.lineWidth = 2; | 
 |     ctx.beginPath(); | 
 |     ctx.moveTo(10, 100); | 
 |     ctx.lineTo(150, 100); | 
 |     ctx.lineTo(150, 15); | 
 |     ctx.lineTo(10, 15); | 
 |     ctx.closePath(); | 
 |  | 
 |     // no extra moveTo here | 
 |     // ctx.moveTo(175, 125); | 
 |  | 
 |     ctx.stroke(); | 
 |     ctx.restore(); | 
 |  | 
 |     ctx.fillText("As Expected", 10, 10); | 
 |  | 
 | #if 0 | 
 |     // TEST 2 - Includes an extra moveTo call before stroke; the rectangle appears larger | 
 |     canvas = document.createElement('canvas'); | 
 |     document.body.appendChild(canvas); | 
 |     ctx = canvas.getContext("2d"); | 
 | #else | 
 |     canvas->translate(200, 0); | 
 | #endif | 
 |  | 
 |     ctx.save(); | 
 |     ctx.lineWidth = 2; | 
 |     ctx.beginPath(); | 
 |     ctx.moveTo(10, 100); | 
 |     ctx.lineTo(150, 100); | 
 |     ctx.lineTo(150, 15); | 
 |     ctx.lineTo(10, 15); | 
 |     ctx.closePath(); | 
 |  | 
 |     ctx.moveTo(175, 125); | 
 |  | 
 |     ctx.stroke(); | 
 |     ctx.restore(); | 
 |  | 
 |     ctx.fillText("Larger Rectangle", 10, 10); | 
 |  | 
 | #if 0 | 
 |     // TEST 3 - Identical to test 2 except the line width is 1 | 
 |     canvas = document.createElement('canvas'); | 
 |     document.body.appendChild(canvas); | 
 |     ctx = canvas.getContext("2d"); | 
 | #else | 
 |     canvas->translate(200, 0); | 
 | #endif | 
 |  | 
 |     ctx.save(); | 
 |     ctx.lineWidth = 1; | 
 |     ctx.beginPath(); | 
 |     ctx.moveTo(10, 100); | 
 |     ctx.lineTo(150, 100); | 
 |     ctx.lineTo(150, 15); | 
 |     ctx.lineTo(10, 15); | 
 |     ctx.closePath(); | 
 |  | 
 |     ctx.moveTo(175, 125); | 
 |  | 
 |     ctx.stroke(); | 
 |     ctx.restore(); | 
 |  | 
 |     ctx.fillText("As Expected - line width 1", 10, 10); | 
 | } | 
 |  | 
 | class Poly2PolyGM : public skiagm::GM { | 
 | public: | 
 |     Poly2PolyGM() {} | 
 |  | 
 | protected: | 
 |     virtual uint32_t onGetFlags() const SK_OVERRIDE { | 
 |         return kSkipTiled_Flag; | 
 |     } | 
 |  | 
 |     virtual SkString onShortName() SK_OVERRIDE { | 
 |         return SkString("poly2poly"); | 
 |     } | 
 |  | 
 |     virtual SkISize onISize() SK_OVERRIDE { | 
 |         return SkISize::Make(835, 840); | 
 |     } | 
 |  | 
 |     static void doDraw(SkCanvas* canvas, SkPaint* paint, const int isrc[], | 
 |                        const int idst[], int count) { | 
 |         SkMatrix matrix; | 
 |         SkPoint src[4], dst[4]; | 
 |  | 
 |         for (int i = 0; i < count; i++) { | 
 |             src[i].set(SkIntToScalar(isrc[2*i+0]), SkIntToScalar(isrc[2*i+1])); | 
 |             dst[i].set(SkIntToScalar(idst[2*i+0]), SkIntToScalar(idst[2*i+1])); | 
 |         } | 
 |  | 
 |         canvas->save(); | 
 |         matrix.setPolyToPoly(src, dst, count); | 
 |         canvas->concat(matrix); | 
 |  | 
 |         paint->setColor(SK_ColorGRAY); | 
 |         paint->setStyle(SkPaint::kStroke_Style); | 
 |         const SkScalar D = SkIntToScalar(64); | 
 |         canvas->drawRectCoords(0, 0, D, D, *paint); | 
 |         canvas->drawLine(0, 0, D, D, *paint); | 
 |         canvas->drawLine(0, D, D, 0, *paint); | 
 |  | 
 |         SkPaint::FontMetrics fm; | 
 |         paint->getFontMetrics(&fm); | 
 |         paint->setColor(SK_ColorRED); | 
 |         paint->setStyle(SkPaint::kFill_Style); | 
 |         SkScalar x = D/2; | 
 |         SkScalar y = D/2 - (fm.fAscent + fm.fDescent)/2; | 
 |         SkString str; | 
 |         str.appendS32(count); | 
 |         canvas->drawText(str.c_str(), str.size(), x, y, *paint); | 
 |  | 
 |         canvas->restore(); | 
 |     } | 
 |  | 
 |     virtual void onDraw(SkCanvas* canvas) SK_OVERRIDE { | 
 |         if (false) { test_stroke(canvas); return; } | 
 |  | 
 |         SkPaint paint; | 
 |         paint.setAntiAlias(true); | 
 |         paint.setStrokeWidth(SkIntToScalar(4)); | 
 |         paint.setTextSize(SkIntToScalar(40)); | 
 |         paint.setTextAlign(SkPaint::kCenter_Align); | 
 |  | 
 |         canvas->save(); | 
 |         canvas->translate(SkIntToScalar(10), SkIntToScalar(10)); | 
 |         // translate (1 point) | 
 |         const int src1[] = { 0, 0 }; | 
 |         const int dst1[] = { 5, 5 }; | 
 |         doDraw(canvas, &paint, src1, dst1, 1); | 
 |         canvas->restore(); | 
 |  | 
 |         canvas->save(); | 
 |         canvas->translate(SkIntToScalar(160), SkIntToScalar(10)); | 
 |         // rotate/uniform-scale (2 points) | 
 |         const int src2[] = { 32, 32, 64, 32 }; | 
 |         const int dst2[] = { 32, 32, 64, 48 }; | 
 |         doDraw(canvas, &paint, src2, dst2, 2); | 
 |         canvas->restore(); | 
 |  | 
 |         canvas->save(); | 
 |         canvas->translate(SkIntToScalar(10), SkIntToScalar(110)); | 
 |         // rotate/skew (3 points) | 
 |         const int src3[] = { 0, 0, 64, 0, 0, 64 }; | 
 |         const int dst3[] = { 0, 0, 96, 0, 24, 64 }; | 
 |         doDraw(canvas, &paint, src3, dst3, 3); | 
 |         canvas->restore(); | 
 |  | 
 |         canvas->save(); | 
 |         canvas->translate(SkIntToScalar(160), SkIntToScalar(110)); | 
 |         // perspective (4 points) | 
 |         const int src4[] = { 0, 0, 64, 0, 64, 64, 0, 64 }; | 
 |         const int dst4[] = { 0, 0, 96, 0, 64, 96, 0, 64 }; | 
 |         doDraw(canvas, &paint, src4, dst4, 4); | 
 |         canvas->restore(); | 
 |     } | 
 |  | 
 | private: | 
 |     typedef skiagm::GM INHERITED; | 
 | }; | 
 |  | 
 | ////////////////////////////////////////////////////////////////////////////// | 
 |  | 
 | DEF_GM( return new Poly2PolyGM; ) |