blob: a3ca0caf3dc03082ade76f4e544168d6d24644a9 [file] [log] [blame]
[email protected]a9c060ca2012-01-05 20:43:411// Copyright (c) 2012 The Chromium Authors. All rights reserved.
license.botbf09a502008-08-24 00:55:552// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
initial.commit09911bf2008-07-26 23:55:294
[email protected]d9fde8d2009-10-08 19:59:305#include "build/build_config.h"
6
[email protected]a9c060ca2012-01-05 20:43:417#include "base/bind.h"
[email protected]3985ba82010-07-29 21:44:128#include "base/file_util.h"
[email protected]f07d7bf2010-04-06 08:02:429#include "base/format_macros.h"
[email protected]8bcdec92009-02-25 16:15:1810#include "base/message_loop.h"
[email protected]3985ba82010-07-29 21:44:1211#include "base/path_service.h"
[email protected]a711df102010-12-14 09:47:0312#include "base/string_number_conversions.h"
[email protected]477ae052011-11-18 23:53:5713#include "base/stringprintf.h"
[email protected]be1ce6a72010-08-03 14:35:2214#include "base/utf_string_conversions.h"
[email protected]7b5dc002010-11-16 23:08:1015#include "chrome/browser/ui/browser.h"
[email protected]a37d4b02012-06-25 21:56:1016#include "chrome/browser/ui/browser_commands.h"
[email protected]52877dbc62012-06-29 22:22:0317#include "chrome/browser/ui/browser_tabstrip.h"
[email protected]00070c732011-04-09 15:31:3318#include "chrome/browser/ui/browser_window.h"
[email protected]5d9cace72012-06-21 16:07:1219#include "chrome/browser/ui/chrome_pages.h"
[email protected]871dc682012-06-11 19:35:3320#include "chrome/browser/ui/tab_contents/tab_contents.h"
[email protected]b56e2e32012-05-11 21:18:0421#include "chrome/browser/ui/tabs/tab_strip_model.h"
[email protected]6a3ec2312010-12-02 19:30:1922#include "chrome/browser/ui/view_ids.h"
[email protected]432115822011-07-10 15:52:2723#include "chrome/common/chrome_notification_types.h"
[email protected]9e0c83a2009-05-06 19:44:3724#include "chrome/common/chrome_paths.h"
[email protected]cecc93a2010-10-05 15:58:5525#include "chrome/common/url_constants.h"
[email protected]af44e7fb2011-07-29 18:32:3226#include "chrome/test/base/in_process_browser_test.h"
[email protected]477ae052011-11-18 23:53:5727#include "chrome/test/base/ui_test_utils.h"
[email protected]cadaec52012-02-08 21:53:1328#include "content/public/browser/interstitial_page.h"
[email protected]85f0a572012-02-07 22:20:1329#include "content/public/browser/interstitial_page_delegate.h"
[email protected]477ae052011-11-18 23:53:5730#include "content/public/browser/notification_service.h"
[email protected]9c1662b2012-03-06 15:44:3331#include "content/public/browser/render_view_host.h"
[email protected]5626b0892012-02-20 14:46:5832#include "content/public/browser/render_widget_host_view.h"
[email protected]6acde6352012-01-04 16:52:2033#include "content/public/browser/web_contents.h"
[email protected]8643e6d2012-01-18 20:26:1034#include "content/public/browser/web_contents_view.h"
[email protected]3985ba82010-07-29 21:44:1235#include "net/test/test_server.h"
[email protected]853300a82010-07-27 21:17:5736
37#if defined(TOOLKIT_VIEWS) || defined(OS_WIN)
[email protected]477ae052011-11-18 23:53:5738#include "ui/views/focus/focus_manager.h"
[email protected]5025f862011-11-30 23:35:2039#include "ui/views/view.h"
[email protected]853300a82010-07-27 21:17:5740#endif
initial.commit09911bf2008-07-26 23:55:2941
[email protected]134c47b92009-08-19 03:33:4442#if defined(TOOLKIT_VIEWS)
[email protected]9e790bd2011-01-10 23:48:5443#include "chrome/browser/ui/views/frame/browser_view.h"
[email protected]134c47b92009-08-19 03:33:4444#endif
45
[email protected]a13283cc2012-04-05 00:21:2246#if defined(TOOLKIT_GTK)
[email protected]93270d002011-01-19 22:32:5947#include "chrome/browser/ui/gtk/view_id_util.h"
[email protected]b9821882009-08-17 22:25:1748#endif
49
[email protected]a711df102010-12-14 09:47:0350#if defined(OS_WIN)
[email protected]a711df102010-12-14 09:47:0351#include <Psapi.h>
[email protected]93270d002011-01-19 22:32:5952#include <windows.h>
[email protected]1870d5cf2011-05-12 01:55:4053#include "base/string_util.h"
[email protected]a711df102010-12-14 09:47:0354#endif
55
[email protected]cadaec52012-02-08 21:53:1356using content::InterstitialPage;
[email protected]c5eed492012-01-04 17:07:5057using content::NavigationController;
[email protected]eaabba22012-03-07 15:02:1158using content::RenderViewHost;
[email protected]4b19ea52012-01-02 20:15:2559using content::WebContents;
60
[email protected]3a3cf4d2011-12-09 00:39:3561#if defined(OS_MACOSX)
62// TODO(suzhe): http://crbug.com/60973
[email protected]a5eb2d672010-10-22 07:24:4863#define MAYBE_FocusTraversal DISABLED_FocusTraversal
64#define MAYBE_FocusTraversalOnInterstitial DISABLED_FocusTraversalOnInterstitial
[email protected]871153922012-01-10 20:13:4865#elif defined(OS_WIN)
66// http://crbug.com/109770
[email protected]2f2bf1172012-01-10 22:17:1867#define MAYBE_FocusTraversal FocusTraversal
[email protected]871153922012-01-10 20:13:4868#define MAYBE_FocusTraversalOnInterstitial DISABLED_FocusTraversalOnInterstitial
[email protected]3a3cf4d2011-12-09 00:39:3569#else
70#define MAYBE_FocusTraversal FocusTraversal
71#define MAYBE_FocusTraversalOnInterstitial FocusTraversalOnInterstitial
72#endif
73
74#if defined(OS_LINUX) || defined(OS_MACOSX)
75// TODO(jcampan): http://crbug.com/23683 for linux.
76// TODO(suzhe): http://crbug.com/49737 for mac.
[email protected]853300a82010-07-27 21:17:5777#define MAYBE_TabsRememberFocusFindInPage FAILS_TabsRememberFocusFindInPage
78#elif defined(OS_WIN)
[email protected]c92928672010-11-09 18:31:0779// Flaky, http://crbug.com/62537.
[email protected]4d2451652012-02-14 23:54:2680#define MAYBE_TabsRememberFocusFindInPage DISABLED_TabsRememberFocusFindInPage
[email protected]fc2e0872009-08-21 22:14:4181#endif
82
initial.commit09911bf2008-07-26 23:55:2983namespace {
84
[email protected]8bcdec92009-02-25 16:15:1885// The delay waited in some cases where we don't have a notifications for an
86// action we take.
initial.commit09911bf2008-07-26 23:55:2987const int kActionDelayMs = 500;
88
[email protected]a9c060ca2012-01-05 20:43:4189// Maxiumum time to wait until the focus is moved to expected view.
90const int kFocusChangeTimeoutMs = 500;
91
[email protected]f72a1cc2010-04-30 07:17:3092const char kSimplePage[] = "files/focus/page_with_focus.html";
93const char kStealFocusPage[] = "files/focus/page_steals_focus.html";
94const char kTypicalPage[] = "files/focus/typical_page.html";
[email protected]b65de8b92009-09-14 19:36:3195const char kTypicalPageName[] = "typical_page.html";
initial.commit09911bf2008-07-26 23:55:2996
[email protected]a711df102010-12-14 09:47:0397// Test to make sure Chrome is in the foreground as we start testing. This is
98// required for tests that synthesize input to the Chrome window.
99bool ChromeInForeground() {
100#if defined(OS_WIN)
101 HWND window = ::GetForegroundWindow();
102 std::wstring caption;
103 std::wstring filename;
104 int len = ::GetWindowTextLength(window) + 1;
[email protected]fdce4782011-11-29 20:06:18105 if (len > 1)
106 ::GetWindowText(window, WriteInto(&caption, len), len);
[email protected]a711df102010-12-14 09:47:03107 bool chrome_window_in_foreground =
108 EndsWith(caption, L" - Google Chrome", true) ||
109 EndsWith(caption, L" - Chromium", true);
110 if (!chrome_window_in_foreground) {
111 DWORD process_id;
112 int thread_id = ::GetWindowThreadProcessId(window, &process_id);
113
114 base::ProcessHandle process;
115 if (base::OpenProcessHandleWithAccess(process_id,
116 PROCESS_QUERY_LIMITED_INFORMATION,
117 &process)) {
[email protected]fdce4782011-11-29 20:06:18118 if (!GetProcessImageFileName(process, WriteInto(&filename, MAX_PATH),
119 MAX_PATH)) {
[email protected]a711df102010-12-14 09:47:03120 int error = GetLastError();
121 filename = std::wstring(L"Unable to read filename for process id '" +
122 base::IntToString16(process_id) +
123 L"' (error ") +
124 base::IntToString16(error) + L")";
125 }
126 base::CloseProcessHandle(process);
127 }
128 }
129 EXPECT_TRUE(chrome_window_in_foreground)
130 << "Chrome must be in the foreground when running interactive tests\n"
131 << "Process in foreground: " << filename.c_str() << "\n"
132 << "Window: " << window << "\n"
133 << "Caption: " << caption.c_str();
134 return chrome_window_in_foreground;
135#else
136 // Windows only at the moment.
137 return true;
138#endif
139}
140
[email protected]a9c060ca2012-01-05 20:43:41141// Wait the focus change in message loop.
142void CheckFocus(Browser* browser, ViewID id, const base::Time& timeout) {
143 if (ui_test_utils::IsViewFocused(browser, id) ||
144 base::Time::Now() > timeout) {
145 MessageLoop::current()->PostTask(FROM_HERE, MessageLoop::QuitClosure());
146 } else {
147 MessageLoop::current()->PostDelayedTask(
148 FROM_HERE,
149 base::Bind(&CheckFocus, browser, id, timeout),
[email protected]7e560102012-03-08 20:58:42150 base::TimeDelta::FromMilliseconds(10));
[email protected]a9c060ca2012-01-05 20:43:41151 }
152};
153
[email protected]8bcdec92009-02-25 16:15:18154class BrowserFocusTest : public InProcessBrowserTest {
initial.commit09911bf2008-07-26 23:55:29155 public:
[email protected]0b8fa8b2011-12-07 00:54:52156 BrowserFocusTest() :
157#if defined(USE_AURA)
158 location_bar_focus_view_id_(VIEW_ID_OMNIBOX)
159#else
160 location_bar_focus_view_id_(VIEW_ID_LOCATION_BAR)
161#endif
[email protected]90ca44272012-07-18 18:15:48162 {}
[email protected]b9821882009-08-17 22:25:17163
[email protected]21abcc742009-10-23 02:52:06164 bool IsViewFocused(ViewID vid) {
165 return ui_test_utils::IsViewFocused(browser(), vid);
[email protected]b9821882009-08-17 22:25:17166 }
167
[email protected]fc2e0872009-08-21 22:14:41168 void ClickOnView(ViewID vid) {
[email protected]21abcc742009-10-23 02:52:06169 ui_test_utils::ClickOnView(browser(), vid);
[email protected]fc2e0872009-08-21 22:14:41170 }
[email protected]0b8fa8b2011-12-07 00:54:52171
[email protected]a9c060ca2012-01-05 20:43:41172 bool WaitForFocusChange(ViewID vid) {
173 const base::Time timeout = base::Time::Now() +
174 base::TimeDelta::FromMilliseconds(kFocusChangeTimeoutMs);
175 MessageLoop::current()->PostDelayedTask(
176 FROM_HERE,
177 base::Bind(&CheckFocus, browser(), vid, timeout),
[email protected]7e560102012-03-08 20:58:42178 base::TimeDelta::FromMilliseconds(100));
[email protected]a9c060ca2012-01-05 20:43:41179 ui_test_utils::RunMessageLoop();
180 return IsViewFocused(vid);
181 }
182
[email protected]0b8fa8b2011-12-07 00:54:52183 ViewID location_bar_focus_view_id_;
initial.commit09911bf2008-07-26 23:55:29184};
185
[email protected]85f0a572012-02-07 22:20:13186class TestInterstitialPage : public content::InterstitialPageDelegate {
[email protected]9e0c83a2009-05-06 19:44:37187 public:
[email protected]85f0a572012-02-07 22:20:13188 TestInterstitialPage(WebContents* tab, bool new_navigation, const GURL& url) {
[email protected]b65de8b92009-09-14 19:36:31189 FilePath file_path;
[email protected]9e0c83a2009-05-06 19:44:37190 bool r = PathService::Get(chrome::DIR_TEST_DATA, &file_path);
191 EXPECT_TRUE(r);
[email protected]b65de8b92009-09-14 19:36:31192 file_path = file_path.AppendASCII("focus");
193 file_path = file_path.AppendASCII(kTypicalPageName);
[email protected]9e0c83a2009-05-06 19:44:37194 r = file_util::ReadFileToString(file_path, &html_contents_);
195 EXPECT_TRUE(r);
[email protected]85f0a572012-02-07 22:20:13196 interstitial_page_ = InterstitialPage::Create(
197 tab, new_navigation, url , this);
198 interstitial_page_->Show();
[email protected]9e0c83a2009-05-06 19:44:37199 }
200
201 virtual std::string GetHTMLContents() {
202 return html_contents_;
203 }
204
[email protected]85f0a572012-02-07 22:20:13205 RenderViewHost* render_view_host() {
[email protected]cadaec52012-02-08 21:53:13206 return interstitial_page_->GetRenderViewHostForTesting();
[email protected]9e0c83a2009-05-06 19:44:37207 }
208
[email protected]85f0a572012-02-07 22:20:13209 void DontProceed() {
210 interstitial_page_->DontProceed();
[email protected]4e85c112011-01-26 22:27:21211 }
212
[email protected]9e0c83a2009-05-06 19:44:37213 bool HasFocus() {
[email protected]9f76c1e2012-03-05 15:15:58214 return render_view_host()->GetView()->HasFocus();
[email protected]9e0c83a2009-05-06 19:44:37215 }
216
217 private:
218 std::string html_contents_;
[email protected]85f0a572012-02-07 22:20:13219 InterstitialPage* interstitial_page_; // Owns us.
[email protected]9e0c83a2009-05-06 19:44:37220};
[email protected]b9821882009-08-17 22:25:17221
[email protected]c39e233d2012-05-18 13:24:12222// Flaky on mac. http://crbug.com/67301.
223#if defined(OS_MACOSX)
[email protected]839e0802012-05-18 14:45:23224#define MAYBE_ClickingMovesFocus DISABLED_ClickingMovesFocus
[email protected]c39e233d2012-05-18 13:24:12225#else
[email protected]839e0802012-05-18 14:45:23226#define MAYBE_ClickingMovesFocus ClickingMovesFocus
[email protected]c39e233d2012-05-18 13:24:12227#endif
228IN_PROC_BROWSER_TEST_F(BrowserFocusTest, MAYBE_ClickingMovesFocus) {
[email protected]a6e602f2010-09-28 22:28:30229 ASSERT_TRUE(ui_test_utils::BringBrowserWindowToFront(browser()));
[email protected]0ff0ff32010-12-21 19:34:42230#if defined(OS_POSIX)
[email protected]fc2e0872009-08-21 22:14:41231 // It seems we have to wait a little bit for the widgets to spin up before
232 // we can start clicking on them.
[email protected]7e560102012-03-08 20:58:42233 MessageLoop::current()->PostDelayedTask(
234 FROM_HERE,
235 MessageLoop::QuitClosure(),
236 base::TimeDelta::FromMilliseconds(kActionDelayMs));
[email protected]fc2e0872009-08-21 22:14:41237 ui_test_utils::RunMessageLoop();
[email protected]0ff0ff32010-12-21 19:34:42238#endif // defined(OS_POSIX)
[email protected]fc2e0872009-08-21 22:14:41239
[email protected]0b8fa8b2011-12-07 00:54:52240 ASSERT_TRUE(IsViewFocused(location_bar_focus_view_id_));
[email protected]186f13f2009-08-19 20:34:00241
[email protected]fc2e0872009-08-21 22:14:41242 ClickOnView(VIEW_ID_TAB_CONTAINER);
[email protected]f2159ba2012-04-17 19:13:21243 ASSERT_TRUE(IsViewFocused(VIEW_ID_TAB_CONTAINER));
[email protected]186f13f2009-08-19 20:34:00244
[email protected]fc2e0872009-08-21 22:14:41245 ClickOnView(VIEW_ID_LOCATION_BAR);
[email protected]0b8fa8b2011-12-07 00:54:52246 ASSERT_TRUE(IsViewFocused(location_bar_focus_view_id_));
[email protected]186f13f2009-08-19 20:34:00247}
[email protected]186f13f2009-08-19 20:34:00248
[email protected]320948e02011-01-10 08:21:12249// Flaky, http://crbug.com/69034.
[email protected]4d2451652012-02-14 23:54:26250IN_PROC_BROWSER_TEST_F(BrowserFocusTest, DISABLED_BrowsersRememberFocus) {
[email protected]a6e602f2010-09-28 22:28:30251 ASSERT_TRUE(ui_test_utils::BringBrowserWindowToFront(browser()));
[email protected]95409e12010-08-17 20:07:11252 ASSERT_TRUE(test_server()->Start());
initial.commit09911bf2008-07-26 23:55:29253
254 // First we navigate to our test page.
[email protected]95409e12010-08-17 20:07:11255 GURL url = test_server()->GetURL(kSimplePage);
[email protected]8bcdec92009-02-25 16:15:18256 ui_test_utils::NavigateToURL(browser(), url);
initial.commit09911bf2008-07-26 23:55:29257
[email protected]90556dd2012-06-07 20:26:18258 gfx::NativeWindow window = browser()->window()->GetNativeWindow();
[email protected]186f13f2009-08-19 20:34:00259
initial.commit09911bf2008-07-26 23:55:29260 // The focus should be on the Tab contents.
[email protected]f2159ba2012-04-17 19:13:21261 ASSERT_TRUE(IsViewFocused(VIEW_ID_TAB_CONTAINER));
initial.commit09911bf2008-07-26 23:55:29262 // Now hide the window, show it again, the focus should not have changed.
[email protected]853300a82010-07-27 21:17:57263 ui_test_utils::HideNativeWindow(window);
[email protected]7807088c2011-11-09 15:16:29264 ASSERT_TRUE(ui_test_utils::ShowAndFocusNativeWindow(window));
[email protected]f2159ba2012-04-17 19:13:21265 ASSERT_TRUE(IsViewFocused(VIEW_ID_TAB_CONTAINER));
initial.commit09911bf2008-07-26 23:55:29266
[email protected]a37d4b02012-06-25 21:56:10267 chrome::FocusLocationBar(browser());
[email protected]0b8fa8b2011-12-07 00:54:52268 ASSERT_TRUE(IsViewFocused(location_bar_focus_view_id_));
initial.commit09911bf2008-07-26 23:55:29269 // Hide the window, show it again, the focus should not have changed.
[email protected]853300a82010-07-27 21:17:57270 ui_test_utils::HideNativeWindow(window);
[email protected]7807088c2011-11-09 15:16:29271 ASSERT_TRUE(ui_test_utils::ShowAndFocusNativeWindow(window));
[email protected]0b8fa8b2011-12-07 00:54:52272 ASSERT_TRUE(IsViewFocused(location_bar_focus_view_id_));
initial.commit09911bf2008-07-26 23:55:29273
[email protected]186f13f2009-08-19 20:34:00274 // The rest of this test does not make sense on Linux because the behavior
275 // of Activate() is not well defined and can vary by window manager.
276#if defined(OS_WIN)
initial.commit09911bf2008-07-26 23:55:29277 // Open a new browser window.
[email protected]8bcdec92009-02-25 16:15:18278 Browser* browser2 = Browser::Create(browser()->profile());
279 ASSERT_TRUE(browser2);
[email protected]855370052012-07-10 19:30:32280 chrome::AddBlankTab(browser2, true);
[email protected]8bcdec92009-02-25 16:15:18281 browser2->window()->Show();
282 ui_test_utils::NavigateToURL(browser2, url);
initial.commit09911bf2008-07-26 23:55:29283
[email protected]90556dd2012-06-07 20:26:18284 gfx::NativeWindow window2 = browser2->window()->GetNativeWindow();
[email protected]4a507a62009-05-28 00:10:00285 BrowserView* browser_view2 =
[email protected]075b204c2011-09-30 19:02:16286 BrowserView::GetBrowserViewForBrowser(browser2);
[email protected]8bcdec92009-02-25 16:15:18287 ASSERT_TRUE(browser_view2);
[email protected]0aaa5282011-10-26 18:54:50288 const views::Widget* widget2 =
289 views::Widget::GetWidgetForNativeWindow(window2);
[email protected]0c966682011-08-02 18:22:10290 ASSERT_TRUE(widget2);
[email protected]0aaa5282011-10-26 18:54:50291 const views::FocusManager* focus_manager2 = widget2->GetFocusManager();
[email protected]8bcdec92009-02-25 16:15:18292 ASSERT_TRUE(focus_manager2);
[email protected]7e383692009-06-12 19:14:54293 EXPECT_EQ(browser_view2->GetTabContentsContainerView(),
[email protected]610d36a2009-05-22 23:00:38294 focus_manager2->GetFocusedView());
initial.commit09911bf2008-07-26 23:55:29295
296 // Switch to the 1st browser window, focus should still be on the location
297 // bar and the second browser should have nothing focused.
[email protected]8bcdec92009-02-25 16:15:18298 browser()->window()->Activate();
[email protected]0b8fa8b2011-12-07 00:54:52299 ASSERT_TRUE(IsViewFocused(location_bar_focus_view_id_));
[email protected]8bcdec92009-02-25 16:15:18300 EXPECT_EQ(NULL, focus_manager2->GetFocusedView());
initial.commit09911bf2008-07-26 23:55:29301
302 // Switch back to the second browser, focus should still be on the page.
[email protected]8bcdec92009-02-25 16:15:18303 browser2->window()->Activate();
[email protected]0c966682011-08-02 18:22:10304 views::Widget* widget = views::Widget::GetWidgetForNativeWindow(window);
305 ASSERT_TRUE(widget);
306 EXPECT_EQ(NULL, widget->GetFocusManager()->GetFocusedView());
[email protected]7e383692009-06-12 19:14:54307 EXPECT_EQ(browser_view2->GetTabContentsContainerView(),
[email protected]610d36a2009-05-22 23:00:38308 focus_manager2->GetFocusedView());
[email protected]8bcdec92009-02-25 16:15:18309
310 // Close the 2nd browser to avoid a DCHECK().
311 browser_view2->Close();
[email protected]186f13f2009-08-19 20:34:00312#endif
initial.commit09911bf2008-07-26 23:55:29313}
314
315// Tabs remember focus.
[email protected]26153732010-11-09 18:47:39316// Disabled, http://crbug.com/62542.
317IN_PROC_BROWSER_TEST_F(BrowserFocusTest, DISABLED_TabsRememberFocus) {
[email protected]a6e602f2010-09-28 22:28:30318 ASSERT_TRUE(ui_test_utils::BringBrowserWindowToFront(browser()));
[email protected]95409e12010-08-17 20:07:11319 ASSERT_TRUE(test_server()->Start());
initial.commit09911bf2008-07-26 23:55:29320
321 // First we navigate to our test page.
[email protected]95409e12010-08-17 20:07:11322 GURL url = test_server()->GetURL(kSimplePage);
[email protected]8bcdec92009-02-25 16:15:18323 ui_test_utils::NavigateToURL(browser(), url);
324
initial.commit09911bf2008-07-26 23:55:29325 // Create several tabs.
[email protected]52877dbc62012-06-29 22:22:03326 for (int i = 0; i < 4; ++i) {
327 chrome::AddSelectedTabWithURL(browser(), url,
328 content::PAGE_TRANSITION_TYPED);
329 }
initial.commit09911bf2008-07-26 23:55:29330
331 // Alternate focus for the tab.
332 const bool kFocusPage[3][5] = {
333 { true, true, true, true, false },
334 { false, false, false, false, false },
335 { false, true, false, true, false }
336 };
337
338 for (int i = 1; i < 3; i++) {
339 for (int j = 0; j < 5; j++) {
[email protected]8bcdec92009-02-25 16:15:18340 // Activate the tab.
[email protected]52877dbc62012-06-29 22:22:03341 chrome::ActivateTabAt(browser(), j, true);
initial.commit09911bf2008-07-26 23:55:29342
343 // Activate the location bar or the page.
[email protected]7e383692009-06-12 19:14:54344 if (kFocusPage[i][j]) {
[email protected]52877dbc62012-06-29 22:22:03345 chrome::GetWebContentsAt(browser(), j)->GetView()->Focus();
[email protected]7e383692009-06-12 19:14:54346 } else {
[email protected]a37d4b02012-06-25 21:56:10347 chrome::FocusLocationBar(browser());
[email protected]7e383692009-06-12 19:14:54348 }
initial.commit09911bf2008-07-26 23:55:29349 }
350
351 // Now come back to the tab and check the right view is focused.
352 for (int j = 0; j < 5; j++) {
[email protected]8bcdec92009-02-25 16:15:18353 // Activate the tab.
[email protected]52877dbc62012-06-29 22:22:03354 chrome::ActivateTabAt(browser(), j, true);
initial.commit09911bf2008-07-26 23:55:29355
[email protected]f2159ba2012-04-17 19:13:21356 ViewID vid = kFocusPage[i][j] ? VIEW_ID_TAB_CONTAINER :
[email protected]0b8fa8b2011-12-07 00:54:52357 location_bar_focus_view_id_;
[email protected]21abcc742009-10-23 02:52:06358 ASSERT_TRUE(IsViewFocused(vid));
initial.commit09911bf2008-07-26 23:55:29359 }
[email protected]cb7e2542009-12-14 22:02:35360
[email protected]52877dbc62012-06-29 22:22:03361 chrome::ActivateTabAt(browser(), 0, true);
[email protected]cb7e2542009-12-14 22:02:35362 // Try the above, but with ctrl+tab. Since tab normally changes focus,
363 // this has regressed in the past. Loop through several times to be sure.
364 for (int j = 0; j < 15; j++) {
[email protected]f2159ba2012-04-17 19:13:21365 ViewID vid = kFocusPage[i][j % 5] ? VIEW_ID_TAB_CONTAINER :
[email protected]0b8fa8b2011-12-07 00:54:52366 location_bar_focus_view_id_;
[email protected]cb7e2542009-12-14 22:02:35367 ASSERT_TRUE(IsViewFocused(vid));
368
[email protected]1d000682010-08-23 16:21:28369 ASSERT_TRUE(ui_test_utils::SendKeyPressSync(
[email protected]b6d81262011-01-13 17:36:09370 browser(), ui::VKEY_TAB, true, false, false, false));
[email protected]cb7e2542009-12-14 22:02:35371 }
372
373 // As above, but with ctrl+shift+tab.
[email protected]52877dbc62012-06-29 22:22:03374 chrome::ActivateTabAt(browser(), 4, true);
[email protected]cb7e2542009-12-14 22:02:35375 for (int j = 14; j >= 0; --j) {
[email protected]f2159ba2012-04-17 19:13:21376 ViewID vid = kFocusPage[i][j % 5] ? VIEW_ID_TAB_CONTAINER :
[email protected]0b8fa8b2011-12-07 00:54:52377 location_bar_focus_view_id_;
[email protected]cb7e2542009-12-14 22:02:35378 ASSERT_TRUE(IsViewFocused(vid));
379
[email protected]1d000682010-08-23 16:21:28380 ASSERT_TRUE(ui_test_utils::SendKeyPressSync(
[email protected]b6d81262011-01-13 17:36:09381 browser(), ui::VKEY_TAB, true, true, false, false));
[email protected]cb7e2542009-12-14 22:02:35382 }
initial.commit09911bf2008-07-26 23:55:29383 }
384}
385
[email protected]ae40b572009-10-02 21:17:45386// Tabs remember focus with find-in-page box.
[email protected]cb7e2542009-12-14 22:02:35387IN_PROC_BROWSER_TEST_F(BrowserFocusTest, MAYBE_TabsRememberFocusFindInPage) {
[email protected]a6e602f2010-09-28 22:28:30388 ASSERT_TRUE(ui_test_utils::BringBrowserWindowToFront(browser()));
[email protected]95409e12010-08-17 20:07:11389 ASSERT_TRUE(test_server()->Start());
[email protected]ae40b572009-10-02 21:17:45390
391 // First we navigate to our test page.
[email protected]95409e12010-08-17 20:07:11392 GURL url = test_server()->GetURL(kSimplePage);
[email protected]ae40b572009-10-02 21:17:45393 ui_test_utils::NavigateToURL(browser(), url);
394
[email protected]a37d4b02012-06-25 21:56:10395 chrome::Find(browser());
[email protected]52877dbc62012-06-29 22:22:03396 ui_test_utils::FindInPage(chrome::GetActiveTabContents(browser()),
[email protected]ae40b572009-10-02 21:17:45397 ASCIIToUTF16("a"), true, false, NULL);
[email protected]21abcc742009-10-23 02:52:06398 ASSERT_TRUE(IsViewFocused(VIEW_ID_FIND_IN_PAGE_TEXT_FIELD));
[email protected]ae40b572009-10-02 21:17:45399
400 // Focus the location bar.
[email protected]a37d4b02012-06-25 21:56:10401 chrome::FocusLocationBar(browser());
[email protected]ae40b572009-10-02 21:17:45402
403 // Create a 2nd tab.
[email protected]52877dbc62012-06-29 22:22:03404 chrome::AddSelectedTabWithURL(browser(), url, content::PAGE_TRANSITION_TYPED);
[email protected]ae40b572009-10-02 21:17:45405
406 // Focus should be on the recently opened tab page.
[email protected]f2159ba2012-04-17 19:13:21407 ASSERT_TRUE(IsViewFocused(VIEW_ID_TAB_CONTAINER));
[email protected]ae40b572009-10-02 21:17:45408
409 // Select 1st tab, focus should still be on the location-bar.
410 // (bug http://crbug.com/23296)
[email protected]52877dbc62012-06-29 22:22:03411 chrome::ActivateTabAt(browser(), 0, true);
[email protected]0b8fa8b2011-12-07 00:54:52412 ASSERT_TRUE(IsViewFocused(location_bar_focus_view_id_));
[email protected]ae40b572009-10-02 21:17:45413
414 // Now open the find box again, switch to another tab and come back, the focus
415 // should return to the find box.
[email protected]a37d4b02012-06-25 21:56:10416 chrome::Find(browser());
[email protected]21abcc742009-10-23 02:52:06417 ASSERT_TRUE(IsViewFocused(VIEW_ID_FIND_IN_PAGE_TEXT_FIELD));
[email protected]52877dbc62012-06-29 22:22:03418 chrome::ActivateTabAt(browser(), 1, true);
[email protected]f2159ba2012-04-17 19:13:21419 ASSERT_TRUE(IsViewFocused(VIEW_ID_TAB_CONTAINER));
[email protected]52877dbc62012-06-29 22:22:03420 chrome::ActivateTabAt(browser(), 0, true);
[email protected]21abcc742009-10-23 02:52:06421 ASSERT_TRUE(IsViewFocused(VIEW_ID_FIND_IN_PAGE_TEXT_FIELD));
[email protected]ae40b572009-10-02 21:17:45422}
423
initial.commit09911bf2008-07-26 23:55:29424// Background window does not steal focus.
[email protected]c92928672010-11-09 18:31:07425// Flaky, http://crbug.com/62538.
426IN_PROC_BROWSER_TEST_F(BrowserFocusTest,
[email protected]4d2451652012-02-14 23:54:26427 DISABLED_BackgroundBrowserDontStealFocus) {
[email protected]a6e602f2010-09-28 22:28:30428 ASSERT_TRUE(ui_test_utils::BringBrowserWindowToFront(browser()));
[email protected]95409e12010-08-17 20:07:11429 ASSERT_TRUE(test_server()->Start());
initial.commit09911bf2008-07-26 23:55:29430
initial.commit09911bf2008-07-26 23:55:29431 // Open a new browser window.
[email protected]8bcdec92009-02-25 16:15:18432 Browser* browser2 = Browser::Create(browser()->profile());
433 ASSERT_TRUE(browser2);
[email protected]855370052012-07-10 19:30:32434 chrome::AddBlankTab(browser2, true);
[email protected]8bcdec92009-02-25 16:15:18435 browser2->window()->Show();
[email protected]186f13f2009-08-19 20:34:00436
[email protected]ed179ee2009-10-03 21:02:51437 Browser* focused_browser = NULL;
438 Browser* unfocused_browser = NULL;
[email protected]753efc42010-03-09 19:52:16439#if defined(USE_X11)
440 // On X11, calling Activate() is not guaranteed to move focus, so we have
[email protected]186f13f2009-08-19 20:34:00441 // to figure out which browser does have focus.
442 if (browser2->window()->IsActive()) {
443 focused_browser = browser2;
444 unfocused_browser = browser();
445 } else if (browser()->window()->IsActive()) {
446 focused_browser = browser();
447 unfocused_browser = browser2;
448 } else {
[email protected]579c3d82010-10-06 03:53:51449 FAIL() << "Could not determine which browser has focus";
[email protected]186f13f2009-08-19 20:34:00450 }
451#elif defined(OS_WIN)
452 focused_browser = browser();
453 unfocused_browser = browser2;
[email protected]853300a82010-07-27 21:17:57454#elif defined(OS_MACOSX)
455 // On Mac, the newly created window always gets the focus.
456 focused_browser = browser2;
457 unfocused_browser = browser();
[email protected]186f13f2009-08-19 20:34:00458#endif
459
[email protected]95409e12010-08-17 20:07:11460 GURL steal_focus_url = test_server()->GetURL(kStealFocusPage);
[email protected]186f13f2009-08-19 20:34:00461 ui_test_utils::NavigateToURL(unfocused_browser, steal_focus_url);
[email protected]1e187af2009-02-25 02:02:46462
[email protected]8bcdec92009-02-25 16:15:18463 // Activate the first browser.
[email protected]186f13f2009-08-19 20:34:00464 focused_browser->window()->Activate();
initial.commit09911bf2008-07-26 23:55:29465
[email protected]579c3d82010-10-06 03:53:51466 ASSERT_TRUE(ui_test_utils::ExecuteJavaScript(
[email protected]52877dbc62012-06-29 22:22:03467 chrome::GetActiveWebContents(unfocused_browser)->GetRenderViewHost(), L"",
[email protected]579c3d82010-10-06 03:53:51468 L"stealFocus();"));
initial.commit09911bf2008-07-26 23:55:29469
[email protected]8bcdec92009-02-25 16:15:18470 // Make sure the first browser is still active.
[email protected]186f13f2009-08-19 20:34:00471 EXPECT_TRUE(focused_browser->window()->IsActive());
initial.commit09911bf2008-07-26 23:55:29472}
473
474// Page cannot steal focus when focus is on location bar.
[email protected]e4f4e0b2009-10-13 19:58:21475IN_PROC_BROWSER_TEST_F(BrowserFocusTest, LocationBarLockFocus) {
[email protected]a6e602f2010-09-28 22:28:30476 ASSERT_TRUE(ui_test_utils::BringBrowserWindowToFront(browser()));
[email protected]95409e12010-08-17 20:07:11477 ASSERT_TRUE(test_server()->Start());
initial.commit09911bf2008-07-26 23:55:29478
479 // Open the page that steals focus.
[email protected]95409e12010-08-17 20:07:11480 GURL url = test_server()->GetURL(kStealFocusPage);
[email protected]8bcdec92009-02-25 16:15:18481 ui_test_utils::NavigateToURL(browser(), url);
initial.commit09911bf2008-07-26 23:55:29482
[email protected]a37d4b02012-06-25 21:56:10483 chrome::FocusLocationBar(browser());
initial.commit09911bf2008-07-26 23:55:29484
[email protected]579c3d82010-10-06 03:53:51485 ASSERT_TRUE(ui_test_utils::ExecuteJavaScript(
[email protected]52877dbc62012-06-29 22:22:03486 chrome::GetActiveWebContents(browser())->GetRenderViewHost(), L"",
[email protected]579c3d82010-10-06 03:53:51487 L"stealFocus();"));
initial.commit09911bf2008-07-26 23:55:29488
489 // Make sure the location bar is still focused.
[email protected]0b8fa8b2011-12-07 00:54:52490 ASSERT_TRUE(IsViewFocused(location_bar_focus_view_id_));
initial.commit09911bf2008-07-26 23:55:29491}
492
[email protected]9e0c83a2009-05-06 19:44:37493// Focus traversal on a regular page.
[email protected]130efb02009-09-18 18:54:35494// Note that this test relies on a notification from the renderer that the
495// focus has changed in the page. The notification in the renderer may change
496// at which point this test would fail (see comment in
497// RenderWidget::didFocus()).
[email protected]853300a82010-07-27 21:17:57498IN_PROC_BROWSER_TEST_F(BrowserFocusTest, MAYBE_FocusTraversal) {
[email protected]a6e602f2010-09-28 22:28:30499 ASSERT_TRUE(ui_test_utils::BringBrowserWindowToFront(browser()));
[email protected]95409e12010-08-17 20:07:11500 ASSERT_TRUE(test_server()->Start());
initial.commit09911bf2008-07-26 23:55:29501
[email protected]8bcdec92009-02-25 16:15:18502 // First we navigate to our test page.
[email protected]95409e12010-08-17 20:07:11503 GURL url = test_server()->GetURL(kTypicalPage);
[email protected]8bcdec92009-02-25 16:15:18504 ui_test_utils::NavigateToURL(browser(), url);
initial.commit09911bf2008-07-26 23:55:29505
[email protected]a37d4b02012-06-25 21:56:10506 chrome::FocusLocationBar(browser());
initial.commit09911bf2008-07-26 23:55:29507
[email protected]546ae4e02010-12-08 14:57:19508 const char* kTextElementID = "textEdit";
[email protected]8bcdec92009-02-25 16:15:18509 const char* kExpElementIDs[] = {
510 "", // Initially no element in the page should be focused
511 // (the location bar is focused).
[email protected]546ae4e02010-12-08 14:57:19512 kTextElementID, "searchButton", "luckyButton", "googleLink", "gmailLink",
[email protected]8bcdec92009-02-25 16:15:18513 "gmapLink"
initial.commit09911bf2008-07-26 23:55:29514 };
515
516 // Test forward focus traversal.
517 for (int i = 0; i < 3; ++i) {
[email protected]1870d5cf2011-05-12 01:55:40518 SCOPED_TRACE(base::StringPrintf("outer loop: %d", i));
initial.commit09911bf2008-07-26 23:55:29519 // Location bar should be focused.
[email protected]0b8fa8b2011-12-07 00:54:52520 ASSERT_TRUE(IsViewFocused(location_bar_focus_view_id_));
initial.commit09911bf2008-07-26 23:55:29521
[email protected]911696b2011-01-28 02:36:49522 // Move the caret to the end, otherwise the next Tab key may not move focus.
523 ASSERT_TRUE(ui_test_utils::SendKeyPressSync(
524 browser(), ui::VKEY_END, false, false, false, false));
525
initial.commit09911bf2008-07-26 23:55:29526 // Now let's press tab to move the focus.
[email protected]130efb02009-09-18 18:54:35527 for (size_t j = 0; j < arraysize(kExpElementIDs); ++j) {
[email protected]1870d5cf2011-05-12 01:55:40528 SCOPED_TRACE(base::StringPrintf("inner loop %" PRIuS, j));
initial.commit09911bf2008-07-26 23:55:29529 // Let's make sure the focus is on the expected element in the page.
[email protected]45671612009-04-29 22:24:01530 std::string actual;
531 ASSERT_TRUE(ui_test_utils::ExecuteJavaScriptAndExtractString(
[email protected]52877dbc62012-06-29 22:22:03532 chrome::GetActiveWebContents(browser())->GetRenderViewHost(),
[email protected]8bcdec92009-02-25 16:15:18533 L"",
[email protected]45671612009-04-29 22:24:01534 L"window.domAutomationController.send(getFocusedElement());",
535 &actual));
initial.commit09911bf2008-07-26 23:55:29536 ASSERT_STREQ(kExpElementIDs[j], actual.c_str());
537
[email protected]130efb02009-09-18 18:54:35538 if (j < arraysize(kExpElementIDs) - 1) {
[email protected]546ae4e02010-12-08 14:57:19539 // If the next element is the kTextElementID, we expect to be
540 // notified we have switched to an editable node.
541 bool is_editable_node =
542 (strcmp(kTextElementID, kExpElementIDs[j + 1]) == 0);
[email protected]6c2381d2011-10-19 02:52:53543 content::Details<bool> details(&is_editable_node);
[email protected]546ae4e02010-12-08 14:57:19544
545 ASSERT_TRUE(ui_test_utils::SendKeyPressAndWaitWithDetails(
[email protected]b6d81262011-01-13 17:36:09546 browser(), ui::VKEY_TAB, false, false, false, false,
[email protected]432115822011-07-10 15:52:27547 content::NOTIFICATION_FOCUS_CHANGED_IN_PAGE,
[email protected]d53092652011-12-06 06:12:56548 content::NotificationSource(content::Source<RenderViewHost>(
[email protected]52877dbc62012-06-29 22:22:03549 chrome::GetActiveWebContents(browser())->GetRenderViewHost())),
[email protected]546ae4e02010-12-08 14:57:19550 details));
[email protected]130efb02009-09-18 18:54:35551 } else {
552 // On the last tab key press, the focus returns to the browser.
[email protected]546ae4e02010-12-08 14:57:19553 ASSERT_TRUE(ui_test_utils::SendKeyPressAndWait(
[email protected]b6d81262011-01-13 17:36:09554 browser(), ui::VKEY_TAB, false, false, false, false,
[email protected]432115822011-07-10 15:52:27555 chrome::NOTIFICATION_FOCUS_RETURNED_TO_BROWSER,
[email protected]6c2381d2011-10-19 02:52:53556 content::NotificationSource(content::Source<Browser>(browser()))));
[email protected]130efb02009-09-18 18:54:35557 }
initial.commit09911bf2008-07-26 23:55:29558 }
[email protected]8bcdec92009-02-25 16:15:18559
560 // At this point the renderer has sent us a message asking to advance the
561 // focus (as the end of the focus loop was reached in the renderer).
562 // We need to run the message loop to process it.
[email protected]f07d7bf2010-04-06 08:02:42563 ui_test_utils::RunAllPendingInMessageLoop();
initial.commit09911bf2008-07-26 23:55:29564 }
565
566 // Now let's try reverse focus traversal.
567 for (int i = 0; i < 3; ++i) {
[email protected]1870d5cf2011-05-12 01:55:40568 SCOPED_TRACE(base::StringPrintf("outer loop: %d", i));
initial.commit09911bf2008-07-26 23:55:29569 // Location bar should be focused.
[email protected]0b8fa8b2011-12-07 00:54:52570 ASSERT_TRUE(IsViewFocused(location_bar_focus_view_id_));
initial.commit09911bf2008-07-26 23:55:29571
[email protected]911696b2011-01-28 02:36:49572 // Move the caret to the end, otherwise the next Tab key may not move focus.
573 ASSERT_TRUE(ui_test_utils::SendKeyPressSync(
574 browser(), ui::VKEY_END, false, false, false, false));
575
[email protected]8bcdec92009-02-25 16:15:18576 // Now let's press shift-tab to move the focus in reverse.
[email protected]a6e602f2010-09-28 22:28:30577 for (size_t j = 0; j < arraysize(kExpElementIDs); ++j) {
[email protected]1870d5cf2011-05-12 01:55:40578 SCOPED_TRACE(base::StringPrintf("inner loop: %" PRIuS, j));
[email protected]546ae4e02010-12-08 14:57:19579 const char* next_element =
580 kExpElementIDs[arraysize(kExpElementIDs) - 1 - j];
[email protected]130efb02009-09-18 18:54:35581
582 if (j < arraysize(kExpElementIDs) - 1) {
[email protected]546ae4e02010-12-08 14:57:19583 // If the next element is the kTextElementID, we expect to be
584 // notified we have switched to an editable node.
585 bool is_editable_node = (strcmp(kTextElementID, next_element) == 0);
[email protected]6c2381d2011-10-19 02:52:53586 content::Details<bool> details(&is_editable_node);
[email protected]546ae4e02010-12-08 14:57:19587
588 ASSERT_TRUE(ui_test_utils::SendKeyPressAndWaitWithDetails(
[email protected]b6d81262011-01-13 17:36:09589 browser(), ui::VKEY_TAB, false, true, false, false,
[email protected]432115822011-07-10 15:52:27590 content::NOTIFICATION_FOCUS_CHANGED_IN_PAGE,
[email protected]d53092652011-12-06 06:12:56591 content::NotificationSource(content::Source<RenderViewHost>(
[email protected]52877dbc62012-06-29 22:22:03592 chrome::GetActiveWebContents(browser())->GetRenderViewHost())),
[email protected]546ae4e02010-12-08 14:57:19593 details));
[email protected]130efb02009-09-18 18:54:35594 } else {
595 // On the last tab key press, the focus returns to the browser.
[email protected]546ae4e02010-12-08 14:57:19596 ASSERT_TRUE(ui_test_utils::SendKeyPressAndWait(
[email protected]b6d81262011-01-13 17:36:09597 browser(), ui::VKEY_TAB, false, true, false, false,
[email protected]432115822011-07-10 15:52:27598 chrome::NOTIFICATION_FOCUS_RETURNED_TO_BROWSER,
[email protected]6c2381d2011-10-19 02:52:53599 content::NotificationSource(content::Source<Browser>(browser()))));
[email protected]130efb02009-09-18 18:54:35600 }
initial.commit09911bf2008-07-26 23:55:29601
602 // Let's make sure the focus is on the expected element in the page.
[email protected]45671612009-04-29 22:24:01603 std::string actual;
604 ASSERT_TRUE(ui_test_utils::ExecuteJavaScriptAndExtractString(
[email protected]52877dbc62012-06-29 22:22:03605 chrome::GetActiveWebContents(browser())->GetRenderViewHost(),
[email protected]8bcdec92009-02-25 16:15:18606 L"",
[email protected]45671612009-04-29 22:24:01607 L"window.domAutomationController.send(getFocusedElement());",
608 &actual));
[email protected]546ae4e02010-12-08 14:57:19609 ASSERT_STREQ(next_element, actual.c_str());
initial.commit09911bf2008-07-26 23:55:29610 }
[email protected]8bcdec92009-02-25 16:15:18611
612 // At this point the renderer has sent us a message asking to advance the
613 // focus (as the end of the focus loop was reached in the renderer).
614 // We need to run the message loop to process it.
[email protected]f07d7bf2010-04-06 08:02:42615 ui_test_utils::RunAllPendingInMessageLoop();
initial.commit09911bf2008-07-26 23:55:29616 }
617}
618
[email protected]9e0c83a2009-05-06 19:44:37619// Focus traversal while an interstitial is showing.
[email protected]fc2e0872009-08-21 22:14:41620IN_PROC_BROWSER_TEST_F(BrowserFocusTest, MAYBE_FocusTraversalOnInterstitial) {
[email protected]a6e602f2010-09-28 22:28:30621 ASSERT_TRUE(ui_test_utils::BringBrowserWindowToFront(browser()));
[email protected]95409e12010-08-17 20:07:11622 ASSERT_TRUE(test_server()->Start());
[email protected]9e0c83a2009-05-06 19:44:37623
624 // First we navigate to our test page.
[email protected]95409e12010-08-17 20:07:11625 GURL url = test_server()->GetURL(kSimplePage);
[email protected]9e0c83a2009-05-06 19:44:37626 ui_test_utils::NavigateToURL(browser(), url);
627
[email protected]9e0c83a2009-05-06 19:44:37628 // Focus should be on the page.
[email protected]f2159ba2012-04-17 19:13:21629 ASSERT_TRUE(IsViewFocused(VIEW_ID_TAB_CONTAINER));
[email protected]9e0c83a2009-05-06 19:44:37630
631 // Let's show an interstitial.
632 TestInterstitialPage* interstitial_page =
[email protected]52877dbc62012-06-29 22:22:03633 new TestInterstitialPage(chrome::GetActiveWebContents(browser()),
[email protected]9e0c83a2009-05-06 19:44:37634 true, GURL("http://interstitial.com"));
[email protected]9e0c83a2009-05-06 19:44:37635 // Give some time for the interstitial to show.
636 MessageLoop::current()->PostDelayedTask(FROM_HERE,
[email protected]a778709f2011-12-10 00:28:17637 MessageLoop::QuitClosure(),
[email protected]7e560102012-03-08 20:58:42638 base::TimeDelta::FromSeconds(1));
[email protected]9e0c83a2009-05-06 19:44:37639 ui_test_utils::RunMessageLoop();
640
[email protected]a37d4b02012-06-25 21:56:10641 chrome::FocusLocationBar(browser());
[email protected]9e0c83a2009-05-06 19:44:37642
643 const char* kExpElementIDs[] = {
644 "", // Initially no element in the page should be focused
645 // (the location bar is focused).
646 "textEdit", "searchButton", "luckyButton", "googleLink", "gmailLink",
647 "gmapLink"
648 };
649
650 // Test forward focus traversal.
651 for (int i = 0; i < 2; ++i) {
652 // Location bar should be focused.
[email protected]0b8fa8b2011-12-07 00:54:52653 ASSERT_TRUE(IsViewFocused(location_bar_focus_view_id_));
[email protected]9e0c83a2009-05-06 19:44:37654
[email protected]911696b2011-01-28 02:36:49655 // Move the caret to the end, otherwise the next Tab key may not move focus.
656 ASSERT_TRUE(ui_test_utils::SendKeyPressSync(
657 browser(), ui::VKEY_END, false, false, false, false));
658
[email protected]9e0c83a2009-05-06 19:44:37659 // Now let's press tab to move the focus.
[email protected]130efb02009-09-18 18:54:35660 for (size_t j = 0; j < 7; ++j) {
[email protected]9e0c83a2009-05-06 19:44:37661 // Let's make sure the focus is on the expected element in the page.
[email protected]a6e602f2010-09-28 22:28:30662 std::string actual;
663 ASSERT_TRUE(ui_test_utils::ExecuteJavaScriptAndExtractString(
664 interstitial_page->render_view_host(), L"",
665 L"window.domAutomationController.send(getFocusedElement());",
666 &actual));
[email protected]9e0c83a2009-05-06 19:44:37667 ASSERT_STREQ(kExpElementIDs[j], actual.c_str());
668
[email protected]432115822011-07-10 15:52:27669 int notification_type;
[email protected]6c2381d2011-10-19 02:52:53670 content::NotificationSource notification_source =
[email protected]ad50def52011-10-19 23:17:07671 content::NotificationService::AllSources();
[email protected]130efb02009-09-18 18:54:35672 if (j < arraysize(kExpElementIDs) - 1) {
[email protected]432115822011-07-10 15:52:27673 notification_type = content::NOTIFICATION_FOCUS_CHANGED_IN_PAGE;
[email protected]d53092652011-12-06 06:12:56674 notification_source = content::Source<RenderViewHost>(
675 interstitial_page->render_view_host());
[email protected]130efb02009-09-18 18:54:35676 } else {
677 // On the last tab key press, the focus returns to the browser.
[email protected]432115822011-07-10 15:52:27678 notification_type = chrome::NOTIFICATION_FOCUS_RETURNED_TO_BROWSER;
[email protected]6c2381d2011-10-19 02:52:53679 notification_source = content::Source<Browser>(browser());
[email protected]130efb02009-09-18 18:54:35680 }
[email protected]a6e602f2010-09-28 22:28:30681
682 ASSERT_TRUE(ui_test_utils::SendKeyPressAndWait(
[email protected]b6d81262011-01-13 17:36:09683 browser(), ui::VKEY_TAB, false, false, false, false,
[email protected]a6e602f2010-09-28 22:28:30684 notification_type, notification_source));
[email protected]9e0c83a2009-05-06 19:44:37685 }
686
687 // At this point the renderer has sent us a message asking to advance the
688 // focus (as the end of the focus loop was reached in the renderer).
689 // We need to run the message loop to process it.
[email protected]f07d7bf2010-04-06 08:02:42690 ui_test_utils::RunAllPendingInMessageLoop();
[email protected]9e0c83a2009-05-06 19:44:37691 }
692
693 // Now let's try reverse focus traversal.
694 for (int i = 0; i < 2; ++i) {
695 // Location bar should be focused.
[email protected]0b8fa8b2011-12-07 00:54:52696 ASSERT_TRUE(IsViewFocused(location_bar_focus_view_id_));
[email protected]9e0c83a2009-05-06 19:44:37697
[email protected]911696b2011-01-28 02:36:49698 // Move the caret to the end, otherwise the next Tab key may not move focus.
699 ASSERT_TRUE(ui_test_utils::SendKeyPressSync(
700 browser(), ui::VKEY_END, false, false, false, false));
701
[email protected]9e0c83a2009-05-06 19:44:37702 // Now let's press shift-tab to move the focus in reverse.
[email protected]130efb02009-09-18 18:54:35703 for (size_t j = 0; j < 7; ++j) {
[email protected]432115822011-07-10 15:52:27704 int notification_type;
[email protected]6c2381d2011-10-19 02:52:53705 content::NotificationSource notification_source =
[email protected]ad50def52011-10-19 23:17:07706 content::NotificationService::AllSources();
[email protected]130efb02009-09-18 18:54:35707 if (j < arraysize(kExpElementIDs) - 1) {
[email protected]432115822011-07-10 15:52:27708 notification_type = content::NOTIFICATION_FOCUS_CHANGED_IN_PAGE;
[email protected]d53092652011-12-06 06:12:56709 notification_source = content::Source<RenderViewHost>(
710 interstitial_page->render_view_host());
[email protected]130efb02009-09-18 18:54:35711 } else {
712 // On the last tab key press, the focus returns to the browser.
[email protected]432115822011-07-10 15:52:27713 notification_type = chrome::NOTIFICATION_FOCUS_RETURNED_TO_BROWSER;
[email protected]6c2381d2011-10-19 02:52:53714 notification_source = content::Source<Browser>(browser());
[email protected]130efb02009-09-18 18:54:35715 }
[email protected]9e0c83a2009-05-06 19:44:37716
[email protected]a6e602f2010-09-28 22:28:30717 ASSERT_TRUE(ui_test_utils::SendKeyPressAndWait(
[email protected]b6d81262011-01-13 17:36:09718 browser(), ui::VKEY_TAB, false, true, false, false,
[email protected]a6e602f2010-09-28 22:28:30719 notification_type, notification_source));
720
[email protected]9e0c83a2009-05-06 19:44:37721 // Let's make sure the focus is on the expected element in the page.
[email protected]a6e602f2010-09-28 22:28:30722 std::string actual;
723 ASSERT_TRUE(ui_test_utils::ExecuteJavaScriptAndExtractString(
724 interstitial_page->render_view_host(), L"",
725 L"window.domAutomationController.send(getFocusedElement());",
726 &actual));
[email protected]9e0c83a2009-05-06 19:44:37727 ASSERT_STREQ(kExpElementIDs[6 - j], actual.c_str());
728 }
729
730 // At this point the renderer has sent us a message asking to advance the
731 // focus (as the end of the focus loop was reached in the renderer).
732 // We need to run the message loop to process it.
[email protected]f07d7bf2010-04-06 08:02:42733 ui_test_utils::RunAllPendingInMessageLoop();
[email protected]9e0c83a2009-05-06 19:44:37734 }
735}
736
737// Focus stays on page with interstitials.
[email protected]cb931932011-05-03 19:57:33738// http://crbug.com/81451
[email protected]fd5a53e2011-09-15 14:09:05739#if defined(OS_MACOSX) || defined(OS_WIN)
[email protected]4d2451652012-02-14 23:54:26740IN_PROC_BROWSER_TEST_F(BrowserFocusTest, DISABLED_InterstitialFocus) {
[email protected]cb931932011-05-03 19:57:33741#else
[email protected]e4f4e0b2009-10-13 19:58:21742IN_PROC_BROWSER_TEST_F(BrowserFocusTest, InterstitialFocus) {
[email protected]cb931932011-05-03 19:57:33743#endif
[email protected]a6e602f2010-09-28 22:28:30744 ASSERT_TRUE(ui_test_utils::BringBrowserWindowToFront(browser()));
[email protected]95409e12010-08-17 20:07:11745 ASSERT_TRUE(test_server()->Start());
[email protected]9e0c83a2009-05-06 19:44:37746
747 // First we navigate to our test page.
[email protected]95409e12010-08-17 20:07:11748 GURL url = test_server()->GetURL(kSimplePage);
[email protected]9e0c83a2009-05-06 19:44:37749 ui_test_utils::NavigateToURL(browser(), url);
750
[email protected]9e0c83a2009-05-06 19:44:37751 // Page should have focus.
[email protected]f2159ba2012-04-17 19:13:21752 ASSERT_TRUE(IsViewFocused(VIEW_ID_TAB_CONTAINER));
[email protected]52877dbc62012-06-29 22:22:03753 EXPECT_TRUE(chrome::GetActiveWebContents(browser())->GetRenderViewHost()->
[email protected]9f76c1e2012-03-05 15:15:58754 GetView()->HasFocus());
[email protected]9e0c83a2009-05-06 19:44:37755
[email protected]9d8a4642009-07-29 17:25:30756 // Let's show an interstitial.
[email protected]9e0c83a2009-05-06 19:44:37757 TestInterstitialPage* interstitial_page =
[email protected]52877dbc62012-06-29 22:22:03758 new TestInterstitialPage(chrome::GetActiveWebContents(browser()),
[email protected]9e0c83a2009-05-06 19:44:37759 true, GURL("http://interstitial.com"));
[email protected]9e0c83a2009-05-06 19:44:37760 // Give some time for the interstitial to show.
761 MessageLoop::current()->PostDelayedTask(FROM_HERE,
[email protected]a778709f2011-12-10 00:28:17762 MessageLoop::QuitClosure(),
[email protected]7e560102012-03-08 20:58:42763 base::TimeDelta::FromSeconds(1));
[email protected]9e0c83a2009-05-06 19:44:37764 ui_test_utils::RunMessageLoop();
765
766 // The interstitial should have focus now.
[email protected]f2159ba2012-04-17 19:13:21767 ASSERT_TRUE(IsViewFocused(VIEW_ID_TAB_CONTAINER));
[email protected]9e0c83a2009-05-06 19:44:37768 EXPECT_TRUE(interstitial_page->HasFocus());
769
770 // Hide the interstitial.
771 interstitial_page->DontProceed();
772
773 // Focus should be back on the original page.
[email protected]f2159ba2012-04-17 19:13:21774 ASSERT_TRUE(IsViewFocused(VIEW_ID_TAB_CONTAINER));
[email protected]9e0c83a2009-05-06 19:44:37775}
776
[email protected]9bd491ee2008-12-10 22:31:07777// Make sure Find box can request focus, even when it is already open.
[email protected]2004b762011-05-05 22:43:20778// Flaky on mac and valgrind. http://crbug.com/67301.
779#if defined(OS_MACOSX)
[email protected]4d2451652012-02-14 23:54:26780#define MAYBE_FindFocusTest DISABLED_FindFocusTest
[email protected]2004b762011-05-05 22:43:20781#else
782#define MAYBE_FindFocusTest FindFocusTest
783#endif
[email protected]c39e233d2012-05-18 13:24:12784IN_PROC_BROWSER_TEST_F(BrowserFocusTest, MAYBE_FindFocusTest) {
[email protected]a6e602f2010-09-28 22:28:30785 ASSERT_TRUE(ui_test_utils::BringBrowserWindowToFront(browser()));
[email protected]95409e12010-08-17 20:07:11786 ASSERT_TRUE(test_server()->Start());
license.botbf09a502008-08-24 00:55:55787
[email protected]9bd491ee2008-12-10 22:31:07788 // Open some page (any page that doesn't steal focus).
[email protected]95409e12010-08-17 20:07:11789 GURL url = test_server()->GetURL(kTypicalPage);
[email protected]8bcdec92009-02-25 16:15:18790 ui_test_utils::NavigateToURL(browser(), url);
[email protected]9bd491ee2008-12-10 22:31:07791
[email protected]a711df102010-12-14 09:47:03792 EXPECT_TRUE(ChromeInForeground());
793
[email protected]853300a82010-07-27 21:17:57794#if defined(OS_MACOSX)
795 // Press Cmd+F, which will make the Find box open and request focus.
[email protected]1d000682010-08-23 16:21:28796 ASSERT_TRUE(ui_test_utils::SendKeyPressSync(
[email protected]b6d81262011-01-13 17:36:09797 browser(), ui::VKEY_F, false, false, false, true));
[email protected]853300a82010-07-27 21:17:57798#else
[email protected]9bd491ee2008-12-10 22:31:07799 // Press Ctrl+F, which will make the Find box open and request focus.
[email protected]1d000682010-08-23 16:21:28800 ASSERT_TRUE(ui_test_utils::SendKeyPressSync(
[email protected]b6d81262011-01-13 17:36:09801 browser(), ui::VKEY_F, true, false, false, false));
[email protected]853300a82010-07-27 21:17:57802#endif
[email protected]8bcdec92009-02-25 16:15:18803
[email protected]a9c060ca2012-01-05 20:43:41804 ASSERT_TRUE(WaitForFocusChange(VIEW_ID_FIND_IN_PAGE_TEXT_FIELD));
[email protected]9bd491ee2008-12-10 22:31:07805
[email protected]a37d4b02012-06-25 21:56:10806 chrome::FocusLocationBar(browser());
[email protected]0b8fa8b2011-12-07 00:54:52807 ASSERT_TRUE(IsViewFocused(location_bar_focus_view_id_));
[email protected]9bd491ee2008-12-10 22:31:07808
809 // Now press Ctrl+F again and focus should move to the Find box.
[email protected]853300a82010-07-27 21:17:57810#if defined(OS_MACOSX)
[email protected]1d000682010-08-23 16:21:28811 ASSERT_TRUE(ui_test_utils::SendKeyPressSync(
[email protected]b6d81262011-01-13 17:36:09812 browser(), ui::VKEY_F, false, false, false, true));
[email protected]853300a82010-07-27 21:17:57813#else
[email protected]1d000682010-08-23 16:21:28814 ASSERT_TRUE(ui_test_utils::SendKeyPressSync(
[email protected]b6d81262011-01-13 17:36:09815 browser(), ui::VKEY_F, true, false, false, false));
[email protected]853300a82010-07-27 21:17:57816#endif
[email protected]21abcc742009-10-23 02:52:06817 ASSERT_TRUE(IsViewFocused(VIEW_ID_FIND_IN_PAGE_TEXT_FIELD));
[email protected]9bd491ee2008-12-10 22:31:07818
819 // Set focus to the page.
[email protected]fc2e0872009-08-21 22:14:41820 ClickOnView(VIEW_ID_TAB_CONTAINER);
[email protected]f2159ba2012-04-17 19:13:21821 ASSERT_TRUE(IsViewFocused(VIEW_ID_TAB_CONTAINER));
[email protected]9bd491ee2008-12-10 22:31:07822
823 // Now press Ctrl+F again and focus should move to the Find box.
[email protected]853300a82010-07-27 21:17:57824#if defined(OS_MACOSX)
[email protected]1d000682010-08-23 16:21:28825 ASSERT_TRUE(ui_test_utils::SendKeyPressSync(
[email protected]b6d81262011-01-13 17:36:09826 browser(), ui::VKEY_F, false, false, false, true));
[email protected]853300a82010-07-27 21:17:57827#else
[email protected]1d000682010-08-23 16:21:28828 ASSERT_TRUE(ui_test_utils::SendKeyPressSync(
[email protected]b6d81262011-01-13 17:36:09829 browser(), ui::VKEY_F, true, false, false, false));
[email protected]853300a82010-07-27 21:17:57830#endif
[email protected]8bcdec92009-02-25 16:15:18831
[email protected]a9c060ca2012-01-05 20:43:41832 ASSERT_TRUE(WaitForFocusChange(VIEW_ID_FIND_IN_PAGE_TEXT_FIELD));
[email protected]9bd491ee2008-12-10 22:31:07833}
[email protected]401513c2009-03-12 00:21:28834
835// Makes sure the focus is in the right location when opening the different
836// types of tabs.
[email protected]c92928672010-11-09 18:31:07837// Flaky, http://crbug.com/62539.
[email protected]4d2451652012-02-14 23:54:26838IN_PROC_BROWSER_TEST_F(BrowserFocusTest, DISABLED_TabInitialFocus) {
[email protected]a6e602f2010-09-28 22:28:30839 ASSERT_TRUE(ui_test_utils::BringBrowserWindowToFront(browser()));
[email protected]9ba21ede2010-07-30 01:11:07840
[email protected]401513c2009-03-12 00:21:28841 // Open the history tab, focus should be on the tab contents.
[email protected]5d9cace72012-06-21 16:07:12842 chrome::ShowHistory(browser());
[email protected]a3f343f2010-10-06 23:39:42843 ASSERT_NO_FATAL_FAILURE(ui_test_utils::WaitForLoadStop(
[email protected]52877dbc62012-06-29 22:22:03844 chrome::GetActiveWebContents(browser())));
[email protected]f2159ba2012-04-17 19:13:21845 EXPECT_TRUE(IsViewFocused(VIEW_ID_TAB_CONTAINER));
[email protected]401513c2009-03-12 00:21:28846
847 // Open the new tab, focus should be on the location bar.
[email protected]a37d4b02012-06-25 21:56:10848 chrome::NewTab(browser());
[email protected]a3f343f2010-10-06 23:39:42849 ASSERT_NO_FATAL_FAILURE(ui_test_utils::WaitForLoadStop(
[email protected]52877dbc62012-06-29 22:22:03850 chrome::GetActiveWebContents(browser())));
[email protected]0b8fa8b2011-12-07 00:54:52851 EXPECT_TRUE(IsViewFocused(location_bar_focus_view_id_));
[email protected]401513c2009-03-12 00:21:28852
853 // Open the download tab, focus should be on the tab contents.
[email protected]5d9cace72012-06-21 16:07:12854 chrome::ShowDownloads(browser());
[email protected]a3f343f2010-10-06 23:39:42855 ASSERT_NO_FATAL_FAILURE(ui_test_utils::WaitForLoadStop(
[email protected]52877dbc62012-06-29 22:22:03856 chrome::GetActiveWebContents(browser())));
[email protected]f2159ba2012-04-17 19:13:21857 EXPECT_TRUE(IsViewFocused(VIEW_ID_TAB_CONTAINER));
[email protected]3e3f0eb2009-06-22 18:33:43858
859 // Open about:blank, focus should be on the location bar.
[email protected]52877dbc62012-06-29 22:22:03860 chrome::AddSelectedTabWithURL(browser(), GURL(chrome::kAboutBlankURL),
861 content::PAGE_TRANSITION_LINK);
[email protected]a3f343f2010-10-06 23:39:42862 ASSERT_NO_FATAL_FAILURE(ui_test_utils::WaitForLoadStop(
[email protected]52877dbc62012-06-29 22:22:03863 chrome::GetActiveWebContents(browser())));
[email protected]0b8fa8b2011-12-07 00:54:52864 EXPECT_TRUE(IsViewFocused(location_bar_focus_view_id_));
[email protected]401513c2009-03-12 00:21:28865}
[email protected]9d8a4642009-07-29 17:25:30866
867// Tests that focus goes where expected when using reload.
[email protected]e4f4e0b2009-10-13 19:58:21868IN_PROC_BROWSER_TEST_F(BrowserFocusTest, FocusOnReload) {
[email protected]a6e602f2010-09-28 22:28:30869 ASSERT_TRUE(ui_test_utils::BringBrowserWindowToFront(browser()));
[email protected]95409e12010-08-17 20:07:11870 ASSERT_TRUE(test_server()->Start());
[email protected]9d8a4642009-07-29 17:25:30871
[email protected]9d8a4642009-07-29 17:25:30872 // Open the new tab, reload.
[email protected]6b4e8e42011-08-17 19:36:06873 {
[email protected]a7fe9112012-07-20 02:34:45874 content::WindowedNotificationObserver observer(
[email protected]6b4e8e42011-08-17 19:36:06875 content::NOTIFICATION_LOAD_STOP,
[email protected]ad50def52011-10-19 23:17:07876 content::NotificationService::AllSources());
[email protected]a37d4b02012-06-25 21:56:10877 chrome::NewTab(browser());
[email protected]6b4e8e42011-08-17 19:36:06878 observer.Wait();
879 }
[email protected]f07d7bf2010-04-06 08:02:42880 ui_test_utils::RunAllPendingInMessageLoop();
881
[email protected]6b4e8e42011-08-17 19:36:06882 {
[email protected]a7fe9112012-07-20 02:34:45883 content::WindowedNotificationObserver observer(
[email protected]6b4e8e42011-08-17 19:36:06884 content::NOTIFICATION_LOAD_STOP,
[email protected]c5eed492012-01-04 17:07:50885 content::Source<NavigationController>(
[email protected]52877dbc62012-06-29 22:22:03886 &chrome::GetActiveWebContents(browser())->GetController()));
[email protected]a37d4b02012-06-25 21:56:10887 chrome::Reload(browser(), CURRENT_TAB);
[email protected]6b4e8e42011-08-17 19:36:06888 observer.Wait();
889 }
[email protected]9d8a4642009-07-29 17:25:30890 // Focus should stay on the location bar.
[email protected]0b8fa8b2011-12-07 00:54:52891 ASSERT_TRUE(IsViewFocused(location_bar_focus_view_id_));
[email protected]9d8a4642009-07-29 17:25:30892
893 // Open a regular page, focus the location bar, reload.
[email protected]95409e12010-08-17 20:07:11894 ui_test_utils::NavigateToURL(browser(), test_server()->GetURL(kSimplePage));
[email protected]a37d4b02012-06-25 21:56:10895 chrome::FocusLocationBar(browser());
[email protected]0b8fa8b2011-12-07 00:54:52896 ASSERT_TRUE(IsViewFocused(location_bar_focus_view_id_));
[email protected]6b4e8e42011-08-17 19:36:06897 {
[email protected]a7fe9112012-07-20 02:34:45898 content::WindowedNotificationObserver observer(
[email protected]6b4e8e42011-08-17 19:36:06899 content::NOTIFICATION_LOAD_STOP,
[email protected]c5eed492012-01-04 17:07:50900 content::Source<NavigationController>(
[email protected]52877dbc62012-06-29 22:22:03901 &chrome::GetActiveWebContents(browser())->GetController()));
[email protected]a37d4b02012-06-25 21:56:10902 chrome::Reload(browser(), CURRENT_TAB);
[email protected]6b4e8e42011-08-17 19:36:06903 observer.Wait();
904 }
[email protected]9ba21ede2010-07-30 01:11:07905
[email protected]9d8a4642009-07-29 17:25:30906 // Focus should now be on the tab contents.
[email protected]5d9cace72012-06-21 16:07:12907 chrome::ShowDownloads(browser());
[email protected]f2159ba2012-04-17 19:13:21908 ASSERT_TRUE(IsViewFocused(VIEW_ID_TAB_CONTAINER));
[email protected]9d8a4642009-07-29 17:25:30909}
910
911// Tests that focus goes where expected when using reload on a crashed tab.
[email protected]160f29ad2010-10-28 15:43:27912IN_PROC_BROWSER_TEST_F(BrowserFocusTest, DISABLED_FocusOnReloadCrashedTab) {
[email protected]a6e602f2010-09-28 22:28:30913 ASSERT_TRUE(ui_test_utils::BringBrowserWindowToFront(browser()));
[email protected]95409e12010-08-17 20:07:11914 ASSERT_TRUE(test_server()->Start());
[email protected]9d8a4642009-07-29 17:25:30915
[email protected]9d8a4642009-07-29 17:25:30916 // Open a regular page, crash, reload.
[email protected]95409e12010-08-17 20:07:11917 ui_test_utils::NavigateToURL(browser(), test_server()->GetURL(kSimplePage));
[email protected]52877dbc62012-06-29 22:22:03918 ui_test_utils::CrashTab(chrome::GetActiveWebContents(browser()));
[email protected]6b4e8e42011-08-17 19:36:06919 {
[email protected]a7fe9112012-07-20 02:34:45920 content::WindowedNotificationObserver observer(
[email protected]6b4e8e42011-08-17 19:36:06921 content::NOTIFICATION_LOAD_STOP,
[email protected]c5eed492012-01-04 17:07:50922 content::Source<NavigationController>(
[email protected]52877dbc62012-06-29 22:22:03923 &chrome::GetActiveWebContents(browser())->GetController()));
[email protected]a37d4b02012-06-25 21:56:10924 chrome::Reload(browser(), CURRENT_TAB);
[email protected]6b4e8e42011-08-17 19:36:06925 observer.Wait();
926 }
[email protected]9ba21ede2010-07-30 01:11:07927
[email protected]9d8a4642009-07-29 17:25:30928 // Focus should now be on the tab contents.
[email protected]5d9cace72012-06-21 16:07:12929 chrome::ShowDownloads(browser());
[email protected]f2159ba2012-04-17 19:13:21930 ASSERT_TRUE(IsViewFocused(VIEW_ID_TAB_CONTAINER));
[email protected]9d8a4642009-07-29 17:25:30931}
[email protected]629e0342010-07-27 23:30:13932
933} // namespace