102 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
			
		
		
	
	
			102 lines
		
	
	
		
			2.9 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;
 | 
						||
  }
 | 
						||
} |