|  | // Copyright 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 "build/build_config.h" | 
|  | #include "chrome/browser/apps/app_browsertest_util.h" | 
|  | #include "chrome/browser/profiles/profile.h" | 
|  | #include "chrome/browser/ui/browser.h" | 
|  | #include "chrome/browser/ui/extensions/app_launch_params.h" | 
|  | #include "chrome/browser/ui/extensions/application_launch.h" | 
|  | #include "content/public/browser/notification_service.h" | 
|  | #include "content/public/test/test_utils.h" | 
|  | #include "extensions/browser/app_window/app_window_geometry_cache.h" | 
|  | #include "extensions/common/constants.h" | 
|  | #include "extensions/common/extension.h" | 
|  | #include "extensions/test/extension_test_message_listener.h" | 
|  | #include "extensions/test/result_catcher.h" | 
|  |  | 
|  | using extensions::AppWindowGeometryCache; | 
|  | using extensions::ResultCatcher; | 
|  |  | 
|  | // This helper class can be used to wait for changes in the app window | 
|  | // geometry cache registry for a specific window in a specific extension. | 
|  | class GeometryCacheChangeHelper : AppWindowGeometryCache::Observer { | 
|  | public: | 
|  | GeometryCacheChangeHelper(AppWindowGeometryCache* cache, | 
|  | const std::string& extension_id, | 
|  | const std::string& window_id, | 
|  | const gfx::Rect& bounds) | 
|  | : cache_(cache), | 
|  | extension_id_(extension_id), | 
|  | window_id_(window_id), | 
|  | bounds_(bounds), | 
|  | satisfied_(false), | 
|  | waiting_(false) { | 
|  | cache_->AddObserver(this); | 
|  | } | 
|  |  | 
|  | // This method will block until the app window geometry cache registry will | 
|  | // provide a bound for |window_id_| that is entirely different (as in x/y/w/h) | 
|  | // from the initial |bounds_|. | 
|  | void WaitForEntirelyChanged() { | 
|  | if (satisfied_) | 
|  | return; | 
|  |  | 
|  | waiting_ = true; | 
|  | content::RunMessageLoop(); | 
|  | } | 
|  |  | 
|  | // Implements the content::NotificationObserver interface. | 
|  | void OnGeometryCacheChanged(const std::string& extension_id, | 
|  | const std::string& window_id, | 
|  | const gfx::Rect& bounds) override { | 
|  | if (extension_id != extension_id_ || window_id != window_id_) | 
|  | return; | 
|  |  | 
|  | if (bounds_.x() != bounds.x() && | 
|  | bounds_.y() != bounds.y() && | 
|  | bounds_.width() != bounds.width() && | 
|  | bounds_.height() != bounds.height()) { | 
|  | satisfied_ = true; | 
|  | cache_->RemoveObserver(this); | 
|  |  | 
|  | if (waiting_) | 
|  | base::MessageLoopForUI::current()->QuitWhenIdle(); | 
|  | } | 
|  | } | 
|  |  | 
|  | private: | 
|  | AppWindowGeometryCache* cache_; | 
|  | std::string extension_id_; | 
|  | std::string window_id_; | 
|  | gfx::Rect bounds_; | 
|  | bool satisfied_; | 
|  | bool waiting_; | 
|  | }; | 
|  |  | 
|  | // Helper class for tests related to the Apps Window API (chrome.app.window). | 
|  | class AppWindowAPITest : public extensions::PlatformAppBrowserTest { | 
|  | protected: | 
|  | bool RunAppWindowAPITest(const char* testName) { | 
|  | if (!BeginAppWindowAPITest(testName)) | 
|  | return false; | 
|  |  | 
|  | ResultCatcher catcher; | 
|  | if (!catcher.GetNextResult()) { | 
|  | message_ = catcher.message(); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool RunAppWindowAPITestAndWaitForRoundTrip(const char* testName) { | 
|  | if (!BeginAppWindowAPITest(testName)) | 
|  | return false; | 
|  |  | 
|  | ExtensionTestMessageListener round_trip_listener("WaitForRoundTrip", true); | 
|  | if (!round_trip_listener.WaitUntilSatisfied()) { | 
|  | message_ = "Did not get the 'WaitForRoundTrip' message."; | 
|  | return false; | 
|  | } | 
|  |  | 
|  | round_trip_listener.Reply(""); | 
|  |  | 
|  | ResultCatcher catcher; | 
|  | if (!catcher.GetNextResult()) { | 
|  | message_ = catcher.message(); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | private: | 
|  | bool BeginAppWindowAPITest(const char* testName) { | 
|  | ExtensionTestMessageListener launched_listener("Launched", true); | 
|  | LoadAndLaunchPlatformApp("window_api", &launched_listener); | 
|  | if (!launched_listener.WaitUntilSatisfied()) { | 
|  | message_ = "Did not get the 'Launched' message."; | 
|  | return false; | 
|  | } | 
|  |  | 
|  | launched_listener.Reply(testName); | 
|  | return true; | 
|  | } | 
|  | }; | 
|  |  | 
|  | // These tests are flaky after https://codereview.chromium.org/57433010/. | 
|  | // See http://crbug.com/319613. | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(AppWindowAPITest, TestCreate) { | 
|  | ASSERT_TRUE(RunAppWindowAPITest("testCreate")) << message_; | 
|  | } | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(AppWindowAPITest, TestSingleton) { | 
|  | ASSERT_TRUE(RunAppWindowAPITest("testSingleton")) << message_; | 
|  | } | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(AppWindowAPITest, TestCloseEvent) { | 
|  | ASSERT_TRUE(RunAppWindowAPITest("testCloseEvent")) << message_; | 
|  | } | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(AppWindowAPITest, DISABLED_TestMaximize) { | 
|  | ASSERT_TRUE(RunAppWindowAPITest("testMaximize")) << message_; | 
|  | } | 
|  |  | 
|  | // Flaky on Linux. http://crbug.com/424399. | 
|  | #if defined(OS_LINUX) && !defined(OS_CHROMEOS) | 
|  | #define MAYBE_TestMinimize DISABLED_TestMinimize | 
|  | #else | 
|  | #define MAYBE_TestMinimize TestMinimize | 
|  | #endif | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(AppWindowAPITest, MAYBE_TestMinimize) { | 
|  | ASSERT_TRUE(RunAppWindowAPITest("testMinimize")) << message_; | 
|  | } | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(AppWindowAPITest, DISABLED_TestRestore) { | 
|  | ASSERT_TRUE(RunAppWindowAPITest("testRestore")) << message_; | 
|  | } | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(AppWindowAPITest, DISABLED_TestRestoreAfterClose) { | 
|  | ASSERT_TRUE(RunAppWindowAPITest("testRestoreAfterClose")) << message_; | 
|  | } | 
|  |  | 
|  | // These tests will be flaky in Linux as window bounds change asynchronously. | 
|  | #if defined(OS_LINUX) | 
|  | #define MAYBE_TestDeprecatedBounds DISABLED_TestDeprecatedBounds | 
|  | #define MAYBE_TestInitialBounds DISABLED_TestInitialBounds | 
|  | #define MAYBE_TestInitialConstraints DISABLED_TestInitialConstraints | 
|  | #define MAYBE_TestSetBounds DISABLED_TestSetBounds | 
|  | #define MAYBE_TestSetSizeConstraints DISABLED_TestSetSizeConstraints | 
|  | #else | 
|  | #define MAYBE_TestDeprecatedBounds TestDeprecatedBounds | 
|  | #define MAYBE_TestInitialBounds TestInitialBounds | 
|  | #define MAYBE_TestInitialConstraints TestInitialConstraints | 
|  | #define MAYBE_TestSetBounds TestSetBounds | 
|  | // Disabled as flakey, see http://crbug.com/434532 for details. | 
|  | #define MAYBE_TestSetSizeConstraints DISABLED_TestSetSizeConstraints | 
|  | #endif | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(AppWindowAPITest, MAYBE_TestDeprecatedBounds) { | 
|  | ASSERT_TRUE(RunAppWindowAPITest("testDeprecatedBounds")) << message_; | 
|  | } | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(AppWindowAPITest, MAYBE_TestInitialBounds) { | 
|  | ASSERT_TRUE(RunAppWindowAPITest("testInitialBounds")) << message_; | 
|  | } | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(AppWindowAPITest, MAYBE_TestInitialConstraints) { | 
|  | ASSERT_TRUE(RunAppWindowAPITest("testInitialConstraints")) << message_; | 
|  | } | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(AppWindowAPITest, MAYBE_TestSetBounds) { | 
|  | ASSERT_TRUE(RunAppWindowAPITest("testSetBounds")) << message_; | 
|  | } | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(AppWindowAPITest, MAYBE_TestSetSizeConstraints) { | 
|  | ASSERT_TRUE(RunAppWindowAPITest("testSetSizeConstraints")) << message_; | 
|  | } | 
|  |  | 
|  | // Flaky failures on mac_rel and WinXP, see http://crbug.com/324915. | 
|  | IN_PROC_BROWSER_TEST_F(AppWindowAPITest, | 
|  | DISABLED_TestRestoreGeometryCacheChange) { | 
|  | // This test is similar to the other AppWindowAPI tests except that at some | 
|  | // point the app will send a 'ListenGeometryChange' message at which point the | 
|  | // test will check if the geometry cache entry for the test window has | 
|  | // changed. When the change happens, the test will let the app know so it can | 
|  | // continue running. | 
|  | ExtensionTestMessageListener launched_listener("Launched", true); | 
|  |  | 
|  | content::WindowedNotificationObserver app_loaded_observer( | 
|  | content::NOTIFICATION_LOAD_COMPLETED_MAIN_FRAME, | 
|  | content::NotificationService::AllSources()); | 
|  |  | 
|  | const extensions::Extension* extension = LoadExtension( | 
|  | test_data_dir_.AppendASCII("platform_apps").AppendASCII("window_api")); | 
|  | EXPECT_TRUE(extension); | 
|  |  | 
|  | OpenApplication(AppLaunchParams(browser()->profile(), extension, | 
|  | extensions::LAUNCH_CONTAINER_NONE, NEW_WINDOW, | 
|  | extensions::SOURCE_TEST)); | 
|  |  | 
|  | ExtensionTestMessageListener geometry_listener("ListenGeometryChange", true); | 
|  |  | 
|  | ASSERT_TRUE(launched_listener.WaitUntilSatisfied()); | 
|  | launched_listener.Reply("testRestoreAfterGeometryCacheChange"); | 
|  |  | 
|  | ASSERT_TRUE(geometry_listener.WaitUntilSatisfied()); | 
|  |  | 
|  | GeometryCacheChangeHelper geo_change_helper_1( | 
|  | AppWindowGeometryCache::Get(browser()->profile()), | 
|  | extension->id(), | 
|  | // The next line has information that has to stay in sync with the app. | 
|  | "test-ra", | 
|  | gfx::Rect(200, 200, 200, 200)); | 
|  |  | 
|  | GeometryCacheChangeHelper geo_change_helper_2( | 
|  | AppWindowGeometryCache::Get(browser()->profile()), | 
|  | extension->id(), | 
|  | // The next line has information that has to stay in sync with the app. | 
|  | "test-rb", | 
|  | gfx::Rect(200, 200, 200, 200)); | 
|  |  | 
|  | // These calls will block until the app window geometry cache will change. | 
|  | geo_change_helper_1.WaitForEntirelyChanged(); | 
|  | geo_change_helper_2.WaitForEntirelyChanged(); | 
|  |  | 
|  | ResultCatcher catcher; | 
|  | geometry_listener.Reply(""); | 
|  | ASSERT_TRUE(catcher.GetNextResult()); | 
|  | } | 
|  |  | 
|  | // TODO(benwells): Implement on Mac. | 
|  | #if defined(USE_AURA) | 
|  | IN_PROC_BROWSER_TEST_F(AppWindowAPITest, TestFrameColors) { | 
|  | ASSERT_TRUE(RunAppWindowAPITest("testFrameColors")) << message_; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(AppWindowAPITest, TestVisibleOnAllWorkspaces) { | 
|  | ASSERT_TRUE( | 
|  | RunAppWindowAPITestAndWaitForRoundTrip("testVisibleOnAllWorkspaces")) | 
|  | << message_; | 
|  | } |