| // Copyright (c) 2013 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "ui/snapshot/snapshot.h" |
| |
| #include "base/bind.h" |
| #include "base/test/test_simple_task_runner.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "ui/aura/test/aura_test_helper.h" |
| #include "ui/aura/test/test_screen.h" |
| #include "ui/aura/test/test_window_delegate.h" |
| #include "ui/aura/test/test_windows.h" |
| #include "ui/aura/window.h" |
| #include "ui/aura/window_event_dispatcher.h" |
| #include "ui/compositor/compositor.h" |
| #include "ui/compositor/layer.h" |
| #include "ui/compositor/test/context_factories_for_test.h" |
| #include "ui/compositor/test/draw_waiter_for_test.h" |
| #include "ui/gfx/canvas.h" |
| #include "ui/gfx/geometry/rect.h" |
| #include "ui/gfx/geometry/size_conversions.h" |
| #include "ui/gfx/gfx_paths.h" |
| #include "ui/gfx/image/image.h" |
| #include "ui/gfx/transform.h" |
| #include "ui/gl/gl_implementation.h" |
| #include "ui/wm/core/default_activation_client.h" |
| |
| namespace ui { |
| namespace { |
| |
| SkColor GetExpectedColorForPoint(int x, int y) { |
| return SkColorSetRGB(std::min(x, 255), std::min(y, 255), 0); |
| } |
| |
| // Paint simple rectangle on the specified aura window. |
| class TestPaintingWindowDelegate : public aura::test::TestWindowDelegate { |
| public: |
| explicit TestPaintingWindowDelegate(const gfx::Size& window_size) |
| : window_size_(window_size) { |
| } |
| |
| ~TestPaintingWindowDelegate() override {} |
| |
| void OnPaint(gfx::Canvas* canvas) override { |
| for (int y = 0; y < window_size_.height(); ++y) { |
| for (int x = 0; x < window_size_.width(); ++x) |
| canvas->FillRect(gfx::Rect(x, y, 1, 1), GetExpectedColorForPoint(x, y)); |
| } |
| } |
| |
| private: |
| gfx::Size window_size_; |
| |
| DISALLOW_COPY_AND_ASSIGN(TestPaintingWindowDelegate); |
| }; |
| |
| size_t GetFailedPixelsCountWithScaleFactor(const gfx::Image& image, |
| int scale_factor) { |
| const SkBitmap* bitmap = image.ToSkBitmap(); |
| uint32* bitmap_data = reinterpret_cast<uint32*>( |
| bitmap->pixelRef()->pixels()); |
| size_t result = 0; |
| for (int y = 0; y < bitmap->height(); y += scale_factor) { |
| for (int x = 0; x < bitmap->width(); x += scale_factor) { |
| if (static_cast<SkColor>(bitmap_data[x + y * bitmap->width()]) != |
| GetExpectedColorForPoint(x / scale_factor, y / scale_factor)) { |
| ++result; |
| } |
| } |
| } |
| return result; |
| } |
| |
| size_t GetFailedPixelsCount(const gfx::Image& image) { |
| return GetFailedPixelsCountWithScaleFactor(image, 1); |
| } |
| |
| } // namespace |
| |
| class SnapshotAuraTest : public testing::Test { |
| public: |
| SnapshotAuraTest() {} |
| ~SnapshotAuraTest() override {} |
| |
| void SetUp() override { |
| testing::Test::SetUp(); |
| |
| // The ContextFactory must exist before any Compositors are created. |
| // Snapshot test tests real drawing and readback, so needs pixel output. |
| bool enable_pixel_output = true; |
| ui::ContextFactory* context_factory = |
| ui::InitializeContextFactoryForTests(enable_pixel_output); |
| |
| helper_.reset( |
| new aura::test::AuraTestHelper(base::MessageLoopForUI::current())); |
| helper_->SetUp(context_factory); |
| new ::wm::DefaultActivationClient(helper_->root_window()); |
| } |
| |
| void TearDown() override { |
| test_window_.reset(); |
| delegate_.reset(); |
| helper_->RunAllPendingInMessageLoop(); |
| helper_->TearDown(); |
| ui::TerminateContextFactoryForTests(); |
| testing::Test::TearDown(); |
| } |
| |
| protected: |
| aura::Window* test_window() { return test_window_.get(); } |
| aura::Window* root_window() { return helper_->root_window(); } |
| aura::TestScreen* test_screen() { return helper_->test_screen(); } |
| |
| void WaitForDraw() { |
| helper_->host()->compositor()->ScheduleDraw(); |
| ui::DrawWaiterForTest::WaitForCompositingEnded( |
| helper_->host()->compositor()); |
| } |
| |
| void SetupTestWindow(const gfx::Rect& window_bounds) { |
| delegate_.reset(new TestPaintingWindowDelegate(window_bounds.size())); |
| test_window_.reset(aura::test::CreateTestWindowWithDelegate( |
| delegate_.get(), 0, window_bounds, root_window())); |
| } |
| |
| gfx::Image GrabSnapshotForTestWindow() { |
| gfx::Rect source_rect(test_window_->bounds().size()); |
| aura::Window::ConvertRectToTarget( |
| test_window(), root_window(), &source_rect); |
| |
| scoped_refptr<base::TestSimpleTaskRunner> task_runner( |
| new base::TestSimpleTaskRunner()); |
| scoped_refptr<SnapshotHolder> holder(new SnapshotHolder); |
| ui::GrabWindowSnapshotAsync( |
| root_window(), |
| source_rect, |
| task_runner, |
| base::Bind(&SnapshotHolder::SnapshotCallback, holder)); |
| |
| // Wait for copy response. |
| WaitForDraw(); |
| // Run internal snapshot callback to scale/rotate response image. |
| task_runner->RunUntilIdle(); |
| // Run SnapshotHolder callback. |
| helper_->RunAllPendingInMessageLoop(); |
| |
| if (holder->completed()) |
| return holder->image(); |
| |
| // Callback never called. |
| NOTREACHED(); |
| return gfx::Image(); |
| } |
| |
| private: |
| class SnapshotHolder : public base::RefCountedThreadSafe<SnapshotHolder> { |
| public: |
| SnapshotHolder() : completed_(false) {} |
| |
| void SnapshotCallback(scoped_refptr<base::RefCountedBytes> png_data) { |
| DCHECK(!completed_); |
| image_ = gfx::Image::CreateFrom1xPNGBytes(&(png_data->data()[0]), |
| png_data->size()); |
| completed_ = true; |
| } |
| bool completed() const { |
| return completed_; |
| }; |
| const gfx::Image& image() const { return image_; } |
| |
| private: |
| friend class base::RefCountedThreadSafe<SnapshotHolder>; |
| |
| virtual ~SnapshotHolder() {} |
| |
| gfx::Image image_; |
| bool completed_; |
| }; |
| |
| scoped_ptr<aura::test::AuraTestHelper> helper_; |
| scoped_ptr<aura::Window> test_window_; |
| scoped_ptr<TestPaintingWindowDelegate> delegate_; |
| std::vector<unsigned char> png_representation_; |
| |
| DISALLOW_COPY_AND_ASSIGN(SnapshotAuraTest); |
| }; |
| |
| TEST_F(SnapshotAuraTest, FullScreenWindow) { |
| SetupTestWindow(root_window()->bounds()); |
| WaitForDraw(); |
| |
| gfx::Image snapshot = GrabSnapshotForTestWindow(); |
| EXPECT_EQ(test_window()->bounds().size().ToString(), |
| snapshot.Size().ToString()); |
| EXPECT_EQ(0u, GetFailedPixelsCount(snapshot)); |
| } |
| |
| TEST_F(SnapshotAuraTest, PartialBounds) { |
| gfx::Rect test_bounds(100, 100, 300, 200); |
| SetupTestWindow(test_bounds); |
| WaitForDraw(); |
| |
| gfx::Image snapshot = GrabSnapshotForTestWindow(); |
| EXPECT_EQ(test_bounds.size().ToString(), snapshot.Size().ToString()); |
| EXPECT_EQ(0u, GetFailedPixelsCount(snapshot)); |
| } |
| |
| TEST_F(SnapshotAuraTest, Rotated) { |
| test_screen()->SetDisplayRotation(gfx::Display::ROTATE_90); |
| |
| gfx::Rect test_bounds(100, 100, 300, 200); |
| SetupTestWindow(test_bounds); |
| WaitForDraw(); |
| |
| gfx::Image snapshot = GrabSnapshotForTestWindow(); |
| EXPECT_EQ(test_bounds.size().ToString(), snapshot.Size().ToString()); |
| EXPECT_EQ(0u, GetFailedPixelsCount(snapshot)); |
| } |
| |
| TEST_F(SnapshotAuraTest, UIScale) { |
| const float kUIScale = 1.25f; |
| test_screen()->SetUIScale(kUIScale); |
| |
| gfx::Rect test_bounds(100, 100, 300, 200); |
| SetupTestWindow(test_bounds); |
| WaitForDraw(); |
| |
| // Snapshot always captures the physical pixels. |
| gfx::SizeF snapshot_size(test_bounds.size()); |
| |
| gfx::Image snapshot = GrabSnapshotForTestWindow(); |
| EXPECT_EQ(gfx::ToRoundedSize(snapshot_size).ToString(), |
| snapshot.Size().ToString()); |
| EXPECT_EQ(0u, GetFailedPixelsCount(snapshot)); |
| } |
| |
| TEST_F(SnapshotAuraTest, DeviceScaleFactor) { |
| test_screen()->SetDeviceScaleFactor(2.0f); |
| |
| gfx::Rect test_bounds(100, 100, 150, 100); |
| SetupTestWindow(test_bounds); |
| WaitForDraw(); |
| |
| // Snapshot always captures the physical pixels. |
| gfx::SizeF snapshot_size(test_bounds.size()); |
| snapshot_size.Scale(2.0f); |
| |
| gfx::Image snapshot = GrabSnapshotForTestWindow(); |
| EXPECT_EQ(gfx::ToRoundedSize(snapshot_size).ToString(), |
| snapshot.Size().ToString()); |
| EXPECT_EQ(0u, GetFailedPixelsCountWithScaleFactor(snapshot, 2)); |
| } |
| |
| TEST_F(SnapshotAuraTest, RotateAndUIScale) { |
| const float kUIScale = 1.25f; |
| test_screen()->SetUIScale(kUIScale); |
| test_screen()->SetDisplayRotation(gfx::Display::ROTATE_90); |
| |
| gfx::Rect test_bounds(100, 100, 300, 200); |
| SetupTestWindow(test_bounds); |
| WaitForDraw(); |
| |
| // Snapshot always captures the physical pixels. |
| gfx::SizeF snapshot_size(test_bounds.size()); |
| |
| gfx::Image snapshot = GrabSnapshotForTestWindow(); |
| EXPECT_EQ(gfx::ToRoundedSize(snapshot_size).ToString(), |
| snapshot.Size().ToString()); |
| EXPECT_EQ(0u, GetFailedPixelsCount(snapshot)); |
| } |
| |
| TEST_F(SnapshotAuraTest, RotateAndUIScaleAndScaleFactor) { |
| test_screen()->SetDeviceScaleFactor(2.0f); |
| const float kUIScale = 1.25f; |
| test_screen()->SetUIScale(kUIScale); |
| test_screen()->SetDisplayRotation(gfx::Display::ROTATE_90); |
| |
| gfx::Rect test_bounds(20, 30, 150, 100); |
| SetupTestWindow(test_bounds); |
| WaitForDraw(); |
| |
| // Snapshot always captures the physical pixels. |
| gfx::SizeF snapshot_size(test_bounds.size()); |
| snapshot_size.Scale(2.0f); |
| |
| gfx::Image snapshot = GrabSnapshotForTestWindow(); |
| EXPECT_EQ(gfx::ToRoundedSize(snapshot_size).ToString(), |
| snapshot.Size().ToString()); |
| EXPECT_EQ(0u, GetFailedPixelsCountWithScaleFactor(snapshot, 2)); |
| } |
| |
| } // namespace ui |