import { locate } from 'locate-character'; import fragment from './state/fragment.js'; import { whitespace } from './patterns.js'; import { trimStart, trimEnd } from './utils/trim.js'; import spaces from '../utils/spaces.js'; import hash from './utils/hash.js'; function tabsToSpaces ( str ) { return str.replace( /^\t+/, match => match.split( '\t' ).join( ' ' ) ); } function ParseError ( message, template, index ) { const { line, column } = locate( template, index ); const lines = template.split( '\n' ); const frameStart = Math.max( 0, line - 2 ); const frameEnd = Math.min( line + 3, lines.length ); const digits = String( frameEnd + 1 ).length; const frame = lines .slice( frameStart, frameEnd ) .map( ( str, i ) => { const isErrorLine = frameStart + i === line; let lineNum = String( i + frameStart + 1 ); while ( lineNum.length < digits ) lineNum = ` ${lineNum}`; if ( isErrorLine ) { const indicator = spaces( digits + 2 + tabsToSpaces( str.slice( 0, column ) ).length ) + '^'; return `${lineNum}: ${tabsToSpaces( str )}\n${indicator}`; } return `${lineNum}: ${tabsToSpaces( str )}`; }) .join( '\n' ); this.name = 'ParseError'; this.message = `${message} (${line + 1}:${column})\n${frame}`; this.loc = { line: line + 1, column }; this.pos = index; this.shortMessage = message; } export default function parse ( template ) { if ( typeof template !== 'string' ) { throw new TypeError( 'Template must be a string' ); } const parser = { index: 0, template, stack: [], current () { return this.stack[ this.stack.length - 1 ]; }, acornError ( err ) { parser.error( err.message.replace( / \(\d+:\d+\)$/, '' ), err.pos ); }, error ( message, index = this.index ) { throw new ParseError( message, this.template, index ); }, eat ( str, required ) { if ( this.match( str ) ) { this.index += str.length; return true; } if ( required ) { this.error( `Expected ${str}` ); } }, match ( str ) { return this.template.slice( this.index, this.index + str.length ) === str; }, allowWhitespace () { while ( this.index < this.template.length && whitespace.test( this.template[ this.index ] ) ) { this.index++; } }, read ( pattern ) { const match = pattern.exec( this.template.slice( this.index ) ); if ( !match || match.index !== 0 ) return null; parser.index += match[0].length; return match[0]; }, readUntil ( pattern ) { if ( this.index >= this.template.length ) parser.error( 'Unexpected end of input' ); const start = this.index; const match = pattern.exec( this.template.slice( start ) ); if ( match ) { const start = this.index; this.index = start + match.index; return this.template.slice( start, this.index ); } this.index = this.template.length; return this.template.slice( start ); }, remaining () { return this.template.slice( this.index ); }, requireWhitespace () { if ( !whitespace.test( this.template[ this.index ] ) ) { this.error( `Expected whitespace` ); } this.allowWhitespace(); }, html: { start: null, end: null, type: 'Fragment', children: [] }, css: null, js: null }; parser.stack.push( parser.html ); let state = fragment; while ( parser.index < parser.template.length ) { state = state( parser ) || fragment; } if ( state !== fragment || parser.stack.length > 1 ) { parser.error( 'Unexpected end of input' ); } // trim unnecessary whitespace while ( parser.html.children.length ) { const firstChild = parser.html.children[0]; parser.html.start = firstChild.start; if ( firstChild.type !== 'Text' ) break; const length = firstChild.data.length; firstChild.data = trimStart( firstChild.data ); if ( firstChild.data === '' ) { parser.html.children.shift(); } else { parser.html.start += length - firstChild.data.length; break; } } while ( parser.html.children.length ) { const lastChild = parser.html.children[ parser.html.children.length - 1 ]; parser.html.end = lastChild.end; if ( lastChild.type !== 'Text' ) break; const length = lastChild.data.length; lastChild.data = trimEnd( lastChild.data ); if ( lastChild.data === '' ) { parser.html.children.pop(); } else { parser.html.end -= length - lastChild.data.length; break; } } return { hash: hash( template ), html: parser.html, css: parser.css, js: parser.js }; }