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; } /** |