|
|
var List = require('css-tree').List; var clone = require('css-tree').clone; var usageUtils = require('./usage'); var clean = require('./clean'); var replace = require('./replace'); var restructure = require('./restructure'); var walk = require('css-tree').walk;
function readChunk(children, specialComments) { var buffer = new List(); var nonSpaceTokenInBuffer = false; var protectedComment;
children.nextUntil(children.head, function(node, item, list) { if (node.type === 'Comment') { if (!specialComments || node.value.charAt(0) !== '!') { list.remove(item); return; }
if (nonSpaceTokenInBuffer || protectedComment) { return true; }
list.remove(item); protectedComment = node; return; }
if (node.type !== 'WhiteSpace') { nonSpaceTokenInBuffer = true; }
buffer.insert(list.remove(item)); });
return { comment: protectedComment, stylesheet: { type: 'StyleSheet', loc: null, children: buffer } }; }
function compressChunk(ast, firstAtrulesAllowed, num, options) { options.logger('Compress block #' + num, null, true);
var seed = 1;
if (ast.type === 'StyleSheet') { ast.firstAtrulesAllowed = firstAtrulesAllowed; ast.id = seed++; }
walk(ast, { visit: 'Atrule', enter: function markScopes(node) { if (node.block !== null) { node.block.id = seed++; } } }); options.logger('init', ast);
// remove redundant
clean(ast, options); options.logger('clean', ast);
// replace nodes for shortened forms
replace(ast, options); options.logger('replace', ast);
// structure optimisations
if (options.restructuring) { restructure(ast, options); }
return ast; }
function getCommentsOption(options) { var comments = 'comments' in options ? options.comments : 'exclamation';
if (typeof comments === 'boolean') { comments = comments ? 'exclamation' : false; } else if (comments !== 'exclamation' && comments !== 'first-exclamation') { comments = false; }
return comments; }
function getRestructureOption(options) { if ('restructure' in options) { return options.restructure; }
return 'restructuring' in options ? options.restructuring : true; }
function wrapBlock(block) { return new List().appendData({ type: 'Rule', loc: null, prelude: { type: 'SelectorList', loc: null, children: new List().appendData({ type: 'Selector', loc: null, children: new List().appendData({ type: 'TypeSelector', loc: null, name: 'x' }) }) }, block: block }); }
module.exports = function compress(ast, options) { ast = ast || { type: 'StyleSheet', loc: null, children: new List() }; options = options || {};
var compressOptions = { logger: typeof options.logger === 'function' ? options.logger : function() {}, restructuring: getRestructureOption(options), forceMediaMerge: Boolean(options.forceMediaMerge), usage: options.usage ? usageUtils.buildIndex(options.usage) : false }; var specialComments = getCommentsOption(options); var firstAtrulesAllowed = true; var input; var output = new List(); var chunk; var chunkNum = 1; var chunkChildren;
if (options.clone) { ast = clone(ast); }
if (ast.type === 'StyleSheet') { input = ast.children; ast.children = output; } else { input = wrapBlock(ast); }
do { chunk = readChunk(input, Boolean(specialComments)); compressChunk(chunk.stylesheet, firstAtrulesAllowed, chunkNum++, compressOptions); chunkChildren = chunk.stylesheet.children;
if (chunk.comment) { // add \n before comment if there is another content in output
if (!output.isEmpty()) { output.insert(List.createItem({ type: 'Raw', value: '\n' })); }
output.insert(List.createItem(chunk.comment));
// add \n after comment if chunk is not empty
if (!chunkChildren.isEmpty()) { output.insert(List.createItem({ type: 'Raw', value: '\n' })); } }
if (firstAtrulesAllowed && !chunkChildren.isEmpty()) { var lastRule = chunkChildren.last();
if (lastRule.type !== 'Atrule' || (lastRule.name !== 'import' && lastRule.name !== 'charset')) { firstAtrulesAllowed = false; } }
if (specialComments !== 'exclamation') { specialComments = false; }
output.appendList(chunkChildren); } while (!input.isEmpty());
return { ast: ast }; };
|