electron/shell/common/gin_helper/constructor.h

193 lines
7.3 KiB
C
Raw Normal View History

// 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 08:59:50 +00:00
#ifndef ELECTRON_SHELL_COMMON_GIN_HELPER_CONSTRUCTOR_H_
#define ELECTRON_SHELL_COMMON_GIN_HELPER_CONSTRUCTOR_H_
2014-04-15 07:30:46 +00:00
#include "shell/common/gin_helper/function_template.h"
#include "shell/common/gin_helper/wrappable_base.h"
2014-04-15 07:30:46 +00:00
namespace gin_helper {
2014-04-15 07:30:46 +00:00
namespace internal {
// This set of templates invokes a base::RepeatingCallback by converting the
// Arguments into native types. It relies on the function_template.h to provide
// helper templates.
2016-04-25 01:17:39 +00:00
inline WrappableBase* InvokeFactory(
gin::Arguments* args,
const base::RepeatingCallback<WrappableBase*()>& callback) {
return callback.Run();
2019-02-27 19:14:23 +00:00
}
2019-02-27 19:14:23 +00:00
template <typename P1>
2016-04-25 01:17:39 +00:00
inline WrappableBase* InvokeFactory(
gin::Arguments* args,
const base::RepeatingCallback<WrappableBase*(P1)>& callback) {
typename CallbackParamTraits<P1>::LocalType a1;
if (!gin_helper::GetNextArgument(args, {.holder_is_first_argument = true}, 0,
&a1))
2016-04-25 01:17:39 +00:00
return nullptr;
return callback.Run(a1);
2019-02-27 19:14:23 +00:00
}
2019-02-27 19:14:23 +00:00
template <typename P1, typename P2>
2016-04-25 01:17:39 +00:00
inline WrappableBase* InvokeFactory(
gin::Arguments* args,
const base::RepeatingCallback<WrappableBase*(P1, P2)>& callback) {
typename CallbackParamTraits<P1>::LocalType a1;
typename CallbackParamTraits<P2>::LocalType a2;
if (!gin_helper::GetNextArgument(args, {.holder_is_first_argument = true}, 0,
&a1) ||
!gin_helper::GetNextArgument(args, {.holder_is_first_argument = false}, 0,
&a2))
2016-04-25 01:17:39 +00:00
return nullptr;
return callback.Run(a1, a2);
2019-02-27 19:14:23 +00:00
}
2019-02-27 19:14:23 +00:00
template <typename P1, typename P2, typename P3>
2016-04-25 01:17:39 +00:00
inline WrappableBase* InvokeFactory(
gin::Arguments* args,
const base::RepeatingCallback<WrappableBase*(P1, P2, P3)>& callback) {
typename CallbackParamTraits<P1>::LocalType a1;
typename CallbackParamTraits<P2>::LocalType a2;
typename CallbackParamTraits<P3>::LocalType a3;
if (!gin_helper::GetNextArgument(args, {.holder_is_first_argument = true}, 0,
&a1) ||
!gin_helper::GetNextArgument(args, {.holder_is_first_argument = false}, 0,
&a2) ||
!gin_helper::GetNextArgument(args, {.holder_is_first_argument = false}, 0,
&a3))
2016-04-25 01:17:39 +00:00
return nullptr;
return callback.Run(a1, a2, a3);
2019-02-27 19:14:23 +00:00
}
2019-02-27 19:14:23 +00:00
template <typename P1, typename P2, typename P3, typename P4>
2016-04-25 01:17:39 +00:00
inline WrappableBase* InvokeFactory(
gin::Arguments* args,
const base::RepeatingCallback<WrappableBase*(P1, P2, P3, P4)>& callback) {
typename CallbackParamTraits<P1>::LocalType a1;
typename CallbackParamTraits<P2>::LocalType a2;
typename CallbackParamTraits<P3>::LocalType a3;
typename CallbackParamTraits<P4>::LocalType a4;
if (!gin_helper::GetNextArgument(args, {.holder_is_first_argument = true}, 0,
&a1) ||
!gin_helper::GetNextArgument(args, {.holder_is_first_argument = false}, 0,
&a2) ||
!gin_helper::GetNextArgument(args, {.holder_is_first_argument = false}, 0,
&a3) ||
!gin_helper::GetNextArgument(args, {.holder_is_first_argument = false}, 0,
&a4))
2016-04-25 01:17:39 +00:00
return nullptr;
return callback.Run(a1, a2, a3, a4);
2019-02-27 19:14:23 +00:00
}
2019-02-27 19:14:23 +00:00
template <typename P1, typename P2, typename P3, typename P4, typename P5>
2016-04-25 01:17:39 +00:00
inline WrappableBase* InvokeFactory(
gin::Arguments* args,
const base::RepeatingCallback<WrappableBase*(P1, P2, P3, P4, P5)>&
callback) {
typename CallbackParamTraits<P1>::LocalType a1;
typename CallbackParamTraits<P2>::LocalType a2;
typename CallbackParamTraits<P3>::LocalType a3;
typename CallbackParamTraits<P4>::LocalType a4;
typename CallbackParamTraits<P5>::LocalType a5;
if (!gin_helper::GetNextArgument(args, {.holder_is_first_argument = true}, 0,
&a1) ||
!gin_helper::GetNextArgument(args, {.holder_is_first_argument = false}, 0,
&a2) ||
!gin_helper::GetNextArgument(args, {.holder_is_first_argument = false}, 0,
&a3) ||
!gin_helper::GetNextArgument(args, {.holder_is_first_argument = false}, 0,
&a4) ||
!gin_helper::GetNextArgument(args, {.holder_is_first_argument = false}, 0,
&a5))
2016-04-25 01:17:39 +00:00
return nullptr;
return callback.Run(a1, a2, a3, a4, a5);
2019-02-27 19:14:23 +00:00
}
2019-02-27 19:14:23 +00:00
template <typename P1,
typename P2,
typename P3,
typename P4,
typename P5,
typename P6>
2016-04-25 01:17:39 +00:00
inline WrappableBase* InvokeFactory(
gin::Arguments* args,
const base::RepeatingCallback<WrappableBase*(P1, P2, P3, P4, P5, P6)>&
callback) {
typename CallbackParamTraits<P1>::LocalType a1;
typename CallbackParamTraits<P2>::LocalType a2;
typename CallbackParamTraits<P3>::LocalType a3;
typename CallbackParamTraits<P4>::LocalType a4;
typename CallbackParamTraits<P5>::LocalType a5;
typename CallbackParamTraits<P6>::LocalType a6;
if (!gin_helper::GetNextArgument(args, {.holder_is_first_argument = true}, 0,
&a1) ||
!gin_helper::GetNextArgument(args, {.holder_is_first_argument = false}, 0,
&a2) ||
!gin_helper::GetNextArgument(args, {.holder_is_first_argument = false}, 0,
&a3) ||
!gin_helper::GetNextArgument(args, {.holder_is_first_argument = false}, 0,
&a4) ||
!gin_helper::GetNextArgument(args, {.holder_is_first_argument = false}, 0,
&a5) ||
!gin_helper::GetNextArgument(args, {.holder_is_first_argument = false}, 0,
&a6))
2016-04-25 01:17:39 +00:00
return nullptr;
return callback.Run(a1, a2, a3, a4, a5, a6);
2019-02-27 19:14:23 +00:00
}
2019-02-27 19:14:23 +00:00
template <typename Sig>
void InvokeNew(const base::RepeatingCallback<Sig>& factory,
2019-02-27 19:14:23 +00:00
v8::Isolate* isolate,
gin_helper::Arguments* args) {
2016-08-02 08:01:19 +00:00
if (!args->IsConstructCall()) {
args->ThrowError("Requires constructor call");
2018-05-17 21:36:15 +00:00
return;
}
2014-04-15 07:30:46 +00:00
2016-08-02 08:01:19 +00:00
WrappableBase* object;
{
// Don't continue if the constructor throws an exception.
2018-04-10 23:17:20 +00:00
v8::TryCatch try_catch(isolate);
2016-08-02 08:01:19 +00:00
object = internal::InvokeFactory(args, factory);
if (try_catch.HasCaught()) {
try_catch.ReThrow();
2018-05-17 21:36:15 +00:00
return;
}
}
2016-08-02 08:01:19 +00:00
if (!object)
args->ThrowError();
2014-04-15 07:30:46 +00:00
2018-05-17 21:36:15 +00:00
return;
}
2016-08-02 08:01:19 +00:00
} // namespace internal
// 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 16:17:57 +00:00
v8::Local<v8::FunctionTemplate> templ = gin_helper::CreateFunctionTemplate(
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 07:30:46 +00:00
#endif // ELECTRON_SHELL_COMMON_GIN_HELPER_CONSTRUCTOR_H_