| // Copyright (c) 2012 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/version_handler.h" |
| |
| #include "base/command_line.h" |
| #include "base/file_util.h" |
| #include "base/metrics/field_trial.h" |
| #include "base/strings/string_util.h" |
| #include "base/strings/utf_string_conversions.h" |
| #include "chrome/browser/plugins/plugin_prefs.h" |
| #include "chrome/browser/profiles/profile.h" |
| #include "components/variations/active_field_trials.h" |
| #include "content/public/browser/browser_thread.h" |
| #include "content/public/browser/plugin_service.h" |
| #include "content/public/browser/web_ui.h" |
| #include "content/public/common/content_constants.h" |
| #include "grit/generated_resources.h" |
| #include "ui/base/l10n/l10n_util.h" |
| #include "url/gurl.h" |
| |
| namespace { |
| |
| // Retrieves the executable and profile paths on the FILE thread. |
| void GetFilePaths(const base::FilePath& profile_path, |
| base::string16* exec_path_out, |
| base::string16* profile_path_out) { |
| DCHECK_CURRENTLY_ON(content::BrowserThread::FILE); |
| |
| base::FilePath executable_path = base::MakeAbsoluteFilePath( |
| CommandLine::ForCurrentProcess()->GetProgram()); |
| if (!executable_path.empty()) { |
| *exec_path_out = executable_path.LossyDisplayName(); |
| } else { |
| *exec_path_out = |
| l10n_util::GetStringUTF16(IDS_ABOUT_VERSION_PATH_NOTFOUND); |
| } |
| |
| base::FilePath profile_path_copy(base::MakeAbsoluteFilePath(profile_path)); |
| if (!profile_path.empty() && !profile_path_copy.empty()) { |
| *profile_path_out = profile_path.LossyDisplayName(); |
| } else { |
| *profile_path_out = |
| l10n_util::GetStringUTF16(IDS_ABOUT_VERSION_PATH_NOTFOUND); |
| } |
| } |
| |
| } // namespace |
| |
| VersionHandler::VersionHandler() |
| : weak_ptr_factory_(this) { |
| } |
| |
| VersionHandler::~VersionHandler() { |
| } |
| |
| void VersionHandler::RegisterMessages() { |
| web_ui()->RegisterMessageCallback( |
| "requestVersionInfo", |
| base::Bind(&VersionHandler::HandleRequestVersionInfo, |
| base::Unretained(this))); |
| } |
| |
| void VersionHandler::HandleRequestVersionInfo(const base::ListValue* args) { |
| #if defined(ENABLE_PLUGINS) |
| // The Flash version information is needed in the response, so make sure |
| // the plugins are loaded. |
| content::PluginService::GetInstance()->GetPlugins( |
| base::Bind(&VersionHandler::OnGotPlugins, |
| weak_ptr_factory_.GetWeakPtr())); |
| #endif |
| |
| // Grab the executable path on the FILE thread. It is returned in |
| // OnGotFilePaths. |
| base::string16* exec_path_buffer = new base::string16; |
| base::string16* profile_path_buffer = new base::string16; |
| content::BrowserThread::PostTaskAndReply( |
| content::BrowserThread::FILE, FROM_HERE, |
| base::Bind(&GetFilePaths, Profile::FromWebUI(web_ui())->GetPath(), |
| base::Unretained(exec_path_buffer), |
| base::Unretained(profile_path_buffer)), |
| base::Bind(&VersionHandler::OnGotFilePaths, |
| weak_ptr_factory_.GetWeakPtr(), |
| base::Owned(exec_path_buffer), |
| base::Owned(profile_path_buffer))); |
| |
| // Respond with the variations info immediately. |
| std::vector<std::string> variations; |
| #if !defined(NDEBUG) |
| base::FieldTrial::ActiveGroups active_groups; |
| base::FieldTrialList::GetActiveFieldTrialGroups(&active_groups); |
| |
| const unsigned char kNonBreakingHyphenUTF8[] = { 0xE2, 0x80, 0x91, '\0' }; |
| const std::string kNonBreakingHyphenUTF8String( |
| reinterpret_cast<const char*>(kNonBreakingHyphenUTF8)); |
| for (size_t i = 0; i < active_groups.size(); ++i) { |
| std::string line = active_groups[i].trial_name + ":" + |
| active_groups[i].group_name; |
| base::ReplaceChars(line, "-", kNonBreakingHyphenUTF8String, &line); |
| variations.push_back(line); |
| } |
| #else |
| // In release mode, display the hashes only. |
| variations::GetFieldTrialActiveGroupIdsAsStrings(&variations); |
| #endif |
| |
| base::ListValue variations_list; |
| for (std::vector<std::string>::const_iterator it = variations.begin(); |
| it != variations.end(); ++it) { |
| variations_list.Append(new base::StringValue(*it)); |
| } |
| |
| // In release mode, this will return an empty list to clear the section. |
| web_ui()->CallJavascriptFunction("returnVariationInfo", variations_list); |
| } |
| |
| void VersionHandler::OnGotFilePaths(base::string16* executable_path_data, |
| base::string16* profile_path_data) { |
| DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
| |
| base::StringValue exec_path(*executable_path_data); |
| base::StringValue profile_path(*profile_path_data); |
| web_ui()->CallJavascriptFunction("returnFilePaths", exec_path, profile_path); |
| } |
| |
| #if defined(ENABLE_PLUGINS) |
| void VersionHandler::OnGotPlugins( |
| const std::vector<content::WebPluginInfo>& plugins) { |
| // Obtain the version of the first enabled Flash plugin. |
| std::vector<content::WebPluginInfo> info_array; |
| content::PluginService::GetInstance()->GetPluginInfoArray( |
| GURL(), content::kFlashPluginSwfMimeType, false, &info_array, NULL); |
| base::string16 flash_version = |
| l10n_util::GetStringUTF16(IDS_PLUGINS_DISABLED_PLUGIN); |
| PluginPrefs* plugin_prefs = |
| PluginPrefs::GetForProfile(Profile::FromWebUI(web_ui())).get(); |
| if (plugin_prefs) { |
| for (size_t i = 0; i < info_array.size(); ++i) { |
| if (plugin_prefs->IsPluginEnabled(info_array[i])) { |
| flash_version = info_array[i].version; |
| break; |
| } |
| } |
| } |
| |
| base::StringValue arg(flash_version); |
| web_ui()->CallJavascriptFunction("returnFlashVersion", arg); |
| } |
| #endif // defined(ENABLE_PLUGINS) |