// Copyright (c) 2011 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 <string>
#include "base/command_line.h"
#include "base/file_util.h"
#include "base/test/test_file_util.h"
#include "base/values.h"
#include "base/memory/scoped_temp_dir.h"
#include "build/build_config.h"
#include "chrome/common/chrome_constants.h"
#include "chrome/common/chrome_switches.h"
#include "chrome/common/pref_names.h"
#include "chrome/test/automation/browser_proxy.h"
#include "chrome/test/automation/window_proxy.h"
#include "chrome/test/ui/ui_test.h"
#include "content/common/json_value_serializer.h"
#include "ui/gfx/rect.h"
class PreferenceServiceTest : public UITest {
public:
void SetUp() {
ASSERT_TRUE(temp_dir_.CreateUniqueTempDir());
FilePath tmp_profile = temp_dir_.path().AppendASCII("tmp_profile");
ASSERT_TRUE(file_util::CreateDirectory(tmp_profile));
FilePath reference_pref_file;
if (new_profile_) {
reference_pref_file = test_data_directory_
.AppendASCII("profiles")
.AppendASCII("window_placement")
.AppendASCII("Default")
.Append(chrome::kPreferencesFilename);
tmp_pref_file_ = tmp_profile.AppendASCII("Default");
ASSERT_TRUE(file_util::CreateDirectory(tmp_pref_file_));
tmp_pref_file_ = tmp_pref_file_.Append(chrome::kPreferencesFilename);
} else {
reference_pref_file = test_data_directory_
.AppendASCII("profiles")
.AppendASCII("window_placement")
.Append(chrome::kLocalStateFilename);
tmp_pref_file_ = tmp_profile.Append(chrome::kLocalStateFilename);
}
ASSERT_TRUE(file_util::PathExists(reference_pref_file));
// Copy only the Preferences file if |new_profile_|, or Local State if not,
// and the rest will be automatically created.
ASSERT_TRUE(file_util::CopyFile(reference_pref_file, tmp_pref_file_));
#if defined(OS_WIN)
// Make the copy writable. On POSIX we assume the umask allows files
// we create to be writable.
ASSERT_TRUE(::SetFileAttributesW(tmp_pref_file_.value().c_str(),
FILE_ATTRIBUTE_NORMAL));
#endif
launch_arguments_.AppendSwitchPath(switches::kUserDataDir, tmp_profile);
}
bool LaunchAppWithProfile() {
if (!file_util::PathExists(tmp_pref_file_))
return false;
UITest::SetUp();
return true;
}
void TearDown() {
UITest::TearDown();
}
public:
bool new_profile_;
FilePath tmp_pref_file_;
private:
ScopedTempDir temp_dir_;
};
#if !defined(OS_LINUX)
// This test verifies that the window position from the prefs file is restored
// when the app restores. This doesn't really make sense on Linux, where
// the window manager might fight with you over positioning. However, we
// might be able to make this work on buildbots.
// TODO(port): revisit this.
TEST_F(PreferenceServiceTest, PreservedWindowPlacementIsLoaded) {
// The window should open with the new reference profile, with window
// placement values stored in the user data directory.
new_profile_ = true;
ASSERT_TRUE(LaunchAppWithProfile());
ASSERT_TRUE(file_util::PathExists(tmp_pref_file_));
JSONFileValueSerializer deserializer(tmp_pref_file_);
scoped_ptr<Value> root(deserializer.Deserialize(NULL, NULL));
ASSERT_TRUE(root.get());
ASSERT_TRUE(root->IsType(Value::TYPE_DICTIONARY));
DictionaryValue* root_dict = static_cast<DictionaryValue*>(root.get());
// Retrieve the screen rect for the launched window
scoped_refptr<BrowserProxy> browser(automation()->GetBrowserWindow(0));
ASSERT_TRUE(browser.get());
scoped_refptr<WindowProxy> window(browser->GetWindow());
ASSERT_TRUE(window.get());
gfx::Rect bounds;
ASSERT_TRUE(window->GetBounds(&bounds));
// Retrieve the expected rect values from "Preferences"
int bottom = 0;
std::string kBrowserWindowPlacement(prefs::kBrowserWindowPlacement);
EXPECT_TRUE(root_dict->GetInteger(kBrowserWindowPlacement + ".bottom",
&bottom));
EXPECT_EQ(bottom, bounds.y() + bounds.height());
int top = 0;
EXPECT_TRUE(root_dict->GetInteger(kBrowserWindowPlacement + ".top",
&top));
EXPECT_EQ(top, bounds.y());
int left = 0;
EXPECT_TRUE(root_dict->GetInteger(kBrowserWindowPlacement + ".left",
&left));
EXPECT_EQ(left, bounds.x());
int right = 0;
EXPECT_TRUE(root_dict->GetInteger(kBrowserWindowPlacement + ".right",
&right));
EXPECT_EQ(right, bounds.x() + bounds.width());
// Find if launched window is maximized.
bool is_window_maximized = false;
ASSERT_TRUE(window->IsMaximized(&is_window_maximized));
bool is_maximized = false;
EXPECT_TRUE(root_dict->GetBoolean(kBrowserWindowPlacement + ".maximized",
&is_maximized));
EXPECT_EQ(is_maximized, is_window_maximized);
}
#endif
#if !defined(OS_LINUX)
TEST_F(PreferenceServiceTest, PreservedWindowPlacementIsMigrated) {
// The window should open with the old reference profile, with window
// placement values stored in Local State.
new_profile_ = false;
ASSERT_TRUE(LaunchAppWithProfile());
ASSERT_TRUE(file_util::PathExists(tmp_pref_file_));
JSONFileValueSerializer deserializer(tmp_pref_file_);
scoped_ptr<Value> root(deserializer.Deserialize(NULL, NULL));
ASSERT_TRUE(root.get());
ASSERT_TRUE(root->IsType(Value::TYPE_DICTIONARY));
// Retrieve the screen rect for the launched window
scoped_refptr<BrowserProxy> browser(automation()->GetBrowserWindow(0));
ASSERT_TRUE(browser.get());
scoped_refptr<WindowProxy> window(browser->GetWindow());
ASSERT_TRUE(window.get());
gfx::Rect bounds;
ASSERT_TRUE(window->GetBounds(&bounds));
// Values from old reference profile in Local State should have been
// correctly migrated to the user's Preferences -- if so, the window
// should be set to values taken from the user's Local State.
DictionaryValue* root_dict = static_cast<DictionaryValue*>(root.get());
// Retrieve the expected rect values from User Preferences, where they
// should have been migrated from Local State.
int bottom = 0;
std::string kBrowserWindowPlacement(prefs::kBrowserWindowPlacement);
EXPECT_TRUE(root_dict->GetInteger(kBrowserWindowPlacement + ".bottom",
&bottom));
EXPECT_EQ(bottom, bounds.y() + bounds.height());
int top = 0;
EXPECT_TRUE(root_dict->GetInteger(kBrowserWindowPlacement + ".top",
&top));
EXPECT_EQ(top, bounds.y());
int left = 0;
EXPECT_TRUE(root_dict->GetInteger(kBrowserWindowPlacement + ".left",
&left));
EXPECT_EQ(left, bounds.x());
int right = 0;
EXPECT_TRUE(root_dict->GetInteger(kBrowserWindowPlacement + ".right",
&right));
EXPECT_EQ(right, bounds.x() + bounds.width());
// Find if launched window is maximized.
bool is_window_maximized = false;
ASSERT_TRUE(window->IsMaximized(&is_window_maximized));
bool is_maximized = false;
EXPECT_TRUE(root_dict->GetBoolean(kBrowserWindowPlacement + ".maximized",
&is_maximized));
EXPECT_EQ(is_maximized, is_window_maximized);
}
#endif