// Copyright 2013 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef GIN_CONVERTER_H_ #define GIN_CONVERTER_H_ #include #include #include #include "base/logging.h" #include "base/strings/string_piece.h" #include "gin/gin_export.h" #include "v8/include/v8.h" namespace gin { template bool SetProperty(v8::Isolate* isolate, v8::Local object, KeyType key, v8::Local value) { auto maybe = object->DefineOwnProperty(isolate->GetCurrentContext(), key, value); return !maybe.IsNothing() && maybe.FromJust(); } template struct ToV8ReturnsMaybe { static const bool value = false; }; template struct Converter {}; template<> struct GIN_EXPORT Converter { static v8::Local ToV8(v8::Isolate* isolate, bool val); static bool FromV8(v8::Isolate* isolate, v8::Local val, bool* out); }; template<> struct GIN_EXPORT Converter { static v8::Local ToV8(v8::Isolate* isolate, int32_t val); static bool FromV8(v8::Isolate* isolate, v8::Local val, int32_t* out); }; template<> struct GIN_EXPORT Converter { static v8::Local ToV8(v8::Isolate* isolate, uint32_t val); static bool FromV8(v8::Isolate* isolate, v8::Local val, uint32_t* out); }; template<> struct GIN_EXPORT Converter { // Warning: JavaScript cannot represent 64 integers precisely. static v8::Local ToV8(v8::Isolate* isolate, int64_t val); static bool FromV8(v8::Isolate* isolate, v8::Local val, int64_t* out); }; template<> struct GIN_EXPORT Converter { // Warning: JavaScript cannot represent 64 integers precisely. static v8::Local ToV8(v8::Isolate* isolate, uint64_t val); static bool FromV8(v8::Isolate* isolate, v8::Local val, uint64_t* out); }; template<> struct GIN_EXPORT Converter { static v8::Local ToV8(v8::Isolate* isolate, float val); static bool FromV8(v8::Isolate* isolate, v8::Local val, float* out); }; template<> struct GIN_EXPORT Converter { static v8::Local ToV8(v8::Isolate* isolate, double val); static bool FromV8(v8::Isolate* isolate, v8::Local val, double* out); }; template<> struct GIN_EXPORT Converter { // This crashes when val.size() > v8::String::kMaxLength. static v8::Local ToV8(v8::Isolate* isolate, const base::StringPiece& val); // No conversion out is possible because StringPiece does not contain storage. }; template<> struct GIN_EXPORT Converter { // This crashes when val.size() > v8::String::kMaxLength. static v8::Local ToV8(v8::Isolate* isolate, const std::string& val); static bool FromV8(v8::Isolate* isolate, v8::Local val, std::string* out); }; template<> struct GIN_EXPORT Converter > { static bool FromV8(v8::Isolate* isolate, v8::Local val, v8::Local* out); }; template<> struct GIN_EXPORT Converter > { static v8::Local ToV8(v8::Isolate* isolate, v8::Local val); static bool FromV8(v8::Isolate* isolate, v8::Local val, v8::Local* out); }; template<> struct GIN_EXPORT Converter > { static v8::Local ToV8(v8::Isolate* isolate, v8::Local val); static bool FromV8(v8::Isolate* isolate, v8::Local val, v8::Local* out); }; template<> struct GIN_EXPORT Converter > { static v8::Local ToV8(v8::Isolate* isolate, v8::Local val); static bool FromV8(v8::Isolate* isolate, v8::Local val, v8::Local* out); }; template<> struct GIN_EXPORT Converter > { static v8::Local ToV8(v8::Isolate* isolate, v8::Local val); static bool FromV8(v8::Isolate* isolate, v8::Local val, v8::Local* out); }; template struct Converter > { static v8::MaybeLocal ToV8(v8::Local context, const std::vector& val) { v8::Isolate* isolate = context->GetIsolate(); v8::Local result( v8::Array::New(isolate, static_cast(val.size()))); for (uint32_t i = 0; i < val.size(); ++i) { auto maybe = result->Set(context, i, Converter::ToV8(isolate, val[i])); if (maybe.IsNothing() || !maybe.FromJust()) return v8::MaybeLocal(); } return result; } static bool FromV8(v8::Isolate* isolate, v8::Local val, std::vector* out) { if (!val->IsArray()) return false; std::vector result; v8::Local array(v8::Local::Cast(val)); uint32_t length = array->Length(); for (uint32_t i = 0; i < length; ++i) { v8::Local v8_item; if (!array->Get(isolate->GetCurrentContext(), i).ToLocal(&v8_item)) return false; T item; if (!Converter::FromV8(isolate, v8_item, &item)) return false; result.push_back(item); } out->swap(result); return true; } }; template struct ToV8ReturnsMaybe> { static const bool value = true; }; // Convenience functions that deduce T. template v8::Local ConvertToV8(v8::Isolate* isolate, T input) { return Converter::ToV8(isolate, input); } template v8::MaybeLocal ConvertToV8(v8::Local context, T input) { return Converter::ToV8(context, input); } template::value> struct ToV8Traits; template struct ToV8Traits { static bool TryConvertToV8(v8::Isolate* isolate, T input, v8::Local* output) { auto maybe = ConvertToV8(isolate->GetCurrentContext(), input); if (maybe.IsEmpty()) return false; *output = maybe.ToLocalChecked(); return true; } }; template struct ToV8Traits { static bool TryConvertToV8(v8::Isolate* isolate, T input, v8::Local* output) { *output = ConvertToV8(isolate, input); return true; } }; template bool TryConvertToV8(v8::Isolate* isolate, T input, v8::Local* output) { return ToV8Traits::TryConvertToV8(isolate, input, output); } // This crashes when input.size() > v8::String::kMaxLength. GIN_EXPORT inline v8::Local StringToV8( v8::Isolate* isolate, const base::StringPiece& input) { return ConvertToV8(isolate, input).As(); } // This crashes when input.size() > v8::String::kMaxLength. GIN_EXPORT v8::Local StringToSymbol(v8::Isolate* isolate, const base::StringPiece& val); template bool ConvertFromV8(v8::Isolate* isolate, v8::Local input, T* result) { return Converter::FromV8(isolate, input, result); } GIN_EXPORT std::string V8ToString(v8::Local value); } // namespace gin #endif // GIN_CONVERTER_H_