270 lines
8.4 KiB
C++
270 lines
8.4 KiB
C++
// Copyright (c) 2020 Microsoft, Inc. All rights reserved.
|
|
// Use of this source code is governed by the MIT license that can be
|
|
// found in the LICENSE file.
|
|
|
|
#include <memory>
|
|
|
|
#include <string>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#include "shell/browser/file_select_helper.h"
|
|
|
|
#include "base/bind.h"
|
|
#include "base/files/file_enumerator.h"
|
|
#include "base/files/file_util.h"
|
|
#include "base/strings/utf_string_conversions.h"
|
|
#include "chrome/common/pref_names.h"
|
|
#include "components/prefs/pref_service.h"
|
|
#include "shell/browser/electron_browser_context.h"
|
|
#include "shell/browser/javascript_environment.h"
|
|
#include "shell/browser/ui/file_dialog.h"
|
|
#include "shell/common/gin_converters/callback_converter.h"
|
|
#include "shell/common/gin_converters/file_path_converter.h"
|
|
|
|
using blink::mojom::FileChooserFileInfo;
|
|
using blink::mojom::FileChooserFileInfoPtr;
|
|
using blink::mojom::FileChooserParams;
|
|
using blink::mojom::NativeFileInfo;
|
|
|
|
namespace {
|
|
void DeleteFiles(std::vector<base::FilePath> paths) {
|
|
for (auto& file_path : paths)
|
|
base::DeleteFile(file_path);
|
|
}
|
|
} // namespace
|
|
|
|
FileSelectHelper::FileSelectHelper(
|
|
content::RenderFrameHost* render_frame_host,
|
|
scoped_refptr<content::FileSelectListener> listener,
|
|
FileChooserParams::Mode mode)
|
|
: render_frame_host_(render_frame_host),
|
|
listener_(std::move(listener)),
|
|
mode_(mode) {
|
|
DCHECK(render_frame_host_);
|
|
DCHECK(listener_);
|
|
|
|
web_contents_ = content::WebContents::FromRenderFrameHost(render_frame_host);
|
|
DCHECK(web_contents_);
|
|
|
|
content::WebContentsObserver::Observe(web_contents_);
|
|
observer_.Add(render_frame_host_->GetRenderViewHost()->GetWidget());
|
|
}
|
|
|
|
FileSelectHelper::~FileSelectHelper() = default;
|
|
|
|
void FileSelectHelper::ShowOpenDialog(
|
|
const file_dialog::DialogSettings& settings) {
|
|
v8::Isolate* isolate = electron::JavascriptEnvironment::GetIsolate();
|
|
v8::HandleScope scope(isolate);
|
|
gin_helper::Promise<gin_helper::Dictionary> promise(isolate);
|
|
|
|
auto callback = base::BindOnce(&FileSelectHelper::OnOpenDialogDone,
|
|
weak_ptr_factory_.GetWeakPtr());
|
|
ignore_result(promise.Then(std::move(callback)));
|
|
|
|
file_dialog::ShowOpenDialog(settings, std::move(promise));
|
|
}
|
|
|
|
void FileSelectHelper::ShowSaveDialog(
|
|
const file_dialog::DialogSettings& settings) {
|
|
v8::Isolate* isolate = electron::JavascriptEnvironment::GetIsolate();
|
|
v8::HandleScope scope(isolate);
|
|
gin_helper::Promise<gin_helper::Dictionary> promise(isolate);
|
|
|
|
auto callback = base::BindOnce(&FileSelectHelper::OnSaveDialogDone,
|
|
weak_ptr_factory_.GetWeakPtr());
|
|
ignore_result(promise.Then(std::move(callback)));
|
|
|
|
file_dialog::ShowSaveDialog(settings, std::move(promise));
|
|
}
|
|
|
|
// net::DirectoryLister::DirectoryListerDelegate
|
|
void FileSelectHelper::OnListFile(
|
|
const net::DirectoryLister::DirectoryListerData& data) {
|
|
if (!render_frame_host_ || !web_contents_) {
|
|
// If the frame or webcontents was destroyed under us. We
|
|
// must notify |listener_| and release our reference to
|
|
// ourself. RunFileChooserEnd() performs this.
|
|
RunFileChooserEnd();
|
|
return;
|
|
}
|
|
// We don't want to return directory paths, only file paths
|
|
if (data.info.IsDirectory())
|
|
return;
|
|
|
|
lister_paths_.push_back(data.path);
|
|
}
|
|
|
|
void FileSelectHelper::RunFileChooserEnd() {
|
|
// If there are temporary files, then this instance needs to stick around
|
|
// until web_contents_ is destroyed, so that this instance can delete the
|
|
// temporary files.
|
|
if (!temporary_files_.empty())
|
|
return;
|
|
|
|
if (listener_)
|
|
listener_->FileSelectionCanceled();
|
|
|
|
render_frame_host_ = nullptr;
|
|
web_contents_ = nullptr;
|
|
|
|
delete this;
|
|
}
|
|
|
|
// net::DirectoryLister::DirectoryListerDelegate
|
|
void FileSelectHelper::OnListDone(int error) {
|
|
if (!render_frame_host_ || !web_contents_) {
|
|
// If the frame or webcontents was destroyed under us. We
|
|
// must notify |listener_| and release our reference to
|
|
// ourself. RunFileChooserEnd() performs this.
|
|
RunFileChooserEnd();
|
|
return;
|
|
}
|
|
|
|
std::vector<FileChooserFileInfoPtr> file_info;
|
|
for (const auto& path : lister_paths_)
|
|
file_info.push_back(FileChooserFileInfo::NewNativeFile(
|
|
NativeFileInfo::New(path, std::u16string())));
|
|
|
|
OnFilesSelected(std::move(file_info), lister_base_dir_);
|
|
}
|
|
|
|
void FileSelectHelper::DeleteTemporaryFiles() {
|
|
base::ThreadPool::PostTask(
|
|
FROM_HERE,
|
|
{base::MayBlock(), base::TaskPriority::BEST_EFFORT,
|
|
base::TaskShutdownBehavior::BLOCK_SHUTDOWN},
|
|
base::BindOnce(&DeleteFiles, std::move(temporary_files_)));
|
|
}
|
|
|
|
void FileSelectHelper::EnumerateDirectory() {
|
|
// Ensure that this fn is only called once
|
|
DCHECK(!lister_);
|
|
DCHECK(!lister_base_dir_.empty());
|
|
DCHECK(lister_paths_.empty());
|
|
|
|
lister_ = std::make_unique<net::DirectoryLister>(
|
|
lister_base_dir_, net::DirectoryLister::NO_SORT_RECURSIVE, this);
|
|
lister_->Start();
|
|
}
|
|
|
|
void FileSelectHelper::OnOpenDialogDone(gin_helper::Dictionary result) {
|
|
bool canceled = true;
|
|
result.Get("canceled", &canceled);
|
|
|
|
if (!render_frame_host_ || canceled) {
|
|
RunFileChooserEnd();
|
|
} else {
|
|
std::vector<base::FilePath> paths;
|
|
if (result.Get("filePaths", &paths)) {
|
|
std::vector<ui::SelectedFileInfo> files =
|
|
ui::FilePathListToSelectedFileInfoList(paths);
|
|
// If we are uploading a folder we need to enumerate its contents
|
|
if (mode_ == FileChooserParams::Mode::kUploadFolder &&
|
|
paths.size() >= 1) {
|
|
lister_base_dir_ = paths[0];
|
|
EnumerateDirectory();
|
|
} else {
|
|
#if defined(OS_MAC)
|
|
base::ThreadPool::PostTask(
|
|
FROM_HERE,
|
|
{base::MayBlock(),
|
|
base::TaskShutdownBehavior::CONTINUE_ON_SHUTDOWN},
|
|
base::BindOnce(&FileSelectHelper::ProcessSelectedFilesMac,
|
|
base::Unretained(this), files));
|
|
#else
|
|
ConvertToFileChooserFileInfoList(files);
|
|
#endif
|
|
}
|
|
|
|
if (render_frame_host_ && !paths.empty()) {
|
|
auto* browser_context = static_cast<electron::ElectronBrowserContext*>(
|
|
render_frame_host_->GetProcess()->GetBrowserContext());
|
|
browser_context->prefs()->SetFilePath(prefs::kSelectFileLastDirectory,
|
|
paths[0].DirName());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void FileSelectHelper::ConvertToFileChooserFileInfoList(
|
|
const std::vector<ui::SelectedFileInfo>& files) {
|
|
std::vector<FileChooserFileInfoPtr> file_info;
|
|
|
|
for (const auto& file : files) {
|
|
file_info.push_back(FileChooserFileInfo::NewNativeFile(NativeFileInfo::New(
|
|
file.local_path, base::FilePath(file.display_name).AsUTF16Unsafe())));
|
|
}
|
|
|
|
OnFilesSelected(std::move(file_info), lister_base_dir_);
|
|
}
|
|
|
|
void FileSelectHelper::OnSaveDialogDone(gin_helper::Dictionary result) {
|
|
std::vector<FileChooserFileInfoPtr> file_info;
|
|
bool canceled = true;
|
|
result.Get("canceled", &canceled);
|
|
|
|
if (!render_frame_host_ || canceled) {
|
|
RunFileChooserEnd();
|
|
} else {
|
|
base::FilePath path;
|
|
if (result.Get("filePath", &path)) {
|
|
file_info.push_back(FileChooserFileInfo::NewNativeFile(
|
|
NativeFileInfo::New(path, path.BaseName().AsUTF16Unsafe())));
|
|
}
|
|
// We should only call this if we have not cancelled the dialog.
|
|
OnFilesSelected(std::move(file_info), base::FilePath());
|
|
}
|
|
}
|
|
|
|
void FileSelectHelper::OnFilesSelected(
|
|
std::vector<FileChooserFileInfoPtr> file_info,
|
|
base::FilePath base_dir) {
|
|
if (listener_) {
|
|
listener_->FileSelected(std::move(file_info), base_dir, mode_);
|
|
listener_.reset();
|
|
}
|
|
|
|
render_frame_host_ = nullptr;
|
|
|
|
delete this;
|
|
}
|
|
|
|
void FileSelectHelper::RenderWidgetHostDestroyed(
|
|
content::RenderWidgetHost* widget_host) {
|
|
render_frame_host_ = nullptr;
|
|
observer_.Remove(widget_host);
|
|
}
|
|
|
|
// content::WebContentsObserver:
|
|
void FileSelectHelper::RenderFrameHostChanged(
|
|
content::RenderFrameHost* old_host,
|
|
content::RenderFrameHost* new_host) {
|
|
if (!render_frame_host_)
|
|
return;
|
|
// The |old_host| and its children are now pending deletion. Do not give
|
|
// them file access past this point.
|
|
if (render_frame_host_ == old_host ||
|
|
render_frame_host_->IsDescendantOf(old_host)) {
|
|
render_frame_host_ = nullptr;
|
|
}
|
|
}
|
|
|
|
// content::WebContentsObserver:
|
|
void FileSelectHelper::RenderFrameDeleted(
|
|
content::RenderFrameHost* deleted_host) {
|
|
if (deleted_host == render_frame_host_)
|
|
render_frame_host_ = nullptr;
|
|
}
|
|
|
|
// content::WebContentsObserver:
|
|
void FileSelectHelper::WebContentsDestroyed() {
|
|
render_frame_host_ = nullptr;
|
|
web_contents_ = nullptr;
|
|
|
|
DeleteTemporaryFiles();
|
|
|
|
if (!lister_)
|
|
delete this;
|
|
}
|