refactor: ginify DownloadItem (#22924)

This commit is contained in:
Jeremy Apthorp 2020-04-02 17:22:46 -07:00 committed by GitHub
parent 6159066c26
commit 0a78ab4b98
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
9 changed files with 153 additions and 101 deletions

View file

@ -138,7 +138,3 @@ for (const name of events) {
// Deprecate allowRendererProcessReuse but only if they set it to false, no need to log if // Deprecate allowRendererProcessReuse but only if they set it to false, no need to log if
// they are setting it to true // they are setting it to true
deprecate.removeProperty(app, 'allowRendererProcessReuse', [false]); deprecate.removeProperty(app, 'allowRendererProcessReuse', [false]);
// Wrappers for native classes.
const { DownloadItem } = process.electronBinding('download_item');
Object.setPrototypeOf(DownloadItem.prototype, EventEmitter.prototype);

View file

@ -5,6 +5,7 @@
#include "shell/browser/api/electron_api_download_item.h" #include "shell/browser/api/electron_api_download_item.h"
#include <map> #include <map>
#include <memory>
#include "base/strings/utf_string_conversions.h" #include "base/strings/utf_string_conversions.h"
#include "base/threading/thread_task_runner_handle.h" #include "base/threading/thread_task_runner_handle.h"
@ -53,16 +54,41 @@ namespace api {
namespace { namespace {
std::map<uint32_t, v8::Global<v8::Value>> g_download_item_objects; // Ordinarily base::SupportsUserData only supports strong links, where the
// thing to which the user data is attached owns the user data. But we can't
// make the api::DownloadItem owned by the DownloadItem, since it's owned by
// V8. So this makes a weak link. The lifetimes of download::DownloadItem and
// api::DownloadItem are fully independent, and either one may be destroyed
// before the other.
struct UserDataLink : base::SupportsUserData::Data {
explicit UserDataLink(base::WeakPtr<DownloadItem> item)
: download_item(item) {}
base::WeakPtr<DownloadItem> download_item;
};
const void* kElectronApiDownloadItemKey = &kElectronApiDownloadItemKey;
} // namespace } // namespace
gin::WrapperInfo DownloadItem::kWrapperInfo = {gin::kEmbedderNativeGin};
// static
DownloadItem* DownloadItem::FromDownloadItem(
download::DownloadItem* download_item) {
// ^- say that 7 times fast in a row
UserDataLink* data = static_cast<UserDataLink*>(
download_item->GetUserData(kElectronApiDownloadItemKey));
return data ? data->download_item.get() : nullptr;
}
DownloadItem::DownloadItem(v8::Isolate* isolate, DownloadItem::DownloadItem(v8::Isolate* isolate,
download::DownloadItem* download_item) download::DownloadItem* download_item)
: download_item_(download_item) { : download_item_(download_item) {
download_item_->AddObserver(this); download_item_->AddObserver(this);
Init(isolate); download_item_->SetUserData(
AttachAsUserData(download_item); kElectronApiDownloadItemKey,
std::make_unique<UserDataLink>(weak_factory_.GetWeakPtr()));
} }
DownloadItem::~DownloadItem() { DownloadItem::~DownloadItem() {
@ -71,17 +97,23 @@ DownloadItem::~DownloadItem() {
download_item_->RemoveObserver(this); download_item_->RemoveObserver(this);
download_item_->Remove(); download_item_->Remove();
} }
}
// Remove from the global map. bool DownloadItem::CheckAlive() const {
g_download_item_objects.erase(weak_map_id()); if (!download_item_) {
gin_helper::ErrorThrower(v8::Isolate::GetCurrent())
.ThrowError("DownloadItem used after being destroyed");
return false;
}
return true;
} }
void DownloadItem::OnDownloadUpdated(download::DownloadItem* item) { void DownloadItem::OnDownloadUpdated(download::DownloadItem* item) {
if (!CheckAlive())
return;
if (download_item_->IsDone()) { if (download_item_->IsDone()) {
Emit("done", item->GetState()); Emit("done", item->GetState());
// Destroy the item once item is downloaded. Unpin();
base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE,
GetDestroyClosure());
} else { } else {
Emit("updated", item->GetState()); Emit("updated", item->GetState());
} }
@ -89,47 +121,66 @@ void DownloadItem::OnDownloadUpdated(download::DownloadItem* item) {
void DownloadItem::OnDownloadDestroyed(download::DownloadItem* download_item) { void DownloadItem::OnDownloadDestroyed(download::DownloadItem* download_item) {
download_item_ = nullptr; download_item_ = nullptr;
// Destroy the native class immediately when downloadItem is destroyed. Unpin();
delete this;
} }
void DownloadItem::Pause() { void DownloadItem::Pause() {
if (!CheckAlive())
return;
download_item_->Pause(); download_item_->Pause();
} }
bool DownloadItem::IsPaused() const { bool DownloadItem::IsPaused() const {
if (!CheckAlive())
return false;
return download_item_->IsPaused(); return download_item_->IsPaused();
} }
void DownloadItem::Resume() { void DownloadItem::Resume() {
if (!CheckAlive())
return;
download_item_->Resume(true /* user_gesture */); download_item_->Resume(true /* user_gesture */);
} }
bool DownloadItem::CanResume() const { bool DownloadItem::CanResume() const {
if (!CheckAlive())
return false;
return download_item_->CanResume(); return download_item_->CanResume();
} }
void DownloadItem::Cancel() { void DownloadItem::Cancel() {
if (!CheckAlive())
return;
download_item_->Cancel(true); download_item_->Cancel(true);
} }
int64_t DownloadItem::GetReceivedBytes() const { int64_t DownloadItem::GetReceivedBytes() const {
if (!CheckAlive())
return 0;
return download_item_->GetReceivedBytes(); return download_item_->GetReceivedBytes();
} }
int64_t DownloadItem::GetTotalBytes() const { int64_t DownloadItem::GetTotalBytes() const {
if (!CheckAlive())
return 0;
return download_item_->GetTotalBytes(); return download_item_->GetTotalBytes();
} }
std::string DownloadItem::GetMimeType() const { std::string DownloadItem::GetMimeType() const {
if (!CheckAlive())
return "";
return download_item_->GetMimeType(); return download_item_->GetMimeType();
} }
bool DownloadItem::HasUserGesture() const { bool DownloadItem::HasUserGesture() const {
if (!CheckAlive())
return false;
return download_item_->HasUserGesture(); return download_item_->HasUserGesture();
} }
std::string DownloadItem::GetFilename() const { std::string DownloadItem::GetFilename() const {
if (!CheckAlive())
return "";
return base::UTF16ToUTF8( return base::UTF16ToUTF8(
net::GenerateFileName(GetURL(), GetContentDisposition(), std::string(), net::GenerateFileName(GetURL(), GetContentDisposition(), std::string(),
download_item_->GetSuggestedFilename(), download_item_->GetSuggestedFilename(),
@ -138,22 +189,32 @@ std::string DownloadItem::GetFilename() const {
} }
std::string DownloadItem::GetContentDisposition() const { std::string DownloadItem::GetContentDisposition() const {
if (!CheckAlive())
return "";
return download_item_->GetContentDisposition(); return download_item_->GetContentDisposition();
} }
const GURL& DownloadItem::GetURL() const { const GURL& DownloadItem::GetURL() const {
if (!CheckAlive())
return GURL::EmptyGURL();
return download_item_->GetURL(); return download_item_->GetURL();
} }
const std::vector<GURL>& DownloadItem::GetURLChain() const { v8::Local<v8::Value> DownloadItem::GetURLChain(v8::Isolate* isolate) const {
return download_item_->GetUrlChain(); if (!CheckAlive())
return v8::Local<v8::Value>();
return gin::ConvertToV8(isolate, download_item_->GetUrlChain());
} }
download::DownloadItem::DownloadState DownloadItem::GetState() const { download::DownloadItem::DownloadState DownloadItem::GetState() const {
if (!CheckAlive())
return download::DownloadItem::IN_PROGRESS;
return download_item_->GetState(); return download_item_->GetState();
} }
bool DownloadItem::IsDone() const { bool DownloadItem::IsDone() const {
if (!CheckAlive())
return false;
return download_item_->IsDone(); return download_item_->IsDone();
} }
@ -175,23 +236,28 @@ void DownloadItem::SetSaveDialogOptions(
} }
std::string DownloadItem::GetLastModifiedTime() const { std::string DownloadItem::GetLastModifiedTime() const {
if (!CheckAlive())
return "";
return download_item_->GetLastModifiedTime(); return download_item_->GetLastModifiedTime();
} }
std::string DownloadItem::GetETag() const { std::string DownloadItem::GetETag() const {
if (!CheckAlive())
return "";
return download_item_->GetETag(); return download_item_->GetETag();
} }
double DownloadItem::GetStartTime() const { double DownloadItem::GetStartTime() const {
if (!CheckAlive())
return 0;
return download_item_->GetStartTime().ToDoubleT(); return download_item_->GetStartTime().ToDoubleT();
} }
// static // static
void DownloadItem::BuildPrototype(v8::Isolate* isolate, gin::ObjectTemplateBuilder DownloadItem::GetObjectTemplateBuilder(
v8::Local<v8::FunctionTemplate> prototype) { v8::Isolate* isolate) {
prototype->SetClassName(gin::StringToV8(isolate, "DownloadItem")); return gin_helper::EventEmitterMixin<DownloadItem>::GetObjectTemplateBuilder(
gin_helper::Destroyable::MakeDestroyable(isolate, prototype); isolate)
gin_helper::ObjectTemplateBuilder(isolate, prototype->PrototypeTemplate())
.SetMethod("pause", &DownloadItem::Pause) .SetMethod("pause", &DownloadItem::Pause)
.SetMethod("isPaused", &DownloadItem::IsPaused) .SetMethod("isPaused", &DownloadItem::IsPaused)
.SetMethod("resume", &DownloadItem::Resume) .SetMethod("resume", &DownloadItem::Resume)
@ -218,38 +284,25 @@ void DownloadItem::BuildPrototype(v8::Isolate* isolate,
.SetMethod("getStartTime", &DownloadItem::GetStartTime); .SetMethod("getStartTime", &DownloadItem::GetStartTime);
} }
const char* DownloadItem::GetTypeName() {
return "DownloadItem";
}
// static // static
gin::Handle<DownloadItem> DownloadItem::Create(v8::Isolate* isolate, gin::Handle<DownloadItem> DownloadItem::FromOrCreate(
v8::Isolate* isolate,
download::DownloadItem* item) { download::DownloadItem* item) {
auto* existing = TrackableObject::FromWrappedClass(isolate, item); DownloadItem* existing = FromDownloadItem(item);
if (existing) if (existing)
return gin::CreateHandle(isolate, static_cast<DownloadItem*>(existing)); return gin::CreateHandle(isolate, existing);
auto handle = gin::CreateHandle(isolate, new DownloadItem(isolate, item)); auto handle = gin::CreateHandle(isolate, new DownloadItem(isolate, item));
// Reference this object in case it got garbage collected. handle->Pin(isolate);
g_download_item_objects[handle->weak_map_id()] =
v8::Global<v8::Value>(isolate, handle.ToV8());
return handle; return handle;
} }
} // namespace api } // namespace api
} // namespace electron } // namespace electron
namespace {
void Initialize(v8::Local<v8::Object> exports,
v8::Local<v8::Value> unused,
v8::Local<v8::Context> context,
void* priv) {
v8::Isolate* isolate = context->GetIsolate();
gin_helper::Dictionary(isolate, exports)
.Set("DownloadItem", electron::api::DownloadItem::GetConstructor(isolate)
->GetFunction(context)
.ToLocalChecked());
}
} // namespace
NODE_LINKED_MODULE_CONTEXT_AWARE(electron_browser_download_item, Initialize)

View file

@ -9,25 +9,51 @@
#include <vector> #include <vector>
#include "base/files/file_path.h" #include "base/files/file_path.h"
#include "base/memory/weak_ptr.h"
#include "components/download/public/common/download_item.h" #include "components/download/public/common/download_item.h"
#include "gin/handle.h" #include "gin/handle.h"
#include "gin/wrappable.h"
#include "shell/browser/event_emitter_mixin.h"
#include "shell/browser/ui/file_dialog.h" #include "shell/browser/ui/file_dialog.h"
#include "shell/common/gin_helper/trackable_object.h" #include "shell/common/gin_helper/pinnable.h"
#include "url/gurl.h" #include "url/gurl.h"
namespace electron { namespace electron {
namespace api { namespace api {
class DownloadItem : public gin_helper::TrackableObject<DownloadItem>, class DownloadItem : public gin::Wrappable<DownloadItem>,
public gin_helper::Pinnable<DownloadItem>,
public gin_helper::EventEmitterMixin<DownloadItem>,
public download::DownloadItem::Observer { public download::DownloadItem::Observer {
public: public:
static gin::Handle<DownloadItem> Create(v8::Isolate* isolate, static gin::Handle<DownloadItem> FromOrCreate(v8::Isolate* isolate,
download::DownloadItem* item); download::DownloadItem* item);
static void BuildPrototype(v8::Isolate* isolate, static DownloadItem* FromDownloadItem(download::DownloadItem*);
v8::Local<v8::FunctionTemplate> prototype);
// gin::Wrappable
static gin::WrapperInfo kWrapperInfo;
gin::ObjectTemplateBuilder GetObjectTemplateBuilder(
v8::Isolate* isolate) override;
const char* GetTypeName() override;
// JS API, but also C++ calls it sometimes
void SetSavePath(const base::FilePath& path);
base::FilePath GetSavePath() const;
file_dialog::DialogSettings GetSaveDialogOptions() const;
private:
DownloadItem(v8::Isolate* isolate, download::DownloadItem* download_item);
~DownloadItem() override;
bool CheckAlive() const;
// download::DownloadItem::Observer
void OnDownloadUpdated(download::DownloadItem* download) override;
void OnDownloadDestroyed(download::DownloadItem* download) override;
// JS API
void Pause(); void Pause();
bool IsPaused() const; bool IsPaused() const;
void Resume(); void Resume();
@ -40,30 +66,20 @@ class DownloadItem : public gin_helper::TrackableObject<DownloadItem>,
std::string GetFilename() const; std::string GetFilename() const;
std::string GetContentDisposition() const; std::string GetContentDisposition() const;
const GURL& GetURL() const; const GURL& GetURL() const;
const std::vector<GURL>& GetURLChain() const; v8::Local<v8::Value> GetURLChain(v8::Isolate*) const;
download::DownloadItem::DownloadState GetState() const; download::DownloadItem::DownloadState GetState() const;
bool IsDone() const; bool IsDone() const;
void SetSavePath(const base::FilePath& path);
base::FilePath GetSavePath() const;
file_dialog::DialogSettings GetSaveDialogOptions() const;
void SetSaveDialogOptions(const file_dialog::DialogSettings& options); void SetSaveDialogOptions(const file_dialog::DialogSettings& options);
std::string GetLastModifiedTime() const; std::string GetLastModifiedTime() const;
std::string GetETag() const; std::string GetETag() const;
double GetStartTime() const; double GetStartTime() const;
protected:
DownloadItem(v8::Isolate* isolate, download::DownloadItem* download_item);
~DownloadItem() override;
// Override download::DownloadItem::Observer methods
void OnDownloadUpdated(download::DownloadItem* download) override;
void OnDownloadDestroyed(download::DownloadItem* download) override;
private:
base::FilePath save_path_; base::FilePath save_path_;
file_dialog::DialogSettings dialog_options_; file_dialog::DialogSettings dialog_options_;
download::DownloadItem* download_item_; download::DownloadItem* download_item_;
base::WeakPtrFactory<DownloadItem> weak_factory_{this};
DISALLOW_COPY_AND_ASSIGN(DownloadItem); DISALLOW_COPY_AND_ASSIGN(DownloadItem);
}; };

View file

@ -294,7 +294,7 @@ void Session::OnDownloadCreated(content::DownloadManager* manager,
v8::Locker locker(isolate()); v8::Locker locker(isolate());
v8::HandleScope handle_scope(isolate()); v8::HandleScope handle_scope(isolate());
auto handle = DownloadItem::Create(isolate(), item); auto handle = DownloadItem::FromOrCreate(isolate(), item);
if (item->GetState() == download::DownloadItem::INTERRUPTED) if (item->GetState() == download::DownloadItem::INTERRUPTED)
handle->SetSavePath(item->GetTargetFilePath()); handle->SetSavePath(item->GetTargetFilePath());
content::WebContents* web_contents = content::WebContents* web_contents =
@ -641,7 +641,7 @@ void Session::CreateInterruptedDownload(const gin_helper::Dictionary& options) {
} }
auto* download_manager = auto* download_manager =
content::BrowserContext::GetDownloadManager(browser_context()); content::BrowserContext::GetDownloadManager(browser_context());
download_manager->GetDelegate()->GetNextId(base::BindRepeating( download_manager->GetNextId(base::BindRepeating(
&DownloadIdCallback, download_manager, path, url_chain, mime_type, offset, &DownloadIdCallback, download_manager, path, url_chain, mime_type, offset,
length, last_modified, etag, base::Time::FromDoubleT(start_time))); length, last_modified, etag, base::Time::FromDoubleT(start_time)));
} }

View file

@ -170,7 +170,7 @@ bool Tray::IsDestroyed() {
} }
void Tray::SetImage(gin::Handle<NativeImage> image) { void Tray::SetImage(gin::Handle<NativeImage> image) {
if (!CheckDestroyed()) if (!CheckAlive())
return; return;
#if defined(OS_WIN) #if defined(OS_WIN)
tray_icon_->SetImage(image->GetHICON(GetSystemMetrics(SM_CXSMICON))); tray_icon_->SetImage(image->GetHICON(GetSystemMetrics(SM_CXSMICON)));
@ -180,7 +180,7 @@ void Tray::SetImage(gin::Handle<NativeImage> image) {
} }
void Tray::SetPressedImage(gin::Handle<NativeImage> image) { void Tray::SetPressedImage(gin::Handle<NativeImage> image) {
if (!CheckDestroyed()) if (!CheckAlive())
return; return;
#if defined(OS_WIN) #if defined(OS_WIN)
tray_icon_->SetPressedImage(image->GetHICON(GetSystemMetrics(SM_CXSMICON))); tray_icon_->SetPressedImage(image->GetHICON(GetSystemMetrics(SM_CXSMICON)));
@ -190,13 +190,13 @@ void Tray::SetPressedImage(gin::Handle<NativeImage> image) {
} }
void Tray::SetToolTip(const std::string& tool_tip) { void Tray::SetToolTip(const std::string& tool_tip) {
if (!CheckDestroyed()) if (!CheckAlive())
return; return;
tray_icon_->SetToolTip(tool_tip); tray_icon_->SetToolTip(tool_tip);
} }
void Tray::SetTitle(const std::string& title) { void Tray::SetTitle(const std::string& title) {
if (!CheckDestroyed()) if (!CheckAlive())
return; return;
#if defined(OS_MACOSX) #if defined(OS_MACOSX)
tray_icon_->SetTitle(title); tray_icon_->SetTitle(title);
@ -204,7 +204,7 @@ void Tray::SetTitle(const std::string& title) {
} }
std::string Tray::GetTitle() { std::string Tray::GetTitle() {
if (!CheckDestroyed()) if (!CheckAlive())
return std::string(); return std::string();
#if defined(OS_MACOSX) #if defined(OS_MACOSX)
return tray_icon_->GetTitle(); return tray_icon_->GetTitle();
@ -214,7 +214,7 @@ std::string Tray::GetTitle() {
} }
void Tray::SetIgnoreDoubleClickEvents(bool ignore) { void Tray::SetIgnoreDoubleClickEvents(bool ignore) {
if (!CheckDestroyed()) if (!CheckAlive())
return; return;
#if defined(OS_MACOSX) #if defined(OS_MACOSX)
tray_icon_->SetIgnoreDoubleClickEvents(ignore); tray_icon_->SetIgnoreDoubleClickEvents(ignore);
@ -222,7 +222,7 @@ void Tray::SetIgnoreDoubleClickEvents(bool ignore) {
} }
bool Tray::GetIgnoreDoubleClickEvents() { bool Tray::GetIgnoreDoubleClickEvents() {
if (!CheckDestroyed()) if (!CheckAlive())
return false; return false;
#if defined(OS_MACOSX) #if defined(OS_MACOSX)
return tray_icon_->GetIgnoreDoubleClickEvents(); return tray_icon_->GetIgnoreDoubleClickEvents();
@ -233,7 +233,7 @@ bool Tray::GetIgnoreDoubleClickEvents() {
void Tray::DisplayBalloon(gin_helper::ErrorThrower thrower, void Tray::DisplayBalloon(gin_helper::ErrorThrower thrower,
const gin_helper::Dictionary& options) { const gin_helper::Dictionary& options) {
if (!CheckDestroyed()) if (!CheckAlive())
return; return;
TrayIcon::BalloonOptions balloon_options; TrayIcon::BalloonOptions balloon_options;
@ -263,19 +263,19 @@ void Tray::DisplayBalloon(gin_helper::ErrorThrower thrower,
} }
void Tray::RemoveBalloon() { void Tray::RemoveBalloon() {
if (!CheckDestroyed()) if (!CheckAlive())
return; return;
tray_icon_->RemoveBalloon(); tray_icon_->RemoveBalloon();
} }
void Tray::Focus() { void Tray::Focus() {
if (!CheckDestroyed()) if (!CheckAlive())
return; return;
tray_icon_->Focus(); tray_icon_->Focus();
} }
void Tray::PopUpContextMenu(gin::Arguments* args) { void Tray::PopUpContextMenu(gin::Arguments* args) {
if (!CheckDestroyed()) if (!CheckAlive())
return; return;
gin::Handle<Menu> menu; gin::Handle<Menu> menu;
gfx::Point pos; gfx::Point pos;
@ -298,14 +298,14 @@ void Tray::PopUpContextMenu(gin::Arguments* args) {
} }
void Tray::CloseContextMenu() { void Tray::CloseContextMenu() {
if (!CheckDestroyed()) if (!CheckAlive())
return; return;
tray_icon_->CloseContextMenu(); tray_icon_->CloseContextMenu();
} }
void Tray::SetContextMenu(gin_helper::ErrorThrower thrower, void Tray::SetContextMenu(gin_helper::ErrorThrower thrower,
v8::Local<v8::Value> arg) { v8::Local<v8::Value> arg) {
if (!CheckDestroyed()) if (!CheckAlive())
return; return;
gin::Handle<Menu> menu; gin::Handle<Menu> menu;
if (arg->IsNull()) { if (arg->IsNull()) {
@ -320,12 +320,12 @@ void Tray::SetContextMenu(gin_helper::ErrorThrower thrower,
} }
gfx::Rect Tray::GetBounds() { gfx::Rect Tray::GetBounds() {
if (!CheckDestroyed()) if (!CheckAlive())
return gfx::Rect(); return gfx::Rect();
return tray_icon_->GetBounds(); return tray_icon_->GetBounds();
} }
bool Tray::CheckDestroyed() { bool Tray::CheckAlive() {
if (!tray_icon_) { if (!tray_icon_) {
v8::Isolate* isolate = v8::Isolate::GetCurrent(); v8::Isolate* isolate = v8::Isolate::GetCurrent();
v8::Locker locker(isolate); v8::Locker locker(isolate);

View file

@ -99,7 +99,7 @@ class Tray : public gin::Wrappable<Tray>,
v8::Local<v8::Value> arg); v8::Local<v8::Value> arg);
gfx::Rect GetBounds(); gfx::Rect GetBounds();
bool CheckDestroyed(); bool CheckAlive();
v8::Global<v8::Value> menu_; v8::Global<v8::Value> menu_;
std::unique_ptr<TrayIcon> tray_icon_; std::unique_ptr<TrayIcon> tray_icon_;

View file

@ -65,11 +65,7 @@ ElectronDownloadManagerDelegate::~ElectronDownloadManagerDelegate() {
void ElectronDownloadManagerDelegate::GetItemSavePath( void ElectronDownloadManagerDelegate::GetItemSavePath(
download::DownloadItem* item, download::DownloadItem* item,
base::FilePath* path) { base::FilePath* path) {
v8::Isolate* isolate = v8::Isolate::GetCurrent(); api::DownloadItem* download = api::DownloadItem::FromDownloadItem(item);
v8::Locker locker(isolate);
v8::HandleScope handle_scope(isolate);
api::DownloadItem* download =
api::DownloadItem::FromWrappedClass(isolate, item);
if (download) if (download)
*path = download->GetSavePath(); *path = download->GetSavePath();
} }
@ -77,11 +73,7 @@ void ElectronDownloadManagerDelegate::GetItemSavePath(
void ElectronDownloadManagerDelegate::GetItemSaveDialogOptions( void ElectronDownloadManagerDelegate::GetItemSaveDialogOptions(
download::DownloadItem* item, download::DownloadItem* item,
file_dialog::DialogSettings* options) { file_dialog::DialogSettings* options) {
v8::Isolate* isolate = v8::Isolate::GetCurrent(); api::DownloadItem* download = api::DownloadItem::FromDownloadItem(item);
v8::Locker locker(isolate);
v8::HandleScope handle_scope(isolate);
api::DownloadItem* download =
api::DownloadItem::FromWrappedClass(isolate, item);
if (download) if (download)
*options = download->GetSaveDialogOptions(); *options = download->GetSaveDialogOptions();
} }
@ -165,13 +157,9 @@ void ElectronDownloadManagerDelegate::OnDownloadSaveDialogDone(
browser_context->prefs()->SetFilePath(prefs::kDownloadDefaultDirectory, browser_context->prefs()->SetFilePath(prefs::kDownloadDefaultDirectory,
path.DirName()); path.DirName());
v8::Isolate* isolate = v8::Isolate::GetCurrent(); api::DownloadItem* download = api::DownloadItem::FromDownloadItem(item);
v8::Locker locker(isolate); if (download)
v8::HandleScope handle_scope(isolate); download->SetSavePath(path);
api::DownloadItem* download_item =
api::DownloadItem::FromWrappedClass(isolate, item);
if (download_item)
download_item->SetSavePath(path);
} }
} }

View file

@ -38,7 +38,6 @@
V(electron_browser_browser_view) \ V(electron_browser_browser_view) \
V(electron_browser_content_tracing) \ V(electron_browser_content_tracing) \
V(electron_browser_dialog) \ V(electron_browser_dialog) \
V(electron_browser_download_item) \
V(electron_browser_event) \ V(electron_browser_event) \
V(electron_browser_event_emitter) \ V(electron_browser_event_emitter) \
V(electron_browser_global_shortcut) \ V(electron_browser_global_shortcut) \

View file

@ -311,7 +311,7 @@ describe('session module', () => {
const { item, itemUrl, itemFilename } = await downloadPrevented; const { item, itemUrl, itemFilename } = await downloadPrevented;
expect(itemUrl).to.equal(url); expect(itemUrl).to.equal(url);
expect(itemFilename).to.equal('mockFile.txt'); expect(itemFilename).to.equal('mockFile.txt');
expect(() => item.getURL()).to.throw('Object has been destroyed'); expect(() => item.getURL()).to.throw('DownloadItem used after being destroyed');
}); });
}); });