| // Copyright 2018 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 "chrome/browser/ui/exclusive_access/keyboard_lock_controller.h" |
| |
| #include "base/bind.h" |
| #include "base/feature_list.h" |
| #include "base/metrics/histogram_macros.h" |
| #include "base/time/default_tick_clock.h" |
| #include "base/time/time.h" |
| #include "chrome/browser/chrome_notification_types.h" |
| #include "chrome/browser/ui/exclusive_access/exclusive_access_bubble_hide_callback.h" |
| #include "chrome/browser/ui/exclusive_access/exclusive_access_manager.h" |
| #include "chrome/browser/ui/exclusive_access/fullscreen_controller.h" |
| #include "chrome/common/chrome_features.h" |
| #include "content/public/browser/native_web_keyboard_event.h" |
| #include "content/public/browser/notification_service.h" |
| #include "content/public/browser/web_contents.h" |
| #include "content/public/common/content_features.h" |
| #include "ui/events/keycodes/keyboard_codes.h" |
| |
| using base::TimeDelta; |
| using base::TimeTicks; |
| using content::WebContents; |
| |
| namespace { |
| |
| const char kBubbleReshowsHistogramName[] = |
| "ExclusiveAccess.BubbleReshowsPerSession.KeyboardLock"; |
| |
| const char kForcedBubbleReshowsHistogramName[] = |
| "ExclusiveAccess.BubbleReshowsPerSession.KeyboardLockForced"; |
| |
| // Amount of time the user must hold ESC to exit full screen. |
| constexpr TimeDelta kHoldEscapeTime = TimeDelta::FromMilliseconds(1500); |
| |
| // Amount of time to look for ESC key presses to reshow the exit instructions. |
| constexpr TimeDelta kDefaultEscRepeatWindow = TimeDelta::FromSeconds(1); |
| |
| // Number of times ESC must be pressed within |kDefaultEscRepeatWindow| to |
| // trigger the exit instructions to be shown again. |
| constexpr int kEscRepeatCountToTriggerUiReshow = 3; |
| |
| } // namespace |
| |
| KeyboardLockController::KeyboardLockController(ExclusiveAccessManager* manager) |
| : ExclusiveAccessControllerBase(manager), |
| esc_repeat_window_(kDefaultEscRepeatWindow), |
| esc_repeat_tick_clock_(base::DefaultTickClock::GetInstance()) {} |
| |
| KeyboardLockController::~KeyboardLockController() = default; |
| |
| bool KeyboardLockController::HandleUserPressedEscape() { |
| if (!IsKeyboardLockActive()) |
| return false; |
| |
| UnlockKeyboard(); |
| return true; |
| } |
| |
| void KeyboardLockController::ExitExclusiveAccessToPreviousState() { |
| UnlockKeyboard(); |
| } |
| |
| void KeyboardLockController::ExitExclusiveAccessIfNecessary() { |
| UnlockKeyboard(); |
| } |
| |
| void KeyboardLockController::NotifyTabExclusiveAccessLost() { |
| UnlockKeyboard(); |
| } |
| |
| void KeyboardLockController::RecordBubbleReshowsHistogram(int reshow_count) { |
| UMA_HISTOGRAM_COUNTS_100(kBubbleReshowsHistogramName, reshow_count); |
| } |
| |
| void KeyboardLockController::RecordForcedBubbleReshowsHistogram() { |
| UMA_HISTOGRAM_COUNTS_100(kForcedBubbleReshowsHistogramName, |
| forced_reshow_count_); |
| forced_reshow_count_ = 0; |
| } |
| |
| bool KeyboardLockController::IsKeyboardLockActive() const { |
| DCHECK_EQ(keyboard_lock_state_ == KeyboardLockState::kUnlocked, |
| exclusive_access_tab() == nullptr); |
| return keyboard_lock_state_ != KeyboardLockState::kUnlocked; |
| } |
| |
| bool KeyboardLockController::RequiresPressAndHoldEscToExit() const { |
| DCHECK_EQ(keyboard_lock_state_ == KeyboardLockState::kUnlocked, |
| exclusive_access_tab() == nullptr); |
| return keyboard_lock_state_ == KeyboardLockState::kLockedWithEsc; |
| } |
| |
| void KeyboardLockController::RequestKeyboardLock(WebContents* web_contents, |
| bool esc_key_locked) { |
| if (!exclusive_access_manager() |
| ->fullscreen_controller() |
| ->IsFullscreenForTabOrPending(web_contents)) { |
| return; |
| } |
| |
| DCHECK(!exclusive_access_tab() || exclusive_access_tab() == web_contents); |
| |
| LockKeyboard(web_contents, esc_key_locked); |
| } |
| |
| bool KeyboardLockController::HandleKeyEvent( |
| const content::NativeWebKeyboardEvent& event) { |
| DCHECK_EQ(ui::VKEY_ESCAPE, event.windows_key_code); |
| // This method handles the press and hold gesture used for exiting fullscreen. |
| // If we don't have a feature which requires press and hold, or there isn't an |
| // active keyboard lock request which requires press and hold, then we just |
| // return as the simple 'press esc to exit' case is handled by the caller |
| // (which is the ExclusiveAccessManager in this case). |
| if (!RequiresPressAndHoldEscToExit()) |
| return false; |
| |
| // Note: This logic handles exiting fullscreen but the UI feedback element is |
| // created and managed by the FullscreenControlHost class. |
| if (event.GetType() == content::NativeWebKeyboardEvent::kKeyUp && |
| hold_timer_.IsRunning()) { |
| // Seeing a key up event on Esc with the hold timer running cancels the |
| // timer and doesn't exit. This means the user pressed Esc, but not long |
| // enough to trigger an exit |
| hold_timer_.Stop(); |
| ReShowExitBubbleIfNeeded(); |
| } else if (event.GetType() == content::NativeWebKeyboardEvent::kRawKeyDown && |
| !hold_timer_.IsRunning()) { |
| // Seeing a key down event on Esc when the hold timer is stopped starts |
| // the timer. When the timer fires, the callback will trigger an exit from |
| // fullscreen/mouselock/keyboardlock. |
| hold_timer_.Start( |
| FROM_HERE, kHoldEscapeTime, |
| base::BindRepeating(&KeyboardLockController::HandleUserHeldEscape, |
| base::Unretained(this))); |
| } |
| |
| return true; |
| } |
| |
| void KeyboardLockController::CancelKeyboardLockRequest(WebContents* tab) { |
| if (tab == exclusive_access_tab()) |
| UnlockKeyboard(); |
| } |
| |
| void KeyboardLockController::LostKeyboardLock() { |
| UnlockKeyboard(); |
| } |
| |
| void KeyboardLockController::LockKeyboard(content::WebContents* web_contents, |
| bool esc_key_locked) { |
| if (web_contents->GotResponseToKeyboardLockRequest(true)) { |
| KeyboardLockState new_lock_state = |
| esc_key_locked ? KeyboardLockState::kLockedWithEsc |
| : KeyboardLockState::kLockedWithoutEsc; |
| // Only re-show the exit bubble if the requesting web_contents has changed |
| // (or is new) or if the esc key lock state has changed. |
| bool reshow_exit_bubble = exclusive_access_tab() != web_contents || |
| new_lock_state != keyboard_lock_state_; |
| keyboard_lock_state_ = new_lock_state; |
| SetTabWithExclusiveAccess(web_contents); |
| if (reshow_exit_bubble) { |
| exclusive_access_manager()->UpdateExclusiveAccessExitBubbleContent( |
| bubble_hide_callback_for_test_ |
| ? base::BindOnce(bubble_hide_callback_for_test_) |
| : ExclusiveAccessBubbleHideCallback()); |
| } |
| } else { |
| UnlockKeyboard(); |
| } |
| } |
| |
| void KeyboardLockController::UnlockKeyboard() { |
| if (!exclusive_access_tab()) |
| return; |
| |
| RecordExitingUMA(); |
| RecordForcedBubbleReshowsHistogram(); |
| keyboard_lock_state_ = KeyboardLockState::kUnlocked; |
| |
| exclusive_access_tab()->GotResponseToKeyboardLockRequest(false); |
| SetTabWithExclusiveAccess(nullptr); |
| exclusive_access_manager()->UpdateExclusiveAccessExitBubbleContent( |
| ExclusiveAccessBubbleHideCallback()); |
| } |
| |
| void KeyboardLockController::HandleUserHeldEscape() { |
| ExclusiveAccessManager* const manager = exclusive_access_manager(); |
| manager->fullscreen_controller()->HandleUserPressedEscape(); |
| manager->mouse_lock_controller()->HandleUserPressedEscape(); |
| HandleUserPressedEscape(); |
| } |
| |
| void KeyboardLockController::ReShowExitBubbleIfNeeded() { |
| TimeTicks now = esc_repeat_tick_clock_->NowTicks(); |
| TimeTicks esc_repeat_window = now - esc_repeat_window_; |
| // Remove any events which are outside of the window. |
| while (!esc_keypress_tracker_.empty() && |
| esc_keypress_tracker_.front() < esc_repeat_window) { |
| esc_keypress_tracker_.pop_front(); |
| } |
| |
| esc_keypress_tracker_.push_back(now); |
| if (esc_keypress_tracker_.size() >= kEscRepeatCountToTriggerUiReshow) { |
| exclusive_access_manager()->UpdateExclusiveAccessExitBubbleContent( |
| ExclusiveAccessBubbleHideCallback(), /*force_update=*/true); |
| forced_reshow_count_++; |
| esc_keypress_tracker_.clear(); |
| |
| if (esc_repeat_triggered_for_test_) |
| std::move(esc_repeat_triggered_for_test_).Run(); |
| } |
| } |