about summary refs log tree commit diff stats
diff options
context:
space:
mode:
authorbptato <nincsnevem662@gmail.com>2023-01-02 21:19:46 +0100
committerbptato <nincsnevem662@gmail.com>2023-01-02 21:19:46 +0100
commit62983cdc609aacc47ade0a8a4a4c6b85dd26199a (patch)
treeab0518ac0f556a499107887fb524beddb80ba104
parentf2bf1572456412f109d06c6a49e34cfbe924bbfa (diff)
downloadchawan-62983cdc609aacc47ade0a8a4a4c6b85dd26199a.tar.gz
dom: add better attribute reflection
Instead of creating a new function for each attribute, use a single
magic function for reflected attributes.
-rw-r--r--src/buffer/buffer.nim5
-rw-r--r--src/css/cascade.nim6
-rw-r--r--src/html/dom.nim326
-rw-r--r--src/html/env.nim8
-rw-r--r--src/html/tags.nim5
-rw-r--r--src/js/javascript.nim38
6 files changed, 267 insertions, 121 deletions
diff --git a/src/buffer/buffer.nim b/src/buffer/buffer.nim
index f7ad6537..34674062 100644
--- a/src/buffer/buffer.nim
+++ b/src/buffer/buffer.nim
@@ -512,8 +512,9 @@ proc updateHover*(buffer: Buffer, cursorx, cursory: int): UpdateHoverResult {.pr
   buffer.prevnode = thisnode
 
 proc loadResource(buffer: Buffer, document: Document, elem: HTMLLinkElement) =
-  if elem.href == "": return
-  let url = parseURL(elem.href, document.url.some)
+  let href = elem.attr("href")
+  if href == "": return
+  let url = parseURL(href, document.url.some)
   if url.isSome:
     let url = url.get
     if url.scheme == buffer.url.scheme:
diff --git a/src/css/cascade.nim b/src/css/cascade.nim
index acf2ddf4..68b9ebee 100644
--- a/src/css/cascade.nim
+++ b/src/css/cascade.nim
@@ -160,8 +160,10 @@ func calcPresentationalHints(element: Element): CSSComputedValues =
     map_text
   of TAG_TEXTAREA:
     let textarea = HTMLTextAreaElement(element)
-    set_cv "width", CSSLength(unit: UNIT_CH, num: float64(textarea.cols))
-    set_cv "height", CSSLength(unit: UNIT_EM, num: float64(textarea.rows))
+    let cols = textarea.attri("cols").get(20)
+    let rows = textarea.attri("rows").get(1)
+    set_cv "width", CSSLength(unit: UNIT_CH, num: float64(cols))
+    set_cv "height", CSSLength(unit: UNIT_EM, num: float64(rows))
   of TAG_FONT:
     map_color
   else: discard
diff --git a/src/html/dom.nim b/src/html/dom.nim
index 1a831a5e..6f98d9d4 100644
--- a/src/html/dom.nim
+++ b/src/html/dom.nim
@@ -221,8 +221,6 @@ type
   HTMLInputElement* = ref object of FormAssociatedElement
     form* {.jsget.}: HTMLFormElement
     inputType*: InputType
-    autofocus*: bool
-    required*: bool
     value* {.jsget.}: string
     checked*: bool
     xcoord*: int
@@ -267,7 +265,6 @@ type
     name*: string
     smethod*: string
     enctype*: string
-    target*: string
     novalidate*: bool
     constructingentrylist*: bool
     controls*: seq[FormAssociatedElement]
@@ -305,6 +302,71 @@ type
 
   HTMLLabelElement* = ref object of HTMLElement
 
+# Reflected attributes.
+type
+  ReflectType = enum
+    REFLECT_STR, REFLECT_BOOL, REFLECT_INT, REFLECT_INT_GREATER_ZERO,
+    REFLECT_INT_GREATER_EQUAL_ZERO
+
+  ReflectEntry = tuple[
+    attrname: string,
+    funcname: string,
+    t: ReflectType,
+    tags: set[TagType],
+    i: int
+  ]
+
+template toset(ts: openarray[TagType]): set[TagType] =
+  var tags: set[TagType]
+  for tag in ts:
+    tags.incl(tag)
+  tags
+
+template makes(name: string, ts: set[TagType]): ReflectEntry =
+  (name, name, REFLECT_STR, ts, 0)
+
+template makes(attrname: string, funcname: string, ts: set[TagType]): ReflectEntry =
+  (attrname, funcname, REFLECT_STR, ts, 0)
+
+template makes(name: string, ts: varargs[TagType]): ReflectEntry =
+  makes(name, toset(ts))
+
+template makes(attrname: string, funcname: string, ts: varargs[TagType]): ReflectEntry =
+  makes(attrname, funcname, toset(ts))
+
+template makeb(name: string, ts: varargs[TagType]): ReflectEntry =
+  (name, name, REFLECT_BOOL, toset(ts), 0)
+
+template makei(name: string, ts: varargs[TagType], default = 0): ReflectEntry =
+  (name, name, REFLECT_INT, toset(ts), default)
+
+template makeigz(name: string, ts: varargs[TagType], default = 0): ReflectEntry =
+  (name, name, REFLECT_INT_GREATER_ZERO, toset(ts), default)
+
+template makeigez(name: string, ts: varargs[TagType], default = 0): ReflectEntry =
+  (name, name, REFLECT_INT_GREATER_EQUAL_ZERO, toset(ts), default)
+
+const ReflectTable0 = [
+  # non-global attributes
+  makes("target", TAG_A, TAG_AREA, TAG_LABEL, TAG_LINK),
+  makes("href", TAG_LINK),
+  makeb("required", TAG_INPUT, TAG_SELECT, TAG_TEXTAREA),
+  makes("rel", "relList", TAG_A, TAG_LINK, TAG_LABEL),
+  makes("for", "htmlFor", TAG_LABEL),
+  makeigz("cols", TAG_TEXTAREA, 20),
+  makeigz("rows", TAG_TEXTAREA, 1),
+# <SELECT>:
+#> For historical reasons, the default value of the size IDL attribute does
+#> not return the actual size used, which, in the absence of the size content
+#> attribute, is either 1 or 4 depending on the presence of the multiple
+#> attribute.
+  makeigz("size", TAG_SELECT, 0),
+  makeigz("size", TAG_INPUT, 20),
+  # "super-global" attributes
+  makes("slot", AllTagTypes),
+  makes("class", "className", AllTagTypes)
+]
+
 # Forward declarations
 func attrb*(element: Element, s: string): bool
 proc attr*(element: Element, name, value: string)
@@ -990,25 +1052,6 @@ func attrb*(element: Element, s: string): bool =
   return false
 
 # Element attribute reflection (getters)
-func className(element: Element): string {.jsfget.} =
-  element.attr("class")
-
-func size*(element: HTMLInputElement): int {.jsfget.} =
-  element.attrigz("size").get(20)
-
-#> For historical reasons, the default value of the size IDL attribute does
-#> not return the actual size used, which, in the absence of the size content
-#> attribute, is either 1 or 4 depending on the presence of the multiple
-#> attribute.
-func size*(element: HTMLSelectElement): int {.jsfget.} =
-  element.attrigz("size").get(0)
-
-func cols*(element: HTMLTextAreaElement): int {.jsfget.} =
-  element.attrigz("cols").get(20)
-
-func rows*(element: HTMLTextAreaElement): int {.jsfget.} =
-  element.attrigz("rows").get(1)
-
 func innerHTML*(element: Element): string {.jsfget.} =
   for child in element.childList:
     result &= $child
@@ -1047,9 +1090,9 @@ func inputString*(input: HTMLInputElement): string =
     if input.checked: "*"
     else: " "
   of INPUT_SEARCH, INPUT_TEXT:
-    input.value.padToWidth(input.size)
+    input.value.padToWidth(input.attri("size").get(20))
   of INPUT_PASSWORD:
-    '*'.repeat(input.value.len).padToWidth(input.size)
+    '*'.repeat(input.value.len).padToWidth(input.attri("size").get(20))
   of INPUT_RESET:
     if input.value != "": input.value
     else: "RESET"
@@ -1057,18 +1100,22 @@ func inputString*(input: HTMLInputElement): string =
     if input.value != "": input.value
     else: "SUBMIT"
   of INPUT_FILE:
-    if input.file.isnone: "".padToWidth(input.size)
-    else: input.file.get.path.serialize_unicode().padToWidth(input.size)
+    if input.file.isnone:
+      "".padToWidth(input.attri("size").get(20))
+    else:
+      input.file.get.path.serialize_unicode().padToWidth(input.attri("size").get(20))
   else: input.value
 
 func textAreaString*(textarea: HTMLTextAreaElement): string =
   let split = textarea.value.split('\n')
-  for i in 0 ..< textarea.rows:
-    if textarea.cols > 2:
+  let rows = textarea.attri("rows").get(1)
+  for i in 0 ..< rows:
+    let cols = textarea.attri("cols").get(20)
+    if cols > 2:
       if i < split.len:
-        result &= '[' & split[i].padToWidth(textarea.cols - 2) & "]\n"
+        result &= '[' & split[i].padToWidth(cols - 2) & "]\n"
       else:
-        result &= '[' & ' '.repeat(textarea.cols - 2) & "]\n"
+        result &= '[' & ' '.repeat(cols - 2) & "]\n"
     else:
       result &= "[]\n"
 
@@ -1160,12 +1207,6 @@ func href0[T: HTMLAnchorElement|HTMLAreaElement](element: T): string =
       return $url.get
 
 # <base>
-func target(base: HTMLBaseElement): string {.jsfget.} =
-  base.attr("target")
-
-proc target(base: HTMLBaseElement, target: string) {.jsfset.} =
-  base.attr("target", target)
-
 func href(base: HTMLBaseElement): string {.jsfget.} =
   if base.attrb("href"):
     #TODO with fallback base url
@@ -1174,12 +1215,6 @@ func href(base: HTMLBaseElement): string {.jsfget.} =
       return $url.get
 
 # <a>
-func target(anchor: HTMLAnchorElement): string {.jsfget.} =
-  anchor.attr("target")
-
-proc target(anchor: HTMLAnchorElement, target: string) {.jsfset.} =
-  anchor.attr("target", target)
-
 func href*(anchor: HTMLAnchorElement): string {.jsfget.} =
   anchor.href0
 
@@ -1199,28 +1234,9 @@ proc href(area: HTMLAreaElement, href: string) {.jsfset.} =
 func `$`(area: HTMLAreaElement): string {.jsfunc.} =
   area.href
 
-# <link>
-func href*(link: HTMLLinkElement): string {.jsfget.} =
-  link.attr("href")
-
-proc href*(link: HTMLLinkElement, href: string) {.jsfset.} =
-  link.attr("href", href)
-
-func target(link: HTMLLinkElement): string {.jsfget.} =
-  link.attr("target")
-
-proc target(link: HTMLLinkElement, target: string) {.jsfset.} =
-  link.attr("target", target)
-
 # <label>
-func htmlFor(label: HTMLLabelElement): string {.jsfget.} =
-  label.attr("for")
-
-proc htmlFor(label: HTMLLabelElement, htmlFor: string) {.jsfset.} =
-  label.attr("for", htmlFor)
-
 func control*(label: HTMLLabelElement): FormAssociatedElement {.jsfget.} =
-  let f = label.htmlFor
+  let f = label.attr("for")
   if f != "":
     let elem = label.document.getElementById(f)
     #TODO the supported check shouldn't be needed, just labelable
@@ -1505,9 +1521,16 @@ proc attr*(element: Element, name, value: string) =
     element.attributes.attrlist.add(element.newAttr(name, value))
   element.attr0(name, value)
 
+proc attri(element: Element, name: string, value: int) =
+  element.attr(name, $value)
+
 proc attrigz(element: Element, name: string, value: int) =
   if value > 0:
-    element.attr(name, $value)
+    element.attri(name, value)
+
+proc attrigez(element: Element, name: string, value: int) =
+  if value >= 0:
+    element.attri(name, value)
 
 proc setAttribute(element: Element, qualifiedName, value: string) {.jserr, jsfunc.} =
   if not qualifiedName.matchNameProduction():
@@ -1574,22 +1597,6 @@ proc value(attr: Attr, s: string) {.jsfset.} =
   if attr.ownerElement != nil:
     attr.ownerElement.attr0(attr.name, s)
 
-# Element attribute reflection (setters)
-proc className(element: Element, s: string) {.jsfset.} =
-  element.attr("class", s)
-
-proc size(element: HTMLInputElement, n: int) {.jsfset.} =
-  element.attrigz("size", n)
-
-proc size(element: HTMLSelectElement, n: int) {.jsfset.} =
-  element.attrigz("size", n)
-
-proc cols(element: HTMLTextAreaElement, n: int) {.jsfset.} =
-  element.attrigz("cols", n)
-
-proc rows(element: HTMLTextAreaElement, n: int) {.jsfset.} =
-  element.attrigz("rows", n)
-
 proc setNamedItem(map: NamedNodeMap, attr: Attr): Option[Attr] {.jserr, jsfunc.} =
   if attr.ownerElement != nil and attr.ownerElement != map.element:
     #TODO should be DOMException
@@ -2191,10 +2198,145 @@ proc querySelectorAll*(node: Node, q: string): seq[Element] {.jsfunc.} =
 proc querySelector*(node: Node, q: string): Element {.jsfunc.} =
   return doqs(node, q)
 
+const (ReflectTable, TagReflectMap, ReflectAllStartIndex) = (func(): (
+    seq[ReflectEntry],
+    Table[TagType, seq[uint16]],
+    uint16) =
+  var i: uint16 = 0
+  while i < ReflectTable0.len:
+    let x = ReflectTable0[i]
+    result[0].add(x)
+    if x.tags == AllTagTypes:
+      break
+    for tag in result[0][i].tags:
+      if tag notin result[1]:
+        result[1][tag] = newSeq[uint16]()
+      result[1][tag].add(i)
+    assert result[0][i].tags.len != 0
+    inc i
+  result[2] = i
+  while i < ReflectTable0.len:
+    let x = ReflectTable0[i]
+    assert x.tags == AllTagTypes
+    result[0].add(x)
+    inc i
+)()
+
+proc jsReflectGet(ctx: JSContext, this: JSValue, magic: cint): JSValue {.cdecl.} =
+  let entry = ReflectTable[uint16(magic)]
+  let op = getOpaque0(this)
+  if unlikely(not ctx.isInstanceOf(this, "Element") or op == nil):
+    return JS_ThrowTypeError(ctx, "Reflected getter called on a value that is not an element")
+  let element = cast[Element](op)
+  if element.tagType notin entry.tags:
+    return JS_ThrowTypeError(ctx, "Invalid tag type %s", element.tagType)
+  case entry.t
+  of REFLECT_STR:
+    let x = toJS(ctx, element.attr(entry.attrname))
+    return x
+  of REFLECT_BOOl:
+    return toJS(ctx, element.attrb(entry.attrname))
+  of REFLECT_INT:
+    return toJS(ctx, element.attri(entry.attrname).get(entry.i))
+  of REFLECT_INT_GREATER_ZERO:
+    return toJS(ctx, element.attrigz(entry.attrname).get(entry.i))
+  of REFLECT_INT_GREATER_EQUAL_ZERO:
+    return toJS(ctx, element.attrigez(entry.attrname).get(entry.i))
+
+proc jsReflectSet(ctx: JSContext, this, val: JSValue, magic: cint): JSValue {.cdecl.} =
+  if unlikely(not ctx.isInstanceOf(this, "Element")):
+    return JS_ThrowTypeError(ctx, "Reflected getter called on a value that is not an element")
+  let entry = ReflectTable[uint16(magic)]
+  let op = getOpaque0(this)
+  assert op != nil
+  let element = cast[Element](op)
+  if element.tagType notin entry.tags:
+    return JS_ThrowTypeError(ctx, "Invalid tag type %s", element.tagType)
+  case entry.t
+  of REFLECT_STR:
+    let x = toString(ctx, val)
+    if x.isSome:
+      element.attr(entry.attrname, x.get)
+  of REFLECT_BOOL:
+    let x = fromJS[bool](ctx, val)
+    if x.isSome:
+      if x.get:
+        element.attr(entry.attrname, "")
+      else:
+        element.delAttr(entry.attrname)
+  of REFLECT_INT:
+    let x = fromJS[int](ctx, val)
+    if x.isSome:
+      element.attri(entry.attrname, x.get)
+  of REFLECT_INT_GREATER_ZERO:
+    let x = fromJS[int](ctx, val)
+    if x.isSome:
+      element.attrigz(entry.attrname, x.get)
+  of REFLECT_INT_GREATER_EQUAL_ZERO:
+    let x = fromJS[int](ctx, val)
+    if x.isSome:
+      element.attrigez(entry.attrname, x.get)
+  return JS_DupValue(ctx, val)
+
 proc addconsoleModule*(ctx: JSContext) =
   #TODO console should not have a prototype
   ctx.registerType(console, nointerface = true)
 
+func getReflectFunctions(tags: set[TagType]): seq[TabGetSet] =
+  for tag in tags:
+    if tag in TagReflectMap:
+      for i in TagReflectMap[tag]:
+        result.add(TabGetSet(
+          name: ReflectTable[i].funcname,
+          get: jsReflectGet,
+          set: jsReflectSet,
+          magic: i
+        ))
+  return result
+
+func getElementReflectFunctions(): seq[TabGetSet] =
+  var i: uint16 = ReflectAllStartIndex
+  while i < ReflectTable.len:
+    let entry = ReflectTable[i]
+    assert entry.tags == AllTagTypes
+    result.add(TabGetSet(name: ReflectTable[i].funcname, get: jsReflectGet, set: jsReflectSet, magic: i))
+    inc i
+
+proc registerElements(ctx: JSContext, nodeCID: JSClassID) =
+  let elementCID = ctx.registerType(Element, parent = nodeCID)
+  const extra_getset = getElementReflectFunctions()
+  let htmlElementCID = ctx.registerType(HTMLElement, parent = elementCID,
+    extra_getset = extra_getset)
+  template register(t: typed, tags: set[TagType]) =
+    const extra_getset = getReflectFunctions(tags)
+    ctx.registerType(t, parent = htmlElementCID,
+      extra_getset = extra_getset)
+  template register(t: typed, tag: TagType) =
+    register(t, {tag})
+  register(HTMLInputElement, TAG_INPUT)
+  register(HTMLAnchorElement, TAG_A)
+  register(HTMLSelectElement, TAG_SELECT)
+  register(HTMLSpanElement, TAG_SPAN)
+  register(HTMLOptGroupElement, TAG_OPTGROUP)
+  register(HTMLOptionElement, TAG_OPTION)
+  register(HTMLHeadingElement, {TAG_H1, TAG_H2, TAG_H3, TAG_H4, TAG_H5, TAG_H6})
+  register(HTMLBRElement, TAG_BR)
+  register(HTMLMenuElement, TAG_MENU)
+  register(HTMLUListElement, TAG_UL)
+  register(HTMLOListElement, TAG_OL)
+  register(HTMLLIElement, TAG_LI)
+  register(HTMLStyleElement, TAG_STYLE)
+  register(HTMLLinkElement, TAG_LINK)
+  register(HTMLFormElement, TAG_FORM)
+  register(HTMLTemplateElement, TAG_TEMPLATE)
+  register(HTMLUnknownElement, TAG_UNKNOWN)
+  register(HTMLScriptElement, TAG_SCRIPT)
+  register(HTMLBaseElement, TAG_BASE)
+  register(HTMLAreaElement, TAG_AREA)
+  register(HTMLButtonElement, TAG_BUTTON)
+  register(HTMLTextAreaElement, TAG_TEXTAREA)
+  register(HTMLLabelElement, TAG_LABEL)
+
 proc addDOMModule*(ctx: JSContext) =
   let eventTargetCID = ctx.registerType(EventTarget)
   let nodeCID = ctx.registerType(Node, parent = eventTargetCID)
@@ -2210,30 +2352,6 @@ proc addDOMModule*(ctx: JSContext) =
   ctx.registerType(ProcessingInstruction, parent = characterDataCID)
   ctx.registerType(Text, parent = characterDataCID)
   ctx.registerType(DocumentType, parent = nodeCID)
-  let elementCID = ctx.registerType(Element, parent = nodeCID)
   ctx.registerType(Attr, parent = nodeCID)
   ctx.registerType(NamedNodeMap)
-  let htmlElementCID = ctx.registerType(HTMLElement, parent = elementCID)
-  ctx.registerType(HTMLInputElement, parent = htmlElementCID)
-  ctx.registerType(HTMLAnchorElement, parent = htmlElementCID)
-  ctx.registerType(HTMLSelectElement, parent = htmlElementCID)
-  ctx.registerType(HTMLSpanElement, parent = htmlElementCID)
-  ctx.registerType(HTMLOptGroupElement, parent = htmlElementCID)
-  ctx.registerType(HTMLOptionElement, parent = htmlElementCID)
-  ctx.registerType(HTMLHeadingElement, parent = htmlElementCID)
-  ctx.registerType(HTMLBRElement, parent = htmlElementCID)
-  ctx.registerType(HTMLMenuElement, parent = htmlElementCID)
-  ctx.registerType(HTMLUListElement, parent = htmlElementCID)
-  ctx.registerType(HTMLOListElement, parent = htmlElementCID)
-  ctx.registerType(HTMLLIElement, parent = htmlElementCID)
-  ctx.registerType(HTMLStyleElement, parent = htmlElementCID)
-  ctx.registerType(HTMLLinkElement, parent = htmlElementCID)
-  ctx.registerType(HTMLFormElement, parent = htmlElementCID)
-  ctx.registerType(HTMLTemplateElement, parent = htmlElementCID)
-  ctx.registerType(HTMLUnknownElement, parent = htmlElementCID)
-  ctx.registerType(HTMLScriptElement, parent = htmlElementCID)
-  ctx.registerType(HTMLBaseElement, parent = htmlElementCID)
-  ctx.registerType(HTMLAreaElement, parent = htmlElementCID)
-  ctx.registerType(HTMLButtonElement, parent = htmlElementCID)
-  ctx.registerType(HTMLTextAreaElement, parent = htmlElementCID)
-  ctx.registerType(HTMLLabelElement, parent = htmlElementCID)
+  ctx.registerElements(nodeCID)
diff --git a/src/html/env.nim b/src/html/env.nim
index 3c1668c7..a232baa5 100644
--- a/src/html/env.nim
+++ b/src/html/env.nim
@@ -22,13 +22,14 @@ proc language(navigator: Navigator): string {.jsfget.} = "en-US"
 proc languages(navigator: Navigator): seq[string] {.jsfget.} = @["en-US"] #TODO frozen array?
 
 # NavigatorOnline
-proc onLine(navigator: Navigator): bool {.jsfget.} = true # none of your business :)
+proc onLine(navigator: Navigator): bool {.jsfget.} =
+  true # at the very least, the terminal is on-line :)
 
 #TODO NavigatorContentUtils
 
 # NavigatorCookies
 # "this website needs cookies to be enabled to function correctly"
-# I'll take your incorrectly functioning website over the tracking any day.
+# It's probably better to lie here.
 proc cookieEnabled(navigator: Navigator): bool {.jsfget.} = true
 
 # NavigatorPlugins
@@ -48,9 +49,6 @@ proc addNavigatorModule(ctx: JSContext) =
   ctx.registerType(PluginArray)
   ctx.registerType(MimeTypeArray)
 
-#func `$`(window: Window): string {.jsfunc.} =
-#  "[object Window]"
-
 proc newWindow*(scripting: bool, loader = none(FileLoader)): Window =
   result = Window(
     console: console(),
diff --git a/src/html/tags.nim b/src/html/tags.nim
index 8b0713d7..c85fe2f7 100644
--- a/src/html/tags.nim
+++ b/src/html/tags.nim
@@ -83,6 +83,11 @@ const tagNameMap = (func(): Table[TagType, string] =
     result[v] = k
 )()
 
+const AllTagTypes* = (func(): set[TagType] =
+  for tag in TagType:
+    result.incl(tag)
+)()
+
 func tagName*(t: TagType): string =
   return tagNameMap[t]
 
diff --git a/src/js/javascript.nim b/src/js/javascript.nim
index 30ece22d..8dbf87f5 100644
--- a/src/js/javascript.nim
+++ b/src/js/javascript.nim
@@ -50,11 +50,11 @@ export
   JS_EVAL_FLAG_STRIP,
   JS_EVAL_FLAG_COMPILE_ONLY
 
-export JSRuntime, JSContext, JSValue
+export JSRuntime, JSContext, JSValue, JSClassID
 
 export
   JS_GetGlobalObject, JS_FreeValue, JS_IsException, JS_GetPropertyStr,
-  JS_IsFunction, JS_NewCFunctionData, JS_Call
+  JS_IsFunction, JS_NewCFunctionData, JS_Call, JS_DupValue
 
 type
   JSContextOpaque* = ref object
@@ -195,6 +195,11 @@ func getClassID*(val: JSValue): JSClassID =
               sizeof(uint8) # bit field
   return cast[ptr uint16](cast[int](JS_VALUE_GET_PTR(val)) + index)[]
 
+# getOpaque, but doesn't work for global objects.
+func getOpaque0*(val: JSValue): pointer =
+  if JS_VALUE_GET_TAG(val) == JS_TAG_OBJECT:
+    return JS_GetOpaque(val, val.getClassID())
+
 func getOpaque*(ctx: JSContext, val: JSValue, class: string): pointer =
   # Unfortunately, we can't change the global object's class.
   #TODO: or maybe we can, but I'm afraid of breaking something.
@@ -204,9 +209,7 @@ func getOpaque*(ctx: JSContext, val: JSValue, class: string): pointer =
     let opaque = JS_GetOpaque(global, 1) # JS_CLASS_OBJECT
     JS_FreeValue(ctx, global)
     return opaque
-  if JS_VALUE_GET_TAG(val) == JS_TAG_OBJECT:
-    return JS_GetOpaque(val, val.getClassID())
-  return nil
+  return getOpaque0(val)
 
 proc setInterruptHandler*(rt: JSRuntime, cb: JSInterruptHandler, opaque: pointer = nil) =
   JS_SetInterruptHandler(rt, cb, opaque)
@@ -359,7 +362,7 @@ func fromJSInt[T: SomeInteger](ctx: JSContext, val: JSValue): Option[T] =
       return none(T)
     return some(cast[uint64](ret))
 
-proc fromJS[T](ctx: JSContext, val: JSValue): Option[T]
+proc fromJS*[T](ctx: JSContext, val: JSValue): Option[T]
 
 macro len(t: type tuple): int =
   let i = t.getType()[1].len - 1 # - tuple
@@ -576,7 +579,7 @@ macro unpackArg0(f: typed) =
   let rvv = rv[1]
   result = quote do: `rvv`
 
-proc fromJS[T](ctx: JSContext, val: JSValue): Option[T] =
+proc fromJS*[T](ctx: JSContext, val: JSValue): Option[T] =
   when T is string:
     return toString(ctx, val)
   elif T is char:
@@ -1543,7 +1546,15 @@ proc findPragmas(t: NimNode): JSObjectPragmas =
             of "jsget": result.jsget.add(varName)
             of "jsset": result.jsset.add(varName)
 
-macro registerType*(ctx: typed, t: typed, parent: JSClassID = 0, asglobal = false, nointerface = false, name: static string = ""): JSClassID =
+type TabGetSet* = object
+  name*: string
+  get*: JSGetterMagicFunction
+  set*: JSSetterMagicFunction
+  magic*: uint16
+
+macro registerType*(ctx: typed, t: typed, parent: JSClassID = 0, asglobal =
+                   false, nointerface = false, name: static string = "",
+                   extra_getset: static openarray[TabGetSet] = []): JSClassID =
   result = newStmtList()
   let tname = t.strVal # the nim type's name.
   let name = if name == "": tname else: name # possibly a different name, e.g. Buffer for Container
@@ -1628,6 +1639,17 @@ macro registerType*(ctx: typed, t: typed, parent: JSClassID = 0, asglobal = fals
     if k notin getters:
       tabList.add(quote do: JS_CGETSET_DEF(`k`, nil, `v`))
 
+  for x in extra_getset:
+    #TODO TODO TODO what the hell is this...
+    # For some reason, extra_getset gets weird contents when nothing is
+    # passed to it.
+    if repr(x) != "" and repr(x) != "[]":
+      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)