import algorithm import streams import sugar import css/cssparser import css/mediaquery import css/select import css/selectorparser import css/sheet import css/stylednode import css/values import html/dom import html/tags type ApplyResult = object normal: seq[CSSDeclaration] important: seq[CSSDeclaration] DeclarationList* = array[PseudoElem, seq[CSSDeclaration]] proc applyProperty(styledNode: StyledNode, parent: CSSComputedValues, d: CSSDeclaration) = styledNode.computed.applyValue(parent, d) if styledNode.node != nil: Element(styledNode.node).cssapplied = true func applies(mq: MediaQuery): bool = case mq.t of CONDITION_MEDIA: case mq.media of MEDIA_TYPE_ALL: return true of MEDIA_TYPE_PRINT: return false of MEDIA_TYPE_SCREEN: return true of MEDIA_TYPE_SPEECH: return false of MEDIA_TYPE_TTY: return true of MEDIA_TYPE_UNKNOWN: return false of CONDITION_NOT: return not mq.n.applies() of CONDITION_AND: return mq.anda.applies() and mq.andb.applies() of CONDITION_OR: return mq.ora.applies() or mq.orb.applies() of CONDITION_FEATURE: case mq.feature.t of FEATURE_COLOR: return true #TODO of FEATURE_GRID: return mq.feature.b of FEATURE_HOVER: return mq.feature.b of FEATURE_PREFERS_COLOR_SCHEME: return mq.feature.b func applies(mqlist: MediaQueryList): bool = for mq in mqlist: if mq.applies(): return true return false type ToSorts = array[PseudoElem, seq[(int, seq[CSSDeclaration])]] proc calcRule(tosorts: var ToSorts, elem: Element, rule: CSSRuleDef) = for sel in rule.sels: if elem.selectorsMatch(sel): let spec = getSpecificity(sel) tosorts[sel.pseudo].add((spec,rule.decls)) func calcRules(elem: Element, sheet: CSSStylesheet): DeclarationList = var tosorts: ToSorts for rule in sheet.gen_rules(elem.tagType, elem.id, elem.classList): tosorts.calcRule(elem, rule) for i in PseudoElem: tosorts[i].sort((x, y) => cmp(x[0], y[0])) result[i] = collect(newSeq): for item in tosorts[i]: for dl in item[1]: dl proc applyNormal(ares: var ApplyResult, decls: seq[CSSDeclaration]) = for decl in decls: if not decl.important: ares.normal.add(decl) proc applyImportant(ares: var ApplyResult, decls: seq[CSSDeclaration]) = for decl in decls: if decl.important: ares.important.add(decl) # Always returns a new styled node, with the passed declarations applied. proc applyDeclarations(elem: Element, parent: CSSComputedValues, ua, user: DeclarationList, author: seq[DeclarationList]): StyledNode = let pseudo = PSEUDO_NONE var ares: ApplyResult ares.applyNormal(ua[pseudo]) ares.applyNormal(user[pseudo]) for rule in author: ares.applyNormal(rule[pseudo]) for rule in author: ares.applyImportant(rule[pseudo]) let style = elem.attr("style") if style.len > 0: let inline_rules = newStringStream(style).parseListOfDeclarations2() ares.applyNormal(inline_rules) ares.applyImportant(inline_rules) ares.applyImportant(user[pseudo]) ares.applyImportant(ua[pseudo]) result = StyledNode(t: STYLED_ELEMENT, node: elem, computed: parent.inheritProperties()) for rule in ares.normal: result.applyProperty(parent, rule) for rule in ares.important: result.applyProperty(parent, rule) # Either returns a new styled node or nil. proc applyDeclarations(pseudo: PseudoElem, parent: CSSComputedValues, ua, user: DeclarationList, author: seq[DeclarationList]): StyledNode = var ares: ApplyResult ares.applyNormal(ua[pseudo]) ares.applyNormal(user[pseudo]) for rule in author: ares.applyNormal(rule[pseudo]) for rule in author: ares.applyImportant(rule[pseudo]) ares.applyImportant(user[pseudo]) ares.applyImportant(ua[pseudo]) if ares.normal.len > 0 or ares.important.len > 0: result = StyledNode(t: STYLED_ELEMENT, node: nil, computed: parent.inheritProperties(), pseudo: pseudo) for rule in ares.normal: result.applyProperty(parent, rule) for rule in ares.important: result.applyProperty(parent, rule) func applyMediaQuery(ss: CSSStylesheet): CSSStylesheet = result = ss for mq in ss.mq_list: if mq.query.applies(): result.add(mq.children.applyMediaQuery()) func calcRules(elem: Element, ua, user: CSSStylesheet, author: seq[CSSStylesheet]): tuple[uadecls, userdecls: DeclarationList, authordecls: seq[DeclarationList]] = result.uadecls = calcRules(elem, ua) result.userdecls = calcRules(elem, user) for rule in author: result.authordecls.add(calcRules(elem, rule)) proc applyStyle(parent: StyledNode, elem: Element, uadecls, userdecls: DeclarationList, authordecls: seq[DeclarationList]): StyledNode = let parentComputed = if parent != nil: parent.computed else: rootProperties() result = elem.applyDeclarations(parentComputed, uadecls, userdecls, authordecls) assert result != nil # Builds a StyledNode tree, optionally based on a previously cached version. # This was originally a recursive algorithm; it had to be rewritten iteratively proc applyRules(document: Document, ua, user: CSSStylesheet, cachedTree: StyledNode): StyledNode = if document.html == nil: return var author: seq[CSSStylesheet] if document.head != nil: for sheet in document.head.sheets: author.add(sheet) var lenstack = newSeqOfCap[int](256) var styledStack: seq[(StyledNode, Node, PseudoElem, StyledNode)] styledStack.add((nil, document.html, PSEUDO_NONE, cachedTree)) while styledStack.len > 0: let (styledParent, child, pseudo, cachedChild) = styledStack.pop() # Remove stylesheets on nil if pseudo == PSEUDO_NONE and child == nil: let len = lenstack.pop() author.setLen(author.len - len) continue var styledChild: StyledNode if cachedChild != nil and (cachedChild.node == nil or cachedChild.node.nodeType != ELEMENT_NODE or Element(cachedChild.node).cssapplied): if cachedChild.t == STYLED_ELEMENT: styledChild = StyledNode(t: STYLED_ELEMENT, pseudo: cachedChild.pseudo, computed: cachedChild.computed, node: cachedChild.node) if cachedChild.pseudo != PSEUDO_NONE: styledChild.children = cachedChild.children #TODO does this actually refresh pseudo elems when needed? else: # Text styledChild = StyledNode(t: STYLED_TEXT, text: cachedChild.text, node: cachedChild.node) if styledParent == nil: # Root element result = styledChild else: styledParent.children.add(styledChild) else: if pseudo != PSEUDO_NONE: let (ua, user, authordecls) = Element(styledParent.node).calcRules(ua, user, author) case pseudo of PSEUDO_BEFORE, PSEUDO_AFTER: let styledPseudo = pseudo.applyDeclarations(styledParent.computed, ua, user, authordecls) if styledPseudo != nil: styledParent.children.add(styledPseudo) let content = styledPseudo.computed{"content"} if content.len > 0: styledPseudo.children.add(StyledNode(t: STYLED_TEXT, text: content)) of PSEUDO_INPUT_TEXT: let content = HTMLInputElement(styledParent.node).inputString() if content.len > 0: styledChild = StyledNode(t: STYLED_TEXT, text: content) styledParent.children.add(styledChild) of PSEUDO_NONE: discard else: assert child != nil if styledParent != nil: if child.nodeType == ELEMENT_NODE: let (ua, user, authordecls) = Element(child).calcRules(ua, user, author) styledChild = applyStyle(styledParent, Element(child), ua, user, authordecls) styledParent.children.add(styledChild) elif child.nodeType == TEXT_NODE: let text = Text(child) styledChild = StyledNode(t: STYLED_TEXT, node: child, text: text.data) styledParent.children.add(styledChild) else: # Root element assert result == nil let (ua, user, authordecls) = Element(child).calcRules(ua, user, author) styledChild = applyStyle(styledParent, Element(child), ua, user, authordecls) result = styledChild if styledChild != nil and styledChild.node != nil and styledChild.node.nodeType == ELEMENT_NODE: template stack_append(styledParent: StyledNode, child: Node) = if cachedChild != nil: var cached: StyledNode for it in cachedChild.children: if it.node == child: cached = it break styledStack.add((styledParent, child, PSEUDO_NONE, cached)) else: styledStack.add((styledParent, child, PSEUDO_NONE, nil)) template stack_append(styledParent: StyledNode, ps: PseudoElem) = if cachedChild != nil: var cached: StyledNode for it in cachedChild.children: if it.t == STYLED_ELEMENT and it.pseudo == ps: cached = it break styledStack.add((styledParent, nil, ps, cached)) else: styledStack.add((styledParent, nil, ps, nil)) let elem = Element(styledChild.node) # Add a nil before the last element (in-stack), so we can remove the # stylesheets if elem.sheets.len > 0: author.add(elem.sheets) lenstack.add(elem.sheets.len) styledStack.add((nil, nil, PSEUDO_NONE, nil)) stack_append styledChild, PSEUDO_AFTER for i in countdown(elem.childNodes.high, 0): stack_append styledChild, elem.childNodes[i] if elem.tagType == TAG_INPUT: stack_append styledChild, PSEUDO_INPUT_TEXT stack_append styledChild, PSEUDO_BEFORE proc applyStylesheets*(document: Document, uass, userss: CSSStylesheet, previousStyled: StyledNode): StyledNode = let uass = uass.applyMediaQuery() let userss = userss.applyMediaQuery() return document.applyRules(uass, userss, previousStyled) proc refreshStyle*(elem: Element) = elem.cssapplied = false for child in elem.children: child.refreshStyle()