about summary refs log tree commit diff stats
path: root/src/js/javascript.nim
diff options
context:
space:
mode:
Diffstat (limited to 'src/js/javascript.nim')
-rw-r--r--src/js/javascript.nim170
1 files changed, 95 insertions, 75 deletions
diff --git a/src/js/javascript.nim b/src/js/javascript.nim
index 49d0b311..c0c55237 100644
--- a/src/js/javascript.nim
+++ b/src/js/javascript.nim
@@ -21,6 +21,7 @@
 #   nothing. The key should probably be either a string or an integer.
 import macros
 import options
+import sets
 import streams
 import strformat
 import strutils
@@ -61,6 +62,7 @@ type
     done: JSAtom
     next: JSAtom
     value: JSAtom
+    Array_prototype_values: JSValue
 
   JSRuntimeOpaque* = ref object
     plist: Table[pointer, pointer]
@@ -88,7 +90,7 @@ proc newJSContext*(rt: JSRuntime): JSContext =
   var opaque = new(JSContextOpaque)
   GC_ref(opaque)
 
-  block:
+  block: # get well-known symbols and other functions
     let global = JS_GetGlobalObject(ctx)
     block:
       let sym = JS_GetPropertyStr(ctx, global, "Symbol")
@@ -116,6 +118,11 @@ proc newJSContext*(rt: JSRuntime): JSContext =
       block:
         let s = "next"
         opaque.next = JS_NewAtomLen(ctx, cstring(s), csize_t(s.len))
+      block:
+        # 2 - JS_CLASS_ARRAY
+        let arrproto = JS_GetClassProto(ctx, 2)
+        opaque.Array_prototype_values = JS_GetPropertyStr(ctx, arrproto, "values")
+        JS_FreeValue(ctx, arrproto)
       JS_FreeValue(ctx, sym)
     JS_FreeValue(ctx, global)
 
@@ -149,6 +156,7 @@ proc free*(ctx: var JSContext) =
     JS_FreeAtom(ctx, opaque.sym_toStringTag)
     JS_FreeAtom(ctx, opaque.done)
     JS_FreeAtom(ctx, opaque.next)
+    JS_FreeValue(ctx, opaque.Array_prototype_values)
     GC_unref(opaque)
   JS_FreeContext(ctx)
   ctx = nil
