blob: 7c3fd78062c1d25654d9ccc008b53684f9c6e9b6 [file] [log] [blame]
// Copyright (c) 2014 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/settings_window_manager_chromeos.h"
#include "ash/constants/app_types.h"
#include "ash/public/cpp/resources/grit/ash_public_unscaled_resources.h"
#include "chrome/browser/app_mode/app_mode_utils.h"
#include "chrome/browser/apps/app_service/launch_utils.h"
#include "chrome/browser/ash/system_web_apps/types/system_web_app_type.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/ui/ash/window_properties.h"
#include "chrome/browser/ui/browser.h"
#include "chrome/browser/ui/browser_finder.h"
#include "chrome/browser/ui/browser_navigator.h"
#include "chrome/browser/ui/browser_navigator_params.h"
#include "chrome/browser/ui/browser_window.h"
#include "chrome/browser/ui/chrome_pages.h"
#include "chrome/browser/ui/settings_window_manager_observer_chromeos.h"
#include "chrome/browser/ui/tabs/tab_strip_model.h"
#include "chrome/browser/ui/web_applications/app_browser_controller.h"
#include "chrome/browser/ui/web_applications/system_web_app_ui_utils.h"
#include "chrome/browser/web_applications/web_app_utils.h"
#include "chrome/common/webui_url_constants.h"
#include "content/public/browser/web_contents.h"
#include "ui/aura/client/aura_constants.h"
#include "url/gurl.h"
namespace chrome {
namespace {
bool g_force_deprecated_settings_window_for_testing = false;
SettingsWindowManager* g_settings_window_manager_for_testing = nullptr;
} // namespace
// static
SettingsWindowManager* SettingsWindowManager::GetInstance() {
return g_settings_window_manager_for_testing
? g_settings_window_manager_for_testing
: base::Singleton<SettingsWindowManager>::get();
}
// static
void SettingsWindowManager::SetInstanceForTesting(
SettingsWindowManager* manager) {
g_settings_window_manager_for_testing = manager;
}
// static
void SettingsWindowManager::ForceDeprecatedSettingsWindowForTesting() {
g_force_deprecated_settings_window_for_testing = true;
}
// static
bool SettingsWindowManager::UseDeprecatedSettingsWindow(
const Profile* profile) {
return !web_app::AreWebAppsEnabled(profile) ||
chrome::IsRunningInForcedAppMode() ||
g_force_deprecated_settings_window_for_testing;
}
void SettingsWindowManager::AddObserver(
SettingsWindowManagerObserver* observer) {
observers_.AddObserver(observer);
}
void SettingsWindowManager::RemoveObserver(
SettingsWindowManagerObserver* observer) {
observers_.RemoveObserver(observer);
}
void SettingsWindowManager::ShowChromePageForProfile(Profile* profile,
const GURL& gurl,
int64_t display_id) {
// Use the original (non off-the-record) profile for settings unless
// this is a guest session.
if (!profile->IsGuestSession() && profile->IsOffTheRecord())
profile = profile->GetOriginalProfile();
// If this profile isn't allowed to create browser windows (e.g. the login
// screen profile) then bail out. Neither the new SWA code path nor the legacy
// code path can successfully open the window for these profiles.
if (Browser::GetCreationStatusForProfile(profile) !=
Browser::CreationStatus::kOk) {
LOG(ERROR) << "Unable to open settings for this profile, url "
<< gurl.spec();
return;
}
// TODO(crbug.com/1067073): Remove legacy Settings Window.
if (!UseDeprecatedSettingsWindow(profile)) {
web_app::SystemAppLaunchParams params;
params.url = gurl;
web_app::LaunchSystemWebAppAsync(profile, ash::SystemWebAppType::SETTINGS,
params, apps::MakeWindowInfo(display_id));
// SWA OS Settings don't use SettingsWindowManager to manage windows, don't
// notify SettingsWindowObservers.
return;
}
// Look for an existing browser window.
Browser* browser = FindBrowserForProfile(profile);
if (browser) {
DCHECK(browser->profile() == profile);
content::WebContents* web_contents =
browser->tab_strip_model()->GetWebContentsAt(0);
if (web_contents && web_contents->GetURL() == gurl) {
browser->window()->Show();
return;
}
NavigateParams params(browser, gurl, ui::PAGE_TRANSITION_AUTO_BOOKMARK);
params.window_action = NavigateParams::SHOW_WINDOW;
params.user_gesture = true;
Navigate(&params);
return;
}
// No existing browser window, create one.
NavigateParams params(profile, gurl, ui::PAGE_TRANSITION_AUTO_BOOKMARK);
params.disposition = WindowOpenDisposition::NEW_POPUP;
params.trusted_source = true;
params.window_action = NavigateParams::SHOW_WINDOW;
params.user_gesture = true;
params.path_behavior = NavigateParams::IGNORE_AND_NAVIGATE;
Navigate(&params);
browser = params.browser;
CHECK(browser); // See https://crbug.com/1174525
// operator[] not used because SessionID has no default constructor.
settings_session_map_.emplace(profile, SessionID::InvalidValue())
.first->second = browser->session_id();
DCHECK(browser->is_trusted_source());
auto* window = browser->window()->GetNativeWindow();
window->SetProperty(aura::client::kAppType,
static_cast<int>(ash::AppType::CHROME_APP));
window->SetProperty(kOverrideWindowIconResourceIdKey, IDR_SETTINGS_LOGO_192);
for (SettingsWindowManagerObserver& observer : observers_)
observer.OnNewSettingsWindow(browser);
}
void SettingsWindowManager::ShowOSSettings(Profile* profile,
int64_t display_id) {
ShowOSSettings(profile, std::string(), display_id);
}
void SettingsWindowManager::ShowOSSettings(Profile* profile,
const std::string& sub_page,
int64_t display_id) {
ShowChromePageForProfile(profile, chrome::GetOSSettingsUrl(sub_page),
display_id);
}
Browser* SettingsWindowManager::FindBrowserForProfile(Profile* profile) {
if (!UseDeprecatedSettingsWindow(profile)) {
return web_app::FindSystemWebAppBrowser(profile,
ash::SystemWebAppType::SETTINGS);
}
auto iter = settings_session_map_.find(profile);
if (iter != settings_session_map_.end())
return chrome::FindBrowserWithID(iter->second);
return nullptr;
}
bool SettingsWindowManager::IsSettingsBrowser(Browser* browser) const {
DCHECK(browser);
Profile* profile = browser->profile();
if (!UseDeprecatedSettingsWindow(profile)) {
if (!browser->app_controller())
return false;
// TODO(calamity): Determine whether, during startup, we need to wait for
// app install and then provide a valid answer here.
absl::optional<std::string> settings_app_id =
web_app::GetAppIdForSystemWebApp(profile,
ash::SystemWebAppType::SETTINGS);
return settings_app_id &&
browser->app_controller()->app_id() == settings_app_id.value();
} else {
auto iter = settings_session_map_.find(profile);
return iter != settings_session_map_.end() &&
iter->second == browser->session_id();
}
}
SettingsWindowManager::SettingsWindowManager() {}
SettingsWindowManager::~SettingsWindowManager() {}
} // namespace chrome