| // Copyright 2010 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include <stdint.h> |
| |
| #include "base/memory/platform_shared_memory_region.h" |
| #include "build/build_config.h" |
| #include "skia/ext/platform_canvas.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "ui/gfx/blit.h" |
| #include "ui/gfx/geometry/point.h" |
| #include "ui/gfx/geometry/rect.h" |
| |
| namespace { |
| |
| // A TestPixelMap is a thin shim layer that allows writing a square canvas of |
| // 32-bit pixels as a flat array of 8-bit value, which makes the tests a bit |
| // less fiddly to read and maintain. At construction time, each 8-bit value is |
| // used to fill in each of the ARGB channels of the 32-bit pixels. |
| struct TestPixelMap { |
| static constexpr size_t kWidth = 5; |
| static constexpr size_t kHeight = 5; |
| |
| static uint32_t ExpandByte(uint8_t b) { |
| return (b << 24) | (b << 16) | (b << 8) | b; |
| } |
| |
| TestPixelMap(std::initializer_list<uint8_t> values) { |
| std::transform(values.begin(), values.end(), pixels.begin(), |
| &TestPixelMap::ExpandByte); |
| } |
| |
| uint32_t AtXY(size_t x, size_t y) const { return pixels[y * kWidth + x]; } |
| |
| std::array<uint32_t, kWidth * kHeight> pixels; |
| }; |
| |
| // Fills the given canvas with the values by duplicating the values into each |
| // color channel for the corresponding pixel. |
| // |
| // Example values = {{0x0, 0x01}, {0x12, 0xFF}} would give a canvas with: |
| // 0x00000000 0x01010101 |
| // 0x12121212 0xFFFFFFFF |
| void SetToCanvas(SkCanvas* canvas, const TestPixelMap& values) { |
| ASSERT_EQ(TestPixelMap::kHeight, |
| base::checked_cast<size_t>(canvas->imageInfo().height())); |
| ASSERT_EQ(TestPixelMap::kWidth, |
| base::checked_cast<size_t>(canvas->imageInfo().width())); |
| |
| SkImageInfo info = |
| SkImageInfo::MakeN32Premul(TestPixelMap::kWidth, TestPixelMap::kHeight); |
| canvas->writePixels(info, values.pixels.data(), TestPixelMap::kWidth * 4, 0, |
| 0); |
| } |
| |
| // Checks each pixel in the given canvas and see if it is made up of the given |
| // values, where each value has been duplicated into each channel of the given |
| // bitmap (see SetToCanvas above). |
| void VerifyCanvasValues(SkCanvas* canvas, const TestPixelMap& values) { |
| SkBitmap bitmap = skia::ReadPixels(canvas); |
| ASSERT_EQ(TestPixelMap::kHeight, base::checked_cast<size_t>(bitmap.height())); |
| ASSERT_EQ(TestPixelMap::kWidth, base::checked_cast<size_t>(bitmap.width())); |
| |
| for (size_t y = 0; y < TestPixelMap::kHeight; y++) { |
| for (size_t x = 0; x < TestPixelMap::kWidth; x++) { |
| ASSERT_EQ(values.AtXY(x, y), *bitmap.getAddr32(x, y)); |
| } |
| } |
| } |
| |
| } // namespace |
| |
| TEST(Blit, ScrollCanvas) { |
| const int kCanvasWidth = 5; |
| const int kCanvasHeight = 5; |
| std::unique_ptr<SkCanvas> canvas = |
| skia::CreatePlatformCanvas(kCanvasWidth, kCanvasHeight, false); |
| const TestPixelMap initial_values({0x00, 0x01, 0x02, 0x03, 0x04, 0x10, 0x11, |
| 0x12, 0x13, 0x14, 0x20, 0x21, 0x22, 0x23, |
| 0x24, 0x30, 0x31, 0x32, 0x33, 0x34, 0x40, |
| 0x41, 0x42, 0x43, 0x44}); |
| |
| SetToCanvas(canvas.get(), initial_values); |
| VerifyCanvasValues(canvas.get(), initial_values); |
| |
| // Scroll none and make sure it's a NOP. |
| gfx::ScrollCanvas(canvas.get(), |
| gfx::Rect(0, 0, kCanvasWidth, kCanvasHeight), |
| gfx::Vector2d(0, 0)); |
| VerifyCanvasValues(canvas.get(), initial_values); |
| |
| // Scroll with a empty clip and make sure it's a NOP. |
| gfx::Rect empty_clip(1, 1, 0, 0); |
| gfx::ScrollCanvas(canvas.get(), empty_clip, gfx::Vector2d(0, 1)); |
| VerifyCanvasValues(canvas.get(), initial_values); |
| |
| // Scroll the center 3 pixels up one. |
| gfx::Rect center_three(1, 1, 3, 3); |
| gfx::ScrollCanvas(canvas.get(), center_three, gfx::Vector2d(0, -1)); |
| const TestPixelMap scroll_up_expected( |
| {0x00, 0x01, 0x02, 0x03, 0x04, 0x10, 0x21, 0x22, 0x23, |
| 0x14, 0x20, 0x31, 0x32, 0x33, 0x24, 0x30, 0x31, 0x32, |
| 0x33, 0x34, 0x40, 0x41, 0x42, 0x43, 0x44}); |
| VerifyCanvasValues(canvas.get(), scroll_up_expected); |
| |
| // Reset and scroll the center 3 pixels down one. |
| SetToCanvas(canvas.get(), initial_values); |
| gfx::ScrollCanvas(canvas.get(), center_three, gfx::Vector2d(0, 1)); |
| const TestPixelMap scroll_down_expected( |
| {0x00, 0x01, 0x02, 0x03, 0x04, 0x10, 0x11, 0x12, 0x13, |
| 0x14, 0x20, 0x11, 0x12, 0x13, 0x24, 0x30, 0x21, 0x22, |
| 0x23, 0x34, 0x40, 0x41, 0x42, 0x43, 0x44}); |
| VerifyCanvasValues(canvas.get(), scroll_down_expected); |
| |
| // Reset and scroll the center 3 pixels right one. |
| SetToCanvas(canvas.get(), initial_values); |
| gfx::ScrollCanvas(canvas.get(), center_three, gfx::Vector2d(1, 0)); |
| const TestPixelMap scroll_right_expected( |
| {0x00, 0x01, 0x02, 0x03, 0x04, 0x10, 0x11, 0x11, 0x12, |
| 0x14, 0x20, 0x21, 0x21, 0x22, 0x24, 0x30, 0x31, 0x31, |
| 0x32, 0x34, 0x40, 0x41, 0x42, 0x43, 0x44}); |
| VerifyCanvasValues(canvas.get(), scroll_right_expected); |
| |
| // Reset and scroll the center 3 pixels left one. |
| SetToCanvas(canvas.get(), initial_values); |
| gfx::ScrollCanvas(canvas.get(), center_three, gfx::Vector2d(-1, 0)); |
| const TestPixelMap scroll_left_expected( |
| {0x00, 0x01, 0x02, 0x03, 0x04, 0x10, 0x12, 0x13, 0x13, |
| 0x14, 0x20, 0x22, 0x23, 0x23, 0x24, 0x30, 0x32, 0x33, |
| 0x33, 0x34, 0x40, 0x41, 0x42, 0x43, 0x44}); |
| VerifyCanvasValues(canvas.get(), scroll_left_expected); |
| |
| // Diagonal scroll. |
| SetToCanvas(canvas.get(), initial_values); |
| gfx::ScrollCanvas(canvas.get(), center_three, gfx::Vector2d(2, 2)); |
| const TestPixelMap scroll_diagonal_expected( |
| {0x00, 0x01, 0x02, 0x03, 0x04, 0x10, 0x11, 0x12, 0x13, |
| 0x14, 0x20, 0x21, 0x22, 0x23, 0x24, 0x30, 0x31, 0x32, |
| 0x11, 0x34, 0x40, 0x41, 0x42, 0x43, 0x44}); |
| VerifyCanvasValues(canvas.get(), scroll_diagonal_expected); |
| } |
| |
| #if BUILDFLAG(IS_WIN) |
| |
| TEST(Blit, WithSharedMemory) { |
| const int kCanvasWidth = 5; |
| const int kCanvasHeight = 5; |
| base::subtle::PlatformSharedMemoryRegion section = |
| base::subtle::PlatformSharedMemoryRegion::CreateWritable(kCanvasWidth * |
| kCanvasHeight); |
| ASSERT_TRUE(section.IsValid()); |
| std::unique_ptr<SkCanvas> canvas = |
| skia::CreatePlatformCanvasWithSharedSection( |
| kCanvasWidth, kCanvasHeight, false, section.GetPlatformHandle(), |
| skia::RETURN_NULL_ON_FAILURE); |
| ASSERT_TRUE(canvas); |
| // Closes a HANDLE associated with |section|, |canvas| must remain valid. |
| section = base::subtle::PlatformSharedMemoryRegion(); |
| |
| const TestPixelMap initial_values({0x00, 0x01, 0x02, 0x03, 0x04, 0x10, 0x11, |
| 0x12, 0x13, 0x14, 0x20, 0x21, 0x22, 0x23, |
| 0x24, 0x30, 0x31, 0x32, 0x33, 0x34, 0x40, |
| 0x41, 0x42, 0x43, 0x44}); |
| SetToCanvas(canvas.get(), initial_values); |
| VerifyCanvasValues(canvas.get(), initial_values); |
| } |
| |
| #endif |
| |