123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285 |
- // Copyright (c) 2014 GitHub, Inc. All rights reserved.
- // Use of this source code is governed by the MIT license that can be
- // found in the LICENSE file.
- #include "atom/browser/web_dialog_helper.h"
- #include <string>
- #include <utility>
- #include <vector>
- #include "atom/browser/atom_browser_context.h"
- #include "atom/browser/native_window.h"
- #include "atom/browser/ui/file_dialog.h"
- #include "atom/common/native_mate_converters/callback.h"
- #include "base/bind.h"
- #include "base/files/file_enumerator.h"
- #include "base/files/file_path.h"
- #include "base/strings/utf_string_conversions.h"
- #include "chrome/common/pref_names.h"
- #include "components/prefs/pref_service.h"
- #include "content/public/browser/file_select_listener.h"
- #include "content/public/browser/render_frame_host.h"
- #include "content/public/browser/render_process_host.h"
- #include "content/public/browser/render_view_host.h"
- #include "content/public/browser/web_contents.h"
- #include "content/public/browser/web_contents_observer.h"
- #include "native_mate/dictionary.h"
- #include "net/base/mime_util.h"
- #include "ui/shell_dialogs/selected_file_info.h"
- using blink::mojom::FileChooserFileInfo;
- using blink::mojom::FileChooserFileInfoPtr;
- using blink::mojom::FileChooserParams;
- namespace {
- class FileSelectHelper : public base::RefCounted<FileSelectHelper>,
- public content::WebContentsObserver {
- public:
- FileSelectHelper(content::RenderFrameHost* render_frame_host,
- std::unique_ptr<content::FileSelectListener> listener,
- blink::mojom::FileChooserParams::Mode mode)
- : render_frame_host_(render_frame_host),
- listener_(std::move(listener)),
- mode_(mode) {
- auto* web_contents =
- content::WebContents::FromRenderFrameHost(render_frame_host);
- content::WebContentsObserver::Observe(web_contents);
- }
- void ShowOpenDialog(const file_dialog::DialogSettings& settings) {
- v8::Isolate* isolate = v8::Isolate::GetCurrent();
- atom::util::Promise promise(isolate);
- auto callback = base::Bind(&FileSelectHelper::OnOpenDialogDone, this);
- ignore_result(promise.Then(callback));
- file_dialog::ShowOpenDialog(settings, std::move(promise));
- }
- void ShowSaveDialog(const file_dialog::DialogSettings& settings) {
- v8::Isolate* isolate = v8::Isolate::GetCurrent();
- v8::Local<v8::Context> context = isolate->GetCurrentContext();
- atom::util::Promise promise(isolate);
- v8::Local<v8::Promise> handle = promise.GetHandle();
- file_dialog::ShowSaveDialog(settings, std::move(promise));
- ignore_result(handle->Then(
- context,
- v8::Local<v8::Function>::Cast(mate::ConvertToV8(
- isolate, base::Bind(&FileSelectHelper::OnSaveDialogDone, this)))));
- }
- private:
- friend class base::RefCounted<FileSelectHelper>;
- ~FileSelectHelper() override {}
- void OnOpenDialogDone(mate::Dictionary result) {
- std::vector<FileChooserFileInfoPtr> file_info;
- bool canceled = true;
- result.Get("canceled", &canceled);
- if (!canceled) {
- std::vector<base::FilePath> paths;
- if (result.Get("filePaths", &paths)) {
- for (auto& path : paths) {
- file_info.push_back(FileChooserFileInfo::NewNativeFile(
- blink::mojom::NativeFileInfo::New(
- path, path.BaseName().AsUTF16Unsafe())));
- }
- if (render_frame_host_ && !paths.empty()) {
- auto* browser_context = static_cast<atom::AtomBrowserContext*>(
- render_frame_host_->GetProcess()->GetBrowserContext());
- browser_context->prefs()->SetFilePath(prefs::kSelectFileLastDirectory,
- paths[0].DirName());
- }
- }
- }
- OnFilesSelected(std::move(file_info));
- }
- void OnSaveDialogDone(mate::Dictionary result) {
- std::vector<FileChooserFileInfoPtr> file_info;
- bool canceled = true;
- result.Get("canceled", &canceled);
- if (!canceled) {
- base::FilePath path;
- if (result.Get("filePath", &path)) {
- file_info.push_back(FileChooserFileInfo::NewNativeFile(
- blink::mojom::NativeFileInfo::New(
- path, path.BaseName().AsUTF16Unsafe())));
- }
- }
- OnFilesSelected(std::move(file_info));
- }
- void OnFilesSelected(std::vector<FileChooserFileInfoPtr> file_info) {
- if (listener_) {
- listener_->FileSelected(std::move(file_info), base::FilePath(), mode_);
- listener_.reset();
- }
- render_frame_host_ = nullptr;
- }
- // content::WebContentsObserver:
- void RenderFrameHostChanged(content::RenderFrameHost* old_host,
- content::RenderFrameHost* new_host) override {
- if (old_host == render_frame_host_)
- render_frame_host_ = nullptr;
- }
- // content::WebContentsObserver:
- void RenderFrameDeleted(content::RenderFrameHost* deleted_host) override {
- if (deleted_host == render_frame_host_)
- render_frame_host_ = nullptr;
- }
- // content::WebContentsObserver:
- void WebContentsDestroyed() override { render_frame_host_ = nullptr; }
- content::RenderFrameHost* render_frame_host_;
- std::unique_ptr<content::FileSelectListener> listener_;
- blink::mojom::FileChooserParams::Mode mode_;
- };
- file_dialog::Filters GetFileTypesFromAcceptType(
- const std::vector<base::string16>& accept_types) {
- file_dialog::Filters filters;
- if (accept_types.empty())
- return filters;
- std::vector<base::FilePath::StringType> extensions;
- int valid_type_count = 0;
- std::string description;
- for (const auto& accept_type : accept_types) {
- std::string ascii_type = base::UTF16ToASCII(accept_type);
- auto old_extension_size = extensions.size();
- if (ascii_type[0] == '.') {
- // If the type starts with a period it is assumed to be a file extension,
- // like `.txt`, // so we just have to add it to the list.
- base::FilePath::StringType extension(ascii_type.begin(),
- ascii_type.end());
- // Skip the first character.
- extensions.push_back(extension.substr(1));
- } else {
- if (ascii_type == "image/*")
- description = "Image Files";
- else if (ascii_type == "audio/*")
- description = "Audio Files";
- else if (ascii_type == "video/*")
- description = "Video Files";
- // For MIME Type, `audio/*, video/*, image/*
- net::GetExtensionsForMimeType(ascii_type, &extensions);
- }
- if (extensions.size() > old_extension_size)
- valid_type_count++;
- }
- // If no valid exntesion is added, return empty filters.
- if (extensions.empty())
- return filters;
- filters.push_back(file_dialog::Filter());
- if (valid_type_count > 1 || (valid_type_count == 1 && description.empty()))
- description = "Custom Files";
- DCHECK(!description.empty());
- filters[0].first = description;
- for (const auto& extension : extensions) {
- #if defined(OS_WIN)
- filters[0].second.push_back(base::UTF16ToASCII(extension));
- #else
- filters[0].second.push_back(extension);
- #endif
- }
- // Allow all files when extension is specified.
- filters.push_back(file_dialog::Filter());
- filters.back().first = "All Files";
- filters.back().second.push_back("*");
- return filters;
- }
- } // namespace
- namespace atom {
- WebDialogHelper::WebDialogHelper(NativeWindow* window, bool offscreen)
- : window_(window), offscreen_(offscreen), weak_factory_(this) {}
- WebDialogHelper::~WebDialogHelper() {}
- void WebDialogHelper::RunFileChooser(
- content::RenderFrameHost* render_frame_host,
- std::unique_ptr<content::FileSelectListener> listener,
- const blink::mojom::FileChooserParams& params) {
- file_dialog::DialogSettings settings;
- settings.force_detached = offscreen_;
- settings.filters = GetFileTypesFromAcceptType(params.accept_types);
- settings.parent_window = window_;
- settings.title = base::UTF16ToUTF8(params.title);
- scoped_refptr<FileSelectHelper> file_select_helper(new FileSelectHelper(
- render_frame_host, std::move(listener), params.mode));
- if (params.mode == FileChooserParams::Mode::kSave) {
- settings.default_path = params.default_file_name;
- file_select_helper->ShowSaveDialog(settings);
- } else {
- int flags = file_dialog::FILE_DIALOG_CREATE_DIRECTORY;
- switch (params.mode) {
- case FileChooserParams::Mode::kOpenMultiple:
- flags |= file_dialog::FILE_DIALOG_MULTI_SELECTIONS;
- FALLTHROUGH;
- case FileChooserParams::Mode::kOpen:
- flags |= file_dialog::FILE_DIALOG_OPEN_FILE;
- flags |= file_dialog::FILE_DIALOG_TREAT_PACKAGE_APP_AS_DIRECTORY;
- break;
- case FileChooserParams::Mode::kUploadFolder:
- flags |= file_dialog::FILE_DIALOG_OPEN_DIRECTORY;
- break;
- default:
- NOTREACHED();
- }
- auto* browser_context = static_cast<atom::AtomBrowserContext*>(
- render_frame_host->GetProcess()->GetBrowserContext());
- settings.default_path = browser_context->prefs()
- ->GetFilePath(prefs::kSelectFileLastDirectory)
- .Append(params.default_file_name);
- settings.properties = flags;
- file_select_helper->ShowOpenDialog(settings);
- }
- }
- void WebDialogHelper::EnumerateDirectory(
- content::WebContents* web_contents,
- std::unique_ptr<content::FileSelectListener> listener,
- const base::FilePath& dir) {
- int types = base::FileEnumerator::FILES | base::FileEnumerator::DIRECTORIES |
- base::FileEnumerator::INCLUDE_DOT_DOT;
- base::FileEnumerator file_enum(dir, false, types);
- base::FilePath path;
- std::vector<FileChooserFileInfoPtr> file_info;
- while (!(path = file_enum.Next()).empty()) {
- file_info.push_back(FileChooserFileInfo::NewNativeFile(
- blink::mojom::NativeFileInfo::New(path, base::string16())));
- }
- listener->FileSelected(std::move(file_info), dir,
- FileChooserParams::Mode::kUploadFolder);
- }
- } // namespace atom
|