/*
* 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 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; )