2019-12-05 18:46:34 +09:00
|
|
|
// Copyright (c) 2018 GitHub, Inc.
|
|
|
|
// Use of this source code is governed by the MIT license that can be
|
|
|
|
// found in the LICENSE file.
|
2014-08-10 16:59:50 +08:00
|
|
|
|
2021-11-22 08:34:31 +01:00
|
|
|
#ifndef ELECTRON_SHELL_COMMON_GIN_HELPER_CONSTRUCTOR_H_
|
|
|
|
#define ELECTRON_SHELL_COMMON_GIN_HELPER_CONSTRUCTOR_H_
|
2014-04-15 15:30:46 +08:00
|
|
|
|
2025-03-13 23:41:22 -05:00
|
|
|
#include <tuple>
|
|
|
|
|
2019-10-15 10:15:23 +09:00
|
|
|
#include "shell/common/gin_helper/function_template.h"
|
2019-12-05 18:46:34 +09:00
|
|
|
#include "shell/common/gin_helper/wrappable_base.h"
|
2014-04-15 15:30:46 +08:00
|
|
|
|
2019-12-05 18:46:34 +09:00
|
|
|
namespace gin_helper {
|
2014-04-15 15:30:46 +08:00
|
|
|
|
2014-04-16 09:50:51 +08:00
|
|
|
namespace internal {
|
|
|
|
|
2025-03-13 23:41:22 -05:00
|
|
|
// Convert a `gin::Argument`'s arguments into a tuple of native types
|
|
|
|
// by iteratively calling gin_helper::GetNextArgument().
|
|
|
|
template <typename... Types>
|
|
|
|
class GinArgumentsToTuple {
|
|
|
|
public:
|
|
|
|
[[nodiscard]] static std::pair<bool /*ok*/, std::tuple<Types...>> GetArgs(
|
|
|
|
gin::Arguments* args) {
|
|
|
|
bool ok = true;
|
|
|
|
InvokerOptions opts{.holder_is_first_argument = true};
|
|
|
|
auto tup = std::make_tuple(GetNextArg<Types>(args, opts, ok)...);
|
|
|
|
return {ok, std::move(tup)};
|
|
|
|
}
|
2014-04-16 09:50:51 +08:00
|
|
|
|
2025-03-13 23:41:22 -05:00
|
|
|
private:
|
|
|
|
template <typename T>
|
|
|
|
static T GetNextArg(gin::Arguments* args, InvokerOptions& opts, bool& ok) {
|
|
|
|
auto val = T{};
|
|
|
|
ok = ok && gin_helper::GetNextArgument(args, opts, 0, &val);
|
|
|
|
opts.holder_is_first_argument = false;
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
};
|
2014-04-16 09:50:51 +08:00
|
|
|
|
2025-03-13 23:41:22 -05:00
|
|
|
// Invoke a callback with arguments extracted from `args`.
|
|
|
|
template <typename... Types>
|
|
|
|
WrappableBase* InvokeFactory(
|
2019-12-05 18:46:34 +09:00
|
|
|
gin::Arguments* args,
|
2025-03-13 23:41:22 -05:00
|
|
|
const base::RepeatingCallback<WrappableBase*(Types...)>& callback) {
|
|
|
|
auto [ok, tup] = GinArgumentsToTuple<Types...>::GetArgs(args);
|
|
|
|
if (!ok)
|
|
|
|
return {};
|
|
|
|
return std::apply(
|
|
|
|
[&callback](Types... args) { return callback.Run(std::move(args)...); },
|
|
|
|
std::move(tup));
|
2019-02-27 11:14:23 -08:00
|
|
|
}
|
2014-04-16 09:50:51 +08:00
|
|
|
|
2019-02-27 11:14:23 -08:00
|
|
|
template <typename Sig>
|
2021-05-06 15:01:04 -07:00
|
|
|
void InvokeNew(const base::RepeatingCallback<Sig>& factory,
|
2019-02-27 11:14:23 -08:00
|
|
|
v8::Isolate* isolate,
|
2019-12-05 18:46:34 +09:00
|
|
|
gin_helper::Arguments* args) {
|
2016-08-02 17:01:19 +09:00
|
|
|
if (!args->IsConstructCall()) {
|
|
|
|
args->ThrowError("Requires constructor call");
|
2018-05-17 23:36:15 +02:00
|
|
|
return;
|
2014-04-16 09:50:51 +08:00
|
|
|
}
|
2014-04-15 15:30:46 +08:00
|
|
|
|
2016-08-02 17:01:19 +09:00
|
|
|
WrappableBase* object;
|
|
|
|
{
|
|
|
|
// Don't continue if the constructor throws an exception.
|
2025-03-13 23:41:22 -05:00
|
|
|
v8::TryCatch try_catch{isolate};
|
2016-08-02 17:01:19 +09:00
|
|
|
object = internal::InvokeFactory(args, factory);
|
|
|
|
if (try_catch.HasCaught()) {
|
|
|
|
try_catch.ReThrow();
|
2018-05-17 23:36:15 +02:00
|
|
|
return;
|
2014-10-30 21:54:29 +08:00
|
|
|
}
|
2014-04-16 09:50:51 +08:00
|
|
|
}
|
|
|
|
|
2016-08-02 17:01:19 +09:00
|
|
|
if (!object)
|
|
|
|
args->ThrowError();
|
2014-04-15 15:30:46 +08:00
|
|
|
|
2018-05-17 23:36:15 +02:00
|
|
|
return;
|
2014-04-16 09:50:51 +08:00
|
|
|
}
|
|
|
|
|
2016-08-02 17:01:19 +09:00
|
|
|
} // namespace internal
|
|
|
|
|
2019-12-05 18:46:34 +09:00
|
|
|
// Create a FunctionTemplate that can be "new"ed in JavaScript.
|
|
|
|
// It is user's responsibility to ensure this function is called for one type
|
|
|
|
// only ONCE in the program's whole lifetime, otherwise we would have memory
|
|
|
|
// leak.
|
|
|
|
template <typename T, typename Sig>
|
|
|
|
v8::Local<v8::Function> CreateConstructor(
|
|
|
|
v8::Isolate* isolate,
|
|
|
|
const base::RepeatingCallback<Sig>& func) {
|
|
|
|
#ifndef NDEBUG
|
|
|
|
static bool called = false;
|
|
|
|
CHECK(!called) << "CreateConstructor can only be called for one type once";
|
|
|
|
called = true;
|
|
|
|
#endif
|
2020-07-30 09:17:57 -07:00
|
|
|
v8::Local<v8::FunctionTemplate> templ = gin_helper::CreateFunctionTemplate(
|
2019-12-05 18:46:34 +09:00
|
|
|
isolate, base::BindRepeating(&internal::InvokeNew<Sig>, func));
|
|
|
|
templ->InstanceTemplate()->SetInternalFieldCount(1);
|
|
|
|
T::BuildPrototype(isolate, templ);
|
|
|
|
return templ->GetFunction(isolate->GetCurrentContext()).ToLocalChecked();
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace gin_helper
|
2014-04-15 15:30:46 +08:00
|
|
|
|
2021-11-22 08:34:31 +01:00
|
|
|
#endif // ELECTRON_SHELL_COMMON_GIN_HELPER_CONSTRUCTOR_H_
|