103 lines
2.8 KiB
JavaScript
103 lines
2.8 KiB
JavaScript
|
/*
|
|||
|
解析和匹配 Css 的选择器
|
|||
|
github:https://github.com/jin-yufeng/Parser
|
|||
|
docs:https://jin-yufeng.github.io/Parser
|
|||
|
author:JinYufeng
|
|||
|
update:2020/03/15
|
|||
|
*/
|
|||
|
var cfg = require('./config.js');
|
|||
|
class CssHandler {
|
|||
|
constructor(tagStyle) {
|
|||
|
var styles = Object.assign({}, cfg.userAgentStyles);
|
|||
|
for (var item in tagStyle)
|
|||
|
styles[item] = (styles[item] ? styles[item] + ';' : '') + tagStyle[item];
|
|||
|
this.styles = styles;
|
|||
|
}
|
|||
|
getStyle = data => this.styles = new CssParser(data, this.styles).parse();
|
|||
|
match(name, attrs) {
|
|||
|
var tmp, matched = (tmp = this.styles[name]) ? tmp + ';' : '';
|
|||
|
if (attrs.class) {
|
|||
|
var items = attrs.class.split(' ');
|
|||
|
for (var i = 0, item; item = items[i]; i++)
|
|||
|
if (tmp = this.styles['.' + item])
|
|||
|
matched += tmp + ';';
|
|||
|
}
|
|||
|
if (tmp = this.styles['#' + attrs.id])
|
|||
|
matched += tmp + ';';
|
|||
|
return matched;
|
|||
|
}
|
|||
|
}
|
|||
|
module.exports = CssHandler;
|
|||
|
class CssParser {
|
|||
|
constructor(data, init) {
|
|||
|
this.data = data;
|
|||
|
this.floor = 0;
|
|||
|
this.i = 0;
|
|||
|
this.list = [];
|
|||
|
this.res = init;
|
|||
|
this.state = this.Space;
|
|||
|
}
|
|||
|
parse() {
|
|||
|
for (var c; c = this.data[this.i]; this.i++)
|
|||
|
this.state(c);
|
|||
|
return this.res;
|
|||
|
}
|
|||
|
section = () => this.data.substring(this.start, this.i);
|
|||
|
isLetter = c => (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z');
|
|||
|
// 状态机
|
|||
|
Space(c) {
|
|||
|
if (c == '.' || c == '#' || this.isLetter(c)) {
|
|||
|
this.start = this.i;
|
|||
|
this.state = this.Name;
|
|||
|
} else if (c == '/' && this.data[this.i + 1] == '*')
|
|||
|
this.Comment();
|
|||
|
else if (!cfg.blankChar[c] && c != ';')
|
|||
|
this.state = this.Ignore;
|
|||
|
}
|
|||
|
Comment() {
|
|||
|
this.i = this.data.indexOf('*/', this.i) + 1;
|
|||
|
if (!this.i) this.i = this.data.length;
|
|||
|
this.state = this.Space;
|
|||
|
}
|
|||
|
Ignore(c) {
|
|||
|
if (c == '{') this.floor++;
|
|||
|
else if (c == '}' && !--this.floor) this.state = this.Space;
|
|||
|
}
|
|||
|
Name(c) {
|
|||
|
if (cfg.blankChar[c]) {
|
|||
|
this.list.push(this.section());
|
|||
|
this.state = this.NameSpace;
|
|||
|
} else if (c == '{') {
|
|||
|
this.list.push(this.section());
|
|||
|
this.Content();
|
|||
|
} else if (c == ',') {
|
|||
|
this.list.push(this.section());
|
|||
|
this.Comma();
|
|||
|
} else if (!this.isLetter(c) && (c < '0' || c > '9') && c != '-' && c != '_')
|
|||
|
this.state = this.Ignore;
|
|||
|
}
|
|||
|
NameSpace(c) {
|
|||
|
if (c == '{') this.Content();
|
|||
|
else if (c == ',') this.Comma();
|
|||
|
else if (!cfg.blankChar[c]) this.state = this.Ignore;
|
|||
|
}
|
|||
|
Comma() {
|
|||
|
while (cfg.blankChar[this.data[++this.i]]);
|
|||
|
if (this.data[this.i] == '{') this.Content();
|
|||
|
else {
|
|||
|
this.start = this.i--;
|
|||
|
this.state = this.Name;
|
|||
|
}
|
|||
|
}
|
|||
|
Content() {
|
|||
|
this.start = ++this.i;
|
|||
|
if ((this.i = this.data.indexOf('}', this.i)) == -1) this.i = this.data.length;
|
|||
|
var content = this.section();
|
|||
|
for (var i = 0, item; item = this.list[i++];)
|
|||
|
if (this.res[item]) this.res[item] += ';' + content;
|
|||
|
else this.res[item] = content;
|
|||
|
this.list = [];
|
|||
|
this.state = this.Space;
|
|||
|
}
|
|||
|
}
|