// 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. #ifndef CHROME_BROWSER_FILE_SELECT_HELPER_H_ #define CHROME_BROWSER_FILE_SELECT_HELPER_H_ #include #include #include "base/compiler_specific.h" #include "base/gtest_prod_util.h" #include "content/public/browser/notification_observer.h" #include "content/public/browser/notification_registrar.h" #include "content/public/browser/web_contents_observer.h" #include "content/public/common/file_chooser_params.h" #include "net/base/directory_lister.h" #include "ui/shell_dialogs/select_file_dialog.h" class Profile; namespace content { struct FileChooserFileInfo; class RenderViewHost; class WebContents; } namespace ui { struct SelectedFileInfo; } // This class handles file-selection requests coming from WebUI elements // (via the extensions::ExtensionHost class). It implements both the // initialisation and listener functions for file-selection dialogs. class FileSelectHelper : public base::RefCountedThreadSafe, public ui::SelectFileDialog::Listener, public content::WebContentsObserver, public content::NotificationObserver { public: // Show the file chooser dialog. static void RunFileChooser(content::WebContents* tab, const content::FileChooserParams& params); // Enumerates all the files in directory. static void EnumerateDirectory(content::WebContents* tab, int request_id, const base::FilePath& path); private: friend class base::RefCountedThreadSafe; FRIEND_TEST_ALL_PREFIXES(FileSelectHelperTest, IsAcceptTypeValid); FRIEND_TEST_ALL_PREFIXES(FileSelectHelperTest, ZipPackage); explicit FileSelectHelper(Profile* profile); ~FileSelectHelper() override; // Utility class which can listen for directory lister events and relay // them to the main object with the correct tracking id. class DirectoryListerDispatchDelegate : public net::DirectoryLister::DirectoryListerDelegate { public: DirectoryListerDispatchDelegate(FileSelectHelper* parent, int id) : parent_(parent), id_(id) {} ~DirectoryListerDispatchDelegate() override {} void OnListFile( const net::DirectoryLister::DirectoryListerData& data) override; void OnListDone(int error) override; private: // This FileSelectHelper owns this object. FileSelectHelper* parent_; int id_; DISALLOW_COPY_AND_ASSIGN(DirectoryListerDispatchDelegate); }; void RunFileChooser(content::RenderViewHost* render_view_host, content::WebContents* web_contents, const content::FileChooserParams& params); void RunFileChooserOnFileThread( const content::FileChooserParams& params); void RunFileChooserOnUIThread( const content::FileChooserParams& params); // Cleans up and releases this instance. This must be called after the last // callback is received from the file chooser dialog. void RunFileChooserEnd(); // SelectFileDialog::Listener overrides. void FileSelected(const base::FilePath& path, int index, void* params) override; void FileSelectedWithExtraInfo(const ui::SelectedFileInfo& file, int index, void* params) override; void MultiFilesSelected(const std::vector& files, void* params) override; void MultiFilesSelectedWithExtraInfo( const std::vector& files, void* params) override; void FileSelectionCanceled(void* params) override; // content::NotificationObserver overrides. void Observe(int type, const content::NotificationSource& source, const content::NotificationDetails& details) override; // content::WebContentsObserver overrides. void RenderViewHostChanged(content::RenderViewHost* old_host, content::RenderViewHost* new_host) override; void WebContentsDestroyed() override; void EnumerateDirectory(int request_id, content::RenderViewHost* render_view_host, const base::FilePath& path); // Kicks off a new directory enumeration. void StartNewEnumeration(const base::FilePath& path, int request_id, content::RenderViewHost* render_view_host); // Callbacks from directory enumeration. virtual void OnListFile( int id, const net::DirectoryLister::DirectoryListerData& data); virtual void OnListDone(int id, int error); // Cleans up and releases this instance. This must be called after the last // callback is received from the enumeration code. void EnumerateDirectoryEnd(); #if defined(OS_MACOSX) && !defined(OS_IOS) // Must be called on the FILE_USER_BLOCKING thread. Each selected file that is // a package will be zipped, and the zip will be passed to the render view // host in place of the package. void ProcessSelectedFilesMac(const std::vector& files); // Saves the paths of |zipped_files| for later deletion. Passes |files| to the // render view host. void ProcessSelectedFilesMacOnUIThread( const std::vector& files, const std::vector& zipped_files); // Zips the package at |path| into a temporary destination. Returns the // temporary destination, if the zip was successful. Otherwise returns an // empty path. static base::FilePath ZipPackage(const base::FilePath& path); #endif // defined(OS_MACOSX) && !defined(OS_IOS) // Utility method that passes |files| to the render view host, and ends the // file chooser. void NotifyRenderViewHostAndEnd( const std::vector& files); // Sends the result to the render process, and call |RunFileChooserEnd|. void NotifyRenderViewHostAndEndAfterConversion( const std::vector& list); // Schedules the deletion of the files in |temporary_files_| and clears the // vector. void DeleteTemporaryFiles(); // Cleans up when the RenderViewHost of our WebContents changes. void CleanUpOnRenderViewHostChange(); // Helper method to get allowed extensions for select file dialog from // the specified accept types as defined in the spec: // http://whatwg.org/html/number-state.html#attr-input-accept // |accept_types| contains only valid lowercased MIME types or file extensions // beginning with a period (.). static scoped_ptr GetFileTypesFromAcceptType( const std::vector& accept_types); // Check the accept type is valid. It is expected to be all lower case with // no whitespace. static bool IsAcceptTypeValid(const std::string& accept_type); // Profile used to set/retrieve the last used directory. Profile* profile_; // The RenderViewHost and WebContents for the page showing a file dialog // (may only be one such dialog). content::RenderViewHost* render_view_host_; content::WebContents* web_contents_; // Dialog box used for choosing files to upload from file form fields. scoped_refptr select_file_dialog_; scoped_ptr select_file_types_; // The type of file dialog last shown. ui::SelectFileDialog::Type dialog_type_; // The mode of file dialog last shown. content::FileChooserParams::Mode dialog_mode_; // Maintain a list of active directory enumerations. These could come from // the file select dialog or from drag-and-drop of directories, so there could // be more than one going on at a time. struct ActiveDirectoryEnumeration; std::map directory_enumerations_; // Registrar for notifications regarding our RenderViewHost. content::NotificationRegistrar notification_registrar_; // Temporary files only used on OSX. This class is responsible for deleting // these files when they are no longer needed. std::vector temporary_files_; DISALLOW_COPY_AND_ASSIGN(FileSelectHelper); }; #endif // CHROME_BROWSER_FILE_SELECT_HELPER_H_