blob: 87d40ed6ea40e09bdc2ae238b9ab5a14cedfb3b4 [file] [log] [blame]
// Copyright 2016 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 <memory>
#include <vector>
#include "base/command_line.h"
#include "base/logging.h"
#include "base/memory/ptr_util.h"
#include "base/strings/string_number_conversions.h"
#include "services/ui/display/platform_screen.h"
#include "services/ui/display/platform_screen_ozone.h"
#include "services/ui/display/viewport_metrics.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/display/display_switches.h"
#include "ui/display/fake_display_snapshot.h"
#include "ui/display/types/display_constants.h"
#include "ui/display/types/display_mode.h"
#include "ui/display/types/display_snapshot.h"
#include "ui/ozone/public/ozone_platform.h"
namespace display {
using ui::DisplayMode;
using ui::DisplaySnapshot;
using testing::IsEmpty;
using testing::SizeIs;
namespace {
// Holds info about the display state we want to test.
struct DisplayState {
int64_t id;
ViewportMetrics metrics;
};
// Matchers that operate on DisplayState.
MATCHER_P(DisplayId, display_id, "") {
*result_listener << "has id " << arg.id;
return arg.id == display_id;
}
MATCHER_P(DisplaySize, size_string, "") {
*result_listener << "has size " << arg.metrics.bounds.size().ToString();
return arg.metrics.bounds.size().ToString() == size_string;
}
MATCHER_P(DisplayOrigin, origin_string, "") {
*result_listener << "has origin " << arg.metrics.bounds.origin().ToString();
return arg.metrics.bounds.origin().ToString() == origin_string;
}
// Make a DisplaySnapshot with specified id and size.
std::unique_ptr<DisplaySnapshot> MakeSnapshot(int64_t id,
const gfx::Size& size) {
return FakeDisplaySnapshot::Builder()
.SetId(id)
.SetNativeMode(size)
.SetCurrentMode(size)
.Build();
}
// Test delegate to track what functions calls the delegate receives.
class TestPlatformScreenDelegate : public PlatformScreenDelegate {
public:
TestPlatformScreenDelegate() {}
~TestPlatformScreenDelegate() override {}
const std::vector<DisplayState>& added() const { return added_; }
const std::vector<DisplayState>& modified() const { return modified_; }
// Returns a string containing the function calls that PlatformScreenDelegate
// has received in the order they occured. Each function call will be in the
// form "<action>(<id>)" and multiple function calls will be separated by ";".
// For example, if display 2 was added then display 1 was modified, changes()
// would return "Added(2);Modified(1)".
const std::string& changes() const { return changes_; }
void Reset() {
added_.clear();
modified_.clear();
changes_.clear();
}
private:
void AddChange(const std::string& name, const std::string& value) {
if (!changes_.empty())
changes_ += ";";
changes_ += name + "(" + value + ")";
}
void OnDisplayAdded(int64_t id, const ViewportMetrics& metrics) override {
added_.push_back({id, metrics});
AddChange("Added", base::Int64ToString(id));
}
void OnDisplayRemoved(int64_t id) override {
AddChange("Removed", base::Int64ToString(id));
}
void OnDisplayModified(int64_t id, const ViewportMetrics& metrics) override {
modified_.push_back({id, metrics});
AddChange("Modified", base::Int64ToString(id));
}
void OnPrimaryDisplayChanged(int64_t primary_display_id) override {
AddChange("Primary", base::Int64ToString(primary_display_id));
}
std::vector<DisplayState> added_;
std::vector<DisplayState> modified_;
std::string changes_;
DISALLOW_COPY_AND_ASSIGN(TestPlatformScreenDelegate);
};
} // namespace
// Test fixture with helpers to act like ui::DisplayConfigurator and send
// OnDisplayModeChanged() to PlatformScreenOzone.
class PlatformScreenOzoneTest : public testing::Test {
public:
PlatformScreenOzoneTest() {}
~PlatformScreenOzoneTest() override {}
PlatformScreenOzone* platform_screen() { return platform_screen_.get(); }
TestPlatformScreenDelegate* delegate() { return &delegate_; }
// Adds a display snapshot with specified ID and default size.
void AddDisplay(int64_t id) { return AddDisplay(id, gfx::Size(1024, 768)); }
// Adds a display snapshot with specified ID and size to list of snapshots.
void AddDisplay(int64_t id, const gfx::Size& size) {
snapshots_.push_back(MakeSnapshot(id, size));
TriggerOnDisplayModeChanged();
}
// Removes display snapshot with specified ID.
void RemoveDisplay(int64_t id) {
snapshots_.erase(
std::remove_if(snapshots_.begin(), snapshots_.end(),
[id](std::unique_ptr<DisplaySnapshot>& snapshot) {
return snapshot->display_id() == id;
}));
TriggerOnDisplayModeChanged();
}
// Modify the size of the display snapshot with specified ID.
void ModifyDisplay(int64_t id, const gfx::Size& size) {
DisplaySnapshot* snapshot = GetSnapshotById(id);
const DisplayMode* new_mode = nullptr;
for (auto& mode : snapshot->modes()) {
if (mode->size() == size) {
new_mode = mode.get();
break;
}
}
if (!new_mode) {
snapshot->add_mode(new DisplayMode(size, false, 30.0f));
new_mode = snapshot->modes().back().get();
}
snapshot->set_current_mode(new_mode);
TriggerOnDisplayModeChanged();
}
// Calls OnDisplayModeChanged with our list of display snapshots.
void TriggerOnDisplayModeChanged() {
std::vector<DisplaySnapshot*> snapshots_ptrs;
for (auto& snapshot : snapshots_) {
snapshots_ptrs.push_back(snapshot.get());
}
platform_screen_->OnDisplayModeChanged(snapshots_ptrs);
}
private:
DisplaySnapshot* GetSnapshotById(int64_t id) {
for (auto& snapshot : snapshots_) {
if (snapshot->display_id() == id)
return snapshot.get();
}
return nullptr;
}
// testing::Test:
void SetUp() override {
base::CommandLine::ForCurrentProcess()->AppendSwitchNative(
switches::kScreenConfig, "none");
testing::Test::SetUp();
ui::OzonePlatform::InitializeForUI();
platform_screen_ = base::MakeUnique<PlatformScreenOzone>();
platform_screen_->Init(&delegate_);
// Have all tests start with a 1024x768 display by default.
AddDisplay(1, gfx::Size(1024, 768));
TriggerOnDisplayModeChanged();
// Double check the expected display exists and clear counters.
ASSERT_THAT(delegate()->added(), SizeIs(1));
ASSERT_THAT(delegate_.added()[0], DisplayId(1));
ASSERT_THAT(delegate_.added()[0], DisplayOrigin("0,0"));
ASSERT_THAT(delegate_.added()[0], DisplaySize("1024x768"));
ASSERT_EQ("Added(1);Primary(1)", delegate()->changes());
delegate_.Reset();
}
void TearDown() override {
snapshots_.clear();
delegate_.Reset();
platform_screen_.reset();
}
TestPlatformScreenDelegate delegate_;
std::unique_ptr<PlatformScreenOzone> platform_screen_;
std::vector<std::unique_ptr<DisplaySnapshot>> snapshots_;
};
TEST_F(PlatformScreenOzoneTest, AddDisplay) {
AddDisplay(2);
// Check that display 2 was added.
EXPECT_EQ("Added(2)", delegate()->changes());
}
TEST_F(PlatformScreenOzoneTest, RemoveDisplay) {
AddDisplay(2);
delegate()->Reset();
RemoveDisplay(2);
// Check that display 2 was removed.
EXPECT_EQ("Removed(2)", delegate()->changes());
}
TEST_F(PlatformScreenOzoneTest, RemoveFirstDisplay) {
AddDisplay(2);
delegate()->Reset();
RemoveDisplay(1);
// Check that display 1 was removed and display 2 was modified due to the
// origin changing.
EXPECT_EQ("Primary(2);Removed(1);Modified(2)", delegate()->changes());
ASSERT_THAT(delegate()->modified(), SizeIs(1));
EXPECT_THAT(delegate()->modified()[0], DisplayId(2));
EXPECT_THAT(delegate()->modified()[0], DisplayOrigin("0,0"));
}
TEST_F(PlatformScreenOzoneTest, RemoveMultipleDisplay) {
AddDisplay(2);
AddDisplay(3);
delegate()->Reset();
RemoveDisplay(2);
RemoveDisplay(3);
// Check that display 2 was removed and display 3 is modifed (origin change),
// then display 3 was removed.
EXPECT_EQ("Removed(2);Modified(3);Removed(3)", delegate()->changes());
}
TEST_F(PlatformScreenOzoneTest, ModifyDisplaySize) {
const gfx::Size size1(1920, 1200);
const gfx::Size size2(1680, 1050);
AddDisplay(2, size1);
// Check that display 2 was added with expected size.
ASSERT_THAT(delegate()->added(), SizeIs(1));
EXPECT_THAT(delegate()->added()[0], DisplayId(2));
EXPECT_THAT(delegate()->added()[0], DisplaySize(size1.ToString()));
EXPECT_EQ("Added(2)", delegate()->changes());
delegate()->Reset();
ModifyDisplay(2, size2);
// Check that display 2 was modified to have the new expected size.
ASSERT_THAT(delegate()->modified(), SizeIs(1));
EXPECT_THAT(delegate()->modified()[0], DisplayId(2));
EXPECT_THAT(delegate()->modified()[0], DisplaySize(size2.ToString()));
EXPECT_EQ("Modified(2)", delegate()->changes());
}
TEST_F(PlatformScreenOzoneTest, ModifyFirstDisplaySize) {
const gfx::Size size(1920, 1200);
AddDisplay(2, size);
// Check that display 2 has the expected initial origin.
EXPECT_EQ("Added(2)", delegate()->changes());
ASSERT_THAT(delegate()->added(), SizeIs(1));
EXPECT_THAT(delegate()->added()[0], DisplayOrigin("1024,0"));
delegate()->Reset();
ModifyDisplay(1, size);
// Check that display 1 was modified with a new size and display 2 origin was
// modified after.
EXPECT_EQ("Modified(1);Modified(2)", delegate()->changes());
ASSERT_THAT(delegate()->modified(), SizeIs(2));
EXPECT_THAT(delegate()->modified()[0], DisplayId(1));
EXPECT_THAT(delegate()->modified()[0], DisplaySize(size.ToString()));
EXPECT_THAT(delegate()->modified()[1], DisplayId(2));
EXPECT_THAT(delegate()->modified()[1], DisplayOrigin("1920,0"));
}
TEST_F(PlatformScreenOzoneTest, RemovePrimaryDisplay) {
AddDisplay(2);
delegate()->Reset();
RemoveDisplay(1);
// Check the primary display changed because the old primary was removed.
EXPECT_EQ("Primary(2);Removed(1);Modified(2)", delegate()->changes());
}
TEST_F(PlatformScreenOzoneTest, RemoveLastDisplay) {
RemoveDisplay(1);
// Check that display 1 is removed and no updates for the primary display are
// received.
EXPECT_EQ("Removed(1)", delegate()->changes());
}
TEST_F(PlatformScreenOzoneTest, SwapPrimaryDisplay) {
AddDisplay(2);
delegate()->Reset();
platform_screen()->SwapPrimaryDisplay();
EXPECT_EQ("Primary(2)", delegate()->changes());
}
TEST_F(PlatformScreenOzoneTest, SwapPrimaryThreeDisplays) {
AddDisplay(2);
AddDisplay(3);
EXPECT_EQ("Added(2);Added(3)", delegate()->changes());
delegate()->Reset();
platform_screen()->SwapPrimaryDisplay();
platform_screen()->SwapPrimaryDisplay();
platform_screen()->SwapPrimaryDisplay();
EXPECT_EQ("Primary(2);Primary(3);Primary(1)", delegate()->changes());
}
} // namespace display