| /* |
| * 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 "include/core/SkBitmap.h" |
| #include "include/core/SkBlendMode.h" |
| #include "include/core/SkCanvas.h" |
| #include "include/core/SkColor.h" |
| #include "include/core/SkImageFilter.h" |
| #include "include/core/SkImageInfo.h" |
| #include "include/core/SkPaint.h" |
| #include "include/core/SkPoint.h" |
| #include "include/core/SkRect.h" |
| #include "include/core/SkRefCnt.h" |
| #include "include/core/SkScalar.h" |
| #include "include/core/SkShader.h" |
| #include "include/core/SkString.h" |
| #include "include/core/SkSurface.h" |
| #include "include/core/SkTileMode.h" |
| #include "include/core/SkTypes.h" |
| #include "include/effects/SkGradientShader.h" |
| #include "include/effects/SkImageFilters.h" |
| #include "include/effects/SkRuntimeEffect.h" |
| #include "include/gpu/GpuTypes.h" |
| #include "include/gpu/GrDirectContext.h" |
| #include "include/gpu/ganesh/SkSurfaceGanesh.h" |
| #include "tests/CtsEnforcement.h" |
| #include "tests/Test.h" |
| #include "tools/EncodeUtils.h" |
| #include "tools/ToolUtils.h" |
| |
| #include <vector> |
| |
| struct GrContextOptions; |
| |
| static void test_unscaled(skiatest::Reporter* reporter) { |
| static const int kWidth = 10; |
| static const int kHeight = 10; |
| |
| SkIRect ir = SkIRect::MakeWH(kWidth, kHeight); |
| |
| SkBitmap filterResult, paintResult; |
| |
| filterResult.allocN32Pixels(kWidth, kHeight); |
| SkCanvas canvasFilter(filterResult); |
| canvasFilter.clear(0x00000000); |
| |
| paintResult.allocN32Pixels(kWidth, kHeight); |
| SkCanvas canvasPaint(paintResult); |
| canvasPaint.clear(0x00000000); |
| |
| SkPoint center = SkPoint::Make(SkIntToScalar(5), SkIntToScalar(5)); |
| SkColor colors[] = {SK_ColorBLUE, SK_ColorRED, SK_ColorGREEN}; |
| SkScalar pos[] = {0, SK_ScalarHalf, SK_Scalar1}; |
| SkScalar radius = SkIntToScalar(5); |
| |
| sk_sp<SkShader> gradient = SkGradientShader::MakeRadial( |
| center, radius, colors, pos, std::size(colors), SkTileMode::kClamp); |
| |
| // Test using the image filter |
| { |
| SkPaint paint; |
| paint.setImageFilter(SkImageFilters::Shader(gradient, &ir)); |
| canvasFilter.drawRect(SkRect::Make(ir), paint); |
| } |
| |
| // Test using the paint directly |
| { |
| SkPaint paint; |
| paint.setShader(gradient); |
| canvasPaint.drawRect(SkRect::Make(ir), paint); |
| } |
| |
| // Assert that both paths yielded the same result |
| for (int y = 0; y < kHeight; ++y) { |
| const SkPMColor* filterPtr = filterResult.getAddr32(0, y); |
| const SkPMColor* paintPtr = paintResult.getAddr32(0, y); |
| for (int x = 0; x < kWidth; ++x, ++filterPtr, ++paintPtr) { |
| REPORTER_ASSERT(reporter, *filterPtr == *paintPtr); |
| } |
| } |
| } |
| |
| static void test_scaled(skiatest::Reporter* reporter) { |
| static const int kWidth = 10; |
| static const int kHeight = 10; |
| |
| SkIRect ir = SkIRect::MakeWH(kWidth, kHeight); |
| |
| SkBitmap filterResult, paintResult; |
| |
| filterResult.allocN32Pixels(kWidth, kHeight); |
| SkCanvas canvasFilter(filterResult); |
| canvasFilter.clear(0x00000000); |
| |
| paintResult.allocN32Pixels(kWidth, kHeight); |
| SkCanvas canvasPaint(paintResult); |
| canvasPaint.clear(0x00000000); |
| |
| SkPoint center = SkPoint::Make(SkIntToScalar(5), SkIntToScalar(5)); |
| SkColor colors[] = {SK_ColorBLUE, SK_ColorRED, SK_ColorGREEN}; |
| SkScalar pos[] = {0, SK_ScalarHalf, SK_Scalar1}; |
| SkScalar radius = SkIntToScalar(5); |
| |
| sk_sp<SkShader> gradient = SkGradientShader::MakeRadial( |
| center, radius, colors, pos, std::size(colors), SkTileMode::kClamp); |
| |
| // Test using the image filter |
| { |
| SkPaint paint; |
| paint.setImageFilter(SkImageFilters::Shader(gradient, &ir)); |
| canvasFilter.scale(SkIntToScalar(2), SkIntToScalar(2)); |
| canvasFilter.drawRect(SkRect::Make(ir), paint); |
| } |
| |
| // Test using the paint directly |
| { |
| SkPaint paint; |
| paint.setShader(gradient); |
| canvasPaint.scale(SkIntToScalar(2), SkIntToScalar(2)); |
| canvasPaint.drawRect(SkRect::Make(ir), paint); |
| } |
| |
| // Assert that both paths yielded the same result |
| if (!ToolUtils::equal_pixels(filterResult, paintResult)) { |
| SkString encoded; |
| SkString errString("Image filter doesn't match paint reference"); |
| errString.append("\nExpected: "); |
| if (ToolUtils::BitmapToBase64DataURI(paintResult, &encoded)) { |
| errString.append(encoded); |
| } else { |
| errString.append("failed to encode"); |
| } |
| |
| errString.append("\nActual: "); |
| if (ToolUtils::BitmapToBase64DataURI(filterResult, &encoded)) { |
| errString.append(encoded); |
| } else { |
| errString.append("failed to encode"); |
| } |
| |
| ERRORF(reporter, "%s\n", errString.c_str()); |
| } |
| } |
| |
| DEF_TEST(ShaderImageFilter, reporter) { |
| test_unscaled(reporter); |
| test_scaled(reporter); |
| } |
| |
| static void test_runtime_shader(skiatest::Reporter* r, SkSurface* surface) { |
| sk_sp<SkRuntimeEffect> effect = SkRuntimeEffect::MakeForShader(SkString(R"( |
| uniform shader child; |
| vec4 main(vec2 coord) { |
| return child.eval(coord) * 0.5; |
| } |
| )")) |
| .effect; |
| SkRuntimeShaderBuilder builder(effect); |
| |
| // create a red image filter to feed as input into the SkImageFilters::RuntimeShader |
| sk_sp<SkImageFilter> input = SkImageFilters::Shader(SkShaders::Color(SK_ColorRED)); |
| |
| // Create the different variations of SkImageFilters::RuntimeShader |
| // All variations should produce the same pixel output |
| std::vector<sk_sp<SkImageFilter>> filters = { |
| SkImageFilters::RuntimeShader(builder, /*childShaderName=*/"", input), |
| SkImageFilters::RuntimeShader(builder, /*childShaderName=*/"child", input)}; |
| |
| for (auto&& filter : filters) { |
| auto canvas = surface->getCanvas(); |
| |
| // clear to transparent |
| SkPaint paint; |
| paint.setColor(SK_ColorTRANSPARENT); |
| paint.setBlendMode(SkBlendMode::kSrc); |
| canvas->drawPaint(paint); |
| |
| SkPaint filterPaint; |
| // the green color will be ignored by the filter within the runtime shader |
| filterPaint.setColor(SK_ColorGREEN); |
| filterPaint.setImageFilter(filter); |
| canvas->saveLayer(nullptr, &filterPaint); |
| // the blue color will be ignored by the filter because the input to the image filter is not |
| // null |
| canvas->drawColor(SK_ColorBLUE); |
| canvas->restore(); |
| |
| // This is expected to read back the half transparent red pixel produced by the image filter |
| SkBitmap bitmap; |
| REPORTER_ASSERT(r, bitmap.tryAllocPixels(surface->imageInfo())); |
| REPORTER_ASSERT(r, |
| surface->readPixels(bitmap.info(), |
| bitmap.getPixels(), |
| bitmap.rowBytes(), |
| /*srcX=*/0, |
| /*srcY=*/0)); |
| SkColor color = bitmap.getColor(/*x=*/0, /*y=*/0); |
| |
| // check alpha with a small tolerance |
| SkAlpha alpha = SkColorGetA(color); |
| REPORTER_ASSERT(r, alpha >= 127 && alpha <= 129, "Expected: %d Actual: %d", 128, alpha); |
| |
| // check each color channel |
| color = SkColorSetA(color, 255); |
| REPORTER_ASSERT(r, SK_ColorRED == color, "Expected: %08x Actual: %08x", SK_ColorRED, color); |
| } |
| } |
| |
| DEF_TEST(SkRuntimeShaderImageFilter_CPU, r) { |
| const SkImageInfo info = SkImageInfo::MakeN32Premul(/*width=*/1, /*height=*/1); |
| sk_sp<SkSurface> surface(SkSurfaces::Raster(info)); |
| test_runtime_shader(r, surface.get()); |
| } |
| |
| DEF_GANESH_TEST_FOR_RENDERING_CONTEXTS(SkRuntimeShaderImageFilter_GPU, |
| r, |
| ctxInfo, |
| CtsEnforcement::kApiLevel_T) { |
| const SkImageInfo info = SkImageInfo::MakeN32Premul(/*width=*/1, /*height=*/1); |
| sk_sp<SkSurface> surface( |
| SkSurfaces::RenderTarget(ctxInfo.directContext(), skgpu::Budgeted::kNo, info)); |
| test_runtime_shader(r, surface.get()); |
| } |