mirror of https://github.com/buggins/dlangui.git
css parser
This commit is contained in:
parent
9f768c85af
commit
4f0cced6b4
|
@ -5,6 +5,9 @@ This module contains implementation of CSS support - Cascading Style Sheets.
|
||||||
|
|
||||||
Port of CoolReader Engine written in C++.
|
Port of CoolReader Engine written in C++.
|
||||||
|
|
||||||
|
Supports subset of CSS standards.
|
||||||
|
|
||||||
|
|
||||||
Synopsis:
|
Synopsis:
|
||||||
|
|
||||||
----
|
----
|
||||||
|
@ -324,7 +327,7 @@ public:
|
||||||
@property CssSelectorRule next() { return _next; }
|
@property CssSelectorRule next() { return _next; }
|
||||||
@property void next(CssSelectorRule v) { _next = v; }
|
@property void next(CssSelectorRule v) { _next = v; }
|
||||||
/// check condition for node
|
/// check condition for node
|
||||||
bool check(ref Node node) {
|
bool check(ref Node node) const {
|
||||||
if (!node || !node.parent)
|
if (!node || !node.parent)
|
||||||
return false;
|
return false;
|
||||||
switch (_type) with (CssSelectorRuleType) {
|
switch (_type) with (CssSelectorRuleType) {
|
||||||
|
@ -422,17 +425,17 @@ public:
|
||||||
@property void id(elem_id id) { _id = id; }
|
@property void id(elem_id id) { _id = id; }
|
||||||
|
|
||||||
this() { }
|
this() { }
|
||||||
|
|
||||||
~this() {
|
~this() {
|
||||||
//if (_next)
|
//if (_next)
|
||||||
// destroy(_next);
|
// destroy(_next);
|
||||||
//if (_rules)
|
|
||||||
// destroy(_rules);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void insertRuleStart(CssSelectorRule rule) {
|
void insertRuleStart(CssSelectorRule rule) {
|
||||||
rule.next = _rules;
|
rule.next = _rules;
|
||||||
_rules = rule;
|
_rules = rule;
|
||||||
}
|
}
|
||||||
|
|
||||||
void insertRuleAfterStart(CssSelectorRule rule) {
|
void insertRuleAfterStart(CssSelectorRule rule) {
|
||||||
if (!_rules) {
|
if (!_rules) {
|
||||||
_rules = rule;
|
_rules = rule;
|
||||||
|
@ -442,26 +445,26 @@ public:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@property uint tagId() { return _id; }
|
/// check if selector rules match this node
|
||||||
bool check(const Node node) const {
|
bool check(Node node) const {
|
||||||
// TODO
|
CssSelectorRule rule = cast(CssSelectorRule)_rules;
|
||||||
return false;
|
while (rule && node) {
|
||||||
|
if (!rule.check(node))
|
||||||
|
return false;
|
||||||
|
rule = rule.next;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// apply to style if selector matches
|
/// apply to style if selector matches
|
||||||
void apply(const Node node, CssStyle style) const
|
void apply(Node node, CssStyle style) const {
|
||||||
{
|
|
||||||
if (check(node))
|
if (check(node))
|
||||||
_decl.apply(style);
|
_decl.apply(style);
|
||||||
}
|
}
|
||||||
void setDeclaration(CssDeclaration decl) {
|
|
||||||
|
void setDeclaration(CssDeclaration decl) {
|
||||||
_decl = decl;
|
_decl = decl;
|
||||||
}
|
}
|
||||||
int getSpecificity() {
|
|
||||||
return _specificity;
|
|
||||||
}
|
|
||||||
@property CssSelector next() { return _next; }
|
|
||||||
@property void next(CssSelector next) { _next = next; }
|
|
||||||
//lUInt32 getHash();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
struct CssDeclItem {
|
struct CssDeclItem {
|
||||||
|
@ -555,6 +558,71 @@ class CssDeclaration {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// CSS Style Sheet
|
||||||
|
class StyleSheet {
|
||||||
|
private:
|
||||||
|
CssSelector[elem_id] _selectorMap;
|
||||||
|
int _len;
|
||||||
|
public:
|
||||||
|
/// clears stylesheet
|
||||||
|
void clear() {
|
||||||
|
_selectorMap = null;
|
||||||
|
_len = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// count of selectors in stylesheet
|
||||||
|
@property int length() { return _len; }
|
||||||
|
|
||||||
|
/// add selector to stylesheet
|
||||||
|
void add(CssSelector selector) {
|
||||||
|
elem_id id = selector.id;
|
||||||
|
if (auto p = id in _selectorMap) {
|
||||||
|
for (;;) {
|
||||||
|
if (!(*p) || (*p)._specificity < selector._specificity) {
|
||||||
|
selector._next = (*p);
|
||||||
|
(*p) = selector;
|
||||||
|
_len++;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
p = &((*p)._next);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// first selector for this id
|
||||||
|
_selectorMap[id] = selector;
|
||||||
|
_len++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// apply stylesheet to node style
|
||||||
|
void apply(Node node, CssStyle style) {
|
||||||
|
elem_id id = node.id;
|
||||||
|
CssSelector selector_0, selector_id;
|
||||||
|
if (auto p = 0 in _selectorMap)
|
||||||
|
selector_0 = *p;
|
||||||
|
if (id) {
|
||||||
|
if (auto p = id in _selectorMap)
|
||||||
|
selector_id = *p;
|
||||||
|
}
|
||||||
|
for (;;) {
|
||||||
|
if (selector_0) {
|
||||||
|
if (!selector_id || selector_id._specificity < selector_0._specificity) {
|
||||||
|
selector_0.apply(node, style);
|
||||||
|
selector_0 = selector_0._next;
|
||||||
|
} else {
|
||||||
|
selector_id.apply(node, style);
|
||||||
|
selector_id = selector_id._next;
|
||||||
|
}
|
||||||
|
} else if (selector_id) {
|
||||||
|
selector_id.apply(node, style);
|
||||||
|
selector_id = selector_id._next;
|
||||||
|
} else {
|
||||||
|
// end of lists
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
unittest {
|
unittest {
|
||||||
CssStyle style = new CssStyle();
|
CssStyle style = new CssStyle();
|
||||||
CssWhiteSpace whiteSpace = CssWhiteSpace.inherit;
|
CssWhiteSpace whiteSpace = CssWhiteSpace.inherit;
|
||||||
|
|
|
@ -20,27 +20,42 @@ private char skip(ref string src, int count = 1) {
|
||||||
return src[0];
|
return src[0];
|
||||||
}
|
}
|
||||||
|
|
||||||
/// returns first char of string or 0 if end of string reached
|
/// returns char of string at specified position (first by default) or 0 if end of string reached
|
||||||
private char peek(string str) {
|
private char peek(string str, int offset = 0) {
|
||||||
return str.empty ? 0 : str[0];
|
return offset >= str.length ? 0 : str[offset];
|
||||||
}
|
}
|
||||||
|
|
||||||
/// skip spaces, move to new location, return first character in string, 0 if end of string reached
|
/// skip spaces, move to new location, return first character in string, 0 if end of string reached
|
||||||
private char skipSpaces(ref string src) {
|
private char skipSpaces(ref string str)
|
||||||
for(;;) {
|
{
|
||||||
if (src.empty) {
|
string oldpos = str;
|
||||||
src = null;
|
for (;;) {
|
||||||
|
char ch = str.peek;
|
||||||
|
if (!ch)
|
||||||
return 0;
|
return 0;
|
||||||
|
while (ch == ' ' || ch == '\t' || ch == '\n' || ch == '\r')
|
||||||
|
ch = str.skip;
|
||||||
|
if (str.peek == '/' && str.peek(1) == '*') {
|
||||||
|
// comment found
|
||||||
|
str.skip(2);
|
||||||
|
while (str.peek && (str.peek(0) != '*' || str.peek(1) != '/'))
|
||||||
|
str.skip;
|
||||||
|
if (str.peek == '*' && str.peek(1) == '/' )
|
||||||
|
str.skip(2);
|
||||||
}
|
}
|
||||||
char ch = src[0];
|
ch = str.peek;
|
||||||
if (ch == ' ' || ch == '\t' || ch == '\r' || ch == '\n') {
|
while (ch == ' ' || ch == '\t' || ch == '\n' || ch == '\r')
|
||||||
src.skip;
|
ch = str.skip;
|
||||||
} else {
|
if (oldpos.ptr is str.ptr)
|
||||||
return src.empty ? 0 : src[0];
|
break;
|
||||||
}
|
if (str.empty)
|
||||||
|
return 0;
|
||||||
|
oldpos = str;
|
||||||
}
|
}
|
||||||
|
return str.peek;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private bool isIdentChar(char ch) {
|
private bool isIdentChar(char ch) {
|
||||||
return (ch >= 'A' && ch <='Z') || (ch >= 'a' && ch <='z') || (ch == '-') || (ch == '_');
|
return (ch >= 'A' && ch <='Z') || (ch >= 'a' && ch <='z') || (ch == '-') || (ch == '_');
|
||||||
}
|
}
|
||||||
|
@ -62,7 +77,7 @@ private string parseIdent(ref string src) {
|
||||||
|
|
||||||
private bool skipChar(ref string src, char ch) {
|
private bool skipChar(ref string src, char ch) {
|
||||||
src.skipSpaces;
|
src.skipSpaces;
|
||||||
if (src.length > 0 && src[0] == ch) {
|
if (src.peek == ch) {
|
||||||
src.skip;
|
src.skip;
|
||||||
src.skipSpaces;
|
src.skipSpaces;
|
||||||
return true;
|
return true;
|
||||||
|
@ -506,6 +521,7 @@ private CssSelectorRule parseAttr(ref string str, Document doc)
|
||||||
return rule;
|
return rule;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Parse css properties declaration either in {} or w/o {} - e.g. { width: 40%; margin-top: 3px } -- returns null if parse error occured or property list is empty
|
||||||
CssDeclaration parseCssDeclaration(ref string src, bool mustBeInBrackets = true) {
|
CssDeclaration parseCssDeclaration(ref string src, bool mustBeInBrackets = true) {
|
||||||
if (!src.skipSpaces)
|
if (!src.skipSpaces)
|
||||||
return null;
|
return null;
|
||||||
|
@ -688,7 +704,7 @@ CssDeclaration parseCssDeclaration(ref string src, bool mustBeInBrackets = true)
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// parse Css selector, return selector object if parsed ok
|
/// parse Css selector, return selector object if parsed ok, null if error occured
|
||||||
CssSelector parseCssSelector(ref string str, Document doc) {
|
CssSelector parseCssSelector(ref string str, Document doc) {
|
||||||
if (str.empty)
|
if (str.empty)
|
||||||
return null;
|
return null;
|
||||||
|
@ -756,6 +772,17 @@ CssSelector parseCssSelector(ref string str, Document doc) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// skips until } or end of string, returns true if some characters left in string
|
||||||
|
private bool skipUntilEndOfRule(ref string str)
|
||||||
|
{
|
||||||
|
while (str.length && str[0] != '}')
|
||||||
|
str.skip;
|
||||||
|
if (str.peek == '}')
|
||||||
|
str.skip;
|
||||||
|
return !str.empty;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
unittest {
|
unittest {
|
||||||
Document doc = new Document();
|
Document doc = new Document();
|
||||||
string str;
|
string str;
|
||||||
|
@ -768,4 +795,72 @@ unittest {
|
||||||
str = ".myclass + div { }";
|
str = ".myclass + div { }";
|
||||||
assert(parseCssSelector(str, doc) !is null);
|
assert(parseCssSelector(str, doc) !is null);
|
||||||
assert(parseCssDeclaration(str, true) is null); // empty property decl
|
assert(parseCssDeclaration(str, true) is null); // empty property decl
|
||||||
}
|
destroy(doc);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// parse stylesheet text
|
||||||
|
bool parseStyleSheet(StyleSheet sheet, Document doc, string str) {
|
||||||
|
bool res = false;
|
||||||
|
for(;;) {
|
||||||
|
if (!str.skipSpaces)
|
||||||
|
break;
|
||||||
|
CssSelector[] selectors;
|
||||||
|
for(;;) {
|
||||||
|
CssSelector selector = parseCssSelector(str, doc);
|
||||||
|
if (!selector)
|
||||||
|
break;
|
||||||
|
selectors ~= selector;
|
||||||
|
str.skipChar(',');
|
||||||
|
}
|
||||||
|
if (selectors.length) {
|
||||||
|
if (CssDeclaration decl = parseCssDeclaration(str, true)) {
|
||||||
|
foreach(item; selectors) {
|
||||||
|
item.setDeclaration(decl);
|
||||||
|
sheet.add(item);
|
||||||
|
res = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!skipUntilEndOfRule(str))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
unittest {
|
||||||
|
string src = q{
|
||||||
|
body { width: 50%; color: blue }
|
||||||
|
body > div, body > section {
|
||||||
|
/* some comment
|
||||||
|
goes here */
|
||||||
|
font-family: serif;
|
||||||
|
background-color: yellow;
|
||||||
|
}
|
||||||
|
section {
|
||||||
|
margin-top: 5px
|
||||||
|
}
|
||||||
|
};
|
||||||
|
Document doc = new Document();
|
||||||
|
StyleSheet sheet = new StyleSheet();
|
||||||
|
assert(parseStyleSheet(sheet, doc, src));
|
||||||
|
assert(sheet.length == 2);
|
||||||
|
// check appending of additional source text
|
||||||
|
assert(parseStyleSheet(sheet, doc, "pre { white-space: pre }"));
|
||||||
|
assert(sheet.length == 3);
|
||||||
|
destroy(doc);
|
||||||
|
}
|
||||||
|
|
||||||
|
unittest {
|
||||||
|
Document doc = new Document();
|
||||||
|
StyleSheet sheet = new StyleSheet();
|
||||||
|
assert(parseStyleSheet(sheet, doc, "* { color: #aaa }"));
|
||||||
|
assert(sheet.length == 1);
|
||||||
|
assert(parseStyleSheet(sheet, doc, "div, p { display: block }"));
|
||||||
|
assert(sheet.length == 3);
|
||||||
|
// check appending of additional source text
|
||||||
|
assert(parseStyleSheet(sheet, doc, "pre { white-space: pre }"));
|
||||||
|
assert(sheet.length == 4);
|
||||||
|
assert(parseStyleSheet(sheet, doc, "pre { font-size: 120% }"));
|
||||||
|
assert(sheet.length == 5);
|
||||||
|
destroy(doc);
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue