diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/css/cascade.nim | 26 | ||||
-rw-r--r-- | src/css/selectorparser.nim | 2 | ||||
-rw-r--r-- | src/css/values.nim | 630 | ||||
-rw-r--r-- | src/html/dom.nim | 48 |
4 files changed, 327 insertions, 379 deletions
diff --git a/src/css/cascade.nim b/src/css/cascade.nim index b2d007f2..ad1d537d 100644 --- a/src/css/cascade.nim +++ b/src/css/cascade.nim @@ -388,6 +388,15 @@ proc applyRulesFrameInvalid(frame: CascadeFrame; ua, user: CSSStylesheet; let styledText = styledParent.newStyledReplacement(content) styledText.pseudo = pseudo styledParent.children.add(styledText) + of peCanvas: + let content = CSSContent( + t: ContentImage, + s: "canvas://", + bmp: HTMLCanvasElement(styledParent.node).bitmap + ) + let styledText = styledParent.newStyledReplacement(content) + styledText.pseudo = pseudo + styledParent.children.add(styledText) of peVideo: let content = CSSContent(t: ContentVideo) let styledText = styledParent.newStyledReplacement(content) @@ -491,16 +500,13 @@ proc appendChildren(styledStack: var seq[CascadeFrame]; frame: CascadeFrame; -1 let elem = Element(styledChild.node) styledStack.stackAppend(frame, styledChild, peAfter, idx, parentDeclMap) - if elem.tagType == TAG_TEXTAREA: - styledStack.stackAppend(frame, styledChild, peTextareaText, idx) - elif elem.tagType == TAG_IMG or elem.tagType == TAG_IMAGE: - styledStack.stackAppend(frame, styledChild, peImage, idx) - elif elem.tagType == TAG_VIDEO: - styledStack.stackAppend(frame, styledChild, peVideo, idx) - elif elem.tagType == TAG_AUDIO: - styledStack.stackAppend(frame, styledChild, peAudio, idx) - elif elem.tagType == TAG_BR: - styledStack.stackAppend(frame, styledChild, peNewline, idx) + case elem.tagType + of TAG_TEXTAREA: styledStack.stackAppend(frame, styledChild, peTextareaText, idx) + of TAG_IMG, TAG_IMAGE: styledStack.stackAppend(frame, styledChild, peImage, idx) + of TAG_VIDEO: styledStack.stackAppend(frame, styledChild, peVideo, idx) + of TAG_AUDIO: styledStack.stackAppend(frame, styledChild, peAudio, idx) + of TAG_BR: styledStack.stackAppend(frame, styledChild, peNewline, idx) + of TAG_CANVAS: styledStack.stackAppend(frame, styledChild, peCanvas, idx) else: for i in countdown(elem.childList.high, 0): if elem.childList[i] of Element or elem.childList[i] of Text: diff --git a/src/css/selectorparser.nim b/src/css/selectorparser.nim index 5e466ab3..26940292 100644 --- a/src/css/selectorparser.nim +++ b/src/css/selectorparser.nim @@ -13,7 +13,7 @@ type PseudoElem* = enum peNone, peBefore, peAfter, # internal - peInputText, peTextareaText, peImage, peNewline, peVideo, peAudio + peInputText, peTextareaText, peImage, peNewline, peVideo, peAudio, peCanvas PseudoClass* = enum pcFirstChild, pcLastChild, pcOnlyChild, pcHover, pcRoot, pcNthChild, diff --git a/src/css/values.nim b/src/css/values.nim index 4cca9489..7150de59 100644 --- a/src/css/values.nim +++ b/src/css/values.nim @@ -27,8 +27,22 @@ type cstFlexFlow = "flex-flow" CSSUnit* = enum - cuCm, cuMm, cuIn, cuPx, cuPt, cuPc, cuEm, cuEx, cuCh, cuRem, cuVw, cuVh, - cuVmin, cuVmax, cuPerc, cuIc + cuCm = "cm" + cuMm = "mm" + cuIn = "in" + cuPx = "px" + cuPt = "pt" + cuPc = "pc" + cuEm = "em" + cuEx = "ex" + cuCh = "ch" + cuRem = "rem" + cuVw = "vw" + cuVh = "vh" + cuVmin = "vmin" + cuVmax = "vmax" + cuPerc = "%" + cuIc = "ic" CSSPropertyType* = enum cptNone = "" @@ -119,64 +133,118 @@ type cvtNumber = "number" CSSGlobalType = enum - cgtNoglobal, cgtInitial, cgtInherit, cgtRevert, cgtUnset + cgtNoglobal = "" + cgtInitial = "initial" + cgtInherit = "inherit" + cgtRevert = "revert" + cgtUnset = "unset" CSSDisplay* = enum - DisplayNone, DisplayInline, DisplayBlock, DisplayListItem, - DisplayInlineBlock, DisplayTable, DisplayInlineTable, DisplayTableRowGroup, - DisplayTableHeaderGroup, DisplayTableFooterGroup, DisplayTableColumnGroup, - DisplayTableRow, DisplayTableColumn, DisplayTableCell, DisplayTableCaption, - DisplayFlowRoot, DisplayFlex, DisplayInlineFlex - - CSSWhitespace* = enum - WhitespaceNormal, WhitespaceNowrap, WhitespacePre, WhitespacePreLine, - WhitespacePreWrap + DisplayNone = "none" + DisplayInline = "inline" + DisplayBlock = "block" + DisplayListItem = "list-item" + DisplayInlineBlock = "inline-block" + DisplayTable = "table" + DisplayInlineTable = "inline-table" + DisplayTableRowGroup = "table-row-group" + DisplayTableHeaderGroup = "table-header-group" + DisplayTableFooterGroup = "table-footer-group" + DisplayTableColumnGroup = "table-column-group" + DisplayTableRow = "table-row" + DisplayTableColumn = "table-column" + DisplayTableCell = "table-cell" + DisplayTableCaption = "table-caption" + DisplayFlowRoot = "flow-root" + DisplayFlex = "flex" + DisplayInlineFlex = "inline-flex" + + CSSWhiteSpace* = enum + WhitespaceNormal = "normal" + WhitespaceNowrap = "nowrap" + WhitespacePre = "pre" + WhitespacePreLine = "pre-line" + WhitespacePreWrap = "pre-wrap" CSSFontStyle* = enum - FontStyleNormal, FontStyleItalic, FontStyleOblique + FontStyleNormal = "normal" + FontStyleItalic = "italic" + FontStyleOblique = "oblique" CSSPosition* = enum - PositionStatic, PositionRelative, PositionAbsolute, PositionFixed, - PositionSticky + PositionStatic = "static" + PositionRelative = "relative" + PositionAbsolute = "absolute" + PositionFixed = "fixed" + PositionSticky = "sticky" CSSTextDecoration* = enum - TextDecorationNone, TextDecorationUnderline, TextDecorationOverline, - TextDecorationLineThrough, TextDecorationBlink + TextDecorationNone = "none" + TextDecorationUnderline = "underline" + TextDecorationOverline = "overline" + TextDecorationLineThrough = "line-through" + TextDecorationBlink = "blink" CSSWordBreak* = enum - WordBreakNormal, WordBreakBreakAll, WordBreakKeepAll + WordBreakNormal = "normal" + WordBreakBreakAll = "break-all" + WordBreakKeepAll = "keep-all" CSSListStyleType* = enum - ListStyleTypeNone, ListStyleTypeDisc, ListStyleTypeCircle, - ListStyleTypeSquare, ListStyleTypeDecimal, - ListStyleTypeDisclosureClosed, ListStyleTypeDisclosureOpen, - ListStyleTypeCjkEarthlyBranch, ListStyleTypeCjkHeavenlyStem, - ListStyleTypeLowerRoman, ListStyleTypeUpperRoman, - ListStyleTypeLowerAlpha, ListStyleTypeUpperAlpha, - ListStyleTypeLowerGreek, - ListStyleTypeHiragana, ListStyleTypeHiraganaIroha, - ListStyleTypeKatakana, ListStyleTypeKatakanaIroha, - ListStyleTypeJapaneseInformal + ListStyleTypeNone = "none" + ListStyleTypeDisc = "disc" + ListStyleTypeCircle = "circle" + ListStyleTypeSquare = "square" + ListStyleTypeDecimal = "decimal" + ListStyleTypeDisclosureClosed = "disclosure-closed" + ListStyleTypeDisclosureOpen = "disclosure-open" + ListStyleTypeCjkEarthlyBranch = "cjk-earthly-branch" + ListStyleTypeCjkHeavenlyStem = "cjk-heavenly-stem" + ListStyleTypeLowerRoman = "lower-roman" + ListStyleTypeUpperRoman = "upper-roman" + ListStyleTypeLowerAlpha = "lower-alpha" + ListStyleTypeUpperAlpha = "upper-alpha" + ListStyleTypeLowerGreek = "lower-greek" + ListStyleTypeHiragana = "hiragana" + ListStyleTypeHiraganaIroha = "hiragana-iroha" + ListStyleTypeKatakana = "katakana" + ListStyleTypeKatakanaIroha = "katakana-iroha" + ListStyleTypeJapaneseInformal = "japanese-informal" CSSVerticalAlign2* = enum - VerticalAlignBaseline, VerticalAlignSub, VerticalAlignSuper, - VerticalAlignTextTop, VerticalAlignTextBottom, VerticalAlignMiddle, - VerticalAlignTop, VerticalAlignBottom + VerticalAlignBaseline = "baseline" + VerticalAlignSub = "sub" + VerticalAlignSuper = "super" + VerticalAlignTextTop = "text-top" + VerticalAlignTextBottom = "text-bottom" + VerticalAlignMiddle = "middle" + VerticalAlignTop = "top" + VerticalAlignBottom = "bottom" CSSTextAlign* = enum - TextAlignStart, TextAlignEnd, TextAlignLeft, TextAlignRight, - TextAlignCenter, TextAlignJustify, TextAlignChaCenter, - TextAlignChaLeft, TextAlignChaRight + TextAlignStart = "start" + TextAlignEnd = "end" + TextAlignLeft = "left" + TextAlignRight = "right" + TextAlignCenter = "center" + TextAlignJustify = "justify" + TextAlignChaCenter = "-cha-center" + TextAlignChaLeft = "-cha-left" + TextAlignChaRight = "-cha-right" CSSListStylePosition* = enum - ListStylePositionOutside, ListStylePositionInside + ListStylePositionOutside = "outside" + ListStylePositionInside = "inside" CSSCaptionSide* = enum - CaptionSideTop, CaptionSideBottom, CaptionSideBlockStart, - CaptionSideBlockEnd, + CaptionSideTop = "top" + CaptionSideBottom = "bottom" + CaptionSideBlockStart = "block-start" + CaptionSideBlockEnd = "block-end" CSSBorderCollapse* = enum - BorderCollapseSeparate, BorderCollapseCollapse + BorderCollapseSeparate = "separate" + BorderCollapseCollapse = "collapse" CSSContentType* = enum ContentString, ContentOpenQuote, ContentCloseQuote, @@ -184,29 +252,46 @@ type ContentVideo, ContentAudio, ContentNewline CSSFloat* = enum - FloatNone, FloatLeft, FloatRight + FloatNone = "none" + FloatLeft = "left" + FloatRight = "right" CSSVisibility* = enum - VisibilityVisible, VisibilityHidden, VisibilityCollapse + VisibilityVisible = "visible" + VisibilityHidden = "hidden" + VisibilityCollapse = "collapse" CSSBoxSizing* = enum - BoxSizingContentBox, BoxSizingBorderBox + BoxSizingContentBox = "content-box" + BoxSizingBorderBox = "border-box" CSSClear* = enum - ClearNone, ClearLeft, ClearRight, ClearBoth, ClearInlineStart, - ClearInlineEnd + ClearNone = "none" + ClearLeft = "left" + ClearRight = "right" + ClearBoth = "both" + ClearInlineStart = "inline-start" + ClearInlineEnd = "inline-end" CSSTextTransform* = enum - TextTransformNone, TextTransformCapitalize, TextTransformUppercase, - TextTransformLowercase, TextTransformFullWidth, - TextTransformFullSizeKana, TextTransformChaHalfWidth + TextTransformNone = "none" + TextTransformCapitalize = "capitalize" + TextTransformUppercase = "uppercase" + TextTransformLowercase = "lowercase" + TextTransformFullWidth = "full-width" + TextTransformFullSizeKana = "full-size-kana" + TextTransformChaHalfWidth = "-cha-half-width" CSSFlexDirection* = enum - FlexDirectionRow, FlexDirectionRowReverse, FlexDirectionColumn, - FlexDirectionColumnReverse + FlexDirectionRow = "row" + FlexDirectionRowReverse = "row-reverse" + FlexDirectionColumn = "column" + FlexDirectionColumnReverse = "column-reverse" CSSFlexWrap* = enum - FlexWrapNowrap, FlexWrapWrap, FlexWrapWrapReverse + FlexWrapNowrap = "nowrap" + FlexWrapWrap = "wrap" + FlexWrapWrapReverse = "wrap-reverse" type CSSLength* = object @@ -246,7 +331,7 @@ type of cvtQuotes: quotes*: CSSQuotes of cvtWhiteSpace: - whitespace*: CSSWhitespace + whitespace*: CSSWhiteSpace of cvtInteger: integer*: int of cvtNumber: @@ -300,7 +385,7 @@ type coUser coAuthor - CSSComputedEntry = tuple + CSSComputedEntry* = tuple t: CSSPropertyType val: CSSComputedValue global: CSSGlobalType @@ -415,25 +500,57 @@ func isSupportedProperty*(s: string): bool = func `$`*(length: CSSLength): string = if length.auto: return "auto" - let ss = ($length.unit).split('_') - let us = ss.toOpenArray(1, ss.high).join('_').toLowerAscii() - return $length.num & us + return $length.num & $length.unit func `$`*(content: CSSContent): string = if content.s != "": return "url(" & content.s & ")" return "none" +func `$`(quotes: CSSQuotes): string = + if quotes.auto: + return "auto" + return "auto" #TODO + +func `$`(counterreset: seq[CSSCounterReset]): string = + result = "" + for it in counterreset: + result &= $it.name + result &= ' ' + result &= $it.num + func `$`*(val: CSSComputedValue): string = case val.v - of cvtColor: - result &= $val.color - of cvtImage: - result &= $val.image - of cvtLength: - result &= $val.length - else: - result = $val.v + of cvtNone: return "none" + of cvtColor: return $val.color + of cvtImage: return $val.image + of cvtLength: return $val.length + of cvtDisplay: return $val.display + of cvtFontStyle: return $val.fontstyle + of cvtWhiteSpace: return $val.whitespace + of cvtInteger: return $val.integer + of cvtTextDecoration: return $val.textdecoration + of cvtWordBreak: return $val.wordbreak + of cvtListStyleType: return $val.liststyletype + of cvtVerticalAlign: return $val.verticalalign + of cvtTextAlign: return $val.textalign + of cvtListStylePosition: return $val.liststyleposition + of cvtPosition: return $val.position + of cvtCaptionSide: return $val.captionside + of cvtLength2: return $val.length2.a & " " & $val.length2.b + of cvtBorderCollapse: return $val.bordercollapse + of cvtContent: return $val.content + of cvtQuotes: return $val.quotes + of cvtCounterReset: return $val.counterreset + of cvtFloat: return $val.float + of cvtVisibility: return $val.visibility + of cvtBoxSizing: return $val.boxsizing + of cvtClear: return $val.clear + of cvtTextTransform: return $val.texttransform + of cvtBgcolorIsCanvas: return $val.bgcoloriscanvas + of cvtFlexDirection: return $val.flexdirection + of cvtFlexWrap: return $val.flexwrap + of cvtNumber: return $val.number macro `{}`*(vals: CSSComputedValues; s: static string): untyped = let t = propertyType(s) @@ -454,18 +571,18 @@ macro `{}=`*(vals: CSSComputedValues; s: static string, val: typed) = func inherited(t: CSSPropertyType): bool = return InheritedArray[t] -func em_to_px(em: float64, window: WindowAttributes): LayoutUnit = +func em_to_px(em: float64; window: WindowAttributes): LayoutUnit = em * float64(window.ppl) -func ch_to_px(ch: float64, window: WindowAttributes): LayoutUnit = +func ch_to_px(ch: float64; window: WindowAttributes): LayoutUnit = ch * float64(window.ppc) # 水 width, we assume it's 2 chars -func ic_to_px(ic: float64, window: WindowAttributes): LayoutUnit = +func ic_to_px(ic: float64; window: WindowAttributes): LayoutUnit = ic * float64(window.ppc) * 2 # x-letter height, we assume it's em/2 -func ex_to_px(ex: float64, window: WindowAttributes): LayoutUnit = +func ex_to_px(ex: float64; window: WindowAttributes): LayoutUnit = ex * float64(window.ppc) / 2 func px*(l: CSSLength; window: WindowAttributes; p: LayoutUnit): LayoutUnit @@ -581,30 +698,47 @@ const Colors: Table[string, ARGBColor] = ((func (): Table[string, ARGBColor] = result["transparent"] = rgba(0x00, 0x00, 0x00, 0x00) )()) -const Units = { - "%": cuPerc, - "cm": cuCm, - "mm": cuMm, - "in": cuIn, - "px": cuPx, - "pt": cuPt, - "pc": cuPc, - "em": cuEm, - "ex": cuEx, - "ch": cuCh, - "ic": cuCh, - "rem": cuRem, - "vw": cuVw, - "vh": cuVh, - "vmin": cuVmin, - "vmax": cuVmax, -}.toTable() - -func cssLength(val: float64, unit: string): Opt[CSSLength] = - if unit in Units: - return ok(CSSLength(num: val, unit: Units[unit])) +func isToken(cval: CSSComponentValue): bool {.inline.} = + cval of CSSToken + +func getToken(cval: CSSComponentValue): CSSToken {.inline.} = + CSSToken(cval) + +func parseIdent0[T](map: static openArray[(string, T)]; s: string): Opt[T] = + # cmp when len is small enough, otherwise lowercase & hashmap + when map.len <= 4: + for (k, v) in map: + if k.equalsIgnoreCase(s): + return ok(v) + else: + const MapTable = map.toTable() + let val = s.toLowerAscii() + if val in MapTable: + return ok(MapTable[val]) + return err() + +func parseIdent[T](map: static openArray[(string, T)]; cval: CSSComponentValue): + Opt[T] = + if isToken(cval): + let tok = getToken(cval) + if tok.tokenType == cttIdent: + return parseIdent0[T](map, tok.value) return err() +func getIdentMap[T: enum](e: typedesc[T]): seq[(string, T)] = + result = @[] + for e in T.low .. T.high: + result.add(($e, e)) + +func parseIdent[T: enum](cval: CSSComponentValue): Opt[T] = + const IdentMap = getIdentMap(T) + return IdentMap.parseIdent(cval) + +func cssLength(val: float64; unit: string): Opt[CSSLength] = + const UnitMap = getIdentMap(CSSUnit) + let u = ?UnitMap.parseIdent0(unit) + return ok(CSSLength(num: val, unit: u)) + const CSSLengthAuto* = CSSLength(auto: true) func parseDimensionValues*(s: string): Option[CSSLength] = @@ -740,29 +874,6 @@ func cssColor*(val: CSSComponentValue): Opt[CellColor] = return parseANSI(f.value) return err() -func isToken(cval: CSSComponentValue): bool {.inline.} = - cval of CSSToken - -func getToken(cval: CSSComponentValue): CSSToken {.inline.} = - CSSToken(cval) - -func cssIdent[T](map: static openArray[(string, T)], cval: CSSComponentValue): - Opt[T] = - if isToken(cval): - let tok = getToken(cval) - if tok.tokenType == cttIdent: - # cmp when len is small enough, otherwise lowercase & hashmap - when map.len <= 4: - for (k, v) in map: - if k.equalsIgnoreCase(tok.value): - return ok(v) - else: - const MapTable = map.toTable() - let val = tok.value.toLowerAscii() - if val in MapTable: - return ok(MapTable[val]) - return err() - func cssLength*(val: CSSComponentValue; has_auto = true; allow_negative = true): Opt[CSSLength] = if val of CSSToken: @@ -814,15 +925,9 @@ func cssWordSpacing(cval: CSSComponentValue): Opt[CSSLength] = return err() func cssGlobal(cval: CSSComponentValue): CSSGlobalType = - const GlobalMap = { - "inherit": cgtInherit, - "initial": cgtInitial, - "unset": cgtUnset, - "revert": cgtRevert - } - return cssIdent(GlobalMap, cval).get(cgtNoglobal) - -func cssQuotes(cvals: seq[CSSComponentValue]): Opt[CSSQuotes] = + return parseIdent[CSSGlobalType](cval).get(cgtNoglobal) + +func cssQuotes(cvals: openArray[CSSComponentValue]): Opt[CSSQuotes] = template die = return err() if cvals.len == 0: @@ -857,7 +962,7 @@ func cssQuotes(cvals: seq[CSSComponentValue]): Opt[CSSQuotes] = die return ok(res) -func cssContent(cvals: seq[CSSComponentValue]): seq[CSSContent] = +func cssContent(cvals: openArray[CSSComponentValue]): seq[CSSContent] = for cval in cvals: if isToken(cval): let tok = getToken(cval) @@ -877,47 +982,6 @@ func cssContent(cvals: seq[CSSComponentValue]): seq[CSSContent] = result.add(CSSContent(t: ContentString, s: tok.value)) else: return -func cssDisplay(cval: CSSComponentValue): Opt[CSSDisplay] = - const DisplayMap = { - "block": DisplayBlock, - "inline": DisplayInline, - "list-item": DisplayListItem, - "inline-block": DisplayInlineBlock, - "table": DisplayTable, - "table-row": DisplayTableRow, - "table-cell": DisplayTableCell, - "table-column": DisplayTableColumn, - "table-column-group": DisplayTableColumnGroup, - "inline-table": DisplayInlineTable, - "table-row-group": DisplayTableRowGroup, - "table-header-group": DisplayTableHeaderGroup, - "table-footer-group": DisplayTableFooterGroup, - "table-caption": DisplayTableCaption, - "flow-root": DisplayFlowRoot, - "flex": DisplayFlex, - "inline-flex": DisplayInlineFlex, - "none": DisplayNone - } - return cssIdent(DisplayMap, cval) - -func cssFontStyle(cval: CSSComponentValue): Opt[CSSFontStyle] = - const FontStyleMap = { - "normal": FontStyleNormal, - "italic": FontStyleItalic, - "oblique": FontStyleOblique - } - return cssIdent(FontStyleMap, cval) - -func cssWhiteSpace(cval: CSSComponentValue): Opt[CSSWhitespace] = - const WhiteSpaceMap = { - "normal": WhitespaceNormal, - "nowrap": WhitespaceNowrap, - "pre": WhitespacePre, - "pre-line": WhitespacePreLine, - "pre-wrap": WhitespacePreWrap - } - return cssIdent(WhiteSpaceMap, cval) - func cssFontWeight(cval: CSSComponentValue): Opt[int] = if isToken(cval): let tok = getToken(cval) @@ -928,13 +992,13 @@ func cssFontWeight(cval: CSSComponentValue): Opt[int] = "lighter": 400, "bolder": 700 } - return cssIdent(FontWeightMap, cval) + return FontWeightMap.parseIdent(cval) elif tok.tokenType == cttNumber: if tok.nvalue in 1f64..1000f64: return ok(int(tok.nvalue)) return err() -func cssTextDecoration(cvals: seq[CSSComponentValue]): +func cssTextDecoration(cvals: openArray[CSSComponentValue]): Opt[set[CSSTextDecoration]] = var s: set[CSSTextDecoration] = {} for cval in cvals: @@ -942,73 +1006,20 @@ func cssTextDecoration(cvals: seq[CSSComponentValue]): continue let tok = getToken(cval) if tok.tokenType == cttIdent: - if tok.value.equalsIgnoreCase("none"): + let td = ?parseIdent[CSSTextDecoration](tok) + if td == TextDecorationNone: if cvals.len != 1: return err() return ok(s) - elif tok.value.equalsIgnoreCase("underline"): - s.incl(TextDecorationUnderline) - elif tok.value.equalsIgnoreCase("overline"): - s.incl(TextDecorationOverline) - elif tok.value.equalsIgnoreCase("line-through"): - s.incl(TextDecorationLineThrough) - elif tok.value.equalsIgnoreCase("blink"): - s.incl(TextDecorationBlink) - else: - return err() + s.incl(td) return ok(s) -func cssWordBreak(cval: CSSComponentValue): Opt[CSSWordBreak] = - const WordBreakMap = { - "normal": WordBreakNormal, - "break-all": WordBreakBreakAll, - "keep-all": WordBreakKeepAll - } - return cssIdent(WordBreakMap, cval) - -func cssListStyleType(cval: CSSComponentValue): Opt[CSSListStyleType] = - const ListStyleMap = { - "none": ListStyleTypeNone, - "disc": ListStyleTypeDisc, - "circle": ListStyleTypeCircle, - "square": ListStyleTypeSquare, - "decimal": ListStyleTypeDecimal, - "disclosure-open": ListStyleTypeDisclosureOpen, - "disclosure-closed": ListStyleTypeDisclosureClosed, - "cjk-earthly-branch": ListStyleTypeCjkEarthlyBranch, - "cjk-heavenly-stem": ListStyleTypeCjkHeavenlyStem, - "upper-roman": ListStyleTypeUpperRoman, - "lower-roman": ListStyleTypeLowerRoman, - "upper-latin": ListStyleTypeUpperAlpha, - "lower-latin": ListStyleTypeLowerAlpha, - "upper-alpha": ListStyleTypeUpperAlpha, - "lower-alpha": ListStyleTypeUpperAlpha, - "lower-greek": ListStyleTypeLowerGreek, - "hiragana": ListStyleTypeHiragana, - "hiragana-iroha": ListStyleTypeHiraganaIroha, - "katakana": ListStyleTypeKatakana, - "katakana-iroha": ListStyleTypeKatakanaIroha, - "japanese-informal": ListStyleTypeJapaneseInformal - } - return cssIdent(ListStyleMap, cval) - func cssVerticalAlign(cval: CSSComponentValue): Opt[CSSVerticalAlign] = if isToken(cval): let tok = getToken(cval) if tok.tokenType == cttIdent: - const VerticalAlignMap = { - "baseline": VerticalAlignBaseline, - "sub": VerticalAlignSub, - "super": VerticalAlignSuper, - "text-top": VerticalAlignTextBottom, - "middle": VerticalAlignMiddle, - "top": VerticalAlignTop, - "bottom": VerticalAlignBottom - } - let va2 = ?cssIdent(VerticalAlignMap, cval) - return ok(CSSVerticalAlign( - keyword: va2 - )) + let va2 = ?parseIdent[CSSVerticalAlign2](cval) + return ok(CSSVerticalAlign(keyword: va2)) else: return ok(CSSVerticalAlign( keyword: VerticalAlignBaseline, @@ -1029,52 +1040,8 @@ func cssLineHeight(cval: CSSComponentValue): Opt[CSSLength] = return cssLength(tok, has_auto = false) return err() -func cssTextAlign(cval: CSSComponentValue): Opt[CSSTextAlign] = - const TextAlignMap = { - "start": TextAlignStart, - "end": TextAlignEnd, - "left": TextAlignLeft, - "right": TextAlignRight, - "center": TextAlignCenter, - "justify": TextAlignJustify, - "-cha-center": TextAlignChaCenter - } - return cssIdent(TextAlignMap, cval) - -func cssListStylePosition(cval: CSSComponentValue): Opt[CSSListStylePosition] = - const ListStylePositionMap = { - "inside": ListStylePositionInside, - "outside": ListStylePositionOutside - } - return cssIdent(ListStylePositionMap, cval) - -func cssPosition(cval: CSSComponentValue): Opt[CSSPosition] = - const PositionMap = { - "static": PositionStatic, - "relative": PositionRelative, - "absolute": PositionAbsolute, - "fixed": PositionFixed, - "sticky": PositionSticky - } - return cssIdent(PositionMap, cval) - -func cssCaptionSide(cval: CSSComponentValue): Opt[CSSCaptionSide] = - const CaptionSideMap = { - "top": CaptionSideTop, - "bottom": CaptionSideBottom, - "block-start": CaptionSideBlockStart, - "block-end": CaptionSideBlockEnd, - } - return cssIdent(CaptionSideMap, cval) - -func cssBorderCollapse(cval: CSSComponentValue): Opt[CSSBorderCollapse] = - const BorderCollapseMap = { - "collapse": BorderCollapseCollapse, - "separate": BorderCollapseSeparate - } - return cssIdent(BorderCollapseMap, cval) - -func cssCounterReset(cvals: seq[CSSComponentValue]): Opt[seq[CSSCounterReset]] = +func cssCounterReset(cvals: openArray[CSSComponentValue]): + Opt[seq[CSSCounterReset]] = template die = return err() var r = CSSCounterReset() @@ -1152,61 +1119,6 @@ func cssInteger(cval: CSSComponentValue; range: Slice[int]): Opt[int] = return ok(int(tok.nvalue)) return err() -func cssFloat(cval: CSSComponentValue): Opt[CSSFloat] = - const FloatMap = { - "none": FloatNone, - "left": FloatLeft, - "right": FloatRight - } - return cssIdent(FloatMap, cval) - -func cssVisibility(cval: CSSComponentValue): Opt[CSSVisibility] = - const VisibilityMap = { - "visible": VisibilityVisible, - "hidden": VisibilityHidden, - "collapse": VisibilityCollapse - } - return cssIdent(VisibilityMap, cval) - -func cssBoxSizing(cval: CSSComponentValue): Opt[CSSBoxSizing] = - const BoxSizingMap = { - "border-box": BoxSizingBorderBox, - "content-box": BoxSizingContentBox - } - return cssIdent(BoxSizingMap, cval) - -func cssClear(cval: CSSComponentValue): Opt[CSSClear] = - const ClearMap = { - "none": ClearNone, - "left": ClearLeft, - "right": ClearRight, - "both": ClearBoth, - "inline-start": ClearInlineStart, - "inline-end": ClearInlineEnd - } - return cssIdent(ClearMap, cval) - -func cssTextTransform(cval: CSSComponentValue): Opt[CSSTextTransform] = - const TextTransformMap = { - "none": TextTransformNone, - "capitalize": TextTransformCapitalize, - "uppercase": TextTransformUppercase, - "lowercase": TextTransformLowercase, - "full-width": TextTransformFullWidth, - "full-size-kana": TextTransformFullSizeKana, - "-cha-half-width": TextTransformChaHalfWidth - } - return cssIdent(TextTransformMap, cval) - -func cssFlexDirection(cval: CSSComponentValue): Opt[CSSFlexDirection] = - const FlexDirectionMap = { - "row": FlexDirectionRow, - "row-reverse": FlexDirectionRowReverse, - "column": FlexDirectionColumn, - "column-reverse": FlexDirectionColumnReverse, - } - return cssIdent(FlexDirectionMap, cval) - func cssNumber(cval: CSSComponentValue; positive: bool): Opt[float64] = if isToken(cval): let tok = getToken(cval) @@ -1215,15 +1127,7 @@ func cssNumber(cval: CSSComponentValue; positive: bool): Opt[float64] = return ok(tok.nvalue) return err() -func cssFlexWrap(cval: CSSComponentValue): Opt[CSSFlexWrap] = - const FlexWrapMap = { - "nowrap": FlexWrapNowrap, - "wrap": FlexWrapWrap, - "wrap-reverse": FlexWrapWrapReverse - } - return cssIdent(FlexWrapMap, cval) - -proc parseValue(cvals: seq[CSSComponentValue]; t: CSSPropertyType): +proc parseValue(cvals: openArray[CSSComponentValue]; t: CSSPropertyType): Opt[CSSComputedValue] = var i = 0 cvals.skipWhitespace(i) @@ -1249,10 +1153,10 @@ proc parseValue(cvals: seq[CSSComponentValue]; t: CSSPropertyType): #TODO content for flex-basis else: return_new length, ?cssLength(cval) - of cvtFontStyle: return_new fontstyle, ?cssFontStyle(cval) - of cvtDisplay: return_new display, ?cssDisplay(cval) + of cvtFontStyle: return_new fontstyle, ?parseIdent[CSSFontStyle](cval) + of cvtDisplay: return_new display, ?parseIdent[CSSDisplay](cval) of cvtContent: return_new content, cssContent(cvals) - of cvtWhiteSpace: return_new whitespace, ?cssWhiteSpace(cval) + of cvtWhiteSpace: return_new whitespace, ?parseIdent[CSSWhiteSpace](cval) of cvtInteger: case t of cptFontWeight: return_new integer, ?cssFontWeight(cval) @@ -1260,15 +1164,17 @@ proc parseValue(cvals: seq[CSSComponentValue]; t: CSSPropertyType): of cptChaRowspan: return_new integer, ?cssInteger(cval, 0 .. 65534) else: assert false of cvtTextDecoration: return_new textdecoration, ?cssTextDecoration(cvals) - of cvtWordBreak: return_new wordbreak, ?cssWordBreak(cval) - of cvtListStyleType: return_new liststyletype, ?cssListStyleType(cval) + of cvtWordBreak: return_new wordbreak, ?parseIdent[CSSWordBreak](cval) + of cvtListStyleType: + return_new liststyletype, ?parseIdent[CSSListStyleType](cval) of cvtVerticalAlign: return_new verticalalign, ?cssVerticalAlign(cval) - of cvtTextAlign: return_new textalign, ?cssTextAlign(cval) + of cvtTextAlign: return_new textalign, ?parseIdent[CSSTextAlign](cval) of cvtListStylePosition: - return_new liststyleposition, ?cssListStylePosition(cval) - of cvtPosition: return_new position, ?cssPosition(cval) - of cvtCaptionSide: return_new captionside, ?cssCaptionSide(cval) - of cvtBorderCollapse: return_new bordercollapse, ?cssBorderCollapse(cval) + return_new liststyleposition, ?parseIdent[CSSListStylePosition](cval) + of cvtPosition: return_new position, ?parseIdent[CSSPosition](cval) + of cvtCaptionSide: return_new captionside, ?parseIdent[CSSCaptionSide](cval) + of cvtBorderCollapse: + return_new bordercollapse, ?parseIdent[CSSBorderCollapse](cval) of cvtLength2: let a = ?cssAbsoluteLength(cval) cvals.skipWhitespace(i) @@ -1277,14 +1183,16 @@ proc parseValue(cvals: seq[CSSComponentValue]; t: CSSPropertyType): of cvtQuotes: return_new quotes, ?cssQuotes(cvals) of cvtCounterReset: return_new counterreset, ?cssCounterReset(cvals) of cvtImage: return_new image, ?cssImage(cval) - of cvtFloat: return_new float, ?cssFloat(cval) - of cvtVisibility: return_new visibility, ?cssVisibility(cval) - of cvtBoxSizing: return_new boxsizing, ?cssBoxSizing(cval) - of cvtClear: return_new clear, ?cssClear(cval) - of cvtTextTransform: return_new texttransform, ?cssTextTransform(cval) + of cvtFloat: return_new float, ?parseIdent[CSSFloat](cval) + of cvtVisibility: return_new visibility, ?parseIdent[CSSVisibility](cval) + of cvtBoxSizing: return_new boxsizing, ?parseIdent[CSSBoxSizing](cval) + of cvtClear: return_new clear, ?parseIdent[CSSClear](cval) + of cvtTextTransform: + return_new texttransform, ?parseIdent[CSSTextTransform](cval) of cvtBgcolorIsCanvas: return err() # internal value - of cvtFlexDirection: return_new flexdirection, ?cssFlexDirection(cval) - of cvtFlexWrap: return_new flexwrap, ?cssFlexWrap(cval) + of cvtFlexDirection: + return_new flexdirection, ?parseIdent[CSSFlexDirection](cval) + of cvtFlexWrap: return_new flexwrap, ?parseIdent[CSSFlexWrap](cval) of cvtNumber: return_new number, ?cssNumber(cval, t == cptFlexGrow) of cvtNone: return err() @@ -1348,7 +1256,7 @@ template getDefault(t: CSSPropertyType): CSSComputedValue = {.cast(noSideEffect).}: defaultTable[t] -func lengthShorthand(cvals: seq[CSSComponentValue]; +func lengthShorthand(cvals: openArray[CSSComponentValue]; props: array[4, CSSPropertyType]; global: CSSGlobalType; has_auto = true): Opt[seq[CSSComputedEntry]] = var res: seq[CSSComputedEntry] = @[] @@ -1395,8 +1303,8 @@ const PropertyPaddingSpec = [ cptPaddingTop, cptPaddingRight, cptPaddingBottom, cptPaddingLeft ] -proc getComputedValues(res: var seq[CSSComputedEntry]; name: string; - cvals: seq[CSSComponentValue]): Err[void] = +proc parseComputedValues*(res: var seq[CSSComputedEntry]; name: string; + cvals: openArray[CSSComponentValue]): Err[void] = var i = 0 cvals.skipWhitespace(i) if i >= cvals.len: @@ -1446,12 +1354,12 @@ proc getComputedValues(res: var seq[CSSComputedEntry]; name: string; for tok in cvals: if tok == cttWhitespace: continue - if (let r = cssListStylePosition(tok); r.isOk): + if (let r = parseIdent[CSSListStylePosition](tok); r.isOk): positionVal = CSSComputedValue( v: cvtListStylePosition, liststyleposition: r.get ) - elif (let r = cssListStyleType(tok); r.isOk): + elif (let r = parseIdent[CSSListStyleType](tok); r.isOk): typeVal = CSSComputedValue( v: cvtListStyleType, liststyletype: r.get @@ -1510,14 +1418,14 @@ proc getComputedValues(res: var seq[CSSComputedEntry]; name: string; cvals.skipWhitespace(i) if i >= cvals.len: return err() - if (let dir = cssFlexDirection(cvals[i]); dir.isSome): + if (let dir = parseIdent[CSSFlexDirection](cvals[i]); dir.isSome): # flex-direction let val = CSSComputedValue(v: cvtFlexDirection, flexdirection: dir.get) res.add((cptFlexDirection, val, global)) inc i cvals.skipWhitespace(i) if i < cvals.len: - let wrap = ?cssFlexWrap(cvals[i]) + let wrap = ?parseIdent[CSSFlexWrap](cvals[i]) let val = CSSComputedValue(v: cvtFlexWrap, flexwrap: wrap) res.add((cptFlexWrap, val, global)) else: @@ -1525,19 +1433,21 @@ proc getComputedValues(res: var seq[CSSComputedEntry]; name: string; res.add((cptFlexWrap, getDefault(cptFlexWrap), global)) return ok() -proc getComputedValues(d: CSSDeclaration): seq[CSSComputedEntry] = +proc parseComputedValues*(name: string; value: seq[CSSComponentValue]): + seq[CSSComputedEntry] = var res: seq[CSSComputedEntry] = @[] - if res.getComputedValues(d.name, d.value).isOk: + if res.parseComputedValues(name, value).isOk: return res return @[] proc addValues*(builder: var CSSComputedValuesBuilder; decls: seq[CSSDeclaration]; origin: CSSOrigin) = for decl in decls: + let vals = parseComputedValues(decl.name, decl.value) if decl.important: - builder.importantProperties[origin].add(getComputedValues(decl)) + builder.importantProperties[origin].add(vals) else: - builder.normalProperties[origin].add(getComputedValues(decl)) + builder.normalProperties[origin].add(vals) proc applyValue(vals: CSSComputedValues; entry: CSSComputedEntry; parent: CSSComputedValues; previousOrigin: CSSComputedValues) = diff --git a/src/html/dom.nim b/src/html/dom.nim index 0c009b6b..e7d7e28e 100644 --- a/src/html/dom.nim +++ b/src/html/dom.nim @@ -320,7 +320,7 @@ type HTMLCanvasElement* = ref object of HTMLElement ctx2d: CanvasRenderingContext2D - bitmap: Bitmap + bitmap*: Bitmap DrawingState = object # CanvasTransform @@ -436,8 +436,7 @@ proc resetState(state: var DrawingState) = state.path = newPath() proc create2DContext*(jctx: JSContext; target: HTMLCanvasElement; - options: Option[JSValue]): - CanvasRenderingContext2D = + options: Option[JSValue]): CanvasRenderingContext2D = let ctx = CanvasRenderingContext2D( bitmap: target.bitmap, canvas: target @@ -2577,7 +2576,8 @@ proc newHTMLElement*(document: Document; localName: CAtom; of TAG_LABEL: result = HTMLLabelElement() of TAG_CANVAS: - result = HTMLCanvasElement() + let bitmap = if document.scriptingEnabled: newBitmap(300, 150) else: nil + result = HTMLCanvasElement(bitmap: bitmap) of TAG_IMG: result = HTMLImageElement() of TAG_VIDEO: @@ -2750,10 +2750,15 @@ func item(this: CSSStyleDeclaration; u: uint32): Option[string] = return some(this.decls[int(u)].name) return none(string) -proc getPropertyValue(this: CSSStyleDeclaration; s: string): string = - for decl in this.decls: +func find(this: CSSStyleDeclaration; s: string): int = + for i, decl in this.decls: if decl.name == s: - return $decl.value + return i + return -1 + +proc getPropertyValue(this: CSSStyleDeclaration; s: string): string = + if (let i = this.find(s); i != -1): + return $this.decls[i].value return "" # https://drafts.csswg.org/cssom/#idl-attribute-to-css-property @@ -2778,6 +2783,33 @@ proc getter[T: uint32|string](this: CSSStyleDeclaration; u: T): return some(this.getPropertyValue(u)) return none(string) +proc setValue(this: CSSStyleDeclaration; i: int; cvals: seq[CSSComponentValue]): + Err[void] = + if i notin 0 .. this.decls.high: + return err() + var dummy: seq[CSSComputedEntry] + ?parseComputedValues(dummy, this.decls[i].name, cvals) + this.decls[i].value = cvals + return ok() + +proc setter[T: uint32|string](this: CSSStyleDeclaration; u: T; + value: string) {.jssetprop.} = + let cvals = parseListOfComponentValues(newStringStream(value)) + when u is uint32: + if this.setValue(int(u), cvals).isErr: + return + else: + if (let i = this.find(u); i != -1): + if this.setValue(i, cvals).isErr: + return + else: + var dummy: seq[CSSComputedEntry] + let val0 = parseComputedValues(dummy, u, cvals) + if val0.isErr: + return + this.decls.add(CSSDeclaration(name: u, value: cvals)) + this.element.attr(satStyle, $this.decls) + proc style*(element: Element): CSSStyleDeclaration {.jsfget.} = if element.style_cached == nil: element.style_cached = CSSStyleDeclaration(element: element) @@ -2952,7 +2984,7 @@ proc reflectAttrs(element: Element; name: CAtom; value: string) = let area = HTMLAreaElement(element) area.reflect_domtoklist satRel, relList of TAG_CANVAS: - if element.scriptingEnabled and (name == satWidth or name == satHeight): + if element.scriptingEnabled and name in {satWidth, satHeight}: let w = element.attrul(satWidth).get(300) let h = element.attrul(satHeight).get(150) let canvas = HTMLCanvasElement(element) |