| Leo Repp | 58b9f11 | 2021-11-22 11:57:47 +0100 | [diff] [blame^] | 1 | 'use strict'; |
| 2 | |
| 3 | var use = require('use'); |
| 4 | var define = require('define-property'); |
| 5 | var debug = require('debug')('snapdragon:compiler'); |
| 6 | var utils = require('./utils'); |
| 7 | |
| 8 | /** |
| 9 | * Create a new `Compiler` with the given `options`. |
| 10 | * @param {Object} `options` |
| 11 | */ |
| 12 | |
| 13 | function Compiler(options, state) { |
| 14 | debug('initializing', __filename); |
| 15 | this.options = utils.extend({source: 'string'}, options); |
| 16 | this.state = state || {}; |
| 17 | this.compilers = {}; |
| 18 | this.output = ''; |
| 19 | this.set('eos', function(node) { |
| 20 | return this.emit(node.val, node); |
| 21 | }); |
| 22 | this.set('noop', function(node) { |
| 23 | return this.emit(node.val, node); |
| 24 | }); |
| 25 | this.set('bos', function(node) { |
| 26 | return this.emit(node.val, node); |
| 27 | }); |
| 28 | use(this); |
| 29 | } |
| 30 | |
| 31 | /** |
| 32 | * Prototype methods |
| 33 | */ |
| 34 | |
| 35 | Compiler.prototype = { |
| 36 | |
| 37 | /** |
| 38 | * Throw an error message with details including the cursor position. |
| 39 | * @param {String} `msg` Message to use in the Error. |
| 40 | */ |
| 41 | |
| 42 | error: function(msg, node) { |
| 43 | var pos = node.position || {start: {column: 0}}; |
| 44 | var message = this.options.source + ' column:' + pos.start.column + ': ' + msg; |
| 45 | |
| 46 | var err = new Error(message); |
| 47 | err.reason = msg; |
| 48 | err.column = pos.start.column; |
| 49 | err.source = this.pattern; |
| 50 | |
| 51 | if (this.options.silent) { |
| 52 | this.errors.push(err); |
| 53 | } else { |
| 54 | throw err; |
| 55 | } |
| 56 | }, |
| 57 | |
| 58 | /** |
| 59 | * Define a non-enumberable property on the `Compiler` instance. |
| 60 | * |
| 61 | * ```js |
| 62 | * compiler.define('foo', 'bar'); |
| 63 | * ``` |
| 64 | * @name .define |
| 65 | * @param {String} `key` propery name |
| 66 | * @param {any} `val` property value |
| 67 | * @return {Object} Returns the Compiler instance for chaining. |
| 68 | * @api public |
| 69 | */ |
| 70 | |
| 71 | define: function(key, val) { |
| 72 | define(this, key, val); |
| 73 | return this; |
| 74 | }, |
| 75 | |
| 76 | /** |
| 77 | * Emit `node.val` |
| 78 | */ |
| 79 | |
| 80 | emit: function(str, node) { |
| 81 | this.output += str; |
| 82 | return str; |
| 83 | }, |
| 84 | |
| 85 | /** |
| 86 | * Add a compiler `fn` with the given `name` |
| 87 | */ |
| 88 | |
| 89 | set: function(name, fn) { |
| 90 | this.compilers[name] = fn; |
| 91 | return this; |
| 92 | }, |
| 93 | |
| 94 | /** |
| 95 | * Get compiler `name`. |
| 96 | */ |
| 97 | |
| 98 | get: function(name) { |
| 99 | return this.compilers[name]; |
| 100 | }, |
| 101 | |
| 102 | /** |
| 103 | * Get the previous AST node. |
| 104 | */ |
| 105 | |
| 106 | prev: function(n) { |
| 107 | return this.ast.nodes[this.idx - (n || 1)] || { type: 'bos', val: '' }; |
| 108 | }, |
| 109 | |
| 110 | /** |
| 111 | * Get the next AST node. |
| 112 | */ |
| 113 | |
| 114 | next: function(n) { |
| 115 | return this.ast.nodes[this.idx + (n || 1)] || { type: 'eos', val: '' }; |
| 116 | }, |
| 117 | |
| 118 | /** |
| 119 | * Visit `node`. |
| 120 | */ |
| 121 | |
| 122 | visit: function(node, nodes, i) { |
| 123 | var fn = this.compilers[node.type]; |
| 124 | this.idx = i; |
| 125 | |
| 126 | if (typeof fn !== 'function') { |
| 127 | throw this.error('compiler "' + node.type + '" is not registered', node); |
| 128 | } |
| 129 | return fn.call(this, node, nodes, i); |
| 130 | }, |
| 131 | |
| 132 | /** |
| 133 | * Map visit over array of `nodes`. |
| 134 | */ |
| 135 | |
| 136 | mapVisit: function(nodes) { |
| 137 | if (!Array.isArray(nodes)) { |
| 138 | throw new TypeError('expected an array'); |
| 139 | } |
| 140 | var len = nodes.length; |
| 141 | var idx = -1; |
| 142 | while (++idx < len) { |
| 143 | this.visit(nodes[idx], nodes, idx); |
| 144 | } |
| 145 | return this; |
| 146 | }, |
| 147 | |
| 148 | /** |
| 149 | * Compile `ast`. |
| 150 | */ |
| 151 | |
| 152 | compile: function(ast, options) { |
| 153 | var opts = utils.extend({}, this.options, options); |
| 154 | this.ast = ast; |
| 155 | this.parsingErrors = this.ast.errors; |
| 156 | this.output = ''; |
| 157 | |
| 158 | // source map support |
| 159 | if (opts.sourcemap) { |
| 160 | var sourcemaps = require('./source-maps'); |
| 161 | sourcemaps(this); |
| 162 | this.mapVisit(this.ast.nodes); |
| 163 | this.applySourceMaps(); |
| 164 | this.map = opts.sourcemap === 'generator' ? this.map : this.map.toJSON(); |
| 165 | return this; |
| 166 | } |
| 167 | |
| 168 | this.mapVisit(this.ast.nodes); |
| 169 | return this; |
| 170 | } |
| 171 | }; |
| 172 | |
| 173 | /** |
| 174 | * Expose `Compiler` |
| 175 | */ |
| 176 | |
| 177 | module.exports = Compiler; |