about summary refs log tree commit diff stats
diff options
context:
space:
mode:
-rw-r--r--src/bindings/quickjs.nim18
-rw-r--r--src/html/dom.nim13
-rw-r--r--src/html/event.nim225
-rw-r--r--src/js/javascript.nim544
-rw-r--r--src/js/regex.nim22
5 files changed, 617 insertions, 205 deletions
diff --git a/src/bindings/quickjs.nim b/src/bindings/quickjs.nim
index d415c17e..9db75a05 100644
--- a/src/bindings/quickjs.nim
+++ b/src/bindings/quickjs.nim
@@ -262,7 +262,8 @@ const
 const
   JS_PARSE_JSON_EXT* = (1 shl 0)
 
-template JS_CFUNC_DEF*(n: string, len: uint8, func1: JSCFunction): JSCFunctionListEntry = 
+template JS_CFUNC_DEF*(n: string, len: uint8, func1: JSCFunction):
+    JSCFunctionListEntry =
   JSCFunctionListEntry(name: cstring(n),
                        prop_flags: JS_PROP_WRITABLE or JS_PROP_CONFIGURABLE,
                        def_type: JS_DEF_CFUNC,
@@ -272,7 +273,8 @@ template JS_CFUNC_DEF*(n: string, len: uint8, func1: JSCFunction): JSCFunctionLi
                            cproto: JS_CFUNC_generic,
                            cfunc: JSCFunctionType(generic: func1))))
 
-template JS_CGETSET_DEF*(n: string, fgetter, fsetter: untyped): JSCFunctionListEntry =
+template JS_CGETSET_DEF*(n: string, fgetter: JSGetterFunction,
+    fsetter: JSSetterFunction): JSCFunctionListEntry =
   JSCFunctionListEntry(name: cstring(n),
                        prop_flags: JS_PROP_CONFIGURABLE,
                        def_type: JS_DEF_CGETSET,
@@ -281,7 +283,17 @@ template JS_CGETSET_DEF*(n: string, fgetter, fsetter: untyped): JSCFunctionListE
                            get: JSCFunctionType(getter: fgetter),
                            set: JSCFunctionType(setter: fsetter))))
 
