refactor: ginify DownloadItem (#22924)
This commit is contained in:
parent
6159066c26
commit
0a78ab4b98
9 changed files with 153 additions and 101 deletions
|
@ -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);
|
|
||||||
|
|
|
@ -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)
|
|
||||||
|
|
|
@ -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);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -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)));
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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_;
|
||||||
|
|
|
@ -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);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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) \
|
||||||
|
|
|
@ -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');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue