forked from GNUsocial/gnu-social
		
	Added minify plugin
This commit is contained in:
		
							
								
								
									
										1370
									
								
								plugins/Minify/extlib/minify/min/lib/FirePHP.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1370
									
								
								plugins/Minify/extlib/minify/min/lib/FirePHP.php
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										348
									
								
								plugins/Minify/extlib/minify/min/lib/HTTP/ConditionalGet.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										348
									
								
								plugins/Minify/extlib/minify/min/lib/HTTP/ConditionalGet.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,348 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class HTTP_ConditionalGet  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @subpackage HTTP
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Implement conditional GET via a timestamp or hash of content
 | 
			
		||||
 *
 | 
			
		||||
 * E.g. Content from DB with update time:
 | 
			
		||||
 * <code>
 | 
			
		||||
 * list($updateTime, $content) = getDbUpdateAndContent();
 | 
			
		||||
 * $cg = new HTTP_ConditionalGet(array(
 | 
			
		||||
 *     'lastModifiedTime' => $updateTime
 | 
			
		||||
 *     ,'isPublic' => true
 | 
			
		||||
 * ));
 | 
			
		||||
 * $cg->sendHeaders();
 | 
			
		||||
 * if ($cg->cacheIsValid) {
 | 
			
		||||
 *     exit();
 | 
			
		||||
 * }
 | 
			
		||||
 * echo $content;
 | 
			
		||||
 * </code>
 | 
			
		||||
 * 
 | 
			
		||||
 * E.g. Shortcut for the above
 | 
			
		||||
 * <code>
 | 
			
		||||
 * HTTP_ConditionalGet::check($updateTime, true); // exits if client has cache
 | 
			
		||||
 * echo $content;
 | 
			
		||||
 * </code>
 | 
			
		||||
 *
 | 
			
		||||
 * E.g. Content from DB with no update time:
 | 
			
		||||
 * <code>
 | 
			
		||||
 * $content = getContentFromDB();
 | 
			
		||||
 * $cg = new HTTP_ConditionalGet(array(
 | 
			
		||||
 *     'contentHash' => md5($content)
 | 
			
		||||
 * ));
 | 
			
		||||
 * $cg->sendHeaders();
 | 
			
		||||
 * if ($cg->cacheIsValid) {
 | 
			
		||||
 *     exit();
 | 
			
		||||
 * }
 | 
			
		||||
 * echo $content;
 | 
			
		||||
 * </code>
 | 
			
		||||
 * 
 | 
			
		||||
 * E.g. Static content with some static includes:
 | 
			
		||||
 * <code>
 | 
			
		||||
 * // before content
 | 
			
		||||
 * $cg = new HTTP_ConditionalGet(array(
 | 
			
		||||
 *     'lastUpdateTime' => max(
 | 
			
		||||
 *         filemtime(__FILE__)
 | 
			
		||||
 *         ,filemtime('/path/to/header.inc')
 | 
			
		||||
 *         ,filemtime('/path/to/footer.inc')
 | 
			
		||||
 *     )
 | 
			
		||||
 * ));
 | 
			
		||||
 * $cg->sendHeaders();
 | 
			
		||||
 * if ($cg->cacheIsValid) {
 | 
			
		||||
 *     exit();
 | 
			
		||||
 * }
 | 
			
		||||
 * </code>
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @subpackage HTTP
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 */
 | 
			
		||||
class HTTP_ConditionalGet {
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Does the client have a valid copy of the requested resource?
 | 
			
		||||
     * 
 | 
			
		||||
     * You'll want to check this after instantiating the object. If true, do
 | 
			
		||||
     * not send content, just call sendHeaders() if you haven't already.
 | 
			
		||||
     *
 | 
			
		||||
     * @var bool
 | 
			
		||||
     */
 | 
			
		||||
    public $cacheIsValid = null;
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * @param array $spec options
 | 
			
		||||
     * 
 | 
			
		||||
     * 'isPublic': (bool) if true, the Cache-Control header will contain 
 | 
			
		||||
     * "public", allowing proxies to cache the content. Otherwise "private" will 
 | 
			
		||||
     * be sent, allowing only browser caching. (default false)
 | 
			
		||||
     * 
 | 
			
		||||
     * 'lastModifiedTime': (int) if given, both ETag AND Last-Modified headers
 | 
			
		||||
     * will be sent with content. This is recommended.
 | 
			
		||||
     *
 | 
			
		||||
     * 'encoding': (string) if set, the header "Vary: Accept-Encoding" will
 | 
			
		||||
     * always be sent and a truncated version of the encoding will be appended
 | 
			
		||||
     * to the ETag. E.g. "pub123456;gz". This will also trigger a more lenient 
 | 
			
		||||
     * checking of the client's If-None-Match header, as the encoding portion of
 | 
			
		||||
     * the ETag will be stripped before comparison.
 | 
			
		||||
     * 
 | 
			
		||||
     * 'contentHash': (string) if given, only the ETag header can be sent with
 | 
			
		||||
     * content (only HTTP1.1 clients can conditionally GET). The given string 
 | 
			
		||||
     * should be short with no quote characters and always change when the 
 | 
			
		||||
     * resource changes (recommend md5()). This is not needed/used if 
 | 
			
		||||
     * lastModifiedTime is given.
 | 
			
		||||
     * 
 | 
			
		||||
     * 'eTag': (string) if given, this will be used as the ETag header rather
 | 
			
		||||
     * than values based on lastModifiedTime or contentHash. Also the encoding
 | 
			
		||||
     * string will not be appended to the given value as described above.
 | 
			
		||||
     * 
 | 
			
		||||
     * 'invalidate': (bool) if true, the client cache will be considered invalid
 | 
			
		||||
     * without testing. Effectively this disables conditional GET. 
 | 
			
		||||
     * (default false)
 | 
			
		||||
     * 
 | 
			
		||||
     * 'maxAge': (int) if given, this will set the Cache-Control max-age in 
 | 
			
		||||
     * seconds, and also set the Expires header to the equivalent GMT date. 
 | 
			
		||||
     * After the max-age period has passed, the browser will again send a 
 | 
			
		||||
     * conditional GET to revalidate its cache.
 | 
			
		||||
     * 
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    public function __construct($spec)
 | 
			
