166 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			166 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
var TYPE = require('../../tokenizer').TYPE;
 | 
						|
 | 
						|
var IDENT = TYPE.Ident;
 | 
						|
var STRING = TYPE.String;
 | 
						|
var COLON = TYPE.Colon;
 | 
						|
var LEFTSQUAREBRACKET = TYPE.LeftSquareBracket;
 | 
						|
var RIGHTSQUAREBRACKET = TYPE.RightSquareBracket;
 | 
						|
var DOLLARSIGN = 0x0024;       // U+0024 DOLLAR SIGN ($)
 | 
						|
var ASTERISK = 0x002A;         // U+002A ASTERISK (*)
 | 
						|
var EQUALSSIGN = 0x003D;       // U+003D EQUALS SIGN (=)
 | 
						|
var CIRCUMFLEXACCENT = 0x005E; // U+005E (^)
 | 
						|
var VERTICALLINE = 0x007C;     // U+007C VERTICAL LINE (|)
 | 
						|
var TILDE = 0x007E;            // U+007E TILDE (~)
 | 
						|
 | 
						|
function getAttributeName() {
 | 
						|
    if (this.scanner.eof) {
 | 
						|
        this.error('Unexpected end of input');
 | 
						|
    }
 | 
						|
 | 
						|
    var start = this.scanner.tokenStart;
 | 
						|
    var expectIdent = false;
 | 
						|
    var checkColon = true;
 | 
						|
 | 
						|
    if (this.scanner.isDelim(ASTERISK)) {
 | 
						|
        expectIdent = true;
 | 
						|
        checkColon = false;
 | 
						|
        this.scanner.next();
 | 
						|
    } else if (!this.scanner.isDelim(VERTICALLINE)) {
 | 
						|
        this.eat(IDENT);
 | 
						|
    }
 | 
						|
 | 
						|
    if (this.scanner.isDelim(VERTICALLINE)) {
 | 
						|
        if (this.scanner.source.charCodeAt(this.scanner.tokenStart + 1) !== EQUALSSIGN) {
 | 
						|
            this.scanner.next();
 | 
						|
            this.eat(IDENT);
 | 
						|
        } else if (expectIdent) {
 | 
						|
            this.error('Identifier is expected', this.scanner.tokenEnd);
 | 
						|
        }
 | 
						|
    } else if (expectIdent) {
 | 
						|
        this.error('Vertical line is expected');
 | 
						|
    }
 | 
						|
 | 
						|
    if (checkColon && this.scanner.tokenType === COLON) {
 | 
						|
        this.scanner.next();
 | 
						|
        this.eat(IDENT);
 | 
						|
    }
 | 
						|
 | 
						|
    return {
 | 
						|
        type: 'Identifier',
 | 
						|
        loc: this.getLocation(start, this.scanner.tokenStart),
 | 
						|
        name: this.scanner.substrToCursor(start)
 | 
						|
    };
 | 
						|
}
 | 
						|
 | 
						|
function getOperator() {
 | 
						|
    var start = this.scanner.tokenStart;
 | 
						|
    var code = this.scanner.source.charCodeAt(start);
 | 
						|
 | 
						|
    if (code !== EQUALSSIGN &&        // =
 | 
						|
        code !== TILDE &&             // ~=
 | 
						|
        code !== CIRCUMFLEXACCENT &&  // ^=
 | 
						|
        code !== DOLLARSIGN &&        // $=
 | 
						|
        code !== ASTERISK &&          // *=
 | 
						|
        code !== VERTICALLINE         // |=
 | 
						|
    ) {
 | 
						|
        this.error('Attribute selector (=, ~=, ^=, $=, *=, |=) is expected');
 | 
						|
    }
 | 
						|
 | 
						|
    this.scanner.next();
 | 
						|
 | 
						|
    if (code !== EQUALSSIGN) {
 | 
						|
        if (!this.scanner.isDelim(EQUALSSIGN)) {
 | 
						|
            this.error('Equal sign is expected');
 | 
						|
        }
 | 
						|
 | 
						|
        this.scanner.next();
 | 
						|
    }
 | 
						|
 | 
						|
    return this.scanner.substrToCursor(start);
 | 
						|
}
 | 
						|
 | 
						|
// '[' <wq-name> ']'
 | 
						|
// '[' <wq-name> <attr-matcher> [ <string-token> | <ident-token> ] <attr-modifier>? ']'
 | 
						|
module.exports = {
 | 
						|
    name: 'AttributeSelector',
 | 
						|
    structure: {
 | 
						|
        name: 'Identifier',
 | 
						|
        matcher: [String, null],
 | 
						|
        value: ['String', 'Identifier', null],
 | 
						|
        flags: [String, null]
 | 
						|
    },
 | 
						|
    parse: function() {
 | 
						|
        var start = this.scanner.tokenStart;
 | 
						|
        var name;
 | 
						|
        var matcher = null;
 | 
						|
        var value = null;
 | 
						|
        var flags = null;
 | 
						|
 | 
						|
        this.eat(LEFTSQUAREBRACKET);
 | 
						|
        this.scanner.skipSC();
 | 
						|
 | 
						|
        name = getAttributeName.call(this);
 | 
						|
        this.scanner.skipSC();
 | 
						|
 | 
						|
        if (this.scanner.tokenType !== RIGHTSQUAREBRACKET) {
 | 
						|
            // avoid case `[name i]`
 | 
						|
            if (this.scanner.tokenType !== IDENT) {
 | 
						|
                matcher = getOperator.call(this);
 | 
						|
 | 
						|
                this.scanner.skipSC();
 | 
						|
 | 
						|
                value = this.scanner.tokenType === STRING
 | 
						|
                    ? this.String()
 | 
						|
                    : this.Identifier();
 | 
						|
 | 
						|
                this.scanner.skipSC();
 | 
						|
            }
 | 
						|
 | 
						|
            // attribute flags
 | 
						|
            if (this.scanner.tokenType === IDENT) {
 | 
						|
                flags = this.scanner.getTokenValue();
 | 
						|
                this.scanner.next();
 | 
						|
 | 
						|
                this.scanner.skipSC();
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        this.eat(RIGHTSQUAREBRACKET);
 | 
						|
 | 
						|
        return {
 | 
						|
            type: 'AttributeSelector',
 | 
						|
            loc: this.getLocation(start, this.scanner.tokenStart),
 | 
						|
            name: name,
 | 
						|
            matcher: matcher,
 | 
						|
            value: value,
 | 
						|
            flags: flags
 | 
						|
        };
 | 
						|
    },
 | 
						|
    generate: function(node) {
 | 
						|
        var flagsPrefix = ' ';
 | 
						|
 | 
						|
        this.chunk('[');
 | 
						|
        this.node(node.name);
 | 
						|
 | 
						|
        if (node.matcher !== null) {
 | 
						|
            this.chunk(node.matcher);
 | 
						|
 | 
						|
            if (node.value !== null) {
 | 
						|
                this.node(node.value);
 | 
						|
 | 
						|
                // space between string and flags is not required
 | 
						|
                if (node.value.type === 'String') {
 | 
						|
                    flagsPrefix = '';
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        if (node.flags !== null) {
 | 
						|
            this.chunk(flagsPrefix);
 | 
						|
            this.chunk(node.flags);
 | 
						|
        }
 | 
						|
 | 
						|
        this.chunk(']');
 | 
						|
    }
 | 
						|
};
 |