import std/options
import std/streams
import std/strutils
import std/tables
import css/cssparser
import css/selectorparser
import css/stylednode
import html/catom
import html/dom
import utils/twtstr
import chame/tags
#TODO rfNone should match insensitively for certain properties
func attrSelectorMatches(elem: Element; sel: Selector): bool =
case sel.rel.t
of rtExists: return elem.attrb(sel.attr)
of rtEquals:
case sel.rel.flag
of rfNone: return elem.attr(sel.attr) == sel.value
of rfI: return elem.attr(sel.attr).equalsIgnoreCase(sel.value)
of rfS: return elem.attr(sel.attr) == sel.value
of rtToken:
let val = elem.attr(sel.attr)
case sel.rel.flag
of rfNone: return sel.value in val.split(AsciiWhitespace)
of rfI:
let val = val.toLowerAscii()
let selval = sel.value.toLowerAscii()
return selval in val.split(AsciiWhitespace)
of rfS: return sel.value in val.split(AsciiWhitespace)
of rtBeginDash:
let val = elem.attr(sel.attr)
case sel.rel.flag
of rfNone:
return val == sel.value or sel.value.startsWith(val & '-')
of rfI:
return val.equalsIgnoreCase(sel.value) or
sel.value.startsWithIgnoreCase(val & '-')
of rfS:
return val == sel.value or sel.value.startsWith(val & '-')
of rtStartsWith:
let val = elem.attr(sel.attr)
case sel.rel.flag
of rfNone: return val.startsWith(sel.value)
of rfI: return val.startsWithIgnoreCase(sel.value)
of rfS: return val.startsWith(sel.value)
of rtEndsWith:
let val = elem.attr(sel.attr)
case sel.rel.flag
of rfNone: return val.endsWith(sel.value)
of rfI: return val.endsWithIgnoreCase(sel.value)
of rfS: return val.endsWith(sel.value)
of rtContains:
let val = elem.attr(sel.attr)
case sel.rel.flag
of rfNone: return val.contains(sel.value)
of rfI:
let val = val.toLowerAscii()
let selval = sel.value.toLowerAscii()
return val.contains(selval)
of rfS: return val.contains(sel.value)
func selectorsMatch*[T: Element|StyledNode](elem: T; cxsel: ComplexSelector;
felem: T = nil): bool
func selectorsMatch*[T: Element|StyledNode](elem: T; slist: SelectorList;
felem: T = nil): bool =
for cxsel in slist:
if elem.selectorsMatch(cxsel, felem):
return true
return false
func pseudoSelectorMatches[T: Element|StyledNode](elem: T; sel: Selector;
felem: T): bool =
let selem = elem
when elem is StyledNode:
let elem = Element(elem.node)
case sel.pseudo.t
of pcFirstChild: return elem.parentNode.firstElementChild == elem
of pcLastChild: return elem.parentNode.lastElementChild == elem
of pcOnlyChild:
return elem.parentNode.firstElementChild == elem and
elem.parentNode.lastElementChild == elem
of pcHover:
when selem is StyledNode: felem.addDependency(selem, dtHover)
return elem.hover
of pcRoot: return elem == elem.document.html
of pcNthChild:
if sel.pseudo.ofsels.len != 0 and
not selem.selectorsMatch(sel.pseudo.ofsels, felem):
return false
let A = sel.pseudo.anb.A # step
let B = sel.pseudo.anb.B # start
var i = 1
let parent = when selem is StyledNode: selem.parent
else: selem.parentNode
if parent == nil: return false
for child in parent.elementList:
when selem is StyledNode:
if not child.isDomElement: continue
if child == selem:
if A == 0:
return i == B
if A < 0:
return (i - B) <= 0 and (i - B) mod A == 0
return (i - B) >= 0 and (i - B) mod A == 0
if sel.pseudo.ofsels.len == 0 or
child.selectorsMatch(sel.pseudo.ofsels, felem):
inc i
return false
of pcNthLastChild:
if sel.pseudo.ofsels.len == 0 and
not selem.selectorsMatch(sel.pseudo.ofsels, felem):
return false
let A = sel.pseudo.anb.A # step
let B = sel.pseudo.anb.B # start
var i = 1
let parent = when selem is StyledNode: selem.parent
else: selem.parentNode
if parent == nil: return false
for child in parent.elementList_rev:
when selem is StyledNode:
if not child.isDomElement: continue
if child == selem:
if A == 0:
return i == B
if A < 0:
return (i - B) <= 0 and (i - B) mod A == 0
return (i - B) >= 0 and (i - B) mod A == 0
if sel.pseudo.ofsels.len != 0 or
child.selectorsMatch(sel.pseudo.ofsels, felem):
inc i
return false
of pcChecked:
when selem is StyledNode: felem.addDependency(selem, dtChecked)
if elem.tagType == TAG_INPUT:
return HTMLInputElement(elem).checked
elif elem.tagType == TAG_OPTION:
return HTMLOptionElement(elem).selected
return false
of pcFocus:
when selem is StyledNode: felem.addDependency(selem, dtFocus)
return elem.document.focus == elem
of pcNot:
return not selem.selectorsMatch(sel.pseudo.fsels, felem)
of pcIs, pcWhere:
return selem.selectorsMatch(sel.pseudo.fsels, felem)
of pcLang:
return sel.pseudo.s == "en" #TODO languages?
of pcLink:
return elem.tagType in {TAG_A, TAG_AREA} and elem.attrb(satHref)
of pcVisited:
return false
func selectorMatches[T: Element|StyledNode](elem: T; sel: Selector;
felem: T = nil): bool =
let selem = elem
when elem is StyledNode:
let elem = Element(selem.node)
case sel.t
of stType:
return elem.localName == sel.tag
of stClass:
return sel.class in elem.classList
of stId:
return sel.id == elem.id
of stAttr:
return elem.attrSelectorMatches(sel)
of stPseudoClass:
return pseudoSelectorMatches(selem, sel, felem)
of stPseudoElement:
return true
of stUniversal:
return true
func selectorsMatch[T: Element|StyledNode](elem: T; sels: CompoundSelector;
felem: T): bool =
for sel in sels:
if not selectorMatches(elem, sel, felem):
return false
return true
func complexSelectorMatches[T: Element|StyledNode](elem: T;
cxsel: ComplexSelector; felem: T = nil): bool =
var e = elem
for i in countdown(cxsel.high, 0):
let sels = cxsel[i]
if e == nil:
return false
var match = false
case sels.ct
of ctNone:
match = e.selectorsMatch(sels, felem)
of ctDescendant:
e = e.parentElement
while e != nil:
if e.selectorsMatch(sels, felem):
match = true
break
e = e.parentElement
of ctChild:
e = e.parentElement
if e != nil:
match = e.selectorsMatch(sels, felem)
of ctNextSibling:
if e.parentElement == nil: return false
var found = false
for child in e.parentElement.elementList_rev:
when elem is StyledNode:
if not child.isDomElement: continue
if e == child:
found = true
continue
if found:
e = child
match = e.selectorsMatch(sels, felem)
break
of ctSubsequentSibling:
var found = false
if e.parentElement == nil: return false
for child in e.parentElement.elementList_rev:
when elem is StyledNode:
if not child.isDomElement: continue
if child == elem:
found = true
continue
if not found: continue
if child.selectorsMatch(sels, felem):
e = child
match = true
break
if not match:
return false
return true
# WARNING for StyledNode, this has the side effect of modifying depends.
#TODO make that an explicit flag or something, also get rid of the Element case
func selectorsMatch*[T: Element|StyledNode](elem: T; cxsel: ComplexSelector;
felem: T = nil): bool =
var felem = if felem != nil:
felem
else:
elem
return elem.complexSelectorMatches(cxsel, felem)
proc querySelectorAll(node: Node; q: string): seq[Element] =
let selectors = parseSelectors(newStringStream(q), node.document.factory)
for element in node.elements:
if element.selectorsMatch(selectors):
result.add(element)
doqsa = (proc(node: Node, q: string): seq[Element] = querySelectorAll(node, q))
proc querySelector(node: Node; q: string): Element =
let selectors = parseSelectors(newStringStream(q), node.document.factory)
for element in node.elements:
if element.selectorsMatch(selectors):
return element
return nil
doqs = (proc(node: Node, q: string): Element = querySelector(node, q))