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.
		
		
		
		
		
			
		
			
				
					
					
						
							145 lines
						
					
					
						
							3.5 KiB
						
					
					
				
			
		
		
	
	
							145 lines
						
					
					
						
							3.5 KiB
						
					
					
				'use strict'; | 
						|
 | 
						|
var fs = require('fs'); | 
						|
var path = require('path'); | 
						|
var define = require('define-property'); | 
						|
var utils = require('./utils'); | 
						|
 | 
						|
/** | 
						|
 * Expose `mixin()`. | 
						|
 * This code is based on `source-maps-support.js` in reworkcss/css | 
						|
 * https://github.com/reworkcss/css/blob/master/lib/stringify/source-map-support.js | 
						|
 * Copyright (c) 2012 TJ Holowaychuk <tj@vision-media.ca> | 
						|
 */ | 
						|
 | 
						|
module.exports = mixin; | 
						|
 | 
						|
/** | 
						|
 * Mixin source map support into `compiler`. | 
						|
 * | 
						|
 * @param {Object} `compiler` | 
						|
 * @api public | 
						|
 */ | 
						|
 | 
						|
function mixin(compiler) { | 
						|
  define(compiler, '_comment', compiler.comment); | 
						|
  compiler.map = new utils.SourceMap.SourceMapGenerator(); | 
						|
  compiler.position = { line: 1, column: 1 }; | 
						|
  compiler.content = {}; | 
						|
  compiler.files = {}; | 
						|
 | 
						|
  for (var key in exports) { | 
						|
    define(compiler, key, exports[key]); | 
						|
  } | 
						|
} | 
						|
 | 
						|
/** | 
						|
 * Update position. | 
						|
 * | 
						|
 * @param {String} str | 
						|
 */ | 
						|
 | 
						|
exports.updatePosition = function(str) { | 
						|
  var lines = str.match(/\n/g); | 
						|
  if (lines) this.position.line += lines.length; | 
						|
  var i = str.lastIndexOf('\n'); | 
						|
  this.position.column = ~i ? str.length - i : this.position.column + str.length; | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Emit `str` with `position`. | 
						|
 * | 
						|
 * @param {String} str | 
						|
 * @param {Object} [pos] | 
						|
 * @return {String} | 
						|
 */ | 
						|
 | 
						|
exports.emit = function(str, node) { | 
						|
  var position = node.position || {}; | 
						|
  var source = position.source; | 
						|
  if (source) { | 
						|
    if (position.filepath) { | 
						|
      source = utils.unixify(position.filepath); | 
						|
    } | 
						|
 | 
						|
    this.map.addMapping({ | 
						|
      source: source, | 
						|
      generated: { | 
						|
        line: this.position.line, | 
						|
        column: Math.max(this.position.column - 1, 0) | 
						|
      }, | 
						|
      original: { | 
						|
        line: position.start.line, | 
						|
        column: position.start.column - 1 | 
						|
      } | 
						|
    }); | 
						|
 | 
						|
    if (position.content) { | 
						|
      this.addContent(source, position); | 
						|
    } | 
						|
    if (position.filepath) { | 
						|
      this.addFile(source, position); | 
						|
    } | 
						|
 | 
						|
    this.updatePosition(str); | 
						|
    this.output += str; | 
						|
  } | 
						|
  return str; | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Adds a file to the source map output if it has not already been added | 
						|
 * @param {String} `file` | 
						|
 * @param {Object} `pos` | 
						|
 */ | 
						|
 | 
						|
exports.addFile = function(file, position) { | 
						|
  if (typeof position.content !== 'string') return; | 
						|
  if (Object.prototype.hasOwnProperty.call(this.files, file)) return; | 
						|
  this.files[file] = position.content; | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Adds a content source to the source map output if it has not already been added | 
						|
 * @param {String} `source` | 
						|
 * @param {Object} `position` | 
						|
 */ | 
						|
 | 
						|
exports.addContent = function(source, position) { | 
						|
  if (typeof position.content !== 'string') return; | 
						|
  if (Object.prototype.hasOwnProperty.call(this.content, source)) return; | 
						|
  this.map.setSourceContent(source, position.content); | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Applies any original source maps to the output and embeds the source file | 
						|
 * contents in the source map. | 
						|
 */ | 
						|
 | 
						|
exports.applySourceMaps = function() { | 
						|
  Object.keys(this.files).forEach(function(file) { | 
						|
    var content = this.files[file]; | 
						|
    this.map.setSourceContent(file, content); | 
						|
 | 
						|
    if (this.options.inputSourcemaps === true) { | 
						|
      var originalMap = utils.sourceMapResolve.resolveSync(content, file, fs.readFileSync); | 
						|
      if (originalMap) { | 
						|
        var map = new utils.SourceMap.SourceMapConsumer(originalMap.map); | 
						|
        var relativeTo = originalMap.sourcesRelativeTo; | 
						|
        this.map.applySourceMap(map, file, utils.unixify(path.dirname(relativeTo))); | 
						|
      } | 
						|
    } | 
						|
  }, this); | 
						|
}; | 
						|
 | 
						|
/** | 
						|
 * Process comments, drops sourceMap comments. | 
						|
 * @param {Object} node | 
						|
 */ | 
						|
 | 
						|
exports.comment = function(node) { | 
						|
  if (/^# sourceMappingURL=/.test(node.comment)) { | 
						|
    return this.emit('', node.position); | 
						|
  } | 
						|
  return this._comment(node); | 
						|
};
 | 
						|
 |