skia2/tests/ShaderTest.cpp
reed 320a40d773 Always return ImageShader, even from SkShader::MakeBitmapShader
Lessons learned

1. ImageShader (correctly) always compresses (typically via PNG) during serialization. This has the surprise results of
- if the image was marked opaque, but has some non-opaque pixels (i.e. bug in blitter or caller), then compressing may "fix" those pixels, making the deserialized version draw differently. bug filed.
- 565 compressess/decompresses to 8888 (at least on Mac), which draws differently (esp. under some filters). bug filed.

2. BitmapShader did not enforce a copy for mutable bitmaps, but ImageShader does (since it creates an Image). Thus the former would see subsequent changes to the pixels after shader creation, while the latter does not, hence the change to the BlitRow test to avoid this modify-after-create pattern. I sure hope this prev. behavior was a bug/undefined-behavior, since this CL changes that.

BUG=skia:5595
GOLD_TRYBOT_URL= https://gold.skia.org/search?issue=2195893002

Review-Url: https://codereview.chromium.org/2195893002
2016-08-02 06:12:06 -07:00

58 lines
2.1 KiB
C++

/*
* Copyright 2016 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "Test.h"
#include "SkBitmap.h"
#include "SkImage.h"
#include "SkShader.h"
#include "SkSurface.h"
#include "SkData.h"
static void check_isabitmap(skiatest::Reporter* reporter, SkShader* shader,
int expectedW, int expectedH,
SkShader::TileMode expectedX, SkShader::TileMode expectedY,
const SkMatrix& expectedM) {
SkBitmap bm;
SkShader::TileMode tileModes[2];
SkMatrix localM;
REPORTER_ASSERT(reporter, shader->isABitmap(&bm, &localM, tileModes));
REPORTER_ASSERT(reporter, bm.width() == expectedW);
REPORTER_ASSERT(reporter, bm.height() == expectedH);
REPORTER_ASSERT(reporter, localM == expectedM);
REPORTER_ASSERT(reporter, tileModes[0] == expectedX);
REPORTER_ASSERT(reporter, tileModes[1] == expectedY);
// wack these so we don't get a false positive
localM.setScale(9999, -9999);
tileModes[0] = tileModes[1] = (SkShader::TileMode)99;
SkImage* image = shader->isAImage(&localM, tileModes);
REPORTER_ASSERT(reporter, image);
REPORTER_ASSERT(reporter, image->width() == expectedW);
REPORTER_ASSERT(reporter, image->height() == expectedH);
REPORTER_ASSERT(reporter, localM == expectedM);
REPORTER_ASSERT(reporter, tileModes[0] == expectedX);
REPORTER_ASSERT(reporter, tileModes[1] == expectedY);
}
DEF_TEST(Shader_isABitmap, reporter) {
const int W = 100;
const int H = 100;
SkBitmap bm;
bm.allocN32Pixels(W, H);
auto img = SkImage::MakeFromBitmap(bm);
const SkMatrix localM = SkMatrix::MakeScale(2, 3);
const SkShader::TileMode tmx = SkShader::kRepeat_TileMode;
const SkShader::TileMode tmy = SkShader::kMirror_TileMode;
auto shader0 = SkShader::MakeBitmapShader(bm, tmx, tmy, &localM);
auto shader1 = SkImage::MakeFromBitmap(bm)->makeShader(tmx, tmy, &localM);
check_isabitmap(reporter, shader0.get(), W, H, tmx, tmy, localM);
check_isabitmap(reporter, shader1.get(), W, H, tmx, tmy, localM);
}