123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365 |
- /*
- ** node-jsmin - a js minifier based on Doug Crockford's jsmin.c, based
- ** on Franck Marica's jsmin.js by Peteris Krumins.
- */
- /*!
- jsmin.js - 2010-01-15
- Author: NanaLich (http://www.cnblogs.com/NanaLich)
- Another patched version for jsmin.js patched by Billy Hoffman,
- this version will try to keep CR LF pairs inside the important comments
- away from being changed into double LF pairs.
- jsmin.js - 2009-11-05
- Author: Billy Hoffman
- This is a patched version of jsmin.js created by Franck Marcia which
- supports important comments denoted with /*! ...
- Permission is hereby granted to use the Javascript version under the same
- conditions as the jsmin.js on which it is based.
- jsmin.js - 2006-08-31
- Author: Franck Marcia
- This work is an adaptation of jsminc.c published by Douglas Crockford.
- Permission is hereby granted to use the Javascript version under the same
- conditions as the jsmin.c on which it is based.
- jsmin.c
- 2006-05-04
- Copyright (c) 2002 Douglas Crockford (www.crockford.com)
- Permission is hereby granted, free of charge, to any person obtaining a copy of
- this software and associated documentation files (the "Software"), to deal in
- the Software without restriction, including without limitation the rights to
- use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
- of the Software, and to permit persons to whom the Software is furnished to do
- so, subject to the following conditions:
- The above copyright notice and this permission notice shall be included in all
- copies or substantial portions of the Software.
- The Software shall be used for Good, not Evil.
- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
- SOFTWARE.
- Update:
- add level:
- 1: minimal, keep linefeeds if single
- 2: normal, the standard algorithm
- 3: agressive, remove any linefeed and doesn't take care of potential
- missing semicolons (can be regressive)
- */
- String.prototype.has = function(c) {
- return this.indexOf(c) > -1;
- };
- exports.jsmin = jsmin;
- function jsmin(input, level, comment) {
- if (!input) return '';
- if (!level) level = 2;
- if (!comment) comment = '';
- var a = '',
- b = '',
- EOF = -1,
- LETTERS = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz',
- DIGITS = '0123456789',
- ALNUM = LETTERS + DIGITS + '_$\\',
- theLookahead = EOF;
- /* isAlphanum -- return true if the character is a letter, digit, underscore,
- dollar sign, or non-ASCII character.
- */
- function isAlphanum(c) {
- return c != EOF && (ALNUM.has(c) || c.charCodeAt(0) > 126);
- }
- /* getc(IC) -- return the next character. Watch out for lookahead. If the
- character is a control character, translate it to a space or
- linefeed.
- */
- var iChar = 0, lInput = input.length;
- function getc() {
- var c = theLookahead;
- if(iChar == lInput) {
- return EOF;
- }
- theLookahead = EOF;
- if(c == EOF) {
- c = input.charAt(iChar);
- ++iChar;
- }
- if(c >= ' ' || c == '\n') {
- return c;
- }
- if(c == '\r') {
- return '\n';
- }
- return ' ';
- }
- function getcIC() {
- var c = theLookahead;
- if(iChar == lInput) {
- return EOF;
- }
- theLookahead = EOF;
- if(c == EOF) {
- c = input.charAt(iChar);
- ++iChar;
- }
- if(c >= ' ' || c == '\n' || c == '\r') {
- return c;
- }
- return ' ';
- }
- /* peek -- get the next character without getting it.
- */
- function peek() {
- theLookahead = getc();
- return theLookahead;
- }
- /* next -- get the next character, excluding comments. peek() is used to see
- if a '/' is followed by a '/' or '*'.
- */
- function next() {
- var c = getc();
- if(c == '/') {
- switch(peek()) {
- case '/':
- for(; ; ) {
- c = getc();
- if(c <= '\n') {
- return c;
- }
- }
- break;
- case '*':
- //this is a comment. What kind?
- getc();
- if(peek() == '!') {
- // kill the extra one
- getc();
- //important comment
- var d = '/*!';
- for(; ; ) {
- c = getcIC(); // let it know it's inside an important comment
- switch(c) {
- case '*':
- if(peek() == '/') {
- getc();
- return d + '*/';
- }
- break;
- case EOF:
- throw 'Error: Unterminated comment.';
- default:
- //modern JS engines handle string concats much better than the
- //array+push+join hack.
- d += c;
- }
- }
- } else {
- //unimportant comment
- for(; ; ) {
- switch(getc()) {
- case '*':
- if(peek() == '/') {
- getc();
- return ' ';
- }
- break;
- case EOF:
- throw 'Error: Unterminated comment.';
- }
- }
- }
- break;
- default:
- return c;
- }
- }
- return c;
- }
- /* action -- do something! What you do is determined by the argument:
- 1 Output A. Copy B to A. Get the next B.
- 2 Copy B to A. Get the next B. (Delete A).
- 3 Get the next B. (Delete B).
- action treats a string as a single character. Wow!
- action recognizes a regular expression if it is preceded by ( or , or =.
- */
- function action(d) {
- var r = [];
- if(d == 1) {
- r.push(a);
- }
- if(d < 3) {
- a = b;
- if(a == '\'' || a == '"') {
- for(; ; ) {
- r.push(a);
- a = getc();
- if(a == b) {
- break;
- }
- if(a <= '\n') {
- throw 'Error: unterminated string literal: ' + a;
- }
- if(a == '\\') {
- r.push(a);
- a = getc();
- }
- }
- }
- }
- b = next();
- if(b == '/' && '(,=:[!&|'.has(a)) {
- r.push(a);
- r.push(b);
- for(; ; ) {
- a = getc();
- if(a == '/') {
- break;
- } else if(a == '\\') {
- r.push(a);
- a = getc();
- } else if(a <= '\n') {
- throw 'Error: unterminated Regular Expression literal';
- }
- r.push(a);
- }
- b = next();
- }
- return r.join('');
- }
- /* m -- Copy the input to the output, deleting the characters which are
- insignificant to JavaScript. Comments will be removed. Tabs will be
- replaced with spaces. Carriage returns will be replaced with
- linefeeds.
- Most spaces and linefeeds will be removed.
- */
- function m() {
- var r = [];
- a = '';
- r.push(action(3));
- while(a != EOF) {
- switch(a) {
- case ' ':
- if(isAlphanum(b)) {
- r.push(action(1));
- } else {
- r.push(action(2));
- }
- break;
- case '\n':
- switch(b) {
- case '{':
- case '[':
- case '(':
- case '+':
- case '-':
- r.push(action(1));
- break;
- case ' ':
- r.push(action(3));
- break;
- default:
- if(isAlphanum(b)) {
- r.push(action(1));
- } else {
- if(level == 1 && b != '\n') {
- r.push(action(1));
- } else {
- r.push(action(2));
- }
- }
- }
- break;
- default:
- switch(b) {
- case ' ':
- if(isAlphanum(a)) {
- r.push(action(1));
- break;
- }
- r.push(action(3));
- break;
- case '\n':
- if(level == 1 && a != '\n') {
- r.push(action(1));
- } else {
- switch(a) {
- case '}':
- case ']':
- case ')':
- case '+':
- case '-':
- case '"':
- case '\'':
- if(level == 3) {
- r.push(action(3));
- } else {
- r.push(action(1));
- }
- break;
- default:
- if(isAlphanum(a)) {
- r.push(action(1));
- } else {
- r.push(action(3));
- }
- }
- }
- break;
- default:
- r.push(action(1));
- break;
- }
- }
- }
- return r.join('');
- }
- ret = m(input);
- if (comment) {
- return comment + '\n' + ret;
- }
- return ret;
- }
|