|
|
- '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);
- };
|