You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
			
				
					134 lines
				
				3.3 KiB
			
		
		
			
		
	
	
					134 lines
				
				3.3 KiB
			| 
											10 months ago
										 | 'use strict' | ||
|  | 
 | ||
|  | let pico = require('picocolors') | ||
|  | 
 | ||
|  | let terminalHighlight = require('./terminal-highlight') | ||
|  | 
 | ||
|  | class CssSyntaxError extends Error { | ||
|  |   constructor(message, line, column, source, file, plugin) { | ||
|  |     super(message) | ||
|  |     this.name = 'CssSyntaxError' | ||
|  |     this.reason = message | ||
|  | 
 | ||
|  |     if (file) { | ||
|  |       this.file = file | ||
|  |     } | ||
|  |     if (source) { | ||
|  |       this.source = source | ||
|  |     } | ||
|  |     if (plugin) { | ||
|  |       this.plugin = plugin | ||
|  |     } | ||
|  |     if (typeof line !== 'undefined' && typeof column !== 'undefined') { | ||
|  |       if (typeof line === 'number') { | ||
|  |         this.line = line | ||
|  |         this.column = column | ||
|  |       } else { | ||
|  |         this.line = line.line | ||
|  |         this.column = line.column | ||
|  |         this.endLine = column.line | ||
|  |         this.endColumn = column.column | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     this.setMessage() | ||
|  | 
 | ||
|  |     if (Error.captureStackTrace) { | ||
|  |       Error.captureStackTrace(this, CssSyntaxError) | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   setMessage() { | ||
|  |     this.message = this.plugin ? this.plugin + ': ' : '' | ||
|  |     this.message += this.file ? this.file : '<css input>' | ||
|  |     if (typeof this.line !== 'undefined') { | ||
|  |       this.message += ':' + this.line + ':' + this.column | ||
|  |     } | ||
|  |     this.message += ': ' + this.reason | ||
|  |   } | ||
|  | 
 | ||
|  |   showSourceCode(color) { | ||
|  |     if (!this.source) return '' | ||
|  | 
 | ||
|  |     let css = this.source | ||
|  |     if (color == null) color = pico.isColorSupported | ||
|  | 
 | ||
|  |     let aside = text => text | ||
|  |     let mark = text => text | ||
|  |     let highlight = text => text | ||
|  |     if (color) { | ||
|  |       let { bold, gray, red } = pico.createColors(true) | ||
|  |       mark = text => bold(red(text)) | ||
|  |       aside = text => gray(text) | ||
|  |       if (terminalHighlight) { | ||
|  |         highlight = text => terminalHighlight(text) | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     let lines = css.split(/\r?\n/) | ||
|  |     let start = Math.max(this.line - 3, 0) | ||
|  |     let end = Math.min(this.line + 2, lines.length) | ||
|  |     let maxWidth = String(end).length | ||
|  | 
 | ||
|  |     return lines | ||
|  |       .slice(start, end) | ||
|  |       .map((line, index) => { | ||
|  |         let number = start + 1 + index | ||
|  |         let gutter = ' ' + (' ' + number).slice(-maxWidth) + ' | ' | ||
|  |         if (number === this.line) { | ||
|  |           if (line.length > 160) { | ||
|  |             let padding = 20 | ||
|  |             let subLineStart = Math.max(0, this.column - padding) | ||
|  |             let subLineEnd = Math.max( | ||
|  |               this.column + padding, | ||
|  |               this.endColumn + padding | ||
|  |             ) | ||
|  |             let subLine = line.slice(subLineStart, subLineEnd) | ||
|  | 
 | ||
|  |             let spacing = | ||
|  |               aside(gutter.replace(/\d/g, ' ')) + | ||
|  |               line | ||
|  |                 .slice(0, Math.min(this.column - 1, padding - 1)) | ||
|  |                 .replace(/[^\t]/g, ' ') | ||
|  | 
 | ||
|  |             return ( | ||
|  |               mark('>') + | ||
|  |               aside(gutter) + | ||
|  |               highlight(subLine) + | ||
|  |               '\n ' + | ||
|  |               spacing + | ||
|  |               mark('^') | ||
|  |             ) | ||
|  |           } | ||
|  | 
 | ||
|  |           let spacing = | ||
|  |             aside(gutter.replace(/\d/g, ' ')) + | ||
|  |             line.slice(0, this.column - 1).replace(/[^\t]/g, ' ') | ||
|  | 
 | ||
|  |           return ( | ||
|  |             mark('>') + | ||
|  |             aside(gutter) + | ||
|  |             highlight(line) + | ||
|  |             '\n ' + | ||
|  |             spacing + | ||
|  |             mark('^') | ||
|  |           ) | ||
|  |         } | ||
|  | 
 | ||
|  |         return ' ' + aside(gutter) + highlight(line) | ||
|  |       }) | ||
|  |       .join('\n') | ||
|  |   } | ||
|  | 
 | ||
|  |   toString() { | ||
|  |     let code = this.showSourceCode() | ||
|  |     if (code) { | ||
|  |       code = '\n\n' + code + '\n' | ||
|  |     } | ||
|  |     return this.name + ': ' + this.message + code | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | module.exports = CssSyntaxError | ||
|  | CssSyntaxError.default = CssSyntaxError |