mirror of https://github.com/buggins/dlangui.git
CSS parser
This commit is contained in:
parent
2037fcfe23
commit
1686fde76d
|
@ -247,6 +247,7 @@
|
|||
<ItemGroup>
|
||||
<Compile Include="src\dlangui\core\collections.d" />
|
||||
<Compile Include="src\dlangui\core\css.d" />
|
||||
<Compile Include="src\dlangui\core\cssparser.d" />
|
||||
<Compile Include="src\dlangui\core\dom.d" />
|
||||
<Compile Include="src\dlangui\core\editable.d" />
|
||||
<Compile Include="src\dlangui\core\events.d" />
|
||||
|
@ -338,12 +339,6 @@
|
|||
<Compile Include="deps\DerelictUtil\source\derelict\util\system.d" />
|
||||
<Compile Include="deps\DerelictUtil\source\derelict\util\wintypes.d" />
|
||||
<Compile Include="deps\DerelictUtil\source\derelict\util\xtypes.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\frustum.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\linalg.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\math.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\plane.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\util.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\aabb.d" />
|
||||
<Compile Include="3rdparty\fontconfig\functions.d" />
|
||||
<Compile Include="3rdparty\fontconfig\package.d" />
|
||||
<Compile Include="3rdparty\fontconfig\types.d" />
|
||||
|
|
|
@ -106,6 +106,7 @@
|
|||
<Compile Include="src\dlangui\core\collections.d" />
|
||||
<Compile Include="src\dlangui\core\config.d" />
|
||||
<Compile Include="src\dlangui\core\css.d" />
|
||||
<Compile Include="src\dlangui\core\cssparser.d" />
|
||||
<Compile Include="src\dlangui\core\dom.d" />
|
||||
<Compile Include="src\dlangui\core\editable.d" />
|
||||
<Compile Include="src\dlangui\core\events.d" />
|
||||
|
@ -195,10 +196,6 @@
|
|||
<Compile Include="deps\DerelictFT\source\derelict\freetype\ft.d" />
|
||||
<Compile Include="deps\DerelictFT\source\derelict\freetype\functions.d" />
|
||||
<Compile Include="deps\DerelictFT\source\derelict\freetype\types.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\linalg.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\math.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\util.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\plane.d" />
|
||||
<Compile Include="src\dlangui\graphics\xpm\colors.d" />
|
||||
<Compile Include="3rdparty\fontconfig\functions.d" />
|
||||
<Compile Include="3rdparty\fontconfig\package.d" />
|
||||
|
|
|
@ -15,7 +15,6 @@
|
|||
<Path>deps\DerelictSDL2\source</Path>
|
||||
<Path>deps\DerelictGL3\source</Path>
|
||||
<Path>deps\DerelictUtil\source</Path>
|
||||
<Path>deps\gl3n</Path>
|
||||
<Path>deps\dlib</Path>
|
||||
</Includes>
|
||||
</Includes>
|
||||
|
@ -83,6 +82,7 @@
|
|||
<Compile Include="src\dlangui\package.d" />
|
||||
<Compile Include="src\dlangui\core\collections.d" />
|
||||
<Compile Include="src\dlangui\core\css.d" />
|
||||
<Compile Include="src\dlangui\core\cssparser.d" />
|
||||
<Compile Include="src\dlangui\core\dom.d" />
|
||||
<Compile Include="src\dlangui\core\editable.d" />
|
||||
<Compile Include="src\dlangui\core\events.d" />
|
||||
|
@ -120,6 +120,7 @@
|
|||
<Compile Include="src\dlangui\graphics\xpm\colors.d" />
|
||||
<Compile Include="src\dlangui\graphics\xpm\reader.d" />
|
||||
<Compile Include="src\dlangui\platforms\common\platform.d" />
|
||||
<Compile Include="src\dlangui\platforms\common\startup.d" />
|
||||
<Compile Include="src\dlangui\platforms\sdl\sdlapp.d" />
|
||||
<Compile Include="src\dlangui\platforms\windows\win32drawbuf.d" />
|
||||
<Compile Include="src\dlangui\platforms\windows\win32fonts.d" />
|
||||
|
@ -200,6 +201,9 @@
|
|||
<Compile Include="3rdparty\win32\winver.d" />
|
||||
<Compile Include="3rdparty\win32\ws2tcpip.d" />
|
||||
<Compile Include="3rdparty\win32\wtypes.d" />
|
||||
<Compile Include="3rdparty\fontconfig\functions.d" />
|
||||
<Compile Include="3rdparty\fontconfig\package.d" />
|
||||
<Compile Include="3rdparty\fontconfig\types.d" />
|
||||
<Compile Include="deps\DerelictFT\source\derelict\freetype\ft.d" />
|
||||
<Compile Include="deps\DerelictFT\source\derelict\freetype\functions.d" />
|
||||
<Compile Include="deps\DerelictFT\source\derelict\freetype\types.d" />
|
||||
|
@ -231,51 +235,6 @@
|
|||
<Compile Include="deps\DerelictUtil\source\derelict\util\system.d" />
|
||||
<Compile Include="deps\DerelictUtil\source\derelict\util\wintypes.d" />
|
||||
<Compile Include="deps\DerelictUtil\source\derelict\util\xtypes.d" />
|
||||
<Compile Include="deps\dlib\dlib\image\io\bmp.d" />
|
||||
<Compile Include="deps\dlib\dlib\image\io\idct.d" />
|
||||
<Compile Include="deps\dlib\dlib\image\io\io.d" />
|
||||
<Compile Include="deps\dlib\dlib\image\io\jpeg.d" />
|
||||
<Compile Include="deps\dlib\dlib\image\io\png.d" />
|
||||
<Compile Include="deps\dlib\dlib\image\io\tga.d" />
|
||||
<Compile Include="deps\dlib\dlib\image\io\utils.d" />
|
||||
<Compile Include="deps\dlib\dlib\image\arithmetics.d" />
|
||||
<Compile Include="deps\dlib\dlib\image\color.d" />
|
||||
<Compile Include="deps\dlib\dlib\image\image.d" />
|
||||
<Compile Include="deps\dlib\dlib\core\bitio.d" />
|
||||
<Compile Include="deps\dlib\dlib\core\compound.d" />
|
||||
<Compile Include="deps\dlib\dlib\core\memory.d" />
|
||||
<Compile Include="deps\dlib\dlib\core\oop.d" />
|
||||
<Compile Include="deps\dlib\dlib\core\tuple.d" />
|
||||
<Compile Include="deps\dlib\dlib\filesystem\delegaterange.d" />
|
||||
<Compile Include="deps\dlib\dlib\filesystem\dirrange.d" />
|
||||
<Compile Include="deps\dlib\dlib\filesystem\filesystem.d" />
|
||||
<Compile Include="deps\dlib\dlib\filesystem\local.d" />
|
||||
<Compile Include="deps\dlib\dlib\functional\combinators.d" />
|
||||
<Compile Include="deps\dlib\dlib\functional\range.d" />
|
||||
<Compile Include="deps\dlib\dlib\math\decomposition.d" />
|
||||
<Compile Include="deps\dlib\dlib\math\interpolation.d" />
|
||||
<Compile Include="deps\dlib\dlib\math\linsolve.d" />
|
||||
<Compile Include="deps\dlib\dlib\math\matrix.d" />
|
||||
<Compile Include="deps\dlib\dlib\math\utils.d" />
|
||||
<Compile Include="deps\dlib\dlib\math\vector.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\aabb.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\frustum.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\interpolate.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\linalg.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\math.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\plane.d" />
|
||||
<Compile Include="deps\gl3n\gl3n\util.d" />
|
||||
<Compile Include="deps\dlib\dlib\core\stream.d" />
|
||||
<Compile Include="deps\dlib\dlib\filesystem\windows\common.d" />
|
||||
<Compile Include="deps\dlib\dlib\filesystem\windows\directory.d" />
|
||||
<Compile Include="deps\dlib\dlib\filesystem\windows\file.d" />
|
||||
<Compile Include="deps\dlib\dlib\coding\huffman.d" />
|
||||
<Compile Include="deps\dlib\dlib\coding\zlib.d" />
|
||||
<Compile Include="deps\dlib\dlib\container\array.d" />
|
||||
<Compile Include="3rdparty\fontconfig\functions.d" />
|
||||
<Compile Include="3rdparty\fontconfig\package.d" />
|
||||
<Compile Include="3rdparty\fontconfig\types.d" />
|
||||
<Compile Include="src\dlangui\platforms\common\startup.d" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<None Include="3rdparty\win32\makefile" />
|
||||
|
|
|
@ -738,6 +738,7 @@
|
|||
<File path="src\dlangui\core\collections.d" />
|
||||
<File path="src\dlangui\core\config.d" />
|
||||
<File path="src\dlangui\core\css.d" />
|
||||
<File path="src\dlangui\core\cssparser.d" />
|
||||
<File path="src\dlangui\core\dom.d" />
|
||||
<File path="src\dlangui\core\editable.d" />
|
||||
<File path="src\dlangui\core\events.d" />
|
||||
|
|
|
@ -367,7 +367,7 @@ public:
|
|||
case attrhas: // E[foo~="value"]
|
||||
// one of space separated values
|
||||
string val = node.attrValue(Ns.any, _attrid);
|
||||
int p = val.indexOf(_value);
|
||||
int p = cast(int)val.indexOf(_value);
|
||||
if (p < 0)
|
||||
return false;
|
||||
if ( (p > 0 && val[p - 1] != ' ')
|
||||
|
@ -398,6 +398,8 @@ public:
|
|||
}
|
||||
}
|
||||
|
||||
import dlangui.core.cssparser;
|
||||
|
||||
/** simple CSS selector
|
||||
|
||||
Currently supports only element name and universal selector.
|
||||
|
@ -413,6 +415,20 @@ private:
|
|||
int _specificity;
|
||||
CssSelector _next;
|
||||
CssSelectorRule _rules;
|
||||
public:
|
||||
/// get element tag id (0 - any tag)
|
||||
@property elem_id id() { return _id; }
|
||||
/// set element tag id (0 - any tag)
|
||||
@property void id(elem_id id) { _id = id; }
|
||||
|
||||
this() { }
|
||||
~this() {
|
||||
//if (_next)
|
||||
// destroy(_next);
|
||||
//if (_rules)
|
||||
// destroy(_rules);
|
||||
}
|
||||
|
||||
void insertRuleStart(CssSelectorRule rule) {
|
||||
rule.next = _rules;
|
||||
_rules = rule;
|
||||
|
@ -425,83 +441,6 @@ private:
|
|||
_rules.next = rule;
|
||||
}
|
||||
}
|
||||
public:
|
||||
this(CssSelector v) {
|
||||
_id = v._id;
|
||||
_decl = v._decl;
|
||||
_specificity = v._specificity;
|
||||
}
|
||||
this() { }
|
||||
~this() {
|
||||
//if (_next)
|
||||
// destroy(_next);
|
||||
//if (_rules)
|
||||
// destroy(_rules);
|
||||
}
|
||||
|
||||
bool parse(ref string str, Document doc) { //, lxmlDocBase * doc
|
||||
if (str.empty)
|
||||
return false;
|
||||
for (;;)
|
||||
{
|
||||
if (!skipSpaces(str))
|
||||
return false;
|
||||
char ch = str[0];
|
||||
string ident = parseIdent(str);
|
||||
if (ch == '*') { // universal selector
|
||||
str = str[1 .. $];
|
||||
skipSpaces(str);
|
||||
_id = 0;
|
||||
} else if (ch == '.') { // classname follows
|
||||
_id = 0;
|
||||
// will be parsed as attribute
|
||||
} else if (!ident.empty) {
|
||||
// ident
|
||||
_id = doc.tagId(ident);
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
if (ch == ',' || ch == '{' )
|
||||
return true;
|
||||
// one or more attribute rules
|
||||
bool attr_rule = false;
|
||||
while (ch == '[' || ch == '.' || ch == '#') {
|
||||
CssSelectorRule rule = parseAttr(str, doc);
|
||||
if (!rule)
|
||||
return false;
|
||||
insertRuleStart(rule); //insertRuleAfterStart
|
||||
skipSpaces(str);
|
||||
attr_rule = true;
|
||||
//continue;
|
||||
}
|
||||
// element relation
|
||||
if (ch == '>') {
|
||||
str = str[1 .. $];
|
||||
CssSelectorRule rule = new CssSelectorRule(CssSelectorRuleType.parent);
|
||||
rule.id = _id;
|
||||
insertRuleStart(rule);
|
||||
_id=0;
|
||||
continue;
|
||||
} else if (ch == '+') {
|
||||
str = str[1 .. $];
|
||||
CssSelectorRule rule = new CssSelectorRule(CssSelectorRuleType.predecessor);
|
||||
rule.id = _id;
|
||||
insertRuleStart(rule);
|
||||
_id=0;
|
||||
continue;
|
||||
} else if (ch.isAlpha) {
|
||||
CssSelectorRule rule = new CssSelectorRule(CssSelectorRuleType.ancessor);
|
||||
rule.id = _id;
|
||||
insertRuleStart(rule);
|
||||
_id=0;
|
||||
continue;
|
||||
}
|
||||
if ( !attr_rule )
|
||||
return false;
|
||||
else if (str.length > 0 && (str[0] == ',' || str[0] == '{'))
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
@property uint tagId() { return _id; }
|
||||
bool check(const Node node) const {
|
||||
|
@ -591,14 +530,18 @@ struct CssDeclItem {
|
|||
class CssDeclaration {
|
||||
private CssDeclItem[] _list;
|
||||
|
||||
private void addLengthDecl(CssDeclType type, CssValue len) {
|
||||
@property bool empty() {
|
||||
return _list.length == 0;
|
||||
}
|
||||
|
||||
void addLengthDecl(CssDeclType type, CssValue len) {
|
||||
CssDeclItem item;
|
||||
item.type = type;
|
||||
item.length = len;
|
||||
_list ~= item;
|
||||
}
|
||||
|
||||
private void addDecl(CssDeclType type, int value, string str) {
|
||||
void addDecl(CssDeclType type, int value, string str) {
|
||||
CssDeclItem item;
|
||||
item.type = type;
|
||||
item.value = value;
|
||||
|
@ -610,679 +553,10 @@ class CssDeclaration {
|
|||
foreach(item; _list)
|
||||
item.apply(style);
|
||||
}
|
||||
|
||||
bool parse(ref string src, bool mustBeInBrackets = true) {
|
||||
if (!skipSpaces(src))
|
||||
return false;
|
||||
if (mustBeInBrackets && !skipChar(src, '{'))
|
||||
return false; // decl must start with {
|
||||
for (;;) {
|
||||
CssDeclType propId = parseCssDeclType(src);
|
||||
if (src.empty)
|
||||
break;
|
||||
if (propId != CssDeclType.unknown) {
|
||||
int n = -1;
|
||||
string s = null;
|
||||
switch(propId) with(CssDeclType) {
|
||||
case display: n = parseEnumItem!CssDisplay(src, -1); break;
|
||||
case white_space: n = parseEnumItem!CssWhiteSpace(src, -1); break;
|
||||
case text_align: n = parseEnumItem!CssTextAlign(src, -1); break;
|
||||
case text_align_last: n = parseEnumItem!CssTextAlign(src, -1); break;
|
||||
case text_decoration: n = parseEnumItem!CssTextDecoration(src, -1); break;
|
||||
case hyphenate:
|
||||
case _webkit_hyphens: // -webkit-hyphens
|
||||
case adobe_hyphenate: // adobe-hyphenate
|
||||
case adobe_text_layout: // adobe-text-layout
|
||||
n = parseEnumItem!CssHyphenate(src, -1);
|
||||
break; // hyphenate
|
||||
case color:
|
||||
case background_color:
|
||||
CssValue v;
|
||||
if (parseColor(src, v)) {
|
||||
addLengthDecl(propId, v);
|
||||
}
|
||||
break;
|
||||
case vertical_align: n = parseEnumItem!CssVerticalAlign(src, -1); break;
|
||||
case font_family: // id families like serif, sans-serif
|
||||
string[] list;
|
||||
string[] faceList;
|
||||
if (splitPropertyValueList(src, list)) {
|
||||
foreach(item; list) {
|
||||
string name = item;
|
||||
int family = parseEnumItem!CssFontFamily(name, -1);
|
||||
if (family != -1) {
|
||||
// family name, e.g. sans-serif
|
||||
n = family;
|
||||
} else {
|
||||
faceList ~= item;
|
||||
}
|
||||
}
|
||||
}
|
||||
s = joinPropertyValueList(faceList);
|
||||
break;
|
||||
case font_style: n = parseEnumItem!CssFontStyle(src, -1); break;
|
||||
case font_weight:
|
||||
n = parseEnumItem!CssFontWeight(src, -1);
|
||||
if (n < 0) {
|
||||
CssValue value;
|
||||
if (parseLength(src, value)) {
|
||||
if (value.type == CssValueType.px) {
|
||||
if (value.value < 150)
|
||||
n = CssFontWeight.fw_100;
|
||||
else if (value.value < 250)
|
||||
n = CssFontWeight.fw_200;
|
||||
else if (value.value < 350)
|
||||
n = CssFontWeight.fw_300;
|
||||
else if (value.value < 450)
|
||||
n = CssFontWeight.fw_400;
|
||||
else if (value.value < 550)
|
||||
n = CssFontWeight.fw_500;
|
||||
else if (value.value < 650)
|
||||
n = CssFontWeight.fw_600;
|
||||
else if (value.value < 750)
|
||||
n = CssFontWeight.fw_700;
|
||||
else if (value.value < 850)
|
||||
n = CssFontWeight.fw_800;
|
||||
else
|
||||
n = CssFontWeight.fw_900;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
//n = parseEnumItem!Css(src, -1);
|
||||
break;
|
||||
case text_indent:
|
||||
{
|
||||
// read length
|
||||
CssValue len;
|
||||
bool negative = false;
|
||||
if (src[0] == '-') {
|
||||
src = src[1 .. $];
|
||||
negative = true;
|
||||
}
|
||||
if (parseLength(src, len)) {
|
||||
// read optional "hanging" flag
|
||||
skipSpaces(src);
|
||||
string attr = parseIdent(src);
|
||||
if (attr == "hanging")
|
||||
len.value = -len.value;
|
||||
addLengthDecl(propId, len);
|
||||
}
|
||||
}
|
||||
break;
|
||||
case line_height:
|
||||
case letter_spacing:
|
||||
case font_size:
|
||||
case width:
|
||||
case height:
|
||||
case margin_left:
|
||||
case margin_right:
|
||||
case margin_top:
|
||||
case margin_bottom:
|
||||
case padding_left:
|
||||
case padding_right:
|
||||
case padding_top:
|
||||
case padding_bottom:
|
||||
// parse length
|
||||
CssValue value;
|
||||
if (parseLength(src, value))
|
||||
addLengthDecl(propId, value);
|
||||
break;
|
||||
case margin:
|
||||
case padding:
|
||||
//n = parseEnumItem!Css(src, -1);
|
||||
CssValue[4] len;
|
||||
int i;
|
||||
for (i = 0; i < 4; ++i)
|
||||
if (!parseLength(src, len[i]))
|
||||
break;
|
||||
if (i) {
|
||||
switch (i) {
|
||||
case 1:
|
||||
len[1] = len[0];
|
||||
goto case; /* fall through */
|
||||
case 2:
|
||||
len[2] = len[0];
|
||||
goto case; /* fall through */
|
||||
case 3:
|
||||
len[3] = len[1];
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
if (propId == margin) {
|
||||
addLengthDecl(margin_left, len[0]);
|
||||
addLengthDecl(margin_top, len[1]);
|
||||
addLengthDecl(margin_right, len[2]);
|
||||
addLengthDecl(margin_bottom, len[3]);
|
||||
} else {
|
||||
addLengthDecl(padding_left, len[0]);
|
||||
addLengthDecl(padding_top, len[1]);
|
||||
addLengthDecl(padding_right, len[2]);
|
||||
addLengthDecl(padding_bottom, len[3]);
|
||||
}
|
||||
}
|
||||
break;
|
||||
case page_break_before:
|
||||
case page_break_inside:
|
||||
case page_break_after:
|
||||
n = parseEnumItem!CssPageBreak(src, -1);
|
||||
break;
|
||||
case list_style:
|
||||
//n = parseEnumItem!Css(src, -1);
|
||||
break;
|
||||
case list_style_type: n = parseEnumItem!CssListStyleType(src, -1); break;
|
||||
case list_style_position: n = parseEnumItem!CssListStylePosition(src, -1); break;
|
||||
case list_style_image:
|
||||
//n = parseEnumItem!CssListStyleImage(src, -1);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
if (n >= 0 || !s.empty)
|
||||
addDecl(propId, n, s);
|
||||
}
|
||||
if (!nextProperty(src))
|
||||
break;
|
||||
}
|
||||
if (mustBeInBrackets && !skipChar(src, '}'))
|
||||
return false;
|
||||
return _list.length > 0;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/// skip spaces, move to new location, return true if there are some characters left in source line
|
||||
private bool skipSpaces(ref string src) {
|
||||
for(;;) {
|
||||
if (src.empty) {
|
||||
src = null;
|
||||
return false;
|
||||
}
|
||||
char ch = src[0];
|
||||
if (ch == ' ' || ch == '\t' || ch == '\r' || ch == '\n') {
|
||||
src = src[1 .. $];
|
||||
} else {
|
||||
return !src.empty;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private bool isIdentChar(char ch) {
|
||||
return (ch >= 'A' && ch <='Z') || (ch >= 'a' && ch <='z') || (ch == '-') || (ch == '_');
|
||||
}
|
||||
|
||||
/// parse css identifier
|
||||
private string parseIdent(ref string src) {
|
||||
int pos = 0;
|
||||
for ( ; pos < src.length; pos++) {
|
||||
if (!src[pos].isIdentChar)
|
||||
break;
|
||||
}
|
||||
if (!pos)
|
||||
return null;
|
||||
string res = src[0 .. pos];
|
||||
if (pos < src.length)
|
||||
src = src[pos .. $];
|
||||
else
|
||||
src = null;
|
||||
skipSpaces(src);
|
||||
return res;
|
||||
}
|
||||
|
||||
private bool skipChar(ref string src, char ch) {
|
||||
skipSpaces(src);
|
||||
if (src.length > 0 && src[0] == ch) {
|
||||
src = src[1 .. $];
|
||||
skipSpaces(src);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
private string replaceChar(string s, char from, char to) {
|
||||
foreach(ch; s) {
|
||||
if (ch == from) {
|
||||
char[] buf;
|
||||
foreach(c; s)
|
||||
if (c == from)
|
||||
buf ~= to;
|
||||
else
|
||||
buf ~= c;
|
||||
return buf.dup;
|
||||
}
|
||||
}
|
||||
return s;
|
||||
}
|
||||
|
||||
/// remove trailing _ from string, e.g. "body_" -> "body"
|
||||
private string removeTrailingUnderscore(string s) {
|
||||
if (s.endsWith("_"))
|
||||
return s[0..$-1];
|
||||
return s;
|
||||
}
|
||||
|
||||
private int parseEnumItem(E)(ref string src, int defValue = -1) if (is(E == enum)) {
|
||||
string ident = replaceChar(parseIdent(src), '-', '_');
|
||||
foreach(member; EnumMembers!E) {
|
||||
if (ident == removeTrailingUnderscore(member.to!string)) {
|
||||
return member.to!int;
|
||||
}
|
||||
}
|
||||
return defValue;
|
||||
}
|
||||
|
||||
private CssDeclType parseCssDeclType(ref string src) {
|
||||
int n = parseEnumItem!CssDeclType(src, -1);
|
||||
if (n < 0)
|
||||
return CssDeclType.unknown;
|
||||
if (!skipChar(src, ':')) // no : after identifier
|
||||
return CssDeclType.unknown;
|
||||
return cast(CssDeclType)n;
|
||||
}
|
||||
|
||||
private bool nextProperty(ref string str) {
|
||||
int pos = 0;
|
||||
for (; pos < str.length; pos++) {
|
||||
char ch = str[pos];
|
||||
if (ch == '}')
|
||||
break;
|
||||
if (ch == ';') {
|
||||
pos++;
|
||||
break;
|
||||
}
|
||||
}
|
||||
str = pos < str.length ? str[pos .. $] : null;
|
||||
skipSpaces(str);
|
||||
return !str.empty && str[0] != '}';
|
||||
}
|
||||
|
||||
|
||||
private int hexDigit( char c )
|
||||
{
|
||||
if ( c >= '0' && c <= '9' )
|
||||
return c-'0';
|
||||
if ( c >= 'A' && c <= 'F' )
|
||||
return c - 'A' + 10;
|
||||
if ( c >= 'a' && c <= 'f' )
|
||||
return c - 'a' + 10;
|
||||
return -1;
|
||||
}
|
||||
|
||||
private int parseStandardColor(string ident) {
|
||||
switch(ident) {
|
||||
case "black": return 0x000000;
|
||||
case "green": return 0x008000;
|
||||
case "silver": return 0xC0C0C0;
|
||||
case "lime": return 0x00FF00;
|
||||
case "gray": return 0x808080;
|
||||
case "olive": return 0x808000;
|
||||
case "white": return 0xFFFFFF;
|
||||
case "yellow": return 0xFFFF00;
|
||||
case "maroon": return 0x800000;
|
||||
case "navy": return 0x000080;
|
||||
case "red": return 0xFF0000;
|
||||
case "blue": return 0x0000FF;
|
||||
case "purple": return 0x800080;
|
||||
case "teal": return 0x008080;
|
||||
case "fuchsia": return 0xFF00FF;
|
||||
case "aqua": return 0x00FFFF;
|
||||
default: return -1;
|
||||
}
|
||||
}
|
||||
|
||||
private bool parseColor(ref string src, ref CssValue value)
|
||||
{
|
||||
value.type = CssValueType.unspecified;
|
||||
value.value = 0;
|
||||
skipSpaces(src);
|
||||
if (src.empty)
|
||||
return false;
|
||||
string ident = parseIdent(src);
|
||||
if (!ident.empty) {
|
||||
switch(ident) {
|
||||
case "inherited":
|
||||
value.type = CssValueType.inherited;
|
||||
return true;
|
||||
case "none":
|
||||
return true;
|
||||
default:
|
||||
int v = parseStandardColor(ident);
|
||||
if (v >= 0) {
|
||||
value.value = v;
|
||||
value.type = CssValueType.color;
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
}
|
||||
char ch = src[0];
|
||||
if (ch == '#') {
|
||||
// #rgb or #rrggbb colors
|
||||
src = src[1 .. $];
|
||||
int nDigits = 0;
|
||||
for ( ; nDigits < src.length && hexDigit(src[nDigits])>=0; nDigits++ ) {
|
||||
}
|
||||
if ( nDigits==3 ) {
|
||||
int r = hexDigit( src[0] );
|
||||
int g = hexDigit( src[1] );
|
||||
int b = hexDigit( src[2] );
|
||||
value.type = CssValueType.color;
|
||||
value.value = (((r + r*16) * 256) | (g + g*16)) * 256 | (b + b*16);
|
||||
src = src[3..$];
|
||||
return true;
|
||||
} else if ( nDigits==6 ) {
|
||||
int r = hexDigit( src[0] ) * 16;
|
||||
r += hexDigit( src[1] );
|
||||
int g = hexDigit( src[2] ) * 16;
|
||||
g += hexDigit( src[3] );
|
||||
int b = hexDigit( src[4] ) * 16;
|
||||
b += hexDigit( src[5] );
|
||||
value.type = CssValueType.color;
|
||||
value.value = ((r * 256) | g) * 256 | b;
|
||||
src = src[6..$];
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
private bool parseLength(ref string src, ref CssValue value)
|
||||
{
|
||||
value.type = CssValueType.unspecified;
|
||||
value.value = 0;
|
||||
skipSpaces(src);
|
||||
string ident = parseIdent(src);
|
||||
if (!ident.empty) {
|
||||
switch(ident) {
|
||||
case "inherited":
|
||||
value.type = CssValueType.inherited;
|
||||
return true;
|
||||
default:
|
||||
return false;
|
||||
}
|
||||
}
|
||||
if (src.empty)
|
||||
return false;
|
||||
int n = 0;
|
||||
char ch = src[0];
|
||||
if (ch != '.') {
|
||||
if (ch < '0' || ch > '9') {
|
||||
return false; // not a number
|
||||
}
|
||||
while (ch >= '0' && ch <= '9') {
|
||||
n = n*10 + (ch - '0');
|
||||
src = src[1 .. $];
|
||||
if (src.empty)
|
||||
break;
|
||||
ch = src[0];
|
||||
}
|
||||
}
|
||||
int frac = 0;
|
||||
int frac_div = 1;
|
||||
if (ch == '.') {
|
||||
src = src[1 .. $];
|
||||
if (!src.empty) {
|
||||
ch = src[0];
|
||||
while (ch >= '0' && ch <= '9') {
|
||||
frac = frac*10 + (ch - '0');
|
||||
frac_div *= 10;
|
||||
src = src[1 .. $];
|
||||
if (src.empty)
|
||||
break;
|
||||
ch = src[0];
|
||||
}
|
||||
}
|
||||
}
|
||||
if (ch == '%') {
|
||||
value.type = CssValueType.percent;
|
||||
src = src[1 .. $];
|
||||
} else {
|
||||
ident = parseIdent(src);
|
||||
if (!ident.empty) {
|
||||
switch(ident) {
|
||||
case "em": value.type = CssValueType.em; break;
|
||||
case "pt": value.type = CssValueType.pt; break;
|
||||
case "ex": value.type = CssValueType.ex; break;
|
||||
case "px": value.type = CssValueType.px; break;
|
||||
case "in": value.type = CssValueType.in_; break;
|
||||
case "cm": value.type = CssValueType.cm; break;
|
||||
case "mm": value.type = CssValueType.mm; break;
|
||||
case "pc": value.type = CssValueType.pc; break;
|
||||
default:
|
||||
return false;
|
||||
}
|
||||
} else {
|
||||
value.type = CssValueType.px;
|
||||
}
|
||||
}
|
||||
if ( value.type == CssValueType.px || value.type == CssValueType.percent )
|
||||
value.value = n; // normal
|
||||
else
|
||||
value.value = n * 256 + 256 * frac / frac_div; // *256
|
||||
return true;
|
||||
}
|
||||
|
||||
private void appendItem(ref string[] list, ref char[] item) {
|
||||
if (!item.empty) {
|
||||
list ~= item.dup;
|
||||
item.length = 0;
|
||||
}
|
||||
}
|
||||
|
||||
/// splits string like "Arial", Times New Roman, Courier; into list, stops on ; and }
|
||||
/// returns true if at least one item added to list; moves str to new position
|
||||
bool splitPropertyValueList(ref string str, ref string[] list)
|
||||
{
|
||||
int i=0;
|
||||
char quote_char = 0;
|
||||
char[] name;
|
||||
bool last_space = false;
|
||||
for (i=0; i < str.length; i++) {
|
||||
char ch = str[i];
|
||||
switch(ch) {
|
||||
case '\'':
|
||||
case '\"':
|
||||
if (quote_char == 0) {
|
||||
if (!name.empty)
|
||||
appendItem(list, name);
|
||||
quote_char = ch;
|
||||
} else if (quote_char == ch) {
|
||||
if (!name.empty)
|
||||
appendItem(list, name);
|
||||
quote_char = 0;
|
||||
} else {
|
||||
// append char
|
||||
name ~= ch;
|
||||
}
|
||||
last_space = false;
|
||||
break;
|
||||
case ',':
|
||||
{
|
||||
if (quote_char==0) {
|
||||
if (!name.empty)
|
||||
appendItem(list, name);
|
||||
} else {
|
||||
// inside quotation: append char
|
||||
name ~= ch;
|
||||
}
|
||||
last_space = false;
|
||||
}
|
||||
break;
|
||||
case '\t':
|
||||
case ' ':
|
||||
{
|
||||
if (quote_char != 0)
|
||||
name ~= ch;
|
||||
last_space = true;
|
||||
}
|
||||
break;
|
||||
case ';':
|
||||
case '}':
|
||||
if (quote_char==0) {
|
||||
if (!name.empty)
|
||||
appendItem(list, name);
|
||||
str = i < str.length ? str[i .. $] : null;
|
||||
return list.length > 0;
|
||||
} else {
|
||||
// inside quotation: append char
|
||||
name ~= ch;
|
||||
last_space = false;
|
||||
}
|
||||
break;
|
||||
default:
|
||||
if (last_space && !name.empty && quote_char == 0)
|
||||
name ~= ' ';
|
||||
name ~= ch;
|
||||
last_space = false;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (!name.empty)
|
||||
appendItem(list, name);
|
||||
str = i < str.length ? str[i .. $] : null;
|
||||
return list.length > 0;
|
||||
}
|
||||
|
||||
unittest {
|
||||
string src = "Arial, 'Times New Roman', \"Arial Black\", sans-serif; next-property: }";
|
||||
string[] list;
|
||||
assert(splitPropertyValueList(src, list));
|
||||
assert(list.length == 4);
|
||||
assert(list[0] == "Arial");
|
||||
assert(list[1] == "Times New Roman");
|
||||
assert(list[2] == "Arial Black");
|
||||
assert(list[3] == "sans-serif");
|
||||
}
|
||||
|
||||
/// joins list of items into comma separated string, each item in quotation marks
|
||||
string joinPropertyValueList(string[] list) {
|
||||
if (list.empty)
|
||||
return null;
|
||||
char[] res;
|
||||
|
||||
for (int i = 0; i < list.length; i++) {
|
||||
if (i > 0)
|
||||
res ~= ", ";
|
||||
res ~= "\"";
|
||||
res ~= list[i];
|
||||
res ~= "\"";
|
||||
}
|
||||
|
||||
return res.dup;
|
||||
}
|
||||
|
||||
unittest {
|
||||
assert(joinPropertyValueList(["item1", "item 2"]) == "\"item1\", \"item 2\"");
|
||||
}
|
||||
|
||||
|
||||
private bool parseAttrValue(ref string str, ref string attrvalue)
|
||||
{
|
||||
char[] buf;
|
||||
int pos = 0;
|
||||
if (!skipSpaces(str))
|
||||
return false;
|
||||
char ch = str[0];
|
||||
if (ch == '\"') {
|
||||
str = str[1 .. $];
|
||||
for ( ; pos < str.length && str[pos] != '\"'; pos++) {
|
||||
if (pos >= 1000)
|
||||
return false;
|
||||
}
|
||||
if (pos >= str.length || str[pos] != '\"')
|
||||
return false;
|
||||
buf ~= str[0 .. pos];
|
||||
str = str[pos + 1 .. $];
|
||||
if (!skipSpaces(str))
|
||||
return false;
|
||||
if (str[0] != ']')
|
||||
return false;
|
||||
str = str[1 .. $];
|
||||
attrvalue = buf.dup;
|
||||
return true;
|
||||
} else {
|
||||
for ( ; pos < str.length && str[pos] != ' ' && str[pos] != '\t' && str[pos] != ']'; pos++) {
|
||||
if (pos >= 1000)
|
||||
return false;
|
||||
}
|
||||
if (pos >= str.length || str[pos] != ']')
|
||||
return false;
|
||||
buf ~= str[0 .. pos];
|
||||
str = str[pos + 1 .. $];
|
||||
attrvalue = buf.dup;
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
private CssSelectorRule parseAttr(ref string str, Document doc)
|
||||
{
|
||||
CssSelectorRuleType st = CssSelectorRuleType.universal;
|
||||
char ch = str[0];
|
||||
if (ch == '.') {
|
||||
// E.class
|
||||
str = str[1 .. $];
|
||||
skipSpaces(str);
|
||||
string attrvalue = parseIdent(str);
|
||||
if (attrvalue.empty)
|
||||
return null;
|
||||
CssSelectorRule rule = new CssSelectorRule(CssSelectorRuleType.class_);
|
||||
rule.setAttr(Attr.class_, attrvalue.toLower);
|
||||
return rule;
|
||||
} else if (ch == '#') {
|
||||
// E#id
|
||||
str = str[1 .. $];
|
||||
skipSpaces(str);
|
||||
string attrvalue = parseIdent(str);
|
||||
if (attrvalue.empty)
|
||||
return null;
|
||||
CssSelectorRule rule = new CssSelectorRule(CssSelectorRuleType.id);
|
||||
rule.setAttr(Attr.id, attrvalue.toLower);
|
||||
return rule;
|
||||
} else if (ch != '[')
|
||||
return null;
|
||||
// [.....] rule
|
||||
str = str[1 .. $]; // skip [
|
||||
skipSpaces(str);
|
||||
string attrname = parseIdent(str);
|
||||
if (attrname.empty)
|
||||
return null;
|
||||
if (!skipSpaces(str))
|
||||
return null;
|
||||
string attrvalue = null;
|
||||
ch = str[0];
|
||||
if (ch == ']') {
|
||||
// empty []
|
||||
st = CssSelectorRuleType.attrset;
|
||||
str = str[1 .. $]; // skip ]
|
||||
} else if (ch == '=') {
|
||||
str = str[1 .. $]; // skip =
|
||||
if (!parseAttrValue(str, attrvalue))
|
||||
return null;
|
||||
st = CssSelectorRuleType.attreq;
|
||||
} else if (ch == '~' && str.length > 1 && str[1] == '=') {
|
||||
str = str[2 .. $]; // skip ~=
|
||||
if (!parseAttrValue(str, attrvalue))
|
||||
return null;
|
||||
st = CssSelectorRuleType.attrhas;
|
||||
} else if (ch == '|' && str.length > 1 && str[1] == '=') {
|
||||
str = str[2 .. $]; // skip |=
|
||||
if (!parseAttrValue(str, attrvalue))
|
||||
return null;
|
||||
st = CssSelectorRuleType.attrstarts;
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
CssSelectorRule rule = new CssSelectorRule(st);
|
||||
attr_id id = doc.attrId(attrname);
|
||||
rule.setAttr(id, attrvalue);
|
||||
return rule;
|
||||
}
|
||||
|
||||
|
||||
unittest {
|
||||
CssStyle style = new CssStyle();
|
||||
CssDeclaration decl = new CssDeclaration();
|
||||
CssWhiteSpace whiteSpace = CssWhiteSpace.inherit;
|
||||
CssTextAlign textAlign = CssTextAlign.inherit;
|
||||
CssTextAlign textAlignLast = CssTextAlign.inherit;
|
||||
|
@ -1291,8 +565,9 @@ unittest {
|
|||
string src = "{ display: inline; text-decoration: underline; white-space: pre; text-align: right; text-align-last: left; "
|
||||
"hyphenate: auto; width: 70%; height: 1.5pt; margin-left: 2.0em; "
|
||||
"font-family: Arial, 'Times New Roman', sans-serif; font-size: 18pt; line-height: 120%; letter-spacing: 2px; font-weight: 300; "
|
||||
" }t";
|
||||
assert(decl.parse(src, true));
|
||||
" }tail";
|
||||
CssDeclaration decl = parseCssDeclaration(src, true);
|
||||
assert(decl !is null);
|
||||
assert(!src.empty && src[0] == 't');
|
||||
assert(style.display == CssDisplay.block);
|
||||
assert(style.textDecoration == CssTextDecoration.inherit);
|
||||
|
|
|
@ -0,0 +1,762 @@
|
|||
module dlangui.core.cssparser;
|
||||
|
||||
import std.traits;
|
||||
import std.conv : to;
|
||||
import std.string;
|
||||
import std.array : empty;
|
||||
import std.algorithm : equal;
|
||||
import std.ascii : isAlpha;
|
||||
|
||||
import dlangui.core.dom;
|
||||
import dlangui.core.css;
|
||||
|
||||
/// skip spaces, move to new location, return true if there are some characters left in source line
|
||||
private bool skipSpaces(ref string src) {
|
||||
for(;;) {
|
||||
if (src.empty) {
|
||||
src = null;
|
||||
return false;
|
||||
}
|
||||
char ch = src[0];
|
||||
if (ch == ' ' || ch == '\t' || ch == '\r' || ch == '\n') {
|
||||
src = src[1 .. $];
|
||||
} else {
|
||||
return !src.empty;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private bool isIdentChar(char ch) {
|
||||
return (ch >= 'A' && ch <='Z') || (ch >= 'a' && ch <='z') || (ch == '-') || (ch == '_');
|
||||
}
|
||||
|
||||
/// parse css identifier
|
||||
private string parseIdent(ref string src) {
|
||||
int pos = 0;
|
||||
for ( ; pos < src.length; pos++) {
|
||||
if (!src[pos].isIdentChar)
|
||||
break;
|
||||
}
|
||||
if (!pos)
|
||||
return null;
|
||||
string res = src[0 .. pos];
|
||||
if (pos < src.length)
|
||||
src = src[pos .. $];
|
||||
else
|
||||
src = null;
|
||||
skipSpaces(src);
|
||||
return res;
|
||||
}
|
||||
|
||||
private bool skipChar(ref string src, char ch) {
|
||||
skipSpaces(src);
|
||||
if (src.length > 0 && src[0] == ch) {
|
||||
src = src[1 .. $];
|
||||
skipSpaces(src);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
private string replaceChar(string s, char from, char to) {
|
||||
foreach(ch; s) {
|
||||
if (ch == from) {
|
||||
char[] buf;
|
||||
foreach(c; s)
|
||||
if (c == from)
|
||||
buf ~= to;
|
||||
else
|
||||
buf ~= c;
|
||||
return buf.dup;
|
||||
}
|
||||
}
|
||||
return s;
|
||||
}
|
||||
|
||||
/// remove trailing _ from string, e.g. "body_" -> "body"
|
||||
private string removeTrailingUnderscore(string s) {
|
||||
if (s.endsWith("_"))
|
||||
return s[0..$-1];
|
||||
return s;
|
||||
}
|
||||
|
||||
private int parseEnumItem(E)(ref string src, int defValue = -1) if (is(E == enum)) {
|
||||
string ident = replaceChar(parseIdent(src), '-', '_');
|
||||
foreach(member; EnumMembers!E) {
|
||||
if (ident == removeTrailingUnderscore(member.to!string)) {
|
||||
return member.to!int;
|
||||
}
|
||||
}
|
||||
return defValue;
|
||||
}
|
||||
|
||||
private CssDeclType parseCssDeclType(ref string src) {
|
||||
int n = parseEnumItem!CssDeclType(src, -1);
|
||||
if (n < 0)
|
||||
return CssDeclType.unknown;
|
||||
if (!skipChar(src, ':')) // no : after identifier
|
||||
return CssDeclType.unknown;
|
||||
return cast(CssDeclType)n;
|
||||
}
|
||||
|
||||
private bool nextProperty(ref string str) {
|
||||
int pos = 0;
|
||||
for (; pos < str.length; pos++) {
|
||||
char ch = str[pos];
|
||||
if (ch == '}')
|
||||
break;
|
||||
if (ch == ';') {
|
||||
pos++;
|
||||
break;
|
||||
}
|
||||
}
|
||||
str = pos < str.length ? str[pos .. $] : null;
|
||||
skipSpaces(str);
|
||||
return !str.empty && str[0] != '}';
|
||||
}
|
||||
|
||||
|
||||
private int hexDigit( char c )
|
||||
{
|
||||
if ( c >= '0' && c <= '9' )
|
||||
return c-'0';
|
||||
if ( c >= 'A' && c <= 'F' )
|
||||
return c - 'A' + 10;
|
||||
if ( c >= 'a' && c <= 'f' )
|
||||
return c - 'a' + 10;
|
||||
return -1;
|
||||
}
|
||||
|
||||
private int parseStandardColor(string ident) {
|
||||
switch(ident) {
|
||||
case "black": return 0x000000;
|
||||
case "green": return 0x008000;
|
||||
case "silver": return 0xC0C0C0;
|
||||
case "lime": return 0x00FF00;
|
||||
case "gray": return 0x808080;
|
||||
case "olive": return 0x808000;
|
||||
case "white": return 0xFFFFFF;
|
||||
case "yellow": return 0xFFFF00;
|
||||
case "maroon": return 0x800000;
|
||||
case "navy": return 0x000080;
|
||||
case "red": return 0xFF0000;
|
||||
case "blue": return 0x0000FF;
|
||||
case "purple": return 0x800080;
|
||||
case "teal": return 0x008080;
|
||||
case "fuchsia": return 0xFF00FF;
|
||||
case "aqua": return 0x00FFFF;
|
||||
default: return -1;
|
||||
}
|
||||
}
|
||||
|
||||
private bool parseColor(ref string src, ref CssValue value)
|
||||
{
|
||||
value.type = CssValueType.unspecified;
|
||||
value.value = 0;
|
||||
skipSpaces(src);
|
||||
if (src.empty)
|
||||
return false;
|
||||
string ident = parseIdent(src);
|
||||
if (!ident.empty) {
|
||||
switch(ident) {
|
||||
case "inherited":
|
||||
value.type = CssValueType.inherited;
|
||||
return true;
|
||||
case "none":
|
||||
return true;
|
||||
default:
|
||||
int v = parseStandardColor(ident);
|
||||
if (v >= 0) {
|
||||
value.value = v;
|
||||
value.type = CssValueType.color;
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
}
|
||||
char ch = src[0];
|
||||
if (ch == '#') {
|
||||
// #rgb or #rrggbb colors
|
||||
src = src[1 .. $];
|
||||
int nDigits = 0;
|
||||
for ( ; nDigits < src.length && hexDigit(src[nDigits])>=0; nDigits++ ) {
|
||||
}
|
||||
if ( nDigits==3 ) {
|
||||
int r = hexDigit( src[0] );
|
||||
int g = hexDigit( src[1] );
|
||||
int b = hexDigit( src[2] );
|
||||
value.type = CssValueType.color;
|
||||
value.value = (((r + r*16) * 256) | (g + g*16)) * 256 | (b + b*16);
|
||||
src = src[3..$];
|
||||
return true;
|
||||
} else if ( nDigits==6 ) {
|
||||
int r = hexDigit( src[0] ) * 16;
|
||||
r += hexDigit( src[1] );
|
||||
int g = hexDigit( src[2] ) * 16;
|
||||
g += hexDigit( src[3] );
|
||||
int b = hexDigit( src[4] ) * 16;
|
||||
b += hexDigit( src[5] );
|
||||
value.type = CssValueType.color;
|
||||
value.value = ((r * 256) | g) * 256 | b;
|
||||
src = src[6..$];
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
private bool parseLength(ref string src, ref CssValue value)
|
||||
{
|
||||
value.type = CssValueType.unspecified;
|
||||
value.value = 0;
|
||||
skipSpaces(src);
|
||||
string ident = parseIdent(src);
|
||||
if (!ident.empty) {
|
||||
switch(ident) {
|
||||
case "inherited":
|
||||
value.type = CssValueType.inherited;
|
||||
return true;
|
||||
default:
|
||||
return false;
|
||||
}
|
||||
}
|
||||
if (src.empty)
|
||||
return false;
|
||||
int n = 0;
|
||||
char ch = src[0];
|
||||
if (ch != '.') {
|
||||
if (ch < '0' || ch > '9') {
|
||||
return false; // not a number
|
||||
}
|
||||
while (ch >= '0' && ch <= '9') {
|
||||
n = n*10 + (ch - '0');
|
||||
src = src[1 .. $];
|
||||
if (src.empty)
|
||||
break;
|
||||
ch = src[0];
|
||||
}
|
||||
}
|
||||
int frac = 0;
|
||||
int frac_div = 1;
|
||||
if (ch == '.') {
|
||||
src = src[1 .. $];
|
||||
if (!src.empty) {
|
||||
ch = src[0];
|
||||
while (ch >= '0' && ch <= '9') {
|
||||
frac = frac*10 + (ch - '0');
|
||||
frac_div *= 10;
|
||||
src = src[1 .. $];
|
||||
if (src.empty)
|
||||
break;
|
||||
ch = src[0];
|
||||
}
|
||||
}
|
||||
}
|
||||
if (ch == '%') {
|
||||
value.type = CssValueType.percent;
|
||||
src = src[1 .. $];
|
||||
} else {
|
||||
ident = parseIdent(src);
|
||||
if (!ident.empty) {
|
||||
switch(ident) {
|
||||
case "em": value.type = CssValueType.em; break;
|
||||
case "pt": value.type = CssValueType.pt; break;
|
||||
case "ex": value.type = CssValueType.ex; break;
|
||||
case "px": value.type = CssValueType.px; break;
|
||||
case "in": value.type = CssValueType.in_; break;
|
||||
case "cm": value.type = CssValueType.cm; break;
|
||||
case "mm": value.type = CssValueType.mm; break;
|
||||
case "pc": value.type = CssValueType.pc; break;
|
||||
default:
|
||||
return false;
|
||||
}
|
||||
} else {
|
||||
value.type = CssValueType.px;
|
||||
}
|
||||
}
|
||||
if ( value.type == CssValueType.px || value.type == CssValueType.percent )
|
||||
value.value = n; // normal
|
||||
else
|
||||
value.value = n * 256 + 256 * frac / frac_div; // *256
|
||||
return true;
|
||||
}
|
||||
|
||||
private void appendItem(ref string[] list, ref char[] item) {
|
||||
if (!item.empty) {
|
||||
list ~= item.dup;
|
||||
item.length = 0;
|
||||
}
|
||||
}
|
||||
|
||||
/// splits string like "Arial", Times New Roman, Courier; into list, stops on ; and }
|
||||
/// returns true if at least one item added to list; moves str to new position
|
||||
bool splitPropertyValueList(ref string str, ref string[] list)
|
||||
{
|
||||
int i=0;
|
||||
char quote_char = 0;
|
||||
char[] name;
|
||||
bool last_space = false;
|
||||
for (i=0; i < str.length; i++) {
|
||||
char ch = str[i];
|
||||
switch(ch) {
|
||||
case '\'':
|
||||
case '\"':
|
||||
if (quote_char == 0) {
|
||||
if (!name.empty)
|
||||
appendItem(list, name);
|
||||
quote_char = ch;
|
||||
} else if (quote_char == ch) {
|
||||
if (!name.empty)
|
||||
appendItem(list, name);
|
||||
quote_char = 0;
|
||||
} else {
|
||||
// append char
|
||||
name ~= ch;
|
||||
}
|
||||
last_space = false;
|
||||
break;
|
||||
case ',':
|
||||
{
|
||||
if (quote_char==0) {
|
||||
if (!name.empty)
|
||||
appendItem(list, name);
|
||||
} else {
|
||||
// inside quotation: append char
|
||||
name ~= ch;
|
||||
}
|
||||
last_space = false;
|
||||
}
|
||||
break;
|
||||
case '\t':
|
||||
case ' ':
|
||||
{
|
||||
if (quote_char != 0)
|
||||
name ~= ch;
|
||||
last_space = true;
|
||||
}
|
||||
break;
|
||||
case ';':
|
||||
case '}':
|
||||
if (quote_char==0) {
|
||||
if (!name.empty)
|
||||
appendItem(list, name);
|
||||
str = i < str.length ? str[i .. $] : null;
|
||||
return list.length > 0;
|
||||
} else {
|
||||
// inside quotation: append char
|
||||
name ~= ch;
|
||||
last_space = false;
|
||||
}
|
||||
break;
|
||||
default:
|
||||
if (last_space && !name.empty && quote_char == 0)
|
||||
name ~= ' ';
|
||||
name ~= ch;
|
||||
last_space = false;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (!name.empty)
|
||||
appendItem(list, name);
|
||||
str = i < str.length ? str[i .. $] : null;
|
||||
return list.length > 0;
|
||||
}
|
||||
|
||||
unittest {
|
||||
string src = "Arial, 'Times New Roman', \"Arial Black\", sans-serif; next-property: }";
|
||||
string[] list;
|
||||
assert(splitPropertyValueList(src, list));
|
||||
assert(list.length == 4);
|
||||
assert(list[0] == "Arial");
|
||||
assert(list[1] == "Times New Roman");
|
||||
assert(list[2] == "Arial Black");
|
||||
assert(list[3] == "sans-serif");
|
||||
}
|
||||
|
||||
/// joins list of items into comma separated string, each item in quotation marks
|
||||
string joinPropertyValueList(string[] list) {
|
||||
if (list.empty)
|
||||
return null;
|
||||
char[] res;
|
||||
|
||||
for (int i = 0; i < list.length; i++) {
|
||||
if (i > 0)
|
||||
res ~= ", ";
|
||||
res ~= "\"";
|
||||
res ~= list[i];
|
||||
res ~= "\"";
|
||||
}
|
||||
|
||||
return res.dup;
|
||||
}
|
||||
|
||||
unittest {
|
||||
assert(joinPropertyValueList(["item1", "item 2"]) == "\"item1\", \"item 2\"");
|
||||
}
|
||||
|
||||
|
||||
private bool parseAttrValue(ref string str, ref string attrvalue)
|
||||
{
|
||||
char[] buf;
|
||||
int pos = 0;
|
||||
if (!skipSpaces(str))
|
||||
return false;
|
||||
char ch = str[0];
|
||||
if (ch == '\"') {
|
||||
str = str[1 .. $];
|
||||
for ( ; pos < str.length && str[pos] != '\"'; pos++) {
|
||||
if (pos >= 1000)
|
||||
return false;
|
||||
}
|
||||
if (pos >= str.length || str[pos] != '\"')
|
||||
return false;
|
||||
buf ~= str[0 .. pos];
|
||||
str = str[pos + 1 .. $];
|
||||
if (!skipSpaces(str))
|
||||
return false;
|
||||
if (str[0] != ']')
|
||||
return false;
|
||||
str = str[1 .. $];
|
||||
attrvalue = buf.dup;
|
||||
return true;
|
||||
} else {
|
||||
for ( ; pos < str.length && str[pos] != ' ' && str[pos] != '\t' && str[pos] != ']'; pos++) {
|
||||
if (pos >= 1000)
|
||||
return false;
|
||||
}
|
||||
if (pos >= str.length || str[pos] != ']')
|
||||
return false;
|
||||
buf ~= str[0 .. pos];
|
||||
str = str[pos + 1 .. $];
|
||||
attrvalue = buf.dup;
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
private CssSelectorRule parseAttr(ref string str, Document doc)
|
||||
{
|
||||
CssSelectorRuleType st = CssSelectorRuleType.universal;
|
||||
char ch = str[0];
|
||||
if (ch == '.') {
|
||||
// E.class
|
||||
str = str[1 .. $];
|
||||
skipSpaces(str);
|
||||
string attrvalue = parseIdent(str);
|
||||
if (attrvalue.empty)
|
||||
return null;
|
||||
CssSelectorRule rule = new CssSelectorRule(CssSelectorRuleType.class_);
|
||||
rule.setAttr(Attr.class_, attrvalue.toLower);
|
||||
return rule;
|
||||
} else if (ch == '#') {
|
||||
// E#id
|
||||
str = str[1 .. $];
|
||||
skipSpaces(str);
|
||||
string attrvalue = parseIdent(str);
|
||||
if (attrvalue.empty)
|
||||
return null;
|
||||
CssSelectorRule rule = new CssSelectorRule(CssSelectorRuleType.id);
|
||||
rule.setAttr(Attr.id, attrvalue.toLower);
|
||||
return rule;
|
||||
} else if (ch != '[')
|
||||
return null;
|
||||
// [.....] rule
|
||||
str = str[1 .. $]; // skip [
|
||||
skipSpaces(str);
|
||||
string attrname = parseIdent(str);
|
||||
if (attrname.empty)
|
||||
return null;
|
||||
if (!skipSpaces(str))
|
||||
return null;
|
||||
string attrvalue = null;
|
||||
ch = str[0];
|
||||
if (ch == ']') {
|
||||
// empty []
|
||||
st = CssSelectorRuleType.attrset;
|
||||
str = str[1 .. $]; // skip ]
|
||||
} else if (ch == '=') {
|
||||
str = str[1 .. $]; // skip =
|
||||
if (!parseAttrValue(str, attrvalue))
|
||||
return null;
|
||||
st = CssSelectorRuleType.attreq;
|
||||
} else if (ch == '~' && str.length > 1 && str[1] == '=') {
|
||||
str = str[2 .. $]; // skip ~=
|
||||
if (!parseAttrValue(str, attrvalue))
|
||||
return null;
|
||||
st = CssSelectorRuleType.attrhas;
|
||||
} else if (ch == '|' && str.length > 1 && str[1] == '=') {
|
||||
str = str[2 .. $]; // skip |=
|
||||
if (!parseAttrValue(str, attrvalue))
|
||||
return null;
|
||||
st = CssSelectorRuleType.attrstarts;
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
CssSelectorRule rule = new CssSelectorRule(st);
|
||||
attr_id id = doc.attrId(attrname);
|
||||
rule.setAttr(id, attrvalue);
|
||||
return rule;
|
||||
}
|
||||
|
||||
CssDeclaration parseCssDeclaration(ref string src, bool mustBeInBrackets = true) {
|
||||
if (!skipSpaces(src))
|
||||
return null;
|
||||
if (mustBeInBrackets && !skipChar(src, '{'))
|
||||
return null; // decl must start with {
|
||||
CssDeclaration res = new CssDeclaration();
|
||||
for (;;) {
|
||||
CssDeclType propId = parseCssDeclType(src);
|
||||
if (src.empty)
|
||||
break;
|
||||
if (propId != CssDeclType.unknown) {
|
||||
int n = -1;
|
||||
string s = null;
|
||||
switch(propId) with(CssDeclType) {
|
||||
case display: n = parseEnumItem!CssDisplay(src, -1); break;
|
||||
case white_space: n = parseEnumItem!CssWhiteSpace(src, -1); break;
|
||||
case text_align: n = parseEnumItem!CssTextAlign(src, -1); break;
|
||||
case text_align_last: n = parseEnumItem!CssTextAlign(src, -1); break;
|
||||
case text_decoration: n = parseEnumItem!CssTextDecoration(src, -1); break;
|
||||
case hyphenate:
|
||||
case _webkit_hyphens: // -webkit-hyphens
|
||||
case adobe_hyphenate: // adobe-hyphenate
|
||||
case adobe_text_layout: // adobe-text-layout
|
||||
n = parseEnumItem!CssHyphenate(src, -1);
|
||||
break; // hyphenate
|
||||
case color:
|
||||
case background_color:
|
||||
CssValue v;
|
||||
if (parseColor(src, v)) {
|
||||
res.addLengthDecl(propId, v);
|
||||
}
|
||||
break;
|
||||
case vertical_align: n = parseEnumItem!CssVerticalAlign(src, -1); break;
|
||||
case font_family: // id families like serif, sans-serif
|
||||
string[] list;
|
||||
string[] faceList;
|
||||
if (splitPropertyValueList(src, list)) {
|
||||
foreach(item; list) {
|
||||
string name = item;
|
||||
int family = parseEnumItem!CssFontFamily(name, -1);
|
||||
if (family != -1) {
|
||||
// family name, e.g. sans-serif
|
||||
n = family;
|
||||
} else {
|
||||
faceList ~= item;
|
||||
}
|
||||
}
|
||||
}
|
||||
s = joinPropertyValueList(faceList);
|
||||
break;
|
||||
case font_style: n = parseEnumItem!CssFontStyle(src, -1); break;
|
||||
case font_weight:
|
||||
n = parseEnumItem!CssFontWeight(src, -1);
|
||||
if (n < 0) {
|
||||
CssValue value;
|
||||
if (parseLength(src, value)) {
|
||||
if (value.type == CssValueType.px) {
|
||||
if (value.value < 150)
|
||||
n = CssFontWeight.fw_100;
|
||||
else if (value.value < 250)
|
||||
n = CssFontWeight.fw_200;
|
||||
else if (value.value < 350)
|
||||
n = CssFontWeight.fw_300;
|
||||
else if (value.value < 450)
|
||||
n = CssFontWeight.fw_400;
|
||||
else if (value.value < 550)
|
||||
n = CssFontWeight.fw_500;
|
||||
else if (value.value < 650)
|
||||
n = CssFontWeight.fw_600;
|
||||
else if (value.value < 750)
|
||||
n = CssFontWeight.fw_700;
|
||||
else if (value.value < 850)
|
||||
n = CssFontWeight.fw_800;
|
||||
else
|
||||
n = CssFontWeight.fw_900;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
//n = parseEnumItem!Css(src, -1);
|
||||
break;
|
||||
case text_indent:
|
||||
{
|
||||
// read length
|
||||
CssValue len;
|
||||
bool negative = false;
|
||||
if (src[0] == '-') {
|
||||
src = src[1 .. $];
|
||||
negative = true;
|
||||
}
|
||||
if (parseLength(src, len)) {
|
||||
// read optional "hanging" flag
|
||||
skipSpaces(src);
|
||||
string attr = parseIdent(src);
|
||||
if (attr == "hanging")
|
||||
len.value = -len.value;
|
||||
res.addLengthDecl(propId, len);
|
||||
}
|
||||
}
|
||||
break;
|
||||
case line_height:
|
||||
case letter_spacing:
|
||||
case font_size:
|
||||
case width:
|
||||
case height:
|
||||
case margin_left:
|
||||
case margin_right:
|
||||
case margin_top:
|
||||
case margin_bottom:
|
||||
case padding_left:
|
||||
case padding_right:
|
||||
case padding_top:
|
||||
case padding_bottom:
|
||||
// parse length
|
||||
CssValue value;
|
||||
if (parseLength(src, value))
|
||||
res.addLengthDecl(propId, value);
|
||||
break;
|
||||
case margin:
|
||||
case padding:
|
||||
//n = parseEnumItem!Css(src, -1);
|
||||
CssValue[4] len;
|
||||
int i;
|
||||
for (i = 0; i < 4; ++i)
|
||||
if (!parseLength(src, len[i]))
|
||||
break;
|
||||
if (i) {
|
||||
switch (i) {
|
||||
case 1:
|
||||
len[1] = len[0];
|
||||
goto case; /* fall through */
|
||||
case 2:
|
||||
len[2] = len[0];
|
||||
goto case; /* fall through */
|
||||
case 3:
|
||||
len[3] = len[1];
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
if (propId == margin) {
|
||||
res.addLengthDecl(margin_left, len[0]);
|
||||
res.addLengthDecl(margin_top, len[1]);
|
||||
res.addLengthDecl(margin_right, len[2]);
|
||||
res.addLengthDecl(margin_bottom, len[3]);
|
||||
} else {
|
||||
res.addLengthDecl(padding_left, len[0]);
|
||||
res.addLengthDecl(padding_top, len[1]);
|
||||
res.addLengthDecl(padding_right, len[2]);
|
||||
res.addLengthDecl(padding_bottom, len[3]);
|
||||
}
|
||||
}
|
||||
break;
|
||||
case page_break_before:
|
||||
case page_break_inside:
|
||||
case page_break_after:
|
||||
n = parseEnumItem!CssPageBreak(src, -1);
|
||||
break;
|
||||
case list_style:
|
||||
//n = parseEnumItem!Css(src, -1);
|
||||
break;
|
||||
case list_style_type: n = parseEnumItem!CssListStyleType(src, -1); break;
|
||||
case list_style_position: n = parseEnumItem!CssListStylePosition(src, -1); break;
|
||||
case list_style_image:
|
||||
//n = parseEnumItem!CssListStyleImage(src, -1);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
if (n >= 0 || !s.empty)
|
||||
res.addDecl(propId, n, s);
|
||||
}
|
||||
if (!nextProperty(src))
|
||||
break;
|
||||
}
|
||||
if (mustBeInBrackets && !skipChar(src, '}'))
|
||||
return null;
|
||||
if (res.empty)
|
||||
return null;
|
||||
return res;
|
||||
}
|
||||
|
||||
/// parse Css selector, return selector object if parsed ok
|
||||
CssSelector parseCssSelector(ref string str, Document doc) {
|
||||
if (str.empty)
|
||||
return null;
|
||||
CssSelector res = new CssSelector();
|
||||
for (;;) {
|
||||
if (!skipSpaces(str))
|
||||
return null;
|
||||
char ch = str[0];
|
||||
string ident = parseIdent(str);
|
||||
if (ch == '*') { // universal selector
|
||||
str = str[1 .. $];
|
||||
skipSpaces(str);
|
||||
res.id = 0;
|
||||
} else if (ch == '.') { // classname follows
|
||||
res.id = 0;
|
||||
// will be parsed as attribute
|
||||
} else if (!ident.empty) {
|
||||
// ident
|
||||
res.id = doc.tagId(ident);
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
if (!str.skipSpaces)
|
||||
return null;
|
||||
ch = str[0];
|
||||
if (ch == ',' || ch == '{')
|
||||
return res;
|
||||
// one or more attribute rules
|
||||
bool attr_rule = false;
|
||||
while (ch == '[' || ch == '.' || ch == '#') {
|
||||
CssSelectorRule rule = parseAttr(str, doc);
|
||||
if (!rule)
|
||||
return null;
|
||||
res.insertRuleStart(rule); //insertRuleAfterStart
|
||||
skipSpaces(str);
|
||||
attr_rule = true;
|
||||
//continue;
|
||||
}
|
||||
// element relation
|
||||
if (ch == '>') {
|
||||
str = str[1 .. $];
|
||||
CssSelectorRule rule = new CssSelectorRule(CssSelectorRuleType.parent);
|
||||
rule.id = res.id;
|
||||
res.insertRuleStart(rule);
|
||||
res.id = 0;
|
||||
continue;
|
||||
} else if (ch == '+') {
|
||||
str = str[1 .. $];
|
||||
CssSelectorRule rule = new CssSelectorRule(CssSelectorRuleType.predecessor);
|
||||
rule.id = res.id;
|
||||
res.insertRuleStart(rule);
|
||||
res.id = 0;
|
||||
continue;
|
||||
} else if (ch.isAlpha) {
|
||||
CssSelectorRule rule = new CssSelectorRule(CssSelectorRuleType.ancessor);
|
||||
rule.id = res.id;
|
||||
res.insertRuleStart(rule);
|
||||
res.id = 0;
|
||||
continue;
|
||||
}
|
||||
if (!attr_rule)
|
||||
return null;
|
||||
else if (str.length > 0 && (str[0] == ',' || str[0] == '{'))
|
||||
return res;
|
||||
}
|
||||
}
|
||||
|
||||
unittest {
|
||||
Document doc = new Document();
|
||||
string str;
|
||||
str = "body { width: 50% }";
|
||||
assert(parseCssSelector(str, doc) !is null);
|
||||
assert(parseCssDeclaration(str, true) !is null);
|
||||
str = "body > p { font-family: sans-serif }";
|
||||
assert(parseCssSelector(str, doc) !is null);
|
||||
assert(parseCssDeclaration(str, true) !is null);
|
||||
str = ".myclass + div { }";
|
||||
assert(parseCssSelector(str, doc) !is null);
|
||||
assert(parseCssDeclaration(str, true) is null); // empty property decl
|
||||
}
|
Loading…
Reference in New Issue