// 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/file_select_helper.h" #include #include "base/bind.h" #include "base/file_util.h" #include "base/files/file_enumerator.h" #include "base/platform_file.h" #include "base/strings/string_split.h" #include "base/strings/string_util.h" #include "base/strings/utf_string_conversions.h" #include "chrome/browser/platform_util.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/ui/browser.h" #include "chrome/browser/ui/browser_list.h" #include "chrome/browser/ui/chrome_select_file_policy.h" #include "content/public/browser/browser_thread.h" #include "content/public/browser/notification_details.h" #include "content/public/browser/notification_source.h" #include "content/public/browser/notification_types.h" #include "content/public/browser/render_view_host.h" #include "content/public/browser/render_widget_host_view.h" #include "content/public/browser/web_contents.h" #include "content/public/common/file_chooser_params.h" #include "grit/generated_resources.h" #include "net/base/mime_util.h" #include "ui/base/l10n/l10n_util.h" #include "ui/shell_dialogs/selected_file_info.h" using content::BrowserThread; using content::FileChooserParams; using content::RenderViewHost; using content::RenderWidgetHost; using content::WebContents; namespace { // There is only one file-selection happening at any given time, // so we allocate an enumeration ID for that purpose. All IDs from // the renderer must start at 0 and increase. const int kFileSelectEnumerationId = -1; void NotifyRenderViewHost(RenderViewHost* render_view_host, const std::vector& files, ui::SelectFileDialog::Type dialog_type) { const int kReadFilePermissions = base::PLATFORM_FILE_OPEN | base::PLATFORM_FILE_READ | base::PLATFORM_FILE_EXCLUSIVE_READ | base::PLATFORM_FILE_ASYNC; const int kWriteFilePermissions = base::PLATFORM_FILE_CREATE | base::PLATFORM_FILE_CREATE_ALWAYS | base::PLATFORM_FILE_OPEN | base::PLATFORM_FILE_OPEN_ALWAYS | base::PLATFORM_FILE_OPEN_TRUNCATED | base::PLATFORM_FILE_WRITE | base::PLATFORM_FILE_WRITE_ATTRIBUTES | base::PLATFORM_FILE_ASYNC; int permissions = kReadFilePermissions; if (dialog_type == ui::SelectFileDialog::SELECT_SAVEAS_FILE) permissions = kWriteFilePermissions; render_view_host->FilesSelectedInChooser(files, permissions); } // Converts a list of FilePaths to a list of ui::SelectedFileInfo. std::vector FilePathListToSelectedFileInfoList( const std::vector& paths) { std::vector selected_files; for (size_t i = 0; i < paths.size(); ++i) { selected_files.push_back( ui::SelectedFileInfo(paths[i], paths[i])); } return selected_files; } } // namespace struct FileSelectHelper::ActiveDirectoryEnumeration { ActiveDirectoryEnumeration() : rvh_(NULL) {} scoped_ptr delegate_; scoped_ptr lister_; RenderViewHost* rvh_; std::vector results_; }; FileSelectHelper::FileSelectHelper(Profile* profile) : profile_(profile), render_view_host_(NULL), web_contents_(NULL), select_file_dialog_(), select_file_types_(), dialog_type_(ui::SelectFileDialog::SELECT_OPEN_FILE) { } FileSelectHelper::~FileSelectHelper() { // There may be pending file dialogs, we need to tell them that we've gone // away so they don't try and call back to us. if (select_file_dialog_.get()) select_file_dialog_->ListenerDestroyed(); // Stop any pending directory enumeration, prevent a callback, and free // allocated memory. std::map::iterator iter; for (iter = directory_enumerations_.begin(); iter != directory_enumerations_.end(); ++iter) { iter->second->lister_.reset(); delete iter->second; } } void FileSelectHelper::DirectoryListerDispatchDelegate::OnListFile( const net::DirectoryLister::DirectoryListerData& data) { parent_->OnListFile(id_, data); } void FileSelectHelper::DirectoryListerDispatchDelegate::OnListDone(int error) { parent_->OnListDone(id_, error); } void FileSelectHelper::FileSelected(const base::FilePath& path, int index, void* params) { FileSelectedWithExtraInfo(ui::SelectedFileInfo(path, path), index, params); } void FileSelectHelper::FileSelectedWithExtraInfo( const ui::SelectedFileInfo& file, int index, void* params) { if (!render_view_host_) return; profile_->set_last_selected_directory(file.file_path.DirName()); const base::FilePath& path = file.local_path; if (dialog_type_ == ui::SelectFileDialog::SELECT_FOLDER) { StartNewEnumeration(path, kFileSelectEnumerationId, render_view_host_); return; } std::vector files; files.push_back(file); NotifyRenderViewHost(render_view_host_, files, dialog_type_); // No members should be accessed from here on. RunFileChooserEnd(); } void FileSelectHelper::MultiFilesSelected( const std::vector& files, void* params) { std::vector selected_files = FilePathListToSelectedFileInfoList(files); MultiFilesSelectedWithExtraInfo(selected_files, params); } void FileSelectHelper::MultiFilesSelectedWithExtraInfo( const std::vector& files, void* params) { if (!files.empty()) profile_->set_last_selected_directory(files[0].file_path.DirName()); if (!render_view_host_) return; NotifyRenderViewHost(render_view_host_, files, dialog_type_); // No members should be accessed from here on. RunFileChooserEnd(); } void FileSelectHelper::FileSelectionCanceled(void* params) { if (!render_view_host_) return; // If the user cancels choosing a file to upload we pass back an // empty vector. NotifyRenderViewHost( render_view_host_, std::vector(), dialog_type_); // No members should be accessed from here on. RunFileChooserEnd(); } void FileSelectHelper::StartNewEnumeration(const base::FilePath& path, int request_id, RenderViewHost* render_view_host) { scoped_ptr entry(new ActiveDirectoryEnumeration); entry->rvh_ = render_view_host; entry->delegate_.reset(new DirectoryListerDispatchDelegate(this, request_id)); entry->lister_.reset(new net::DirectoryLister(path, true, net::DirectoryLister::NO_SORT, entry->delegate_.get())); if (!entry->lister_->Start()) { if (request_id == kFileSelectEnumerationId) FileSelectionCanceled(NULL); else render_view_host->DirectoryEnumerationFinished(request_id, entry->results_); } else { directory_enumerations_[request_id] = entry.release(); } } void FileSelectHelper::OnListFile( int id, const net::DirectoryLister::DirectoryListerData& data) { ActiveDirectoryEnumeration* entry = directory_enumerations_[id]; // Directory upload returns directories via a "." file, so that // empty directories are included. This util call just checks // the flags in the structure; there's no file I/O going on. if (data.info.IsDirectory()) entry->results_.push_back(data.path.Append(FILE_PATH_LITERAL("."))); else entry->results_.push_back(data.path); } void FileSelectHelper::OnListDone(int id, int error) { // This entry needs to be cleaned up when this function is done. scoped_ptr entry(directory_enumerations_[id]); directory_enumerations_.erase(id); if (!entry->rvh_) return; if (error) { FileSelectionCanceled(NULL); return; } std::vector selected_files = FilePathListToSelectedFileInfoList(entry->results_); if (id == kFileSelectEnumerationId) NotifyRenderViewHost(entry->rvh_, selected_files, dialog_type_); else entry->rvh_->DirectoryEnumerationFinished(id, entry->results_); EnumerateDirectoryEnd(); } scoped_ptr FileSelectHelper::GetFileTypesFromAcceptType( const std::vector& accept_types) { scoped_ptr base_file_type( new ui::SelectFileDialog::FileTypeInfo()); base_file_type->support_drive = true; if (accept_types.empty()) return base_file_type.Pass(); // Create FileTypeInfo and pre-allocate for the first extension list. scoped_ptr file_type( new ui::SelectFileDialog::FileTypeInfo(*base_file_type)); file_type->include_all_files = true; file_type->extensions.resize(1); std::vector* extensions = &file_type->extensions.back(); // Find the corresponding extensions. int valid_type_count = 0; int description_id = 0; for (size_t i = 0; i < accept_types.size(); ++i) { std::string ascii_type = UTF16ToASCII(accept_types[i]); if (!IsAcceptTypeValid(ascii_type)) continue; size_t old_extension_size = extensions->size(); if (ascii_type[0] == '.') { // If the type starts with a period it is assumed to be a file extension // so we just have to add it to the list. base::FilePath::StringType ext(ascii_type.begin(), ascii_type.end()); extensions->push_back(ext.substr(1)); } else { if (ascii_type == "image/*") description_id = IDS_IMAGE_FILES; else if (ascii_type == "audio/*") description_id = IDS_AUDIO_FILES; else if (ascii_type == "video/*") description_id = IDS_VIDEO_FILES; net::GetExtensionsForMimeType(ascii_type, extensions); } if (extensions->size() > old_extension_size) valid_type_count++; } // If no valid extension is added, bail out. if (valid_type_count == 0) return base_file_type.Pass(); // Use a generic description "Custom Files" if either of the following is // true: // 1) There're multiple types specified, like "audio/*,video/*" // 2) There're multiple extensions for a MIME type without parameter, like // "ehtml,shtml,htm,html" for "text/html". On Windows, the select file // dialog uses the first extension in the list to form the description, // like "EHTML Files". This is not what we want. if (valid_type_count > 1 || (valid_type_count == 1 && description_id == 0 && extensions->size() > 1)) description_id = IDS_CUSTOM_FILES; if (description_id) { file_type->extension_description_overrides.push_back( l10n_util::GetStringUTF16(description_id)); } return file_type.Pass(); } // static void FileSelectHelper::RunFileChooser(content::WebContents* tab, const FileChooserParams& params) { Profile* profile = Profile::FromBrowserContext(tab->GetBrowserContext()); // FileSelectHelper will keep itself alive until it sends the result message. scoped_refptr file_select_helper( new FileSelectHelper(profile)); file_select_helper->RunFileChooser(tab->GetRenderViewHost(), tab, params); } // static void FileSelectHelper::EnumerateDirectory(content::WebContents* tab, int request_id, const base::FilePath& path) { Profile* profile = Profile::FromBrowserContext(tab->GetBrowserContext()); // FileSelectHelper will keep itself alive until it sends the result message. scoped_refptr file_select_helper( new FileSelectHelper(profile)); file_select_helper->EnumerateDirectory( request_id, tab->GetRenderViewHost(), path); } void FileSelectHelper::RunFileChooser(RenderViewHost* render_view_host, content::WebContents* web_contents, const FileChooserParams& params) { DCHECK(!render_view_host_); DCHECK(!web_contents_); render_view_host_ = render_view_host; web_contents_ = web_contents; notification_registrar_.RemoveAll(); notification_registrar_.Add( this, content::NOTIFICATION_RENDER_WIDGET_HOST_DESTROYED, content::Source(render_view_host_)); notification_registrar_.Add( this, content::NOTIFICATION_WEB_CONTENTS_DESTROYED, content::Source(web_contents_)); BrowserThread::PostTask( BrowserThread::FILE, FROM_HERE, base::Bind(&FileSelectHelper::RunFileChooserOnFileThread, this, params)); // Because this class returns notifications to the RenderViewHost, it is // difficult for callers to know how long to keep a reference to this // instance. We AddRef() here to keep the instance alive after we return // to the caller, until the last callback is received from the file dialog. // At that point, we must call RunFileChooserEnd(). AddRef(); } void FileSelectHelper::RunFileChooserOnFileThread( const FileChooserParams& params) { select_file_types_ = GetFileTypesFromAcceptType(params.accept_types); BrowserThread::PostTask( BrowserThread::UI, FROM_HERE, base::Bind(&FileSelectHelper::RunFileChooserOnUIThread, this, params)); } void FileSelectHelper::RunFileChooserOnUIThread( const FileChooserParams& params) { if (!render_view_host_ || !web_contents_) { // If the renderer was destroyed before we started, just cancel the // operation. RunFileChooserEnd(); return; } select_file_dialog_ = ui::SelectFileDialog::Create( this, new ChromeSelectFilePolicy(web_contents_)); switch (params.mode) { case FileChooserParams::Open: dialog_type_ = ui::SelectFileDialog::SELECT_OPEN_FILE; break; case FileChooserParams::OpenMultiple: dialog_type_ = ui::SelectFileDialog::SELECT_OPEN_MULTI_FILE; break; case FileChooserParams::OpenFolder: dialog_type_ = ui::SelectFileDialog::SELECT_FOLDER; break; case FileChooserParams::Save: dialog_type_ = ui::SelectFileDialog::SELECT_SAVEAS_FILE; break; default: // Prevent warning. dialog_type_ = ui::SelectFileDialog::SELECT_OPEN_FILE; NOTREACHED(); } base::FilePath default_file_name = params.default_file_name.IsAbsolute() ? params.default_file_name : profile_->last_selected_directory().Append(params.default_file_name); gfx::NativeWindow owning_window = platform_util::GetTopLevel(render_view_host_->GetView()->GetNativeView()); #if defined(OS_ANDROID) // Android needs the original MIME types and an additional capture value. std::vector accept_types(params.accept_types); accept_types.push_back(params.capture); #endif select_file_dialog_->SelectFile( dialog_type_, params.title, default_file_name, select_file_types_.get(), select_file_types_.get() && !select_file_types_->extensions.empty() ? 1 : 0, // 1-based index of default extension to show. base::FilePath::StringType(), owning_window, #if defined(OS_ANDROID) &accept_types); #else NULL); #endif select_file_types_.reset(); } // This method is called when we receive the last callback from the file // chooser dialog. Perform any cleanup and release the reference we added // in RunFileChooser(). void FileSelectHelper::RunFileChooserEnd() { render_view_host_ = NULL; web_contents_ = NULL; Release(); } void FileSelectHelper::EnumerateDirectory(int request_id, RenderViewHost* render_view_host, const base::FilePath& path) { // Because this class returns notifications to the RenderViewHost, it is // difficult for callers to know how long to keep a reference to this // instance. We AddRef() here to keep the instance alive after we return // to the caller, until the last callback is received from the enumeration // code. At that point, we must call EnumerateDirectoryEnd(). AddRef(); StartNewEnumeration(path, request_id, render_view_host); } // This method is called when we receive the last callback from the enumeration // code. Perform any cleanup and release the reference we added in // EnumerateDirectory(). void FileSelectHelper::EnumerateDirectoryEnd() { Release(); } void FileSelectHelper::Observe(int type, const content::NotificationSource& source, const content::NotificationDetails& details) { switch (type) { case content::NOTIFICATION_RENDER_WIDGET_HOST_DESTROYED: { DCHECK(content::Source(source).ptr() == render_view_host_); render_view_host_ = NULL; break; } case content::NOTIFICATION_WEB_CONTENTS_DESTROYED: { DCHECK(content::Source(source).ptr() == web_contents_); web_contents_ = NULL; break; } default: NOTREACHED(); } } // static bool FileSelectHelper::IsAcceptTypeValid(const std::string& accept_type) { // TODO(raymes): This only does some basic checks, extend to test more cases. // A 1 character accept type will always be invalid (either a "." in the case // of an extension or a "/" in the case of a MIME type). std::string unused; if (accept_type.length() <= 1 || StringToLowerASCII(accept_type) != accept_type || TrimWhitespaceASCII(accept_type, TRIM_ALL, &unused) != TRIM_NONE) { return false; } return true; }