Switched jshrink for JSMinPlus
Compacts javascript better, and with caching makes for faster downloads
This commit is contained in:
parent
004944fdd8
commit
2160f23235
1872
config/JSMinPlus.php
Normal file
1872
config/JSMinPlus.php
Normal file
File diff suppressed because it is too large
Load Diff
@ -1,333 +0,0 @@
|
|||||||
<?php
|
|
||||||
/*
|
|
||||||
JShrink
|
|
||||||
|
|
||||||
Copyright (c) 2009, Robert Hafner
|
|
||||||
All rights reserved.
|
|
||||||
|
|
||||||
Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
|
|
||||||
|
|
||||||
* Redistributions of source code must retain the above copyright notice, this list of conditions and the following
|
|
||||||
disclaimer.
|
|
||||||
* Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the
|
|
||||||
following disclaimer in the documentation and/or other materials provided with the distribution.
|
|
||||||
* Neither the name of the <ORGANIZATION> nor the names of its contributors may be used to endorse or promote
|
|
||||||
products derived from this software without specific prior written permission.
|
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
|
|
||||||
INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
||||||
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
||||||
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
||||||
WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
||||||
THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* JShrink
|
|
||||||
*
|
|
||||||
* Usage - JShrink::minify($js);
|
|
||||||
* Usage - JShrink::minify($js, $options);
|
|
||||||
* Usage - JShrink::minify($js, array('flaggedComments' => false));
|
|
||||||
*
|
|
||||||
* @version 0.2
|
|
||||||
* @package JShrink
|
|
||||||
* @author Robert Hafner <tedivm@tedivm.com>
|
|
||||||
* @license http://www.opensource.org/licenses/bsd-license.php
|
|
||||||
*/
|
|
||||||
class JShrink
|
|
||||||
{
|
|
||||||
protected $input;
|
|
||||||
protected $index = 0;
|
|
||||||
|
|
||||||
protected $a = '';
|
|
||||||
protected $b = '';
|
|
||||||
protected $c;
|
|
||||||
|
|
||||||
protected $options;
|
|
||||||
|
|
||||||
static protected $defaultOptions = array('flaggedComments' => true);
|
|
||||||
|
|
||||||
static public function minify($js, $options = array())
|
|
||||||
{
|
|
||||||
try{
|
|
||||||
$currentOptions = array_merge(self::$defaultOptions, $options);
|
|
||||||
|
|
||||||
ob_start();
|
|
||||||
$currentOptions = array_merge(self::$defaultOptions, $options);
|
|
||||||
$me = new JShrink();
|
|
||||||
$me->breakdownScript($js, $currentOptions);
|
|
||||||
$output = ob_get_clean();
|
|
||||||
return $output;
|
|
||||||
|
|
||||||
}catch(Exception $e){
|
|
||||||
ob_end_clean();
|
|
||||||
throw $e;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function breakdownScript($js, $currentOptions)
|
|
||||||
{
|
|
||||||
$this->options = $currentOptions;
|
|
||||||
|
|
||||||
$js = str_replace("\r\n", "\n", $js);
|
|
||||||
$this->input = str_replace("\r", "\n", $js);
|
|
||||||
|
|
||||||
$this->a = $this->getReal();
|
|
||||||
|
|
||||||
// the only time the length can be higher than 1 is if a conditional comment needs to be displayed
|
|
||||||
// and the only time that can happen for $a is on the very first run
|
|
||||||
while(strlen($this->a) > 1)
|
|
||||||
{
|
|
||||||
echo $this->a;
|
|
||||||
$this->a = $this->getReal();
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->b = $this->getReal();
|
|
||||||
|
|
||||||
while($this->a !== false && !is_null($this->a) && $this->a !== '')
|
|
||||||
{
|
|
||||||
|
|
||||||
// now we give $b the same check for conditional comments we gave $a before we began looping
|
|
||||||
if(strlen($this->b) > 1)
|
|
||||||
{
|
|
||||||
echo $this->a . $this->b;
|
|
||||||
$this->a = $this->getReal();
|
|
||||||
$this->b = $this->getReal();
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
switch($this->a)
|
|
||||||
{
|
|
||||||
// new lines
|
|
||||||
case "\n":
|
|
||||||
// if the next line is something that can't stand alone preserver the newline
|
|
||||||
if(strpos('(-+{[@', $this->b) !== false)
|
|
||||||
{
|
|
||||||
echo $this->a;
|
|
||||||
$this->saveString();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// if its a space we move down to the string test below
|
|
||||||
if($this->b === ' ')
|
|
||||||
break;
|
|
||||||
|
|
||||||
// otherwise we treat the newline like a space
|
|
||||||
|
|
||||||
case ' ':
|
|
||||||
if(self::isAlphaNumeric($this->b))
|
|
||||||
echo $this->a;
|
|
||||||
|
|
||||||
$this->saveString();
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
switch($this->b)
|
|
||||||
{
|
|
||||||
case "\n":
|
|
||||||
if(strpos('}])+-"\'', $this->a) !== false)
|
|
||||||
{
|
|
||||||
echo $this->a;
|
|
||||||
$this->saveString();
|
|
||||||
break;
|
|
||||||
}else{
|
|
||||||
if(self::isAlphaNumeric($this->a))
|
|
||||||
{
|
|
||||||
echo $this->a;
|
|
||||||
$this->saveString();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case ' ':
|
|
||||||
if(!self::isAlphaNumeric($this->a))
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
// check for some regex that breaks stuff
|
|
||||||
if($this->a == '/' && ($this->b == '\'' || $this->b == '"'))
|
|
||||||
{
|
|
||||||
$this->saveRegex();
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
echo $this->a;
|
|
||||||
$this->saveString();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// do reg check of doom
|
|
||||||
$this->b = $this->getReal();
|
|
||||||
|
|
||||||
if(($this->b == '/' && strpos('(,=:[!&|?', $this->a) !== false))
|
|
||||||
$this->saveRegex();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function getChar()
|
|
||||||
{
|
|
||||||
if(isset($this->c))
|
|
||||||
{
|
|
||||||
$char = $this->c;
|
|
||||||
unset($this->c);
|
|
||||||
}else{
|
|
||||||
if(isset($this->input[$this->index]))
|
|
||||||
{
|
|
||||||
$char = $this->input[$this->index];
|
|
||||||
$this->index++;
|
|
||||||
}else{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if($char === "\n" || ord($char) >= 32)
|
|
||||||
return $char;
|
|
||||||
|
|
||||||
return ' ';
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function getReal()
|
|
||||||
{
|
|
||||||
$startIndex = $this->index;
|
|
||||||
$char = $this->getChar();
|
|
||||||
|
|
||||||
if($char == '/')
|
|
||||||
{
|
|
||||||
$this->c = $this->getChar();
|
|
||||||
|
|
||||||
if($this->c == '/')
|
|
||||||
{
|
|
||||||
$thirdCommentString = $this->input[$this->index];
|
|
||||||
|
|
||||||
// kill rest of line
|
|
||||||
$char = $this->getNext("\n");
|
|
||||||
|
|
||||||
if($thirdCommentString == '@')
|
|
||||||
{
|
|
||||||
$endPoint = ($this->index) - $startIndex;
|
|
||||||
unset($this->c);
|
|
||||||
$char = "\n" . substr($this->input, $startIndex, $endPoint);// . "\n";
|
|
||||||
}else{
|
|
||||||
$char = $this->getChar();
|
|
||||||
$char = $this->getChar();
|
|
||||||
}
|
|
||||||
|
|
||||||
}elseif($this->c == '*'){
|
|
||||||
|
|
||||||
$this->getChar(); // current C
|
|
||||||
$thirdCommentString = $this->getChar();
|
|
||||||
|
|
||||||
if($thirdCommentString == '@')
|
|
||||||
{
|
|
||||||
// we're gonna back up a bit and and send the comment back, where the first
|
|
||||||
// char will be echoed and the rest will be treated like a string
|
|
||||||
$this->index = $this->index-2;
|
|
||||||
return '/';
|
|
||||||
|
|
||||||
}elseif($this->getNext('*/')){
|
|
||||||
// kill everything up to the next */
|
|
||||||
|
|
||||||
$this->getChar(); // get *
|
|
||||||
$this->getChar(); // get /
|
|
||||||
|
|
||||||
$char = $this->getChar(); // get next real charactor
|
|
||||||
|
|
||||||
// if YUI-style comments are enabled we reinsert it into the stream
|
|
||||||
if($this->options['flaggedComments'] && $thirdCommentString == '!')
|
|
||||||
{
|
|
||||||
$endPoint = ($this->index - 1) - $startIndex;
|
|
||||||
echo "\n" . substr($this->input, $startIndex, $endPoint) . "\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
}else{
|
|
||||||
$char = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if($char === false)
|
|
||||||
throw new JShrinkException('Stray comment. ' . $this->index);
|
|
||||||
|
|
||||||
// if we're here c is part of the comment and therefore tossed
|
|
||||||
if(isset($this->c))
|
|
||||||
unset($this->c);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return $char;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function getNext($string)
|
|
||||||
{
|
|
||||||
$pos = strpos($this->input, $string, $this->index);
|
|
||||||
|
|
||||||
if($pos === false)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
$this->index = $pos ;
|
|
||||||
return $this->input[$this->index];
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function saveString()
|
|
||||||
{
|
|
||||||
$this->a = $this->b;
|
|
||||||
if($this->a == '\'' || $this->a == '"')
|
|
||||||
{
|
|
||||||
// save literal string
|
|
||||||
$stringType = $this->a;
|
|
||||||
|
|
||||||
while(1)
|
|
||||||
{
|
|
||||||
echo $this->a;
|
|
||||||
$this->a = $this->getChar();
|
|
||||||
|
|
||||||
switch($this->a)
|
|
||||||
{
|
|
||||||
case $stringType:
|
|
||||||
break 2;
|
|
||||||
|
|
||||||
case "\n":
|
|
||||||
throw new JShrinkException('Unclosed string. ' . $this->index);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case '\\':
|
|
||||||
echo $this->a;
|
|
||||||
$this->a = $this->getChar();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function saveRegex()
|
|
||||||
{
|
|
||||||
echo $this->a . $this->b;
|
|
||||||
|
|
||||||
while(($this->a = $this->getChar()) !== false)
|
|
||||||
{
|
|
||||||
if($this->a == '/')
|
|
||||||
break;
|
|
||||||
|
|
||||||
if($this->a == '\\')
|
|
||||||
{
|
|
||||||
echo $this->a;
|
|
||||||
$this->a = $this->getChar();
|
|
||||||
}
|
|
||||||
|
|
||||||
if($this->a == "\n")
|
|
||||||
throw new JShrinkException('Stray regex pattern. ' . $this->index);
|
|
||||||
|
|
||||||
echo $this->a;
|
|
||||||
}
|
|
||||||
$this->b = $this->getReal();
|
|
||||||
}
|
|
||||||
|
|
||||||
static protected function isAlphaNumeric($char)
|
|
||||||
{
|
|
||||||
return preg_match('/^[\w\$]$/', $char) === 1 || $char == '/';
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// Adding a custom exception handler for your own projects just means changing this line
|
|
||||||
class JShrinkException extends Exception {}
|
|
||||||
?>
|
|
4
js.php
4
js.php
@ -1,7 +1,7 @@
|
|||||||
<?php
|
<?php
|
||||||
//Get config files
|
//Get config files
|
||||||
require('./config/config.php');
|
require('./config/config.php');
|
||||||
require('./config/jshrink.php');
|
require('./config/JSMinPlus.php');
|
||||||
|
|
||||||
//Include the js groups
|
//Include the js groups
|
||||||
$groups = require("./config/js_groups.php");
|
$groups = require("./config/js_groups.php");
|
||||||
@ -107,7 +107,7 @@ if($last_modified === $requested_time)
|
|||||||
//Determine what to do: rebuild cache, send files as is, or send cache.
|
//Determine what to do: rebuild cache, send files as is, or send cache.
|
||||||
if($cache_modified < $last_modified)
|
if($cache_modified < $last_modified)
|
||||||
{
|
{
|
||||||
$js = trim(JShrink::minify(get_files(), array('flaggedComments' => false)));
|
$js = JSMinPlus::minify(get_files());
|
||||||
$cs = file_put_contents($cache_file, $js);
|
$cs = file_put_contents($cache_file, $js);
|
||||||
|
|
||||||
//Make sure cache file gets created/updated
|
//Make sure cache file gets created/updated
|
||||||
|
Loading…
Reference in New Issue
Block a user