import chame/tags
import html/catom
import html/dom
import html/enums
import io/dynstream
import io/posixstream
import js/base64
import js/domexception
import monoucha/javascript
import monoucha/tojs
import types/blob
import types/formdata
import types/opt
import utils/twtstr
proc constructEntryList*(form: HTMLFormElement; submitter: Element = nil;
encoding = "UTF-8"): seq[FormDataEntry]
var urandom* {.global.}: PosixStream
proc generateBoundary(): string =
var s: array[33, uint8]
urandom.recvDataLoop(s)
# 33 * 4 / 3 = 44 + prefix string is 22 bytes = 66 bytes
return "----WebKitFormBoundary" & btoa(s)
proc newFormData0*(): FormData =
return FormData(boundary: generateBoundary())
proc newFormData*(form: HTMLFormElement = nil; submitter: HTMLElement = nil):
DOMResult[FormData] {.jsctor.} =
let this = newFormData0()
if form != nil:
if submitter != nil:
if not submitter.isSubmitButton():
return errDOMException("Submitter must be a submit button",
"InvalidStateError")
if FormAssociatedElement(submitter).form != form:
return errDOMException("Submitter's form owner is not form",
"InvalidStateError")
if not form.constructingEntryList:
this.entries = constructEntryList(form, submitter)
return ok(this)
#TODO filename should not be allowed for string entries
# in other words, this should be an overloaded function, not just an or type
proc append*[T: string|Blob](this: FormData; name: string; value: T;
filename = none(string)) {.jsfunc.} =
when T is Blob:
let filename = if filename.isSome:
filename.get
elif value of WebFile:
WebFile(value).name
else:
"blob"
this.entries.add(FormDataEntry(
name: name,
isstr: false,
value: value,
filename: filename
))
else: # string
this.entries.add(FormDataEntry(
name: name,
isstr: true,
svalue: value
))
proc delete(this: FormData; name: string) {.jsfunc.} =
for i in countdown(this.entries.high, 0):
if this.entries[i].name == name:
this.entries.delete(i)
proc get(ctx: JSContext; this: FormData; name: string): JSValue {.jsfunc.} =
for entry in this.entries:
if entry.name == name:
if entry.isstr:
return toJS(ctx, entry.svalue)
else:
return toJS(ctx, entry.value)
return JS_NULL
proc getAll(ctx: JSContext; this: FormData; name: string): seq[JSValue]
{.jsfunc.} =
for entry in this.entries:
if entry.name == name:
if entry.isstr:
result.add(toJS(ctx, entry.svalue))
else:
result.add(toJS(ctx, entry.value))
proc add(list: var seq[FormDataEntry], entry: tuple[name, value: string]) =
list.add(FormDataEntry(
name: entry.name,
isstr: true,
svalue: entry.value
))
func toNameValuePairs*(list: seq[FormDataEntry]):
seq[tuple[name, value: string]] =
for entry in list:
if entry.isstr:
result.add((entry.name, entry.svalue))
else:
result.add((entry.name, entry.name))
# https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#constructing-the-form-data-set
# Warning: we skip the first "constructing entry list" check; the caller must
# do it.
proc constructEntryList*(form: HTMLFormElement; submitter: Element = nil;
encoding = "UTF-8"): seq[FormDataEntry] =
assert not form.constructingEntryList
form.constructingEntryList = true
var entrylist: seq[FormDataEntry] = @[]
for field in form.controls:
if field.findAncestor({TAG_DATALIST}) != nil or
field.attrb(satDisabled) or
field.isButton() and Element(field) != submitter:
continue
if field of HTMLInputElement:
let field = HTMLInputElement(field)
if field.inputType in {itCheckbox, itRadio} and not field.checked:
continue
if field.inputType == itImage:
var name = field.attr(satName)
if name != "":
name &= '.'
entrylist.add((name & 'x', $field.xcoord))
entrylist.add((name & 'y', $field.ycoord))
continue
#TODO custom elements
let name = field.attr(satName)
if name == "":
continue
if field of HTMLSelectElement:
let field = HTMLSelectElement(field)
for option in field.options:
if option.selected and not option.isDisabled:
entrylist.add((name, option.value))
elif field of HTMLInputElement:
let field = HTMLInputElement(field)
case field.inputType
of itCheckbox, itRadio:
let v = field.attr(satValue)
let value = if v != "":
v
else:
"on"
entrylist.add((name, value))
of itFile:
if field.file != nil:
entrylist.add(FormDataEntry(
name: name,
filename: field.file.name,
isstr: false,
value: field.file
))
of itHidden:
if name.equalsIgnoreCase("_charset_"):
entrylist.add((name, encoding))
else:
entrylist.add((name, field.value))
else:
entrylist.add((name, field.value))
elif field of HTMLButtonElement:
entrylist.add((name, HTMLButtonElement(field).value))
elif field of HTMLTextAreaElement:
entrylist.add((name, HTMLTextAreaElement(field).value))
else:
assert false, "Tag type " & $field.tagType &
" not accounted for in constructEntryList"
if field of HTMLTextAreaElement or
field of HTMLInputElement and
HTMLInputElement(field).inputType in AutoDirInput:
let dirname = field.attr(satDirname)
if dirname != "":
let dir = "ltr" #TODO bidi
entrylist.add((dirname, dir))
form.constructingEntryList = false
return entrylist
proc addFormDataModule*(ctx: JSContext) =
ctx.registerType(FormData)