blob: d4324ccb8ce804a59dbf210336fdc1d1e668dac6 [file] [log] [blame]
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/ui/exclusive_access/fullscreen_controller.h"
#include "base/functional/bind.h"
#include "base/test/simple_test_tick_clock.h"
#include "base/test/test_mock_time_task_runner.h"
#include "base/time/time.h"
#include "build/build_config.h"
#include "chrome/browser/content_settings/host_content_settings_map_factory.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/ui/browser.h"
#include "chrome/browser/ui/browser_commands.h"
#include "chrome/browser/ui/browser_tabstrip.h"
#include "chrome/browser/ui/browser_window.h"
#include "chrome/browser/ui/browser_window/public/browser_window_features.h"
#include "chrome/browser/ui/exclusive_access/exclusive_access_context.h"
#include "chrome/browser/ui/exclusive_access/exclusive_access_manager.h"
#include "chrome/browser/ui/exclusive_access/exclusive_access_test.h"
#include "chrome/browser/ui/tabs/tab_strip_model.h"
#include "chrome/browser/ui/ui_features.h"
#include "chrome/common/pref_names.h"
#include "chrome/test/base/ui_test_utils.h"
#include "components/content_settings/core/browser/host_content_settings_map.h"
#include "components/input/native_web_keyboard_event.h"
#include "components/prefs/pref_service.h"
#include "content/public/browser/web_contents.h"
#include "content/public/common/url_constants.h"
#include "content/public/test/browser_test.h"
#include "net/test/embedded_test_server/embedded_test_server.h"
#include "ui/events/event.h"
#include "ui/events/event_constants.h"
#include "ui/events/keycodes/keyboard_codes.h"
#if BUILDFLAG(IS_CHROMEOS)
#include "ash/wm/window_pin_util.h"
#endif
using content::WebContents;
using ui::PAGE_TRANSITION_TYPED;
using FullscreenControllerTest = ExclusiveAccessTest;
namespace {
// In some environments (Linux and Mac) the operation is finished asynchronously
// and we have to wait until the state change has occurred.
void WaitForDisplayed(Browser* browser) {
base::RunLoop outer_loop;
auto wait_for_state = base::BindRepeating(
[](base::RunLoop* outer_loop, Browser* browser) {
ExclusiveAccessManager* manager =
browser->GetFeatures().exclusive_access_manager();
if (manager->context()->IsExclusiveAccessBubbleDisplayed()) {
outer_loop->Quit();
}
},
&outer_loop, browser);
base::RepeatingTimer timer;
timer.Start(FROM_HERE, base::Milliseconds(1), std::move(wait_for_state));
outer_loop.Run();
}
} // namespace
//
// Fullscreen tests.
//
#if BUILDFLAG(IS_MAC)
#define MAYBE_FullscreenOnFileURL DISABLED_FullscreenOnFileURL
#else
#define MAYBE_FullscreenOnFileURL FullscreenOnFileURL
#endif
// TODO(https://crbug.com/330729275): Re-enable when fixed on macOS 14.
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest, MAYBE_FullscreenOnFileURL) {
static const base::FilePath::CharType* kEmptyFile =
FILE_PATH_LITERAL("empty.html");
GURL file_url(ui_test_utils::GetTestUrl(
base::FilePath(base::FilePath::kCurrentDirectory),
base::FilePath(kEmptyFile)));
ASSERT_TRUE(AddTabAtIndex(0, file_url, PAGE_TRANSITION_TYPED));
GetFullscreenController()->EnterFullscreenModeForTab(
browser()
->tab_strip_model()
->GetActiveWebContents()
->GetPrimaryMainFrame());
WaitForDisplayed(browser());
ASSERT_TRUE(IsExclusiveAccessBubbleDisplayed());
}
//
// KeyboardLock fullscreen tests.
//
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest, KeyboardLockWithEscLocked) {
EnterActiveTabFullscreen();
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
ASSERT_EQ(EXCLUSIVE_ACCESS_BUBBLE_TYPE_KEYBOARD_LOCK_EXIT_INSTRUCTION,
GetExclusiveAccessBubbleType());
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest, KeyboardLockWithEscUnlocked) {
EnterActiveTabFullscreen();
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/false));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
ASSERT_EQ(EXCLUSIVE_ACCESS_BUBBLE_TYPE_FULLSCREEN_EXIT_INSTRUCTION,
GetExclusiveAccessBubbleType());
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest,
KeyboardLockOnFileURLWithEscLocked) {
static const base::FilePath::CharType* kEmptyFile =
FILE_PATH_LITERAL("empty.html");
GURL file_url(ui_test_utils::GetTestUrl(
base::FilePath(base::FilePath::kCurrentDirectory),
base::FilePath(kEmptyFile)));
ASSERT_TRUE(AddTabAtIndex(0, file_url, PAGE_TRANSITION_TYPED));
EnterActiveTabFullscreen();
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
ASSERT_EQ(EXCLUSIVE_ACCESS_BUBBLE_TYPE_KEYBOARD_LOCK_EXIT_INSTRUCTION,
GetExclusiveAccessBubbleType());
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest,
KeyboardLockOnFileURLWithEscUnlocked) {
static const base::FilePath::CharType* kEmptyFile =
FILE_PATH_LITERAL("empty.html");
GURL file_url(ui_test_utils::GetTestUrl(
base::FilePath(base::FilePath::kCurrentDirectory),
base::FilePath(kEmptyFile)));
ASSERT_TRUE(AddTabAtIndex(0, file_url, PAGE_TRANSITION_TYPED));
EnterActiveTabFullscreen();
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/false));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
ASSERT_EQ(EXCLUSIVE_ACCESS_BUBBLE_TYPE_FULLSCREEN_EXIT_INSTRUCTION,
GetExclusiveAccessBubbleType());
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest,
KeyboardLockNotLockedInWindowMode) {
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
ASSERT_FALSE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
ASSERT_EQ(EXCLUSIVE_ACCESS_BUBBLE_TYPE_NONE, GetExclusiveAccessBubbleType());
EnterActiveTabFullscreen();
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest,
KeyboardLockExitsOnEscPressWhenEscNotLocked) {
EnterActiveTabFullscreen();
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/false));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
SendEscapeToExclusiveAccessManager();
ASSERT_FALSE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest,
KeyboardLockDoesNotExitOnEscPressWhenEscIsLocked) {
EnterActiveTabFullscreen();
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
SendEscapeToExclusiveAccessManager();
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest,
KeyboardLockNotLockedInExtensionFullscreenMode) {
EnterExtensionInitiatedFullscreen();
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
ASSERT_FALSE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
ASSERT_TRUE(IsExclusiveAccessBubbleDisplayed());
ASSERT_NE(EXCLUSIVE_ACCESS_BUBBLE_TYPE_KEYBOARD_LOCK_EXIT_INSTRUCTION,
GetExclusiveAccessBubbleType());
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest,
KeyboardLockNotLockedAfterFullscreenTransition) {
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
EnterActiveTabFullscreen();
ASSERT_FALSE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
ASSERT_TRUE(IsExclusiveAccessBubbleDisplayed());
ASSERT_NE(EXCLUSIVE_ACCESS_BUBBLE_TYPE_KEYBOARD_LOCK_EXIT_INSTRUCTION,
GetExclusiveAccessBubbleType());
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest,
KeyboardLockBubbleHideCallbackUnlock) {
EnterActiveTabFullscreen();
keyboard_lock_bubble_hide_reason_recorder_.clear();
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
ASSERT_EQ(0ul, keyboard_lock_bubble_hide_reason_recorder_.size());
CancelKeyboardLock();
ASSERT_EQ(1ul, keyboard_lock_bubble_hide_reason_recorder_.size());
ASSERT_EQ(ExclusiveAccessBubbleHideReason::kInterrupted,
keyboard_lock_bubble_hide_reason_recorder_[0]);
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest, FastKeyboardLockUnlockRelock) {
EnterActiveTabFullscreen();
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
// Shorter than `ExclusiveAccessBubble::kShowTime`.
Wait(ExclusiveAccessBubble::kShowTime / 2);
CancelKeyboardLock();
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
ASSERT_EQ(EXCLUSIVE_ACCESS_BUBBLE_TYPE_KEYBOARD_LOCK_EXIT_INSTRUCTION,
GetExclusiveAccessBubbleType());
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest, SlowKeyboardLockUnlockRelock) {
EnterActiveTabFullscreen();
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
// Longer than `ExclusiveAccessBubble::kShowTime`.
Wait(ExclusiveAccessBubble::kShowTime * 2);
CancelKeyboardLock();
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
ASSERT_EQ(EXCLUSIVE_ACCESS_BUBBLE_TYPE_KEYBOARD_LOCK_EXIT_INSTRUCTION,
GetExclusiveAccessBubbleType());
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest,
RepeatedEscEventsWithinWindowReshowsExitBubble) {
EnterActiveTabFullscreen();
base::SimpleTestTickClock clock;
SetEscRepeatTestTickClock(&clock);
bool esc_threshold_reached = false;
SetEscRepeatThresholdReachedCallback(base::BindOnce(
[](bool* triggered) { *triggered = true; }, &esc_threshold_reached));
// Set the window to a known value for testing.
SetEscRepeatWindowLength(base::Seconds(1));
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
input::NativeWebKeyboardEvent key_down_event(
blink::WebKeyboardEvent::Type::kRawKeyDown,
blink::WebInputEvent::kNoModifiers,
blink::WebInputEvent::GetStaticTimeStampForTests());
key_down_event.windows_key_code = ui::VKEY_ESCAPE;
input::NativeWebKeyboardEvent key_up_event(
blink::WebKeyboardEvent::Type::kKeyUp, blink::WebInputEvent::kNoModifiers,
blink::WebInputEvent::GetStaticTimeStampForTests());
key_up_event.windows_key_code = ui::VKEY_ESCAPE;
// Total time for keypress events is 400ms which is inside the window.
GetExclusiveAccessManager()->HandleUserKeyEvent(key_down_event);
// Keypresses are counted on the keyup event.
GetExclusiveAccessManager()->HandleUserKeyEvent(key_up_event);
ASSERT_FALSE(esc_threshold_reached);
clock.Advance(base::Milliseconds(100));
GetExclusiveAccessManager()->HandleUserKeyEvent(key_down_event);
clock.Advance(base::Milliseconds(100));
GetExclusiveAccessManager()->HandleUserKeyEvent(key_up_event);
ASSERT_FALSE(esc_threshold_reached);
clock.Advance(base::Milliseconds(100));
GetExclusiveAccessManager()->HandleUserKeyEvent(key_down_event);
clock.Advance(base::Milliseconds(100));
GetExclusiveAccessManager()->HandleUserKeyEvent(key_up_event);
ASSERT_TRUE(esc_threshold_reached);
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest,
RepeatedEscEventsOutsideWindowDoesNotShowExitBubble) {
EnterActiveTabFullscreen();
base::SimpleTestTickClock clock;
SetEscRepeatTestTickClock(&clock);
bool esc_threshold_reached = false;
SetEscRepeatThresholdReachedCallback(base::BindOnce(
[](bool* triggered) { *triggered = true; }, &esc_threshold_reached));
// Set the window to a known value for testing.
SetEscRepeatWindowLength(base::Seconds(1));
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
input::NativeWebKeyboardEvent key_down_event(
blink::WebKeyboardEvent::Type::kRawKeyDown,
blink::WebInputEvent::kNoModifiers,
blink::WebInputEvent::GetStaticTimeStampForTests());
key_down_event.windows_key_code = ui::VKEY_ESCAPE;
input::NativeWebKeyboardEvent key_up_event(
blink::WebKeyboardEvent::Type::kKeyUp, blink::WebInputEvent::kNoModifiers,
blink::WebInputEvent::GetStaticTimeStampForTests());
key_up_event.windows_key_code = ui::VKEY_ESCAPE;
// Total time for keypress events is 1200ms which is outside the window.
GetExclusiveAccessManager()->HandleUserKeyEvent(key_down_event);
// Keypresses are counted on the keyup event.
GetExclusiveAccessManager()->HandleUserKeyEvent(key_up_event);
ASSERT_FALSE(esc_threshold_reached);
clock.Advance(base::Milliseconds(400));
GetExclusiveAccessManager()->HandleUserKeyEvent(key_down_event);
clock.Advance(base::Milliseconds(200));
GetExclusiveAccessManager()->HandleUserKeyEvent(key_up_event);
ASSERT_FALSE(esc_threshold_reached);
clock.Advance(base::Milliseconds(400));
GetExclusiveAccessManager()->HandleUserKeyEvent(key_down_event);
clock.Advance(base::Milliseconds(200));
GetExclusiveAccessManager()->HandleUserKeyEvent(key_up_event);
ASSERT_FALSE(esc_threshold_reached);
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest, KeyboardLockAfterPointerLock) {
EnterActiveTabFullscreen();
RequestToLockPointer(/*user_gesture=*/true,
/*last_unlocked_by_target=*/false);
ASSERT_TRUE(IsExclusiveAccessBubbleDisplayed());
ASSERT_TRUE(GetExclusiveAccessManager()
->pointer_lock_controller()
->IsPointerLocked());
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/false));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
ASSERT_TRUE(GetExclusiveAccessManager()
->pointer_lock_controller()
->IsPointerLocked());
ASSERT_TRUE(IsExclusiveAccessBubbleDisplayed());
ASSERT_NE(EXCLUSIVE_ACCESS_BUBBLE_TYPE_KEYBOARD_LOCK_EXIT_INSTRUCTION,
GetExclusiveAccessBubbleType());
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest,
KeyboardLockAfterPointerLockWithEscLocked) {
EnterActiveTabFullscreen();
RequestToLockPointer(/*user_gesture=*/true,
/*last_unlocked_by_target=*/false);
ASSERT_TRUE(IsExclusiveAccessBubbleDisplayed());
ASSERT_TRUE(GetExclusiveAccessManager()
->pointer_lock_controller()
->IsPointerLocked());
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
ASSERT_EQ(EXCLUSIVE_ACCESS_BUBBLE_TYPE_KEYBOARD_LOCK_EXIT_INSTRUCTION,
GetExclusiveAccessBubbleType());
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest,
KeyboardLockCycleWithMixedEscLockStates) {
EnterActiveTabFullscreen();
keyboard_lock_bubble_hide_reason_recorder_.clear();
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
ASSERT_EQ(EXCLUSIVE_ACCESS_BUBBLE_TYPE_KEYBOARD_LOCK_EXIT_INSTRUCTION,
GetExclusiveAccessBubbleType());
ASSERT_EQ(0ul, keyboard_lock_bubble_hide_reason_recorder_.size());
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/false));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
ASSERT_EQ(EXCLUSIVE_ACCESS_BUBBLE_TYPE_FULLSCREEN_EXIT_INSTRUCTION,
GetExclusiveAccessBubbleType());
ASSERT_EQ(1ul, keyboard_lock_bubble_hide_reason_recorder_.size());
ASSERT_EQ(ExclusiveAccessBubbleHideReason::kInterrupted,
keyboard_lock_bubble_hide_reason_recorder_[0]);
keyboard_lock_bubble_hide_reason_recorder_.clear();
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/false));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
ASSERT_EQ(EXCLUSIVE_ACCESS_BUBBLE_TYPE_FULLSCREEN_EXIT_INSTRUCTION,
GetExclusiveAccessBubbleType());
ASSERT_EQ(0ul, keyboard_lock_bubble_hide_reason_recorder_.size());
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
ASSERT_EQ(EXCLUSIVE_ACCESS_BUBBLE_TYPE_KEYBOARD_LOCK_EXIT_INSTRUCTION,
GetExclusiveAccessBubbleType());
ASSERT_EQ(1ul, keyboard_lock_bubble_hide_reason_recorder_.size());
ASSERT_EQ(ExclusiveAccessBubbleHideReason::kInterrupted,
keyboard_lock_bubble_hide_reason_recorder_[0]);
keyboard_lock_bubble_hide_reason_recorder_.clear();
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
ASSERT_EQ(EXCLUSIVE_ACCESS_BUBBLE_TYPE_KEYBOARD_LOCK_EXIT_INSTRUCTION,
GetExclusiveAccessBubbleType());
ASSERT_EQ(0ul, keyboard_lock_bubble_hide_reason_recorder_.size());
}
// Test whether the top view's status is correct during various transitions
// among normal state, browser fullscreen mode, and tab fullscreen mode.
// Sheriff: http://crbug.com/925928
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest, DISABLED_TopViewStatusChange) {
ExclusiveAccessContext* context = GetExclusiveAccessManager()->context();
#if BUILDFLAG(IS_MAC)
// First, set the preference to true so we expect to see the top view in
// fullscreen mode.
PrefService* prefs = browser()->profile()->GetPrefs();
prefs->SetBoolean(prefs::kShowFullscreenToolbar, true);
#endif
// Test Normal state <--> Tab fullscreen mode.
EXPECT_FALSE(context->IsFullscreen());
EXPECT_TRUE(browser()->window()->IsToolbarVisible());
EnterActiveTabFullscreen();
EXPECT_TRUE(context->IsFullscreen());
EXPECT_FALSE(browser()->window()->IsToolbarVisible());
SendEscapeToExclusiveAccessManager();
EXPECT_FALSE(context->IsFullscreen());
EXPECT_TRUE(browser()->window()->IsToolbarVisible());
// Test Normal state <--> Browser fullscreen mode <--> Tab fullscreen mode.
ui_test_utils::ToggleFullscreenModeAndWait(browser());
EXPECT_TRUE(context->IsFullscreen());
#if BUILDFLAG(IS_MAC) || BUILDFLAG(IS_CHROMEOS)
bool should_show_top_ui = true;
#else
bool should_show_top_ui = false;
#endif
EXPECT_EQ(should_show_top_ui, browser()->window()->IsToolbarVisible());
EnterActiveTabFullscreen();
EXPECT_TRUE(context->IsFullscreen());
#if BUILDFLAG(IS_CHROMEOS)
EXPECT_TRUE(browser()->window()->IsToolbarVisible());
#else
EXPECT_FALSE(browser()->window()->IsToolbarVisible());
#endif
SendEscapeToExclusiveAccessManager();
EXPECT_TRUE(context->IsFullscreen());
EXPECT_EQ(should_show_top_ui, browser()->window()->IsToolbarVisible());
ui_test_utils::ToggleFullscreenModeAndWait(browser());
EXPECT_FALSE(context->IsFullscreen());
EXPECT_TRUE(browser()->window()->IsToolbarVisible());
// Test exiting tab fullscreen mode by toggling browser fullscreen mode.
// This is to simulate pressing fullscreen shortcut key during tab fullscreen
// mode across all platforms.
// On Mac, this happens by clicking green traffic light button to exit
// tab fullscreen.
EnterActiveTabFullscreen();
EXPECT_TRUE(context->IsFullscreen());
EXPECT_FALSE(browser()->window()->IsToolbarVisible());
ui_test_utils::ToggleFullscreenModeAndWait(browser());
EXPECT_FALSE(context->IsFullscreen());
EXPECT_TRUE(browser()->window()->IsToolbarVisible());
ui_test_utils::ToggleFullscreenModeAndWait(browser());
EXPECT_TRUE(context->IsFullscreen());
EXPECT_EQ(should_show_top_ui, browser()->window()->IsToolbarVisible());
}
// The controller must |CanEnterFullscreenModeForTab| while in fullscreen.
// While an element is in fullscreen, requesting fullscreen for a different
// element in the tab is handled in the renderer process if both elements are in
// the same process. But the request will come to the browser when the element
// is in a different process, such as OOPIF, because the renderer doesn't know
// if an element in other renderer process is in fullscreen. crbug.com/1298081
IN_PROC_BROWSER_TEST_F(FullscreenControllerTest,
EnterFullscreenWhenInFullscreen) {
EnterActiveTabFullscreen();
EXPECT_TRUE(GetFullscreenController()->CanEnterFullscreenModeForTab(
browser()
->tab_strip_model()
->GetActiveWebContents()
->GetPrimaryMainFrame()));
}
class FullscreenControllerPressAndHoldEscTest
: public FullscreenControllerTest {
public:
FullscreenControllerPressAndHoldEscTest() {
task_runner_ = base::MakeRefCounted<base::TestMockTimeTaskRunner>();
}
base::TestMockTimeTaskRunner* task_runner() { return task_runner_.get(); }
private:
base::test::ScopedFeatureList scoped_feature_list_{
features::kPressAndHoldEscToExitBrowserFullscreen};
scoped_refptr<base::TestMockTimeTaskRunner> task_runner_;
};
IN_PROC_BROWSER_TEST_F(FullscreenControllerPressAndHoldEscTest,
ExitBrowserFullscreenOnPressAndHoldEsc) {
// Enter browser fullscreen.
ui_test_utils::ToggleFullscreenModeAndWait(browser());
ASSERT_FALSE(IsWindowFullscreenForTabOrPending());
// Short-press Esc key won't exit browser fullscreen.
SendEscapeToExclusiveAccessManager(/*is_key_down=*/true);
SendEscapeToExclusiveAccessManager(/*is_key_down=*/false);
EXPECT_TRUE(IsFullscreenForBrowser());
// Press-and-hold Esc will exit browser fullscreen.
{
base::TestMockTimeTaskRunner::ScopedContext scoped_context(task_runner());
SendEscapeToExclusiveAccessManager(/*is_key_down=*/true);
task_runner()->FastForwardBy(base::Seconds(2));
}
WaitAndVerifyFullscreenState(/*browser_fullscreen=*/false,
/*tab_fullscreen=*/false);
}
#if BUILDFLAG(IS_CHROMEOS)
IN_PROC_BROWSER_TEST_F(FullscreenControllerPressAndHoldEscTest,
NotExitBrowserLockedFullscreenOnPressEsc) {
// Enter browser locked fullscreen.
ash::PinWindow(browser()->window()->GetNativeWindow(), /*trusted=*/true);
ASSERT_FALSE(IsWindowFullscreenForTabOrPending());
// Short-press Esc key won't exit browser locked fullscreen.
SendEscapeToExclusiveAccessManager(/*is_key_down=*/true);
SendEscapeToExclusiveAccessManager(/*is_key_down=*/false);
EXPECT_TRUE(IsFullscreenForBrowser());
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerPressAndHoldEscTest,
NotExitBrowserLockedFullscreenOnPressAndHoldEsc) {
// Enter browser locked fullscreen.
ash::PinWindow(browser()->window()->GetNativeWindow(), /*trusted=*/true);
ASSERT_FALSE(IsWindowFullscreenForTabOrPending());
// Press-and-hold Esc will not exit browser locked fullscreen.
{
base::TestMockTimeTaskRunner::ScopedContext scoped_context(task_runner());
SendEscapeToExclusiveAccessManager(/*is_key_down=*/true);
task_runner()->FastForwardBy(base::Seconds(2));
}
EXPECT_TRUE(IsFullscreenForBrowser());
}
#endif
IN_PROC_BROWSER_TEST_F(FullscreenControllerPressAndHoldEscTest,
ExitBrowserFullscreenOnMultipleEscKeyDown) {
// Enter browser fullscreen.
ui_test_utils::ToggleFullscreenModeAndWait(browser());
ASSERT_FALSE(IsWindowFullscreenForTabOrPending());
// Send repeating keydown events to simulate platform-specific behavior.
const base::Time start = task_runner()->Now();
{
base::TestMockTimeTaskRunner::ScopedContext scoped_context(task_runner());
while (IsFullscreenForBrowser()) {
SendEscapeToExclusiveAccessManager(/*is_key_down=*/true);
task_runner()->FastForwardBy(base::Milliseconds(300));
}
}
const base::TimeDelta time_to_exit = task_runner()->Now() - start;
// Fullscreen should exit about 1.5 seconds after the first keypress.
EXPECT_GT(time_to_exit, base::Seconds(1));
// Allow some time for the async `IsFullscreenForBrowser()` change.
EXPECT_LT(time_to_exit, base::Seconds(3));
}
IN_PROC_BROWSER_TEST_F(FullscreenControllerPressAndHoldEscTest,
ExitBrowserAndTabFullscreenOnPressAndHoldEsc) {
// Enter tab fullscreen and browser fullscreen.
GetFullscreenController()->ToggleBrowserFullscreenMode(
/*user_initiated=*/false);
GetFullscreenController()->EnterFullscreenModeForTab(
browser()
->tab_strip_model()
->GetActiveWebContents()
->GetPrimaryMainFrame(),
{});
WaitAndVerifyFullscreenState(/*browser_fullscreen=*/true,
/*tab_fullscreen=*/true);
// The first Esc key down event will exit tab fullscreen, but not browser
// fullscreen. Note that the key hasn't been released yet.
SendEscapeToExclusiveAccessManager(/*is_key_down=*/true);
WaitAndVerifyFullscreenState(/*browser_fullscreen=*/true,
/*tab_fullscreen=*/false);
// Press-and-hold Esc will exit browser fullscreen.
{
base::TestMockTimeTaskRunner::ScopedContext scoped_context(task_runner());
SendEscapeToExclusiveAccessManager(/*is_key_down=*/true);
task_runner()->FastForwardBy(base::Seconds(2));
}
WaitAndVerifyFullscreenState(/*browser_fullscreen=*/false,
/*tab_fullscreen=*/false);
}
IN_PROC_BROWSER_TEST_F(
FullscreenControllerPressAndHoldEscTest,
ExitBrowserFullscreenAndUnlockKeyboardOnPressAndHoldEsc) {
// Enter tab fullscreen and browser fullscreen. Then request keyboard lock
// with Esc locked.
GetFullscreenController()->ToggleBrowserFullscreenMode(
/*user_initiated=*/false);
GetFullscreenController()->EnterFullscreenModeForTab(
browser()
->tab_strip_model()
->GetActiveWebContents()
->GetPrimaryMainFrame(),
{});
WaitAndVerifyFullscreenState(/*browser_fullscreen=*/true,
/*tab_fullscreen=*/true);
ASSERT_TRUE(RequestKeyboardLock(/*esc_key_locked=*/true));
// Short-press Esc key will not do anything.
SendEscapeToExclusiveAccessManager(/*is_key_down=*/true);
SendEscapeToExclusiveAccessManager(/*is_key_down=*/false);
EXPECT_TRUE(IsWindowFullscreenForTabOrPending());
EXPECT_TRUE(IsFullscreenForBrowser());
ASSERT_TRUE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
// Press-and-hold Esc key will exit fullscreen and unlock the keyboard.
{
base::TestMockTimeTaskRunner::ScopedContext scoped_context(task_runner());
SendEscapeToExclusiveAccessManager(/*is_key_down=*/true);
task_runner()->FastForwardBy(base::Seconds(2));
}
WaitAndVerifyFullscreenState(/*browser_fullscreen=*/false,
/*tab_fullscreen=*/false);
EXPECT_FALSE(GetExclusiveAccessManager()
->keyboard_lock_controller()
->IsKeyboardLockActive());
}