// 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/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 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(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::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& plugins) { // Obtain the version of the first enabled Flash plugin. std::vector 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)