2011-12-10 19:36:56 +00:00
|
|
|
/*
|
|
|
|
* 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"
|
|
|
|
#include "SkCanvas.h"
|
|
|
|
#include "SkColorPriv.h"
|
|
|
|
#include "SkShader.h"
|
|
|
|
|
|
|
|
#include "SkArithmeticMode.h"
|
|
|
|
#include "SkGradientShader.h"
|
|
|
|
#define WW 100
|
|
|
|
#define HH 32
|
|
|
|
|
|
|
|
static SkBitmap make_bm() {
|
|
|
|
SkBitmap bm;
|
2014-01-25 16:46:20 +00:00
|
|
|
bm.allocN32Pixels(WW, HH);
|
2012-12-06 21:47:40 +00:00
|
|
|
bm.eraseColor(SK_ColorTRANSPARENT);
|
2011-12-10 19:36:56 +00:00
|
|
|
return bm;
|
|
|
|
}
|
|
|
|
|
|
|
|
static SkBitmap make_src() {
|
|
|
|
SkBitmap bm = make_bm();
|
|
|
|
SkCanvas canvas(bm);
|
|
|
|
SkPaint paint;
|
2012-01-06 19:22:11 +00:00
|
|
|
SkPoint pts[] = { {0, 0}, {SkIntToScalar(WW), SkIntToScalar(HH)} };
|
2011-12-10 19:36:56 +00:00
|
|
|
SkColor colors[] = {
|
2013-05-28 19:43:05 +00:00
|
|
|
SK_ColorTRANSPARENT, SK_ColorGREEN, SK_ColorCYAN,
|
|
|
|
SK_ColorRED, SK_ColorMAGENTA, SK_ColorWHITE,
|
2011-12-10 19:36:56 +00:00
|
|
|
};
|
2016-03-09 02:50:00 +00:00
|
|
|
paint.setShader(SkGradientShader::MakeLinear(pts, colors, nullptr, SK_ARRAY_COUNT(colors),
|
|
|
|
SkShader::kClamp_TileMode));
|
2011-12-10 19:36:56 +00:00
|
|
|
canvas.drawPaint(paint);
|
|
|
|
return bm;
|
|
|
|
}
|
|
|
|
|
|
|
|
static SkBitmap make_dst() {
|
|
|
|
SkBitmap bm = make_bm();
|
|
|
|
SkCanvas canvas(bm);
|
|
|
|
SkPaint paint;
|
2012-01-06 19:22:11 +00:00
|
|
|
SkPoint pts[] = { {0, SkIntToScalar(HH)}, {SkIntToScalar(WW), 0} };
|
2011-12-10 19:36:56 +00:00
|
|
|
SkColor colors[] = {
|
2015-07-17 16:39:16 +00:00
|
|
|
SK_ColorBLUE, SK_ColorYELLOW, SK_ColorBLACK, SK_ColorGREEN,
|
|
|
|
sk_tool_utils::color_to_565(SK_ColorGRAY)
|
2011-12-10 19:36:56 +00:00
|
|
|
};
|
2016-03-09 02:50:00 +00:00
|
|
|
paint.setShader(SkGradientShader::MakeLinear(pts, colors, nullptr, SK_ARRAY_COUNT(colors),
|
|
|
|
SkShader::kClamp_TileMode));
|
2011-12-10 19:36:56 +00:00
|
|
|
canvas.drawPaint(paint);
|
|
|
|
return bm;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void show_k_text(SkCanvas* canvas, SkScalar x, SkScalar y, const SkScalar k[]) {
|
|
|
|
SkPaint paint;
|
2011-12-12 22:01:06 +00:00
|
|
|
paint.setTextSize(SkIntToScalar(24));
|
2011-12-10 19:36:56 +00:00
|
|
|
paint.setAntiAlias(true);
|
2015-07-24 19:09:25 +00:00
|
|
|
sk_tool_utils::set_portable_typeface(&paint);
|
2011-12-10 19:36:56 +00:00
|
|
|
for (int i = 0; i < 4; ++i) {
|
|
|
|
SkString str;
|
|
|
|
str.appendScalar(k[i]);
|
|
|
|
SkScalar width = paint.measureText(str.c_str(), str.size());
|
2011-12-12 22:01:06 +00:00
|
|
|
canvas->drawText(str.c_str(), str.size(), x, y + paint.getTextSize(), paint);
|
|
|
|
x += width + SkIntToScalar(10);
|
2011-12-10 19:36:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class ArithmodeGM : public skiagm::GM {
|
|
|
|
public:
|
|
|
|
ArithmodeGM () {}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
|
|
|
virtual SkString onShortName() {
|
|
|
|
return SkString("arithmode");
|
|
|
|
}
|
|
|
|
|
2015-10-19 21:41:11 +00:00
|
|
|
virtual SkISize onISize() { return SkISize::Make(640, 572); }
|
2011-12-10 19:36:56 +00:00
|
|
|
|
|
|
|
virtual void onDraw(SkCanvas* canvas) {
|
|
|
|
SkBitmap src = make_src();
|
|
|
|
SkBitmap dst = make_dst();
|
2012-08-23 18:19:56 +00:00
|
|
|
|
2016-09-01 18:24:54 +00:00
|
|
|
constexpr SkScalar one = SK_Scalar1;
|
|
|
|
constexpr SkScalar K[] = {
|
2011-12-10 19:36:56 +00:00
|
|
|
0, 0, 0, 0,
|
2011-12-12 22:01:06 +00:00
|
|
|
0, 0, 0, one,
|
|
|
|
0, one, 0, 0,
|
|
|
|
0, 0, one, 0,
|
|
|
|
0, one, one, 0,
|
|
|
|
0, one, -one, 0,
|
|
|
|
0, one/2, one/2, 0,
|
|
|
|
0, one/2, one/2, one/4,
|
|
|
|
0, one/2, one/2, -one/4,
|
|
|
|
one/4, one/2, one/2, 0,
|
|
|
|
-one/4, one/2, one/2, 0,
|
2011-12-10 19:36:56 +00:00
|
|
|
};
|
2012-08-23 18:19:56 +00:00
|
|
|
|
2011-12-10 19:36:56 +00:00
|
|
|
const SkScalar* k = K;
|
|
|
|
const SkScalar* stop = k + SK_ARRAY_COUNT(K);
|
|
|
|
SkScalar y = 0;
|
2011-12-12 22:15:18 +00:00
|
|
|
SkScalar gap = SkIntToScalar(src.width() + 20);
|
2011-12-10 19:36:56 +00:00
|
|
|
while (k < stop) {
|
|
|
|
SkScalar x = 0;
|
2015-08-27 14:41:13 +00:00
|
|
|
canvas->drawBitmap(src, x, y, nullptr);
|
2011-12-10 19:36:56 +00:00
|
|
|
x += gap;
|
2015-08-27 14:41:13 +00:00
|
|
|
canvas->drawBitmap(dst, x, y, nullptr);
|
2011-12-10 19:36:56 +00:00
|
|
|
x += gap;
|
2013-05-29 20:55:09 +00:00
|
|
|
SkRect rect = SkRect::MakeXYWH(x, y, SkIntToScalar(WW), SkIntToScalar(HH));
|
2015-08-27 14:41:13 +00:00
|
|
|
canvas->saveLayer(&rect, nullptr);
|
|
|
|
canvas->drawBitmap(dst, x, y, nullptr);
|
2013-05-29 20:55:09 +00:00
|
|
|
SkPaint paint;
|
2016-03-29 18:32:50 +00:00
|
|
|
paint.setXfermode(SkArithmeticMode::Make(k[0], k[1], k[2], k[3]));
|
2013-05-29 20:55:09 +00:00
|
|
|
canvas->drawBitmap(src, x, y, &paint);
|
|
|
|
canvas->restore();
|
2011-12-10 19:36:56 +00:00
|
|
|
x += gap;
|
|
|
|
show_k_text(canvas, x, y, k);
|
|
|
|
k += 4;
|
2011-12-12 22:15:18 +00:00
|
|
|
y += SkIntToScalar(src.height() + 12);
|
2011-12-10 19:36:56 +00:00
|
|
|
}
|
2015-10-19 21:41:11 +00:00
|
|
|
|
|
|
|
// Draw two special cases to test enforcePMColor. In these cases, we
|
|
|
|
// draw the dst bitmap twice, the first time it is halved and inverted,
|
|
|
|
// leading to invalid premultiplied colors. If we enforcePMColor, these
|
|
|
|
// invalid values should be clamped, and will not contribute to the
|
|
|
|
// second draw.
|
|
|
|
for (int i = 0; i < 2; i++) {
|
|
|
|
const bool enforcePMColor = (i == 0);
|
|
|
|
SkScalar x = gap;
|
|
|
|
canvas->drawBitmap(dst, x, y, nullptr);
|
|
|
|
x += gap;
|
|
|
|
SkRect rect = SkRect::MakeXYWH(x, y, SkIntToScalar(WW), SkIntToScalar(HH));
|
|
|
|
canvas->saveLayer(&rect, nullptr);
|
|
|
|
SkPaint paint1;
|
2016-03-29 18:32:50 +00:00
|
|
|
paint1.setXfermode(SkArithmeticMode::Make(0, -one / 2, 0, 1, enforcePMColor));
|
2015-10-19 21:41:11 +00:00
|
|
|
canvas->drawBitmap(dst, x, y, &paint1);
|
|
|
|
SkPaint paint2;
|
2016-03-29 18:32:50 +00:00
|
|
|
paint2.setXfermode(SkArithmeticMode::Make(0, one / 2, -one, 1));
|
2015-10-19 21:41:11 +00:00
|
|
|
canvas->drawBitmap(dst, x, y, &paint2);
|
|
|
|
canvas->restore();
|
|
|
|
x += gap;
|
|
|
|
|
|
|
|
// Label
|
|
|
|
SkPaint paint;
|
|
|
|
paint.setTextSize(SkIntToScalar(24));
|
|
|
|
paint.setAntiAlias(true);
|
|
|
|
sk_tool_utils::set_portable_typeface(&paint);
|
|
|
|
SkString str(enforcePMColor ? "enforcePM" : "no enforcePM");
|
|
|
|
canvas->drawText(str.c_str(), str.size(), x, y + paint.getTextSize(), paint);
|
|
|
|
|
|
|
|
y += SkIntToScalar(src.height() + 12);
|
|
|
|
}
|
2011-12-10 19:36:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
typedef GM INHERITED;
|
|
|
|
};
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2015-12-10 21:31:59 +00:00
|
|
|
DEF_GM( return new ArithmodeGM; )
|