| // Copyright 2020 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 "ash/session/fullscreen_controller.h" |
| |
| #include <limits> |
| |
| #include "ash/constants/ash_features.h" |
| #include "ash/constants/ash_pref_names.h" |
| #include "ash/session/fullscreen_notification_bubble.h" |
| #include "ash/session/session_controller_impl.h" |
| #include "ash/shelf/shelf.h" |
| #include "ash/shell.h" |
| #include "ash/shell_delegate.h" |
| #include "ash/wm/window_state.h" |
| #include "ash/wm/wm_event.h" |
| #include "base/check.h" |
| #include "chromeos/dbus/power_manager/backlight.pb.h" |
| #include "chromeos/dbus/power_manager/idle.pb.h" |
| #include "components/prefs/pref_registry_simple.h" |
| #include "components/prefs/pref_service.h" |
| #include "components/url_matcher/url_matcher.h" |
| #include "components/url_matcher/url_util.h" |
| |
| namespace ash { |
| |
| FullscreenController::FullscreenController( |
| SessionControllerImpl* session_controller) |
| : session_controller_(session_controller) { |
| auto* power_manager = chromeos::PowerManagerClient::Get(); |
| // Might be nullptr in tests. |
| if (power_manager) { |
| power_manager->AddObserver(this); |
| } |
| } |
| |
| FullscreenController::~FullscreenController() { |
| auto* power_manager = chromeos::PowerManagerClient::Get(); |
| if (power_manager) { |
| power_manager->RemoveObserver(this); |
| } |
| } |
| |
| // static |
| void FullscreenController::MaybeExitFullscreen() { |
| // If the active window is fullscreen, exit fullscreen to avoid the web page |
| // or app mimicking the lock screen. Do not exit fullscreen if the shelf is |
| // visible while in fullscreen because the shelf makes it harder for a web |
| // page or app to mimic the lock screen. |
| WindowState* active_window_state = WindowState::ForActiveWindow(); |
| if (!active_window_state || !active_window_state->IsFullscreen()) |
| return; |
| |
| Shelf* shelf = Shelf::ForWindow(active_window_state->window()); |
| const bool shelf_visible = |
| shelf->GetVisibilityState() == ShelfVisibilityState::SHELF_VISIBLE; |
| |
| if (shelf_visible && !active_window_state->GetHideShelfWhenFullscreen()) |
| return; |
| |
| const WMEvent event(WM_EVENT_TOGGLE_FULLSCREEN); |
| active_window_state->OnWMEvent(&event); |
| } |
| |
| void FullscreenController::MaybeShowNotification() { |
| if (!features::IsFullscreenAlertBubbleEnabled()) |
| return; |
| |
| auto* session_controller = Shell::Get()->session_controller(); |
| |
| // Check if a user session is active to exclude OOBE process. |
| if (session_controller->GetSessionState() != |
| session_manager::SessionState::ACTIVE) { |
| return; |
| } |
| |
| auto* prefs = session_controller->GetPrimaryUserPrefService(); |
| |
| if (!prefs->GetBoolean(prefs::kFullscreenAlertEnabled)) |
| return; |
| |
| // Check if the activate window is fullscreen. |
| WindowState* active_window_state = WindowState::ForActiveWindow(); |
| if (!active_window_state || !active_window_state->IsFullscreen()) |
| return; |
| |
| // Check if the shelf is visible. |
| Shelf* shelf = Shelf::ForWindow(active_window_state->window()); |
| const bool shelf_visible = |
| shelf->GetVisibilityState() == ShelfVisibilityState::SHELF_VISIBLE; |
| |
| if (shelf_visible && !active_window_state->GetHideShelfWhenFullscreen()) |
| return; |
| |
| if (!bubble_) |
| bubble_ = std::make_unique<FullscreenNotificationBubble>(); |
| |
| bubble_->ShowForWindowState(active_window_state); |
| } |
| |
| // static |
| void FullscreenController::RegisterProfilePrefs(PrefRegistrySimple* registry) { |
| registry->RegisterBooleanPref(prefs::kFullscreenAlertEnabled, true, |
| PrefRegistry::PUBLIC); |
| registry->RegisterListPref( |
| prefs::kKeepFullscreenWithoutNotificationUrlAllowList, |
| PrefRegistry::PUBLIC); |
| } |
| |
| void FullscreenController::SuspendImminent( |
| power_manager::SuspendImminent::Reason reason) { |
| if (session_controller_->login_status() != LoginStatus::GUEST) |
| return; |
| |
| MaybeExitFullscreen(); |
| } |
| |
| void FullscreenController::ScreenIdleStateChanged( |
| const power_manager::ScreenIdleState& proto) { |
| if (session_controller_->login_status() != LoginStatus::GUEST) |
| return; |
| |
| if (proto.off() || proto.dimmed()) |
| MaybeExitFullscreen(); |
| } |
| |
| void FullscreenController::ScreenBrightnessChanged( |
| const power_manager::BacklightBrightnessChange& change) { |
| // Show alert when the device returns from low (epsilon) brightness which |
| // covers three cases. |
| // 1. The device returns from sleep. |
| // 2. The device lid is opended (with sleep on). |
| // 3. The device returns from low display brightness. |
| double epsilon = std::numeric_limits<double>::epsilon(); |
| if (change.percent() <= epsilon) { |
| device_in_dark_ = true; |
| } else { |
| if (device_in_dark_) |
| MaybeShowNotification(); |
| device_in_dark_ = false; |
| } |
| } |
| |
| void FullscreenController::LidEventReceived( |
| chromeos::PowerManagerClient::LidState state, |
| base::TimeTicks timestamp) { |
| // Show alert when the lid is opened. This also covers the case when the user |
| // turns off "Sleep when cover is closed". |
| if (state == chromeos::PowerManagerClient::LidState::OPEN) |
| MaybeShowNotification(); |
| } |
| |
| // static |
| bool FullscreenController::ShouldExitFullscreenBeforeLock() { |
| // Check whether it is allowed to keep full screen on unlock. |
| if (!features::IsFullscreenAfterUnlockAllowed()) |
| return true; |
| |
| // Nothing to do if the active window is not in full screen mode. |
| WindowState* active_window_state = WindowState::ForActiveWindow(); |
| if (!active_window_state || !active_window_state->IsFullscreen()) |
| return false; |
| |
| // Always exit full screen if the allowlist policy is unset. |
| auto* prefs = Shell::Get()->session_controller()->GetPrimaryUserPrefService(); |
| const auto* url_allow_list = |
| prefs->GetList(prefs::kKeepFullscreenWithoutNotificationUrlAllowList); |
| if (url_allow_list->GetListDeprecated().size() == 0) |
| return true; |
| |
| // Get the URL of the active window from the shell delegate. |
| const GURL& url = |
| Shell::Get()->shell_delegate()->GetLastCommittedURLForWindowIfAny( |
| active_window_state->window()); |
| |
| // Check if it is allowed by user pref to keep full screen for the active URL. |
| url_matcher::URLMatcher url_matcher; |
| url_matcher::util::AddAllowFilters( |
| &url_matcher, &base::Value::AsListValue(*url_allow_list)); |
| return url_matcher.MatchURL(url).empty(); |
| } |
| |
| } // namespace ash |