// Copyright (c) 2010 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/download/download_prefs.h" #include "base/file_util.h" #include "base/string_split.h" #include "base/string_util.h" #include "base/sys_string_conversions.h" #include "base/utf_string_conversions.h" #include "chrome/browser/browser_thread.h" #include "chrome/browser/download/download_extensions.h" #include "chrome/browser/download/download_util.h" #include "chrome/browser/prefs/pref_service.h" #include "chrome/common/pref_names.h" DownloadPrefs::DownloadPrefs(PrefService* prefs) : prefs_(prefs) { prompt_for_download_.Init(prefs::kPromptForDownload, prefs, NULL); download_path_.Init(prefs::kDownloadDefaultDirectory, prefs, NULL); // We store any file extension that should be opened automatically at // download completion in this pref. std::string extensions_to_open = prefs->GetString(prefs::kDownloadExtensionsToOpen); std::vector extensions; base::SplitString(extensions_to_open, ':', &extensions); for (size_t i = 0; i < extensions.size(); ++i) { #if defined(OS_POSIX) FilePath path(extensions[i]); #elif defined(OS_WIN) FilePath path(UTF8ToWide(extensions[i])); #endif if (!extensions[i].empty() && download_util::IsFileSafe(path)) auto_open_.insert(path.value()); } } DownloadPrefs::~DownloadPrefs() { SaveAutoOpenState(); } // static void DownloadPrefs::RegisterUserPrefs(PrefService* prefs) { prefs->RegisterBooleanPref(prefs::kPromptForDownload, false); prefs->RegisterStringPref(prefs::kDownloadExtensionsToOpen, ""); prefs->RegisterBooleanPref(prefs::kDownloadDirUpgraded, false); // The default download path is userprofile\download. const FilePath& default_download_path = download_util::GetDefaultDownloadDirectory(); prefs->RegisterFilePathPref(prefs::kDownloadDefaultDirectory, default_download_path); #if defined(OS_CHROMEOS) // Ensure that the download directory specified in the preferences exists. BrowserThread::PostTask( BrowserThread::FILE, FROM_HERE, NewRunnableFunction(&file_util::CreateDirectory, default_download_path)); #endif // defined(OS_CHROMEOS) // If the download path is dangerous we forcefully reset it. But if we do // so we set a flag to make sure we only do it once, to avoid fighting // the user if he really wants it on an unsafe place such as the desktop. if (!prefs->GetBoolean(prefs::kDownloadDirUpgraded)) { FilePath current_download_dir = prefs->GetFilePath( prefs::kDownloadDefaultDirectory); if (download_util::DownloadPathIsDangerous(current_download_dir)) { prefs->SetFilePath(prefs::kDownloadDefaultDirectory, default_download_path); } prefs->SetBoolean(prefs::kDownloadDirUpgraded, true); } } bool DownloadPrefs::IsAutoOpenUsed() const { return !auto_open_.empty(); } bool DownloadPrefs::IsAutoOpenEnabledForExtension( const FilePath::StringType& extension) const { return auto_open_.find(extension) != auto_open_.end(); } bool DownloadPrefs::EnableAutoOpenBasedOnExtension(const FilePath& file_name) { FilePath::StringType extension = file_name.Extension(); if (extension.empty()) return false; DCHECK(extension[0] == FilePath::kExtensionSeparator); extension.erase(0, 1); if (!download_util::IsFileExtensionSafe(extension)) return false; auto_open_.insert(extension); SaveAutoOpenState(); return true; } void DownloadPrefs::DisableAutoOpenBasedOnExtension(const FilePath& file_name) { FilePath::StringType extension = file_name.Extension(); if (extension.empty()) return; DCHECK(extension[0] == FilePath::kExtensionSeparator); extension.erase(0, 1); auto_open_.erase(extension); SaveAutoOpenState(); } void DownloadPrefs::ResetToDefaults() { // TODO(phajdan.jr): Should we reset rest of prefs here? ResetAutoOpen(); } void DownloadPrefs::ResetAutoOpen() { auto_open_.clear(); SaveAutoOpenState(); } void DownloadPrefs::SaveAutoOpenState() { std::string extensions; for (AutoOpenSet::iterator it = auto_open_.begin(); it != auto_open_.end(); ++it) { #if defined(OS_POSIX) std::string this_extension = *it; #elif defined(OS_WIN) // TODO(phajdan.jr): Why we're using Sys conversion here, but not in ctor? std::string this_extension = base::SysWideToUTF8(*it); #endif extensions += this_extension + ":"; } if (!extensions.empty()) extensions.erase(extensions.size() - 1); prefs_->SetString(prefs::kDownloadExtensionsToOpen, extensions); } bool DownloadPrefs::AutoOpenCompareFunctor::operator()( const FilePath::StringType& a, const FilePath::StringType& b) const { return FilePath::CompareLessIgnoreCase(a, b); }