		||||
    {
 | 
			
		||||
        $scope = (isset($spec['isPublic']) && $spec['isPublic'])
 | 
			
		||||
            ? 'public'
 | 
			
		||||
            : 'private';
 | 
			
		||||
        $maxAge = 0;
 | 
			
		||||
        // backwards compatibility (can be removed later)
 | 
			
		||||
        if (isset($spec['setExpires']) 
 | 
			
		||||
            && is_numeric($spec['setExpires'])
 | 
			
		||||
            && ! isset($spec['maxAge'])) {
 | 
			
		||||
            $spec['maxAge'] = $spec['setExpires'] - $_SERVER['REQUEST_TIME'];
 | 
			
		||||
        }
 | 
			
		||||
        if (isset($spec['maxAge'])) {
 | 
			
		||||
            $maxAge = $spec['maxAge'];
 | 
			
		||||
            $this->_headers['Expires'] = self::gmtDate(
 | 
			
		||||
                $_SERVER['REQUEST_TIME'] + $spec['maxAge'] 
 | 
			
		||||
            );
 | 
			
		||||
        }
 | 
			
		||||
        $etagAppend = '';
 | 
			
		||||
        if (isset($spec['encoding'])) {
 | 
			
		||||
            $this->_stripEtag = true;
 | 
			
		||||
            $this->_headers['Vary'] = 'Accept-Encoding';
 | 
			
		||||
            if ('' !== $spec['encoding']) {
 | 
			
		||||
                if (0 === strpos($spec['encoding'], 'x-')) {
 | 
			
		||||
                    $spec['encoding'] = substr($spec['encoding'], 2);
 | 
			
		||||
                }
 | 
			
		||||
                $etagAppend = ';' . substr($spec['encoding'], 0, 2);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        if (isset($spec['lastModifiedTime'])) {
 | 
			
		||||
            $this->_setLastModified($spec['lastModifiedTime']);
 | 
			
		||||
            if (isset($spec['eTag'])) { // Use it
 | 
			
		||||
                $this->_setEtag($spec['eTag'], $scope);
 | 
			
		||||
            } else { // base both headers on time
 | 
			
		||||
                $this->_setEtag($spec['lastModifiedTime'] . $etagAppend, $scope);
 | 
			
		||||
            }
 | 
			
		||||
        } elseif (isset($spec['eTag'])) { // Use it
 | 
			
		||||
            $this->_setEtag($spec['eTag'], $scope);
 | 
			
		||||
        } elseif (isset($spec['contentHash'])) { // Use the hash as the ETag
 | 
			
		||||
            $this->_setEtag($spec['contentHash'] . $etagAppend, $scope);
 | 
			
		||||
        }
 | 
			
		||||
        $this->_headers['Cache-Control'] = "max-age={$maxAge}, {$scope}";
 | 
			
		||||
        // invalidate cache if disabled, otherwise check
 | 
			
		||||
        $this->cacheIsValid = (isset($spec['invalidate']) && $spec['invalidate'])
 | 
			
		||||
            ? false
 | 
			
		||||
            : $this->_isCacheValid();
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Get array of output headers to be sent
 | 
			
		||||
     * 
 | 
			
		||||
     * In the case of 304 responses, this array will only contain the response
 | 
			
		||||
     * code header: array('_responseCode' => 'HTTP/1.0 304 Not Modified')
 | 
			
		||||
     * 
 | 
			
		||||
     * Otherwise something like: 
 | 
			
		||||
     * <code>
 | 
			
		||||
     * array(
 | 
			
		||||
     *     'Cache-Control' => 'max-age=0, public'
 | 
			
		||||
     *     ,'ETag' => '"foobar"'
 | 
			
		||||
     * )
 | 
			
		||||
     * </code>
 | 
			
		||||
     *
 | 
			
		||||
     * @return array 
 | 
			
		||||
     */
 | 
			
		||||
    public function getHeaders()
 | 
			
		||||
    {
 | 
			
		||||
        return $this->_headers;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Set the Content-Length header in bytes
 | 
			
		||||
     * 
 | 
			
		||||
     * With most PHP configs, as long as you don't flush() output, this method
 | 
			
		||||
     * is not needed and PHP will buffer all output and set Content-Length for 
 | 
			
		||||
     * you. Otherwise you'll want to call this to let the client know up front.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param int $bytes
 | 
			
		||||
     * 
 | 
			
		||||
     * @return int copy of input $bytes
 | 
			
		||||
     */
 | 
			
		||||
    public function setContentLength($bytes)
 | 
			
		||||
    {
 | 
			
		||||
        return $this->_headers['Content-Length'] = $bytes;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Send headers
 | 
			
		||||
     * 
 | 
			
		||||
     * @see getHeaders()
 | 
			
		||||
     * 
 | 
			
		||||
     * Note this doesn't "clear" the headers. Calling sendHeaders() will
 | 
			
		||||
     * call header() again (but probably have not effect) and getHeaders() will
 | 
			
		||||
     * still return the headers.
 | 
			
		||||
     *
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    public function sendHeaders()
 | 
			
		||||
    {
 | 
			
		||||
        $headers = $this->_headers;
 | 
			
		||||
        if (array_key_exists('_responseCode', $headers)) {
 | 
			
		||||
            header($headers['_responseCode']);
 | 
			
		||||
            unset($headers['_responseCode']);
 | 
			
		||||
        }
 | 
			
		||||
        foreach ($headers as $name => $val) {
 | 
			
		||||
            header($name . ': ' . $val);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Exit if the client's cache is valid for this resource
 | 
			
		||||
     *
 | 
			
		||||
     * This is a convenience method for common use of the class
 | 
			
		||||
     *
 | 
			
		||||
     * @param int $lastModifiedTime if given, both ETag AND Last-Modified headers
 | 
			
		||||
     * will be sent with content. This is recommended.
 | 
			
		||||
     *
 | 
			
		||||
     * @param bool $isPublic (default false) if true, the Cache-Control header 
 | 
			
		||||
     * will contain "public", allowing proxies to cache the content. Otherwise 
 | 
			
		||||
     * "private" will be sent, allowing only browser caching.
 | 
			
		||||
     *
 | 
			
		||||
     * @param array $options (default empty) additional options for constructor
 | 
			
		||||
     *
 | 
			
		||||
     * @return null     
 | 
			
		||||
     */
 | 
			
		||||
    public static function check($lastModifiedTime = null, $isPublic = false, $options = array())
 | 
			
		||||
    {
 | 
			
		||||
        if (null !== $lastModifiedTime) {
 | 
			
		||||
            $options['lastModifiedTime'] = (int)$lastModifiedTime;
 | 
			
		||||
        }
 | 
			
		||||
        $options['isPublic'] = (bool)$isPublic;
 | 
			
		||||
        $cg = new HTTP_ConditionalGet($options);
 | 
			
		||||
        $cg->sendHeaders();
 | 
			
		||||
        if ($cg->cacheIsValid) {
 | 
			
		||||
            exit();
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Get a GMT formatted date for use in HTTP headers
 | 
			
		||||
     * 
 | 
			
		||||
     * <code>
 | 
			
		||||
     * header('Expires: ' . HTTP_ConditionalGet::gmtdate($time));
 | 
			
		||||
     * </code>  
 | 
			
		||||
     *
 | 
			
		||||
     * @param int $time unix timestamp
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public static function gmtDate($time)
 | 
			
		||||
    {
 | 
			
		||||
        return gmdate('D, d M Y H:i:s \G\M\T', $time);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    protected $_headers = array();
 | 
			
		||||
    protected $_lmTime = null;
 | 
			
		||||
    protected $_etag = null;
 | 
			
		||||
    protected $_stripEtag = false;
 | 
			
		||||
    
 | 
			
		||||
    protected function _setEtag($hash, $scope)
 | 
			
		||||
    {
 | 
			
		||||
        $this->_etag = '"' . substr($scope, 0, 3) . $hash . '"';
 | 
			
		||||
        $this->_headers['ETag'] = $this->_etag;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    protected function _setLastModified($time)
 | 
			
		||||
    {
 | 
			
		||||
        $this->_lmTime = (int)$time;
 | 
			
		||||
        $this->_headers['Last-Modified'] = self::gmtDate($time);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Determine validity of client cache and queue 304 header if valid
 | 
			
		||||
     */
 | 
			
		||||
    protected function _isCacheValid()
 | 
			
		||||
    {
 | 
			
		||||
        if (null === $this->_etag) {
 | 
			
		||||
            // lmTime is copied to ETag, so this condition implies that the
 | 
			
		||||
            // server sent neither ETag nor Last-Modified, so the client can't 
 | 
			
		||||
            // possibly has a valid cache.
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
        $isValid = ($this->resourceMatchedEtag() || $this->resourceNotModified());
 | 
			
		||||
        if ($isValid) {
 | 
			
		||||
            $this->_headers['_responseCode'] = 'HTTP/1.0 304 Not Modified';
 | 
			
		||||
        }
 | 
			
		||||
        return $isValid;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    protected function resourceMatchedEtag()
 | 
			
		||||
    {
 | 
			
		||||
        if (!isset($_SERVER['HTTP_IF_NONE_MATCH'])) {
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
        $clientEtagList = get_magic_quotes_gpc()
 | 
			
		||||
            ? stripslashes($_SERVER['HTTP_IF_NONE_MATCH'])
 | 
			
		||||
            : $_SERVER['HTTP_IF_NONE_MATCH'];
 | 
			
		||||
        $clientEtags = explode(',', $clientEtagList);
 | 
			
		||||
        
 | 
			
		||||
        $compareTo = $this->normalizeEtag($this->_etag);
 | 
			
		||||
        foreach ($clientEtags as $clientEtag) {
 | 
			
		||||
            if ($this->normalizeEtag($clientEtag) === $compareTo) {
 | 
			
		||||
                // respond with the client's matched ETag, even if it's not what
 | 
			
		||||
                // we would've sent by default
 | 
			
		||||
                $this->_headers['ETag'] = trim($clientEtag);
 | 
			
		||||
                return true;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        return false;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    protected function normalizeEtag($etag) {
 | 
			
		||||
        $etag = trim($etag);
 | 
			
		||||
        return $this->_stripEtag
 | 
			
		||||
            ? preg_replace('/;\\w\\w"$/', '"', $etag)
 | 
			
		||||
            : $etag;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    protected function resourceNotModified()
 | 
			
		||||
    {
 | 
			
		||||
        if (!isset($_SERVER['HTTP_IF_MODIFIED_SINCE'])) {
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
        $ifModifiedSince = $_SERVER['HTTP_IF_MODIFIED_SINCE'];
 | 
			
		||||
        if (false !== ($semicolon = strrpos($ifModifiedSince, ';'))) {
 | 
			
		||||
            // IE has tacked on extra data to this header, strip it
 | 
			
		||||
            $ifModifiedSince = substr($ifModifiedSince, 0, $semicolon);
 | 
			
		||||
        }
 | 
			
		||||
        if ($ifModifiedSince == self::gmtDate($this->_lmTime)) {
 | 
			
		||||
            // Apache 2.2's behavior. If there was no ETag match, send the 
 | 
			
		||||
            // non-encoded version of the ETag value.
 | 
			
		||||
            $this->_headers['ETag'] = $this->normalizeEtag($this->_etag);
 | 
			
		||||
            return true;
 | 
			
		||||
        }
 | 
			
		||||
        return false;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										326
									
								
								plugins/Minify/extlib/minify/min/lib/HTTP/Encoder.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										326
									
								
								plugins/Minify/extlib/minify/min/lib/HTTP/Encoder.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,326 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class HTTP_Encoder  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @subpackage HTTP
 | 
			
		||||
 */
 | 
			
		||||
 
 | 
			
		||||
/**
 | 
			
		||||
 * Encode and send gzipped/deflated content
 | 
			
		||||
 *
 | 
			
		||||
 * The "Vary: Accept-Encoding" header is sent. If the client allows encoding, 
 | 
			
		||||
 * Content-Encoding and Content-Length are added.
 | 
			
		||||
 *
 | 
			
		||||
 * <code>
 | 
			
		||||
 * // Send a CSS file, compressed if possible
 | 
			
		||||
 * $he = new HTTP_Encoder(array(
 | 
			
		||||
 *     'content' => file_get_contents($cssFile)
 | 
			
		||||
 *     ,'type' => 'text/css'
 | 
			
		||||
 * ));
 | 
			
		||||
 * $he->encode();
 | 
			
		||||
 * $he->sendAll();
 | 
			
		||||
 * </code>
 | 
			
		||||
 *
 | 
			
		||||
 * <code>
 | 
			
		||||
 * // Shortcut to encoding output
 | 
			
		||||
 * header('Content-Type: text/css'); // needed if not HTML
 | 
			
		||||
 * HTTP_Encoder::output($css);
 | 
			
		||||
 * </code>
 | 
			
		||||
 * 
 | 
			
		||||
 * <code>
 | 
			
		||||
 * // Just sniff for the accepted encoding
 | 
			
		||||
 * $encoding = HTTP_Encoder::getAcceptedEncoding();
 | 
			
		||||
 * </code>
 | 
			
		||||
 *
 | 
			
		||||
 * For more control over headers, use getHeaders() and getData() and send your
 | 
			
		||||
 * own output.
 | 
			
		||||
 * 
 | 
			
		||||
 * Note: If you don't need header mgmt, use PHP's native gzencode, gzdeflate, 
 | 
			
		||||
 * and gzcompress functions for gzip, deflate, and compress-encoding
 | 
			
		||||
 * respectively.
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @subpackage HTTP
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 */
 | 
			
		||||
class HTTP_Encoder {
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Should the encoder allow HTTP encoding to IE6? 
 | 
			
		||||
     * 
 | 
			
		||||
     * If you have many IE6 users and the bandwidth savings is worth troubling 
 | 
			
		||||
     * some of them, set this to true.
 | 
			
		||||
     * 
 | 
			
		||||
     * By default, encoding is only offered to IE7+. When this is true,
 | 
			
		||||
     * getAcceptedEncoding() will return an encoding for IE6 if its user agent
 | 
			
		||||
     * string contains "SV1". This has been documented in many places as "safe",
 | 
			
		||||
     * but there seem to be remaining, intermittent encoding bugs in patched 
 | 
			
		||||
     * IE6 on the wild web.
 | 
			
		||||
     * 
 | 
			
		||||
     * @var bool
 | 
			
		||||
     */
 | 
			
		||||
    public static $encodeToIe6 = false;
 | 
			
		||||
    
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Default compression level for zlib operations
 | 
			
		||||
     * 
 | 
			
		||||
     * This level is used if encode() is not given a $compressionLevel
 | 
			
		||||
     * 
 | 
			
		||||
     * @var int
 | 
			
		||||
     */
 | 
			
		||||
    public static $compressionLevel = 6;
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Get an HTTP Encoder object
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $spec options
 | 
			
		||||
     * 
 | 
			
		||||
     * 'content': (string required) content to be encoded
 | 
			
		||||
     * 
 | 
			
		||||
     * 'type': (string) if set, the Content-Type header will have this value.
 | 
			
		||||
     * 
 | 
			
		||||
     * 'method: (string) only set this if you are forcing a particular encoding
 | 
			
		||||
     * method. If not set, the best method will be chosen by getAcceptedEncoding()
 | 
			
		||||
     * The available methods are 'gzip', 'deflate', 'compress', and '' (no
 | 
			
		||||
     * encoding)
 | 
			
		||||
     * 
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    public function __construct($spec) 
 | 
			
		||||
    {
 | 
			
		||||
        $this->_content = $spec['content'];
 | 
			
		||||
        $this->_headers['Content-Length'] = (string)strlen($this->_content);
 | 
			
		||||
        if (isset($spec['type'])) {
 | 
			
		||||
            $this->_headers['Content-Type'] = $spec['type'];
 | 
			
		||||
        }
 | 
			
		||||
        if (isset($spec['method'])
 | 
			
		||||
            && in_array($spec['method'], array('gzip', 'deflate', 'compress', '')))
 | 
			
		||||
        {
 | 
			
		||||
            $this->_encodeMethod = array($spec['method'], $spec['method']);
 | 
			
		||||
        } else {
 | 
			
		||||
            $this->_encodeMethod = self::getAcceptedEncoding();
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Get content in current form
 | 
			
		||||
     * 
 | 
			
		||||
     * Call after encode() for encoded content.
 | 
			
		||||
     * 
 | 
			
		||||
     * return string
 | 
			
		||||
     */
 | 
			
		||||
    public function getContent() 
 | 
			
		||||
    {
 | 
			
		||||
        return $this->_content;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Get array of output headers to be sent
 | 
			
		||||
     * 
 | 
			
		||||
     * E.g.
 | 
			
		||||
     * <code>
 | 
			
		||||
     * array(
 | 
			
		||||
     *     'Content-Length' => '615'
 | 
			
		||||
     *     ,'Content-Encoding' => 'x-gzip'
 | 
			
		||||
     *     ,'Vary' => 'Accept-Encoding'
 | 
			
		||||
     * )
 | 
			
		||||
     * </code>
 | 
			
		||||
     *
 | 
			
		||||
     * @return array 
 | 
			
		||||
     */
 | 
			
		||||
    public function getHeaders()
 | 
			
		||||
    {
 | 
			
		||||
        return $this->_headers;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Send output headers
 | 
			
		||||
     * 
 | 
			
		||||
     * You must call this before headers are sent and it probably cannot be
 | 
			
		||||
     * used in conjunction with zlib output buffering / mod_gzip. Errors are
 | 
			
		||||
     * not handled purposefully.
 | 
			
		||||
     * 
 | 
			
		||||
     * @see getHeaders()
 | 
			
		||||
     * 
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    public function sendHeaders()
 | 
			
		||||
    {
 | 
			
		||||
        foreach ($this->_headers as $name => $val) {
 | 
			
		||||
            header($name . ': ' . $val);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Send output headers and content
 | 
			
		||||
     * 
 | 
			
		||||
     * A shortcut for sendHeaders() and echo getContent()
 | 
			
		||||
     *
 | 
			
		||||
     * You must call this before headers are sent and it probably cannot be
 | 
			
		||||
     * used in conjunction with zlib output buffering / mod_gzip. Errors are
 | 
			
		||||
     * not handled purposefully.
 | 
			
		||||
     * 
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    public function sendAll()
 | 
			
		||||
    {
 | 
			
		||||
        $this->sendHeaders();
 | 
			
		||||
        echo $this->_content;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Determine the client's best encoding method from the HTTP Accept-Encoding 
 | 
			
		||||
     * header.
 | 
			
		||||
     * 
 | 
			
		||||
     * If no Accept-Encoding header is set, or the browser is IE before v6 SP2,
 | 
			
		||||
     * this will return ('', ''), the "identity" encoding.
 | 
			
		||||
     * 
 | 
			
		||||
     * A syntax-aware scan is done of the Accept-Encoding, so the method must
 | 
			
		||||
     * be non 0. The methods are favored in order of gzip, deflate, then 
 | 
			
		||||
     * compress. Deflate is always smallest and generally faster, but is 
 | 
			
		||||
     * rarely sent by servers, so client support could be buggier.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param bool $allowCompress allow the older compress encoding
 | 
			
		||||
     * 
 | 
			
		||||
     * @param bool $allowDeflate allow the more recent deflate encoding
 | 
			
		||||
     * 
 | 
			
		||||
     * @return array two values, 1st is the actual encoding method, 2nd is the
 | 
			
		||||
     * alias of that method to use in the Content-Encoding header (some browsers
 | 
			
		||||
     * call gzip "x-gzip" etc.)
 | 
			
		||||
     */
 | 
			
		||||
    public static function getAcceptedEncoding($allowCompress = true, $allowDeflate = true)
 | 
			
		||||
    {
 | 
			
		||||
        // @link http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html
 | 
			
		||||
        
 | 
			
		||||
        if (! isset($_SERVER['HTTP_ACCEPT_ENCODING'])
 | 
			
		||||
            || self::_isBuggyIe())
 | 
			
		||||
        {
 | 
			
		||||
            return array('', '');
 | 
			
		||||
        }
 | 
			
		||||
        $ae = $_SERVER['HTTP_ACCEPT_ENCODING'];
 | 
			
		||||
        // gzip checks (quick)
 | 
			
		||||
        if (0 === strpos($ae, 'gzip,')             // most browsers
 | 
			
		||||
            || 0 === strpos($ae, 'deflate, gzip,') // opera
 | 
			
		||||
        ) {
 | 
			
		||||
            return array('gzip', 'gzip');
 | 
			
		||||
        }
 | 
			
		||||
        // gzip checks (slow)
 | 
			
		||||
        if (preg_match(
 | 
			
		||||
                '@(?:^|,)\\s*((?:x-)?gzip)\\s*(?:$|,|;\\s*q=(?:0\\.|1))@'
 | 
			
		||||
                ,$ae
 | 
			
		||||
                ,$m)) {
 | 
			
		||||
            return array('gzip', $m[1]);
 | 
			
		||||
        }
 | 
			
		||||
        if ($allowDeflate) {
 | 
			
		||||
            // deflate checks    
 | 
			
		||||
            $aeRev = strrev($ae);
 | 
			
		||||
            if (0 === strpos($aeRev, 'etalfed ,') // ie, webkit
 | 
			
		||||
                || 0 === strpos($aeRev, 'etalfed,') // gecko
 | 
			
		||||
                || 0 === strpos($ae, 'deflate,') // opera
 | 
			
		||||
                // slow parsing
 | 
			
		||||
                || preg_match(
 | 
			
		||||
                    '@(?:^|,)\\s*deflate\\s*(?:$|,|;\\s*q=(?:0\\.|1))@', $ae)) {
 | 
			
		||||
                return array('deflate', 'deflate');
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        if ($allowCompress && preg_match(
 | 
			
		||||
                '@(?:^|,)\\s*((?:x-)?compress)\\s*(?:$|,|;\\s*q=(?:0\\.|1))@'
 | 
			
		||||
                ,$ae
 | 
			
		||||
                ,$m)) {
 | 
			
		||||
            return array('compress', $m[1]);
 | 
			
		||||
        }
 | 
			
		||||
        return array('', '');
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Encode (compress) the content
 | 
			
		||||
     * 
 | 
			
		||||
     * If the encode method is '' (none) or compression level is 0, or the 'zlib'
 | 
			
		||||
     * extension isn't loaded, we return false.
 | 
			
		||||
     * 
 | 
			
		||||
     * Then the appropriate gz_* function is called to compress the content. If
 | 
			
		||||
     * this fails, false is returned.
 | 
			
		||||
     * 
 | 
			
		||||
     * The header "Vary: Accept-Encoding" is added. If encoding is successful, 
 | 
			
		||||
     * the Content-Length header is updated, and Content-Encoding is also added.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param int $compressionLevel given to zlib functions. If not given, the
 | 
			
		||||
     * class default will be used.
 | 
			
		||||
     * 
 | 
			
		||||
     * @return bool success true if the content was actually compressed
 | 
			
		||||
     */
 | 
			
		||||
    public function encode($compressionLevel = null)
 | 
			
		||||
    {
 | 
			
		||||
        $this->_headers['Vary'] = 'Accept-Encoding';
 | 
			
		||||
        if (null === $compressionLevel) {
 | 
			
		||||
            $compressionLevel = self::$compressionLevel;
 | 
			
		||||
        }
 | 
			
		||||
        if ('' === $this->_encodeMethod[0]
 | 
			
		||||
            || ($compressionLevel == 0)
 | 
			
		||||
            || !extension_loaded('zlib'))
 | 
			
		||||
        {
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
        if ($this->_encodeMethod[0] === 'deflate') {
 | 
			
		||||
            $encoded = gzdeflate($this->_content, $compressionLevel);
 | 
			
		||||
        } elseif ($this->_encodeMethod[0] === 'gzip') {
 | 
			
		||||
            $encoded = gzencode($this->_content, $compressionLevel);
 | 
			
		||||
        } else {
 | 
			
		||||
            $encoded = gzcompress($this->_content, $compressionLevel);
 | 
			
		||||
        }
 | 
			
		||||
        if (false === $encoded) {
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
        $this->_headers['Content-Length'] = strlen($encoded);
 | 
			
		||||
        $this->_headers['Content-Encoding'] = $this->_encodeMethod[1];
 | 
			
		||||
        $this->_content = $encoded;
 | 
			
		||||
        return true;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Encode and send appropriate headers and content
 | 
			
		||||
     *
 | 
			
		||||
     * This is a convenience method for common use of the class
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $content
 | 
			
		||||
     * 
 | 
			
		||||
     * @param int $compressionLevel given to zlib functions. If not given, the
 | 
			
		||||
     * class default will be used.
 | 
			
		||||
     * 
 | 
			
		||||
     * @return bool success true if the content was actually compressed
 | 
			
		||||
     */
 | 
			
		||||
    public static function output($content, $compressionLevel = null)
 | 
			
		||||
    {
 | 
			
		||||
        if (null === $compressionLevel) {
 | 
			
		||||
            $compressionLevel = self::$compressionLevel;
 | 
			
		||||
        }
 | 
			
		||||
        $he = new HTTP_Encoder(array('content' => $content));
 | 
			
		||||
        $ret = $he->encode($compressionLevel);
 | 
			
		||||
        $he->sendAll();
 | 
			
		||||
        return $ret;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    protected $_content = '';
 | 
			
		||||
    protected $_headers = array();
 | 
			
		||||
    protected $_encodeMethod = array('', '');
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Is the browser an IE version earlier than 6 SP2?  
 | 
			
		||||
     */
 | 
			
		||||
    protected static function _isBuggyIe()
 | 
			
		||||
    {
 | 
			
		||||
        $ua = $_SERVER['HTTP_USER_AGENT'];
 | 
			
		||||
        // quick escape for non-IEs
 | 
			
		||||
        if (0 !== strpos($ua, 'Mozilla/4.0 (compatible; MSIE ')
 | 
			
		||||
            || false !== strpos($ua, 'Opera')) {
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
        // no regex = faaast
 | 
			
		||||
        $version = (float)substr($ua, 30); 
 | 
			
		||||
        return self::$encodeToIe6
 | 
			
		||||
            ? ($version < 6 || ($version == 6 && false === strpos($ua, 'SV1')))
 | 
			
		||||
            : ($version < 7);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										314
									
								
								plugins/Minify/extlib/minify/min/lib/JSMin.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										314
									
								
								plugins/Minify/extlib/minify/min/lib/JSMin.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,314 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * jsmin.php - PHP implementation of Douglas Crockford's JSMin.
 | 
			
		||||
 *
 | 
			
		||||
 * This is a direct port of jsmin.c to PHP with a few PHP performance tweaks and
 | 
			
		||||
 * modifications to preserve some comments (see below). Also, rather than using
 | 
			
		||||
 * stdin/stdout, JSMin::minify() accepts a string as input and returns another
 | 
			
		||||
 * string as output.
 | 
			
		||||
 * 
 | 
			
		||||
 * Comments containing IE conditional compilation are preserved, as are multi-line
 | 
			
		||||
 * comments that begin with "/*!" (for documentation purposes). In the latter case
 | 
			
		||||
 * newlines are inserted around the comment to enhance readability.
 | 
			
		||||
 *
 | 
			
		||||
 * PHP 5 or higher is required.
 | 
			
		||||
 *
 | 
			
		||||
 * Permission is hereby granted to use this version of the library under the
 | 
			
		||||
 * same terms as jsmin.c, which has the following license:
 | 
			
		||||
 *
 | 
			
		||||
 * --
 | 
			
		||||
 * 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.
 | 
			
		||||
 * --
 | 
			
		||||
 *
 | 
			
		||||
 * @package JSMin
 | 
			
		||||
 * @author Ryan Grove <ryan@wonko.com> (PHP port)
 | 
			
		||||
 * @author Steve Clay <steve@mrclay.org> (modifications + cleanup)
 | 
			
		||||
 * @author Andrea Giammarchi <http://www.3site.eu> (spaceBeforeRegExp)
 | 
			
		||||
 * @copyright 2002 Douglas Crockford <douglas@crockford.com> (jsmin.c)
 | 
			
		||||
 * @copyright 2008 Ryan Grove <ryan@wonko.com> (PHP port)
 | 
			
		||||
 * @license http://opensource.org/licenses/mit-license.php MIT License
 | 
			
		||||
 * @link http://code.google.com/p/jsmin-php/
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
class JSMin {
 | 
			
		||||
    const ORD_LF            = 10;
 | 
			
		||||
    const ORD_SPACE         = 32;
 | 
			
		||||
    const ACTION_KEEP_A     = 1;
 | 
			
		||||
    const ACTION_DELETE_A   = 2;
 | 
			
		||||
    const ACTION_DELETE_A_B = 3;
 | 
			
		||||
    
 | 
			
		||||
    protected $a           = "\n";
 | 
			
		||||
    protected $b           = '';
 | 
			
		||||
    protected $input       = '';
 | 
			
		||||
    protected $inputIndex  = 0;
 | 
			
		||||
    protected $inputLength = 0;
 | 
			
		||||
    protected $lookAhead   = null;
 | 
			
		||||
    protected $output      = '';
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Minify Javascript
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $js Javascript to be minified
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public static function minify($js)
 | 
			
		||||
    {
 | 
			
		||||
        $jsmin = new JSMin($js);
 | 
			
		||||
        return $jsmin->min();
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Setup process
 | 
			
		||||
     */
 | 
			
		||||
    public function __construct($input)
 | 
			
		||||
    {
 | 
			
		||||
        $this->input       = str_replace("\r\n", "\n", $input);
 | 
			
		||||
        $this->inputLength = strlen($this->input);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Perform minification, return result
 | 
			
		||||
     */
 | 
			
		||||
    public function min()
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->output !== '') { // min already run
 | 
			
		||||
            return $this->output;
 | 
			
		||||
        }
 | 
			
		||||
        $this->action(self::ACTION_DELETE_A_B);
 | 
			
		||||
        
 | 
			
		||||
        while ($this->a !== null) {
 | 
			
		||||
            // determine next command
 | 
			
		||||
            $command = self::ACTION_KEEP_A; // default
 | 
			
		||||
            if ($this->a === ' ') {
 | 
			
		||||
                if (! $this->isAlphaNum($this->b)) {
 | 
			
		||||
                    $command = self::ACTION_DELETE_A;
 | 
			
		||||
                }
 | 
			
		||||
            } elseif ($this->a === "\n") {
 | 
			
		||||
                if ($this->b === ' ') {
 | 
			
		||||
                    $command = self::ACTION_DELETE_A_B;
 | 
			
		||||
                } elseif (false === strpos('{[(+-', $this->b) 
 | 
			
		||||
                          && ! $this->isAlphaNum($this->b)) {
 | 
			
		||||
                    $command = self::ACTION_DELETE_A;
 | 
			
		||||
                }
 | 
			
		||||
            } elseif (! $this->isAlphaNum($this->a)) {
 | 
			
		||||
                if ($this->b === ' '
 | 
			
		||||
                    || ($this->b === "\n" 
 | 
			
		||||
                        && (false === strpos('}])+-"\'', $this->a)))) {
 | 
			
		||||
                    $command = self::ACTION_DELETE_A_B;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            $this->action($command);
 | 
			
		||||
        }
 | 
			
		||||
        $this->output = trim($this->output);
 | 
			
		||||
        return $this->output;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * ACTION_KEEP_A = Output A. Copy B to A. Get the next B.
 | 
			
		||||
     * ACTION_DELETE_A = Copy B to A. Get the next B.
 | 
			
		||||
     * ACTION_DELETE_A_B = Get the next B.
 | 
			
		||||
     */
 | 
			
		||||
    protected function action($command)
 | 
			
		||||
    {
 | 
			
		||||
        switch ($command) {
 | 
			
		||||
            case self::ACTION_KEEP_A:
 | 
			
		||||
                $this->output .= $this->a;
 | 
			
		||||
                // fallthrough
 | 
			
		||||
            case self::ACTION_DELETE_A:
 | 
			
		||||
                $this->a = $this->b;
 | 
			
		||||
                if ($this->a === "'" || $this->a === '"') { // string literal
 | 
			
		||||
                    $str = $this->a; // in case needed for exception
 | 
			
		||||
                    while (true) {
 | 
			
		||||
                        $this->output .= $this->a;
 | 
			
		||||
                        $this->a       = $this->get();
 | 
			
		||||
                        if ($this->a === $this->b) { // end quote
 | 
			
		||||
                            break;
 | 
			
		||||
                        }
 | 
			
		||||
                        if (ord($this->a) <= self::ORD_LF) {
 | 
			
		||||
                            throw new JSMin_UnterminatedStringException(
 | 
			
		||||
                                'Unterminated String: ' . var_export($str, true));
 | 
			
		||||
                        }
 | 
			
		||||
                        $str .= $this->a;
 | 
			
		||||
                        if ($this->a === '\\') {
 | 
			
		||||
                            $this->output .= $this->a;
 | 
			
		||||
                            $this->a       = $this->get();
 | 
			
		||||
                            $str .= $this->a;
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
                // fallthrough
 | 
			
		||||
            case self::ACTION_DELETE_A_B:
 | 
			
		||||
                $this->b = $this->next();
 | 
			
		||||
                if ($this->b === '/' && $this->isRegexpLiteral()) { // RegExp literal
 | 
			
		||||
                    $this->output .= $this->a . $this->b;
 | 
			
		||||
                    $pattern = '/'; // in case needed for exception
 | 
			
		||||
                    while (true) {
 | 
			
		||||
                        $this->a = $this->get();
 | 
			
		||||
                        $pattern .= $this->a;
 | 
			
		||||
                        if ($this->a === '/') { // end pattern
 | 
			
		||||
                            break; // while (true)
 | 
			
		||||
                        } elseif ($this->a === '\\') {
 | 
			
		||||
                            $this->output .= $this->a;
 | 
			
		||||
                            $this->a       = $this->get();
 | 
			
		||||
                            $pattern      .= $this->a;
 | 
			
		||||
                        } elseif (ord($this->a) <= self::ORD_LF) {
 | 
			
		||||
                            throw new JSMin_UnterminatedRegExpException(
 | 
			
		||||
                                'Unterminated RegExp: '. var_export($pattern, true));
 | 
			
		||||
                        }
 | 
			
		||||
                        $this->output .= $this->a;
 | 
			
		||||
                    }
 | 
			
		||||
                    $this->b = $this->next();
 | 
			
		||||
                }
 | 
			
		||||
            // end case ACTION_DELETE_A_B
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    protected function isRegexpLiteral()
 | 
			
		||||
    {
 | 
			
		||||
        if (false !== strpos("\n{;(,=:[!&|?", $this->a)) { // we aren't dividing
 | 
			
		||||
            return true;
 | 
			
		||||
        }
 | 
			
		||||
        if (' ' === $this->a) {
 | 
			
		||||
            $length = strlen($this->output);
 | 
			
		||||
            if ($length < 2) { // weird edge case
 | 
			
		||||
                return true;
 | 
			
		||||
            }
 | 
			
		||||
            // you can't divide a keyword
 | 
			
		||||
            if (preg_match('/(?:case|else|in|return|typeof)$/', $this->output, $m)) {
 | 
			
		||||
                if ($this->output === $m[0]) { // odd but could happen
 | 
			
		||||
                    return true;
 | 
			
		||||
                }
 | 
			
		||||
                // make sure it's a keyword, not end of an identifier
 | 
			
		||||
                $charBeforeKeyword = substr($this->output, $length - strlen($m[0]) - 1, 1);
 | 
			
		||||
                if (! $this->isAlphaNum($charBeforeKeyword)) {
 | 
			
		||||
                    return true;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        return false;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Get next char. Convert ctrl char to space.
 | 
			
		||||
     */
 | 
			
		||||
    protected function get()
 | 
			
		||||
    {
 | 
			
		||||
        $c = $this->lookAhead;
 | 
			
		||||
        $this->lookAhead = null;
 | 
			
		||||
        if ($c === null) {
 | 
			
		||||
            if ($this->inputIndex < $this->inputLength) {
 | 
			
		||||
                $c = $this->input[$this->inputIndex];
 | 
			
		||||
                $this->inputIndex += 1;
 | 
			
		||||
            } else {
 | 
			
		||||
                return null;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        if ($c === "\r" || $c === "\n") {
 | 
			
		||||
            return "\n";
 | 
			
		||||
        }
 | 
			
		||||
        if (ord($c) < self::ORD_SPACE) { // control char
 | 
			
		||||
            return ' ';
 | 
			
		||||
        }
 | 
			
		||||
        return $c;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Get next char. If is ctrl character, translate to a space or newline.
 | 
			
		||||
     */
 | 
			
		||||
    protected function peek()
 | 
			
		||||
    {
 | 
			
		||||
        $this->lookAhead = $this->get();
 | 
			
		||||
        return $this->lookAhead;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Is $c a letter, digit, underscore, dollar sign, escape, or non-ASCII?
 | 
			
		||||
     */
 | 
			
		||||
    protected function isAlphaNum($c)
 | 
			
		||||
    {
 | 
			
		||||
        return (preg_match('/^[0-9a-zA-Z_\\$\\\\]$/', $c) || ord($c) > 126);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    protected function singleLineComment()
 | 
			
		||||
    {
 | 
			
		||||
        $comment = '';
 | 
			
		||||
        while (true) {
 | 
			
		||||
            $get = $this->get();
 | 
			
		||||
            $comment .= $get;
 | 
			
		||||
            if (ord($get) <= self::ORD_LF) { // EOL reached
 | 
			
		||||
                // if IE conditional comment
 | 
			
		||||
                if (preg_match('/^\\/@(?:cc_on|if|elif|else|end)\\b/', $comment)) {
 | 
			
		||||
                    return "/{$comment}";
 | 
			
		||||
                }
 | 
			
		||||
                return $get;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    protected function multipleLineComment()
 | 
			
		||||
    {
 | 
			
		||||
        $this->get();
 | 
			
		||||
        $comment = '';
 | 
			
		||||
        while (true) {
 | 
			
		||||
            $get = $this->get();
 | 
			
		||||
            if ($get === '*') {
 | 
			
		||||
                if ($this->peek() === '/') { // end of comment reached
 | 
			
		||||
                    $this->get();
 | 
			
		||||
                    // if comment preserved by YUI Compressor
 | 
			
		||||
                    if (0 === strpos($comment, '!')) {
 | 
			
		||||
                        return "\n/*" . substr($comment, 1) . "*/\n";
 | 
			
		||||
                    }
 | 
			
		||||
                    // if IE conditional comment
 | 
			
		||||
                    if (preg_match('/^@(?:cc_on|if|elif|else|end)\\b/', $comment)) {
 | 
			
		||||
                        return "/*{$comment}*/";
 | 
			
		||||
                    }
 | 
			
		||||
                    return ' ';
 | 
			
		||||
                }
 | 
			
		||||
            } elseif ($get === null) {
 | 
			
		||||
                throw new JSMin_UnterminatedCommentException('Unterminated Comment: ' . var_export('/*' . $comment, true));
 | 
			
		||||
            }
 | 
			
		||||
            $comment .= $get;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Get the next character, skipping over comments.
 | 
			
		||||
     * Some comments may be preserved.
 | 
			
		||||
     */
 | 
			
		||||
    protected function next()
 | 
			
		||||
    {
 | 
			
		||||
        $get = $this->get();
 | 
			
		||||
        if ($get !== '/') {
 | 
			
		||||
            return $get;
 | 
			
		||||
        }
 | 
			
		||||
        switch ($this->peek()) {
 | 
			
		||||
            case '/': return $this->singleLineComment();
 | 
			
		||||
            case '*': return $this->multipleLineComment();
 | 
			
		||||
            default: return $get;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
class JSMin_UnterminatedStringException extends Exception {}
 | 
			
		||||
class JSMin_UnterminatedCommentException extends Exception {}
 | 
			
		||||
class JSMin_UnterminatedRegExpException extends Exception {}
 | 
			
		||||
							
								
								
									
										1872
									
								
								plugins/Minify/extlib/minify/min/lib/JSMinPlus.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1872
									
								
								plugins/Minify/extlib/minify/min/lib/JSMinPlus.php
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										532
									
								
								plugins/Minify/extlib/minify/min/lib/Minify.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										532
									
								
								plugins/Minify/extlib/minify/min/lib/Minify.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,532 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Minify_Source
 | 
			
		||||
 */
 | 
			
		||||
require_once 'Minify/Source.php';
 | 
			
		||||
 
 | 
			
		||||
/**
 | 
			
		||||
 * Minify - Combines, minifies, and caches JavaScript and CSS files on demand.
 | 
			
		||||
 *
 | 
			
		||||
 * See README for usage instructions (for now).
 | 
			
		||||
 *
 | 
			
		||||
 * This library was inspired by {@link mailto:flashkot@mail.ru jscsscomp by Maxim Martynyuk}
 | 
			
		||||
 * and by the article {@link http://www.hunlock.com/blogs/Supercharged_Javascript "Supercharged JavaScript" by Patrick Hunlock}.
 | 
			
		||||
 *
 | 
			
		||||
 * Requires PHP 5.1.0.
 | 
			
		||||
 * Tested on PHP 5.1.6.
 | 
			
		||||
 *
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Ryan Grove <ryan@wonko.com>
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 * @copyright 2008 Ryan Grove, Stephen Clay. All rights reserved.
 | 
			
		||||
 * @license http://opensource.org/licenses/bsd-license.php  New BSD License
 | 
			
		||||
 * @link http://code.google.com/p/minify/
 | 
			
		||||
 */
 | 
			
		||||
class Minify {
 | 
			
		||||
    
 | 
			
		||||
    const VERSION = '2.1.3';
 | 
			
		||||
    const TYPE_CSS = 'text/css';
 | 
			
		||||
    const TYPE_HTML = 'text/html';
 | 
			
		||||
    // there is some debate over the ideal JS Content-Type, but this is the
 | 
			
		||||
    // Apache default and what Yahoo! uses..
 | 
			
		||||
    const TYPE_JS = 'application/x-javascript';
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * How many hours behind are the file modification times of uploaded files?
 | 
			
		||||
     * 
 | 
			
		||||
     * If you upload files from Windows to a non-Windows server, Windows may report
 | 
			
		||||
     * incorrect mtimes for the files. Immediately after modifying and uploading a 
 | 
			
		||||
     * file, use the touch command to update the mtime on the server. If the mtime 
 | 
			
		||||
     * jumps ahead by a number of hours, set this variable to that number. If the mtime 
 | 
			
		||||
     * moves back, this should not be needed.
 | 
			
		||||
     *
 | 
			
		||||
     * @var int $uploaderHoursBehind
 | 
			
		||||
     */
 | 
			
		||||
    public static $uploaderHoursBehind = 0;
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * If this string is not empty AND the serve() option 'bubbleCssImports' is
 | 
			
		||||
     * NOT set, then serve() will check CSS files for @import declarations that
 | 
			
		||||
     * appear too late in the combined stylesheet. If found, serve() will prepend
 | 
			
		||||
     * the output with this warning.
 | 
			
		||||
     *
 | 
			
		||||
     * @var string $importWarning
 | 
			
		||||
     */
 | 
			
		||||
    public static $importWarning = "/* See http://code.google.com/p/minify/wiki/CommonProblems#@imports_can_appear_in_invalid_locations_in_combined_CSS_files */\n";
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Specify a cache object (with identical interface as Minify_Cache_File) or
 | 
			
		||||
     * a path to use with Minify_Cache_File.
 | 
			
		||||
     * 
 | 
			
		||||
     * If not called, Minify will not use a cache and, for each 200 response, will 
 | 
			
		||||
     * need to recombine files, minify and encode the output.
 | 
			
		||||
     *
 | 
			
		||||
     * @param mixed $cache object with identical interface as Minify_Cache_File or
 | 
			
		||||
     * a directory path, or null to disable caching. (default = '')
 | 
			
		||||
     * 
 | 
			
		||||
     * @param bool $fileLocking (default = true) This only applies if the first
 | 
			
		||||
     * parameter is a string.
 | 
			
		||||
     *
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    public static function setCache($cache = '', $fileLocking = true)
 | 
			
		||||
    {
 | 
			
		||||
        if (is_string($cache)) {
 | 
			
		||||
            require_once 'Minify/Cache/File.php';
 | 
			
		||||
            self::$_cache = new Minify_Cache_File($cache, $fileLocking);
 | 
			
		||||
        } else {
 | 
			
		||||
            self::$_cache = $cache;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Serve a request for a minified file. 
 | 
			
		||||
     * 
 | 
			
		||||
     * Here are the available options and defaults in the base controller:
 | 
			
		||||
     * 
 | 
			
		||||
     * 'isPublic' : send "public" instead of "private" in Cache-Control 
 | 
			
		||||
     * headers, allowing shared caches to cache the output. (default true)
 | 
			
		||||
     * 
 | 
			
		||||
     * 'quiet' : set to true to have serve() return an array rather than sending
 | 
			
		||||
     * any headers/output (default false)
 | 
			
		||||
     * 
 | 
			
		||||
     * 'encodeOutput' : set to false to disable content encoding, and not send
 | 
			
		||||
     * the Vary header (default true)
 | 
			
		||||
     * 
 | 
			
		||||
     * 'encodeMethod' : generally you should let this be determined by 
 | 
			
		||||
     * HTTP_Encoder (leave null), but you can force a particular encoding
 | 
			
		||||
     * to be returned, by setting this to 'gzip' or '' (no encoding)
 | 
			
		||||
     * 
 | 
			
		||||
     * 'encodeLevel' : level of encoding compression (0 to 9, default 9)
 | 
			
		||||
     * 
 | 
			
		||||
     * 'contentTypeCharset' : appended to the Content-Type header sent. Set to a falsey
 | 
			
		||||
     * value to remove. (default 'utf-8')  
 | 
			
		||||
     * 
 | 
			
		||||
     * 'maxAge' : set this to the number of seconds the client should use its cache
 | 
			
		||||
     * before revalidating with the server. This sets Cache-Control: max-age and the
 | 
			
		||||
     * Expires header. Unlike the old 'setExpires' setting, this setting will NOT
 | 
			
		||||
     * prevent conditional GETs. Note this has nothing to do with server-side caching.
 | 
			
		||||
     * 
 | 
			
		||||
     * 'rewriteCssUris' : If true, serve() will automatically set the 'currentDir'
 | 
			
		||||
     * minifier option to enable URI rewriting in CSS files (default true)
 | 
			
		||||
     * 
 | 
			
		||||
     * 'bubbleCssImports' : If true, all @import declarations in combined CSS
 | 
			
		||||
     * files will be move to the top. Note this may alter effective CSS values
 | 
			
		||||
     * due to a change in order. (default false)
 | 
			
		||||
     * 
 | 
			
		||||
     * 'debug' : set to true to minify all sources with the 'Lines' controller, which
 | 
			
		||||
     * eases the debugging of combined files. This also prevents 304 responses.
 | 
			
		||||
     * @see Minify_Lines::minify()
 | 
			
		||||
     * 
 | 
			
		||||
     * 'minifiers' : to override Minify's default choice of minifier function for 
 | 
			
		||||
     * a particular content-type, specify your callback under the key of the 
 | 
			
		||||
     * content-type:
 | 
			
		||||
     * <code>
 | 
			
		||||
     * // call customCssMinifier($css) for all CSS minification
 | 
			
		||||
     * $options['minifiers'][Minify::TYPE_CSS] = 'customCssMinifier';
 | 
			
		||||
     * 
 | 
			
		||||
     * // don't minify Javascript at all
 | 
			
		||||
     * $options['minifiers'][Minify::TYPE_JS] = '';
 | 
			
		||||
     * </code>
 | 
			
		||||
     * 
 | 
			
		||||
     * 'minifierOptions' : to send options to the minifier function, specify your options
 | 
			
		||||
     * under the key of the content-type. E.g. To send the CSS minifier an option: 
 | 
			
		||||
     * <code>
 | 
			
		||||
     * // give CSS minifier array('optionName' => 'optionValue') as 2nd argument 
 | 
			
		||||
     * $options['minifierOptions'][Minify::TYPE_CSS]['optionName'] = 'optionValue';
 | 
			
		||||
     * </code>
 | 
			
		||||
     * 
 | 
			
		||||
     * 'contentType' : (optional) this is only needed if your file extension is not 
 | 
			
		||||
     * js/css/html. The given content-type will be sent regardless of source file
 | 
			
		||||
     * extension, so this should not be used in a Groups config with other
 | 
			
		||||
     * Javascript/CSS files.
 | 
			
		||||
     * 
 | 
			
		||||
     * Any controller options are documented in that controller's setupSources() method.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param mixed instance of subclass of Minify_Controller_Base or string name of
 | 
			
		||||
     * controller. E.g. 'Files'
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $options controller/serve options
 | 
			
		||||
     * 
 | 
			
		||||
     * @return mixed null, or, if the 'quiet' option is set to true, an array
 | 
			
		||||
     * with keys "success" (bool), "statusCode" (int), "content" (string), and
 | 
			
		||||
     * "headers" (array).
 | 
			
		||||
     */
 | 
			
		||||
    public static function serve($controller, $options = array())
 | 
			
		||||
    {
 | 
			
		||||
        if (is_string($controller)) {
 | 
			
		||||
            // make $controller into object
 | 
			
		||||
            $class = 'Minify_Controller_' . $controller;
 | 
			
		||||
            if (! class_exists($class, false)) {
 | 
			
		||||
                require_once "Minify/Controller/" 
 | 
			
		||||
                    . str_replace('_', '/', $controller) . ".php";    
 | 
			
		||||
            }
 | 
			
		||||
            $controller = new $class();
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        // set up controller sources and mix remaining options with
 | 
			
		||||
        // controller defaults
 | 
			
		||||
        $options = $controller->setupSources($options);
 | 
			
		||||
        $options = $controller->analyzeSources($options);
 | 
			
		||||
        self::$_options = $controller->mixInDefaultOptions($options);
 | 
			
		||||
        
 | 
			
		||||
        // check request validity
 | 
			
		||||
        if (! $controller->sources) {
 | 
			
		||||
            // invalid request!
 | 
			
		||||
            if (! self::$_options['quiet']) {
 | 
			
		||||
                header(self::$_options['badRequestHeader']);
 | 
			
		||||
                echo self::$_options['badRequestHeader'];
 | 
			
		||||
                return;
 | 
			
		||||
            } else {
 | 
			
		||||
                list(,$statusCode) = explode(' ', self::$_options['badRequestHeader']);
 | 
			
		||||
                return array(
 | 
			
		||||
                    'success' => false
 | 
			
		||||
                    ,'statusCode' => (int)$statusCode
 | 
			
		||||
                    ,'content' => ''
 | 
			
		||||
                    ,'headers' => array()
 | 
			
		||||
                );
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        self::$_controller = $controller;
 | 
			
		||||
        
 | 
			
		||||
        if (self::$_options['debug']) {
 | 
			
		||||
            self::_setupDebug($controller->sources);
 | 
			
		||||
            self::$_options['maxAge'] = 0;
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        // determine encoding
 | 
			
		||||
        if (self::$_options['encodeOutput']) {
 | 
			
		||||
            if (self::$_options['encodeMethod'] !== null) {
 | 
			
		||||
                // controller specifically requested this
 | 
			
		||||
                $contentEncoding = self::$_options['encodeMethod'];
 | 
			
		||||
            } else {
 | 
			
		||||
                // sniff request header
 | 
			
		||||
                require_once 'HTTP/Encoder.php';
 | 
			
		||||
                // depending on what the client accepts, $contentEncoding may be 
 | 
			
		||||
                // 'x-gzip' while our internal encodeMethod is 'gzip'. Calling
 | 
			
		||||
                // getAcceptedEncoding(false, false) leaves out compress and deflate as options.
 | 
			
		||||
                list(self::$_options['encodeMethod'], $contentEncoding) = HTTP_Encoder::getAcceptedEncoding(false, false);
 | 
			
		||||
            }
 | 
			
		||||
        } else {
 | 
			
		||||
            self::$_options['encodeMethod'] = ''; // identity (no encoding)
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        // check client cache
 | 
			
		||||
        require_once 'HTTP/ConditionalGet.php';
 | 
			
		||||
        $cgOptions = array(
 | 
			
		||||
            'lastModifiedTime' => self::$_options['lastModifiedTime']
 | 
			
		||||
            ,'isPublic' => self::$_options['isPublic']
 | 
			
		||||
            ,'encoding' => self::$_options['encodeMethod']
 | 
			
		||||
        );
 | 
			
		||||
        if (self::$_options['maxAge'] > 0) {
 | 
			
		||||
            $cgOptions['maxAge'] = self::$_options['maxAge'];
 | 
			
		||||
        }
 | 
			
		||||
        $cg = new HTTP_ConditionalGet($cgOptions);
 | 
			
		||||
        if ($cg->cacheIsValid) {
 | 
			
		||||
            // client's cache is valid
 | 
			
		||||
            if (! self::$_options['quiet']) {
 | 
			
		||||
                $cg->sendHeaders();
 | 
			
		||||
                return;
 | 
			
		||||
            } else {
 | 
			
		||||
                return array(
 | 
			
		||||
                    'success' => true
 | 
			
		||||
                    ,'statusCode' => 304
 | 
			
		||||
                    ,'content' => ''
 | 
			
		||||
                    ,'headers' => $cg->getHeaders()
 | 
			
		||||
                );
 | 
			
		||||
            }
 | 
			
		||||
        } else {
 | 
			
		||||
            // client will need output
 | 
			
		||||
            $headers = $cg->getHeaders();
 | 
			
		||||
            unset($cg);
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        if (self::$_options['contentType'] === self::TYPE_CSS
 | 
			
		||||
            && self::$_options['rewriteCssUris']) {
 | 
			
		||||
            reset($controller->sources);
 | 
			
		||||
            while (list($key, $source) = each($controller->sources)) {
 | 
			
		||||
                if ($source->filepath 
 | 
			
		||||
                    && !isset($source->minifyOptions['currentDir'])
 | 
			
		||||
                    && !isset($source->minifyOptions['prependRelativePath'])
 | 
			
		||||
                ) {
 | 
			
		||||
                    $source->minifyOptions['currentDir'] = dirname($source->filepath);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        // check server cache
 | 
			
		||||
        if (null !== self::$_cache) {
 | 
			
		||||
            // using cache
 | 
			
		||||
            // the goal is to use only the cache methods to sniff the length and 
 | 
			
		||||
            // output the content, as they do not require ever loading the file into
 | 
			
		||||
            // memory.
 | 
			
		||||
            $cacheId = 'minify_' . self::_getCacheId();
 | 
			
		||||
            $fullCacheId = (self::$_options['encodeMethod'])
 | 
			
		||||
                ? $cacheId . '.gz'
 | 
			
		||||
                : $cacheId;
 | 
			
		||||
            // check cache for valid entry
 | 
			
		||||
            $cacheIsReady = self::$_cache->isValid($fullCacheId, self::$_options['lastModifiedTime']); 
 | 
			
		||||
            if ($cacheIsReady) {
 | 
			
		||||
                $cacheContentLength = self::$_cache->getSize($fullCacheId);    
 | 
			
		||||
            } else {
 | 
			
		||||
                // generate & cache content
 | 
			
		||||
                $content = self::_combineMinify();
 | 
			
		||||
                self::$_cache->store($cacheId, $content);
 | 
			
		||||
                if (function_exists('gzencode')) {
 | 
			
		||||
                    self::$_cache->store($cacheId . '.gz', gzencode($content, self::$_options['encodeLevel']));
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        } else {
 | 
			
		||||
            // no cache
 | 
			
		||||
            $cacheIsReady = false;
 | 
			
		||||
            $content = self::_combineMinify();
 | 
			
		||||
        }
 | 
			
		||||
        if (! $cacheIsReady && self::$_options['encodeMethod']) {
 | 
			
		||||
            // still need to encode
 | 
			
		||||
            $content = gzencode($content, self::$_options['encodeLevel']);
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        // add headers
 | 
			
		||||
        $headers['Content-Length'] = $cacheIsReady
 | 
			
		||||
            ? $cacheContentLength
 | 
			
		||||
            : strlen($content);
 | 
			
		||||
        $headers['Content-Type'] = self::$_options['contentTypeCharset']
 | 
			
		||||
            ? self::$_options['contentType'] . '; charset=' . self::$_options['contentTypeCharset']
 | 
			
		||||
            : self::$_options['contentType'];
 | 
			
		||||
        if (self::$_options['encodeMethod'] !== '') {
 | 
			
		||||
            $headers['Content-Encoding'] = $contentEncoding;
 | 
			
		||||
        }
 | 
			
		||||
        if (self::$_options['encodeOutput']) {
 | 
			
		||||
            $headers['Vary'] = 'Accept-Encoding';
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (! self::$_options['quiet']) {
 | 
			
		||||
            // output headers & content
 | 
			
		||||
            foreach ($headers as $name => $val) {
 | 
			
		||||
                header($name . ': ' . $val);
 | 
			
		||||
            }
 | 
			
		||||
            if ($cacheIsReady) {
 | 
			
		||||
                self::$_cache->display($fullCacheId);
 | 
			
		||||
            } else {
 | 
			
		||||
                echo $content;
 | 
			
		||||
            }
 | 
			
		||||
        } else {
 | 
			
		||||
            return array(
 | 
			
		||||
                'success' => true
 | 
			
		||||
                ,'statusCode' => 200
 | 
			
		||||
                ,'content' => $cacheIsReady
 | 
			
		||||
                    ? self::$_cache->fetch($fullCacheId)
 | 
			
		||||
                    : $content
 | 
			
		||||
                ,'headers' => $headers
 | 
			
		||||
            );
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Return combined minified content for a set of sources
 | 
			
		||||
     *
 | 
			
		||||
     * No internal caching will be used and the content will not be HTTP encoded.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $sources array of filepaths and/or Minify_Source objects
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $options (optional) array of options for serve. By default
 | 
			
		||||
     * these are already set: quiet = true, encodeMethod = '', lastModifiedTime = 0.
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public static function combine($sources, $options = array())
 | 
			
		||||
    {
 | 
			
		||||
        $cache = self::$_cache;
 | 
			
		||||
        self::$_cache = null;
 | 
			
		||||
        $options = array_merge(array(
 | 
			
		||||
            'files' => (array)$sources
 | 
			
		||||
            ,'quiet' => true
 | 
			
		||||
            ,'encodeMethod' => ''
 | 
			
		||||
            ,'lastModifiedTime' => 0
 | 
			
		||||
        ), $options);
 | 
			
		||||
        $out = self::serve('Files', $options);
 | 
			
		||||
        self::$_cache = $cache;
 | 
			
		||||
        return $out['content'];
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * On IIS, create $_SERVER['DOCUMENT_ROOT']
 | 
			
		||||
     * 
 | 
			
		||||
     * @param bool $unsetPathInfo (default false) if true, $_SERVER['PATH_INFO']
 | 
			
		||||
     * will be unset (it is inconsistent with Apache's setting)
 | 
			
		||||
     * 
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    public static function setDocRoot($unsetPathInfo = false)
 | 
			
		||||
    {
 | 
			
		||||
        if (isset($_SERVER['SERVER_SOFTWARE'])
 | 
			
		||||
            && 0 === strpos($_SERVER['SERVER_SOFTWARE'], 'Microsoft-IIS/')
 | 
			
		||||
        ) {
 | 
			
		||||
            $_SERVER['DOCUMENT_ROOT'] = rtrim(substr(
 | 
			
		||||
                $_SERVER['PATH_TRANSLATED']
 | 
			
		||||
                ,0
 | 
			
		||||
                ,strlen($_SERVER['PATH_TRANSLATED']) - strlen($_SERVER['SCRIPT_NAME'])
 | 
			
		||||
            ), '\\');
 | 
			
		||||
            if ($unsetPathInfo) {
 | 
			
		||||
                unset($_SERVER['PATH_INFO']);
 | 
			
		||||
            }
 | 
			
		||||
            require_once 'Minify/Logger.php';
 | 
			
		||||
            Minify_Logger::log("setDocRoot() set DOCUMENT_ROOT to \"{$_SERVER['DOCUMENT_ROOT']}\"");
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * @var mixed Minify_Cache_* object or null (i.e. no server cache is used)
 | 
			
		||||
     */
 | 
			
		||||
    private static $_cache = null;
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * @var Minify_Controller active controller for current request
 | 
			
		||||
     */
 | 
			
		||||
    protected static $_controller = null;
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * @var array options for current request
 | 
			
		||||
     */
 | 
			
		||||
    protected static $_options = null;
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Set up sources to use Minify_Lines
 | 
			
		||||
     *
 | 
			
		||||
     * @param array $sources Minify_Source instances
 | 
			
		||||
     *
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    protected static function _setupDebug($sources)
 | 
			
		||||
    {
 | 
			
		||||
        foreach ($sources as $source) {
 | 
			
		||||
            $source->minifier = array('Minify_Lines', 'minify');
 | 
			
		||||
            $id = $source->getId();
 | 
			
		||||
            $source->minifyOptions = array(
 | 
			
		||||
                'id' => (is_file($id) ? basename($id) : $id)
 | 
			
		||||
            );
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Combines sources and minifies the result.
 | 
			
		||||
     *
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    protected static function _combineMinify()
 | 
			
		||||
    {
 | 
			
		||||
        $type = self::$_options['contentType']; // ease readability
 | 
			
		||||
        
 | 
			
		||||
        // when combining scripts, make sure all statements separated and
 | 
			
		||||
        // trailing single line comment is terminated
 | 
			
		||||
        $implodeSeparator = ($type === self::TYPE_JS)
 | 
			
		||||
            ? "\n;"
 | 
			
		||||
            : '';
 | 
			
		||||
        // allow the user to pass a particular array of options to each
 | 
			
		||||
        // minifier (designated by type). source objects may still override
 | 
			
		||||
        // these
 | 
			
		||||
        $defaultOptions = isset(self::$_options['minifierOptions'][$type])
 | 
			
		||||
            ? self::$_options['minifierOptions'][$type]
 | 
			
		||||
            : array();
 | 
			
		||||
        // if minifier not set, default is no minification. source objects
 | 
			
		||||
        // may still override this
 | 
			
		||||
        $defaultMinifier = isset(self::$_options['minifiers'][$type])
 | 
			
		||||
            ? self::$_options['minifiers'][$type]
 | 
			
		||||
            : false;
 | 
			
		||||
       
 | 
			
		||||
        if (Minify_Source::haveNoMinifyPrefs(self::$_controller->sources)) {
 | 
			
		||||
            // all source have same options/minifier, better performance
 | 
			
		||||
            // to combine, then minify once
 | 
			
		||||
            foreach (self::$_controller->sources as $source) {
 | 
			
		||||
                $pieces[] = $source->getContent();
 | 
			
		||||
            }
 | 
			
		||||
            $content = implode($implodeSeparator, $pieces);
 | 
			
		||||
            if ($defaultMinifier) {
 | 
			
		||||
                self::$_controller->loadMinifier($defaultMinifier);
 | 
			
		||||
                $content = call_user_func($defaultMinifier, $content, $defaultOptions);    
 | 
			
		||||
            }
 | 
			
		||||
        } else {
 | 
			
		||||
            // minify each source with its own options and minifier, then combine
 | 
			
		||||
            foreach (self::$_controller->sources as $source) {
 | 
			
		||||
                // allow the source to override our minifier and options
 | 
			
		||||
                $minifier = (null !== $source->minifier)
 | 
			
		||||
                    ? $source->minifier
 | 
			
		||||
                    : $defaultMinifier;
 | 
			
		||||
                $options = (null !== $source->minifyOptions)
 | 
			
		||||
                    ? array_merge($defaultOptions, $source->minifyOptions)
 | 
			
		||||
                    : $defaultOptions;
 | 
			
		||||
                if ($minifier) {
 | 
			
		||||
                    self::$_controller->loadMinifier($minifier);
 | 
			
		||||
                    // get source content and minify it
 | 
			
		||||
                    $pieces[] = call_user_func($minifier, $source->getContent(), $options);     
 | 
			
		||||
                } else {
 | 
			
		||||
                    $pieces[] = $source->getContent();     
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            $content = implode($implodeSeparator, $pieces);
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        if ($type === self::TYPE_CSS && false !== strpos($content, '@import')) {
 | 
			
		||||
            $content = self::_handleCssImports($content);
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        // do any post-processing (esp. for editing build URIs)
 | 
			
		||||
        if (self::$_options['postprocessorRequire']) {
 | 
			
		||||
            require_once self::$_options['postprocessorRequire'];
 | 
			
		||||
        }
 | 
			
		||||
        if (self::$_options['postprocessor']) {
 | 
			
		||||
            $content = call_user_func(self::$_options['postprocessor'], $content, $type);
 | 
			
		||||
        }
 | 
			
		||||
        return $content;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Make a unique cache id for for this request.
 | 
			
		||||
     * 
 | 
			
		||||
     * Any settings that could affect output are taken into consideration  
 | 
			
		||||
     *
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    protected static function _getCacheId()
 | 
			
		||||
    {
 | 
			
		||||
        return md5(serialize(array(
 | 
			
		||||
            Minify_Source::getDigest(self::$_controller->sources)
 | 
			
		||||
            ,self::$_options['minifiers'] 
 | 
			
		||||
            ,self::$_options['minifierOptions']
 | 
			
		||||
            ,self::$_options['postprocessor']
 | 
			
		||||
            ,self::$_options['bubbleCssImports']
 | 
			
		||||
        )));
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Bubble CSS @imports to the top or prepend a warning if an
 | 
			
		||||
     * @import is detected not at the top.
 | 
			
		||||
     */
 | 
			
		||||
    protected static function _handleCssImports($css)
 | 
			
		||||
    {
 | 
			
		||||
        if (self::$_options['bubbleCssImports']) {
 | 
			
		||||
            // bubble CSS imports
 | 
			
		||||
            preg_match_all('/@import.*?;/', $css, $imports);
 | 
			
		||||
            $css = implode('', $imports[0]) . preg_replace('/@import.*?;/', '', $css);
 | 
			
		||||
        } else if ('' !== self::$importWarning) {
 | 
			
		||||
            // remove comments so we don't mistake { in a comment as a block
 | 
			
		||||
            $noCommentCss = preg_replace('@/\\*[\\s\\S]*?\\*/@', '', $css);
 | 
			
		||||
            $lastImportPos = strrpos($noCommentCss, '@import');
 | 
			
		||||
            $firstBlockPos = strpos($noCommentCss, '{');
 | 
			
		||||
            if (false !== $lastImportPos
 | 
			
		||||
                && false !== $firstBlockPos
 | 
			
		||||
                && $firstBlockPos < $lastImportPos
 | 
			
		||||
            ) {
 | 
			
		||||
                // { appears before @import : prepend warning
 | 
			
		||||
                $css = self::$importWarning . $css;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        return $css;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										103
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Build.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										103
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Build.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,103 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_Build  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
require_once 'Minify/Source.php';
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Maintain a single last modification time for a group of Minify sources to
 | 
			
		||||
 * allow use of far off Expires headers in Minify.
 | 
			
		||||
 * 
 | 
			
		||||
 * <code>
 | 
			
		||||
 * // in config file
 | 
			
		||||
 * $groupSources = array(
 | 
			
		||||
 *   'js' => array('file1.js', 'file2.js')
 | 
			
		||||
 *   ,'css' => array('file1.css', 'file2.css', 'file3.css')
 | 
			
		||||
 * )
 | 
			
		||||
 * 
 | 
			
		||||
 * // during HTML generation
 | 
			
		||||
 * $jsBuild = new Minify_Build($groupSources['js']);
 | 
			
		||||
 * $cssBuild = new Minify_Build($groupSources['css']);
 | 
			
		||||
 * 
 | 
			
		||||
 * $script = "<script type='text/javascript' src='"
 | 
			
		||||
 *     . $jsBuild->uri('/min.php/js') . "'></script>";
 | 
			
		||||
 * $link = "<link rel='stylesheet' type='text/css' href='"
 | 
			
		||||
 *     . $cssBuild->uri('/min.php/css') . "'>";
 | 
			
		||||
 * 
 | 
			
		||||
 * // in min.php
 | 
			
		||||
 * Minify::serve('Groups', array(
 | 
			
		||||
 *   'groups' => $groupSources
 | 
			
		||||
 *   ,'setExpires' => (time() + 86400 * 365)
 | 
			
		||||
 * ));
 | 
			
		||||
 * </code>
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 */
 | 
			
		||||
class Minify_Build {
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Last modification time of all files in the build
 | 
			
		||||
     * 
 | 
			
		||||
     * @var int 
 | 
			
		||||
     */
 | 
			
		||||
    public $lastModified = 0;
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * String to use as ampersand in uri(). Set this to '&' if
 | 
			
		||||
     * you are not HTML-escaping URIs.
 | 
			
		||||
     *
 | 
			
		||||
     * @var string
 | 
			
		||||
     */
 | 
			
		||||
    public static $ampersand = '&';
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Get a time-stamped URI
 | 
			
		||||
     * 
 | 
			
		||||
     * <code>
 | 
			
		||||
     * echo $b->uri('/site.js');
 | 
			
		||||
     * // outputs "/site.js?1678242"
 | 
			
		||||
     * 
 | 
			
		||||
     * echo $b->uri('/scriptaculous.js?load=effects');
 | 
			
		||||
     * // outputs "/scriptaculous.js?load=effects&1678242"
 | 
			
		||||
     * </code>
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $uri
 | 
			
		||||
     * @param boolean $forceAmpersand (default = false) Force the use of ampersand to 
 | 
			
		||||
     * append the timestamp to the URI.
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public function uri($uri, $forceAmpersand = false) {
 | 
			
		||||
        $sep = ($forceAmpersand || strpos($uri, '?') !== false)
 | 
			
		||||
            ? self::$ampersand
 | 
			
		||||
            : '?';
 | 
			
		||||
        return "{$uri}{$sep}{$this->lastModified}";
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
     * Create a build object
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $sources array of Minify_Source objects and/or file paths
 | 
			
		||||
     * 
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    public function __construct($sources) 
 | 
			
		||||
    {
 | 
			
		||||
        $max = 0;
 | 
			
		||||
        foreach ((array)$sources as $source) {
 | 
			
		||||
            if ($source instanceof Minify_Source) {
 | 
			
		||||
                $max = max($max, $source->lastModified);
 | 
			
		||||
            } elseif (is_string($source)) {
 | 
			
		||||
                if (0 === strpos($source, '//')) {
 | 
			
		||||
                    $source = $_SERVER['DOCUMENT_ROOT'] . substr($source, 1);
 | 
			
		||||
                }
 | 
			
		||||
                if (is_file($source)) {
 | 
			
		||||
                    $max = max($max, filemtime($source));
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        $this->lastModified = $max;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										83
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/CSS.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/CSS.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,83 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_CSS  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Minify CSS
 | 
			
		||||
 *
 | 
			
		||||
 * This class uses Minify_CSS_Compressor and Minify_CSS_UriRewriter to 
 | 
			
		||||
 * minify CSS and rewrite relative URIs.
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 * @author http://code.google.com/u/1stvamp/ (Issue 64 patch)
 | 
			
		||||
 */
 | 
			
		||||
class Minify_CSS {
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Minify a CSS string
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $css
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $options available options:
 | 
			
		||||
     * 
 | 
			
		||||
     * 'preserveComments': (default true) multi-line comments that begin
 | 
			
		||||
     * with "/*!" will be preserved with newlines before and after to
 | 
			
		||||
     * enhance readability.
 | 
			
		||||
     * 
 | 
			
		||||
     * 'prependRelativePath': (default null) if given, this string will be
 | 
			
		||||
     * prepended to all relative URIs in import/url declarations
 | 
			
		||||
     * 
 | 
			
		||||
     * 'currentDir': (default null) if given, this is assumed to be the
 | 
			
		||||
     * directory of the current CSS file. Using this, minify will rewrite
 | 
			
		||||
     * all relative URIs in import/url declarations to correctly point to
 | 
			
		||||
     * the desired files. For this to work, the files *must* exist and be
 | 
			
		||||
     * visible by the PHP process.
 | 
			
		||||
     *
 | 
			
		||||
     * 'symlinks': (default = array()) If the CSS file is stored in 
 | 
			
		||||
     * a symlink-ed directory, provide an array of link paths to
 | 
			
		||||
     * target paths, where the link paths are within the document root. Because 
 | 
			
		||||
     * paths need to be normalized for this to work, use "//" to substitute 
 | 
			
		||||
     * the doc root in the link paths (the array keys). E.g.:
 | 
			
		||||
     * <code>
 | 
			
		||||
     * array('//symlink' => '/real/target/path') // unix
 | 
			
		||||
     * array('//static' => 'D:\\staticStorage')  // Windows
 | 
			
		||||
     * </code>
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public static function minify($css, $options = array()) 
 | 
			
		||||
    {
 | 
			
		||||
        require_once 'Minify/CSS/Compressor.php';
 | 
			
		||||
        if (isset($options['preserveComments']) 
 | 
			
		||||
            && !$options['preserveComments']) {
 | 
			
		||||
            $css = Minify_CSS_Compressor::process($css, $options);
 | 
			
		||||
        } else {
 | 
			
		||||
            require_once 'Minify/CommentPreserver.php';
 | 
			
		||||
            $css = Minify_CommentPreserver::process(
 | 
			
		||||
                $css
 | 
			
		||||
                ,array('Minify_CSS_Compressor', 'process')
 | 
			
		||||
                ,array($options)
 | 
			
		||||
            );
 | 
			
		||||
        }
 | 
			
		||||
        if (! isset($options['currentDir']) && ! isset($options['prependRelativePath'])) {
 | 
			
		||||
            return $css;
 | 
			
		||||
        }
 | 
			
		||||
        require_once 'Minify/CSS/UriRewriter.php';
 | 
			
		||||
        if (isset($options['currentDir'])) {
 | 
			
		||||
            return Minify_CSS_UriRewriter::rewrite(
 | 
			
		||||
                $css
 | 
			
		||||
                ,$options['currentDir']
 | 
			
		||||
                ,isset($options['docRoot']) ? $options['docRoot'] : $_SERVER['DOCUMENT_ROOT']
 | 
			
		||||
                ,isset($options['symlinks']) ? $options['symlinks'] : array()
 | 
			
		||||
            );  
 | 
			
		||||
        } else {
 | 
			
		||||
            return Minify_CSS_UriRewriter::prepend(
 | 
			
		||||
                $css
 | 
			
		||||
                ,$options['prependRelativePath']
 | 
			
		||||
            );
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										250
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/CSS/Compressor.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										250
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/CSS/Compressor.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,250 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_CSS_Compressor 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Compress CSS
 | 
			
		||||
 *
 | 
			
		||||
 * This is a heavy regex-based removal of whitespace, unnecessary
 | 
			
		||||
 * comments and tokens, and some CSS value minimization, where practical.
 | 
			
		||||
 * Many steps have been taken to avoid breaking comment-based hacks, 
 | 
			
		||||
 * including the ie5/mac filter (and its inversion), but expect tricky
 | 
			
		||||
 * hacks involving comment tokens in 'content' value strings to break
 | 
			
		||||
 * minimization badly. A test suite is available.
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 * @author http://code.google.com/u/1stvamp/ (Issue 64 patch)
 | 
			
		||||
 */
 | 
			
		||||
class Minify_CSS_Compressor {
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Minify a CSS string
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $css
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $options (currently ignored)
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public static function process($css, $options = array())
 | 
			
		||||
    {
 | 
			
		||||
        $obj = new Minify_CSS_Compressor($options);
 | 
			
		||||
        return $obj->_process($css);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * @var array options
 | 
			
		||||
     */
 | 
			
		||||
    protected $_options = null;
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * @var bool Are we "in" a hack?
 | 
			
		||||
     * 
 | 
			
		||||
     * I.e. are some browsers targetted until the next comment?
 | 
			
		||||
     */
 | 
			
		||||
    protected $_inHack = false;
 | 
			
		||||
    
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Constructor
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $options (currently ignored)
 | 
			
		||||
     * 
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    private function __construct($options) {
 | 
			
		||||
        $this->_options = $options;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Minify a CSS string
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $css
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    protected function _process($css)
 | 
			
		||||
    {
 | 
			
		||||
        $css = str_replace("\r\n", "\n", $css);
 | 
			
		||||
        
 | 
			
		||||
        // preserve empty comment after '>'
 | 
			
		||||
        // http://www.webdevout.net/css-hacks#in_css-selectors
 | 
			
		||||
        $css = preg_replace('@>/\\*\\s*\\*/@', '>/*keep*/', $css);
 | 
			
		||||
        
 | 
			
		||||
        // preserve empty comment between property and value
 | 
			
		||||
        // http://css-discuss.incutio.com/?page=BoxModelHack
 | 
			
		||||
        $css = preg_replace('@/\\*\\s*\\*/\\s*:@', '/*keep*/:', $css);
 | 
			
		||||
        $css = preg_replace('@:\\s*/\\*\\s*\\*/@', ':/*keep*/', $css);
 | 
			
		||||
        
 | 
			
		||||
        // apply callback to all valid comments (and strip out surrounding ws
 | 
			
		||||
        $css = preg_replace_callback('@\\s*/\\*([\\s\\S]*?)\\*/\\s*@'
 | 
			
		||||
            ,array($this, '_commentCB'), $css);
 | 
			
		||||
 | 
			
		||||
        // remove ws around { } and last semicolon in declaration block
 | 
			
		||||
        $css = preg_replace('/\\s*{\\s*/', '{', $css);
 | 
			
		||||
        $css = preg_replace('/;?\\s*}\\s*/', '}', $css);
 | 
			
		||||
        
 | 
			
		||||
        // remove ws surrounding semicolons
 | 
			
		||||
        $css = preg_replace('/\\s*;\\s*/', ';', $css);
 | 
			
		||||
        
 | 
			
		||||
        // remove ws around urls
 | 
			
		||||
        $css = preg_replace('/
 | 
			
		||||
                url\\(      # url(
 | 
			
		||||
                \\s*
 | 
			
		||||
                ([^\\)]+?)  # 1 = the URL (really just a bunch of non right parenthesis)
 | 
			
		||||
                \\s*
 | 
			
		||||
                \\)         # )
 | 
			
		||||
            /x', 'url($1)', $css);
 | 
			
		||||
        
 | 
			
		||||
        // remove ws between rules and colons
 | 
			
		||||
        $css = preg_replace('/
 | 
			
		||||
                \\s*
 | 
			
		||||
                ([{;])              # 1 = beginning of block or rule separator 
 | 
			
		||||
                \\s*
 | 
			
		||||
                ([\\*_]?[\\w\\-]+)  # 2 = property (and maybe IE filter)
 | 
			
		||||
                \\s*
 | 
			
		||||
                :
 | 
			
		||||
                \\s*
 | 
			
		||||
                (\\b|[#\'"])        # 3 = first character of a value
 | 
			
		||||
            /x', '$1$2:$3', $css);
 | 
			
		||||
        
 | 
			
		||||
        // remove ws in selectors
 | 
			
		||||
        $css = preg_replace_callback('/
 | 
			
		||||
                (?:              # non-capture
 | 
			
		||||
                    \\s*
 | 
			
		||||
                    [^~>+,\\s]+  # selector part
 | 
			
		||||
                    \\s*
 | 
			
		||||
                    [,>+~]       # combinators
 | 
			
		||||
                )+
 | 
			
		||||
                \\s*
 | 
			
		||||
                [^~>+,\\s]+      # selector part
 | 
			
		||||
                {                # open declaration block
 | 
			
		||||
            /x'
 | 
			
		||||
            ,array($this, '_selectorsCB'), $css);
 | 
			
		||||
        
 | 
			
		||||
        // minimize hex colors
 | 
			
		||||
        $css = preg_replace('/([^=])#([a-f\\d])\\2([a-f\\d])\\3([a-f\\d])\\4([\\s;\\}])/i'
 | 
			
		||||
            , '$1#$2$3$4$5', $css);
 | 
			
		||||
        
 | 
			
		||||
        // remove spaces between font families
 | 
			
		||||
        $css = preg_replace_callback('/font-family:([^;}]+)([;}])/'
 | 
			
		||||
            ,array($this, '_fontFamilyCB'), $css);
 | 
			
		||||
        
 | 
			
		||||
        $css = preg_replace('/@import\\s+url/', '@import url', $css);
 | 
			
		||||
        
 | 
			
		||||
        // replace any ws involving newlines with a single newline
 | 
			
		||||
        $css = preg_replace('/[ \\t]*\\n+\\s*/', "\n", $css);
 | 
			
		||||
        
 | 
			
		||||
        // separate common descendent selectors w/ newlines (to limit line lengths)
 | 
			
		||||
        $css = preg_replace('/([\\w#\\.\\*]+)\\s+([\\w#\\.\\*]+){/', "$1\n$2{", $css);
 | 
			
		||||
        
 | 
			
		||||
        // Use newline after 1st numeric value (to limit line lengths).
 | 
			
		||||
        $css = preg_replace('/
 | 
			
		||||
            ((?:padding|margin|border|outline):\\d+(?:px|em)?) # 1 = prop : 1st numeric value
 | 
			
		||||
            \\s+
 | 
			
		||||
            /x'
 | 
			
		||||
            ,"$1\n", $css);
 | 
			
		||||
        
 | 
			
		||||
        // prevent triggering IE6 bug: http://www.crankygeek.com/ie6pebug/
 | 
			
		||||
        $css = preg_replace('/:first-l(etter|ine)\\{/', ':first-l$1 {', $css);
 | 
			
		||||
            
 | 
			
		||||
        return trim($css);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Replace what looks like a set of selectors  
 | 
			
		||||
     *
 | 
			
		||||
     * @param array $m regex matches
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    protected function _selectorsCB($m)
 | 
			
		||||
    {
 | 
			
		||||
        // remove ws around the combinators
 | 
			
		||||
        return preg_replace('/\\s*([,>+~])\\s*/', '$1', $m[0]);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Process a comment and return a replacement
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $m regex matches
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    protected function _commentCB($m)
 | 
			
		||||
    {
 | 
			
		||||
        $hasSurroundingWs = (trim($m[0]) !== $m[1]);
 | 
			
		||||
        $m = $m[1]; 
 | 
			
		||||
        // $m is the comment content w/o the surrounding tokens, 
 | 
			
		||||
        // but the return value will replace the entire comment.
 | 
			
		||||
        if ($m === 'keep') {
 | 
			
		||||
            return '/**/';
 | 
			
		||||
        }
 | 
			
		||||
        if ($m === '" "') {
 | 
			
		||||
            // component of http://tantek.com/CSS/Examples/midpass.html
 | 
			
		||||
            return '/*" "*/';
 | 
			
		||||
        }
 | 
			
		||||
        if (preg_match('@";\\}\\s*\\}/\\*\\s+@', $m)) {
 | 
			
		||||
            // component of http://tantek.com/CSS/Examples/midpass.html
 | 
			
		||||
            return '/*";}}/* */';
 | 
			
		||||
        }
 | 
			
		||||
        if ($this->_inHack) {
 | 
			
		||||
            // inversion: feeding only to one browser
 | 
			
		||||
            if (preg_match('@
 | 
			
		||||
                    ^/               # comment started like /*/
 | 
			
		||||
                    \\s*
 | 
			
		||||
                    (\\S[\\s\\S]+?)  # has at least some non-ws content
 | 
			
		||||
                    \\s*
 | 
			
		||||
                    /\\*             # ends like /*/ or /**/
 | 
			
		||||
                @x', $m, $n)) {
 | 
			
		||||
                // end hack mode after this comment, but preserve the hack and comment content
 | 
			
		||||
                $this->_inHack = false;
 | 
			
		||||
                return "/*/{$n[1]}/**/";
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        if (substr($m, -1) === '\\') { // comment ends like \*/
 | 
			
		||||
            // begin hack mode and preserve hack
 | 
			
		||||
            $this->_inHack = true;
 | 
			
		||||
            return '/*\\*/';
 | 
			
		||||
        }
 | 
			
		||||
        if ($m !== '' && $m[0] === '/') { // comment looks like /*/ foo */
 | 
			
		||||
            // begin hack mode and preserve hack
 | 
			
		||||
            $this->_inHack = true;
 | 
			
		||||
            return '/*/*/';
 | 
			
		||||
        }
 | 
			
		||||
        if ($this->_inHack) {
 | 
			
		||||
            // a regular comment ends hack mode but should be preserved
 | 
			
		||||
            $this->_inHack = false;
 | 
			
		||||
            return '/**/';
 | 
			
		||||
        }
 | 
			
		||||
        // Issue 107: if there's any surrounding whitespace, it may be important, so 
 | 
			
		||||
        // replace the comment with a single space
 | 
			
		||||
        return $hasSurroundingWs // remove all other comments
 | 
			
		||||
            ? ' '
 | 
			
		||||
            : '';
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Process a font-family listing and return a replacement
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $m regex matches
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string   
 | 
			
		||||
     */
 | 
			
		||||
    protected function _fontFamilyCB($m)
 | 
			
		||||
    {
 | 
			
		||||
        $m[1] = preg_replace('/
 | 
			
		||||
                \\s*
 | 
			
		||||
                (
 | 
			
		||||
                    "[^"]+"      # 1 = family in double qutoes
 | 
			
		||||
                    |\'[^\']+\'  # or 1 = family in single quotes
 | 
			
		||||
                    |[\\w\\-]+   # or 1 = unquoted family
 | 
			
		||||
                )
 | 
			
		||||
                \\s*
 | 
			
		||||
            /x', '$1', $m[1]);
 | 
			
		||||
        return 'font-family:' . $m[1] . $m[2];
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										270
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/CSS/UriRewriter.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										270
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/CSS/UriRewriter.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,270 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_CSS_UriRewriter  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Rewrite file-relative URIs as root-relative in CSS files
 | 
			
		||||
 *
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 */
 | 
			
		||||
class Minify_CSS_UriRewriter {
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Defines which class to call as part of callbacks, change this
 | 
			
		||||
     * if you extend Minify_CSS_UriRewriter
 | 
			
		||||
     * @var string
 | 
			
		||||
     */
 | 
			
		||||
    protected static $className = 'Minify_CSS_UriRewriter';
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * rewrite() and rewriteRelative() append debugging information here
 | 
			
		||||
     * @var string
 | 
			
		||||
     */
 | 
			
		||||
    public static $debugText = '';
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Rewrite file relative URIs as root relative in CSS files
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $css
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $currentDir The directory of the current CSS file.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $docRoot The document root of the web site in which 
 | 
			
		||||
     * the CSS file resides (default = $_SERVER['DOCUMENT_ROOT']).
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $symlinks (default = array()) If the CSS file is stored in 
 | 
			
		||||
     * a symlink-ed directory, provide an array of link paths to
 | 
			
		||||
     * target paths, where the link paths are within the document root. Because 
 | 
			
		||||
     * paths need to be normalized for this to work, use "//" to substitute 
 | 
			
		||||
     * the doc root in the link paths (the array keys). E.g.:
 | 
			
		||||
     * <code>
 | 
			
		||||
     * array('//symlink' => '/real/target/path') // unix
 | 
			
		||||
     * array('//static' => 'D:\\staticStorage')  // Windows
 | 
			
		||||
     * </code>
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public static function rewrite($css, $currentDir, $docRoot = null, $symlinks = array()) 
 | 
			
		||||
    {
 | 
			
		||||
        self::$_docRoot = self::_realpath(
 | 
			
		||||
            $docRoot ? $docRoot : $_SERVER['DOCUMENT_ROOT']
 | 
			
		||||
        );
 | 
			
		||||
        self::$_currentDir = self::_realpath($currentDir);
 | 
			
		||||
        self::$_symlinks = array();
 | 
			
		||||
        
 | 
			
		||||
        // normalize symlinks
 | 
			
		||||
        foreach ($symlinks as $link => $target) {
 | 
			
		||||
            $link = ($link === '//')
 | 
			
		||||
                ? self::$_docRoot
 | 
			
		||||
                : str_replace('//', self::$_docRoot . '/', $link);
 | 
			
		||||
            $link = strtr($link, '/', DIRECTORY_SEPARATOR);
 | 
			
		||||
            self::$_symlinks[$link] = self::_realpath($target);
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        self::$debugText .= "docRoot    : " . self::$_docRoot . "\n"
 | 
			
		||||
                          . "currentDir : " . self::$_currentDir . "\n";
 | 
			
		||||
        if (self::$_symlinks) {
 | 
			
		||||
            self::$debugText .= "symlinks : " . var_export(self::$_symlinks, 1) . "\n";
 | 
			
		||||
        }
 | 
			
		||||
        self::$debugText .= "\n";
 | 
			
		||||
        
 | 
			
		||||
        $css = self::_trimUrls($css);
 | 
			
		||||
        
 | 
			
		||||
        // rewrite
 | 
			
		||||
        $css = preg_replace_callback('/@import\\s+([\'"])(.*?)[\'"]/'
 | 
			
		||||
            ,array(self::$className, '_processUriCB'), $css);
 | 
			
		||||
        $css = preg_replace_callback('/url\\(\\s*([^\\)\\s]+)\\s*\\)/'
 | 
			
		||||
            ,array(self::$className, '_processUriCB'), $css);
 | 
			
		||||
 | 
			
		||||
        return $css;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Prepend a path to relative URIs in CSS files
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $css
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $path The path to prepend.
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public static function prepend($css, $path)
 | 
			
		||||
    {
 | 
			
		||||
        self::$_prependPath = $path;
 | 
			
		||||
        
 | 
			
		||||
        $css = self::_trimUrls($css);
 | 
			
		||||
        
 | 
			
		||||
        // append
 | 
			
		||||
        $css = preg_replace_callback('/@import\\s+([\'"])(.*?)[\'"]/'
 | 
			
		||||
            ,array(self::$className, '_processUriCB'), $css);
 | 
			
		||||
        $css = preg_replace_callback('/url\\(\\s*([^\\)\\s]+)\\s*\\)/'
 | 
			
		||||
            ,array(self::$className, '_processUriCB'), $css);
 | 
			
		||||
 | 
			
		||||
        self::$_prependPath = null;
 | 
			
		||||
        return $css;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * @var string directory of this stylesheet
 | 
			
		||||
     */
 | 
			
		||||
    private static $_currentDir = '';
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * @var string DOC_ROOT
 | 
			
		||||
     */
 | 
			
		||||
    private static $_docRoot = '';
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * @var array directory replacements to map symlink targets back to their
 | 
			
		||||
     * source (within the document root) E.g. '/var/www/symlink' => '/var/realpath'
 | 
			
		||||
     */
 | 
			
		||||
    private static $_symlinks = array();
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * @var string path to prepend
 | 
			
		||||
     */
 | 
			
		||||
    private static $_prependPath = null;
 | 
			
		||||
    
 | 
			
		||||
    private static function _trimUrls($css)
 | 
			
		||||
    {
 | 
			
		||||
        return preg_replace('/
 | 
			
		||||
            url\\(      # url(
 | 
			
		||||
            \\s*
 | 
			
		||||
            ([^\\)]+?)  # 1 = URI (assuming does not contain ")")
 | 
			
		||||
            \\s*
 | 
			
		||||
            \\)         # )
 | 
			
		||||
        /x', 'url($1)', $css);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    private static function _processUriCB($m)
 | 
			
		||||
    {
 | 
			
		||||
        // $m matched either '/@import\\s+([\'"])(.*?)[\'"]/' or '/url\\(\\s*([^\\)\\s]+)\\s*\\)/'
 | 
			
		||||
        $isImport = ($m[0][0] === '@');
 | 
			
		||||
        // determine URI and the quote character (if any)
 | 
			
		||||
        if ($isImport) {
 | 
			
		||||
            $quoteChar = $m[1];
 | 
			
		||||
            $uri = $m[2];
 | 
			
		||||
        } else {
 | 
			
		||||
            // $m[1] is either quoted or not
 | 
			
		||||
            $quoteChar = ($m[1][0] === "'" || $m[1][0] === '"')
 | 
			
		||||
                ? $m[1][0]
 | 
			
		||||
                : '';
 | 
			
		||||
            $uri = ($quoteChar === '')
 | 
			
		||||
                ? $m[1]
 | 
			
		||||
                : substr($m[1], 1, strlen($m[1]) - 2);
 | 
			
		||||
        }
 | 
			
		||||
        // analyze URI
 | 
			
		||||
        if ('/' !== $uri[0]                  // root-relative
 | 
			
		||||
            && false === strpos($uri, '//')  // protocol (non-data)
 | 
			
		||||
            && 0 !== strpos($uri, 'data:')   // data protocol
 | 
			
		||||
        ) {
 | 
			
		||||
            // URI is file-relative: rewrite depending on options
 | 
			
		||||
            $uri = (self::$_prependPath !== null)
 | 
			
		||||
                ? (self::$_prependPath . $uri)
 | 
			
		||||
                : self::rewriteRelative($uri, self::$_currentDir, self::$_docRoot, self::$_symlinks);
 | 
			
		||||
        }
 | 
			
		||||
        return $isImport
 | 
			
		||||
            ? "@import {$quoteChar}{$uri}{$quoteChar}"
 | 
			
		||||
            : "url({$quoteChar}{$uri}{$quoteChar})";
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Rewrite a file relative URI as root relative
 | 
			
		||||
     *
 | 
			
		||||
     * <code>
 | 
			
		||||
     * Minify_CSS_UriRewriter::rewriteRelative(
 | 
			
		||||
     *       '../img/hello.gif'
 | 
			
		||||
     *     , '/home/user/www/css'  // path of CSS file
 | 
			
		||||
     *     , '/home/user/www'      // doc root
 | 
			
		||||
     * );
 | 
			
		||||
     * // returns '/img/hello.gif'
 | 
			
		||||
     * 
 | 
			
		||||
     * // example where static files are stored in a symlinked directory
 | 
			
		||||
     * Minify_CSS_UriRewriter::rewriteRelative(
 | 
			
		||||
     *       'hello.gif'
 | 
			
		||||
     *     , '/var/staticFiles/theme'
 | 
			
		||||
     *     , '/home/user/www'
 | 
			
		||||
     *     , array('/home/user/www/static' => '/var/staticFiles')
 | 
			
		||||
     * );
 | 
			
		||||
     * // returns '/static/theme/hello.gif'
 | 
			
		||||
     * </code>
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $uri file relative URI
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $realCurrentDir realpath of the current file's directory.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $realDocRoot realpath of the site document root.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $symlinks (default = array()) If the file is stored in 
 | 
			
		||||
     * a symlink-ed directory, provide an array of link paths to
 | 
			
		||||
     * real target paths, where the link paths "appear" to be within the document 
 | 
			
		||||
     * root. E.g.:
 | 
			
		||||
     * <code>
 | 
			
		||||
     * array('/home/foo/www/not/real/path' => '/real/target/path') // unix
 | 
			
		||||
     * array('C:\\htdocs\\not\\real' => 'D:\\real\\target\\path')  // Windows
 | 
			
		||||
     * </code>
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public static function rewriteRelative($uri, $realCurrentDir, $realDocRoot, $symlinks = array())
 | 
			
		||||
    {
 | 
			
		||||
        // prepend path with current dir separator (OS-independent)
 | 
			
		||||
        $path = strtr($realCurrentDir, '/', DIRECTORY_SEPARATOR)  
 | 
			
		||||
            . DIRECTORY_SEPARATOR . strtr($uri, '/', DIRECTORY_SEPARATOR);
 | 
			
		||||
        
 | 
			
		||||
        self::$debugText .= "file-relative URI  : {$uri}\n"
 | 
			
		||||
                          . "path prepended     : {$path}\n";
 | 
			
		||||
        
 | 
			
		||||
        // "unresolve" a symlink back to doc root
 | 
			
		||||
        foreach ($symlinks as $link => $target) {
 | 
			
		||||
            if (0 === strpos($path, $target)) {
 | 
			
		||||
                // replace $target with $link
 | 
			
		||||
                $path = $link . substr($path, strlen($target));
 | 
			
		||||
                
 | 
			
		||||
                self::$debugText .= "symlink unresolved : {$path}\n";
 | 
			
		||||
                
 | 
			
		||||
                break;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        // strip doc root
 | 
			
		||||
        $path = substr($path, strlen($realDocRoot));
 | 
			
		||||
        
 | 
			
		||||
        self::$debugText .= "docroot stripped   : {$path}\n";
 | 
			
		||||
        
 | 
			
		||||
        // fix to root-relative URI
 | 
			
		||||
 | 
			
		||||
        $uri = strtr($path, '/\\', '//');
 | 
			
		||||
 | 
			
		||||
        // remove /./ and /../ where possible
 | 
			
		||||
        $uri = str_replace('/./', '/', $uri);
 | 
			
		||||
        // inspired by patch from Oleg Cherniy
 | 
			
		||||
        do {
 | 
			
		||||
            $uri = preg_replace('@/[^/]+/\\.\\./@', '/', $uri, 1, $changed);
 | 
			
		||||
        } while ($changed);
 | 
			
		||||
      
 | 
			
		||||
        self::$debugText .= "traversals removed : {$uri}\n\n";
 | 
			
		||||
        
 | 
			
		||||
        return $uri;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Get realpath with any trailing slash removed. If realpath() fails,
 | 
			
		||||
     * just remove the trailing slash.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $path
 | 
			
		||||
     * 
 | 
			
		||||
     * @return mixed path with no trailing slash
 | 
			
		||||
     */
 | 
			
		||||
    protected static function _realpath($path)
 | 
			
		||||
    {
 | 
			
		||||
        $realPath = realpath($path);
 | 
			
		||||
        if ($realPath !== false) {
 | 
			
		||||
            $path = $realPath;
 | 
			
		||||
        }
 | 
			
		||||
        return rtrim($path, '/\\');
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										130
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Cache/APC.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										130
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Cache/APC.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,130 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_Cache_APC
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * APC-based cache class for Minify
 | 
			
		||||
 * 
 | 
			
		||||
 * <code>
 | 
			
		||||
 * Minify::setCache(new Minify_Cache_APC());
 | 
			
		||||
 * </code>
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Chris Edwards
 | 
			
		||||
 **/
 | 
			
		||||
class Minify_Cache_APC {
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Create a Minify_Cache_APC object, to be passed to
 | 
			
		||||
     * Minify::setCache().
 | 
			
		||||
     *
 | 
			
		||||
     *
 | 
			
		||||
     * @param int $expire seconds until expiration (default = 0
 | 
			
		||||
     * meaning the item will not get an expiration date)
 | 
			
		||||
     *
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    public function __construct($expire = 0)
 | 
			
		||||
    {
 | 
			
		||||
        $this->_exp = $expire;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Write data to cache.
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $id cache id
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $data
 | 
			
		||||
     *
 | 
			
		||||
     * @return bool success
 | 
			
		||||
     */
 | 
			
		||||
    public function store($id, $data)
 | 
			
		||||
    {
 | 
			
		||||
        return apc_store($id, "{$_SERVER['REQUEST_TIME']}|{$data}", $this->_exp);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Get the size of a cache entry
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $id cache id
 | 
			
		||||
     *
 | 
			
		||||
     * @return int size in bytes
 | 
			
		||||
     */
 | 
			
		||||
    public function getSize($id)
 | 
			
		||||
    {
 | 
			
		||||
        return $this->_fetch($id)
 | 
			
		||||
            ? strlen($this->_data)
 | 
			
		||||
            : false;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Does a valid cache entry exist?
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $id cache id
 | 
			
		||||
     *
 | 
			
		||||
     * @param int $srcMtime mtime of the original source file(s)
 | 
			
		||||
     *
 | 
			
		||||
     * @return bool exists
 | 
			
		||||
     */
 | 
			
		||||
    public function isValid($id, $srcMtime)
 | 
			
		||||
    {
 | 
			
		||||
        return ($this->_fetch($id) && ($this->_lm >= $srcMtime));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Send the cached content to output
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $id cache id
 | 
			
		||||
     */
 | 
			
		||||
    public function display($id)
 | 
			
		||||
    {
 | 
			
		||||
        echo $this->_fetch($id)
 | 
			
		||||
            ? $this->_data
 | 
			
		||||
            : '';
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Fetch the cached content
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $id cache id
 | 
			
		||||
     *
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public function fetch($id)
 | 
			
		||||
    {
 | 
			
		||||
        return $this->_fetch($id)
 | 
			
		||||
            ? $this->_data
 | 
			
		||||
            : '';
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private $_exp = null;
 | 
			
		||||
 | 
			
		||||
    // cache of most recently fetched id
 | 
			
		||||
    private $_lm = null;
 | 
			
		||||
    private $_data = null;
 | 
			
		||||
    private $_id = null;
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Fetch data and timestamp from apc, store in instance
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $id
 | 
			
		||||
     *
 | 
			
		||||
     * @return bool success
 | 
			
		||||
     */
 | 
			
		||||
    private function _fetch($id)
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->_id === $id) {
 | 
			
		||||
            return true;
 | 
			
		||||
        }
 | 
			
		||||
        $ret = apc_fetch($id);
 | 
			
		||||
        if (false === $ret) {
 | 
			
		||||
            $this->_id = null;
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
        list($this->_lm, $this->_data) = explode('|', $ret, 2);
 | 
			
		||||
        $this->_id = $id;
 | 
			
		||||
        return true;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										125
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Cache/File.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										125
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Cache/File.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,125 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_Cache_File  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
class Minify_Cache_File {
 | 
			
		||||
    
 | 
			
		||||
    public function __construct($path = '', $fileLocking = false)
 | 
			
		||||
    {
 | 
			
		||||
        if (! $path) {
 | 
			
		||||
            require_once 'Solar/Dir.php';
 | 
			
		||||
            $path = rtrim(Solar_Dir::tmp(), DIRECTORY_SEPARATOR);
 | 
			
		||||
        }
 | 
			
		||||
        $this->_locking = $fileLocking;
 | 
			
		||||
        $this->_path = $path;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Write data to cache.
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $id cache id (e.g. a filename)
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $data
 | 
			
		||||
     * 
 | 
			
		||||
     * @return bool success
 | 
			
		||||
     */
 | 
			
		||||
    public function store($id, $data)
 | 
			
		||||
    {
 | 
			
		||||
        $flag = $this->_locking
 | 
			
		||||
            ? LOCK_EX
 | 
			
		||||
            : null;
 | 
			
		||||
        if (is_file($this->_path . '/' . $id)) {
 | 
			
		||||
            @unlink($this->_path . '/' . $id);
 | 
			
		||||
        }
 | 
			
		||||
        if (! @file_put_contents($this->_path . '/' . $id, $data, $flag)) {
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
        // write control
 | 
			
		||||
        if ($data !== $this->fetch($id)) {
 | 
			
		||||
            @unlink($file);
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
        return true;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Get the size of a cache entry
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $id cache id (e.g. a filename)
 | 
			
		||||
     * 
 | 
			
		||||
     * @return int size in bytes
 | 
			
		||||
     */
 | 
			
		||||
    public function getSize($id)
 | 
			
		||||
    {
 | 
			
		||||
        return filesize($this->_path . '/' . $id);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Does a valid cache entry exist?
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $id cache id (e.g. a filename)
 | 
			
		||||
     * 
 | 
			
		||||
     * @param int $srcMtime mtime of the original source file(s)
 | 
			
		||||
     * 
 | 
			
		||||
     * @return bool exists
 | 
			
		||||
     */
 | 
			
		||||
    public function isValid($id, $srcMtime)
 | 
			
		||||
    {
 | 
			
		||||
        $file = $this->_path . '/' . $id;
 | 
			
		||||
        return (is_file($file) && (filemtime($file) >= $srcMtime));
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Send the cached content to output
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $id cache id (e.g. a filename)
 | 
			
		||||
     */
 | 
			
		||||
    public function display($id)
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->_locking) {
 | 
			
		||||
            $fp = fopen($this->_path . '/' . $id, 'rb');
 | 
			
		||||
            flock($fp, LOCK_SH);
 | 
			
		||||
            fpassthru($fp);
 | 
			
		||||
            flock($fp, LOCK_UN);
 | 
			
		||||
            fclose($fp);
 | 
			
		||||
        } else {
 | 
			
		||||
            readfile($this->_path . '/' . $id);            
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
	/**
 | 
			
		||||
     * Fetch the cached content
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $id cache id (e.g. a filename)
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public function fetch($id)
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->_locking) {
 | 
			
		||||
            $fp = fopen($this->_path . '/' . $id, 'rb');
 | 
			
		||||
            flock($fp, LOCK_SH);
 | 
			
		||||
            $ret = stream_get_contents($fp);
 | 
			
		||||
            flock($fp, LOCK_UN);
 | 
			
		||||
            fclose($fp);
 | 
			
		||||
            return $ret;
 | 
			
		||||
        } else {
 | 
			
		||||
            return file_get_contents($this->_path . '/' . $id);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Fetch the cache path used
 | 
			
		||||
     *
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public function getPath()
 | 
			
		||||
    {
 | 
			
		||||
        return $this->_path;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    private $_path = null;
 | 
			
		||||
    private $_locking = null;
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										137
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Cache/Memcache.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										137
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Cache/Memcache.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,137 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_Cache_Memcache
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Memcache-based cache class for Minify
 | 
			
		||||
 * 
 | 
			
		||||
 * <code>
 | 
			
		||||
 * // fall back to disk caching if memcache can't connect
 | 
			
		||||
 * $memcache = new Memcache;
 | 
			
		||||
 * if ($memcache->connect('localhost', 11211)) {
 | 
			
		||||
 *     Minify::setCache(new Minify_Cache_Memcache($memcache));
 | 
			
		||||
 * } else {
 | 
			
		||||
 *     Minify::setCache();
 | 
			
		||||
 * }
 | 
			
		||||
 * </code>
 | 
			
		||||
 **/
 | 
			
		||||
class Minify_Cache_Memcache {
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Create a Minify_Cache_Memcache object, to be passed to 
 | 
			
		||||
     * Minify::setCache().
 | 
			
		||||
     *
 | 
			
		||||
     * @param Memcache $memcache already-connected instance
 | 
			
		||||
     * 
 | 
			
		||||
     * @param int $expire seconds until expiration (default = 0
 | 
			
		||||
     * meaning the item will not get an expiration date)
 | 
			
		||||
     * 
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    public function __construct($memcache, $expire = 0)
 | 
			
		||||
    {
 | 
			
		||||
        $this->_mc = $memcache;
 | 
			
		||||
        $this->_exp = $expire;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Write data to cache.
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $id cache id
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $data
 | 
			
		||||
     * 
 | 
			
		||||
     * @return bool success
 | 
			
		||||
     */
 | 
			
		||||
    public function store($id, $data)
 | 
			
		||||
    {
 | 
			
		||||
        return $this->_mc->set($id, "{$_SERVER['REQUEST_TIME']}|{$data}", 0, $this->_exp);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Get the size of a cache entry
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $id cache id
 | 
			
		||||
     * 
 | 
			
		||||
     * @return int size in bytes
 | 
			
		||||
     */
 | 
			
		||||
    public function getSize($id)
 | 
			
		||||
    {
 | 
			
		||||
        return $this->_fetch($id)
 | 
			
		||||
            ? strlen($this->_data)
 | 
			
		||||
            : false;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Does a valid cache entry exist?
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $id cache id
 | 
			
		||||
     * 
 | 
			
		||||
     * @param int $srcMtime mtime of the original source file(s)
 | 
			
		||||
     * 
 | 
			
		||||
     * @return bool exists
 | 
			
		||||
     */
 | 
			
		||||
    public function isValid($id, $srcMtime)
 | 
			
		||||
    {
 | 
			
		||||
        return ($this->_fetch($id) && ($this->_lm >= $srcMtime));
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Send the cached content to output
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $id cache id
 | 
			
		||||
     */
 | 
			
		||||
    public function display($id)
 | 
			
		||||
    {
 | 
			
		||||
        echo $this->_fetch($id)
 | 
			
		||||
            ? $this->_data
 | 
			
		||||
            : '';
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
	/**
 | 
			
		||||
     * Fetch the cached content
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $id cache id
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public function fetch($id)
 | 
			
		||||
    {
 | 
			
		||||
        return $this->_fetch($id)
 | 
			
		||||
            ? $this->_data
 | 
			
		||||
            : '';
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    private $_mc = null;
 | 
			
		||||
    private $_exp = null;
 | 
			
		||||
    
 | 
			
		||||
    // cache of most recently fetched id
 | 
			
		||||
    private $_lm = null;
 | 
			
		||||
    private $_data = null;
 | 
			
		||||
    private $_id = null;
 | 
			
		||||
    
 | 
			
		||||
	/**
 | 
			
		||||
     * Fetch data and timestamp from memcache, store in instance
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $id
 | 
			
		||||
     * 
 | 
			
		||||
     * @return bool success
 | 
			
		||||
     */
 | 
			
		||||
    private function _fetch($id)
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->_id === $id) {
 | 
			
		||||
            return true;
 | 
			
		||||
        }
 | 
			
		||||
        $ret = $this->_mc->get($id);
 | 
			
		||||
        if (false === $ret) {
 | 
			
		||||
            $this->_id = null;
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
        list($this->_lm, $this->_data) = explode('|', $ret, 2);
 | 
			
		||||
        $this->_id = $id;
 | 
			
		||||
        return true;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,90 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_CommentPreserver 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Process a string in pieces preserving C-style comments that begin with "/*!"
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 */
 | 
			
		||||
class Minify_CommentPreserver {
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * String to be prepended to each preserved comment
 | 
			
		||||
     *
 | 
			
		||||
     * @var string
 | 
			
		||||
     */
 | 
			
		||||
    public static $prepend = "\n";
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * String to be appended to each preserved comment
 | 
			
		||||
     *
 | 
			
		||||
     * @var string
 | 
			
		||||
     */
 | 
			
		||||
    public static $append = "\n";
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Process a string outside of C-style comments that begin with "/*!"
 | 
			
		||||
     *
 | 
			
		||||
     * On each non-empty string outside these comments, the given processor 
 | 
			
		||||
     * function will be called. The first "!" will be removed from the 
 | 
			
		||||
     * preserved comments, and the comments will be surrounded by 
 | 
			
		||||
     * Minify_CommentPreserver::$preprend and Minify_CommentPreserver::$append.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $content
 | 
			
		||||
     * @param callback $processor function
 | 
			
		||||
     * @param array $args array of extra arguments to pass to the processor 
 | 
			
		||||
     * function (default = array())
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public static function process($content, $processor, $args = array())
 | 
			
		||||
    {
 | 
			
		||||
        $ret = '';
 | 
			
		||||
        while (true) {
 | 
			
		||||
            list($beforeComment, $comment, $afterComment) = self::_nextComment($content);
 | 
			
		||||
            if ('' !== $beforeComment) {
 | 
			
		||||
                $callArgs = $args;
 | 
			
		||||
                array_unshift($callArgs, $beforeComment);
 | 
			
		||||
                $ret .= call_user_func_array($processor, $callArgs);    
 | 
			
		||||
            }
 | 
			
		||||
            if (false === $comment) {
 | 
			
		||||
                break;
 | 
			
		||||
            }
 | 
			
		||||
            $ret .= $comment;
 | 
			
		||||
            $content = $afterComment;
 | 
			
		||||
        }
 | 
			
		||||
        return $ret;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Extract comments that YUI Compressor preserves.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $in input
 | 
			
		||||
     * 
 | 
			
		||||
     * @return array 3 elements are returned. If a YUI comment is found, the
 | 
			
		||||
     * 2nd element is the comment and the 1st and 2nd are the surrounding
 | 
			
		||||
     * strings. If no comment is found, the entire string is returned as the 
 | 
			
		||||
     * 1st element and the other two are false.
 | 
			
		||||
     */
 | 
			
		||||
    private static function _nextComment($in)
 | 
			
		||||
    {
 | 
			
		||||
        if (
 | 
			
		||||
            false === ($start = strpos($in, '/*!'))
 | 
			
		||||
            || false === ($end = strpos($in, '*/', $start + 3))
 | 
			
		||||
        ) {
 | 
			
		||||
            return array($in, false, false);
 | 
			
		||||
        }
 | 
			
		||||
        $ret = array(
 | 
			
		||||
            substr($in, 0, $start)
 | 
			
		||||
            ,self::$prepend . '/*' . substr($in, $start + 3, $end - $start - 1) . self::$append
 | 
			
		||||
        );
 | 
			
		||||
        $endChars = (strlen($in) - $end - 2);
 | 
			
		||||
        $ret[] = (0 === $endChars)
 | 
			
		||||
            ? ''
 | 
			
		||||
            : substr($in, -$endChars);
 | 
			
		||||
        return $ret;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										202
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Controller/Base.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										202
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Controller/Base.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,202 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_Controller_Base  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Base class for Minify controller
 | 
			
		||||
 * 
 | 
			
		||||
 * The controller class validates a request and uses it to create sources
 | 
			
		||||
 * for minification and set options like contentType. It's also responsible
 | 
			
		||||
 * for loading minifier code upon request.
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 */
 | 
			
		||||
abstract class Minify_Controller_Base {
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Setup controller sources and set an needed options for Minify::source
 | 
			
		||||
     * 
 | 
			
		||||
     * You must override this method in your subclass controller to set 
 | 
			
		||||
     * $this->sources. If the request is NOT valid, make sure $this->sources 
 | 
			
		||||
     * is left an empty array. Then strip any controller-specific options from 
 | 
			
		||||
     * $options and return it. To serve files, $this->sources must be an array of
 | 
			
		||||
     * Minify_Source objects.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $options controller and Minify options
 | 
			
		||||
     * 
 | 
			
		||||
     * return array $options Minify::serve options
 | 
			
		||||
     */
 | 
			
		||||
    abstract public function setupSources($options);
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Get default Minify options for this controller.
 | 
			
		||||
     * 
 | 
			
		||||
     * Override in subclass to change defaults
 | 
			
		||||
     *
 | 
			
		||||
     * @return array options for Minify
 | 
			
		||||
     */
 | 
			
		||||
    public function getDefaultMinifyOptions() {
 | 
			
		||||
        return array(
 | 
			
		||||
            'isPublic' => true
 | 
			
		||||
            ,'encodeOutput' => function_exists('gzdeflate')
 | 
			
		||||
            ,'encodeMethod' => null // determine later
 | 
			
		||||
            ,'encodeLevel' => 9
 | 
			
		||||
            ,'minifierOptions' => array() // no minifier options
 | 
			
		||||
            ,'contentTypeCharset' => 'utf-8'
 | 
			
		||||
            ,'maxAge' => 1800 // 30 minutes
 | 
			
		||||
            ,'rewriteCssUris' => true
 | 
			
		||||
            ,'bubbleCssImports' => false
 | 
			
		||||
            ,'quiet' => false // serve() will send headers and output
 | 
			
		||||
            ,'debug' => false
 | 
			
		||||
            
 | 
			
		||||
            // if you override this, the response code MUST be directly after 
 | 
			
		||||
            // the first space.
 | 
			
		||||
            ,'badRequestHeader' => 'HTTP/1.0 400 Bad Request'
 | 
			
		||||
            
 | 
			
		||||
            // callback function to see/modify content of all sources
 | 
			
		||||
            ,'postprocessor' => null
 | 
			
		||||
            // file to require to load preprocessor
 | 
			
		||||
            ,'postprocessorRequire' => null
 | 
			
		||||
        );
 | 
			
		||||
    }  
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Get default minifiers for this controller.
 | 
			
		||||
     * 
 | 
			
		||||
     * Override in subclass to change defaults
 | 
			
		||||
     *
 | 
			
		||||
     * @return array minifier callbacks for common types
 | 
			
		||||
     */
 | 
			
		||||
    public function getDefaultMinifers() {
 | 
			
		||||
        $ret[Minify::TYPE_JS] = array('JSMin', 'minify');
 | 
			
		||||
        $ret[Minify::TYPE_CSS] = array('Minify_CSS', 'minify');
 | 
			
		||||
        $ret[Minify::TYPE_HTML] = array('Minify_HTML', 'minify');
 | 
			
		||||
        return $ret;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Load any code necessary to execute the given minifier callback.
 | 
			
		||||
     * 
 | 
			
		||||
     * The controller is responsible for loading minification code on demand
 | 
			
		||||
     * via this method. This built-in function will only load classes for
 | 
			
		||||
     * static method callbacks where the class isn't already defined. It uses
 | 
			
		||||
     * the PEAR convention, so, given array('Jimmy_Minifier', 'minCss'), this 
 | 
			
		||||
     * function will include 'Jimmy/Minifier.php'.
 | 
			
		||||
     * 
 | 
			
		||||
     * If you need code loaded on demand and this doesn't suit you, you'll need
 | 
			
		||||
     * to override this function in your subclass. 
 | 
			
		||||
     * @see Minify_Controller_Page::loadMinifier()
 | 
			
		||||
     * 
 | 
			
		||||
     * @param callback $minifierCallback callback of minifier function
 | 
			
		||||
     * 
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    public function loadMinifier($minifierCallback)
 | 
			
		||||
    {
 | 
			
		||||
        if (is_array($minifierCallback)
 | 
			
		||||
            && is_string($minifierCallback[0])
 | 
			
		||||
            && !class_exists($minifierCallback[0], false)) {
 | 
			
		||||
            
 | 
			
		||||
            require str_replace('_', '/', $minifierCallback[0]) . '.php';
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Is a user-given file within an allowable directory, existing,
 | 
			
		||||
     * and having an extension js/css/html/txt ?
 | 
			
		||||
     * 
 | 
			
		||||
     * This is a convenience function for controllers that have to accept
 | 
			
		||||
     * user-given paths
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $file full file path (already processed by realpath())
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $safeDirs directories where files are safe to serve. Files can also
 | 
			
		||||
     * be in subdirectories of these directories.
 | 
			
		||||
     * 
 | 
			
		||||
     * @return bool file is safe
 | 
			
		||||
     */
 | 
			
		||||
    public static function _fileIsSafe($file, $safeDirs)
 | 
			
		||||
    {
 | 
			
		||||
        $pathOk = false;
 | 
			
		||||
        foreach ((array)$safeDirs as $safeDir) {
 | 
			
		||||
            if (strpos($file, $safeDir) === 0) {
 | 
			
		||||
                $pathOk = true;
 | 
			
		||||
                break;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        $base = basename($file);
 | 
			
		||||
        if (! $pathOk || ! is_file($file) || $base[0] === '.') {
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
        list($revExt) = explode('.', strrev($base));
 | 
			
		||||
        return in_array(strrev($revExt), array('js', 'css', 'html', 'txt'));
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * @var array instances of Minify_Source, which provide content and
 | 
			
		||||
     * any individual minification needs.
 | 
			
		||||
     * 
 | 
			
		||||
     * @see Minify_Source
 | 
			
		||||
     */
 | 
			
		||||
    public $sources = array();
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Mix in default controller options with user-given options
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $options user options
 | 
			
		||||
     * 
 | 
			
		||||
     * @return array mixed options
 | 
			
		||||
     */
 | 
			
		||||
    public final function mixInDefaultOptions($options)
 | 
			
		||||
    {
 | 
			
		||||
        $ret = array_merge(
 | 
			
		||||
            $this->getDefaultMinifyOptions(), $options
 | 
			
		||||
        );
 | 
			
		||||
        if (! isset($options['minifiers'])) {
 | 
			
		||||
            $options['minifiers'] = array();
 | 
			
		||||
        }
 | 
			
		||||
        $ret['minifiers'] = array_merge(
 | 
			
		||||
            $this->getDefaultMinifers(), $options['minifiers']
 | 
			
		||||
        );
 | 
			
		||||
        return $ret;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Analyze sources (if there are any) and set $options 'contentType' 
 | 
			
		||||
     * and 'lastModifiedTime' if they already aren't.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $options options for Minify
 | 
			
		||||
     * 
 | 
			
		||||
     * @return array options for Minify
 | 
			
		||||
     */
 | 
			
		||||
    public final function analyzeSources($options = array()) 
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->sources) {
 | 
			
		||||
            if (! isset($options['contentType'])) {
 | 
			
		||||
                $options['contentType'] = Minify_Source::getContentType($this->sources);
 | 
			
		||||
            }
 | 
			
		||||
            // last modified is needed for caching, even if setExpires is set
 | 
			
		||||
            if (! isset($options['lastModifiedTime'])) {
 | 
			
		||||
                $max = 0;
 | 
			
		||||
                foreach ($this->sources as $source) {
 | 
			
		||||
                    $max = max($source->lastModified, $max);
 | 
			
		||||
                }
 | 
			
		||||
                $options['lastModifiedTime'] = $max;
 | 
			
		||||
            }    
 | 
			
		||||
        }
 | 
			
		||||
        return $options;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Send message to the Minify logger
 | 
			
		||||
     * @param string $msg
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    protected function log($msg) {
 | 
			
		||||
        require_once 'Minify/Logger.php';
 | 
			
		||||
        Minify_Logger::log($msg);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,78 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_Controller_Files  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
require_once 'Minify/Controller/Base.php';
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Controller class for minifying a set of files
 | 
			
		||||
 * 
 | 
			
		||||
 * E.g. the following would serve the minified Javascript for a site
 | 
			
		||||
 * <code>
 | 
			
		||||
 * Minify::serve('Files', array(
 | 
			
		||||
 *     'files' => array(
 | 
			
		||||
 *         '//js/jquery.js'
 | 
			
		||||
 *         ,'//js/plugins.js'
 | 
			
		||||
 *         ,'/home/username/file.js'
 | 
			
		||||
 *     )
 | 
			
		||||
 * ));
 | 
			
		||||
 * </code>
 | 
			
		||||
 * 
 | 
			
		||||
 * As a shortcut, the controller will replace "//" at the beginning
 | 
			
		||||
 * of a filename with $_SERVER['DOCUMENT_ROOT'] . '/'.
 | 
			
		||||
 *
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 */
 | 
			
		||||
class Minify_Controller_Files extends Minify_Controller_Base {
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Set up file sources
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $options controller and Minify options
 | 
			
		||||
     * @return array Minify options
 | 
			
		||||
     * 
 | 
			
		||||
     * Controller options:
 | 
			
		||||
     * 
 | 
			
		||||
     * 'files': (required) array of complete file paths, or a single path
 | 
			
		||||
     */
 | 
			
		||||
    public function setupSources($options) {
 | 
			
		||||
        // strip controller options
 | 
			
		||||
        
 | 
			
		||||
        $files = $options['files'];
 | 
			
		||||
        // if $files is a single object, casting will break it
 | 
			
		||||
        if (is_object($files)) {
 | 
			
		||||
            $files = array($files);
 | 
			
		||||
        } elseif (! is_array($files)) {
 | 
			
		||||
            $files = (array)$files;
 | 
			
		||||
        }
 | 
			
		||||
        unset($options['files']);
 | 
			
		||||
        
 | 
			
		||||
        $sources = array();
 | 
			
		||||
        foreach ($files as $file) {
 | 
			
		||||
            if ($file instanceof Minify_Source) {
 | 
			
		||||
                $sources[] = $file;
 | 
			
		||||
                continue;
 | 
			
		||||
            }
 | 
			
		||||
            if (0 === strpos($file, '//')) {
 | 
			
		||||
                $file = $_SERVER['DOCUMENT_ROOT'] . substr($file, 1);
 | 
			
		||||
            }
 | 
			
		||||
            $realPath = realpath($file);
 | 
			
		||||
            if (is_file($realPath)) {
 | 
			
		||||
                $sources[] = new Minify_Source(array(
 | 
			
		||||
                    'filepath' => $realPath
 | 
			
		||||
                ));    
 | 
			
		||||
            } else {
 | 
			
		||||
                $this->log("The path \"{$file}\" could not be found (or was not a file)");
 | 
			
		||||
                return $options;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        if ($sources) {
 | 
			
		||||
            $this->sources = $sources;
 | 
			
		||||
        }
 | 
			
		||||
        return $options;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -0,0 +1,94 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_Controller_Groups  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
require_once 'Minify/Controller/Base.php';
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Controller class for serving predetermined groups of minimized sets, selected
 | 
			
		||||
 * by PATH_INFO
 | 
			
		||||
 * 
 | 
			
		||||
 * <code>
 | 
			
		||||
 * Minify::serve('Groups', array( 
 | 
			
		||||
 *     'groups' => array(
 | 
			
		||||
 *         'css' => array('//css/type.css', '//css/layout.css')
 | 
			
		||||
 *        ,'js' => array('//js/jquery.js', '//js/site.js')
 | 
			
		||||
 *     )
 | 
			
		||||
 * ));
 | 
			
		||||
 * </code>
 | 
			
		||||
 * 
 | 
			
		||||
 * If the above code were placed in /serve.php, it would enable the URLs
 | 
			
		||||
 * /serve.php/js and /serve.php/css
 | 
			
		||||
 * 
 | 
			
		||||
 * As a shortcut, the controller will replace "//" at the beginning
 | 
			
		||||
 * of a filename with $_SERVER['DOCUMENT_ROOT'] . '/'.
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 */
 | 
			
		||||
class Minify_Controller_Groups extends Minify_Controller_Base {
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Set up groups of files as sources
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $options controller and Minify options
 | 
			
		||||
     * @return array Minify options
 | 
			
		||||
     * 
 | 
			
		||||
     * Controller options:
 | 
			
		||||
     * 
 | 
			
		||||
     * 'groups': (required) array mapping PATH_INFO strings to arrays
 | 
			
		||||
     * of complete file paths. @see Minify_Controller_Groups 
 | 
			
		||||
     */
 | 
			
		||||
    public function setupSources($options) {
 | 
			
		||||
        // strip controller options
 | 
			
		||||
        $groups = $options['groups'];
 | 
			
		||||
        unset($options['groups']);
 | 
			
		||||
        
 | 
			
		||||
        // mod_fcgid places PATH_INFO in ORIG_PATH_INFO
 | 
			
		||||
        $pi = isset($_SERVER['ORIG_PATH_INFO'])
 | 
			
		||||
            ? substr($_SERVER['ORIG_PATH_INFO'], 1) 
 | 
			
		||||
            : (isset($_SERVER['PATH_INFO'])
 | 
			
		||||
                ? substr($_SERVER['PATH_INFO'], 1) 
 | 
			
		||||
                : false
 | 
			
		||||
            );
 | 
			
		||||
        if (false === $pi || ! isset($groups[$pi])) {
 | 
			
		||||
            // no PATH_INFO or not a valid group
 | 
			
		||||
            $this->log("Missing PATH_INFO or no group set for \"$pi\"");
 | 
			
		||||
            return $options;
 | 
			
		||||
        }
 | 
			
		||||
        $sources = array();
 | 
			
		||||
        
 | 
			
		||||
        $files = $groups[$pi];
 | 
			
		||||
        // if $files is a single object, casting will break it
 | 
			
		||||
        if (is_object($files)) {
 | 
			
		||||
            $files = array($files);
 | 
			
		||||
        } elseif (! is_array($files)) {
 | 
			
		||||
            $files = (array)$files;
 | 
			
		||||
        }
 | 
			
		||||
        foreach ($files as $file) {
 | 
			
		||||
            if ($file instanceof Minify_Source) {
 | 
			
		||||
                $sources[] = $file;
 | 
			
		||||
                continue;
 | 
			
		||||
            }
 | 
			
		||||
            if (0 === strpos($file, '//')) {
 | 
			
		||||
                $file = $_SERVER['DOCUMENT_ROOT'] . substr($file, 1);
 | 
			
		||||
            }
 | 
			
		||||
            $realPath = realpath($file);
 | 
			
		||||
            if (is_file($realPath)) {
 | 
			
		||||
                $sources[] = new Minify_Source(array(
 | 
			
		||||
                    'filepath' => $realPath
 | 
			
		||||
                ));    
 | 
			
		||||
            } else {
 | 
			
		||||
                $this->log("The path \"{$file}\" could not be found (or was not a file)");
 | 
			
		||||
                return $options;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        if ($sources) {
 | 
			
		||||
            $this->sources = $sources;
 | 
			
		||||
        }
 | 
			
		||||
        return $options;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -0,0 +1,132 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_Controller_MinApp  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
require_once 'Minify/Controller/Base.php';
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Controller class for requests to /min/index.php
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 */
 | 
			
		||||
class Minify_Controller_MinApp extends Minify_Controller_Base {
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Set up groups of files as sources
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $options controller and Minify options
 | 
			
		||||
     * @return array Minify options
 | 
			
		||||
     * 
 | 
			
		||||
     */
 | 
			
		||||
    public function setupSources($options) {
 | 
			
		||||
        // filter controller options
 | 
			
		||||
        $cOptions = array_merge(
 | 
			
		||||
            array(
 | 
			
		||||
                'allowDirs' => '//'
 | 
			
		||||
                ,'groupsOnly' => false
 | 
			
		||||
                ,'groups' => array()
 | 
			
		||||
                ,'maxFiles' => 10                
 | 
			
		||||
            )
 | 
			
		||||
            ,(isset($options['minApp']) ? $options['minApp'] : array())
 | 
			
		||||
        );
 | 
			
		||||
        unset($options['minApp']);
 | 
			
		||||
        $sources = array();
 | 
			
		||||
        if (isset($_GET['g'])) {
 | 
			
		||||
            // try groups
 | 
			
		||||
            if (! isset($cOptions['groups'][$_GET['g']])) {
 | 
			
		||||
                $this->log("A group configuration for \"{$_GET['g']}\" was not set");
 | 
			
		||||
                return $options;
 | 
			
		||||
            }
 | 
			
		||||
            
 | 
			
		||||
            $files = $cOptions['groups'][$_GET['g']];
 | 
			
		||||
            // if $files is a single object, casting will break it
 | 
			
		||||
            if (is_object($files)) {
 | 
			
		||||
                $files = array($files);
 | 
			
		||||
            } elseif (! is_array($files)) {
 | 
			
		||||
                $files = (array)$files;
 | 
			
		||||
            }
 | 
			
		||||
            foreach ($files as $file) {
 | 
			
		||||
                if ($file instanceof Minify_Source) {
 | 
			
		||||
                    $sources[] = $file;
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                if (0 === strpos($file, '//')) {
 | 
			
		||||
                    $file = $_SERVER['DOCUMENT_ROOT'] . substr($file, 1);
 | 
			
		||||
                }
 | 
			
		||||
                $file = realpath($file);
 | 
			
		||||
                if (is_file($file)) {
 | 
			
		||||
                    $sources[] = new Minify_Source(array(
 | 
			
		||||
                        'filepath' => $file
 | 
			
		||||
                    ));    
 | 
			
		||||
                } else {
 | 
			
		||||
                    $this->log("The path \"{$file}\" could not be found (or was not a file)");
 | 
			
		||||
                    return $options;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        } elseif (! $cOptions['groupsOnly'] && isset($_GET['f'])) {
 | 
			
		||||
            // try user files
 | 
			
		||||
            // The following restrictions are to limit the URLs that minify will
 | 
			
		||||
            // respond to. Ideally there should be only one way to reference a file.
 | 
			
		||||
            if (// verify at least one file, files are single comma separated, 
 | 
			
		||||
                // and are all same extension
 | 
			
		||||
                ! preg_match('/^[^,]+\\.(css|js)(?:,[^,]+\\.\\1)*$/', $_GET['f'])
 | 
			
		||||
                // no "//"
 | 
			
		||||
                || strpos($_GET['f'], '//') !== false
 | 
			
		||||
                // no "\"
 | 
			
		||||
                || strpos($_GET['f'], '\\') !== false
 | 
			
		||||
                // no "./"
 | 
			
		||||
                || preg_match('/(?:^|[^\\.])\\.\\//', $_GET['f'])
 | 
			
		||||
            ) {
 | 
			
		||||
                $this->log("GET param 'f' invalid (see MinApp.php line 63)");
 | 
			
		||||
                return $options;
 | 
			
		||||
            }
 | 
			
		||||
            $files = explode(',', $_GET['f']);
 | 
			
		||||
            if (count($files) > $cOptions['maxFiles'] || $files != array_unique($files)) {
 | 
			
		||||
                $this->log("Too many or duplicate files specified");
 | 
			
		||||
                return $options;
 | 
			
		||||
            }
 | 
			
		||||
            if (isset($_GET['b'])) {
 | 
			
		||||
                // check for validity
 | 
			
		||||
                if (preg_match('@^[^/]+(?:/[^/]+)*$@', $_GET['b'])
 | 
			
		||||
                    && false === strpos($_GET['b'], '..')
 | 
			
		||||
                    && $_GET['b'] !== '.') {
 | 
			
		||||
                    // valid base
 | 
			
		||||
                    $base = "/{$_GET['b']}/";       
 | 
			
		||||
                } else {
 | 
			
		||||
                    $this->log("GET param 'b' invalid (see MinApp.php line 84)");
 | 
			
		||||
                    return $options;
 | 
			
		||||
                }
 | 
			
		||||
            } else {
 | 
			
		||||
                $base = '/';
 | 
			
		||||
            }
 | 
			
		||||
            $allowDirs = array();
 | 
			
		||||
            foreach ((array)$cOptions['allowDirs'] as $allowDir) {
 | 
			
		||||
                $allowDirs[] = realpath(str_replace('//', $_SERVER['DOCUMENT_ROOT'] . '/', $allowDir));
 | 
			
		||||
            }
 | 
			
		||||
            foreach ($files as $file) {
 | 
			
		||||
                $path = $_SERVER['DOCUMENT_ROOT'] . $base . $file;
 | 
			
		||||
                $file = realpath($path);
 | 
			
		||||
                if (false === $file) {
 | 
			
		||||
                    $this->log("Path \"{$path}\" failed realpath()");
 | 
			
		||||
                    return $options;
 | 
			
		||||
                } elseif (! parent::_fileIsSafe($file, $allowDirs)) {
 | 
			
		||||
                    $this->log("Path \"{$path}\" failed Minify_Controller_Base::_fileIsSafe()");
 | 
			
		||||
                    return $options;
 | 
			
		||||
                } else {
 | 
			
		||||
                    $sources[] = new Minify_Source(array(
 | 
			
		||||
                        'filepath' => $file
 | 
			
		||||
                    ));
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        if ($sources) {
 | 
			
		||||
            $this->sources = $sources;
 | 
			
		||||
        } else {
 | 
			
		||||
            $this->log("No sources to serve");
 | 
			
		||||
        }
 | 
			
		||||
        return $options;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,82 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_Controller_Page  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
require_once 'Minify/Controller/Base.php';
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Controller class for serving a single HTML page
 | 
			
		||||
 * 
 | 
			
		||||
 * @link http://code.google.com/p/minify/source/browse/trunk/web/examples/1/index.php#59
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 */
 | 
			
		||||
class Minify_Controller_Page extends Minify_Controller_Base {
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Set up source of HTML content
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $options controller and Minify options
 | 
			
		||||
     * @return array Minify options
 | 
			
		||||
     * 
 | 
			
		||||
     * Controller options:
 | 
			
		||||
     * 
 | 
			
		||||
     * 'content': (required) HTML markup
 | 
			
		||||
     * 
 | 
			
		||||
     * 'id': (required) id of page (string for use in server-side caching)
 | 
			
		||||
     * 
 | 
			
		||||
     * 'lastModifiedTime': timestamp of when this content changed. This
 | 
			
		||||
     * is recommended to allow both server and client-side caching.
 | 
			
		||||
     * 
 | 
			
		||||
     * 'minifyAll': should all CSS and Javascript blocks be individually 
 | 
			
		||||
     * minified? (default false) 
 | 
			
		||||
     *
 | 
			
		||||
     * @todo Add 'file' option to read HTML file.
 | 
			
		||||
     */
 | 
			
		||||
    public function setupSources($options) {
 | 
			
		||||
        if (isset($options['file'])) {
 | 
			
		||||
            $sourceSpec = array(
 | 
			
		||||
                'filepath' => $options['file']
 | 
			
		||||
            );
 | 
			
		||||
        } else {
 | 
			
		||||
            // strip controller options
 | 
			
		||||
            $sourceSpec = array(
 | 
			
		||||
                'content' => $options['content']
 | 
			
		||||
                ,'id' => $options['id']
 | 
			
		||||
            );
 | 
			
		||||
            unset($options['content'], $options['id']);
 | 
			
		||||
        }
 | 
			
		||||
        if (isset($options['minifyAll'])) {
 | 
			
		||||
            // this will be the 2nd argument passed to Minify_HTML::minify()
 | 
			
		||||
            $sourceSpec['minifyOptions'] = array(
 | 
			
		||||
                'cssMinifier' => array('Minify_CSS', 'minify')
 | 
			
		||||
                ,'jsMinifier' => array('JSMin', 'minify')
 | 
			
		||||
            );
 | 
			
		||||
            $this->_loadCssJsMinifiers = true;
 | 
			
		||||
            unset($options['minifyAll']);
 | 
			
		||||
        }
 | 
			
		||||
        $this->sources[] = new Minify_Source($sourceSpec);
 | 
			
		||||
        
 | 
			
		||||
        $options['contentType'] = Minify::TYPE_HTML;
 | 
			
		||||
        return $options;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    protected $_loadCssJsMinifiers = false;
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * @see Minify_Controller_Base::loadMinifier()
 | 
			
		||||
     */
 | 
			
		||||
    public function loadMinifier($minifierCallback)
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->_loadCssJsMinifiers) {
 | 
			
		||||
            // Minify will not call for these so we must manually load
 | 
			
		||||
            // them when Minify/HTML.php is called for.
 | 
			
		||||
            require_once 'Minify/CSS.php';
 | 
			
		||||
            require_once 'JSMin.php';
 | 
			
		||||
        }
 | 
			
		||||
        parent::loadMinifier($minifierCallback); // load Minify/HTML.php
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -0,0 +1,118 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_Controller_Version1  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
require_once 'Minify/Controller/Base.php';
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Controller class for emulating version 1 of minify.php
 | 
			
		||||
 * 
 | 
			
		||||
 * <code>
 | 
			
		||||
 * Minify::serve('Version1');
 | 
			
		||||
 * </code>
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 */
 | 
			
		||||
class Minify_Controller_Version1 extends Minify_Controller_Base {
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Set up groups of files as sources
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $options controller and Minify options
 | 
			
		||||
     * @return array Minify options
 | 
			
		||||
     * 
 | 
			
		||||
     */
 | 
			
		||||
    public function setupSources($options) {
 | 
			
		||||
        self::_setupDefines();
 | 
			
		||||
        if (MINIFY_USE_CACHE) {
 | 
			
		||||
            $cacheDir = defined('MINIFY_CACHE_DIR')
 | 
			
		||||
                ? MINIFY_CACHE_DIR
 | 
			
		||||
                : '';
 | 
			
		||||
            Minify::setCache($cacheDir);
 | 
			
		||||
        }
 | 
			
		||||
        $options['badRequestHeader'] = 'HTTP/1.0 404 Not Found';
 | 
			
		||||
        $options['contentTypeCharset'] = MINIFY_ENCODING;
 | 
			
		||||
 | 
			
		||||
        // The following restrictions are to limit the URLs that minify will
 | 
			
		||||
        // respond to. Ideally there should be only one way to reference a file.
 | 
			
		||||
        if (! isset($_GET['files'])
 | 
			
		||||
            // verify at least one file, files are single comma separated, 
 | 
			
		||||
            // and are all same extension
 | 
			
		||||
            || ! preg_match('/^[^,]+\\.(css|js)(,[^,]+\\.\\1)*$/', $_GET['files'], $m)
 | 
			
		||||
            // no "//" (makes URL rewriting easier)
 | 
			
		||||
            || strpos($_GET['files'], '//') !== false
 | 
			
		||||
            // no "\"
 | 
			
		||||
            || strpos($_GET['files'], '\\') !== false
 | 
			
		||||
            // no "./"
 | 
			
		||||
            || preg_match('/(?:^|[^\\.])\\.\\//', $_GET['files'])
 | 
			
		||||
        ) {
 | 
			
		||||
            return $options;
 | 
			
		||||
        }
 | 
			
		||||
        $extension = $m[1];
 | 
			
		||||
        
 | 
			
		||||
        $files = explode(',', $_GET['files']);
 | 
			
		||||
        if (count($files) > MINIFY_MAX_FILES) {
 | 
			
		||||
            return $options;
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        // strings for prepending to relative/absolute paths
 | 
			
		||||
        $prependRelPaths = dirname($_SERVER['SCRIPT_FILENAME'])
 | 
			
		||||
            . DIRECTORY_SEPARATOR;
 | 
			
		||||
        $prependAbsPaths = $_SERVER['DOCUMENT_ROOT'];
 | 
			
		||||
        
 | 
			
		||||
        $sources = array();
 | 
			
		||||
        $goodFiles = array();
 | 
			
		||||
        $hasBadSource = false;
 | 
			
		||||
        
 | 
			
		||||
        $allowDirs = isset($options['allowDirs'])
 | 
			
		||||
            ? $options['allowDirs']
 | 
			
		||||
            : MINIFY_BASE_DIR;
 | 
			
		||||
        
 | 
			
		||||
        foreach ($files as $file) {
 | 
			
		||||
            // prepend appropriate string for abs/rel paths
 | 
			
		||||
            $file = ($file[0] === '/' ? $prependAbsPaths : $prependRelPaths) . $file;
 | 
			
		||||
            // make sure a real file!
 | 
			
		||||
            $file = realpath($file);
 | 
			
		||||
            // don't allow unsafe or duplicate files
 | 
			
		||||
            if (parent::_fileIsSafe($file, $allowDirs) 
 | 
			
		||||
                && !in_array($file, $goodFiles)) 
 | 
			
		||||
            {
 | 
			
		||||
                $goodFiles[] = $file;
 | 
			
		||||
                $srcOptions = array(
 | 
			
		||||
                    'filepath' => $file
 | 
			
		||||
                );
 | 
			
		||||
                $this->sources[] = new Minify_Source($srcOptions);
 | 
			
		||||
            } else {
 | 
			
		||||
                $hasBadSource = true;
 | 
			
		||||
                break;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        if ($hasBadSource) {
 | 
			
		||||
            $this->sources = array();
 | 
			
		||||
        }
 | 
			
		||||
        if (! MINIFY_REWRITE_CSS_URLS) {
 | 
			
		||||
            $options['rewriteCssUris'] = false;
 | 
			
		||||
        }
 | 
			
		||||
        return $options;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    private static function _setupDefines()
 | 
			
		||||
    {
 | 
			
		||||
        $defaults = array(
 | 
			
		||||
            'MINIFY_BASE_DIR' => realpath($_SERVER['DOCUMENT_ROOT'])
 | 
			
		||||
            ,'MINIFY_ENCODING' => 'utf-8'
 | 
			
		||||
            ,'MINIFY_MAX_FILES' => 16
 | 
			
		||||
            ,'MINIFY_REWRITE_CSS_URLS' => true
 | 
			
		||||
            ,'MINIFY_USE_CACHE' => true
 | 
			
		||||
        );
 | 
			
		||||
        foreach ($defaults as $const => $val) {
 | 
			
		||||
            if (! defined($const)) {
 | 
			
		||||
                define($const, $val);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										245
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/HTML.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										245
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/HTML.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,245 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_HTML  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Compress HTML
 | 
			
		||||
 *
 | 
			
		||||
 * This is a heavy regex-based removal of whitespace, unnecessary comments and 
 | 
			
		||||
 * tokens. IE conditional comments are preserved. There are also options to have
 | 
			
		||||
 * STYLE and SCRIPT blocks compressed by callback functions. 
 | 
			
		||||
 * 
 | 
			
		||||
 * A test suite is available.
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 */
 | 
			
		||||
class Minify_HTML {
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * "Minify" an HTML page
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $html
 | 
			
		||||
     *
 | 
			
		||||
     * @param array $options
 | 
			
		||||
     *
 | 
			
		||||
     * 'cssMinifier' : (optional) callback function to process content of STYLE
 | 
			
		||||
     * elements.
 | 
			
		||||
     * 
 | 
			
		||||
     * 'jsMinifier' : (optional) callback function to process content of SCRIPT
 | 
			
		||||
     * elements. Note: the type attribute is ignored.
 | 
			
		||||
     * 
 | 
			
		||||
     * 'xhtml' : (optional boolean) should content be treated as XHTML1.0? If
 | 
			
		||||
     * unset, minify will sniff for an XHTML doctype.
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public static function minify($html, $options = array()) {
 | 
			
		||||
        $min = new Minify_HTML($html, $options);
 | 
			
		||||
        return $min->process();
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Create a minifier object
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $html
 | 
			
		||||
     *
 | 
			
		||||
     * @param array $options
 | 
			
		||||
     *
 | 
			
		||||
     * 'cssMinifier' : (optional) callback function to process content of STYLE
 | 
			
		||||
     * elements.
 | 
			
		||||
     * 
 | 
			
		||||
     * 'jsMinifier' : (optional) callback function to process content of SCRIPT
 | 
			
		||||
     * elements. Note: the type attribute is ignored.
 | 
			
		||||
     * 
 | 
			
		||||
     * 'xhtml' : (optional boolean) should content be treated as XHTML1.0? If
 | 
			
		||||
     * unset, minify will sniff for an XHTML doctype.
 | 
			
		||||
     * 
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    public function __construct($html, $options = array())
 | 
			
		||||
    {
 | 
			
		||||
        $this->_html = str_replace("\r\n", "\n", trim($html));
 | 
			
		||||
        if (isset($options['xhtml'])) {
 | 
			
		||||
            $this->_isXhtml = (bool)$options['xhtml'];
 | 
			
		||||
        }
 | 
			
		||||
        if (isset($options['cssMinifier'])) {
 | 
			
		||||
            $this->_cssMinifier = $options['cssMinifier'];
 | 
			
		||||
        }
 | 
			
		||||
        if (isset($options['jsMinifier'])) {
 | 
			
		||||
            $this->_jsMinifier = $options['jsMinifier'];
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Minify the markeup given in the constructor
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public function process()
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->_isXhtml === null) {
 | 
			
		||||
            $this->_isXhtml = (false !== strpos($this->_html, '<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML'));
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        $this->_replacementHash = 'MINIFYHTML' . md5($_SERVER['REQUEST_TIME']);
 | 
			
		||||
        $this->_placeholders = array();
 | 
			
		||||
        
 | 
			
		||||
        // replace SCRIPTs (and minify) with placeholders
 | 
			
		||||
        $this->_html = preg_replace_callback(
 | 
			
		||||
            '/(\\s*)(<script\\b[^>]*?>)([\\s\\S]*?)<\\/script>(\\s*)/i'
 | 
			
		||||
            ,array($this, '_removeScriptCB')
 | 
			
		||||
            ,$this->_html);
 | 
			
		||||
        
 | 
			
		||||
        // replace STYLEs (and minify) with placeholders
 | 
			
		||||
        $this->_html = preg_replace_callback(
 | 
			
		||||
            '/\\s*(<style\\b[^>]*?>)([\\s\\S]*?)<\\/style>\\s*/i'
 | 
			
		||||
            ,array($this, '_removeStyleCB')
 | 
			
		||||
            ,$this->_html);
 | 
			
		||||
        
 | 
			
		||||
        // remove HTML comments (not containing IE conditional comments).
 | 
			
		||||
        $this->_html = preg_replace_callback(
 | 
			
		||||
            '/<!--([\\s\\S]*?)-->/'
 | 
			
		||||
            ,array($this, '_commentCB')
 | 
			
		||||
            ,$this->_html);
 | 
			
		||||
        
 | 
			
		||||
        // replace PREs with placeholders
 | 
			
		||||
        $this->_html = preg_replace_callback('/\\s*(<pre\\b[^>]*?>[\\s\\S]*?<\\/pre>)\\s*/i'
 | 
			
		||||
            ,array($this, '_removePreCB')
 | 
			
		||||
            ,$this->_html);
 | 
			
		||||
        
 | 
			
		||||
        // replace TEXTAREAs with placeholders
 | 
			
		||||
        $this->_html = preg_replace_callback(
 | 
			
		||||
            '/\\s*(<textarea\\b[^>]*?>[\\s\\S]*?<\\/textarea>)\\s*/i'
 | 
			
		||||
            ,array($this, '_removeTextareaCB')
 | 
			
		||||
            ,$this->_html);
 | 
			
		||||
        
 | 
			
		||||
        // trim each line.
 | 
			
		||||
        // @todo take into account attribute values that span multiple lines.
 | 
			
		||||
        $this->_html = preg_replace('/^\\s+|\\s+$/m', '', $this->_html);
 | 
			
		||||
        
 | 
			
		||||
        // remove ws around block/undisplayed elements
 | 
			
		||||
        $this->_html = preg_replace('/\\s+(<\\/?(?:area|base(?:font)?|blockquote|body'
 | 
			
		||||
            .'|caption|center|cite|col(?:group)?|dd|dir|div|dl|dt|fieldset|form'
 | 
			
		||||
            .'|frame(?:set)?|h[1-6]|head|hr|html|legend|li|link|map|menu|meta'
 | 
			
		||||
            .'|ol|opt(?:group|ion)|p|param|t(?:able|body|head|d|h||r|foot|itle)'
 | 
			
		||||
            .'|ul)\\b[^>]*>)/i', '$1', $this->_html);
 | 
			
		||||
        
 | 
			
		||||
        // remove ws outside of all elements
 | 
			
		||||
        $this->_html = preg_replace_callback(
 | 
			
		||||
            '/>([^<]+)</'
 | 
			
		||||
            ,array($this, '_outsideTagCB')
 | 
			
		||||
            ,$this->_html);
 | 
			
		||||
        
 | 
			
		||||
        // use newlines before 1st attribute in open tags (to limit line lengths)
 | 
			
		||||
        $this->_html = preg_replace('/(<[a-z\\-]+)\\s+([^>]+>)/i', "$1\n$2", $this->_html);
 | 
			
		||||
        
 | 
			
		||||
        // fill placeholders
 | 
			
		||||
        $this->_html = str_replace(
 | 
			
		||||
            array_keys($this->_placeholders)
 | 
			
		||||
            ,array_values($this->_placeholders)
 | 
			
		||||
            ,$this->_html
 | 
			
		||||
        );
 | 
			
		||||
        return $this->_html;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    protected function _commentCB($m)
 | 
			
		||||
    {
 | 
			
		||||
        return (0 === strpos($m[1], '[') || false !== strpos($m[1], '<!['))
 | 
			
		||||
            ? $m[0]
 | 
			
		||||
            : '';
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    protected function _reservePlace($content)
 | 
			
		||||
    {
 | 
			
		||||
        $placeholder = '%' . $this->_replacementHash . count($this->_placeholders) . '%';
 | 
			
		||||
        $this->_placeholders[$placeholder] = $content;
 | 
			
		||||
        return $placeholder;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    protected $_isXhtml = null;
 | 
			
		||||
    protected $_replacementHash = null;
 | 
			
		||||
    protected $_placeholders = array();
 | 
			
		||||
    protected $_cssMinifier = null;
 | 
			
		||||
    protected $_jsMinifier = null;
 | 
			
		||||
 | 
			
		||||
    protected function _outsideTagCB($m)
 | 
			
		||||
    {
 | 
			
		||||
        return '>' . preg_replace('/^\\s+|\\s+$/', ' ', $m[1]) . '<';
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    protected function _removePreCB($m)
 | 
			
		||||
    {
 | 
			
		||||
        return $this->_reservePlace($m[1]);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    protected function _removeTextareaCB($m)
 | 
			
		||||
    {
 | 
			
		||||
        return $this->_reservePlace($m[1]);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    protected function _removeStyleCB($m)
 | 
			
		||||
    {
 | 
			
		||||
        $openStyle = $m[1];
 | 
			
		||||
        $css = $m[2];
 | 
			
		||||
        // remove HTML comments
 | 
			
		||||
        $css = preg_replace('/(?:^\\s*<!--|-->\\s*$)/', '', $css);
 | 
			
		||||
        
 | 
			
		||||
        // remove CDATA section markers
 | 
			
		||||
        $css = $this->_removeCdata($css);
 | 
			
		||||
        
 | 
			
		||||
        // minify
 | 
			
		||||
        $minifier = $this->_cssMinifier
 | 
			
		||||
            ? $this->_cssMinifier
 | 
			
		||||
            : 'trim';
 | 
			
		||||
        $css = call_user_func($minifier, $css);
 | 
			
		||||
        
 | 
			
		||||
        return $this->_reservePlace($this->_needsCdata($css)
 | 
			
		||||
            ? "{$openStyle}/*<![CDATA[*/{$css}/*]]>*/</style>"
 | 
			
		||||
            : "{$openStyle}{$css}</style>"
 | 
			
		||||
        );
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    protected function _removeScriptCB($m)
 | 
			
		||||
    {
 | 
			
		||||
        $openScript = $m[2];
 | 
			
		||||
        $js = $m[3];
 | 
			
		||||
        
 | 
			
		||||
        // whitespace surrounding? preserve at least one space
 | 
			
		||||
        $ws1 = ($m[1] === '') ? '' : ' ';
 | 
			
		||||
        $ws2 = ($m[4] === '') ? '' : ' ';
 | 
			
		||||
 
 | 
			
		||||
        // remove HTML comments (and ending "//" if present)
 | 
			
		||||
        $js = preg_replace('/(?:^\\s*<!--\\s*|\\s*(?:\\/\\/)?\\s*-->\\s*$)/', '', $js);
 | 
			
		||||
            
 | 
			
		||||
        // remove CDATA section markers
 | 
			
		||||
        $js = $this->_removeCdata($js);
 | 
			
		||||
        
 | 
			
		||||
        // minify
 | 
			
		||||
        $minifier = $this->_jsMinifier
 | 
			
		||||
            ? $this->_jsMinifier
 | 
			
		||||
            : 'trim'; 
 | 
			
		||||
        $js = call_user_func($minifier, $js);
 | 
			
		||||
        
 | 
			
		||||
        return $this->_reservePlace($this->_needsCdata($js)
 | 
			
		||||
            ? "{$ws1}{$openScript}/*<![CDATA[*/{$js}/*]]>*/</script>{$ws2}"
 | 
			
		||||
            : "{$ws1}{$openScript}{$js}</script>{$ws2}"
 | 
			
		||||
        );
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    protected function _removeCdata($str)
 | 
			
		||||
    {
 | 
			
		||||
        return (false !== strpos($str, '<![CDATA['))
 | 
			
		||||
            ? str_replace(array('<![CDATA[', ']]>'), '', $str)
 | 
			
		||||
            : $str;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    protected function _needsCdata($str)
 | 
			
		||||
    {
 | 
			
		||||
        return ($this->_isXhtml && preg_match('/(?:[<&]|\\-\\-|\\]\\]>)/', $str));
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										157
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/ImportProcessor.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										157
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/ImportProcessor.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,157 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_ImportProcessor  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Linearize a CSS/JS file by including content specified by CSS import
 | 
			
		||||
 * declarations. In CSS files, relative URIs are fixed.
 | 
			
		||||
 * 
 | 
			
		||||
 * @imports will be processed regardless of where they appear in the source 
 | 
			
		||||
 * files; i.e. @imports commented out or in string content will still be
 | 
			
		||||
 * processed!
 | 
			
		||||
 * 
 | 
			
		||||
 * This has a unit test but should be considered "experimental".
 | 
			
		||||
 *
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 */
 | 
			
		||||
class Minify_ImportProcessor {
 | 
			
		||||
    
 | 
			
		||||
    public static $filesIncluded = array();
 | 
			
		||||
    
 | 
			
		||||
    public static function process($file)
 | 
			
		||||
    {
 | 
			
		||||
        self::$filesIncluded = array();
 | 
			
		||||
        self::$_isCss = (strtolower(substr($file, -4)) === '.css');
 | 
			
		||||
        $obj = new Minify_ImportProcessor(dirname($file));
 | 
			
		||||
        return $obj->_getContent($file);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    // allows callback funcs to know the current directory
 | 
			
		||||
    private $_currentDir = null;
 | 
			
		||||
    
 | 
			
		||||
    // allows _importCB to write the fetched content back to the obj
 | 
			
		||||
    private $_importedContent = '';
 | 
			
		||||
    
 | 
			
		||||
    private static $_isCss = null;
 | 
			
		||||
    
 | 
			
		||||
    private function __construct($currentDir)
 | 
			
		||||
    {
 | 
			
		||||
        $this->_currentDir = $currentDir;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    private function _getContent($file)
 | 
			
		||||
    {
 | 
			
		||||
        $file = realpath($file);
 | 
			
		||||
        if (! $file
 | 
			
		||||
            || in_array($file, self::$filesIncluded)
 | 
			
		||||
            || false === ($content = @file_get_contents($file))
 | 
			
		||||
        ) {
 | 
			
		||||
            // file missing, already included, or failed read
 | 
			
		||||
            return '';
 | 
			
		||||
        }
 | 
			
		||||
        self::$filesIncluded[] = realpath($file);
 | 
			
		||||
        $this->_currentDir = dirname($file);
 | 
			
		||||
        
 | 
			
		||||
        // remove UTF-8 BOM if present
 | 
			
		||||
        if (pack("CCC",0xef,0xbb,0xbf) === substr($content, 0, 3)) {
 | 
			
		||||
            $content = substr($content, 3);
 | 
			
		||||
        }
 | 
			
		||||
        // ensure uniform EOLs
 | 
			
		||||
        $content = str_replace("\r\n", "\n", $content);
 | 
			
		||||
        
 | 
			
		||||
        // process @imports
 | 
			
		||||
        $content = preg_replace_callback(
 | 
			
		||||
            '/
 | 
			
		||||
                @import\\s+
 | 
			
		||||
                (?:url\\(\\s*)?      # maybe url(
 | 
			
		||||
                [\'"]?               # maybe quote
 | 
			
		||||
                (.*?)                # 1 = URI
 | 
			
		||||
                [\'"]?               # maybe end quote
 | 
			
		||||
                (?:\\s*\\))?         # maybe )
 | 
			
		||||
                ([a-zA-Z,\\s]*)?     # 2 = media list
 | 
			
		||||
                ;                    # end token
 | 
			
		||||
            /x'
 | 
			
		||||
            ,array($this, '_importCB')
 | 
			
		||||
            ,$content
 | 
			
		||||
        );
 | 
			
		||||
        
 | 
			
		||||
        if (self::$_isCss) {
 | 
			
		||||
            // rewrite remaining relative URIs
 | 
			
		||||
            $content = preg_replace_callback(
 | 
			
		||||
                '/url\\(\\s*([^\\)\\s]+)\\s*\\)/'
 | 
			
		||||
                ,array($this, '_urlCB')
 | 
			
		||||
                ,$content
 | 
			
		||||
            );
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        return $this->_importedContent . $content;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    private function _importCB($m)
 | 
			
		||||
    {
 | 
			
		||||
        $url = $m[1];
 | 
			
		||||
        $mediaList = preg_replace('/\\s+/', '', $m[2]);
 | 
			
		||||
        
 | 
			
		||||
        if (strpos($url, '://') > 0) {
 | 
			
		||||
            // protocol, leave in place for CSS, comment for JS
 | 
			
		||||
            return self::$_isCss
 | 
			
		||||
                ? $m[0]
 | 
			
		||||
                : "/* Minify_ImportProcessor will not include remote content */";
 | 
			
		||||
        }
 | 
			
		||||
        if ('/' === $url[0]) {
 | 
			
		||||
            // protocol-relative or root path
 | 
			
		||||
            $url = ltrim($url, '/');
 | 
			
		||||
            $file = realpath($_SERVER['DOCUMENT_ROOT']) . DIRECTORY_SEPARATOR
 | 
			
		||||
                . strtr($url, '/', DIRECTORY_SEPARATOR);
 | 
			
		||||
        } else {
 | 
			
		||||
            // relative to current path
 | 
			
		||||
            $file = $this->_currentDir . DIRECTORY_SEPARATOR 
 | 
			
		||||
                . strtr($url, '/', DIRECTORY_SEPARATOR);
 | 
			
		||||
        }
 | 
			
		||||
        $obj = new Minify_ImportProcessor(dirname($file));
 | 
			
		||||
        $content = $obj->_getContent($file);
 | 
			
		||||
        if ('' === $content) {
 | 
			
		||||
            // failed. leave in place for CSS, comment for JS
 | 
			
		||||
            return self::$_isCss
 | 
			
		||||
                ? $m[0]
 | 
			
		||||
                : "/* Minify_ImportProcessor could not fetch '{$file}' */";;
 | 
			
		||||
        }
 | 
			
		||||
        return (!self::$_isCss || preg_match('@(?:^$|\\ball\\b)@', $mediaList))
 | 
			
		||||
            ? $content
 | 
			
		||||
            : "@media {$mediaList} {\n{$content}\n}\n";
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    private function _urlCB($m)
 | 
			
		||||
    {
 | 
			
		||||
        // $m[1] is either quoted or not
 | 
			
		||||
        $quote = ($m[1][0] === "'" || $m[1][0] === '"')
 | 
			
		||||
            ? $m[1][0]
 | 
			
		||||
            : '';
 | 
			
		||||
        $url = ($quote === '')
 | 
			
		||||
            ? $m[1]
 | 
			
		||||
            : substr($m[1], 1, strlen($m[1]) - 2);
 | 
			
		||||
        if ('/' !== $url[0]) {
 | 
			
		||||
            if (strpos($url, '//') > 0) {
 | 
			
		||||
                // probably starts with protocol, do not alter
 | 
			
		||||
            } else {
 | 
			
		||||
                // prepend path with current dir separator (OS-independent)
 | 
			
		||||
                $path = $this->_currentDir 
 | 
			
		||||
                    . DIRECTORY_SEPARATOR . strtr($url, '/', DIRECTORY_SEPARATOR);
 | 
			
		||||
                // strip doc root
 | 
			
		||||
                $path = substr($path, strlen(realpath($_SERVER['DOCUMENT_ROOT'])));
 | 
			
		||||
                // fix to absolute URL
 | 
			
		||||
                $url = strtr($path, '/\\', '//');
 | 
			
		||||
                // remove /./ and /../ where possible
 | 
			
		||||
                $url = str_replace('/./', '/', $url);
 | 
			
		||||
                // inspired by patch from Oleg Cherniy
 | 
			
		||||
                do {
 | 
			
		||||
                    $url = preg_replace('@/[^/]+/\\.\\./@', '/', $url, 1, $changed);
 | 
			
		||||
                } while ($changed);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        return "url({$quote}{$url}{$quote})";
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										131
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Lines.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										131
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Lines.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,131 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_Lines  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Add line numbers in C-style comments for easier debugging of combined content
 | 
			
		||||
 *
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 * @author Adam Pedersen (Issue 55 fix)
 | 
			
		||||
 */
 | 
			
		||||
class Minify_Lines {
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Add line numbers in C-style comments
 | 
			
		||||
     *
 | 
			
		||||
     * This uses a very basic parser easily fooled by comment tokens inside
 | 
			
		||||
     * strings or regexes, but, otherwise, generally clean code will not be 
 | 
			
		||||
     * mangled. URI rewriting can also be performed.
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $content
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $options available options:
 | 
			
		||||
     * 
 | 
			
		||||
     * 'id': (optional) string to identify file. E.g. file name/path
 | 
			
		||||
     *
 | 
			
		||||
     * 'currentDir': (default null) if given, this is assumed to be the
 | 
			
		||||
     * directory of the current CSS file. Using this, minify will rewrite
 | 
			
		||||
     * all relative URIs in import/url declarations to correctly point to
 | 
			
		||||
     * the desired files, and prepend a comment with debugging information about
 | 
			
		||||
     * this process.
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string 
 | 
			
		||||
     */
 | 
			
		||||
    public static function minify($content, $options = array()) 
 | 
			
		||||
    {
 | 
			
		||||
        $id = (isset($options['id']) && $options['id'])
 | 
			
		||||
            ? $options['id']
 | 
			
		||||
            : '';
 | 
			
		||||
        $content = str_replace("\r\n", "\n", $content);
 | 
			
		||||
        $lines = explode("\n", $content);
 | 
			
		||||
        $numLines = count($lines);
 | 
			
		||||
        // determine left padding
 | 
			
		||||
        $padTo = strlen($numLines);
 | 
			
		||||
        $inComment = false;
 | 
			
		||||
        $i = 0;
 | 
			
		||||
        $newLines = array();
 | 
			
		||||
        while (null !== ($line = array_shift($lines))) {
 | 
			
		||||
            if (('' !== $id) && (0 == $i % 50)) {
 | 
			
		||||
                array_push($newLines, '', "/* {$id} */", '');
 | 
			
		||||
            }
 | 
			
		||||
            ++$i;
 | 
			
		||||
            $newLines[] = self::_addNote($line, $i, $inComment, $padTo);
 | 
			
		||||
            $inComment = self::_eolInComment($line, $inComment);
 | 
			
		||||
        }
 | 
			
		||||
        $content = implode("\n", $newLines) . "\n";
 | 
			
		||||
        
 | 
			
		||||
        // check for desired URI rewriting
 | 
			
		||||
        if (isset($options['currentDir'])) {
 | 
			
		||||
            require_once 'Minify/CSS/UriRewriter.php';
 | 
			
		||||
            Minify_CSS_UriRewriter::$debugText = '';
 | 
			
		||||
            $content = Minify_CSS_UriRewriter::rewrite(
 | 
			
		||||
                 $content
 | 
			
		||||
                ,$options['currentDir']
 | 
			
		||||
                ,isset($options['docRoot']) ? $options['docRoot'] : $_SERVER['DOCUMENT_ROOT']
 | 
			
		||||
                ,isset($options['symlinks']) ? $options['symlinks'] : array()
 | 
			
		||||
            );
 | 
			
		||||
            $content = "/* Minify_CSS_UriRewriter::\$debugText\n\n" 
 | 
			
		||||
                     . Minify_CSS_UriRewriter::$debugText . "*/\n"
 | 
			
		||||
                     . $content;
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        return $content;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Is the parser within a C-style comment at the end of this line?
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $line current line of code
 | 
			
		||||
     * 
 | 
			
		||||
     * @param bool $inComment was the parser in a comment at the
 | 
			
		||||
     * beginning of the line?
 | 
			
		||||
     * 
 | 
			
		||||
     * @return bool
 | 
			
		||||
     */
 | 
			
		||||
    private static function _eolInComment($line, $inComment)
 | 
			
		||||
    {
 | 
			
		||||
        while (strlen($line)) {
 | 
			
		||||
            $search = $inComment
 | 
			
		||||
                ? '*/'
 | 
			
		||||
                : '/*';
 | 
			
		||||
            $pos = strpos($line, $search);
 | 
			
		||||
            if (false === $pos) {
 | 
			
		||||
                return $inComment;
 | 
			
		||||
            } else {
 | 
			
		||||
                if ($pos == 0
 | 
			
		||||
                    || ($inComment
 | 
			
		||||
                        ? substr($line, $pos, 3)
 | 
			
		||||
                        : substr($line, $pos-1, 3)) != '*/*')
 | 
			
		||||
                {
 | 
			
		||||
                        $inComment = ! $inComment;
 | 
			
		||||
                }
 | 
			
		||||
                $line = substr($line, $pos + 2);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        return $inComment;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Prepend a comment (or note) to the given line
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $line current line of code
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $note content of note/comment
 | 
			
		||||
     * 
 | 
			
		||||
     * @param bool $inComment was the parser in a comment at the
 | 
			
		||||
     * beginning of the line?
 | 
			
		||||
     *
 | 
			
		||||
     * @param int $padTo minimum width of comment
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    private static function _addNote($line, $note, $inComment, $padTo)
 | 
			
		||||
    {
 | 
			
		||||
        return $inComment
 | 
			
		||||
            ? '/* ' . str_pad($note, $padTo, ' ', STR_PAD_RIGHT) . ' *| ' . $line
 | 
			
		||||
            : '/* ' . str_pad($note, $padTo, ' ', STR_PAD_RIGHT) . ' */ ' . $line;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										45
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Logger.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Logger.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,45 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_Logger  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/** 
 | 
			
		||||
 * Message logging class
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 */
 | 
			
		||||
class Minify_Logger {
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Set logger object. 
 | 
			
		||||
     *
 | 
			
		||||
     * The object should have a method "log" that accepts a value as 1st argument and
 | 
			
		||||
     * an optional string label as the 2nd.
 | 
			
		||||
     *
 | 
			
		||||
     * @param mixed $obj or a "falsey" value to disable
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    public static function setLogger($obj = null) {
 | 
			
		||||
        self::$_logger = $obj
 | 
			
		||||
            ? $obj
 | 
			
		||||
            : null;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Pass a message to the logger (if set)
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $msg message to log
 | 
			
		||||
     * @return null
 | 
			
		||||
     */
 | 
			
		||||
    public static function log($msg, $label = 'Minify') {
 | 
			
		||||
        if (! self::$_logger) return;
 | 
			
		||||
        self::$_logger->log($msg, $label);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * @var mixed logger object (like FirePHP) or null (i.e. no logger available)
 | 
			
		||||
     */
 | 
			
		||||
    private static $_logger = null;
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										37
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Packer.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Packer.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,37 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_Packer
 | 
			
		||||
 *
 | 
			
		||||
 * To use this class you must first download the PHP port of Packer
 | 
			
		||||
 * and place the file "class.JavaScriptPacker.php" in /lib (or your
 | 
			
		||||
 * include_path). 
 | 
			
		||||
 * @link http://joliclic.free.fr/php/javascript-packer/en/
 | 
			
		||||
 *
 | 
			
		||||
 * Be aware that, as long as HTTP encoding is used, scripts minified with JSMin
 | 
			
		||||
 * will provide better client-side performance, as they need not be unpacked in
 | 
			
		||||
 * client-side code.
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Minify  
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
if (false === (@include 'class.JavaScriptPacker.php')) {
 | 
			
		||||
    trigger_error(
 | 
			
		||||
        'The script "class.JavaScriptPacker.php" is required. Please see: http:'
 | 
			
		||||
        .'//code.google.com/p/minify/source/browse/trunk/min/lib/Minify/Packer.php'
 | 
			
		||||
        ,E_USER_ERROR
 | 
			
		||||
    );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Minify Javascript using Dean Edward's Packer
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
class Minify_Packer {
 | 
			
		||||
    public static function minify($code, $options = array())
 | 
			
		||||
    {
 | 
			
		||||
        // @todo: set encoding options based on $options :)
 | 
			
		||||
        $packer = new JavascriptPacker($code, 'Normal', true, false);
 | 
			
		||||
        return trim($packer->pack());
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										187
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Source.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										187
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/Source.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,187 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_Source  
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/** 
 | 
			
		||||
 * A content source to be minified by Minify. 
 | 
			
		||||
 * 
 | 
			
		||||
 * This allows per-source minification options and the mixing of files with
 | 
			
		||||
 * content from other sources.
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 */
 | 
			
		||||
class Minify_Source {
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * @var int time of last modification
 | 
			
		||||
     */
 | 
			
		||||
    public $lastModified = null;
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * @var callback minifier function specifically for this source.
 | 
			
		||||
     */
 | 
			
		||||
    public $minifier = null;
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * @var array minification options specific to this source.
 | 
			
		||||
     */
 | 
			
		||||
    public $minifyOptions = null;
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * @var string full path of file
 | 
			
		||||
     */
 | 
			
		||||
    public $filepath = null;
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * @var string HTTP Content Type (Minify requires one of the constants Minify::TYPE_*)
 | 
			
		||||
     */
 | 
			
		||||
    public $contentType = null;
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Create a Minify_Source
 | 
			
		||||
     * 
 | 
			
		||||
     * In the $spec array(), you can either provide a 'filepath' to an existing
 | 
			
		||||
     * file (existence will not be checked!) or give 'id' (unique string for 
 | 
			
		||||
     * the content), 'content' (the string content) and 'lastModified' 
 | 
			
		||||
     * (unixtime of last update).
 | 
			
		||||
     * 
 | 
			
		||||
     * As a shortcut, the controller will replace "//" at the beginning
 | 
			
		||||
     * of a filepath with $_SERVER['DOCUMENT_ROOT'] . '/'.
 | 
			
		||||
     *
 | 
			
		||||
     * @param array $spec options
 | 
			
		||||
     */
 | 
			
		||||
    public function __construct($spec)
 | 
			
		||||
    {
 | 
			
		||||
        if (isset($spec['filepath'])) {
 | 
			
		||||
            if (0 === strpos($spec['filepath'], '//')) {
 | 
			
		||||
                $spec['filepath'] = $_SERVER['DOCUMENT_ROOT'] . substr($spec['filepath'], 1);
 | 
			
		||||
            }
 | 
			
		||||
            $segments = explode('.', $spec['filepath']);
 | 
			
		||||
            $ext = strtolower(array_pop($segments));
 | 
			
		||||
            switch ($ext) {
 | 
			
		||||
            case 'js'   : $this->contentType = 'application/x-javascript';
 | 
			
		||||
                          break;
 | 
			
		||||
            case 'css'  : $this->contentType = 'text/css';
 | 
			
		||||
                          break;
 | 
			
		||||
            case 'htm'  : // fallthrough
 | 
			
		||||
            case 'html' : $this->contentType = 'text/html';
 | 
			
		||||
                          break;
 | 
			
		||||
            }
 | 
			
		||||
            $this->filepath = $spec['filepath'];
 | 
			
		||||
            $this->_id = $spec['filepath'];
 | 
			
		||||
            $this->lastModified = filemtime($spec['filepath'])
 | 
			
		||||
                // offset for Windows uploaders with out of sync clocks
 | 
			
		||||
                + round(Minify::$uploaderHoursBehind * 3600);
 | 
			
		||||
        } elseif (isset($spec['id'])) {
 | 
			
		||||
            $this->_id = 'id::' . $spec['id'];
 | 
			
		||||
            if (isset($spec['content'])) {
 | 
			
		||||
                $this->_content = $spec['content'];
 | 
			
		||||
            } else {
 | 
			
		||||
                $this->_getContentFunc = $spec['getContentFunc'];
 | 
			
		||||
            }
 | 
			
		||||
            $this->lastModified = isset($spec['lastModified'])
 | 
			
		||||
                ? $spec['lastModified']
 | 
			
		||||
                : time();
 | 
			
		||||
        }
 | 
			
		||||
        if (isset($spec['contentType'])) {
 | 
			
		||||
            $this->contentType = $spec['contentType'];
 | 
			
		||||
        }
 | 
			
		||||
        if (isset($spec['minifier'])) {
 | 
			
		||||
            $this->minifier = $spec['minifier'];
 | 
			
		||||
        }
 | 
			
		||||
        if (isset($spec['minifyOptions'])) {
 | 
			
		||||
            $this->minifyOptions = $spec['minifyOptions'];
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Get content
 | 
			
		||||
     *
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public function getContent()
 | 
			
		||||
    {
 | 
			
		||||
        $content = (null !== $this->filepath)
 | 
			
		||||
            ? file_get_contents($this->filepath)
 | 
			
		||||
            : ((null !== $this->_content)
 | 
			
		||||
                ? $this->_content
 | 
			
		||||
                : call_user_func($this->_getContentFunc, $this->_id)
 | 
			
		||||
            );
 | 
			
		||||
        // remove UTF-8 BOM if present
 | 
			
		||||
        return (pack("CCC",0xef,0xbb,0xbf) === substr($content, 0, 3))
 | 
			
		||||
            ? substr($content, 3)
 | 
			
		||||
            : $content;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Get id
 | 
			
		||||
     *
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public function getId()
 | 
			
		||||
    {
 | 
			
		||||
        return $this->_id;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Verifies a single minification call can handle all sources
 | 
			
		||||
     *
 | 
			
		||||
     * @param array $sources Minify_Source instances
 | 
			
		||||
     * 
 | 
			
		||||
     * @return bool true iff there no sources with specific minifier preferences.
 | 
			
		||||
     */
 | 
			
		||||
    public static function haveNoMinifyPrefs($sources)
 | 
			
		||||
    {
 | 
			
		||||
        foreach ($sources as $source) {
 | 
			
		||||
            if (null !== $source->minifier
 | 
			
		||||
                || null !== $source->minifyOptions) {
 | 
			
		||||
                return false;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        return true;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Get unique string for a set of sources
 | 
			
		||||
     *
 | 
			
		||||
     * @param array $sources Minify_Source instances
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public static function getDigest($sources)
 | 
			
		||||
    {
 | 
			
		||||
        foreach ($sources as $source) {
 | 
			
		||||
            $info[] = array(
 | 
			
		||||
                $source->_id, $source->minifier, $source->minifyOptions
 | 
			
		||||
            );
 | 
			
		||||
        }
 | 
			
		||||
        return md5(serialize($info));
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Get content type from a group of sources
 | 
			
		||||
     * 
 | 
			
		||||
     * This is called if the user doesn't pass in a 'contentType' options  
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $sources Minify_Source instances
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string content type. e.g. 'text/css'
 | 
			
		||||
     */
 | 
			
		||||
    public static function getContentType($sources)
 | 
			
		||||
    {
 | 
			
		||||
        foreach ($sources as $source) {
 | 
			
		||||
            if ($source->contentType !== null) {
 | 
			
		||||
                return $source->contentType;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        return 'text/plain';
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    protected $_content = null;
 | 
			
		||||
    protected $_getContentFunc = null;
 | 
			
		||||
    protected $_id = null;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										139
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/YUICompressor.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										139
									
								
								plugins/Minify/extlib/minify/min/lib/Minify/YUICompressor.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,139 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * Class Minify_YUICompressor 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Compress Javascript/CSS using the YUI Compressor
 | 
			
		||||
 * 
 | 
			
		||||
 * You must set $jarFile and $tempDir before calling the minify functions.
 | 
			
		||||
 * Also, depending on your shell's environment, you may need to specify
 | 
			
		||||
 * the full path to java in $javaExecutable or use putenv() to setup the
 | 
			
		||||
 * Java environment.
 | 
			
		||||
 * 
 | 
			
		||||
 * <code>
 | 
			
		||||
 * Minify_YUICompressor::$jarFile = '/path/to/yuicompressor-2.3.5.jar';
 | 
			
		||||
 * Minify_YUICompressor::$tempDir = '/tmp';
 | 
			
		||||
 * $code = Minify_YUICompressor::minifyJs(
 | 
			
		||||
 *   $code
 | 
			
		||||
 *   ,array('nomunge' => true, 'line-break' => 1000)
 | 
			
		||||
 * );
 | 
			
		||||
 * </code>
 | 
			
		||||
 * 
 | 
			
		||||
 * @todo unit tests, $options docs
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Minify
 | 
			
		||||
 * @author Stephen Clay <steve@mrclay.org>
 | 
			
		||||
 */
 | 
			
		||||
class Minify_YUICompressor {
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Filepath of the YUI Compressor jar file. This must be set before
 | 
			
		||||
     * calling minifyJs() or minifyCss().
 | 
			
		||||
     *
 | 
			
		||||
     * @var string
 | 
			
		||||
     */
 | 
			
		||||
    public static $jarFile = null;
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Writable temp directory. This must be set before calling minifyJs()
 | 
			
		||||
     * or minifyCss().
 | 
			
		||||
     *
 | 
			
		||||
     * @var string
 | 
			
		||||
     */
 | 
			
		||||
    public static $tempDir = null;
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Filepath of "java" executable (may be needed if not in shell's PATH)
 | 
			
		||||
     *
 | 
			
		||||
     * @var string
 | 
			
		||||
     */
 | 
			
		||||
    public static $javaExecutable = 'java';
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Minify a Javascript string
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $js
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $options (verbose is ignored)
 | 
			
		||||
     * 
 | 
			
		||||
     * @see http://www.julienlecomte.net/yuicompressor/README
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string 
 | 
			
		||||
     */
 | 
			
		||||
    public static function minifyJs($js, $options = array())
 | 
			
		||||
    {
 | 
			
		||||
        return self::_minify('js', $js, $options);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * Minify a CSS string
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $css
 | 
			
		||||
     * 
 | 
			
		||||
     * @param array $options (verbose is ignored)
 | 
			
		||||
     * 
 | 
			
		||||
     * @see http://www.julienlecomte.net/yuicompressor/README
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string 
 | 
			
		||||
     */
 | 
			
		||||
    public static function minifyCss($css, $options = array())
 | 
			
		||||
    {
 | 
			
		||||
        return self::_minify('css', $css, $options);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    private static function _minify($type, $content, $options)
 | 
			
		||||
    {
 | 
			
		||||
        self::_prepare();
 | 
			
		||||
        if (! ($tmpFile = tempnam(self::$tempDir, 'yuic_'))) {
 | 
			
		||||
            throw new Exception('Minify_YUICompressor : could not create temp file.');
 | 
			
		||||
        }
 | 
			
		||||
        file_put_contents($tmpFile, $content);
 | 
			
		||||
        exec(self::_getCmd($options, $type, $tmpFile), $output);
 | 
			
		||||
        unlink($tmpFile);
 | 
			
		||||
        return implode("\n", $output);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    private static function _getCmd($userOptions, $type, $tmpFile)
 | 
			
		||||
    {
 | 
			
		||||
        $o = array_merge(
 | 
			
		||||
            array(
 | 
			
		||||
                'charset' => ''
 | 
			
		||||
                ,'line-break' => 5000
 | 
			
		||||
                ,'type' => $type
 | 
			
		||||
                ,'nomunge' => false
 | 
			
		||||
                ,'preserve-semi' => false
 | 
			
		||||
                ,'disable-optimizations' => false
 | 
			
		||||
            )
 | 
			
		||||
            ,$userOptions
 | 
			
		||||
        );
 | 
			
		||||
        $cmd = self::$javaExecutable . ' -jar ' . escapeshellarg(self::$jarFile)
 | 
			
		||||
             . " --type {$type}"
 | 
			
		||||
             . (preg_match('/^[a-zA-Z\\-]+$/', $o['charset']) 
 | 
			
		||||
                ? " --charset {$o['charset']}" 
 | 
			
		||||
                : '')
 | 
			
		||||
             . (is_numeric($o['line-break']) && $o['line-break'] >= 0
 | 
			
		||||
                ? ' --line-break ' . (int)$o['line-break']
 | 
			
		||||
                : '');
 | 
			
		||||
        if ($type === 'js') {
 | 
			
		||||
            foreach (array('nomunge', 'preserve-semi', 'disable-optimizations') as $opt) {
 | 
			
		||||
                $cmd .= $o[$opt] 
 | 
			
		||||
                    ? " --{$opt}"
 | 
			
		||||
                    : '';
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        return $cmd . ' ' . escapeshellarg($tmpFile);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    private static function _prepare()
 | 
			
		||||
    {
 | 
			
		||||
        if (! is_file(self::$jarFile) 
 | 
			
		||||
            || ! is_dir(self::$tempDir)
 | 
			
		||||
            || ! is_writable(self::$tempDir)
 | 
			
		||||
        ) {
 | 
			
		||||
            throw new Exception('Minify_YUICompressor : $jarFile and $tempDir must be set.');
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										199
									
								
								plugins/Minify/extlib/minify/min/lib/Solar/Dir.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										199
									
								
								plugins/Minify/extlib/minify/min/lib/Solar/Dir.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,199 @@
 | 
			
		||||
<?php
 | 
			
		||||
/**
 | 
			
		||||
 * 
 | 
			
		||||
 * Utility class for static directory methods.
 | 
			
		||||
 * 
 | 
			
		||||
 * @category Solar
 | 
			
		||||
 * 
 | 
			
		||||
 * @package Solar
 | 
			
		||||
 * 
 | 
			
		||||
 * @author Paul M. Jones <pmjones@solarphp.com>
 | 
			
		||||
 * 
 | 
			
		||||
 * @license http://opensource.org/licenses/bsd-license.php BSD
 | 
			
		||||
 * 
 | 
			
		||||
 * @version $Id: Dir.php 2926 2007-11-09 16:25:44Z pmjones $
 | 
			
		||||
 * 
 | 
			
		||||
 */
 | 
			
		||||
class Solar_Dir {
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * 
 | 
			
		||||
     * The OS-specific temporary directory location.
 | 
			
		||||
     * 
 | 
			
		||||
     * @var string
 | 
			
		||||
     * 
 | 
			
		||||
     */
 | 
			
		||||
    protected static $_tmp;
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * 
 | 
			
		||||
     * Hack for [[php::is_dir() | ]] that checks the include_path.
 | 
			
		||||
     * 
 | 
			
		||||
     * Use this to see if a directory exists anywhere in the include_path.
 | 
			
		||||
     * 
 | 
			
		||||
     * {{code: php
 | 
			
		||||
     *     $dir = Solar_Dir::exists('path/to/dir')
 | 
			
		||||
     *     if ($dir) {
 | 
			
		||||
     *         $files = scandir($dir);
 | 
			
		||||
     *     } else {
 | 
			
		||||
     *         echo "Not found in the include-path.";
 | 
			
		||||
     *     }
 | 
			
		||||
     * }}
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $dir Check for this directory in the include_path.
 | 
			
		||||
     * 
 | 
			
		||||
     * @return mixed If the directory exists in the include_path, returns the
 | 
			
		||||
     * absolute path; if not, returns boolean false.
 | 
			
		||||
     * 
 | 
			
		||||
     */
 | 
			
		||||
    public static function exists($dir)
 | 
			
		||||
    {
 | 
			
		||||
        // no file requested?
 | 
			
		||||
        $dir = trim($dir);
 | 
			
		||||
        if (! $dir) {
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        // using an absolute path for the file?
 | 
			
		||||
        // dual check for Unix '/' and Windows '\',
 | 
			
		||||
        // or Windows drive letter and a ':'.
 | 
			
		||||
        $abs = ($dir[0] == '/' || $dir[0] == '\\' || $dir[1] == ':');
 | 
			
		||||
        if ($abs && is_dir($dir)) {
 | 
			
		||||
            return $dir;
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        // using a relative path on the file
 | 
			
		||||
        $path = explode(PATH_SEPARATOR, ini_get('include_path'));
 | 
			
		||||
        foreach ($path as $base) {
 | 
			
		||||
            // strip Unix '/' and Windows '\'
 | 
			
		||||
            $target = rtrim($base, '\\/') . DIRECTORY_SEPARATOR . $dir;
 | 
			
		||||
            if (is_dir($target)) {
 | 
			
		||||
                return $target;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        // never found it
 | 
			
		||||
        return false;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * 
 | 
			
		||||
     * "Fixes" a directory string for the operating system.
 | 
			
		||||
     * 
 | 
			
		||||
     * Use slashes anywhere you need a directory separator. Then run the
 | 
			
		||||
     * string through fixdir() and the slashes will be converted to the
 | 
			
		||||
     * proper separator (for example '\' on Windows).
 | 
			
		||||
     * 
 | 
			
		||||
     * Always adds a final trailing separator.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $dir The directory string to 'fix'.
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string The "fixed" directory string.
 | 
			
		||||
     * 
 | 
			
		||||
     */
 | 
			
		||||
    public static function fix($dir)
 | 
			
		||||
    {
 | 
			
		||||
        $dir = str_replace('/', DIRECTORY_SEPARATOR, $dir);
 | 
			
		||||
        return rtrim($dir, DIRECTORY_SEPARATOR) . DIRECTORY_SEPARATOR;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * 
 | 
			
		||||
     * Convenience method for dirname() and higher-level directories.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $file Get the dirname() of this file.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param int $up Move up in the directory structure this many 
 | 
			
		||||
     * times, default 0.
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string The dirname() of the file.
 | 
			
		||||
     * 
 | 
			
		||||
     */
 | 
			
		||||
    public static function name($file, $up = 0)
 | 
			
		||||
    {
 | 
			
		||||
        $dir = dirname($file);
 | 
			
		||||
        while ($up --) {
 | 
			
		||||
            $dir = dirname($dir);
 | 
			
		||||
        }
 | 
			
		||||
        return $dir;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * 
 | 
			
		||||
     * Returns the OS-specific directory for temporary files.
 | 
			
		||||
     * 
 | 
			
		||||
     * @param string $sub Add this subdirectory to the returned temporary
 | 
			
		||||
     * directory name.
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string The temporary directory path.
 | 
			
		||||
     * 
 | 
			
		||||
     */
 | 
			
		||||
    public static function tmp($sub = '')
 | 
			
		||||
    {
 | 
			
		||||
        // find the tmp dir if needed
 | 
			
		||||
        if (! Solar_Dir::$_tmp) {
 | 
			
		||||
            
 | 
			
		||||
            // use the system if we can
 | 
			
		||||
            if (function_exists('sys_get_temp_dir')) {
 | 
			
		||||
                $tmp = sys_get_temp_dir();
 | 
			
		||||
            } else {
 | 
			
		||||
                $tmp = Solar_Dir::_tmp();
 | 
			
		||||
            }
 | 
			
		||||
            
 | 
			
		||||
            // remove trailing separator and save
 | 
			
		||||
            Solar_Dir::$_tmp = rtrim($tmp, DIRECTORY_SEPARATOR);
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        // do we have a subdirectory request?
 | 
			
		||||
        $sub = trim($sub);
 | 
			
		||||
        if ($sub) {
 | 
			
		||||
            // remove leading and trailing separators, and force exactly
 | 
			
		||||
            // one trailing separator
 | 
			
		||||
            $sub = trim($sub, DIRECTORY_SEPARATOR)
 | 
			
		||||
                 . DIRECTORY_SEPARATOR;
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        return Solar_Dir::$_tmp . DIRECTORY_SEPARATOR . $sub;
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    /**
 | 
			
		||||
     * 
 | 
			
		||||
     * Returns the OS-specific temporary directory location.
 | 
			
		||||
     * 
 | 
			
		||||
     * @return string The temp directory path.
 | 
			
		||||
     * 
 | 
			
		||||
     */
 | 
			
		||||
    protected static function _tmp()
 | 
			
		||||
    {
 | 
			
		||||
        // non-Windows system?
 | 
			
		||||
        if (strtolower(substr(PHP_OS, 0, 3)) != 'win') {
 | 
			
		||||
            $tmp = empty($_ENV['TMPDIR']) ? getenv('TMPDIR') : $_ENV['TMPDIR'];
 | 
			
		||||
            if ($tmp) {
 | 
			
		||||
                return $tmp;
 | 
			
		||||
            } else {
 | 
			
		||||
                return '/tmp';
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        // Windows 'TEMP'
 | 
			
		||||
        $tmp = empty($_ENV['TEMP']) ? getenv('TEMP') : $_ENV['TEMP'];
 | 
			
		||||
        if ($tmp) {
 | 
			
		||||
            return $tmp;
 | 
			
		||||
        }
 | 
			
		||||
    
 | 
			
		||||
        // Windows 'TMP'
 | 
			
		||||
        $tmp = empty($_ENV['TMP']) ? getenv('TMP') : $_ENV['TMP'];
 | 
			
		||||
        if ($tmp) {
 | 
			
		||||
            return $tmp;
 | 
			
		||||
        }
 | 
			
		||||
    
 | 
			
		||||
        // Windows 'windir'
 | 
			
		||||
        $tmp = empty($_ENV['windir']) ? getenv('windir') : $_ENV['windir'];
 | 
			
		||||
        if ($tmp) {
 | 
			
		||||
            return $tmp;
 | 
			
		||||
        }
 | 
			
		||||
    
 | 
			
		||||
        // final fallback for Windows
 | 
			
		||||
        return getenv('SystemRoot') . '\\temp';
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
		Reference in New Issue
	
	Block a user