mirror of
https://github.com/matrix-construct/construct
synced 2024-11-17 23:40:57 +01:00
ircd::js: Additional object utils; array-object type support.
This commit is contained in:
parent
8a3f592753
commit
42f0c6b06b
2 changed files with 119 additions and 3 deletions
|
@ -30,11 +30,17 @@ template<lifetime L>
|
||||||
struct object
|
struct object
|
||||||
:root<JSObject *, L>
|
:root<JSObject *, L>
|
||||||
{
|
{
|
||||||
|
IRCD_OVERLOAD(array)
|
||||||
|
|
||||||
using handle = typename root<JSObject *, L>::handle;
|
using handle = typename root<JSObject *, L>::handle;
|
||||||
using handle_mutable = typename root<JSObject *, L>::handle_mutable;
|
using handle_mutable = typename root<JSObject *, L>::handle_mutable;
|
||||||
|
|
||||||
operator JS::Value() const;
|
operator JS::Value() const;
|
||||||
|
|
||||||
|
// for array objects
|
||||||
|
uint32_t size() const;
|
||||||
|
void resize(const uint32_t &);
|
||||||
|
|
||||||
// new object
|
// new object
|
||||||
object(const JSClass *const &, const handle &ctor, const JS::HandleValueArray &args);
|
object(const JSClass *const &, const handle &ctor, const JS::HandleValueArray &args);
|
||||||
object(const JSClass *const &, const JS::CallArgs &args);
|
object(const JSClass *const &, const JS::CallArgs &args);
|
||||||
|
@ -43,6 +49,8 @@ struct object
|
||||||
|
|
||||||
template<class T, lifetime LL> object(const root<T, LL> &);
|
template<class T, lifetime LL> object(const root<T, LL> &);
|
||||||
using root<JSObject *, L>::root;
|
using root<JSObject *, L>::root;
|
||||||
|
object(array_t, const size_t &length);
|
||||||
|
object(const JS::HandleValueArray &);
|
||||||
object(const value<L> &);
|
object(const value<L> &);
|
||||||
object(JSObject *const &);
|
object(JSObject *const &);
|
||||||
object(JSObject &);
|
object(JSObject &);
|
||||||
|
@ -57,6 +65,13 @@ template<class T, lifetime L> T &priv(const object<L> &);
|
||||||
template<lifetime L> void priv(object<L> &, void *const &);
|
template<lifetime L> void priv(object<L> &, void *const &);
|
||||||
template<lifetime L> void priv(object<L> &, const void *const &);
|
template<lifetime L> void priv(object<L> &, const void *const &);
|
||||||
|
|
||||||
|
template<lifetime L> bool is_extensible(const object<L> &);
|
||||||
|
template<lifetime L> bool is_array(const object<L> &);
|
||||||
|
template<lifetime L> uint32_t size(const object<L> &);
|
||||||
|
|
||||||
|
template<lifetime L> bool deep_freeze(const object<L> &);
|
||||||
|
template<lifetime L> bool freeze(const object<L> &);
|
||||||
|
|
||||||
} // namespace basic
|
} // namespace basic
|
||||||
|
|
||||||
using object = basic::object<lifetime::stack>;
|
using object = basic::object<lifetime::stack>;
|
||||||
|
@ -77,6 +92,8 @@ object<L>::object()
|
||||||
JS_NewPlainObject(*cx)
|
JS_NewPlainObject(*cx)
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
|
if(unlikely(!this->get()))
|
||||||
|
throw internal_error("NULL object (plain)");
|
||||||
}
|
}
|
||||||
|
|
||||||
template<lifetime L>
|
template<lifetime L>
|
||||||
|
@ -101,6 +118,29 @@ object<L>::object(const value<L> &val)
|
||||||
throw type_error("Value is not an Object");
|
throw type_error("Value is not an Object");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<lifetime L>
|
||||||
|
object<L>::object(const JS::HandleValueArray &values)
|
||||||
|
:object<L>::root::type
|
||||||
|
{
|
||||||
|
JS_NewArrayObject(*cx, values)
|
||||||
|
}
|
||||||
|
{
|
||||||
|
if(unlikely(!this->get()))
|
||||||
|
throw internal_error("NULL object (array)");
|
||||||
|
}
|
||||||
|
|
||||||
|
template<lifetime L>
|
||||||
|
object<L>::object(array_t,
|
||||||
|
const size_t &length)
|
||||||
|
:object<L>::root::type
|
||||||
|
{
|
||||||
|
JS_NewArrayObject(*cx, length)
|
||||||
|
}
|
||||||
|
{
|
||||||
|
if(unlikely(!this->get()))
|
||||||
|
throw internal_error("NULL object (array)");
|
||||||
|
}
|
||||||
|
|
||||||
template<lifetime L>
|
template<lifetime L>
|
||||||
template<class T,
|
template<class T,
|
||||||
lifetime LL>
|
lifetime LL>
|
||||||
|
@ -116,6 +156,8 @@ object<L>::object(const JSClass *const &clasp)
|
||||||
JS_NewObject(*cx, clasp)
|
JS_NewObject(*cx, clasp)
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
|
if(unlikely(!this->get()))
|
||||||
|
throw internal_error("NULL object (clasp)");
|
||||||
}
|
}
|
||||||
|
|
||||||
template<lifetime L>
|
template<lifetime L>
|
||||||
|
@ -126,6 +168,8 @@ object<L>::object(const JSClass *const &clasp,
|
||||||
JS_NewObjectWithGivenProto(*cx, clasp, proto)
|
JS_NewObjectWithGivenProto(*cx, clasp, proto)
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
|
if(unlikely(!this->get()))
|
||||||
|
throw internal_error("NULL object (with given proto)");
|
||||||
}
|
}
|
||||||
|
|
||||||
template<lifetime L>
|
template<lifetime L>
|
||||||
|
@ -136,6 +180,8 @@ object<L>::object(const JSClass *const &clasp,
|
||||||
JS_NewObjectForConstructor(*cx, clasp, args)
|
JS_NewObjectForConstructor(*cx, clasp, args)
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
|
if(unlikely(!this->get()))
|
||||||
|
throw internal_error("NULL object (for constructor)");
|
||||||
}
|
}
|
||||||
|
|
||||||
template<lifetime L>
|
template<lifetime L>
|
||||||
|
@ -147,6 +193,28 @@ object<L>::object(const JSClass *const &clasp,
|
||||||
JS_New(*cx, ctor, args)
|
JS_New(*cx, ctor, args)
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
|
if(unlikely(!this->get()))
|
||||||
|
throw internal_error("NULL object (new)");
|
||||||
|
}
|
||||||
|
|
||||||
|
template<lifetime L>
|
||||||
|
void
|
||||||
|
object<L>::resize(const uint32_t &length)
|
||||||
|
{
|
||||||
|
if(!JS_SetArrayLength(*cx, &(*this), length))
|
||||||
|
throw internal_error("Failed to set array object length");
|
||||||
|
}
|
||||||
|
|
||||||
|
template<lifetime L>
|
||||||
|
uint32_t
|
||||||
|
object<L>::size()
|
||||||
|
const
|
||||||
|
{
|
||||||
|
uint32_t ret;
|
||||||
|
if(!JS_GetArrayLength(*cx, handle(*this), &ret))
|
||||||
|
throw internal_error("Failed to get array object length");
|
||||||
|
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<lifetime L>
|
template<lifetime L>
|
||||||
|
@ -156,6 +224,42 @@ const
|
||||||
return this->get()? JS::ObjectValue(*this->get()) : JS::NullValue();
|
return this->get()? JS::ObjectValue(*this->get()) : JS::NullValue();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<lifetime L>
|
||||||
|
bool
|
||||||
|
freeze(const object<L> & obj)
|
||||||
|
{
|
||||||
|
return JS_FreezeObject(*cx, obj);
|
||||||
|
}
|
||||||
|
|
||||||
|
template<lifetime L>
|
||||||
|
bool
|
||||||
|
deep_freeze(const object<L> & obj)
|
||||||
|
{
|
||||||
|
return JS_DeepFreezeObject(*cx, obj);
|
||||||
|
}
|
||||||
|
|
||||||
|
template<lifetime L>
|
||||||
|
bool
|
||||||
|
is_array(const object<L> & obj)
|
||||||
|
{
|
||||||
|
bool ret;
|
||||||
|
if(!JS_IsArrayObject(*cx, obj, &ret))
|
||||||
|
throw internal_error("Failed to query if object is array");
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<lifetime L>
|
||||||
|
bool
|
||||||
|
is_extensible(const object<L> & obj)
|
||||||
|
{
|
||||||
|
bool ret;
|
||||||
|
if(!JS_IsExtensible(*cx, obj, &ret))
|
||||||
|
throw internal_error("Failed to query object extensibility");
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
template<lifetime L>
|
template<lifetime L>
|
||||||
void
|
void
|
||||||
priv(object<L> &obj,
|
priv(object<L> &obj,
|
||||||
|
|
|
@ -70,6 +70,7 @@ struct value
|
||||||
|
|
||||||
template<lifetime L> JSType type(const value<L> &);
|
template<lifetime L> JSType type(const value<L> &);
|
||||||
template<lifetime L> bool undefined(const value<L> &);
|
template<lifetime L> bool undefined(const value<L> &);
|
||||||
|
template<lifetime L> bool is_array(typename value<L>::handle);
|
||||||
|
|
||||||
} // namespace basic
|
} // namespace basic
|
||||||
|
|
||||||
|
@ -303,10 +304,14 @@ const
|
||||||
}
|
}
|
||||||
|
|
||||||
template<lifetime L>
|
template<lifetime L>
|
||||||
JSType
|
bool
|
||||||
type(const value<L> &val)
|
is_array(const value<L> &val)
|
||||||
{
|
{
|
||||||
return JS_TypeOfValue(*cx, val);
|
bool ret;
|
||||||
|
if(!JS_IsArrayObject(*cx, val, &ret))
|
||||||
|
throw internal_error("Failed to query if value is array");
|
||||||
|
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<lifetime L>
|
template<lifetime L>
|
||||||
|
@ -316,6 +321,13 @@ undefined(const value<L> &val)
|
||||||
return type(val) == JSTYPE_VOID;
|
return type(val) == JSTYPE_VOID;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<lifetime L>
|
||||||
|
JSType
|
||||||
|
type(const value<L> &val)
|
||||||
|
{
|
||||||
|
return JS_TypeOfValue(*cx, val);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace basic
|
} // namespace basic
|
||||||
|
|
||||||
inline JS::Value
|
inline JS::Value
|
||||||
|
|
Loading…
Reference in a new issue