// Copyright (c) 2011 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/webui/flags_ui.h" #include <string> #include "base/memory/singleton.h" #include "base/values.h" #include "chrome/browser/about_flags.h" #include "chrome/browser/browser_process.h" #include "chrome/browser/prefs/pref_service.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/ui/browser_list.h" #include "chrome/browser/ui/webui/chrome_url_data_manager.h" #include "chrome/common/jstemplate_builder.h" #include "chrome/common/pref_names.h" #include "chrome/common/url_constants.h" #include "content/browser/browser_thread.h" #include "content/browser/tab_contents/tab_contents.h" #include "grit/browser_resources.h" #include "grit/chromium_strings.h" #include "grit/generated_resources.h" #include "grit/theme_resources.h" #include "ui/base/l10n/l10n_util.h" #include "ui/base/resource/resource_bundle.h" #if defined(OS_CHROMEOS) #include "chrome/browser/chromeos/user_cros_settings_provider.h" #include "chrome/browser/chromeos/login/user_manager.h" #endif namespace { /////////////////////////////////////////////////////////////////////////////// // // FlagsUIHTMLSource // /////////////////////////////////////////////////////////////////////////////// class FlagsUIHTMLSource : public ChromeURLDataManager::DataSource { public: FlagsUIHTMLSource() : DataSource(chrome::kChromeUIFlagsHost, MessageLoop::current()) {} // Called when the network layer has requested a resource underneath // the path we registered. virtual void StartDataRequest(const std::string& path, bool is_incognito, int request_id); virtual std::string GetMimeType(const std::string&) const { return "text/html"; } private: ~FlagsUIHTMLSource() {} DISALLOW_COPY_AND_ASSIGN(FlagsUIHTMLSource); }; void FlagsUIHTMLSource::StartDataRequest(const std::string& path, bool is_incognito, int request_id) { // Strings used in the JsTemplate file. DictionaryValue localized_strings; localized_strings.SetString("flagsLongTitle", l10n_util::GetStringUTF16(IDS_FLAGS_LONG_TITLE)); localized_strings.SetString("flagsTableTitle", l10n_util::GetStringUTF16(IDS_FLAGS_TABLE_TITLE)); localized_strings.SetString("flagsNoExperimentsAvailable", l10n_util::GetStringUTF16(IDS_FLAGS_NO_EXPERIMENTS_AVAILABLE)); localized_strings.SetString("flagsWarningHeader", l10n_util::GetStringUTF16( IDS_FLAGS_WARNING_HEADER)); localized_strings.SetString("flagsBlurb", l10n_util::GetStringUTF16( IDS_FLAGS_WARNING_TEXT)); localized_strings.SetString("flagsRestartNotice", l10n_util::GetStringFUTF16( IDS_FLAGS_RELAUNCH_NOTICE, l10n_util::GetStringUTF16( #if defined(OS_CHROMEOS) IDS_PRODUCT_OS_NAME #else IDS_PRODUCT_NAME #endif ))); localized_strings.SetString("flagsRestartButton", l10n_util::GetStringUTF16(IDS_FLAGS_RELAUNCH_BUTTON)); localized_strings.SetString("disable", l10n_util::GetStringUTF16(IDS_FLAGS_DISABLE)); localized_strings.SetString("enable", l10n_util::GetStringUTF16(IDS_FLAGS_ENABLE)); base::StringPiece html = ResourceBundle::GetSharedInstance().GetRawDataResource(IDR_FLAGS_HTML); #if defined (OS_CHROMEOS) if (!chromeos::UserManager::Get()->current_user_is_owner()) { html = ResourceBundle::GetSharedInstance().GetRawDataResource( IDR_FLAGS_HTML_WARNING); // Set the strings to show which user can actually change the flags localized_strings.SetString("ownerOnly", l10n_util::GetStringUTF16( IDS_OPTIONS_ACCOUNTS_OWNER_ONLY)); localized_strings.SetString("ownerUserId", UTF8ToUTF16( chromeos::UserCrosSettingsProvider::cached_owner())); } #endif static const base::StringPiece flags_html(html); ChromeURLDataManager::DataSource::SetFontAndTextDirection(&localized_strings); std::string full_html(flags_html.data(), flags_html.size()); jstemplate_builder::AppendJsonHtml(&localized_strings, &full_html); jstemplate_builder::AppendI18nTemplateSourceHtml(&full_html); jstemplate_builder::AppendI18nTemplateProcessHtml(&full_html); jstemplate_builder::AppendJsTemplateSourceHtml(&full_html); scoped_refptr<RefCountedBytes> html_bytes(new RefCountedBytes); html_bytes->data.resize(full_html.size()); std::copy(full_html.begin(), full_html.end(), html_bytes->data.begin()); SendResponse(request_id, html_bytes); } //////////////////////////////////////////////////////////////////////////////// // // FlagsDOMHandler // //////////////////////////////////////////////////////////////////////////////// // The handler for Javascript messages for the about:flags page. class FlagsDOMHandler : public WebUIMessageHandler { public: FlagsDOMHandler() {} virtual ~FlagsDOMHandler() {} // WebUIMessageHandler implementation. virtual void RegisterMessages(); // Callback for the "requestFlagsExperiments" message. void HandleRequestFlagsExperiments(const ListValue* args); // Callback for the "enableFlagsExperiment" message. void HandleEnableFlagsExperimentMessage(const ListValue* args); // Callback for the "restartBrowser" message. Restores all tabs on restart. void HandleRestartBrowser(const ListValue* args); private: DISALLOW_COPY_AND_ASSIGN(FlagsDOMHandler); }; void FlagsDOMHandler::RegisterMessages() { web_ui_->RegisterMessageCallback("requestFlagsExperiments", NewCallback(this, &FlagsDOMHandler::HandleRequestFlagsExperiments)); web_ui_->RegisterMessageCallback("enableFlagsExperiment", NewCallback(this, &FlagsDOMHandler::HandleEnableFlagsExperimentMessage)); web_ui_->RegisterMessageCallback("restartBrowser", NewCallback(this, &FlagsDOMHandler::HandleRestartBrowser)); } void FlagsDOMHandler::HandleRequestFlagsExperiments(const ListValue* args) { DictionaryValue results; results.Set("flagsExperiments", about_flags::GetFlagsExperimentsData( g_browser_process->local_state())); results.SetBoolean("needsRestart", about_flags::IsRestartNeededToCommitChanges()); web_ui_->CallJavascriptFunction("returnFlagsExperiments", results); } void FlagsDOMHandler::HandleEnableFlagsExperimentMessage( const ListValue* args) { DCHECK_EQ(2u, args->GetSize()); if (args->GetSize() != 2) return; std::string experiment_internal_name; std::string enable_str; if (!args->GetString(0, &experiment_internal_name) || !args->GetString(1, &enable_str)) return; about_flags::SetExperimentEnabled( g_browser_process->local_state(), experiment_internal_name, enable_str == "true"); } void FlagsDOMHandler::HandleRestartBrowser(const ListValue* args) { #if !defined(OS_CHROMEOS) // Set the flag to restore state after the restart. PrefService* pref_service = g_browser_process->local_state(); pref_service->SetBoolean(prefs::kRestartLastSessionOnShutdown, true); BrowserList::CloseAllBrowsersAndExit(); #else // For CrOS instead of browser restart (which is not supported) perform a full // sign out. Session will be only restored is user has that setting set. // Same session restore behavior happens in case of full restart after update. BrowserList::GetLastActive()->Exit(); #endif } } // namespace /////////////////////////////////////////////////////////////////////////////// // // FlagsUI // /////////////////////////////////////////////////////////////////////////////// FlagsUI::FlagsUI(TabContents* contents) : WebUI(contents) { AddMessageHandler((new FlagsDOMHandler())->Attach(this)); FlagsUIHTMLSource* html_source = new FlagsUIHTMLSource(); // Set up the about:flags source. contents->profile()->GetChromeURLDataManager()->AddDataSource(html_source); } // static RefCountedMemory* FlagsUI::GetFaviconResourceBytes() { return ResourceBundle::GetSharedInstance(). LoadDataResourceBytes(IDR_FLAGS); } // static void FlagsUI::RegisterPrefs(PrefService* prefs) { prefs->RegisterListPref(prefs::kEnabledLabsExperiments); }