@@ -260,6 +268,10 @@ func newJSClass*(ctx: JSContext, cdef: JSClassDefConst, ctor: JSCFunction, funcs
     #TODO maybe put them in ctxOpaque instead?
     rtOpaque.flist.add(@funcs)
     JS_SetPropertyFunctionList(ctx, proto, addr rtOpaque.flist[^1][0], cint(funcs.len))
+  #TODO check if this is an indexed property getter
+  if cdef.exotic != nil and cdef.exotic.get_own_property != nil:
+    let val = JS_DupValue(ctx, ctxOpaque.Array_prototype_values)
+    assert JS_SetProperty(ctx, proto, ctxOpaque.sym_iterator, val) == 1
   assert JS_SetProperty(ctx, proto, ctxOpaque.sym_toStringTag, JS_NewString(ctx, cdef.class_name)) == 1
   JS_SetClassProto(ctx, result, proto)
   if asglobal:
@@ -269,11 +281,11 @@ func newJSClass*(ctx: JSContext, cdef: JSClassDefConst, ctor: JSCFunction, funcs
     if JS_SetPrototype(ctx, global, proto) != 1:
       raise newException(Defect, "Failed to set global prototype: " & $cdef.class_name)
     JS_FreeValue(ctx, global)
+  let jctor = ctx.newJSCFunction($cdef.class_name, ctor, 0, JS_CFUNC_constructor)
+  JS_SetConstructor(ctx, jctor, proto)
+  ctxOpaque.ctors[result] = JS_DupValue(ctx, jctor)
   if not nointerface:
     let global = JS_GetGlobalObject(ctx)
-    let jctor = ctx.newJSCFunction($cdef.class_name, ctor, 0, JS_CFUNC_constructor)
-    JS_SetConstructor(ctx, jctor, proto)
-    ctxOpaque.ctors[result] = JS_DupValue(ctx, jctor)
     ctx.setProperty(global, $cdef.class_name, jctor)
     JS_FreeValue(ctx, global)
 
@@ -387,7 +399,7 @@ macro fromJSTupleBody(a: tuple) =
         if `done`.isnone: # exception
           return none(T)
         var i = `i`
-        # we're simulating a sequence, so we must query all remaining parameters too:
+        # we're emulating a sequence, so we must query all remaining parameters too:
         while not `done`.get:
           inc i
           let next = JS_Call(ctx, next_method, it, 0, nil)
@@ -600,12 +612,14 @@ proc fromJS[T](ctx: JSContext, val: JSValue): Option[T] =
   elif typeof(result.get) is Table:
     return fromJSTable[typeof(result.get.keys), typeof(result.get.values)](ctx, val)
   elif T is SomeInteger:
-    return fromJSInt[T](ctx, val)
+    if JS_IsNumber(val):
+      return fromJSInt[T](ctx, val)
   elif T is SomeFloat:
-    let f64: float64
-    if JS_ToFloat64(ctx, addr f64, val) < 0:
-      return none(T)
-    return some(cast[T](f64))
+    if JS_IsNumber(val):
+      let f64: float64
+      if JS_ToFloat64(ctx, addr f64, val) < 0:
+        return none(T)
+      return some(cast[T](f64))
   elif T is enum:
     #TODO implement enum handling...
     if JS_IsException(val):
@@ -633,9 +647,18 @@ proc fromJS[T](ctx: JSContext, val: JSValue): Option[T] =
       return none(T)
     return some(op)
 
+const JS_ATOM_TAG_INT = cuint(1u32 shl 31)
+
+func JS_IsNumber(v: JSAtom): JS_BOOL =
+  return (cast[cuint](v) and JS_ATOM_TAG_INT) != 0
+
 func fromJS[T: string|uint32](ctx: JSContext, atom: JSAtom): Option[T] =
-  let val = JS_AtomToValue(ctx, atom)
-  return fromJS[T](ctx, val)
+  when T is SomeNumber:
+    if JS_IsNumber(atom):
+      return some(T(cast[uint32](atom) and (not JS_ATOM_TAG_INT)))
+  else:
+    let val = JS_AtomToValue(ctx, atom)
+    return fromJS[T](ctx, val)
 
 proc getJSFunction*[T, U](ctx: JSContext, val: JSValue): Option[(proc(x: T): Option[U])] =
   return fromJS[(proc(x: T): Option[U])](ctx, val)
@@ -900,13 +923,10 @@ template fromJS_or_return*(t, ctx, val: untyped): untyped =
   )
 
 template fromJS_or_die*(t, ctx, val, ev, dl: untyped): untyped =
-  let valval = when typeof(val) is JSAtom:
-    JS_AtomToValue(ctx, val)
-  else:
-    val
-  if JS_IsException(valval):
-    return ev
-  let x = fromJS[t](ctx, valval)
+  when not (typeof(val) is JSAtom):
+    if JS_IsException(val):
+      return ev
+  let x = fromJS[t](ctx, val)
   if x.isnone:
     break dl
   x.get
@@ -941,11 +961,6 @@ proc addUnionParamBranch(gen: var JSFuncGenerator, query, newBranch: NimNode, fa
     gen.jsFunCallLists[i] = oldBranch
   gen.newBranchList.add(newBranch)
 
-const JS_ATOM_TAG_INT = cuint(1u32 shl 31)
-
-proc JS_IsNumber(v: JSAtom): JS_BOOL =
-  return (cast[cuint](v) and JS_ATOM_TAG_INT) != 0
-
 proc addUnionParam0(gen: var JSFuncGenerator, tt: NimNode, s: NimNode, val: NimNode, fallback: NimNode = nil) =
   # Union types.
   #TODO quite a few types are still missing.
@@ -1111,6 +1126,7 @@ proc finishFunCallList(gen: var JSFuncGenerator) =
     branch.add(gen.jsFunCall)
 
 var js_funcs {.compileTime.}: Table[string, JSFuncGenerator]
+var existing_funcs {.compileTime.}: HashSet[string]
 
 proc registerFunction(typ: string, t: BoundFunctionType, name: string, id: NimNode, magic: uint16 = 0) =
   let nf = BoundFunction(t: t, name: name, id: id, magic: magic)
@@ -1118,11 +1134,15 @@ proc registerFunction(typ: string, t: BoundFunctionType, name: string, id: NimNo
     BoundFunctions[typ] = @[nf]
   else:
     BoundFunctions[typ].add(nf)
+  existing_funcs.incl(id.strVal)
 
-proc registerFunction(gen: JSFuncGenerator) =
+proc registerConstructor(gen: JSFuncGenerator) =
   registerFunction(gen.thisType, gen.t, gen.funcName, ident(gen.newName))
   js_funcs[gen.funcName] = gen
 
+proc registerFunction(gen: JSFuncGenerator) =
+  registerFunction(gen.thisType, gen.t, gen.funcName, ident(gen.newName))
+
 var js_errors {.compileTime.}: Table[string, seq[string]]
 
 export JS_ThrowTypeError, JS_ThrowRangeError, JS_ThrowSyntaxError,
@@ -1258,73 +1278,73 @@ macro jserr*(fun: untyped) =
 
 macro jsctor*(fun: typed) =
   var gen = setupGenerator(fun, CONSTRUCTOR, thisname = none(string))
-  if gen.funcName in js_funcs:
+  if gen.newName in existing_funcs:
     #TODO TODO TODO implement function overloading
     error("Function overloading hasn't been implemented yet...")
-  else:
-    gen.addRequiredParams()
-    gen.addOptionalParams()
-    gen.finishFunCallList()
-    let jfcl = gen.jsFunCallList
-    let dl = gen.dielabel
-    gen.jsCallAndRet = quote do:
-      block `dl`:
-        return ctx.toJS(`jfcl`)
-      return JS_UNDEFINED
-    discard gen.newJSProc(getJSParams())
-    gen.registerFunction()
-    result = newStmtList(fun)
+  gen.addRequiredParams()
+  gen.addOptionalParams()
+  gen.finishFunCallList()
+  let jfcl = gen.jsFunCallList
+  let dl = gen.dielabel
+  gen.jsCallAndRet = quote do:
+    block `dl`:
+      return ctx.toJS(`jfcl`)
+    return JS_UNDEFINED
+  discard gen.newJSProc(getJSParams())
+  gen.registerConstructor()
+  result = newStmtList(fun)
 
 macro jshasprop*(fun: typed) =
   var gen = setupGenerator(fun, PROPERTY_HAS, thisname = some("obj"))
-  if gen.funcName in js_funcs:
+  if gen.newName in existing_funcs:
     #TODO TODO TODO ditto
     error("Function overloading hasn't been implemented yet...")
-  else:
-    gen.addFixParam("obj")
-    gen.addFixParam("atom")
-    gen.finishFunCallList()
-    let jfcl = gen.jsFunCallList
-    let dl = gen.dielabel
-    gen.jsCallAndRet = quote do:
-      block `dl`:
-        let retv = `jfcl`
-        return cint(retv)
-    let jsProc = gen.newJSProc(getJSHasPropParams(), false)
-    gen.registerFunction()
-    result = newStmtList(fun, jsProc)
+  gen.addFixParam("obj")
+  gen.addFixParam("atom")
+  gen.finishFunCallList()
+  let jfcl = gen.jsFunCallList
+  let dl = gen.dielabel
+  gen.jsCallAndRet = quote do:
+    block `dl`:
+      let retv = `jfcl`
+      return cint(retv)
+  let jsProc = gen.newJSProc(getJSHasPropParams(), false)
+  gen.registerFunction()
+  result = newStmtList(fun, jsProc)
 
 macro jsgetprop*(fun: typed) =
   var gen = setupGenerator(fun, PROPERTY_GET, thisname = some("obj"))
-  if gen.funcName in js_funcs:
+  if gen.newName in existing_funcs:
     #TODO TODO TODO ditto
     error("Function overloading hasn't been implemented yet...")
-  else:
-    gen.addFixParam("obj")
-    gen.addFixParam("prop")
-    gen.finishFunCallList()
-    let jfcl = gen.jsFunCallList
-    let dl = gen.dielabel
-    gen.jsCallAndRet = quote do:
-      block `dl`:
-        let retv = ctx.toJS(`jfcl`)
-        if retv != JS_NULL:
-          desc[].setter = JS_UNDEFINED
-          desc[].getter = JS_UNDEFINED
-          desc[].value = retv
-          desc[].flags = 0
-          return cint(1)
-      return cint(0)
-    let jsProc = gen.newJSProc(getJSGetPropParams(), false)
-    gen.registerFunction()
-    result = newStmtList(fun, jsProc)
+  gen.addFixParam("obj")
+  gen.addFixParam("prop")
+  gen.finishFunCallList()
+  let jfcl = gen.jsFunCallList
+  let dl = gen.dielabel
+  gen.jsCallAndRet = quote do:
+    block `dl`:
+      let retv = ctx.toJS(`jfcl`)
+      if retv != JS_NULL:
+        desc[].setter = JS_UNDEFINED
+        desc[].getter = JS_UNDEFINED
+        desc[].value = retv
+        desc[].flags = 0
+        return cint(1)
+    return cint(0)
+  let jsProc = gen.newJSProc(getJSGetPropParams(), false)
+  gen.registerFunction()
+  result = newStmtList(fun, jsProc)
 
 macro jsfget*(fun: typed) =
   var gen = setupGenerator(fun, GETTER)
   if gen.minArgs != 1 or gen.funcParams.len != gen.minArgs:
-    error("jsfget functions must one parameters")
+    error("jsfget functions must only have one parameter.")
   if gen.returnType.isnone:
-    error("jsfget functions must have a return type")
+    error("jsfget functions must have a return type.")
+  if gen.newName in existing_funcs:
+    #TODO TODO TODO ditto
+    error("Function overloading hasn't been implemented yet...")
   gen.addFixParam("this")
   gen.finishFunCallList()
   let jfcl = gen.jsFunCallList