-template JS_CGETSET_MAGIC_DEF*(n: string, fgetter, fsetter: untyped,
+template JS_CGETSET_DEF_NOCONF*(n: string, fgetter: JSGetterFunction,
+    fsetter: JSSetterFunction): JSCFunctionListEntry =
+  JSCFunctionListEntry(name: cstring(n),
+                       prop_flags: JS_PROP_ENUMERABLE,
+                       def_type: JS_DEF_CGETSET,
+                       u: JSCFunctionListEntryU(
+                         getset: JSCFunctionListEntryGetSet(
+                           get: JSCFunctionType(getter: fgetter),
+                           set: JSCFunctionType(setter: fsetter))))
+
+template JS_CGETSET_MAGIC_DEF*(n: string, fgetter, fsetter: typed,
     m: int16): JSCFunctionListEntry =
   JSCFunctionListEntry(name: cstring(n),
                        prop_flags: JS_PROP_CONFIGURABLE,
diff --git a/src/html/dom.nim b/src/html/dom.nim
index 8afb68e6..89fd7748 100644
--- a/src/html/dom.nim
+++ b/src/html/dom.nim
@@ -83,7 +83,7 @@ type
   Location = ref object
     window: Window
 
-  Window* = ref object
+  Window* = ref object of EventTarget
     attrs*: WindowAttributes
     console* {.jsget.}: console
     navigator* {.jsget.}: Navigator
@@ -1784,6 +1784,17 @@ func formmethod*(element: Element): FormMethod =
 
   return FORM_METHOD_GET
 
+# Forward declaration hack
+isDefaultPassive = func (eventTarget: EventTarget): bool =
+  if eventTarget of Window:
+    return true
+  if not (eventTarget of Node):
+    return false
+  let node = Node(eventTarget)
+  return EventTarget(node.document) == eventTarget or
+    EventTarget(node.document.html) == eventTarget or
+    EventTarget(node.document.body) == eventTarget
+
 proc parseColor(element: Element, s: string): RGBAColor =
   let cval = parseComponentValue(newStringStream(s))
   #TODO TODO TODO return element style
diff --git a/src/html/event.nim b/src/html/event.nim
index d862b40c..433e9597 100644
--- a/src/html/event.nim
+++ b/src/html/event.nim
@@ -1,4 +1,9 @@
+import math
+import times
+
+import bindings/quickjs
 import js/javascript
+import utils/opt
 
 type
   EventPhase = enum
@@ -7,25 +12,243 @@ type
     AT_TARGET = 2u16
     BUBBLING_PHASE = 3u16
 
-  Event* = ref object
+  EventFlag = enum
+    FLAG_STOP_PROPAGATION
+    FLAG_STOP_IMMEDIATE_PROPAGATION
+    FLAG_CANCELED
+    FLAG_IN_PASSIVE_LISTENER
+    FLAG_COMPOSED
+    FLAG_INITIALIZED
+    FLAG_DISPATCH
+
+  Event* = ref object of RootObj
     ctype {.jsget: "type".}: string
     target {.jsget.}: EventTarget
     currentTarget {.jsget.}: EventTarget
     eventPhase {.jsget.}: uint16
     bubbles {.jsget.}: bool
     cancelable {.jsget.}: bool
+    #TODO DOMHighResTimeStamp?
+    timeStamp {.jsget.}: float64
+    flags: set[EventFlag]
+    isTrusted {.jsufget.}: bool
+
+  CustomEvent* = ref object of Event
+    ctx: JSContext #TODO get rid of this
+    detail {.jsget.}: JSValue
 
   EventTarget* = ref object of RootObj
+    eventListeners*: seq[EventListener]
 
   EventHandler* = JSValue
 
+  EventListenerCallback = proc (event: Event): Err[JSError]
+
+  EventListener = ref object
+    ctype: string
+    callback: EventListenerCallback
+    capture: bool
+    passive: Opt[bool]
+    once: bool
+    #TODO AbortSignal
+    removed: bool
+
 jsDestructor(Event)
+jsDestructor(CustomEvent)
 jsDestructor(EventTarget)
 
+# Forward declaration hack
+var isDefaultPassive*: proc (eventTarget: EventTarget): bool
+
+# Event
+proc innerEventCreationSteps(event: Event, ctx: JSContext,
+    eventInitDict: JSValue) =
+  event.flags = {FLAG_INITIALIZED}
+  #TODO this is probably incorrect?
+  # I think it measures the time since the first fork. not sure though
+  event.timeStamp = round(cpuTime())
+  if not JS_IsUndefined(eventInitDict):
+    template set(name: static string, value: var bool) =
+      let prop = JS_GetPropertyStr(ctx, eventInitDict, name)
+      let jsVal = fromJS[bool](ctx, prop)
+      if jsVal.isSome:
+        value = jsVal.get
+    set "bubbles", event.bubbles
+    set "cancelable", event.cancelable
+    var composed: bool
+    set "composed", composed
+    if composed:
+      event.flags.incl(FLAG_COMPOSED)
+
+#TODO eventInitDict type
+proc newEvent(ctx: JSContext, ctype: string, eventInitDict = JS_UNDEFINED):
+    Result[Event, JSError] {.jsctor.} =
+  if not JS_IsUndefined(eventInitDict) and not JS_IsObject(eventInitDict):
+    return err(newTypeError("eventInitDict must be an object"))
+  let event = Event()
+  event.innerEventCreationSteps(ctx, eventInitDict)
+  event.ctype = ctype
+  return ok(event)
+
+proc initialize(this: Event, ctype: string, bubbles, cancelable: bool) =
+  this.flags.incl(FLAG_INITIALIZED)
+  this.isTrusted = false
+  this.target = nil
+  this.ctype = ctype
+  this.bubbles = bubbles
+  this.cancelable = cancelable
+
+proc initEvent(this: Event, ctype: string, bubbles, cancelable: bool)
+    {.jsfunc.} =
+  if FLAG_DISPATCH notin this.flags:
+    this.initialize(ctype, bubbles, cancelable)
+
 func srcElement(this: Event): EventTarget {.jsfget.} =
   return this.target
 
+#TODO shadow DOM etc.
+func composedPath(this: Event): seq[EventTarget] {.jsfunc.} =
+  if this.currentTarget == nil:
+    return @[]
+  return @[this.currentTarget]
+
+proc stopPropagation(this: Event) {.jsfunc.} =
+  this.flags.incl(FLAG_STOP_PROPAGATION)
+
+func cancelBubble(this: Event): bool {.jsfget.} =
+  return FLAG_STOP_PROPAGATION in this.flags
+
+proc cancelBubble(this: Event, cancel: bool) {.jsfset.} =
+  if cancel:
+    this.stopPropagation()
+
+proc stopImmediatePropagation(this: Event) {.jsfunc.} =
+  this.flags.incl({FLAG_STOP_PROPAGATION, FLAG_STOP_IMMEDIATE_PROPAGATION})
+
+proc setCanceledFlag(this: Event) =
+  if this.cancelable and FLAG_IN_PASSIVE_LISTENER notin this.flags:
+    this.flags.incl(FLAG_CANCELED)
+
+proc returnValue(this: Event): bool {.jsfget.} =
+  return FLAG_CANCELED notin this.flags
+
+proc returnValue(this: Event, value: bool) {.jsfset.} =
+  if not value:
+    this.setCanceledFlag()
+
+proc preventDefault(this: Event) {.jsfunc.} =
+  this.flags.incl(FLAG_CANCELED)
+
+func defaultPrevented(this: Event): bool {.jsfget.} =
+  return FLAG_CANCELED in this.flags
+
+func composed(this: Event): bool {.jsfget.} =
+  return FLAG_COMPOSED in this.flags
+
+# CustomEvent
+proc newCustomEvent(ctx: JSContext, ctype: string,
+    eventInitDict = JS_UNDEFINED): Result[CustomEvent, JSError] {.jsctor.} =
+  if not JS_IsUndefined(eventInitDict) and not JS_IsObject(eventInitDict):
+    return err(newTypeError("eventInitDict must be an object"))
+  let event = CustomEvent()
+  event.innerEventCreationSteps(ctx, eventInitDict)
+  event.detail = JS_GetPropertyStr(ctx, eventInitDict, "detail")
+  event.ctx = ctx
+  event.ctype = ctype
+  return ok(event)
+
+proc finalize(this: CustomEvent) {.jsfin.} =
+  JS_FreeValue(this.ctx, this.detail)
+
+proc initCustomEvent(ctx: JSContext, this: CustomEvent, ctype: string,
+    bubbles, cancelable: bool, detail: JSValue) {.jsfunc.} =
+  if FLAG_DISPATCH notin this.flags:
+    this.initialize(ctype, bubbles, cancelable)
+    this.ctx = ctx
+    this.detail = detail
+
+# EventTarget
+proc newEventTarget(): EventTarget {.jsctor.} =
+  return EventTarget()
+
+proc defaultPassiveValue(ctype: string, eventTarget: EventTarget): bool =
+  if ctype in ["touchstart", "touchmove", "wheel", "mousewheel"]:
+    return true
+  return eventTarget.isDefaultPassive()
+
+proc findEventListener(eventTarget: EventTarget, ctype: string,
+    callback: EventListenerCallback, capture: bool): int =
+  for i in 0 ..< eventTarget.eventListeners.len:
+    let it = eventTarget.eventListeners[i]
+    if it.ctype == ctype and it.callback == callback and it.capture == capture:
+      return i
+  return -1
+
+# shared
+proc addAnEventListener(eventTarget: EventTarget, listener: EventListener) =
+  #TODO signals
+  if listener.callback == nil:
+    return
+  if listener.passive.isNone:
+    listener.passive = opt(defaultPassiveValue(listener.ctype, eventTarget))
+  if eventTarget.findEventListener(listener.ctype, listener.callback,
+      listener.capture) == -1: # dedup
+    eventTarget.eventListeners.add(listener)
+  #TODO signals
+
+proc removeAnEventListener(eventTarget: EventTarget, i: int) =
+  eventTarget.eventListeners[i].removed = true
+  eventTarget.eventListeners.delete(i)
+
+proc flatten(ctx: JSContext, options: JSValue): bool =
+  if JS_IsBool(options):
+    return fromJS[bool](ctx, options).get(false)
+  if JS_IsObject(options):
+    let x = JS_GetPropertyStr(ctx, options, "capture")
+    return fromJS[bool](ctx, x).get(false)
+  return false
+
+proc flattenMore(ctx: JSContext, options: JSValue):
+    tuple[
+      capture: bool,
+      once: bool,
+      passive: Opt[bool]
+      #TODO signals
+    ] =
+  if JS_IsUndefined(options):
+    return
+  let capture = flatten(ctx, options)
+  var once = false
+  var passive: Opt[bool]
+  if JS_IsObject(options):
+    once = fromJS[bool](ctx, JS_GetPropertyStr(ctx, options, "once"))
+      .get(false)
+    let x = fromJS[bool](ctx, JS_GetPropertyStr(ctx, options, "passive"))
+    if x.isSome:
+      passive = opt(x.get)
+  return (capture, once, passive)
+
+proc addEventListener(ctx: JSContext, eventTarget: EventTarget,
+    callback: EventListenerCallback, options = JS_UNDEFINED) {.jsfunc.} =
+  let (capture, once, passive) = flattenMore(ctx, options)
+  let listener = EventListener(
+    capture: capture,
+    passive: passive,
+    once: once,
+    callback: callback
+  )
+  eventTarget.addAnEventListener(listener)
+
+proc removeEventListener(ctx: JSContext, eventTarget: EventTarget,
+    ctype: string, callback: EventListenerCallback,
+    options = JS_UNDEFINED) {.jsfunc.} =
+  let capture = flatten(ctx, options)
+  let i = eventTarget.findEventListener(ctype, callback, capture)
+  if i != -1:
+    eventTarget.removeAnEventListener(i)
+
 proc addEventModule*(ctx: JSContext) =
   let eventCID = ctx.registerType(Event)
+  ctx.registerType(CustomEvent, parent = eventCID)
   ctx.defineConsts(eventCID, EventPhase, uint16)
   ctx.registerType(EventTarget)
diff --git a/src/js/javascript.nim b/src/js/javascript.nim
index 0cc1fb83..3531040d 100644
--- a/src/js/javascript.nim
+++ b/src/js/javascript.nim
@@ -14,6 +14,12 @@
 #   following pragmas. As mentioned before, overloading doesn't work but OR
 #   generics do. Bare objects (returned by value) can't be passed either, for
 #   now. Otherwise, most types should work.
+# {.jsget.}, {.jsfget.} must be specified on object fields; these generate
+#   regular getter & setter functions.
+# {.jsufget.} For fields with the [LegacyUnforgeable] WebIDL property.
+#   This makes it so a non-configurable/writable, but enumerable property
+#   is defined on the object when the *constructor* is called (i.e. NOT on
+#   the prototype.)
 # {.jsfget.} and {.jsfset.} for getters/setters. Note the `f'; bare jsget/jsset
 #   can only be used on object fields. (I initially wanted to use the same
 #   keyword, unfortunately that didn't work out.)
@@ -91,6 +97,9 @@ type
     typemap: Table[pointer, JSClassID]
     ctors: Table[JSClassID, JSValue]
     parents: Table[JSClassID, JSClassID]
+    # Note: we assume no zero-list entry is added here.
+    unforgeable: Table[JSClassID, seq[JSCFunctionListEntry]]
+    funmap: Table[pointer, pointer]
     gclaz: string
     sym_refs: array[JSSymbolRefs, JSAtom]
     str_refs: array[JSStrRefs, JSAtom]
@@ -110,6 +119,21 @@ type
     e*: JSErrorEnum
     message*: string
 
+  BoundFunction = object
+    t: BoundFunctionType
+    name: string
+    id: NimNode
+    magic: uint16
+
+  BoundFunctionType = enum
+    FUNCTION = "js_func"
+    CONSTRUCTOR = "js_ctor"
+    GETTER = "js_get"
+    SETTER = "js_set"
+    PROPERTY_GET = "js_prop_get"
+    PROPERTY_HAS = "js_prop_has"
+    FINALIZER = "js_fin"
+
 const QuickJSErrors = [
   JS_EVAL_ERROR0,
   JS_RANGE_ERROR0,
@@ -121,6 +145,31 @@ const QuickJSErrors = [
   JS_AGGREGATE_ERROR0
 ]
 
+proc toJS*(ctx: JSContext, s: cstring): JSValue
+proc toJS*(ctx: JSContext, s: string): JSValue
+proc toJS(ctx: JSContext, r: Rune): JSValue
+proc toJS*(ctx: JSContext, n: int64): JSValue
+proc toJS*(ctx: JSContext, n: int32): JSValue
+proc toJS*(ctx: JSContext, n: int): JSValue
+proc toJS*(ctx: JSContext, n: uint16): JSValue
+proc toJS*(ctx: JSContext, n: uint32): JSValue
+proc toJS*(ctx: JSContext, n: uint64): JSValue
+proc toJS(ctx: JSContext, n: SomeFloat): JSValue
+proc toJS*(ctx: JSContext, b: bool): JSValue
+proc toJS[U, V](ctx: JSContext, t: Table[U, V]): JSValue
+proc toJS*(ctx: JSContext, opt: Option): JSValue
+proc toJS[T, E](ctx: JSContext, opt: Result[T, E]): JSValue
+proc toJS(ctx: JSContext, s: seq): JSValue
+proc toJS(ctx: JSContext, e: enum): JSValue
+proc toJS(ctx: JSContext, j: JSValue): JSValue
+proc toJS[T](ctx: JSContext, promise: Promise[T]): JSValue
+proc toJS[T, E](ctx: JSContext, promise: Promise[Result[T, E]]): JSValue
+proc toJS(ctx: JSContext, promise: EmptyPromise): JSValue
+proc toJSRefObj(ctx: JSContext, obj: ref object): JSValue
+proc toJS*(ctx: JSContext, obj: ref object): JSValue
+proc toJS*(ctx: JSContext, err: JSError): JSValue
+proc toJS*(ctx: JSContext, f: JSCFunction): JSValue
+
 func getOpaque*(ctx: JSContext): JSContextOpaque =
   return cast[JSContextOpaque](JS_GetContextOpaque(ctx))
 
@@ -311,7 +360,10 @@ func isInstanceOf*(ctx: JSContext, val: JSValue, class: static string): bool =
     if classid == tclassid:
       found = true
       break
-    classid = ctxOpaque.parents[classid]
+    ctxOpaque.parents.withValue(classid, val):
+      classid = val[]
+    do:
+      classid = 0 # not defined by Chawan; assume parent is Object.
     if classid == 0:
       break
   return found
@@ -323,27 +375,42 @@ proc setProperty*(ctx: JSContext, val: JSValue, name: string, prop: JSValue) =
 proc setProperty*(ctx: JSContext, val: JSValue, name: string, fun: JSCFunction, argc: int = 0) =
   ctx.setProperty(val, name, ctx.newJSCFunction(name, fun, argc))
 
-proc defineProperty*[T](ctx: JSContext, this: JSValue, name: string,
+proc defineProperty(ctx: JSContext, this: JSValue, name: string,
+    prop: JSValue, flags = cint(0)) =
+  if JS_DefinePropertyValueStr(ctx, this, cstring(name), prop, flags) <= 0:
+    raise newException(Defect, "Failed to define property string: " & name)
+
+proc defineProperty*[T](ctx: JSContext, this: JSValue, name: string, prop: T,
+    flags = cint(0)) =
+  defineProperty(ctx, this, name, toJS(ctx, prop), flags)
+
+proc definePropertyE*[T](ctx: JSContext, this: JSValue, name: string,
     prop: T) =
-  when T is JSValue:
-    if JS_DefinePropertyValueStr(ctx, this, cstring(name), prop, cint(0)) <= 0:
-      raise newException(Defect, "Failed to define property string: " & name)
-  else:
-    defineProperty(ctx, this, name, toJS(ctx, prop))
+  defineProperty(ctx, this, name, prop, JS_PROP_ENUMERABLE)
 
 proc definePropertyCWE*[T](ctx: JSContext, this: JSValue, name: string,
     prop: T) =
-  when T is JSValue:
-    if JS_DefinePropertyValueStr(ctx, this, cstring(name), prop,
-        JS_PROP_C_W_E) <= 0:
-      raise newException(Defect, "Failed to define property string: " & name)
+  defineProperty(ctx, this, name, prop, JS_PROP_C_W_E)
+
+# Get a unique pointer for each type.
+proc getTypePtr[T](x: T): pointer =
+  when T is RootRef:
+    # I'm so sorry.
+    # (This dereferences the object's first member, m_type. Probably.)
+    return cast[ptr pointer](x)[]
   else:
-    definePropertyCWE(ctx, this, name, toJS(ctx, prop))
+    return getTypeInfo(x)
+
+func getTypePtr(t: type): pointer =
+  var x: t
+  new(x)
+  return getTypePtr(x)
 
-func newJSClass*(ctx: JSContext, cdef: JSClassDefConst, tname: string,
-    ctor: JSCFunction, funcs: JSFunctionList, nimt: pointer, parent: JSClassID,
+func newJSClass(ctx: JSContext, cdef: JSClassDefConst, tname: string,
+    nimt: pointer, ctor: JSCFunction, funcs: JSFunctionList, parent: JSClassID,
     asglobal: bool, nointerface: bool, finalizer: proc(val: JSValue),
-    namespace: JSValue, errid: Opt[JSErrorEnum]): JSClassID {.discardable.} =
+    namespace: JSValue, errid: Opt[JSErrorEnum],
+    unforgeable: JSFunctionList): JSClassID {.discardable.} =
   let rt = JS_GetRuntime(ctx)
   discard JS_NewClassID(addr result)
   var ctxOpaque = ctx.getOpaque()
@@ -353,6 +420,8 @@ func newJSClass*(ctx: JSContext, cdef: JSClassDefConst, tname: string,
   ctxOpaque.typemap[nimt] = result
   ctxOpaque.creg[tname] = result
   ctxOpaque.parents[result] = parent
+  if unforgeable.len != 0:
+    ctxOpaque.unforgeable[result] = @unforgeable
   if finalizer != nil:
     rtOpaque.fins[result] = finalizer
   var proto: JSValue
@@ -691,59 +760,58 @@ proc fromJSTable[A, B](ctx: JSContext, val: JSValue):
     res[kn] = vn
   return ok(res)
 
-proc toJS*(ctx: JSContext, s: cstring): JSValue
-proc toJS*(ctx: JSContext, s: string): JSValue
-proc toJS(ctx: JSContext, r: Rune): JSValue
-proc toJS*(ctx: JSContext, n: int64): JSValue
-proc toJS*(ctx: JSContext, n: int32): JSValue
-proc toJS*(ctx: JSContext, n: int): JSValue
-proc toJS*(ctx: JSContext, n: uint16): JSValue
-proc toJS*(ctx: JSContext, n: uint32): JSValue
-proc toJS*(ctx: JSContext, n: uint64): JSValue
-proc toJS(ctx: JSContext, n: SomeFloat): JSValue
-proc toJS*(ctx: JSContext, b: bool): JSValue
-proc toJS[U, V](ctx: JSContext, t: Table[U, V]): JSValue
-proc toJS*(ctx: JSContext, opt: Option): JSValue
-proc toJS[T, E](ctx: JSContext, opt: Result[T, E]): JSValue
-proc toJS(ctx: JSContext, s: seq): JSValue
-proc toJS(ctx: JSContext, e: enum): JSValue
-proc toJS(ctx: JSContext, j: JSValue): JSValue
-proc toJS[T](ctx: JSContext, promise: Promise[T]): JSValue
-proc toJS[T, E](ctx: JSContext, promise: Promise[Result[T, E]]): JSValue
-proc toJS(ctx: JSContext, promise: EmptyPromise): JSValue
-proc toJSRefObj(ctx: JSContext, obj: ref object): JSValue
-proc toJS*(ctx: JSContext, obj: ref object): JSValue
-proc toJS*(ctx: JSContext, err: JSError): JSValue
-proc toJS*(ctx: JSContext, f: JSCFunction): JSValue
-
 #TODO varargs
 proc fromJSFunction1[T, U](ctx: JSContext, val: JSValue):
     proc(x: U): Result[T, JSError] =
   return proc(x: U): Result[T, JSError] =
     var arg1 = toJS(ctx, x)
     let ret = JS_Call(ctx, val, JS_UNDEFINED, 1, addr arg1)
-    return fromJS[T](ctx, ret)
+    when T isnot void:
+      result = fromJS[T](ctx, ret)
+    JS_FreeValue(ctx, ret)
+
+proc isErrType(rt: NimNode): bool =
+  let rtType = rt[0]
+  let errType = getTypeInst(Err)
+  return errType.sameType(rtType) and rtType.sameType(errType)
+
+# unpack brackets
+proc getRealTypeFun(x: NimNode): NimNode =
+  var x = x.getTypeImpl()
+  while true:
+    if x.kind == nnkBracketExpr and x.len == 2:
+      x = x[1].getTypeImpl()
+      continue
+    break
+  return x
 
 macro unpackReturnType(f: typed) =
-  var x = f.getTypeImpl()
-  while x.kind == nnkBracketExpr and x.len == 2:
-    x = x[1].getTypeImpl()
+  var x = f.getRealTypeFun()
   let params = x.findChild(it.kind == nnkFormalParams)
   let rv = params[0]
+  if rv.isErrType():
+    return quote do: void
   doAssert rv[0].strVal == "Result"
   let rvv = rv[1]
   result = quote do: `rvv`
 
 macro unpackArg0(f: typed) =
-  var x = f.getTypeImpl()
-  while x.kind == nnkBracketExpr and x.len == 2:
-    x = x[1].getTypeImpl()
+  var x = f.getRealTypeFun()
   let params = x.findChild(it.kind == nnkFormalParams)
   let rv = params[1]
-  assert rv.kind == nnkIdentDefs
+  doAssert rv.kind == nnkIdentDefs
   let rvv = rv[1]
   result = quote do: `rvv`
 
+proc fromJSFunction[T](ctx: JSContext, val: JSValue):
+    Result[T, JSError] =
+  #TODO all args...
+  return ok(
+    fromJSFunction1[
+      typeof(unpackReturnType(T)),
+      typeof(unpackArg0(T))
+    ](ctx, val))
+
 proc fromJSChar(ctx: JSContext, val: JSValue): Opt[char] =
   let s = ?toString(ctx, val)
   if s.len > 1:
@@ -825,8 +893,7 @@ proc fromJS*[T](ctx: JSContext, val: JSValue): Result[T, JSError] =
   elif T is Rune:
     return fromJSRune(ctx, val)
   elif T is (proc):
-    return ok(fromJSFunction1[typeof(unpackReturnType(T)),
-      typeof(unpackArg0(T))](ctx, val))
+    return fromJSFunction[T](ctx, val)
   elif T is Option:
     return fromJSOption[optionType(T)](ctx, val)
   elif T is Opt: # unwrap
@@ -854,7 +921,7 @@ proc fromJS*[T](ctx: JSContext, val: JSValue): Result[T, JSError] =
     return fromJSObject[T](ctx, val)
   else:
     static:
-      doAssert false
+      error("Unrecognized type " & $T)
 
 const JS_ATOM_TAG_INT = cuint(1u32 shl 31)
 
@@ -945,14 +1012,6 @@ proc toJS(ctx: JSContext, s: seq): JSValue =
         return JS_EXCEPTION
   return a
 
-proc getTypePtr[T](x: T): pointer =
-  when T is RootRef:
-    # I'm so sorry.
-    # (This dereferences the object's first member, m_type. Probably.)
-    return cast[ptr pointer](x)[]
-  else:
-    return getTypeInfo(x)
-
 proc toJSRefObj(ctx: JSContext, obj: ref object): JSValue =
   if obj == nil:
     return JS_NULL
@@ -1044,13 +1103,13 @@ proc defineConsts*[T](ctx: JSContext, classid: JSClassID,
     consts: static openarray[(string, T)]) =
   let proto = ctx.getOpaque().ctors[classid]
   for (k, v) in consts:
-    ctx.defineProperty(proto, k, v)
+    ctx.definePropertyE(proto, k, v)
 
 proc defineConsts*(ctx: JSContext, classid: JSClassID,
     consts: typedesc[enum], astype: typedesc) =
   let proto = ctx.getOpaque().ctors[classid]
   for e in consts:
-    ctx.defineProperty(proto, $e, astype(e))
+    ctx.definePropertyE(proto, $e, astype(e))
 
 type
   JSFuncGenerator = object
@@ -1078,21 +1137,6 @@ type
     j: int # js parameters accounted for (not including fix ones, e.g. `this')
     res: NimNode
 
-  BoundFunction = object
-    t: BoundFunctionType
-    name: string
-    id: NimNode
-    magic: uint16
-
-  BoundFunctionType = enum
-    FUNCTION = "js_func"
-    CONSTRUCTOR = "js_ctor"
-    GETTER = "js_get"
-    SETTER = "js_set"
-    PROPERTY_GET = "js_prop_get"
-    PROPERTY_HAS = "js_prop_has"
-    FINALIZER = "js_fin"
-
 var BoundFunctions {.compileTime.}: Table[string, seq[BoundFunction]]
 
 proc getGenerics(fun: NimNode): Table[string, seq[NimNode]] =
@@ -1443,13 +1487,26 @@ var js_funcs {.compileTime.}: Table[string, JSFuncGenerator]
 var existing_funcs {.compileTime.}: HashSet[string]
 var js_dtors {.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)
-  if typ notin BoundFunctions:
+proc newBoundFunction(t: BoundFunctionType, name: string, id: NimNode,
+    magic: uint16 = 0): BoundFunction =
+  return BoundFunction(
+    t: t,
+    name: name,
+    id: id,
+    magic: magic
+  )
+
+proc registerFunction(typ: string, nf: BoundFunction) =
+  BoundFunctions.withValue(typ, val):
+    val[].add(nf)
+  do:
     BoundFunctions[typ] = @[nf]
-  else:
-    BoundFunctions[typ].add(nf)
-  existing_funcs.incl(id.strVal)
+  existing_funcs.incl(nf.id.strVal)
+
+proc registerFunction(typ: string, t: BoundFunctionType, name: string,
+    id: NimNode, magic: uint16 = 0) =
+  let nf = newBoundFunction(t, name, id, magic)
+  registerFunction(typ, nf)
 
 proc registerConstructor(gen: JSFuncGenerator) =
   registerFunction(gen.thisType, gen.t, gen.funcName, gen.newName)
@@ -1458,8 +1515,6 @@ proc registerConstructor(gen: JSFuncGenerator) =
 proc registerFunction(gen: JSFuncGenerator) =
   registerFunction(gen.thisType, gen.t, gen.funcName, gen.newName)
 
-var js_errors {.compileTime.}: Table[string, seq[string]]
-
 export JS_ThrowTypeError, JS_ThrowRangeError, JS_ThrowSyntaxError,
        JS_ThrowInternalError, JS_ThrowReferenceError
 
@@ -1478,25 +1533,13 @@ proc newJSProcBody(gen: var JSFuncGenerator, isva: bool): NimNode =
     let tn = ident(gen.thisname.get)
     let ev = gen.errval
     result.add(quote do:
-      if not (JS_IsUndefined(`tn`) or ctx.isGlobal(`tt`)) and not isInstanceOf(ctx, `tn`, `tt`):
+      if not (JS_IsUndefined(`tn`) or ctx.isGlobal(`tt`)) and
+          not isInstanceOf(ctx, `tn`, `tt`):
         # undefined -> global.
-        discard JS_ThrowTypeError(ctx, "'%s' called on an object that is not an instance of %s", `fn`, `tt`)
+        discard JS_ThrowTypeError(ctx,
+          "'%s' called on an object that is not an instance of %s", `fn`, `tt`)
         return `ev`
     )
-
-  if gen.funcName in js_errors:
-    var tryWrap = newNimNode(nnkTryStmt)
-    tryWrap.add(gen.jsCallAndRet)
-    for error in js_errors[gen.funcName]:
-      let ename = ident(error)
-      var exceptBranch = newNimNode(nnkExceptBranch)
-      let eid = ident("e")
-      exceptBranch.add(newNimNode(nnkInfix).add(ident("as"), ename, eid))
-      let throwName = ident("JS_Throw" & error.substr("JS_".len))
-      exceptBranch.add(quote do:
-        return `throwName`(ctx, "%s", cstring(`eid`.msg)))
-      tryWrap.add(exceptBranch)
-    gen.jsCallAndRet = tryWrap
   result.add(gen.jsCallAndRet)
 
 proc newJSProc(gen: var JSFuncGenerator, params: openArray[NimNode], isva = true): NimNode =
@@ -1589,6 +1632,30 @@ proc makeJSCallAndRet(gen: var JSFuncGenerator, okstmt, errstmt: NimNode) =
         `okstmt`
       `errstmt`
 
+proc defineUnforgeable*(ctx: JSContext, this: JSValue) =
+  if not JS_IsException(this):
+    let ctxOpaque = ctx.getOpaque()
+    var classid = JS_GetClassID(this)
+    while true:
+      ctxOpaque.unforgeable.withValue(classid, uf):
+        JS_SetPropertyFunctionList(ctx, this, addr uf[][0], cint(uf[].len))
+      ctxOpaque.parents.withValue(classid, val):
+        classid = val[]
+      do:
+        classid = 0 # not defined by Chawan; assume parent is Object.
+      if classid == 0:
+        break
+
+proc makeCtorJSCallAndRet(gen: var JSFuncGenerator, okstmt, errstmt: NimNode) =
+  let jfcl = gen.jsFunCallList
+  let dl = gen.dielabel
+  gen.jsCallAndRet = quote do:
+    block `dl`:
+      let val = ctx.toJS(`jfcl`)
+      defineUnforgeable(ctx, val)
+      return val
+    `errstmt`
+
 macro jsctor*(fun: typed) =
   var gen = setupGenerator(fun, CONSTRUCTOR, thisname = none(string))
   if gen.newName.strVal in existing_funcs:
@@ -1600,7 +1667,7 @@ macro jsctor*(fun: typed) =
   let errstmt = quote do:
     return JS_ThrowTypeError(ctx, "Invalid parameters passed to constructor")
   # no okstmt
-  gen.makeJSCallAndRet(nil, errstmt)
+  gen.makeCtorJSCallAndRet(nil, errstmt)
   discard gen.newJSProc(getJSParams())
   gen.registerConstructor()
   result = newStmtList(fun)
@@ -1743,6 +1810,8 @@ template jsset*() {.pragma.}
 template jsset*(name: string) {.pragma.}
 template jsgetset*() {.pragma.}
 template jsgetset*(name: string) {.pragma.}
+template jsufget*() {.pragma.}
+template jsufget*(name: string) {.pragma.}
 
 proc js_illegal_ctor*(ctx: JSContext, this: JSValue, argc: cint, argv: ptr JSValue): JSValue {.cdecl.} =
   return JS_ThrowTypeError(ctx, "Illegal constructor")
@@ -1751,6 +1820,7 @@ type
   JSObjectPragma = object
     name: string
     varsym: NimNode
+    unforgeable: bool
 
   JSObjectPragmas = object
     jsget: seq[JSObjectPragma]
@@ -1799,13 +1869,16 @@ proc findPragmas(t: NimNode): JSObjectPragmas =
           var varPragmas = varNode[1]
           for varPragma in varPragmas:
             let pragmaName = getPragmaName(varPragma)
-            let op = JSObjectPragma(
+            var op = JSObjectPragma(
               name: getStringFromPragma(varPragma).get($varName),
               varsym: varName
             )
             case pragmaName
             of "jsget": result.jsget.add(op)
             of "jsset": result.jsset.add(op)
+            of "jsufget": # LegacyUnforgeable
+              op.unforgeable = true
+              result.jsget.add(op)
             of "jsgetset":
               result.jsget.add(op)
               result.jsset.add(op)
@@ -1838,55 +1911,107 @@ type
 template jsDestructor*[U](T: typedesc[ref U]) =
   static:
     js_dtors.incl($T)
-  proc `=destroy`(obj: var U) =
-    nim_finalize_for_js(addr obj)
-
-macro registerType*(ctx: typed, t: typed, parent: JSClassID = 0,
-    asglobal = false, nointerface = false, name: static string = "",
-    has_extra_getset: static bool = false,
-    extra_getset: static openarray[TabGetSet] = [],
-    namespace: JSValue = JS_NULL, errid = opt(JSErrorEnum)): JSClassID =
-  result = newStmtList()
-  let tname = t.strVal # the nim type's name.
-  if tname notin js_dtors:
-    warning("No destructor has been defined for type " & tname)
-  let name = if name == "": tname else: name # possibly a different name, e.g. Buffer for Container
-  var sctr = ident("js_illegal_ctor")
-  # constructor
-  var ctorFun: NimNode
-  var ctorImpl: NimNode
-  # custom finalizer
-  var finName = newNilLit()
-  var finFun = newNilLit()
-  # generic property getter (e.g. attribute["id"])
-  var propGetFun = newNilLit()
-  var propHasFun = newNilLit()
-  # property setters/getters declared on classes (with jsget, jsset)
-  var setters, getters: Table[string, NimNode]
-  let tabList = newNimNode(nnkBracket)
-  let pragmas = findPragmas(t)
-  for op in pragmas.jsget:
+  when NimMajor >= 2:
+    proc `=destroy`(obj: U) =
+      nim_finalize_for_js(addr obj)
+  else:
+    proc `=destroy`(obj: var U) =
+      nim_finalize_for_js(addr obj)
+
+type RegistryInfo = object
+  t: NimNode # NimNode of type
+  name: string # JS name, if this is the empty string then it equals tname
+  tabList: NimNode # array of function table
+  ctorImpl: NimNode # definition & body of constructor
+  ctorFun: NimNode # constructor ident
+  getset: Table[string, (NimNode, NimNode)] # name -> get, set
+  propGetFun: NimNode # custom get function ident
+  propHasFun: NimNode # custom has function ident
+  finFun: NimNode # finalizer ident
+  finName: NimNode # finalizer wrapper ident
+  dfin: NimNode # CheckDestroy finalizer ident
+  classDef: NimNode # ClassDef ident
+  tabUnforgeable: NimNode # array of unforgeable function table
+
+func tname(info: RegistryInfo): string =
+  return info.t.strVal
+
+# Differs from tname if the Nim object's name differs from the JS object's
+# name.
+func jsname(info: RegistryInfo): string =
+  if info.name != "":
+    return info.name
+  return info.tname
+
+proc newRegistryInfo(t: NimNode, name: string): RegistryInfo =
+  let info = RegistryInfo(
+    t: t,
+    name: name,
+    dfin: ident("js_" & t.strVal & "ClassCheckDestroy"),
+    classDef: ident("classDef"),
+    tabList: newNimNode(nnkBracket),
+    tabUnforgeable: newNimNode(nnkBracket),
+    finName: newNilLit(),
+    finFun: newNilLit(),
+    propGetFun: newNilLit(),
+    propHasFun: newNilLit()
+  )
+  if info.tname notin js_dtors:
+    warning("No destructor has been defined for type " & info.tname)
+  return info
+
+proc bindConstructor(stmts: NimNode, info: var RegistryInfo): NimNode =
+  if info.ctorFun != nil:
+    stmts.add(info.ctorImpl)
+    return info.ctorFun
+  return ident("js_illegal_ctor")
+
+proc registerGetters(stmts: NimNode, info: RegistryInfo,
+    jsget: seq[JSObjectPragma]) =
+  let t = info.t
+  let tname = info.tname
+  let jsname = info.jsname
+  for op in jsget:
     let node = op.varsym
     let fn = op.name
     let id = ident($GETTER & "_" & tname & "_" & fn)
-    result.add(quote do:
+    stmts.add(quote do:
       proc `id`(ctx: JSContext, this: JSValue): JSValue {.cdecl.} =
-        if not (JS_IsUndefined(this) or ctx.isGlobal(`tname`)) and not ctx.isInstanceOf(this, `tname`):
+        if not (JS_IsUndefined(this) or ctx.isGlobal(`tname`)) and
+            not ctx.isInstanceOf(this, `tname`):
           # undefined -> global.
-          return JS_ThrowTypeError(ctx, "'%s' called on an object that is not an instance of %s", `fn`, `name`)
+          return JS_ThrowTypeError(ctx,
+            "'%s' called on an object that is not an instance of %s", `fn`,
+            `jsname`)
         let arg_0 = fromJS_or_return(`t`, ctx, this)
         return toJS(ctx, arg_0.`node`)
     )
-    registerFunction(tname, GETTER, fn, id)
-  for op in pragmas.jsset:
+    let nf = newBoundFunction(GETTER, fn, id)
+    if op.unforgeable:
+      let f0 = nf.name
+      let f1 = nf.id
+      info.tabUnforgeable.add(quote do: JS_CGETSET_DEF_NOCONF(`f0`, `f1`, nil))
+    else:
+      registerFunction(tname, nf)
+
+proc registerSetters(stmts: NimNode, info: RegistryInfo,
+    jsset: seq[JSObjectPragma]) =
+  let t = info.t
+  let tname = info.tname
+  let jsname = info.jsname
+  for op in jsset:
     let node = op.varsym
     let fn = op.name
     let id = ident($SETTER & "_" & tname & "_" & fn)
-    result.add(quote do:
-      proc `id`(ctx: JSContext, this: JSValue, val: JSValue): JSValue {.cdecl.} =
-        if not (JS_IsUndefined(this) or ctx.isGlobal(`tname`)) and not ctx.isInstanceOf(this, `tname`):
+    stmts.add(quote do:
+      proc `id`(ctx: JSContext, this: JSValue, val: JSValue): JSValue
+          {.cdecl.} =
+        if not (JS_IsUndefined(this) or ctx.isGlobal(`tname`)) and
+            not ctx.isInstanceOf(this, `tname`):
           # undefined -> global.
-          return JS_ThrowTypeError(ctx, "'%s' called on an object that is not an instance of %s", `fn`, `name`)
+          return JS_ThrowTypeError(ctx,
+            "'%s' called on an object that is not an instance of %s", `fn`,
+            `jsname`)
         let arg_0 = fromJS_or_return(`t`, ctx, this)
         let arg_1 = val
         arg_0.`node` = fromJS_or_return(typeof(arg_0.`node`), ctx, arg_1)
@@ -1894,8 +2019,9 @@ macro registerType*(ctx: typed, t: typed, parent: JSClassID = 0,
     )
     registerFunction(tname, SETTER, fn, id)
 
-  if tname in BoundFunctions:
-    for fun in BoundFunctions[tname].mitems:
+proc bindFunctions(stmts: NimNode, info: var RegistryInfo) =
+  BoundFunctions.withValue(info.tname, funs):
+    for fun in funs[].mitems:
       var f0 = fun.name
       let f1 = fun.id
       if fun.name.endsWith("_exceptions"):
@@ -1903,60 +2029,61 @@ macro registerType*(ctx: typed, t: typed, parent: JSClassID = 0,
       case fun.t
       of FUNCTION:
         f0 = fun.name
-        tabList.add(quote do:
+        info.tabList.add(quote do:
           JS_CFUNC_DEF(`f0`, 0, cast[JSCFunction](`f1`)))
       of CONSTRUCTOR:
-        ctorImpl = js_funcs[$f0].res
-        if ctorFun != nil:
-          error("Class " & tname & " has 2+ constructors.")
-        ctorFun = f1
+        info.ctorImpl = js_funcs[$f0].res
+        if info.ctorFun != nil:
+          error("Class " & info.tname & " has 2+ constructors.")
+        info.ctorFun = f1
       of GETTER:
-        getters[f0] = f1
+        info.getset.withValue(f0, exv):
+          exv[0] = f1
+        do:
+          info.getset[f0] = (f1, newNilLit())
       of SETTER:
-        setters[f0] = f1
+        info.getset.withValue(f0, exv):
+          exv[1] = f1
+        do:
+          info.getset[f0] = (newNilLit(), f1)
       of PROPERTY_GET:
-        propGetFun = f1
+        info.propGetFun = f1
       of PROPERTY_HAS:
-        propHasFun = f1
+        info.propHasFun = f1
       of FINALIZER:
         f0 = fun.name
-        finFun = ident(f0)
-        finName = f1
-
-  for k, v in getters:
-    if k in setters:
-      let s = setters[k]
-      tabList.add(quote do: JS_CGETSET_DEF(`k`, `v`, `s`))
-    else:
-      tabList.add(quote do: JS_CGETSET_DEF(`k`, `v`, nil))
-  for k, v in setters:
-    if k notin getters:
-      tabList.add(quote do: JS_CGETSET_DEF(`k`, nil, `v`))
-
-  if has_extra_getset:
-    #HACK: for some reason, extra_getset gets weird contents when nothing is
-    # passed to it.
-    for x in extra_getset:
-      let k = x.name
-      let g = x.get
-      let s = x.set
-      let m = x.magic
-      tabList.add(quote do: JS_CGETSET_MAGIC_DEF(`k`, `g`, `s`, `m`))
-
-  if ctorFun != nil:
-    sctr = ctorFun
-    result.add(ctorImpl)
-
-  if finFun.kind != nnkNilLit:
-    result.add(quote do:
+        info.finFun = ident(f0)
+        info.finName = f1
+
+proc bindGetSet(stmts: NimNode, info: RegistryInfo) =
+  for k, (get, set) in info.getset:
+    info.tabList.add(quote do: JS_CGETSET_DEF(`k`, `get`, `set`))
+
+proc bindExtraGetSet(stmts: NimNode, info: var RegistryInfo,
+    extra_getset: openArray[TabGetSet]) =
+  for x in extra_getset:
+    let k = x.name
+    let g = x.get
+    let s = x.set
+    let m = x.magic
+    info.tabList.add(quote do: JS_CGETSET_MAGIC_DEF(`k`, `g`, `s`, `m`))
+
+proc bindFinalizer(stmts: NimNode, info: RegistryInfo) =
+  if info.finFun.kind != nnkNilLit:
+    let t = info.t
+    let finFun = info.finFun
+    let finName = info.finName
+    stmts.add(quote do:
       proc `finName`(val: JSValue) =
         let opaque = JS_GetOpaque(val, JS_GetClassID(val))
         if opaque != nil:
           `finFun`(cast[`t`](opaque))
     )
 
-  let dfin = ident("js_" & tname & "ClassCheckDestroy")
-  result.add(quote do:
+proc bindCheckDestroy(stmts: NimNode, info: RegistryInfo) =
+  let t = info.t
+  let dfin = info.dfin
+  stmts.add(quote do:
     proc `dfin`(rt: JSRuntime, val: JSValue): JS_BOOL {.cdecl.} =
       let opaque = JS_GetOpaque(val, JS_GetClassID(val))
       if opaque != nil:
@@ -1977,10 +2104,14 @@ macro registerType*(ctx: typed, t: typed, parent: JSClassID = 0,
       return true
   )
 
-  let endstmts = newStmtList()
-  let cdname = "classDef" & name
-  let classDef = ident("classDef")
-  if propGetFun.kind != nnkNilLit or propHasFun.kind != nnkNilLit:
+proc bindEndStmts(endstmts: NimNode, info: RegistryInfo) =
+  let jsname = info.jsname
+  let cdname = "classDef" & jsname
+  let dfin = info.dfin
+  let classDef = info.classDef
+  if info.propGetFun.kind != nnkNilLit or info.propHasFun.kind != nnkNilLit:
+    let propGetFun = info.propGetFun
+    let propHasFun = info.propHasFun
     endstmts.add(quote do:
       # No clue how to do this in pure nim.
       {.emit: ["""
@@ -1989,7 +2120,7 @@ static JSClassExoticMethods exotic = {
 	.has_property = """, `propHasFun`, """
 };
 static JSClassDef """, `cdname`, """ = {
-	""", "\"", `name`, "\"", """,
+	""", "\"", `jsname`, "\"", """,
         .can_destroy = """, `dfin`, """,
 	.exotic = &exotic
 };"""
@@ -2002,18 +2133,45 @@ static JSClassDef """, `cdname`, """ = {
   else:
     endstmts.add(quote do:
       const cd = JSClassDef(
-        class_name: `name`,
+        class_name: `jsname`,
         can_destroy: `dfin`
       )
       let `classDef` = JSClassDefConst(unsafeAddr cd))
 
-  endStmts.add(quote do:
-    var x: `t`
-    new(x)
-    `ctx`.newJSClass(`classDef`, `tname`, `sctr`, `tabList`, getTypePtr(x),
-      `parent`, `asglobal`, `nointerface`, `finName`, `namespace`, `errid`)
+macro registerType*(ctx: typed, t: typed, parent: JSClassID = 0,
+    asglobal = false, nointerface = false, name: static string = "",
+    has_extra_getset: static bool = false,
+    extra_getset: static openarray[TabGetSet] = [],
+    namespace: JSValue = JS_NULL, errid = opt(JSErrorEnum)): JSClassID =
+  var stmts = newStmtList()
+  var info = newRegistryInfo(t, name)
+  let pragmas = findPragmas(t)
+  stmts.registerGetters(info, pragmas.jsget)
+  stmts.registerSetters(info, pragmas.jsset)
+  stmts.bindFunctions(info)
+  stmts.bindGetSet(info)
+  if has_extra_getset:
+    #HACK: for some reason, extra_getset gets weird contents when nothing is
+    # passed to it. So we need an extra flag to signal if anything has
+    # been passed to it at all.
+    stmts.bindExtraGetSet(info, extra_getset)
+  let sctr = stmts.bindConstructor(info)
+  stmts.bindFinalizer(info)
+  stmts.bindCheckDestroy(info)
+  let endstmts = newStmtList()
+  endstmts.bindEndStmts(info)
+  let tabList = info.tabList
+  let finName = info.finName
+  let classDef = info.classDef
+  let tname = info.tname
+  let unforgeable = info.tabUnforgeable
+  endstmts.add(quote do:
+    `ctx`.newJSClass(`classDef`, `tname`, getTypePtr(`t`), `sctr`, `tabList`,
+      `parent`, `asglobal`, `nointerface`, `finName`, `namespace`, `errid`,
+      `unforgeable`)
   )
-  result.add(newBlockStmt(endstmts))
+  stmts.add(newBlockStmt(endstmts))
+  return stmts
 
 proc getMemoryUsage*(rt: JSRuntime): string =
   var m: JSMemoryUsage
diff --git a/src/js/regex.nim b/src/js/regex.nim
index eb6ef80f..d7371428 100644
--- a/src/js/regex.nim
+++ b/src/js/regex.nim
@@ -33,13 +33,21 @@ type
 var dummyRuntime = newJSRuntime()
 var dummyContext = dummyRuntime.newJSContextRaw()
 
-proc `=destroy`*(regex: var Regex) =
-  if regex.bytecode != nil:
-    if regex.clone:
-      dealloc(regex.bytecode)
-    else:
-      dummyRuntime.js_free_rt(regex.bytecode)
-    regex.bytecode = nil
+when NimMajor >= 2:
+  proc `=destroy`*(regex: Regex) =
+    if regex.bytecode != nil:
+      if regex.clone:
+        dealloc(regex.bytecode)
+      else:
+        dummyRuntime.js_free_rt(regex.bytecode)
+else:
+  proc `=destroy`*(regex: var Regex) =
+    if regex.bytecode != nil:
+      if regex.clone:
+        dealloc(regex.bytecode)
+      else:
+        dummyRuntime.js_free_rt(regex.bytecode)
+      regex.bytecode = nil
 
 proc `=copy`*(dest: var Regex, source: Regex) =
   if dest.bytecode != source.bytecode: