electron/shell/common/gin_helper/trackable_object.h

140 lines
3.9 KiB
C
Raw Normal View History

2015-06-24 06:36:05 +00:00
// Copyright (c) 2015 GitHub, Inc.
// Use of this source code is governed by the MIT license that can be
// found in the LICENSE file.
#ifndef SHELL_COMMON_GIN_HELPER_TRACKABLE_OBJECT_H_
#define SHELL_COMMON_GIN_HELPER_TRACKABLE_OBJECT_H_
2015-06-24 06:36:05 +00:00
2015-06-24 09:58:12 +00:00
#include <vector>
#include "base/bind.h"
#include "base/memory/weak_ptr.h"
#include "shell/common/gin_helper/event_emitter.h"
#include "shell/common/key_weak_map.h"
2015-06-24 06:36:05 +00:00
namespace base {
class SupportsUserData;
}
namespace gin_helper {
2015-06-24 06:36:05 +00:00
2015-06-24 09:58:12 +00:00
// Users should use TrackableObject instead.
2016-04-25 01:17:54 +00:00
class TrackableObjectBase {
2015-06-24 06:36:05 +00:00
public:
2015-06-24 09:58:12 +00:00
TrackableObjectBase();
2015-06-24 06:36:05 +00:00
2015-06-24 09:58:12 +00:00
// The ID in weak map.
int32_t weak_map_id() const { return weak_map_id_; }
2015-06-24 09:58:12 +00:00
// Wrap TrackableObject into a class that SupportsUserData.
void AttachAsUserData(base::SupportsUserData* wrapped);
2017-09-12 12:17:11 +00:00
// Get the weak_map_id from SupportsUserData.
static int32_t GetIDFromWrappedClass(base::SupportsUserData* wrapped);
2015-06-24 09:58:12 +00:00
protected:
2016-04-25 01:17:54 +00:00
virtual ~TrackableObjectBase();
// Returns a closure that can destroy the native class.
base::OnceClosure GetDestroyClosure();
2015-06-24 09:58:12 +00:00
int32_t weak_map_id_ = 0;
2015-06-24 09:58:12 +00:00
private:
void Destroy();
base::WeakPtrFactory<TrackableObjectBase> weak_factory_;
2015-06-24 09:58:12 +00:00
DISALLOW_COPY_AND_ASSIGN(TrackableObjectBase);
};
// All instances of TrackableObject will be kept in a weak map and can be got
// from its ID.
template <typename T>
class TrackableObject : public TrackableObjectBase, public EventEmitter<T> {
2015-06-24 09:58:12 +00:00
public:
2016-04-25 01:17:54 +00:00
// Mark the JS object as destroyed.
void MarkDestroyed() {
v8::HandleScope scope(gin_helper::Wrappable<T>::isolate());
v8::Local<v8::Object> wrapper = gin_helper::Wrappable<T>::GetWrapper();
2017-10-24 05:38:55 +00:00
if (!wrapper.IsEmpty()) {
2017-10-27 07:07:54 +00:00
wrapper->SetAlignedPointerInInternalField(0, nullptr);
gin_helper::WrappableBase::wrapper_.ClearWeak();
2017-10-24 05:38:55 +00:00
}
2016-04-25 01:17:54 +00:00
}
bool IsDestroyed() {
v8::HandleScope scope(gin_helper::Wrappable<T>::isolate());
v8::Local<v8::Object> wrapper = gin_helper::Wrappable<T>::GetWrapper();
return wrapper->InternalFieldCount() == 0 ||
wrapper->GetAlignedPointerFromInternalField(0) == nullptr;
}
2015-06-24 09:58:12 +00:00
// Finds out the TrackableObject from its ID in weak map.
static T* FromWeakMapID(v8::Isolate* isolate, int32_t id) {
if (!weak_map_)
return nullptr;
v8::HandleScope scope(isolate);
v8::MaybeLocal<v8::Object> object = weak_map_->Get(isolate, id);
2015-06-24 09:58:12 +00:00
if (object.IsEmpty())
return nullptr;
T* self = nullptr;
gin::ConvertFromV8(isolate, object.ToLocalChecked(), &self);
2015-06-24 09:58:12 +00:00
return self;
}
// Finds out the TrackableObject from the class it wraps.
static T* FromWrappedClass(v8::Isolate* isolate,
base::SupportsUserData* wrapped) {
int32_t id = GetIDFromWrappedClass(wrapped);
if (!id)
return nullptr;
return FromWeakMapID(isolate, id);
}
// Returns all objects in this class's weak map.
static std::vector<v8::Local<v8::Object>> GetAll(v8::Isolate* isolate) {
if (weak_map_)
return weak_map_->Values(isolate);
else
return std::vector<v8::Local<v8::Object>>();
2015-06-24 09:58:12 +00:00
}
// Removes this instance from the weak map.
void RemoveFromWeakMap() {
if (weak_map_ && weak_map_->Has(weak_map_id()))
weak_map_->Remove(weak_map_id());
2015-06-24 09:58:12 +00:00
}
2015-06-24 06:36:05 +00:00
protected:
2018-04-18 01:44:10 +00:00
TrackableObject() { weak_map_id_ = ++next_id_; }
2016-04-25 01:17:54 +00:00
2018-04-18 01:44:10 +00:00
~TrackableObject() override { RemoveFromWeakMap(); }
2015-06-24 06:36:05 +00:00
void InitWith(v8::Isolate* isolate, v8::Local<v8::Object> wrapper) override {
gin_helper::WrappableBase::InitWith(isolate, wrapper);
if (!weak_map_) {
weak_map_ = new electron::KeyWeakMap<int32_t>;
}
weak_map_->Set(isolate, weak_map_id_, wrapper);
2015-06-24 09:58:12 +00:00
}
2015-06-24 06:36:05 +00:00
private:
2016-05-11 06:51:22 +00:00
static int32_t next_id_;
static electron::KeyWeakMap<int32_t>* weak_map_; // leaked on purpose
2015-06-24 06:36:05 +00:00
DISALLOW_COPY_AND_ASSIGN(TrackableObject);
};
template <typename T>
int32_t TrackableObject<T>::next_id_ = 0;
2016-05-11 06:51:22 +00:00
template <typename T>
electron::KeyWeakMap<int32_t>* TrackableObject<T>::weak_map_ = nullptr;
2015-06-24 09:58:12 +00:00
} // namespace gin_helper
2015-06-24 06:36:05 +00:00
#endif // SHELL_COMMON_GIN_HELPER_TRACKABLE_OBJECT_H_