| /* |
| * Copyright (C) 2020 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| // TODO(b/129481165): remove the #pragma below and fix conversion issues |
| #pragma clang diagnostic push |
| #pragma clang diagnostic ignored "-Wconversion" |
| |
| #include "LayerTransactionTest.h" |
| |
| namespace android { |
| |
| class EffectLayerTest : public LayerTransactionTest { |
| protected: |
| virtual void SetUp() { |
| LayerTransactionTest::SetUp(); |
| ASSERT_EQ(NO_ERROR, mClient->initCheck()); |
| |
| const auto ids = SurfaceComposerClient::getPhysicalDisplayIds(); |
| ASSERT_FALSE(ids.empty()); |
| const auto display = SurfaceComposerClient::getPhysicalDisplayToken(ids.front()); |
| ASSERT_FALSE(display == nullptr); |
| |
| mParentLayer = createColorLayer("Parent layer", Color::RED); |
| asTransaction([&](Transaction& t) { |
| t.setDisplayLayerStack(display, ui::DEFAULT_LAYER_STACK); |
| t.setLayer(mParentLayer, INT32_MAX - 2).show(mParentLayer); |
| t.setFlags(mParentLayer, layer_state_t::eLayerOpaque, layer_state_t::eLayerOpaque); |
| }); |
| } |
| |
| virtual void TearDown() { |
| LayerTransactionTest::TearDown(); |
| mParentLayer = 0; |
| } |
| |
| sp<SurfaceControl> mParentLayer; |
| }; |
| |
| TEST_F(EffectLayerTest, DefaultEffectLayerHasSolidBlackFill) { |
| sp<SurfaceControl> effectLayer = |
| mClient->createSurface(String8("Effect Layer"), 0 /* width */, 0 /* height */, |
| PIXEL_FORMAT_RGBA_8888, ISurfaceComposerClient::eFXSurfaceEffect, |
| mParentLayer->getHandle()); |
| |
| EXPECT_NE(nullptr, effectLayer.get()) << "failed to create SurfaceControl"; |
| asTransaction([&](Transaction& t) { |
| t.setCrop(effectLayer, Rect(0, 0, 400, 400)); |
| t.show(effectLayer); |
| }); |
| |
| { |
| SCOPED_TRACE("Default effect Layer has solid black fill"); |
| auto shot = screenshot(); |
| shot->expectColor(Rect(0, 0, 400, 400), Color::BLACK); |
| } |
| } |
| |
| TEST_F(EffectLayerTest, EffectLayerWithNoFill) { |
| sp<SurfaceControl> effectLayer = |
| mClient->createSurface(String8("Effect Layer"), 0 /* width */, 0 /* height */, |
| PIXEL_FORMAT_RGBA_8888, |
| ISurfaceComposerClient::eFXSurfaceEffect | |
| ISurfaceComposerClient::eNoColorFill, |
| mParentLayer->getHandle()); |
| |
| EXPECT_NE(nullptr, effectLayer.get()) << "failed to create SurfaceControl"; |
| asTransaction([&](Transaction& t) { |
| t.setCrop(effectLayer, Rect(0, 0, 400, 400)); |
| t.show(effectLayer); |
| }); |
| |
| { |
| SCOPED_TRACE("Effect layer with nofill option is transparent"); |
| auto shot = screenshot(); |
| shot->expectColor(Rect(0, 0, 400, 400), Color::RED); |
| } |
| } |
| |
| TEST_F(EffectLayerTest, EffectLayerCanSetColor) { |
| sp<SurfaceControl> effectLayer = |
| mClient->createSurface(String8("Effect Layer"), 0 /* width */, 0 /* height */, |
| PIXEL_FORMAT_RGBA_8888, |
| ISurfaceComposerClient::eFXSurfaceEffect | |
| ISurfaceComposerClient::eNoColorFill, |
| mParentLayer->getHandle()); |
| |
| EXPECT_NE(nullptr, effectLayer.get()) << "failed to create SurfaceControl"; |
| asTransaction([&](Transaction& t) { |
| t.setCrop(effectLayer, Rect(0, 0, 400, 400)); |
| t.setColor(effectLayer, |
| half3{Color::GREEN.r / 255.0f, Color::GREEN.g / 255.0f, |
| Color::GREEN.b / 255.0f}); |
| t.show(effectLayer); |
| }); |
| |
| { |
| SCOPED_TRACE("Effect Layer can set color"); |
| auto shot = screenshot(); |
| shot->expectColor(Rect(0, 0, 400, 400), Color::GREEN); |
| } |
| } |
| |
| TEST_F(EffectLayerTest, BlurEffectLayerIsVisible) { |
| if (!deviceSupportsBlurs()) GTEST_SKIP(); |
| if (!deviceUsesSkiaRenderEngine()) GTEST_SKIP(); |
| |
| const auto canvasSize = 256; |
| |
| sp<SurfaceControl> leftLayer = createColorLayer("Left", Color::BLUE); |
| sp<SurfaceControl> rightLayer = createColorLayer("Right", Color::RED); |
| sp<SurfaceControl> blurLayer; |
| const auto leftRect = Rect(0, 0, canvasSize / 2, canvasSize); |
| const auto rightRect = Rect(canvasSize / 2, 0, canvasSize, canvasSize); |
| const auto blurRect = Rect(0, 0, canvasSize, canvasSize); |
| |
| asTransaction([&](Transaction& t) { |
| t.setLayer(leftLayer, mLayerZBase + 1); |
| t.reparent(leftLayer, mParentLayer); |
| t.setCrop(leftLayer, leftRect); |
| t.setLayer(rightLayer, mLayerZBase + 2); |
| t.reparent(rightLayer, mParentLayer); |
| t.setCrop(rightLayer, rightRect); |
| t.show(leftLayer); |
| t.show(rightLayer); |
| }); |
| |
| { |
| auto shot = screenshot(); |
| shot->expectColor(leftRect, Color::BLUE); |
| shot->expectColor(rightRect, Color::RED); |
| } |
| |
| ASSERT_NO_FATAL_FAILURE(blurLayer = createColorLayer("BackgroundBlur", Color::TRANSPARENT)); |
| |
| const auto blurRadius = canvasSize / 4; |
| asTransaction([&](Transaction& t) { |
| t.setLayer(blurLayer, mLayerZBase + 3); |
| t.reparent(blurLayer, mParentLayer); |
| t.setBackgroundBlurRadius(blurLayer, blurRadius); |
| t.setCrop(blurLayer, blurRect); |
| t.setAlpha(blurLayer, 0.0f); |
| t.show(blurLayer); |
| }); |
| |
| { |
| auto shot = screenshot(); |
| |
| const auto stepSize = 1; |
| const auto expectedBlurAreaSize = blurRadius * 1.5f; |
| const auto blurAreaStartX = canvasSize / 2 - expectedBlurAreaSize / 2; |
| const auto blurAreaEndX = canvasSize / 2 + expectedBlurAreaSize / 2; |
| // testAreaEndY is needed because the setBackgroundBlurRadius API blurs everything behind |
| // the surface, which means it samples pixels from outside the canvasSize and we get some |
| // unexpected colors in the screenshot. |
| const auto testAreaEndY = canvasSize - blurRadius * 2; |
| |
| Color previousColor; |
| Color currentColor; |
| for (int y = 0; y < testAreaEndY; y++) { |
| shot->checkPixel(0, y, /* r = */ 0, /* g = */ 0, /* b = */ 255); |
| previousColor = shot->getPixelColor(0, y); |
| for (int x = blurAreaStartX; x < blurAreaEndX; x += stepSize) { |
| currentColor = shot->getPixelColor(x, y); |
| ASSERT_GT(currentColor.r, previousColor.r); |
| ASSERT_LT(currentColor.b, previousColor.b); |
| ASSERT_EQ(0, currentColor.g); |
| } |
| shot->checkPixel(canvasSize - 1, y, 255, 0, 0); |
| } |
| } |
| } |
| |
| TEST_F(EffectLayerTest, EffectLayerWithColorNoCrop) { |
| const auto ids = SurfaceComposerClient::getPhysicalDisplayIds(); |
| ASSERT_FALSE(ids.empty()); |
| const auto display = SurfaceComposerClient::getPhysicalDisplayToken(ids.front()); |
| ui::DisplayMode mode; |
| ASSERT_EQ(NO_ERROR, SurfaceComposerClient::getActiveDisplayMode(display, &mode)); |
| const ui::Size& resolution = mode.resolution; |
| auto shot = screenshot(); |
| shot->expectColor(Rect(0, 0, resolution.getWidth(), resolution.getHeight()), Color::RED); |
| } |
| |
| } // namespace android |
| |
| // TODO(b/129481165): remove the #pragma below and fix conversion issues |
| #pragma clang diagnostic pop // ignored "-Wconversion" |