diff options
| author | ctrlaltca@gmail.com <> | 2011-09-20 09:35:40 +0000 | 
|---|---|---|
| committer | ctrlaltca@gmail.com <> | 2011-09-20 09:35:40 +0000 | 
| commit | 8dcab11beafb1a04b514a3beacc6113f29450d3c (patch) | |
| tree | 408a41fdf9e54dd0075759c67843a0c7ffc103b4 | |
| parent | 026421ecddb050a7156b228c329e13de5f0f160d (diff) | |
patch for #363: dropping clientscripts.php
| -rw-r--r-- | .gitattributes | 5 | ||||
| -rw-r--r-- | build.xml | 2 | ||||
| -rw-r--r-- | buildscripts/classtree/build.php | 2 | ||||
| -rw-r--r-- | framework/Web/Javascripts/TJavaScript.php | 12 | ||||
| -rw-r--r-- | framework/Web/Javascripts/clientscripts.php | 573 | ||||
| -rw-r--r-- | framework/Web/Javascripts/jsmin.php | 290 | ||||
| -rw-r--r-- | framework/Web/Javascripts/packages.php (renamed from framework/Web/Javascripts/source/packages.php) | 0 | ||||
| -rw-r--r-- | framework/Web/Javascripts/source/.htaccess | 16 | ||||
| -rw-r--r-- | framework/Web/UI/TClientScriptManager.php | 135 | 
9 files changed, 366 insertions, 669 deletions
| diff --git a/.gitattributes b/.gitattributes index 64a342a8..d8c68e21 100644 --- a/.gitattributes +++ b/.gitattributes @@ -2805,8 +2805,9 @@ framework/Util/TSimpleDateFormatter.php -text  framework/Util/TVarDumper.php -text  framework/Web/Javascripts/TJSON.php -text  framework/Web/Javascripts/TJavaScript.php -text -framework/Web/Javascripts/clientscripts.php -text -framework/Web/Javascripts/source/packages.php -text +framework/Web/Javascripts/jsmin.php -text +framework/Web/Javascripts/packages.php -text +framework/Web/Javascripts/source/.htaccess -text  framework/Web/Javascripts/source/prado/activecontrols/activecontrols3.js -text  framework/Web/Javascripts/source/prado/activecontrols/activedatepicker.js -text  framework/Web/Javascripts/source/prado/activecontrols/ajax3.js -text @@ -513,7 +513,7 @@ the PRADO distribution:  	<coverage-setup database="${reports.codecoverage.dir}/coverage.db">
  	  <fileset dir="${src.dir}">
  	    <include name="**/*.php"/>
 -	    <exclude name="Web/Javascripts/js/clientscripts.php"/>
 +	    <exclude name="Web/Javascripts/js/jsmin.php"/>
  	    <exclude name="Data/TCache.php"/>
  	    <exclude name="DataAccess/**/*.php"/>
  	    <exclude name="I18N/core/Gettext/MO.php"/>
 diff --git a/buildscripts/classtree/build.php b/buildscripts/classtree/build.php index ac79370d..f91bc629 100644 --- a/buildscripts/classtree/build.php +++ b/buildscripts/classtree/build.php @@ -11,7 +11,7 @@ Prado::using('System.Data.SqlMap.TSqlMapManager');  $exclusions=array(
  	'pradolite.php',
  	'prado-cli.php',
 -	'clientscripts.php',
 +	'jsmin.php',
  	'.svn',
  	'/I18N/core',
  	'/3rdParty',
 diff --git a/framework/Web/Javascripts/TJavaScript.php b/framework/Web/Javascripts/TJavaScript.php index 2df16816..c492c197 100644 --- a/framework/Web/Javascripts/TJavaScript.php +++ b/framework/Web/Javascripts/TJavaScript.php @@ -266,5 +266,17 @@ class TJavaScript  			self::$_json = Prado::createComponent('System.Web.Javascripts.TJSON');
  		return self::$_json->decode($value);
  	}
 +
 +	/**
 +	 * Minimize the size of a javascript script.
 +	 * This method is based on Douglas Crockford's JSMin.
 +	 * @param string code that you want to minimzie 
 +	 * @return minimized version of the code
 +	 */
 +	public static function JSMin($code)
 +	{
 +		include_once('jsmin.php');
 +		return JSMin::minify($code);
 +	}
  }
 diff --git a/framework/Web/Javascripts/clientscripts.php b/framework/Web/Javascripts/clientscripts.php deleted file mode 100644 index 18f4c41b..00000000 --- a/framework/Web/Javascripts/clientscripts.php +++ /dev/null @@ -1,573 +0,0 @@ -<?php
 -
 -/**
 - * Combines multiple javascript files and serve up as gzip if possible.
 - * Allowable scripts and script dependencies can be specified in a "packages.php" file
 - * with the following format. This "packages.php" is optional, if absent the filenames
 - * without ".js" extension are used.
 - *
 - * <code>
 - * <?php
 - *  $packages = array(
 - *     'package1' => array('file1.js', 'file2.js'),
 - *     'package2' => array('file3.js', 'file4.js'));
 - *
 - *  $dependencies = array(
 - *     'package1' => array('package1'),
 - *     'package2' => array('package1', 'package2')); //package2 requires package1 first.
 - * </code>
 - *
 - * To serve up 'package1', specify the url, a maxium of 25 packages separated with commas is allows.
 - *
 - * clientscripts.php?js=package1
 - *
 - * for 'package2' (automatically resolves 'package1') dependency
 - *
 - * clientscripts.php?js=package2
 - *
 - * The scripts comments are removed and whitespaces removed appropriately. The
 - * scripts may be served as zipped if the browser and php server allows it. Cache
 - * headers are also sent to inform the browser and proxies to cache the file.
 - * Moreover, the post-processed (comments removed and zipped) are saved in this
 - * current directory for the next requests.
 - *
 - * If the url contains the parameter "mode=debug", the comments are not removed
 - * and headers invalidating the cache are sent. In debug mode, the script can still
 - * be zipped if the browser and server supports it.
 - *
 - * E.g. clientscripts.php?js=package2&mode=debug
 - *
 - * @link http://www.pradosoft.com/
 - * @copyright Copyright © 2007 PradoSoft
 - * @license http://www.pradosoft.com/license/
 - * @author Wei Zhuo<weizhuo[at]gmail[dot]com>
 - * @version $Id$
 - * @package System.Web.Javascripts
 - * @since 3.1
 - */
 -
 -//run script for as long as needed
 -@set_time_limit(0);
 -
 -//set error_reporting directive
 -@error_reporting(E_ERROR | E_WARNING | E_PARSE);
 -
 -function get_client_script_files()
 -{
 -	$package_file = dirname(__FILE__).'/packages.php';
 -	if(is_file($package_file))
 -		return get_package_files($package_file, get_script_requests());
 -	else
 -		return get_javascript_files(get_script_requests());
 -}
 -
 -/**
 - * @param array list of requested libraries
 - */
 -function get_script_requests($max=25)
 -{
 -	$param = isset($_GET['js']) ? $_GET['js'] : '';
 -	if(preg_match('/([a-zA-z0-9\-_])+(,[a-zA-z0-9\-_]+)*/', $param))
 -		return array_unique(explode(',', $param, $max));
 -	return array();
 -}
 -
 -/**
 - * @param string packages.php filename
 - * @param array packages requests
 - */
 -function get_package_files($package_file, $request)
 -{
 -	list($packages, $dependencies) = include($package_file);
 -	if(!(is_array($packages) && is_array($dependencies)))
 -	{
 -		error_log('Prado client script: invalid package file '.$package_file);
 -		return array();
 -	}
 -	$result = array();
 -	$found = array();
 -	foreach($request as $library)
 -	{
 -		if(isset($dependencies[$library]))
 -		{
 -			foreach($dependencies[$library] as $dep)
 -			{
 -				if(isset($found[$dep]))
 -					continue;
 -				$result = array_merge($result, (array)$packages[$dep]);
 -				$found[$dep]=true;
 -			}
 -		}
 -		else
 -			error_log('Prado client script: no such javascript library "'.$library.'"');
 -	}
 -	return $result;
 -}
 -
 -/**
 - * @param string requested javascript files
 - * @array array list of javascript files.
 - */
 -function get_javascript_files($request)
 -{
 -	$result = array();
 -	foreach($request as $file)
 -		$result[] = $file.'.js';
 -	return $result;
 -}
 -
 -/**
 - * @param array list of javascript files.
 - * @return string combined the available javascript files.
 - */
 -function combine_javascript($files)
 -{
 -	$content = '';
 -	$base = dirname(__FILE__);
 -	foreach($files as $file)
 -	{
 -		$filename = $base.'/'.$file;
 -		if(is_file($filename)) //relies on get_client_script_files() for security
 -			$content .= "\x0D\x0A".file_get_contents($filename); //add CR+LF
 -		else
 -			error_log('Prado client script: missing file '.$filename);
 -	}
 -	return $content;
 -}
 -
 -/**
 - * @param string javascript code
 - * @param array files names
 - * @return string javascript code without comments.
 - */
 -function minify_javascript($content, $files)
 -{
 -	return JSMin::minify($content);
 -}
 -
 -/**
 - * @param boolean true if in debug mode.
 - */
 -function is_debug_mode()
 -{
 -	return isset($_GET['mode']) && $_GET['mode']==='debug';
 -}
 -
 -/**
 - * @param string javascript code
 - * @param string gzip code
 - */
 -function gzip_content($content)
 -{
 -	return gzencode($content, 9, FORCE_GZIP);
 -}
 -
 -/**
 - * @param string javascript code.
 - * @param string filename
 - */
 -function save_javascript($content, $filename)
 -{
 -	file_put_contents($filename, $content);
 -	if(supports_gzip_encoding())
 -		file_put_contents($filename.'.gz', gzip_content($content));
 -}
 -
 -/**
 - * @param string comprssed javascript file to be read
 - * @param string javascript code, null if file is not found.
 - */
 -function get_saved_javascript($filename)
 -{
 -	$fn=$filename;
 -	if(supports_gzip_encoding())
 -		$fn .= '.gz';
 -	if(!is_file($fn))
 -		save_javascript(get_javascript_code(true), $filename);
 -	return file_get_contents($fn);
 -}
 -
 -/**
 - * @return string compressed javascript file name.
 - */
 -function compressed_js_filename()
 -{
 -	$files = get_client_script_files();
 -	if(count($files) > 0)
 -	{
 -		$filename = sprintf('%x',crc32(implode(',',($files))));
 -		return dirname(__FILE__).'/clientscript_'.$filename.'.js';
 -	}
 -}
 -
 -/**
 - * @param boolean true to strip comments from javascript code
 - * @return string javascript code
 - */
 -function get_javascript_code($minify=false)
 -{
 -	$files = get_client_script_files();
 -	if(count($files) > 0)
 -	{
 -		$content = combine_javascript($files);
 -		if($minify)
 -			return minify_javascript($content, $files);
 -		else
 -			return $content;
 -	}
 -}
 -
 -/**
 - * Prints headers to serve javascript
 - *
 - * @param string $filePath (default: NULL)
 - */
 -function print_headers($filePath = null)
 -{
 -	$expiresOffset = is_debug_mode() ? -10000 : 3600 * 24 * 10; //no cache
 -	header("Content-type: text/javascript");
 -	header("Vary: Accept-Encoding");  // Handle proxies
 -	header("Expires: " . @gmdate("D, d M Y H:i:s", @time() + $expiresOffset) . " GMT");
 -	if(null !== $filePath && is_file($filePath)) {
 -		header("Last-Modified: " . @gmdate("D, d M Y H:i:s", filectime($filePath)) . " GMT");
 -	}
 -	if(($enc = supports_gzip_encoding()) !== null)
 -		header("Content-Encoding: " . $enc);
 -}
 -
 -/**
 - * @return string 'x-gzip' or 'gzip' if php supports gzip and browser supports gzip response, null otherwise.
 - */
 -function supports_gzip_encoding()
 -{
 -	if(isset($_GET['gzip']) && $_GET['gzip']==='false')
 -		return false;
 -
 -	if (isset($_SERVER['HTTP_ACCEPT_ENCODING']))
 -	{
 -		$encodings = explode(',', strtolower(preg_replace("/\s+/", "", $_SERVER['HTTP_ACCEPT_ENCODING'])));
 -		$allowsZipEncoding = in_array('gzip', $encodings) || in_array('x-gzip', $encodings) || isset($_SERVER['---------------']);
 -		$hasGzip = function_exists('ob_gzhandler');
 -		$noZipBuffer = !ini_get('zlib.output_compression');
 -		$noZipBufferHandler = ini_get('output_handler') != 'ob_gzhandler';
 -
 -		if ( $allowsZipEncoding && $hasGzip && $noZipBuffer && $noZipBufferHandler)
 -			$enc = in_array('x-gzip', $encodings) ? "x-gzip" : "gzip";
 -		return $enc;
 -	}
 -}
 -
 -/**
 - * jsmin.php - PHP implementation of Douglas Crockford's JSMin.
 - *
 - * This is pretty much a direct port of jsmin.c to PHP with just a few
 - * PHP-specific performance tweaks. Also, whereas jsmin.c reads from stdin and
 - * outputs to stdout, this library accepts a string as input and returns another
 - * string as output.
 - *
 - * 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>
 - * @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
 - * @version 1.1.1 (2008-03-02)
 - * @link http://code.google.com/p/jsmin-php/
 - */
 -
 -class JSMin {
 -  const ORD_LF    = 10;
 -  const ORD_SPACE = 32;
 -
 -  protected $a           = '';
 -  protected $b           = '';
 -  protected $input       = '';
 -  protected $inputIndex  = 0;
 -  protected $inputLength = 0;
 -  protected $lookAhead   = null;
 -  protected $output      = '';
 -
 -  // -- Public Static Methods --------------------------------------------------
 -
 -  public static function minify($js) {
 -    $jsmin = new JSMin($js);
 -    return $jsmin->min();
 -  }
 -
 -  // -- Public Instance Methods ------------------------------------------------
 -
 -  public function __construct($input) {
 -    $this->input       = str_replace("\r\n", "\n", $input);
 -    $this->inputLength = strlen($this->input);
 -  }
 -
 -  // -- Protected Instance Methods ---------------------------------------------
 -
 -  protected function action($d) {
 -    switch($d) {
 -      case 1:
 -        $this->output .= $this->a;
 -
 -      case 2:
 -        $this->a = $this->b;
 -
 -        if ($this->a === "'" || $this->a === '"') {
 -          for (;;) {
 -            $this->output .= $this->a;
 -            $this->a       = $this->get();
 -
 -            if ($this->a === $this->b) {
 -              break;
 -            }
 -
 -            if (ord($this->a) <= self::ORD_LF) {
 -              throw new JSMinException('Unterminated string literal.');
 -            }
 -
 -            if ($this->a === '\\') {
 -              $this->output .= $this->a;
 -              $this->a       = $this->get();
 -            }
 -          }
 -        }
 -
 -      case 3:
 -        $this->b = $this->next();
 -
 -        if ($this->b === '/' && (
 -            $this->a === '(' || $this->a === ',' || $this->a === '=' ||
 -            $this->a === ':' || $this->a === '[' || $this->a === '!' ||
 -            $this->a === '&' || $this->a === '|' || $this->a === '?')) {
 -
 -          $this->output .= $this->a . $this->b;
 -
 -          for (;;) {
 -            $this->a = $this->get();
 -
 -            if ($this->a === '/') {
 -              break;
 -            } elseif ($this->a === '\\') {
 -              $this->output .= $this->a;
 -              $this->a       = $this->get();
 -            } elseif (ord($this->a) <= self::ORD_LF) {
 -              throw new JSMinException('Unterminated regular expression '.
 -                  'literal.');
 -            }
 -
 -            $this->output .= $this->a;
 -          }
 -
 -          $this->b = $this->next();
 -        }
 -    }
 -  }
 -
 -  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 {
 -        $c = null;
 -      }
 -    }
 -
 -    if ($c === "\r") {
 -      return "\n";
 -    }
 -
 -    if ($c === null || $c === "\n" || ord($c) >= self::ORD_SPACE) {
 -      return $c;
 -    }
 -
 -    return ' ';
 -  }
 -
 -  protected function isAlphaNum($c) {
 -    return ord($c) > 126 || $c === '\\' || preg_match('/^[\w\$]$/', $c) === 1;
 -  }
 -
 -  protected function min() {
 -    $this->a = "\n";
 -    $this->action(3);
 -
 -    while ($this->a !== null) {
 -      switch ($this->a) {
 -        case ' ':
 -          if ($this->isAlphaNum($this->b)) {
 -            $this->action(1);
 -          } else {
 -            $this->action(2);
 -          }
 -          break;
 -
 -        case "\n":
 -          switch ($this->b) {
 -            case '{':
 -            case '[':
 -            case '(':
 -            case '+':
 -            case '-':
 -              $this->action(1);
 -              break;
 -
 -            case ' ':
 -              $this->action(3);
 -              break;
 -
 -            default:
 -              if ($this->isAlphaNum($this->b)) {
 -                $this->action(1);
 -              }
 -              else {
 -                $this->action(2);
 -              }
 -          }
 -          break;
 -
 -        default:
 -          switch ($this->b) {
 -            case ' ':
 -              if ($this->isAlphaNum($this->a)) {
 -                $this->action(1);
 -                break;
 -              }
 -
 -              $this->action(3);
 -              break;
 -
 -            case "\n":
 -              switch ($this->a) {
 -                case '}':
 -                case ']':
 -                case ')':
 -                case '+':
 -                case '-':
 -                case '"':
 -                case "'":
 -                  $this->action(1);
 -                  break;
 -
 -                default:
 -                  if ($this->isAlphaNum($this->a)) {
 -                    $this->action(1);
 -                  }
 -                  else {
 -                    $this->action(3);
 -                  }
 -              }
 -              break;
 -
 -            default:
 -              $this->action(1);
 -              break;
 -          }
 -      }
 -    }
 -
 -    return $this->output;
 -  }
 -
 -  protected function next() {
 -    $c = $this->get();
 -
 -    if ($c === '/') {
 -      switch($this->peek()) {
 -        case '/':
 -          for (;;) {
 -            $c = $this->get();
 -
 -            if (ord($c) <= self::ORD_LF) {
 -              return $c;
 -            }
 -          }
 -
 -        case '*':
 -          $this->get();
 -
 -          for (;;) {
 -            switch($this->get()) {
 -              case '*':
 -                if ($this->peek() === '/') {
 -                  $this->get();
 -                  return ' ';
 -                }
 -                break;
 -
 -              case null:
 -                throw new JSMinException('Unterminated comment.');
 -            }
 -          }
 -
 -        default:
 -          return $c;
 -      }
 -    }
 -
 -    return $c;
 -  }
 -
 -  protected function peek() {
 -    $this->lookAhead = $this->get();
 -    return $this->lookAhead;
 -  }
 -}
 -
 -// -- Exceptions ---------------------------------------------------------------
 -class JSMinException extends Exception {}
 -
 -
 -/************** OUTPUT *****************/
 -
 -if(count(get_script_requests()) > 0)
 -{
 -	if(is_debug_mode())
 -	{
 -		if(($code = get_javascript_code()) !== null)
 -		{
 -			print_headers();
 -			echo supports_gzip_encoding() ? gzip_content($code) : $code;
 -		}
 -	}
 -	else
 -	{
 -		if(($filename = compressed_js_filename()) !== null)
 -		{
 -			if(!is_file($filename))
 -				save_javascript(get_javascript_code(true), $filename);
 -			print_headers($filename);
 -			echo get_saved_javascript($filename);
 -		}
 -	}
 -}
 diff --git a/framework/Web/Javascripts/jsmin.php b/framework/Web/Javascripts/jsmin.php new file mode 100644 index 00000000..6ed24033 --- /dev/null +++ b/framework/Web/Javascripts/jsmin.php @@ -0,0 +1,290 @@ +<?php
 +/**
 + * jsmin.php - PHP implementation of Douglas Crockford's JSMin.
 + *
 + * This is pretty much a direct port of jsmin.c to PHP with just a few
 + * PHP-specific performance tweaks. Also, whereas jsmin.c reads from stdin and
 + * outputs to stdout, this library accepts a string as input and returns another
 + * string as output.
 + *
 + * 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>
 + * @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
 + * @version 1.1.1 (2008-03-02)
 + * @link http://code.google.com/p/jsmin-php/
 + */
 +
 +class JSMin {
 +  const ORD_LF    = 10;
 +  const ORD_SPACE = 32;
 +
 +  protected $a           = '';
 +  protected $b           = '';
 +  protected $input       = '';
 +  protected $inputIndex  = 0;
 +  protected $inputLength = 0;
 +  protected $lookAhead   = null;
 +  protected $output      = '';
 +
 +  // -- Public Static Methods --------------------------------------------------
 +
 +  public static function minify($js) {
 +    $jsmin = new JSMin($js);
 +    return $jsmin->min();
 +  }
 +
 +  // -- Public Instance Methods ------------------------------------------------
 +
 +  public function __construct($input) {
 +    $this->input       = str_replace("\r\n", "\n", $input);
 +    $this->inputLength = strlen($this->input);
 +  }
 +
 +  // -- Protected Instance Methods ---------------------------------------------
 +
 +  protected function action($d) {
 +    switch($d) {
 +      case 1:
 +        $this->output .= $this->a;
 +
 +      case 2:
 +        $this->a = $this->b;
 +
 +        if ($this->a === "'" || $this->a === '"') {
 +          for (;;) {
 +            $this->output .= $this->a;
 +            $this->a       = $this->get();
 +
 +            if ($this->a === $this->b) {
 +              break;
 +            }
 +
 +            if (ord($this->a) <= self::ORD_LF) {
 +              throw new JSMinException('Unterminated string literal.');
 +            }
 +
 +            if ($this->a === '\\') {
 +              $this->output .= $this->a;
 +              $this->a       = $this->get();
 +            }
 +          }
 +        }
 +
 +      case 3:
 +        $this->b = $this->next();
 +
 +        if ($this->b === '/' && (
 +            $this->a === '(' || $this->a === ',' || $this->a === '=' ||
 +            $this->a === ':' || $this->a === '[' || $this->a === '!' ||
 +            $this->a === '&' || $this->a === '|' || $this->a === '?')) {
 +
 +          $this->output .= $this->a . $this->b;
 +
 +          for (;;) {
 +            $this->a = $this->get();
 +
 +            if ($this->a === '/') {
 +              break;
 +            } elseif ($this->a === '\\') {
 +              $this->output .= $this->a;
 +              $this->a       = $this->get();
 +            } elseif (ord($this->a) <= self::ORD_LF) {
 +              throw new JSMinException('Unterminated regular expression '.
 +                  'literal.');
 +            }
 +
 +            $this->output .= $this->a;
 +          }
 +
 +          $this->b = $this->next();
 +        }
 +    }
 +  }
 +
 +  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 {
 +        $c = null;
 +      }
 +    }
 +
 +    if ($c === "\r") {
 +      return "\n";
 +    }
 +
 +    if ($c === null || $c === "\n" || ord($c) >= self::ORD_SPACE) {
 +      return $c;
 +    }
 +
 +    return ' ';
 +  }
 +
 +  protected function isAlphaNum($c) {
 +    return ord($c) > 126 || $c === '\\' || preg_match('/^[\w\$]$/', $c) === 1;
 +  }
 +
 +  protected function min() {
 +    $this->a = "\n";
 +    $this->action(3);
 +
 +    while ($this->a !== null) {
 +      switch ($this->a) {
 +        case ' ':
 +          if ($this->isAlphaNum($this->b)) {
 +            $this->action(1);
 +          } else {
 +            $this->action(2);
 +          }
 +          break;
 +
 +        case "\n":
 +          switch ($this->b) {
 +            case '{':
 +            case '[':
 +            case '(':
 +            case '+':
 +            case '-':
 +              $this->action(1);
 +              break;
 +
 +            case ' ':
 +              $this->action(3);
 +              break;
 +
 +            default:
 +              if ($this->isAlphaNum($this->b)) {
 +                $this->action(1);
 +              }
 +              else {
 +                $this->action(2);
 +              }
 +          }
 +          break;
 +
 +        default:
 +          switch ($this->b) {
 +            case ' ':
 +              if ($this->isAlphaNum($this->a)) {
 +                $this->action(1);
 +                break;
 +              }
 +
 +              $this->action(3);
 +              break;
 +
 +            case "\n":
 +              switch ($this->a) {
 +                case '}':
 +                case ']':
 +                case ')':
 +                case '+':
 +                case '-':
 +                case '"':
 +                case "'":
 +                  $this->action(1);
 +                  break;
 +
 +                default:
 +                  if ($this->isAlphaNum($this->a)) {
 +                    $this->action(1);
 +                  }
 +                  else {
 +                    $this->action(3);
 +                  }
 +              }
 +              break;
 +
 +            default:
 +              $this->action(1);
 +              break;
 +          }
 +      }
 +    }
 +
 +    return $this->output;
 +  }
 +
 +  protected function next() {
 +    $c = $this->get();
 +
 +    if ($c === '/') {
 +      switch($this->peek()) {
 +        case '/':
 +          for (;;) {
 +            $c = $this->get();
 +
 +            if (ord($c) <= self::ORD_LF) {
 +              return $c;
 +            }
 +          }
 +
 +        case '*':
 +          $this->get();
 +
 +          for (;;) {
 +            switch($this->get()) {
 +              case '*':
 +                if ($this->peek() === '/') {
 +                  $this->get();
 +                  return ' ';
 +                }
 +                break;
 +
 +              case null:
 +                throw new JSMinException('Unterminated comment.');
 +            }
 +          }
 +
 +        default:
 +          return $c;
 +      }
 +    }
 +
 +    return $c;
 +  }
 +
 +  protected function peek() {
 +    $this->lookAhead = $this->get();
 +    return $this->lookAhead;
 +  }
 +}
 +
 +// -- Exceptions ---------------------------------------------------------------
 +class JSMinException extends Exception {}
 diff --git a/framework/Web/Javascripts/source/packages.php b/framework/Web/Javascripts/packages.php index cff913f8..cff913f8 100644 --- a/framework/Web/Javascripts/source/packages.php +++ b/framework/Web/Javascripts/packages.php diff --git a/framework/Web/Javascripts/source/.htaccess b/framework/Web/Javascripts/source/.htaccess new file mode 100644 index 00000000..9ec96342 --- /dev/null +++ b/framework/Web/Javascripts/source/.htaccess @@ -0,0 +1,16 @@ +<ifModule mod_gzip.c> +  mod_gzip_on Yes +  mod_gzip_dechunk Yes +  mod_gzip_item_include file \.js$ +  mod_gzip_item_exclude rspheader ^Content-Encoding:.*gzip.* +</ifModule> +  +<ifModule mod_expires.c> +  ExpiresActive On +  ExpiresDefault "access plus 864000 seconds" +</ifModule> + +<ifModule mod_headers.c> +  Header set Cache-Control "max-age=864000, private" +</ifModule> + diff --git a/framework/Web/UI/TClientScriptManager.php b/framework/Web/UI/TClientScriptManager.php index 19a8090c..0a673b5b 100644 --- a/framework/Web/UI/TClientScriptManager.php +++ b/framework/Web/UI/TClientScriptManager.php @@ -29,10 +29,9 @@ class TClientScriptManager extends TApplicationComponent  	 */  	const SCRIPT_PATH='Web/Javascripts/source';  	/** -	 * the PHP script for loading Prado javascript files +	 * file containing javascript packages and their cross dependencies  	 */ -	const SCRIPT_LOADER='Web/Javascripts/clientscripts.php'; - +	const PACKAGES_FILE='Web/Javascripts/packages.php';  	/**  	 * @var TPage page who owns this manager  	 */ @@ -74,12 +73,12 @@ class TClientScriptManager extends TApplicationComponent  	 */  	private $_registeredPradoScripts=array();  	/** -	 * Client-side javascript library dependencies, loads from SCRIPT_PATH.'/packages.php'; +	 * Client-side javascript library dependencies, loads from PACKAGES_FILE;  	 * @var array  	 */  	private static $_pradoScripts;  	/** -	 * Client-side javascript library packages, loads from SCRIPT_PATH.'/packages.php'; +	 * Client-side javascript library packages, loads from PACKAGES_FILE;  	 * @var array  	 */  	private static $_pradoPackages; @@ -120,7 +119,7 @@ class TClientScriptManager extends TApplicationComponent  	}  	/** -	 * Registers Prado javascript by library name. See "Web/Javascripts/source/packages.php" +	 * Registers Prado javascript by library name. See "Web/Javascripts/packages.php"  	 * for library names.  	 * @param string script library name.  	 */ @@ -141,7 +140,7 @@ class TClientScriptManager extends TApplicationComponent  		{  			if(self::$_pradoScripts === null)  			{ -				$packageFile = Prado::getFrameworkPath().DIRECTORY_SEPARATOR.self::SCRIPT_PATH.'/packages.php'; +				$packageFile = Prado::getFrameworkPath().DIRECTORY_SEPARATOR.self::PACKAGES_FILE;  				list($packages,$deps)= include($packageFile);  				self::$_pradoScripts = $deps;  				self::$_pradoPackages = $packages; @@ -174,31 +173,38 @@ class TClientScriptManager extends TApplicationComponent  		$addedScripts = array_diff($this->_registeredPradoScripts,$this->_renderedPradoScripts);  		if(($packages=array_keys($addedScripts))!==array())  		{ -			if (Prado::getApplication()->getMode()!==TApplicationMode::Debug) -			{ -				$base = Prado::getFrameworkPath().DIRECTORY_SEPARATOR.self::SCRIPT_PATH; -				$url = $this->registerJavascriptPackages($base, $packages); -				$writer->write(TJavaScript::renderScriptFile($url)); -			} -			else +			$base = Prado::getFrameworkPath().DIRECTORY_SEPARATOR.self::SCRIPT_PATH; +			list($path,$baseUrl)=$this->getPackagePathUrl($base); +			$packagesUrl=array(); +			$isDebug=$this->getApplication()->getMode()===TApplicationMode::Debug; +			foreach ($packages as $p)  			{ -				// In debug mode, we add 1 <script> line by file -				$base = Prado::getFrameworkPath().DIRECTORY_SEPARATOR.self::SCRIPT_PATH; -				list($path,$baseUrl)=$this->getPackagePathUrl($base); -				$packagesUrl=array(); -				foreach ($packages as $p) +				foreach (self::$_pradoScripts[$p] as $dep)  				{ -					foreach (self::$_pradoScripts[$p] as $dep) +					foreach (self::$_pradoPackages[$dep] as $script)  					{ -						foreach (self::$_pradoPackages[$dep] as $script) +						if($isDebug)  						{  							if (!in_array($url=$baseUrl.'/'.$script,$packagesUrl))  								$packagesUrl[]=$url; +						} else { +							if (!in_array($url=$baseUrl.'/min/'.$script,$packagesUrl)) +							{ +								if(!is_file($filePath=$path.'/min/'.$script)) +								{ +									$dirPath=dirname($filePath); +									if(!is_dir($dirPath)) +										mkdir($dirPath, PRADO_CHMOD, true); +									file_put_contents($filePath, TJavaScript::JSMin(file_get_contents($base.'/'.$script))); +									chmod($filePath, PRADO_CHMOD); +								} +								$packagesUrl[]=$url; +							}  						}  					}  				} -				$writer->write(TJavaScript::renderScriptFiles($packagesUrl));  			} +			$writer->write(TJavaScript::renderScriptFiles($packagesUrl));  			$this->_renderedPradoScripts = $this->_registeredPradoScripts;  		}  	} @@ -209,84 +215,29 @@ class TClientScriptManager extends TApplicationComponent  	 */  	public function getScriptUrls()  	{ -		if (Prado::getApplication()->getMode()!==TApplicationMode::Debug) -		{ -			$packages=array_keys($this->_registeredPradoScripts); -			$base = Prado::getFrameworkPath().DIRECTORY_SEPARATOR.self::SCRIPT_PATH; -			return array($this->registerJavascriptPackages($base, $packages)); -		} else { -			$scripts = array(); +		$scripts = array(); -			$packages=array_keys($this->_registeredPradoScripts); -			$base = Prado::getFrameworkPath().DIRECTORY_SEPARATOR.self::SCRIPT_PATH; -			list($path,$baseUrl)=$this->getPackagePathUrl($base); -			foreach ($packages as $p) +		$packages=array_keys($this->_registeredPradoScripts); +		$base = Prado::getFrameworkPath().DIRECTORY_SEPARATOR.self::SCRIPT_PATH; +		list($path,$baseUrl)=$this->getPackagePathUrl($base); +		foreach ($packages as $p) +		{ +			foreach (self::$_pradoScripts[$p] as $dep)  			{ -				foreach (self::$_pradoScripts[$p] as $dep) +				foreach (self::$_pradoPackages[$dep] as $script)  				{ -					foreach (self::$_pradoPackages[$dep] as $script) -					{ -						if (!in_array($url=$baseUrl.'/'.$script,$scripts)) -							$scripts[]=$url; -					} +					if (!in_array($url=$baseUrl.'/'.$script,$scripts)) +						$scripts[]=$url;  				}  			} - -			$scripts = array_merge($scripts, array_values($this->_headScriptFiles)); -			$scripts = array_merge($scripts, array_values($this->_scriptFiles)); - -			$scripts = array_unique($scripts); - -			return $scripts;  		} -	} -	/** -	 * Publishes a javascript library path and register packages to be loaded. -	 * See TClientScriptLoader for component that enables users to register custom javascript libraries. -	 * @param string javascript library base path -	 * @param array list of packages or javascript files (without .js extension) to be loaded. -	 * @param boolean true to enable keep comments in javascript files loaded, null to use application configuration. -	 * @param boolean true to gzip the javascript code if browsers and php supports it. -	 * @return string javascript src url -	 * @since 3.1 -	 */ -	public function registerJavascriptPackages($base, $packages, $debug=null, $gzip=true) -	{ -		list($path,$url) = $this->getPackagePathUrl($base); -		$scriptLoaderPath = $path.'/'.basename(self::SCRIPT_LOADER); -		$scriptLoaderSrc = Prado::getFrameworkPath().DIRECTORY_SEPARATOR.self::SCRIPT_LOADER; -		if(!is_file($scriptLoaderPath)) -		{ -			copy($scriptLoaderSrc, $scriptLoaderPath); -			chmod($scriptLoaderPath, PRADO_CHMOD); -		} -		$url .= '/'.basename(self::SCRIPT_LOADER).'?js='.implode(',', $packages); -		if($debug!==false && $this->getApplication()->getMode()===TApplicationMode::Debug) -		{ -			$this->verifyJavascriptPackages($base,$path,$packages); -			$url.='&mode=debug'; -		} -		if($gzip===false) -			$url.='&gzip=false'; -		return $url; -	} +		$scripts = array_merge($scripts, array_values($this->_headScriptFiles)); +		$scripts = array_merge($scripts, array_values($this->_scriptFiles)); -	/** -	 * @throws TConfigurationException when javascript packages mismatch. -	 */ -	protected function verifyJavascriptPackages($base,$path,$scripts) -	{ -		$file = $path.'/packages.php'; -		if(is_file($file)) -		{ -			list($packs,$deps) = include($file); -			if(count($missing = array_diff($scripts, array_keys($deps))) > 0) -			{ -				throw new TConfigurationException('csmanager_invalid_packages', -					$base.'/packages.php',implode(', ', $missing), implode(', ', array_keys($deps))); -			} -		} +		$scripts = array_unique($scripts); + +		return $scripts;  	}  	/** | 
