Cleanup: Set SVN properties and tags
authorOliver Hader <oliver.hader@typo3.org>
Mon, 31 Aug 2009 12:36:01 +0000 (12:36 +0000)
committerOliver Hader <oliver.hader@typo3.org>
Mon, 31 Aug 2009 12:36:01 +0000 (12:36 +0000)
git-svn-id: https://svn.typo3.org/TYPO3v4/Core/trunk@5855 709f56b5-9817-0410-a4d7-c38de5d9e867

ChangeLog
typo3/contrib/flashmedia/player.txt
typo3/contrib/flashmedia/qtobject/qtobject.js
typo3/contrib/swfupload/plugins/swfobject_license.txt
typo3/contrib/swfupload/plugins/swfupload.cookies.js
typo3/contrib/swfupload/plugins/swfupload.queue.js
typo3/contrib/swfupload/plugins/swfupload.speed.js
typo3/contrib/swfupload/plugins/swfupload.swfobject.js
typo3/contrib/swfupload/swfupload.js
typo3/sysext/cms/tslib/hooks/class.tx_cms_mediaitems.php

index f275f38..ee64f20 100755 (executable)
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,7 @@
+2009-08-31  Oliver Hader  <oliver@typo3.org>
+
+       * Cleanup: Set SVN properties and tags
+
 2009-08-31  Ingo Renner  <ingo@typo3.org>
 
        * Fixed bug #11810: Workspace indicator bar does not cover the full width of the page tree
index b8852a9..ce11165 100644 (file)
@@ -1,25 +1,25 @@
-File: player.swf\r
-Plugin Name: Audio player\r
-Plugin URI: http://www.1pixelout.net/code/audio-player-wordpress-plugin/\r
-Description: Highly configurable single track mp3 player\r
-Version: 1.2\r
-Author: Martin Laine\r
-Author URI: http://www.1pixelout.net\r
-\r
-License:\r
-\r
-    Copyright 2005-2006  Martin Laine  (email : martin@1pixelout.net)\r
-\r
-    This program is free software; you can redistribute it and/or modify\r
-    it under the terms of the GNU General Public License as published by\r
-    the Free Software Foundation; either version 2 of the License, or\r
-    any later version.\r
-\r
-    This program is distributed in the hope that it will be useful,\r
-    but WITHOUT ANY WARRANTY; without even the implied warranty of\r
-    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\r
-    GNU General Public License for more details.\r
-\r
-    You should have received a copy of the GNU General Public License\r
-    along with this program; if not, write to the Free Software\r
-    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA\r
+File: player.swf
+Plugin Name: Audio player
+Plugin URI: http://www.1pixelout.net/code/audio-player-wordpress-plugin/
+Description: Highly configurable single track mp3 player
+Version: 1.2
+Author: Martin Laine
+Author URI: http://www.1pixelout.net
+
+License:
+
+    Copyright 2005-2006  Martin Laine  (email : martin@1pixelout.net)
+
+    This program is free software; you can redistribute it and/or modify
+    it under the terms of the GNU General Public License as published by
+    the Free Software Foundation; either version 2 of the License, or
+    any later version.
+
+    This program is distributed in the hope that it will be useful,
+    but WITHOUT ANY WARRANTY; without even the implied warranty of
+    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+    GNU General Public License for more details.
+
+    You should have received a copy of the GNU General Public License
+    along with this program; if not, write to the Free Software
+    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
index 44bc39f..aebfc9a 100644 (file)
-/*\r
- * QTObject embed\r
- * http://blog.deconcept.com/2005/01/26/web-standards-compliant-javascript-quicktime-detect-and-embed/\r
- *\r
- * by Geoff Stearns (geoff@deconcept.com, http://www.deconcept.com/)\r
- *\r
- * v1.0.2 - 02-16-2005\r
- *\r
- * Embeds a quicktime movie to the page, includes plugin detection\r
- *\r
- * Usage:\r
- *\r
- *     myQTObject = new QTObject("path/to/mov.mov", "movid", "width", "height");\r
+/*
+ * QTObject embed
+ * http://blog.deconcept.com/2005/01/26/web-standards-compliant-javascript-quicktime-detect-and-embed/
+ *
+ * by Geoff Stearns (geoff@deconcept.com, http://www.deconcept.com/)
+ *
+ * v1.0.2 - 02-16-2005
+ *
+ * Embeds a quicktime movie to the page, includes plugin detection
+ *
+ * Usage:
+ *
+ *     myQTObject = new QTObject("path/to/mov.mov", "movid", "width", "height");
  *     myQTObject.altTxt = "Upgrade your Quicktime Player!";    // optional
\r
- *  myQTObject.addParam("controller", "false");              // optional\r
- *     myQTObject.write();\r
- *\r
- */\r
-\r
-QTObject = function(mov, id, w, h) {\r
-       this.mov = mov;\r
-       this.id = id;\r
-       this.width = w;\r
-       this.height = h;\r
-       this.redirect = "";\r
-       this.sq = document.location.search.split("?")[1] || "";\r
-       this.altTxt = "This content requires the QuickTime Plugin. <a href='http://www.apple.com/quicktime/download/'>Download QuickTime Player</a>.";\r
-       this.bypassTxt = "<p>Already have QuickTime Player? <a href='?detectqt=false&"+ this.sq +"'>Click here.</a></p>";\r
-       this.params = new Object();\r
-       this.doDetect = getQueryParamValue('detectqt');\r
-}\r
-\r
-QTObject.prototype.addParam = function(name, value) {\r
-       this.params[name] = value;\r
-}\r
-\r
-QTObject.prototype.getParams = function() {\r
-    return this.params;\r
-}\r
-\r
-QTObject.prototype.getParam = function(name) {\r
-    return this.params[name];\r
-}\r
-\r
-QTObject.prototype.getParamTags = function() {\r
-    var paramTags = "";\r
-    for (var param in this.getParams()) {\r
-        paramTags += '<param name="' + param + '" value="' + this.getParam(param) + '" />';\r
-    }\r
-    if (paramTags == "") {\r
-        paramTags = null;\r
-    }\r
-    return paramTags;\r
-}\r
-\r
-QTObject.prototype.getHTML = function() {\r
-    var qtHTML = "";\r
-       if (navigator.plugins && navigator.plugins.length) { // not ie\r
-        qtHTML += '<embed type="video/quicktime" src="' + this.mov + '" width="' + this.width + '" height="' + this.height + '" id="' + this.id + '"';\r
-        for (var param in this.getParams()) {\r
-            qtHTML += ' ' + param + '="' + this.getParam(param) + '"';\r
-        }\r
-        qtHTML += '></embed>';\r
-    }\r
-    else { // pc ie\r
-        qtHTML += '<object classid="clsid:02BF25D5-8C17-4B23-BC80-D3488ABDDC6B" width="' + this.width + '" height="' + this.height + '" id="' + this.id + '">';\r
-        this.addParam("src", this.mov);\r
-        if (this.getParamTags() != null) {\r
-            qtHTML += this.getParamTags();\r
-        }\r
-        qtHTML += '</object>';\r
-    }\r
-    return qtHTML;\r
-}\r
-\r
-\r
-QTObject.prototype.getVariablePairs = function() {\r
-    var variablePairs = new Array();\r
-    for (var name in this.getVariables()) {\r
-        variablePairs.push(name + "=" + escape(this.getVariable(name)));\r
-    }\r
-    if (variablePairs.length > 0) {\r
-        return variablePairs.join("&");\r
-    }\r
-    else {\r
-        return null;\r
-    }\r
-}\r
-\r
-QTObject.prototype.write = function(elementId) {\r
-       if(isQTInstalled() || this.doDetect=='false') {\r
-               if (elementId) {\r
-                       document.getElementById(elementId).innerHTML = this.getHTML();\r
-               } else {\r
-                       document.write(this.getHTML());\r
-               }\r
-       } else {\r
-               if (this.redirect != "") {\r
-                       document.location.replace(this.redirect);\r
-               } else {\r
-                       if (elementId) {\r
-                               document.getElementById(elementId).innerHTML = this.altTxt +""+ this.bypassTxt;\r
-                       } else {\r
-                               document.write(this.altTxt +""+ this.bypassTxt);\r
-                       }\r
-               }\r
-       }               \r
-}\r
-\r
-function isQTInstalled() {\r
-       var qtInstalled = false;\r
-       qtObj = false;\r
-       if (navigator.plugins && navigator.plugins.length) {\r
-               for (var i=0; i < navigator.plugins.length; i++ ) {\r
-         var plugin = navigator.plugins[i];\r
-         if (plugin.name.indexOf("QuickTime") > -1) {\r
-                       qtInstalled = true;\r
-         }\r
-      }\r
-       } else {\r
-               execScript('on error resume next: qtObj = IsObject(CreateObject("QuickTimeCheckObject.QuickTimeCheck.1"))','VBScript');\r
-               qtInstalled = qtObj;\r
-       }\r
-       return qtInstalled;\r
-}\r
-\r
-/* get value of querystring param */\r
-function getQueryParamValue(param) {\r
-       var q = document.location.search;\r
-       var detectIndex = q.indexOf(param);\r
-       var endIndex = (q.indexOf("&", detectIndex) != -1) ? q.indexOf("&", detectIndex) : q.length;\r
-       if(q.length > 1 && detectIndex != -1) {\r
-               return q.substring(q.indexOf("=", detectIndex)+1, endIndex);\r
-       } else {\r
-               return "";\r
-       }\r
-}\r
+ *  myQTObject.addParam("controller", "false");              // optional
+ *     myQTObject.write();
+ *
+ */
+
+QTObject = function(mov, id, w, h) {
+       this.mov = mov;
+       this.id = id;
+       this.width = w;
+       this.height = h;
+       this.redirect = "";
+       this.sq = document.location.search.split("?")[1] || "";
+       this.altTxt = "This content requires the QuickTime Plugin. <a href='http://www.apple.com/quicktime/download/'>Download QuickTime Player</a>.";
+       this.bypassTxt = "<p>Already have QuickTime Player? <a href='?detectqt=false&"+ this.sq +"'>Click here.</a></p>";
+       this.params = new Object();
+       this.doDetect = getQueryParamValue('detectqt');
+}
+
+QTObject.prototype.addParam = function(name, value) {
+       this.params[name] = value;
+}
+
+QTObject.prototype.getParams = function() {
+    return this.params;
+}
+
+QTObject.prototype.getParam = function(name) {
+    return this.params[name];
+}
+
+QTObject.prototype.getParamTags = function() {
+    var paramTags = "";
+    for (var param in this.getParams()) {
+        paramTags += '<param name="' + param + '" value="' + this.getParam(param) + '" />';
+    }
+    if (paramTags == "") {
+        paramTags = null;
+    }
+    return paramTags;
+}
+
+QTObject.prototype.getHTML = function() {
+    var qtHTML = "";
+       if (navigator.plugins && navigator.plugins.length) { // not ie
+        qtHTML += '<embed type="video/quicktime" src="' + this.mov + '" width="' + this.width + '" height="' + this.height + '" id="' + this.id + '"';
+        for (var param in this.getParams()) {
+            qtHTML += ' ' + param + '="' + this.getParam(param) + '"';
+        }
+        qtHTML += '></embed>';
+    }
+    else { // pc ie
+        qtHTML += '<object classid="clsid:02BF25D5-8C17-4B23-BC80-D3488ABDDC6B" width="' + this.width + '" height="' + this.height + '" id="' + this.id + '">';
+        this.addParam("src", this.mov);
+        if (this.getParamTags() != null) {
+            qtHTML += this.getParamTags();
+        }
+        qtHTML += '</object>';
+    }
+    return qtHTML;
+}
+
+
+QTObject.prototype.getVariablePairs = function() {
+    var variablePairs = new Array();
+    for (var name in this.getVariables()) {
+        variablePairs.push(name + "=" + escape(this.getVariable(name)));
+    }
+    if (variablePairs.length > 0) {
+        return variablePairs.join("&");
+    }
+    else {
+        return null;
+    }
+}
+
+QTObject.prototype.write = function(elementId) {
+       if(isQTInstalled() || this.doDetect=='false') {
+               if (elementId) {
+                       document.getElementById(elementId).innerHTML = this.getHTML();
+               } else {
+                       document.write(this.getHTML());
+               }
+       } else {
+               if (this.redirect != "") {
+                       document.location.replace(this.redirect);
+               } else {
+                       if (elementId) {
+                               document.getElementById(elementId).innerHTML = this.altTxt +""+ this.bypassTxt;
+                       } else {
+                               document.write(this.altTxt +""+ this.bypassTxt);
+                       }
+               }
+       }               
+}
+
+function isQTInstalled() {
+       var qtInstalled = false;
+       qtObj = false;
+       if (navigator.plugins && navigator.plugins.length) {
+               for (var i=0; i < navigator.plugins.length; i++ ) {
+         var plugin = navigator.plugins[i];
+         if (plugin.name.indexOf("QuickTime") > -1) {
+                       qtInstalled = true;
+         }
+      }
+       } else {
+               execScript('on error resume next: qtObj = IsObject(CreateObject("QuickTimeCheckObject.QuickTimeCheck.1"))','VBScript');
+               qtInstalled = qtObj;
+       }
+       return qtInstalled;
+}
+
+/* get value of querystring param */
+function getQueryParamValue(param) {
+       var q = document.location.search;
+       var detectIndex = q.indexOf(param);
+       var endIndex = (q.indexOf("&", detectIndex) != -1) ? q.indexOf("&", detectIndex) : q.length;
+       if(q.length > 1 && detectIndex != -1) {
+               return q.substring(q.indexOf("=", detectIndex)+1, endIndex);
+       } else {
+               return "";
+       }
+}
index c393619..189d009 100755 (executable)
@@ -1,4 +1,4 @@
-/*     SWFObject v2.0 rc4 <http://code.google.com/p/swfobject/>\r
-       Copyright (c) 2007 Geoff Stearns, Michael Williams, and Bobby van der Sluis\r
-       This software is released under the MIT License <http://www.opensource.org/licenses/mit-license.php>\r
-*/\r
+/*     SWFObject v2.0 rc4 <http://code.google.com/p/swfobject/>
+       Copyright (c) 2007 Geoff Stearns, Michael Williams, and Bobby van der Sluis
+       This software is released under the MIT License <http://www.opensource.org/licenses/mit-license.php>
+*/
index b8fa5f5..dd3b78b 100755 (executable)
@@ -1,53 +1,53 @@
-/*\r
-       Cookie Plug-in\r
-       \r
-       This plug in automatically gets all the cookies for this site and adds them to the post_params.\r
-       Cookies are loaded only on initialization.  The refreshCookies function can be called to update the post_params.\r
-       The cookies will override any other post params with the same name.\r
-*/\r
-\r
-var SWFUpload;\r
-if (typeof(SWFUpload) === "function") {\r
-       SWFUpload.prototype.initSettings = function (oldInitSettings) {\r
-               return function () {\r
-                       if (typeof(oldInitSettings) === "function") {\r
-                               oldInitSettings.call(this);\r
-                       }\r
-                       \r
-                       this.refreshCookies(false);     // The false parameter must be sent since SWFUpload has not initialzed at this point\r
-               };\r
-       }(SWFUpload.prototype.initSettings);\r
-       \r
-       // refreshes the post_params and updates SWFUpload.  The sendToFlash parameters is optional and defaults to True\r
-       SWFUpload.prototype.refreshCookies = function (sendToFlash) {\r
-               if (sendToFlash === undefined) {\r
-                       sendToFlash = true;\r
-               }\r
-               sendToFlash = !!sendToFlash;\r
-               \r
-               // Get the post_params object\r
-               var postParams = this.settings.post_params;\r
-               \r
-               // Get the cookies\r
-               var i, cookieArray = document.cookie.split(';'), caLength = cookieArray.length, c, eqIndex, name, value;\r
-               for (i = 0; i < caLength; i++) {\r
-                       c = cookieArray[i];\r
-                       \r
-                       // Left Trim spaces\r
-                       while (c.charAt(0) === " ") {\r
-                               c = c.substring(1, c.length);\r
-                       }\r
-                       eqIndex = c.indexOf("=");\r
-                       if (eqIndex > 0) {\r
-                               name = c.substring(0, eqIndex);\r
-                               value = c.substring(eqIndex + 1);\r
-                               postParams[name] = value;\r
-                       }\r
-               }\r
-               \r
-               if (sendToFlash) {\r
-                       this.setPostParams(postParams);\r
-               }\r
-       };\r
-\r
-}\r
+/*
+       Cookie Plug-in
+       
+       This plug in automatically gets all the cookies for this site and adds them to the post_params.
+       Cookies are loaded only on initialization.  The refreshCookies function can be called to update the post_params.
+       The cookies will override any other post params with the same name.
+*/
+
+var SWFUpload;
+if (typeof(SWFUpload) === "function") {
+       SWFUpload.prototype.initSettings = function (oldInitSettings) {
+               return function () {
+                       if (typeof(oldInitSettings) === "function") {
+                               oldInitSettings.call(this);
+                       }
+                       
+                       this.refreshCookies(false);     // The false parameter must be sent since SWFUpload has not initialzed at this point
+               };
+       }(SWFUpload.prototype.initSettings);
+       
+       // refreshes the post_params and updates SWFUpload.  The sendToFlash parameters is optional and defaults to True
+       SWFUpload.prototype.refreshCookies = function (sendToFlash) {
+               if (sendToFlash === undefined) {
+                       sendToFlash = true;
+               }
+               sendToFlash = !!sendToFlash;
+               
+               // Get the post_params object
+               var postParams = this.settings.post_params;
+               
+               // Get the cookies
+               var i, cookieArray = document.cookie.split(';'), caLength = cookieArray.length, c, eqIndex, name, value;
+               for (i = 0; i < caLength; i++) {
+                       c = cookieArray[i];
+                       
+                       // Left Trim spaces
+                       while (c.charAt(0) === " ") {
+                               c = c.substring(1, c.length);
+                       }
+                       eqIndex = c.indexOf("=");
+                       if (eqIndex > 0) {
+                               name = c.substring(0, eqIndex);
+                               value = c.substring(eqIndex + 1);
+                               postParams[name] = value;
+                       }
+               }
+               
+               if (sendToFlash) {
+                       this.setPostParams(postParams);
+               }
+       };
+
+}
index 00aef32..69e619d 100755 (executable)
@@ -1,98 +1,98 @@
-/*\r
-       Queue Plug-in\r
-       \r
-       Features:\r
-               *Adds a cancelQueue() method for cancelling the entire queue.\r
-               *All queued files are uploaded when startUpload() is called.\r
-               *If false is returned from uploadComplete then the queue upload is stopped.\r
-                If false is not returned (strict comparison) then the queue upload is continued.\r
-               *Adds a QueueComplete event that is fired when all the queued files have finished uploading.\r
-                Set the event handler with the queue_complete_handler setting.\r
-               \r
-       */\r
-\r
-var SWFUpload;\r
-if (typeof(SWFUpload) === "function") {\r
-       SWFUpload.queue = {};\r
-       \r
-       SWFUpload.prototype.initSettings = (function (oldInitSettings) {\r
-               return function () {\r
-                       if (typeof(oldInitSettings) === "function") {\r
-                               oldInitSettings.call(this);\r
-                       }\r
-                       \r
-                       this.queueSettings = {};\r
-                       \r
-                       this.queueSettings.queue_cancelled_flag = false;\r
-                       this.queueSettings.queue_upload_count = 0;\r
-                       \r
-                       this.queueSettings.user_upload_complete_handler = this.settings.upload_complete_handler;\r
-                       this.queueSettings.user_upload_start_handler = this.settings.upload_start_handler;\r
-                       this.settings.upload_complete_handler = SWFUpload.queue.uploadCompleteHandler;\r
-                       this.settings.upload_start_handler = SWFUpload.queue.uploadStartHandler;\r
-                       \r
-                       this.settings.queue_complete_handler = this.settings.queue_complete_handler || null;\r
-               };\r
-       })(SWFUpload.prototype.initSettings);\r
-\r
-       SWFUpload.prototype.startUpload = function (fileID) {\r
-               this.queueSettings.queue_cancelled_flag = false;\r
-               this.callFlash("StartUpload", [fileID]);\r
-       };\r
-\r
-       SWFUpload.prototype.cancelQueue = function () {\r
-               this.queueSettings.queue_cancelled_flag = true;\r
-               this.stopUpload();\r
-               \r
-               var stats = this.getStats();\r
-               while (stats.files_queued > 0) {\r
-                       this.cancelUpload();\r
-                       stats = this.getStats();\r
-               }\r
-       };\r
-       \r
-       SWFUpload.queue.uploadStartHandler = function (file) {\r
-               var returnValue;\r
-               if (typeof(this.queueSettings.user_upload_start_handler) === "function") {\r
-                       returnValue = this.queueSettings.user_upload_start_handler.call(this, file);\r
-               }\r
-               \r
-               // To prevent upload a real "FALSE" value must be returned, otherwise default to a real "TRUE" value.\r
-               returnValue = (returnValue === false) ? false : true;\r
-               \r
-               this.queueSettings.queue_cancelled_flag = !returnValue;\r
-\r
-               return returnValue;\r
-       };\r
-       \r
-       SWFUpload.queue.uploadCompleteHandler = function (file) {\r
-               var user_upload_complete_handler = this.queueSettings.user_upload_complete_handler;\r
-               var continueUpload;\r
-               \r
-               if (file.filestatus === SWFUpload.FILE_STATUS.COMPLETE) {\r
-                       this.queueSettings.queue_upload_count++;\r
-               }\r
-\r
-               if (typeof(user_upload_complete_handler) === "function") {\r
-                       continueUpload = (user_upload_complete_handler.call(this, file) === false) ? false : true;\r
-               } else if (file.filestatus === SWFUpload.FILE_STATUS.QUEUED) {\r
-                       // If the file was stopped and re-queued don't restart the upload\r
-                       continueUpload = false;\r
-               } else {\r
-                       continueUpload = true;\r
-               }\r
-               \r
-               if (continueUpload) {\r
-                       var stats = this.getStats();\r
-                       if (stats.files_queued > 0 && this.queueSettings.queue_cancelled_flag === false) {\r
-                               this.startUpload();\r
-                       } else if (this.queueSettings.queue_cancelled_flag === false) {\r
-                               this.queueEvent("queue_complete_handler", [this.queueSettings.queue_upload_count]);\r
-                               this.queueSettings.queue_upload_count = 0;\r
-                       } else {\r
-                               this.queueSettings.queue_cancelled_flag = false;\r
-                               this.queueSettings.queue_upload_count = 0;\r
-                       }\r
-               }\r
-       };\r
+/*
+       Queue Plug-in
+       
+       Features:
+               *Adds a cancelQueue() method for cancelling the entire queue.
+               *All queued files are uploaded when startUpload() is called.
+               *If false is returned from uploadComplete then the queue upload is stopped.
+                If false is not returned (strict comparison) then the queue upload is continued.
+               *Adds a QueueComplete event that is fired when all the queued files have finished uploading.
+                Set the event handler with the queue_complete_handler setting.
+               
+       */
+
+var SWFUpload;
+if (typeof(SWFUpload) === "function") {
+       SWFUpload.queue = {};
+       
+       SWFUpload.prototype.initSettings = (function (oldInitSettings) {
+               return function () {
+                       if (typeof(oldInitSettings) === "function") {
+                               oldInitSettings.call(this);
+                       }
+                       
+                       this.queueSettings = {};
+                       
+                       this.queueSettings.queue_cancelled_flag = false;
+                       this.queueSettings.queue_upload_count = 0;
+                       
+                       this.queueSettings.user_upload_complete_handler = this.settings.upload_complete_handler;
+                       this.queueSettings.user_upload_start_handler = this.settings.upload_start_handler;
+                       this.settings.upload_complete_handler = SWFUpload.queue.uploadCompleteHandler;
+                       this.settings.upload_start_handler = SWFUpload.queue.uploadStartHandler;
+                       
+                       this.settings.queue_complete_handler = this.settings.queue_complete_handler || null;
+               };
+       })(SWFUpload.prototype.initSettings);
+
+       SWFUpload.prototype.startUpload = function (fileID) {
+               this.queueSettings.queue_cancelled_flag = false;
+               this.callFlash("StartUpload", [fileID]);
+       };
+
+       SWFUpload.prototype.cancelQueue = function () {
+               this.queueSettings.queue_cancelled_flag = true;
+               this.stopUpload();
+               
+               var stats = this.getStats();
+               while (stats.files_queued > 0) {
+                       this.cancelUpload();
+                       stats = this.getStats();
+               }
+       };
+       
+       SWFUpload.queue.uploadStartHandler = function (file) {
+               var returnValue;
+               if (typeof(this.queueSettings.user_upload_start_handler) === "function") {
+                       returnValue = this.queueSettings.user_upload_start_handler.call(this, file);
+               }
+               
+               // To prevent upload a real "FALSE" value must be returned, otherwise default to a real "TRUE" value.
+               returnValue = (returnValue === false) ? false : true;
+               
+               this.queueSettings.queue_cancelled_flag = !returnValue;
+
+               return returnValue;
+       };
+       
+       SWFUpload.queue.uploadCompleteHandler = function (file) {
+               var user_upload_complete_handler = this.queueSettings.user_upload_complete_handler;
+               var continueUpload;
+               
+               if (file.filestatus === SWFUpload.FILE_STATUS.COMPLETE) {
+                       this.queueSettings.queue_upload_count++;
+               }
+
+               if (typeof(user_upload_complete_handler) === "function") {
+                       continueUpload = (user_upload_complete_handler.call(this, file) === false) ? false : true;
+               } else if (file.filestatus === SWFUpload.FILE_STATUS.QUEUED) {
+                       // If the file was stopped and re-queued don't restart the upload
+                       continueUpload = false;
+               } else {
+                       continueUpload = true;
+               }
+               
+               if (continueUpload) {
+                       var stats = this.getStats();
+                       if (stats.files_queued > 0 && this.queueSettings.queue_cancelled_flag === false) {
+                               this.startUpload();
+                       } else if (this.queueSettings.queue_cancelled_flag === false) {
+                               this.queueEvent("queue_complete_handler", [this.queueSettings.queue_upload_count]);
+                               this.queueSettings.queue_upload_count = 0;
+                       } else {
+                               this.queueSettings.queue_cancelled_flag = false;
+                               this.queueSettings.queue_upload_count = 0;
+                       }
+               }
+       };
 }
\ No newline at end of file
index 6b7bf94..3245c9c 100755 (executable)
-/*\r
-       Speed Plug-in\r
-       \r
-       Features:\r
-               *Adds several properties to the 'file' object indicated upload speed, time left, upload time, etc.\r
-                       - currentSpeed -- String indicating the upload speed, bytes per second\r
-                       - averageSpeed -- Overall average upload speed, bytes per second\r
-                       - movingAverageSpeed -- Speed over averaged over the last several measurements, bytes per second\r
-                       - timeRemaining -- Estimated remaining upload time in seconds\r
-                       - timeElapsed -- Number of seconds passed for this upload\r
-                       - percentUploaded -- Percentage of the file uploaded (0 to 100)\r
-                       - sizeUploaded -- Formatted size uploaded so far, bytes\r
-               \r
-               *Adds setting 'moving_average_history_size' for defining the window size used to calculate the moving average speed.\r
-               \r
-               *Adds several Formatting functions for formatting that values provided on the file object.\r
-                       - SWFUpload.speed.formatBPS(bps) -- outputs string formatted in the best units (Gbps, Mbps, Kbps, bps)\r
-                       - SWFUpload.speed.formatTime(seconds) -- outputs string formatted in the best units (x Hr y M z S)\r
-                       - SWFUpload.speed.formatSize(bytes) -- outputs string formatted in the best units (w GB x MB y KB z B )\r
-                       - SWFUpload.speed.formatPercent(percent) -- outputs string formatted with a percent sign (x.xx %)\r
-                       - SWFUpload.speed.formatUnits(baseNumber, divisionArray, unitLabelArray, fractionalBoolean)\r
-                               - Formats a number using the division array to determine how to apply the labels in the Label Array\r
-                               - factionalBoolean indicates whether the number should be returned as a single fractional number with a unit (speed)\r
-                                   or as several numbers labeled with units (time)\r
-       */\r
-\r
-var SWFUpload;\r
-if (typeof(SWFUpload) === "function") {\r
-       SWFUpload.speed = {};\r
-       \r
-       SWFUpload.prototype.initSettings = (function (oldInitSettings) {\r
-               return function () {\r
-                       if (typeof(oldInitSettings) === "function") {\r
-                               oldInitSettings.call(this);\r
-                       }\r
-                       \r
-                       this.ensureDefault = function (settingName, defaultValue) {\r
-                               this.settings[settingName] = (this.settings[settingName] == undefined) ? defaultValue : this.settings[settingName];\r
-                       };\r
-\r
-                       // List used to keep the speed stats for the files we are tracking\r
-                       this.fileSpeedStats = {};\r
-                       this.speedSettings = {};\r
-\r
-                       this.ensureDefault("moving_average_history_size", "10");\r
-                       \r
-                       this.speedSettings.user_file_queued_handler = this.settings.file_queued_handler;\r
-                       this.speedSettings.user_file_queue_error_handler = this.settings.file_queue_error_handler;\r
-                       this.speedSettings.user_upload_start_handler = this.settings.upload_start_handler;\r
-                       this.speedSettings.user_upload_error_handler = this.settings.upload_error_handler;\r
-                       this.speedSettings.user_upload_progress_handler = this.settings.upload_progress_handler;\r
-                       this.speedSettings.user_upload_success_handler = this.settings.upload_success_handler;\r
-                       this.speedSettings.user_upload_complete_handler = this.settings.upload_complete_handler;\r
-                       \r
-                       this.settings.file_queued_handler = SWFUpload.speed.fileQueuedHandler;\r
-                       this.settings.file_queue_error_handler = SWFUpload.speed.fileQueueErrorHandler;\r
-                       this.settings.upload_start_handler = SWFUpload.speed.uploadStartHandler;\r
-                       this.settings.upload_error_handler = SWFUpload.speed.uploadErrorHandler;\r
-                       this.settings.upload_progress_handler = SWFUpload.speed.uploadProgressHandler;\r
-                       this.settings.upload_success_handler = SWFUpload.speed.uploadSuccessHandler;\r
-                       this.settings.upload_complete_handler = SWFUpload.speed.uploadCompleteHandler;\r
-                       \r
-                       delete this.ensureDefault;\r
-               };\r
-       })(SWFUpload.prototype.initSettings);\r
-\r
-       \r
-       SWFUpload.speed.fileQueuedHandler = function (file) {\r
-               if (typeof this.speedSettings.user_file_queued_handler === "function") {\r
-                       file = SWFUpload.speed.extendFile(file);\r
-                       \r
-                       return this.speedSettings.user_file_queued_handler.call(this, file);\r
-               }\r
-       };\r
-       \r
-       SWFUpload.speed.fileQueueErrorHandler = function (file, errorCode, message) {\r
-               if (typeof this.speedSettings.user_file_queue_error_handler === "function") {\r
-                       file = SWFUpload.speed.extendFile(file);\r
-                       \r
-                       return this.speedSettings.user_file_queue_error_handler.call(this, file, errorCode, message);\r
-               }\r
-       };\r
-\r
-       SWFUpload.speed.uploadStartHandler = function (file) {\r
-               if (typeof this.speedSettings.user_upload_start_handler === "function") {\r
-                       file = SWFUpload.speed.extendFile(file, this.fileSpeedStats);\r
-                       return this.speedSettings.user_upload_start_handler.call(this, file);\r
-               }\r
-       };\r
-       \r
-       SWFUpload.speed.uploadErrorHandler = function (file, errorCode, message) {\r
-               file = SWFUpload.speed.extendFile(file, this.fileSpeedStats);\r
-               SWFUpload.speed.removeTracking(file, this.fileSpeedStats);\r
-\r
-               if (typeof this.speedSettings.user_upload_error_handler === "function") {\r
-                       return this.speedSettings.user_upload_error_handler.call(this, file, errorCode, message);\r
-               }\r
-       };\r
-       SWFUpload.speed.uploadProgressHandler = function (file, bytesComplete, bytesTotal) {\r
-               this.updateTracking(file, bytesComplete);\r
-               file = SWFUpload.speed.extendFile(file, this.fileSpeedStats);\r
-\r
-               if (typeof this.speedSettings.user_upload_progress_handler === "function") {\r
-                       return this.speedSettings.user_upload_progress_handler.call(this, file, bytesComplete, bytesTotal);\r
-               }\r
-       };\r
-       \r
-       SWFUpload.speed.uploadSuccessHandler = function (file, serverData) {\r
-               if (typeof this.speedSettings.user_upload_success_handler === "function") {\r
-                       file = SWFUpload.speed.extendFile(file, this.fileSpeedStats);\r
-                       return this.speedSettings.user_upload_success_handler.call(this, file, serverData);\r
-               }\r
-       };\r
-       SWFUpload.speed.uploadCompleteHandler = function (file) {\r
-               file = SWFUpload.speed.extendFile(file, this.fileSpeedStats);\r
-               SWFUpload.speed.removeTracking(file, this.fileSpeedStats);\r
-\r
-               if (typeof this.speedSettings.user_upload_complete_handler === "function") {\r
-                       return this.speedSettings.user_upload_complete_handler.call(this, file);\r
-               }\r
-       };\r
-       \r
-       // Private: extends the file object with the speed plugin values\r
-       SWFUpload.speed.extendFile = function (file, trackingList) {\r
-               var tracking;\r
-               \r
-               if (trackingList) {\r
-                       tracking = trackingList[file.id];\r
-               }\r
-               \r
-               if (tracking) {\r
-                       file.currentSpeed = tracking.currentSpeed;\r
-                       file.averageSpeed = tracking.averageSpeed;\r
-                       file.movingAverageSpeed = tracking.movingAverageSpeed;\r
-                       file.timeRemaining = tracking.timeRemaining;\r
-                       file.timeElapsed = tracking.timeElapsed;\r
-                       file.percentUploaded = tracking.percentUploaded;\r
-                       file.sizeUploaded = tracking.bytesUploaded;\r
-\r
-               } else {\r
-                       file.currentSpeed = 0;\r
-                       file.averageSpeed = 0;\r
-                       file.movingAverageSpeed = 0;\r
-                       file.timeRemaining = 0;\r
-                       file.timeElapsed = 0;\r
-                       file.percentUploaded = 0;\r
-                       file.sizeUploaded = 0;\r
-               }\r
-               \r
-               return file;\r
-       };\r
-       \r
-       // Private: Updates the speed tracking object, or creates it if necessary\r
-       SWFUpload.prototype.updateTracking = function (file, bytesUploaded) {\r
-               var tracking = this.fileSpeedStats[file.id];\r
-               if (!tracking) {\r
-                       this.fileSpeedStats[file.id] = tracking = {};\r
-               }\r
-               \r
-               // Sanity check inputs\r
-               bytesUploaded = bytesUploaded || tracking.bytesUploaded || 0;\r
-               if (bytesUploaded < 0) {\r
-                       bytesUploaded = 0;\r
-               }\r
-               if (bytesUploaded > file.size) {\r
-                       bytesUploaded = file.size;\r
-               }\r
-               \r
-               var tickTime = (new Date()).getTime();\r
-               if (!tracking.startTime) {\r
-                       tracking.startTime = (new Date()).getTime();\r
-                       tracking.lastTime = tracking.startTime;\r
-                       tracking.currentSpeed = 0;\r
-                       tracking.averageSpeed = 0;\r
-                       tracking.movingAverageSpeed = 0;\r
-                       tracking.movingAverageHistory = [];\r
-                       tracking.timeRemaining = 0;\r
-                       tracking.timeElapsed = 0;\r
-                       tracking.percentUploaded = bytesUploaded / file.size;\r
-                       tracking.bytesUploaded = bytesUploaded;\r
-               } else if (tracking.startTime > tickTime) {\r
-                       this.debug("When backwards in time");\r
-               } else {\r
-                       // Get time and deltas\r
-                       var now = (new Date()).getTime();\r
-                       var lastTime = tracking.lastTime;\r
-                       var deltaTime = now - lastTime;\r
-                       var deltaBytes = bytesUploaded - tracking.bytesUploaded;\r
-                       \r
-                       if (deltaBytes === 0 || deltaTime === 0) {\r
-                               return tracking;\r
-                       }\r
-                       \r
-                       // Update tracking object\r
-                       tracking.lastTime = now;\r
-                       tracking.bytesUploaded = bytesUploaded;\r
-                       \r
-                       // Calculate speeds\r
-                       tracking.currentSpeed = (deltaBytes * 8 ) / (deltaTime / 1000);\r
-                       tracking.averageSpeed = (tracking.bytesUploaded * 8) / ((now - tracking.startTime) / 1000);\r
-\r
-                       // Calculate moving average\r
-                       tracking.movingAverageHistory.push(tracking.currentSpeed);\r
-                       if (tracking.movingAverageHistory.length > this.settings.moving_average_history_size) {\r
-                               tracking.movingAverageHistory.shift();\r
-                       }\r
-                       \r
-                       tracking.movingAverageSpeed = SWFUpload.speed.calculateMovingAverage(tracking.movingAverageHistory);\r
-                       \r
-                       // Update times\r
-                       tracking.timeRemaining = (file.size - tracking.bytesUploaded) * 8 / tracking.movingAverageSpeed;\r
-                       tracking.timeElapsed = (now - tracking.startTime) / 1000;\r
-                       \r
-                       // Update percent\r
-                       tracking.percentUploaded = (tracking.bytesUploaded / file.size * 100);\r
-               }\r
-               \r
-               return tracking;\r
-       };\r
-       SWFUpload.speed.removeTracking = function (file, trackingList) {\r
-               try {\r
-                       trackingList[file.id] = null;\r
-                       delete trackingList[file.id];\r
-               } catch (ex) {\r
-               }\r
-       };\r
-       \r
-       SWFUpload.speed.formatUnits = function (baseNumber, unitDivisors, unitLabels, singleFractional) {\r
-               var i, unit, unitDivisor, unitLabel;\r
-\r
-               if (baseNumber === 0) {\r
-                       return "0 " + unitLabels[unitLabels.length - 1];\r
-               }\r
-               \r
-               if (singleFractional) {\r
-                       unit = baseNumber;\r
-                       unitLabel = unitLabels.length >= unitDivisors.length ? unitLabels[unitDivisors.length - 1] : "";\r
-                       for (i = 0; i < unitDivisors.length; i++) {\r
-                               if (baseNumber >= unitDivisors[i]) {\r
-                                       unit = (baseNumber / unitDivisors[i]).toFixed(2);\r
-                                       unitLabel = unitLabels.length >= i ? " " + unitLabels[i] : "";\r
-                                       break;\r
-                               }\r
-                       }\r
-                       \r
-                       return unit + unitLabel;\r
-               } else {\r
-                       var formattedStrings = [];\r
-                       var remainder = baseNumber;\r
-                       \r
-                       for (i = 0; i < unitDivisors.length; i++) {\r
-                               unitDivisor = unitDivisors[i];\r
-                               unitLabel = unitLabels.length > i ? " " + unitLabels[i] : "";\r
-                               \r
-                               unit = remainder / unitDivisor;\r
-                               if (i < unitDivisors.length -1) {\r
-                                       unit = Math.floor(unit);\r
-                               } else {\r
-                                       unit = unit.toFixed(2);\r
-                               }\r
-                               if (unit > 0) {\r
-                                       remainder = remainder % unitDivisor;\r
-                                       \r
-                                       formattedStrings.push(unit + unitLabel);\r
-                               }\r
-                       }\r
-                       \r
-                       return formattedStrings.join(" ");\r
-               }\r
-       };\r
-       \r
-       SWFUpload.speed.formatBPS = function (baseNumber) {\r
-               var bpsUnits = [1073741824, 1048576, 1024, 1], bpsUnitLabels = ["Gbps", "Mbps", "Kbps", "bps"];\r
-               return SWFUpload.speed.formatUnits(baseNumber, bpsUnits, bpsUnitLabels, true);\r
-       \r
-       };\r
-       SWFUpload.speed.formatTime = function (baseNumber) {\r
-               var timeUnits = [86400, 3600, 60, 1], timeUnitLabels = ["d", "h", "m", "s"];\r
-               return SWFUpload.speed.formatUnits(baseNumber, timeUnits, timeUnitLabels, false);\r
-       \r
-       };\r
-       SWFUpload.speed.formatBytes = function (baseNumber) {\r
-               var sizeUnits = [1073741824, 1048576, 1024, 1], sizeUnitLabels = ["GB", "MB", "KB", "bytes"];\r
-               return SWFUpload.speed.formatUnits(baseNumber, sizeUnits, sizeUnitLabels, true);\r
-       \r
-       };\r
-       SWFUpload.speed.formatPercent = function (baseNumber) {\r
-               return baseNumber.toFixed(2) + " %";\r
-       };\r
-       \r
-       SWFUpload.speed.calculateMovingAverage = function (history) {\r
-               var vals = [], size, sum = 0.0, mean = 0.0, varianceTemp = 0.0, variance = 0.0, standardDev = 0.0;\r
-               var i;\r
-               var mSum = 0, mCount = 0;\r
-               \r
-               size = history.length;\r
-               \r
-               // Check for sufficient data\r
-               if (size >= 8) {\r
-                       // Clone the array and Calculate sum of the values \r
-                       for (i = 0; i < size; i++) {\r
-                               vals[i] = history[i];\r
-                               sum += vals[i];\r
-                       }\r
-\r
-                       mean = sum / size;\r
-\r
-                       // Calculate variance for the set\r
-                       for (i = 0; i < size; i++) {\r
-                               varianceTemp += Math.pow((vals[i] - mean), 2);\r
-                       }\r
-\r
-                       variance = varianceTemp / size;\r
-                       standardDev = Math.sqrt(variance);\r
-                       \r
-                       //Standardize the Data\r
-                       for (i = 0; i < size; i++) {\r
-                               vals[i] = (vals[i] - mean) / standardDev;\r
-                       }\r
-\r
-                       // Calculate the average excluding outliers\r
-                       var deviationRange = 2.0;\r
-                       for (i = 0; i < size; i++) {\r
-                               \r
-                               if (vals[i] <= deviationRange && vals[i] >= -deviationRange) {\r
-                                       mCount++;\r
-                                       mSum += history[i];\r
-                               }\r
-                       }\r
-                       \r
-               } else {\r
-                       // Calculate the average (not enough data points to remove outliers)\r
-                       mCount = size;\r
-                       for (i = 0; i < size; i++) {\r
-                               mSum += history[i];\r
-                       }\r
-               }\r
-\r
-               return mSum / mCount;\r
-       };\r
-       \r
+/*
+       Speed Plug-in
+       
+       Features:
+               *Adds several properties to the 'file' object indicated upload speed, time left, upload time, etc.
+                       - currentSpeed -- String indicating the upload speed, bytes per second
+                       - averageSpeed -- Overall average upload speed, bytes per second
+                       - movingAverageSpeed -- Speed over averaged over the last several measurements, bytes per second
+                       - timeRemaining -- Estimated remaining upload time in seconds
+                       - timeElapsed -- Number of seconds passed for this upload
+                       - percentUploaded -- Percentage of the file uploaded (0 to 100)
+                       - sizeUploaded -- Formatted size uploaded so far, bytes
+               
+               *Adds setting 'moving_average_history_size' for defining the window size used to calculate the moving average speed.
+               
+               *Adds several Formatting functions for formatting that values provided on the file object.
+                       - SWFUpload.speed.formatBPS(bps) -- outputs string formatted in the best units (Gbps, Mbps, Kbps, bps)
+                       - SWFUpload.speed.formatTime(seconds) -- outputs string formatted in the best units (x Hr y M z S)
+                       - SWFUpload.speed.formatSize(bytes) -- outputs string formatted in the best units (w GB x MB y KB z B )
+                       - SWFUpload.speed.formatPercent(percent) -- outputs string formatted with a percent sign (x.xx %)
+                       - SWFUpload.speed.formatUnits(baseNumber, divisionArray, unitLabelArray, fractionalBoolean)
+                               - Formats a number using the division array to determine how to apply the labels in the Label Array
+                               - factionalBoolean indicates whether the number should be returned as a single fractional number with a unit (speed)
+                                   or as several numbers labeled with units (time)
+       */
+
+var SWFUpload;
+if (typeof(SWFUpload) === "function") {
+       SWFUpload.speed = {};
+       
+       SWFUpload.prototype.initSettings = (function (oldInitSettings) {
+               return function () {
+                       if (typeof(oldInitSettings) === "function") {
+                               oldInitSettings.call(this);
+                       }
+                       
+                       this.ensureDefault = function (settingName, defaultValue) {
+                               this.settings[settingName] = (this.settings[settingName] == undefined) ? defaultValue : this.settings[settingName];
+                       };
+
+                       // List used to keep the speed stats for the files we are tracking
+                       this.fileSpeedStats = {};
+                       this.speedSettings = {};
+
+                       this.ensureDefault("moving_average_history_size", "10");
+                       
+                       this.speedSettings.user_file_queued_handler = this.settings.file_queued_handler;
+                       this.speedSettings.user_file_queue_error_handler = this.settings.file_queue_error_handler;
+                       this.speedSettings.user_upload_start_handler = this.settings.upload_start_handler;
+                       this.speedSettings.user_upload_error_handler = this.settings.upload_error_handler;
+                       this.speedSettings.user_upload_progress_handler = this.settings.upload_progress_handler;
+                       this.speedSettings.user_upload_success_handler = this.settings.upload_success_handler;
+                       this.speedSettings.user_upload_complete_handler = this.settings.upload_complete_handler;
+                       
+                       this.settings.file_queued_handler = SWFUpload.speed.fileQueuedHandler;
+                       this.settings.file_queue_error_handler = SWFUpload.speed.fileQueueErrorHandler;
+                       this.settings.upload_start_handler = SWFUpload.speed.uploadStartHandler;
+                       this.settings.upload_error_handler = SWFUpload.speed.uploadErrorHandler;
+                       this.settings.upload_progress_handler = SWFUpload.speed.uploadProgressHandler;
+                       this.settings.upload_success_handler = SWFUpload.speed.uploadSuccessHandler;
+                       this.settings.upload_complete_handler = SWFUpload.speed.uploadCompleteHandler;
+                       
+                       delete this.ensureDefault;
+               };
+       })(SWFUpload.prototype.initSettings);
+
+       
+       SWFUpload.speed.fileQueuedHandler = function (file) {
+               if (typeof this.speedSettings.user_file_queued_handler === "function") {
+                       file = SWFUpload.speed.extendFile(file);
+                       
+                       return this.speedSettings.user_file_queued_handler.call(this, file);
+               }
+       };
+       
+       SWFUpload.speed.fileQueueErrorHandler = function (file, errorCode, message) {
+               if (typeof this.speedSettings.user_file_queue_error_handler === "function") {
+                       file = SWFUpload.speed.extendFile(file);
+                       
+                       return this.speedSettings.user_file_queue_error_handler.call(this, file, errorCode, message);
+               }
+       };
+
+       SWFUpload.speed.uploadStartHandler = function (file) {
+               if (typeof this.speedSettings.user_upload_start_handler === "function") {
+                       file = SWFUpload.speed.extendFile(file, this.fileSpeedStats);
+                       return this.speedSettings.user_upload_start_handler.call(this, file);
+               }
+       };
+       
+       SWFUpload.speed.uploadErrorHandler = function (file, errorCode, message) {
+               file = SWFUpload.speed.extendFile(file, this.fileSpeedStats);
+               SWFUpload.speed.removeTracking(file, this.fileSpeedStats);
+
+               if (typeof this.speedSettings.user_upload_error_handler === "function") {
+                       return this.speedSettings.user_upload_error_handler.call(this, file, errorCode, message);
+               }
+       };
+       SWFUpload.speed.uploadProgressHandler = function (file, bytesComplete, bytesTotal) {
+               this.updateTracking(file, bytesComplete);
+               file = SWFUpload.speed.extendFile(file, this.fileSpeedStats);
+
+               if (typeof this.speedSettings.user_upload_progress_handler === "function") {
+                       return this.speedSettings.user_upload_progress_handler.call(this, file, bytesComplete, bytesTotal);
+               }
+       };
+       
+       SWFUpload.speed.uploadSuccessHandler = function (file, serverData) {
+               if (typeof this.speedSettings.user_upload_success_handler === "function") {
+                       file = SWFUpload.speed.extendFile(file, this.fileSpeedStats);
+                       return this.speedSettings.user_upload_success_handler.call(this, file, serverData);
+               }
+       };
+       SWFUpload.speed.uploadCompleteHandler = function (file) {
+               file = SWFUpload.speed.extendFile(file, this.fileSpeedStats);
+               SWFUpload.speed.removeTracking(file, this.fileSpeedStats);
+
+               if (typeof this.speedSettings.user_upload_complete_handler === "function") {
+                       return this.speedSettings.user_upload_complete_handler.call(this, file);
+               }
+       };
+       
+       // Private: extends the file object with the speed plugin values
+       SWFUpload.speed.extendFile = function (file, trackingList) {
+               var tracking;
+               
+               if (trackingList) {
+                       tracking = trackingList[file.id];
+               }
+               
+               if (tracking) {
+                       file.currentSpeed = tracking.currentSpeed;
+                       file.averageSpeed = tracking.averageSpeed;
+                       file.movingAverageSpeed = tracking.movingAverageSpeed;
+                       file.timeRemaining = tracking.timeRemaining;
+                       file.timeElapsed = tracking.timeElapsed;
+                       file.percentUploaded = tracking.percentUploaded;
+                       file.sizeUploaded = tracking.bytesUploaded;
+
+               } else {
+                       file.currentSpeed = 0;
+                       file.averageSpeed = 0;
+                       file.movingAverageSpeed = 0;
+                       file.timeRemaining = 0;
+                       file.timeElapsed = 0;
+                       file.percentUploaded = 0;
+                       file.sizeUploaded = 0;
+               }
+               
+               return file;
+       };
+       
+       // Private: Updates the speed tracking object, or creates it if necessary
+       SWFUpload.prototype.updateTracking = function (file, bytesUploaded) {
+               var tracking = this.fileSpeedStats[file.id];
+               if (!tracking) {
+                       this.fileSpeedStats[file.id] = tracking = {};
+               }
+               
+               // Sanity check inputs
+               bytesUploaded = bytesUploaded || tracking.bytesUploaded || 0;
+               if (bytesUploaded < 0) {
+                       bytesUploaded = 0;
+               }
+               if (bytesUploaded > file.size) {
+                       bytesUploaded = file.size;
+               }
+               
+               var tickTime = (new Date()).getTime();
+               if (!tracking.startTime) {
+                       tracking.startTime = (new Date()).getTime();
+                       tracking.lastTime = tracking.startTime;
+                       tracking.currentSpeed = 0;
+                       tracking.averageSpeed = 0;
+                       tracking.movingAverageSpeed = 0;
+                       tracking.movingAverageHistory = [];
+                       tracking.timeRemaining = 0;
+                       tracking.timeElapsed = 0;
+                       tracking.percentUploaded = bytesUploaded / file.size;
+                       tracking.bytesUploaded = bytesUploaded;
+               } else if (tracking.startTime > tickTime) {
+                       this.debug("When backwards in time");
+               } else {
+                       // Get time and deltas
+                       var now = (new Date()).getTime();
+                       var lastTime = tracking.lastTime;
+                       var deltaTime = now - lastTime;
+                       var deltaBytes = bytesUploaded - tracking.bytesUploaded;
+                       
+                       if (deltaBytes === 0 || deltaTime === 0) {
+                               return tracking;
+                       }
+                       
+                       // Update tracking object
+                       tracking.lastTime = now;
+                       tracking.bytesUploaded = bytesUploaded;
+                       
+                       // Calculate speeds
+                       tracking.currentSpeed = (deltaBytes * 8 ) / (deltaTime / 1000);
+                       tracking.averageSpeed = (tracking.bytesUploaded * 8) / ((now - tracking.startTime) / 1000);
+
+                       // Calculate moving average
+                       tracking.movingAverageHistory.push(tracking.currentSpeed);
+                       if (tracking.movingAverageHistory.length > this.settings.moving_average_history_size) {
+                               tracking.movingAverageHistory.shift();
+                       }
+                       
+                       tracking.movingAverageSpeed = SWFUpload.speed.calculateMovingAverage(tracking.movingAverageHistory);
+                       
+                       // Update times
+                       tracking.timeRemaining = (file.size - tracking.bytesUploaded) * 8 / tracking.movingAverageSpeed;
+                       tracking.timeElapsed = (now - tracking.startTime) / 1000;
+                       
+                       // Update percent
+                       tracking.percentUploaded = (tracking.bytesUploaded / file.size * 100);
+               }
+               
+               return tracking;
+       };
+       SWFUpload.speed.removeTracking = function (file, trackingList) {
+               try {
+                       trackingList[file.id] = null;
+                       delete trackingList[file.id];
+               } catch (ex) {
+               }
+       };
+       
+       SWFUpload.speed.formatUnits = function (baseNumber, unitDivisors, unitLabels, singleFractional) {
+               var i, unit, unitDivisor, unitLabel;
+
+               if (baseNumber === 0) {
+                       return "0 " + unitLabels[unitLabels.length - 1];
+               }
+               
+               if (singleFractional) {
+                       unit = baseNumber;
+                       unitLabel = unitLabels.length >= unitDivisors.length ? unitLabels[unitDivisors.length - 1] : "";
+                       for (i = 0; i < unitDivisors.length; i++) {
+                               if (baseNumber >= unitDivisors[i]) {
+                                       unit = (baseNumber / unitDivisors[i]).toFixed(2);
+                                       unitLabel = unitLabels.length >= i ? " " + unitLabels[i] : "";
+                                       break;
+                               }
+                       }
+                       
+                       return unit + unitLabel;
+               } else {
+                       var formattedStrings = [];
+                       var remainder = baseNumber;
+                       
+                       for (i = 0; i < unitDivisors.length; i++) {
+                               unitDivisor = unitDivisors[i];
+                               unitLabel = unitLabels.length > i ? " " + unitLabels[i] : "";
+                               
+                               unit = remainder / unitDivisor;
+                               if (i < unitDivisors.length -1) {
+                                       unit = Math.floor(unit);
+                               } else {
+                                       unit = unit.toFixed(2);
+                               }
+                               if (unit > 0) {
+                                       remainder = remainder % unitDivisor;
+                                       
+                                       formattedStrings.push(unit + unitLabel);
+                               }
+                       }
+                       
+                       return formattedStrings.join(" ");
+               }
+       };
+       
+       SWFUpload.speed.formatBPS = function (baseNumber) {
+               var bpsUnits = [1073741824, 1048576, 1024, 1], bpsUnitLabels = ["Gbps", "Mbps", "Kbps", "bps"];
+               return SWFUpload.speed.formatUnits(baseNumber, bpsUnits, bpsUnitLabels, true);
+       
+       };
+       SWFUpload.speed.formatTime = function (baseNumber) {
+               var timeUnits = [86400, 3600, 60, 1], timeUnitLabels = ["d", "h", "m", "s"];
+               return SWFUpload.speed.formatUnits(baseNumber, timeUnits, timeUnitLabels, false);
+       
+       };
+       SWFUpload.speed.formatBytes = function (baseNumber) {
+               var sizeUnits = [1073741824, 1048576, 1024, 1], sizeUnitLabels = ["GB", "MB", "KB", "bytes"];
+               return SWFUpload.speed.formatUnits(baseNumber, sizeUnits, sizeUnitLabels, true);
+       
+       };
+       SWFUpload.speed.formatPercent = function (baseNumber) {
+               return baseNumber.toFixed(2) + " %";
+       };
+       
+       SWFUpload.speed.calculateMovingAverage = function (history) {
+               var vals = [], size, sum = 0.0, mean = 0.0, varianceTemp = 0.0, variance = 0.0, standardDev = 0.0;
+               var i;
+               var mSum = 0, mCount = 0;
+               
+               size = history.length;
+               
+               // Check for sufficient data
+               if (size >= 8) {
+                       // Clone the array and Calculate sum of the values 
+                       for (i = 0; i < size; i++) {
+                               vals[i] = history[i];
+                               sum += vals[i];
+                       }
+
+                       mean = sum / size;
+
+                       // Calculate variance for the set
+                       for (i = 0; i < size; i++) {
+                               varianceTemp += Math.pow((vals[i] - mean), 2);
+                       }
+
+                       variance = varianceTemp / size;
+                       standardDev = Math.sqrt(variance);
+                       
+                       //Standardize the Data
+                       for (i = 0; i < size; i++) {
+                               vals[i] = (vals[i] - mean) / standardDev;
+                       }
+
+                       // Calculate the average excluding outliers
+                       var deviationRange = 2.0;
+                       for (i = 0; i < size; i++) {
+                               
+                               if (vals[i] <= deviationRange && vals[i] >= -deviationRange) {
+                                       mCount++;
+                                       mSum += history[i];
+                               }
+                       }
+                       
+               } else {
+                       // Calculate the average (not enough data points to remove outliers)
+                       mCount = size;
+                       for (i = 0; i < size; i++) {
+                               mSum += history[i];
+                       }
+               }
+
+               return mSum / mCount;
+       };
+       
 }
\ No newline at end of file
index a75adbf..ca0ca50 100755 (executable)
-/*\r
-       SWFUpload.SWFObject Plugin\r
-\r
-       Summary:\r
-               This plugin uses SWFObject to embed SWFUpload dynamically in the page.  SWFObject provides accurate Flash Player detection and DOM Ready loading.\r
-               This plugin replaces the Graceful Degradation plugin.\r
-\r
-       Features:\r
-               * swfupload_load_failed_hander event\r
-               * swfupload_pre_load_handler event\r
-               * minimum_flash_version setting (default: "9.0.28")\r
-               * SWFUpload.onload event for early loading\r
-\r
-       Usage:\r
-               Provide handlers and settings as needed.  When using the SWFUpload.SWFObject plugin you should initialize SWFUploading\r
-               in SWFUpload.onload rather than in window.onload.  When initialized this way SWFUpload can load earlier preventing the UI flicker\r
-               that was seen using the Graceful Degradation plugin.\r
-\r
-               <script type="text/javascript">\r
-                       var swfu;\r
-                       SWFUpload.onload = function () {\r
-                               swfu = new SWFUpload({\r
-                                       minimum_flash_version: "9.0.28",\r
-                                       swfupload_pre_load_handler: swfuploadPreLoad,\r
-                                       swfupload_load_failed_handler: swfuploadLoadFailed\r
-                               });\r
-                       };\r
-               </script>\r
-               \r
-       Notes:\r
-               You must provide set minimum_flash_version setting to "8" if you are using SWFUpload for Flash Player 8.\r
-               The swfuploadLoadFailed event is only fired if the minimum version of Flash Player is not met.  Other issues such as missing SWF files, browser bugs\r
-                or corrupt Flash Player installations will not trigger this event.\r
-               The swfuploadPreLoad event is fired as soon as the minimum version of Flash Player is found.  It does not wait for SWFUpload to load and can\r
-                be used to prepare the SWFUploadUI and hide alternate content.\r
-               swfobject's onDomReady event is cross-browser safe but will default to the window.onload event when DOMReady is not supported by the browser.\r
-                Early DOM Loading is supported in major modern browsers but cannot be guaranteed for every browser ever made.\r
-*/\r
-\r
-\r
-/* SWFObject v2.1 <http://code.google.com/p/swfobject/>\r
-       Copyright (c) 2007-2008 Geoff Stearns, Michael Williams, and Bobby van der Sluis\r
-       This software is released under the MIT License <http://www.opensource.org/licenses/mit-license.php>\r
-*/\r
-var swfobject=function(){var b="undefined",Q="object",n="Shockwave Flash",p="ShockwaveFlash.ShockwaveFlash",P="application/x-shockwave-flash",m="SWFObjectExprInst",j=window,K=document,T=navigator,o=[],N=[],i=[],d=[],J,Z=null,M=null,l=null,e=false,A=false;var h=function(){var v=typeof K.getElementById!=b&&typeof K.getElementsByTagName!=b&&typeof K.createElement!=b,AC=[0,0,0],x=null;if(typeof T.plugins!=b&&typeof T.plugins[n]==Q){x=T.plugins[n].description;if(x&&!(typeof T.mimeTypes!=b&&T.mimeTypes[P]&&!T.mimeTypes[P].enabledPlugin)){x=x.replace(/^.*\s+(\S+\s+\S+$)/,"$1");AC[0]=parseInt(x.replace(/^(.*)\..*$/,"$1"),10);AC[1]=parseInt(x.replace(/^.*\.(.*)\s.*$/,"$1"),10);AC[2]=/r/.test(x)?parseInt(x.replace(/^.*r(.*)$/,"$1"),10):0}}else{if(typeof j.ActiveXObject!=b){var y=null,AB=false;try{y=new ActiveXObject(p+".7")}catch(t){try{y=new ActiveXObject(p+".6");AC=[6,0,21];y.AllowScriptAccess="always"}catch(t){if(AC[0]==6){AB=true}}if(!AB){try{y=new ActiveXObject(p)}catch(t){}}}if(!AB&&y){try{x=y.GetVariable("$version");if(x){x=x.split(" ")[1].split(",");AC=[parseInt(x[0],10),parseInt(x[1],10),parseInt(x[2],10)]}}catch(t){}}}}var AD=T.userAgent.toLowerCase(),r=T.platform.toLowerCase(),AA=/webkit/.test(AD)?parseFloat(AD.replace(/^.*webkit\/(\d+(\.\d+)?).*$/,"$1")):false,q=false,z=r?/win/.test(r):/win/.test(AD),w=r?/mac/.test(r):/mac/.test(AD);/*@cc_on q=true;@if(@_win32)z=true;@elif(@_mac)w=true;@end@*/return{w3cdom:v,pv:AC,webkit:AA,ie:q,win:z,mac:w}}();var L=function(){if(!h.w3cdom){return }f(H);if(h.ie&&h.win){try{K.write("<script id=__ie_ondomload defer=true src=//:><\/script>");J=C("__ie_ondomload");if(J){I(J,"onreadystatechange",S)}}catch(q){}}if(h.webkit&&typeof K.readyState!=b){Z=setInterval(function(){if(/loaded|complete/.test(K.readyState)){E()}},10)}if(typeof K.addEventListener!=b){K.addEventListener("DOMContentLoaded",E,null)}R(E)}();function S(){if(J.readyState=="complete"){J.parentNode.removeChild(J);E()}}function E(){if(e){return }if(h.ie&&h.win){var v=a("span");try{var u=K.getElementsByTagName("body")[0].appendChild(v);u.parentNode.removeChild(u)}catch(w){return }}e=true;if(Z){clearInterval(Z);Z=null}var q=o.length;for(var r=0;r<q;r++){o[r]()}}function f(q){if(e){q()}else{o[o.length]=q}}function R(r){if(typeof j.addEventListener!=b){j.addEventListener("load",r,false)}else{if(typeof K.addEventListener!=b){K.addEventListener("load",r,false)}else{if(typeof j.attachEvent!=b){I(j,"onload",r)}else{if(typeof j.onload=="function"){var q=j.onload;j.onload=function(){q();r()}}else{j.onload=r}}}}}function H(){var t=N.length;for(var q=0;q<t;q++){var u=N[q].id;if(h.pv[0]>0){var r=C(u);if(r){N[q].width=r.getAttribute("width")?r.getAttribute("width"):"0";N[q].height=r.getAttribute("height")?r.getAttribute("height"):"0";if(c(N[q].swfVersion)){if(h.webkit&&h.webkit<312){Y(r)}W(u,true)}else{if(N[q].expressInstall&&!A&&c("6.0.65")&&(h.win||h.mac)){k(N[q])}else{O(r)}}}}else{W(u,true)}}}function Y(t){var q=t.getElementsByTagName(Q)[0];if(q){var w=a("embed"),y=q.attributes;if(y){var v=y.length;for(var u=0;u<v;u++){if(y[u].nodeName=="DATA"){w.setAttribute("src",y[u].nodeValue)}else{w.setAttribute(y[u].nodeName,y[u].nodeValue)}}}var x=q.childNodes;if(x){var z=x.length;for(var r=0;r<z;r++){if(x[r].nodeType==1&&x[r].nodeName=="PARAM"){w.setAttribute(x[r].getAttribute("name"),x[r].getAttribute("value"))}}}t.parentNode.replaceChild(w,t)}}function k(w){A=true;var u=C(w.id);if(u){if(w.altContentId){var y=C(w.altContentId);if(y){M=y;l=w.altContentId}}else{M=G(u)}if(!(/%$/.test(w.width))&&parseInt(w.width,10)<310){w.width="310"}if(!(/%$/.test(w.height))&&parseInt(w.height,10)<137){w.height="137"}K.title=K.title.slice(0,47)+" - Flash Player Installation";var z=h.ie&&h.win?"ActiveX":"PlugIn",q=K.title,r="MMredirectURL="+j.location+"&MMplayerType="+z+"&MMdoctitle="+q,x=w.id;if(h.ie&&h.win&&u.readyState!=4){var t=a("div");x+="SWFObjectNew";t.setAttribute("id",x);u.parentNode.insertBefore(t,u);u.style.display="none";var v=function(){u.parentNode.removeChild(u)};I(j,"onload",v)}U({data:w.expressInstall,id:m,width:w.width,height:w.height},{flashvars:r},x)}}function O(t){if(h.ie&&h.win&&t.readyState!=4){var r=a("div");t.parentNode.insertBefore(r,t);r.parentNode.replaceChild(G(t),r);t.style.display="none";var q=function(){t.parentNode.removeChild(t)};I(j,"onload",q)}else{t.parentNode.replaceChild(G(t),t)}}function G(v){var u=a("div");if(h.win&&h.ie){u.innerHTML=v.innerHTML}else{var r=v.getElementsByTagName(Q)[0];if(r){var w=r.childNodes;if(w){var q=w.length;for(var t=0;t<q;t++){if(!(w[t].nodeType==1&&w[t].nodeName=="PARAM")&&!(w[t].nodeType==8)){u.appendChild(w[t].cloneNode(true))}}}}}return u}function U(AG,AE,t){var q,v=C(t);if(v){if(typeof AG.id==b){AG.id=t}if(h.ie&&h.win){var AF="";for(var AB in AG){if(AG[AB]!=Object.prototype[AB]){if(AB.toLowerCase()=="data"){AE.movie=AG[AB]}else{if(AB.toLowerCase()=="styleclass"){AF+=' class="'+AG[AB]+'"'}else{if(AB.toLowerCase()!="classid"){AF+=" "+AB+'="'+AG[AB]+'"'}}}}}var AD="";for(var AA in AE){if(AE[AA]!=Object.prototype[AA]){AD+='<param name="'+AA+'" value="'+AE[AA]+'" />'}}v.outerHTML='<object classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"'+AF+">"+AD+"</object>";i[i.length]=AG.id;q=C(AG.id)}else{if(h.webkit&&h.webkit<312){var AC=a("embed");AC.setAttribute("type",P);for(var z in AG){if(AG[z]!=Object.prototype[z]){if(z.toLowerCase()=="data"){AC.setAttribute("src",AG[z])}else{if(z.toLowerCase()=="styleclass"){AC.setAttribute("class",AG[z])}else{if(z.toLowerCase()!="classid"){AC.setAttribute(z,AG[z])}}}}}for(var y in AE){if(AE[y]!=Object.prototype[y]){if(y.toLowerCase()!="movie"){AC.setAttribute(y,AE[y])}}}v.parentNode.replaceChild(AC,v);q=AC}else{var u=a(Q);u.setAttribute("type",P);for(var x in AG){if(AG[x]!=Object.prototype[x]){if(x.toLowerCase()=="styleclass"){u.setAttribute("class",AG[x])}else{if(x.toLowerCase()!="classid"){u.setAttribute(x,AG[x])}}}}for(var w in AE){if(AE[w]!=Object.prototype[w]&&w.toLowerCase()!="movie"){F(u,w,AE[w])}}v.parentNode.replaceChild(u,v);q=u}}}return q}function F(t,q,r){var u=a("param");u.setAttribute("name",q);u.setAttribute("value",r);t.appendChild(u)}function X(r){var q=C(r);if(q&&(q.nodeName=="OBJECT"||q.nodeName=="EMBED")){if(h.ie&&h.win){if(q.readyState==4){B(r)}else{j.attachEvent("onload",function(){B(r)})}}else{q.parentNode.removeChild(q)}}}function B(t){var r=C(t);if(r){for(var q in r){if(typeof r[q]=="function"){r[q]=null}}r.parentNode.removeChild(r)}}function C(t){var q=null;try{q=K.getElementById(t)}catch(r){}return q}function a(q){return K.createElement(q)}function I(t,q,r){t.attachEvent(q,r);d[d.length]=[t,q,r]}function c(t){var r=h.pv,q=t.split(".");q[0]=parseInt(q[0],10);q[1]=parseInt(q[1],10)||0;q[2]=parseInt(q[2],10)||0;return(r[0]>q[0]||(r[0]==q[0]&&r[1]>q[1])||(r[0]==q[0]&&r[1]==q[1]&&r[2]>=q[2]))?true:false}function V(v,r){if(h.ie&&h.mac){return }var u=K.getElementsByTagName("head")[0],t=a("style");t.setAttribute("type","text/css");t.setAttribute("media","screen");if(!(h.ie&&h.win)&&typeof K.createTextNode!=b){t.appendChild(K.createTextNode(v+" {"+r+"}"))}u.appendChild(t);if(h.ie&&h.win&&typeof K.styleSheets!=b&&K.styleSheets.length>0){var q=K.styleSheets[K.styleSheets.length-1];if(typeof q.addRule==Q){q.addRule(v,r)}}}function W(t,q){var r=q?"visible":"hidden";if(e&&C(t)){C(t).style.visibility=r}else{V("#"+t,"visibility:"+r)}}function g(s){var r=/[\\\"<>\.;]/;var q=r.exec(s)!=null;return q?encodeURIComponent(s):s}var D=function(){if(h.ie&&h.win){window.attachEvent("onunload",function(){var w=d.length;for(var v=0;v<w;v++){d[v][0].detachEvent(d[v][1],d[v][2])}var t=i.length;for(var u=0;u<t;u++){X(i[u])}for(var r in h){h[r]=null}h=null;for(var q in swfobject){swfobject[q]=null}swfobject=null})}}();return{registerObject:function(u,q,t){if(!h.w3cdom||!u||!q){return }var r={};r.id=u;r.swfVersion=q;r.expressInstall=t?t:false;N[N.length]=r;W(u,false)},getObjectById:function(v){var q=null;if(h.w3cdom){var t=C(v);if(t){var u=t.getElementsByTagName(Q)[0];if(!u||(u&&typeof t.SetVariable!=b)){q=t}else{if(typeof u.SetVariable!=b){q=u}}}}return q},embedSWF:function(x,AE,AB,AD,q,w,r,z,AC){if(!h.w3cdom||!x||!AE||!AB||!AD||!q){return }AB+="";AD+="";if(c(q)){W(AE,false);var AA={};if(AC&&typeof AC===Q){for(var v in AC){if(AC[v]!=Object.prototype[v]){AA[v]=AC[v]}}}AA.data=x;AA.width=AB;AA.height=AD;var y={};if(z&&typeof z===Q){for(var u in z){if(z[u]!=Object.prototype[u]){y[u]=z[u]}}}if(r&&typeof r===Q){for(var t in r){if(r[t]!=Object.prototype[t]){if(typeof y.flashvars!=b){y.flashvars+="&"+t+"="+r[t]}else{y.flashvars=t+"="+r[t]}}}}f(function(){U(AA,y,AE);if(AA.id==AE){W(AE,true)}})}else{if(w&&!A&&c("6.0.65")&&(h.win||h.mac)){A=true;W(AE,false);f(function(){var AF={};AF.id=AF.altContentId=AE;AF.width=AB;AF.height=AD;AF.expressInstall=w;k(AF)})}}},getFlashPlayerVersion:function(){return{major:h.pv[0],minor:h.pv[1],release:h.pv[2]}},hasFlashPlayerVersion:c,createSWF:function(t,r,q){if(h.w3cdom){return U(t,r,q)}else{return undefined}},removeSWF:function(q){if(h.w3cdom){X(q)}},createCSS:function(r,q){if(h.w3cdom){V(r,q)}},addDomLoadEvent:f,addLoadEvent:R,getQueryParamValue:function(v){var u=K.location.search||K.location.hash;if(v==null){return g(u)}if(u){var t=u.substring(1).split("&");for(var r=0;r<t.length;r++){if(t[r].substring(0,t[r].indexOf("="))==v){return g(t[r].substring((t[r].indexOf("=")+1)))}}}return""},expressInstallCallback:function(){if(A&&M){var q=C(m);if(q){q.parentNode.replaceChild(M,q);if(l){W(l,true);if(h.ie&&h.win){M.style.display="block"}}M=null;l=null;A=false}}}}}();\r
-\r
-\r
-       \r
-var SWFUpload;\r
-if (typeof(SWFUpload) === "function") {\r
-       SWFUpload.onload = function () {};\r
-       \r
-       swfobject.addDomLoadEvent(function () {\r
-               if (typeof(SWFUpload.onload) === "function") {\r
-                       SWFUpload.onload.call(window);\r
-               }\r
-       });\r
-       \r
-       SWFUpload.prototype.initSettings = (function (oldInitSettings) {\r
-               return function () {\r
-                       if (typeof(oldInitSettings) === "function") {\r
-                               oldInitSettings.call(this);\r
-                       }\r
-\r
-                       this.ensureDefault = function (settingName, defaultValue) {\r
-                               this.settings[settingName] = (this.settings[settingName] == undefined) ? defaultValue : this.settings[settingName];\r
-                       };\r
-\r
-                       this.ensureDefault("minimum_flash_version", "9.0.28");\r
-                       this.ensureDefault("swfupload_pre_load_handler", null);\r
-                       this.ensureDefault("swfupload_load_failed_handler", null);\r
-\r
-                       delete this.ensureDefault;\r
-\r
-               };\r
-       })(SWFUpload.prototype.initSettings);\r
-\r
-\r
-       SWFUpload.prototype.loadFlash = function (oldLoadFlash) {\r
-               return function () {\r
-                       var hasFlash = swfobject.hasFlashPlayerVersion(this.settings.minimum_flash_version);\r
-                       \r
-                       if (hasFlash) {\r
-                               this.queueEvent("swfupload_pre_load_handler");\r
-                               if (typeof(oldLoadFlash) === "function") {\r
-                                       oldLoadFlash.call(this);\r
-                               }\r
-                       } else {\r
-                               this.queueEvent("swfupload_load_failed_handler");\r
-                       }\r
-               };\r
-               \r
-       }(SWFUpload.prototype.loadFlash);\r
-                       \r
-       SWFUpload.prototype.displayDebugInfo = function (oldDisplayDebugInfo) {\r
-               return function () {\r
-                       if (typeof(oldDisplayDebugInfo) === "function") {\r
-                               oldDisplayDebugInfo.call(this);\r
-                       }\r
-                       \r
-                       this.debug(\r
-                               [\r
-                                       "SWFUpload.SWFObject Plugin settings:", "\n",\r
-                                       "\t", "minimum_flash_version:                      ", this.settings.minimum_flash_version, "\n",\r
-                                       "\t", "swfupload_pre_load_handler assigned:     ", (typeof(this.settings.swfupload_pre_load_handler) === "function").toString(), "\n",\r
-                                       "\t", "swfupload_load_failed_handler assigned:     ", (typeof(this.settings.swfupload_load_failed_handler) === "function").toString(), "\n",\r
-                               ].join("")\r
-                       );\r
-               };      \r
-       }(SWFUpload.prototype.displayDebugInfo);\r
-}\r
+/*
+       SWFUpload.SWFObject Plugin
+
+       Summary:
+               This plugin uses SWFObject to embed SWFUpload dynamically in the page.  SWFObject provides accurate Flash Player detection and DOM Ready loading.
+               This plugin replaces the Graceful Degradation plugin.
+
+       Features:
+               * swfupload_load_failed_hander event
+               * swfupload_pre_load_handler event
+               * minimum_flash_version setting (default: "9.0.28")
+               * SWFUpload.onload event for early loading
+
+       Usage:
+               Provide handlers and settings as needed.  When using the SWFUpload.SWFObject plugin you should initialize SWFUploading
+               in SWFUpload.onload rather than in window.onload.  When initialized this way SWFUpload can load earlier preventing the UI flicker
+               that was seen using the Graceful Degradation plugin.
+
+               <script type="text/javascript">
+                       var swfu;
+                       SWFUpload.onload = function () {
+                               swfu = new SWFUpload({
+                                       minimum_flash_version: "9.0.28",
+                                       swfupload_pre_load_handler: swfuploadPreLoad,
+                                       swfupload_load_failed_handler: swfuploadLoadFailed
+                               });
+                       };
+               </script>
+               
+       Notes:
+               You must provide set minimum_flash_version setting to "8" if you are using SWFUpload for Flash Player 8.
+               The swfuploadLoadFailed event is only fired if the minimum version of Flash Player is not met.  Other issues such as missing SWF files, browser bugs
+                or corrupt Flash Player installations will not trigger this event.
+               The swfuploadPreLoad event is fired as soon as the minimum version of Flash Player is found.  It does not wait for SWFUpload to load and can
+                be used to prepare the SWFUploadUI and hide alternate content.
+               swfobject's onDomReady event is cross-browser safe but will default to the window.onload event when DOMReady is not supported by the browser.
+                Early DOM Loading is supported in major modern browsers but cannot be guaranteed for every browser ever made.
+*/
+
+
+/* SWFObject v2.1 <http://code.google.com/p/swfobject/>
+       Copyright (c) 2007-2008 Geoff Stearns, Michael Williams, and Bobby van der Sluis
+       This software is released under the MIT License <http://www.opensource.org/licenses/mit-license.php>
+*/
+var swfobject=function(){var b="undefined",Q="object",n="Shockwave Flash",p="ShockwaveFlash.ShockwaveFlash",P="application/x-shockwave-flash",m="SWFObjectExprInst",j=window,K=document,T=navigator,o=[],N=[],i=[],d=[],J,Z=null,M=null,l=null,e=false,A=false;var h=function(){var v=typeof K.getElementById!=b&&typeof K.getElementsByTagName!=b&&typeof K.createElement!=b,AC=[0,0,0],x=null;if(typeof T.plugins!=b&&typeof T.plugins[n]==Q){x=T.plugins[n].description;if(x&&!(typeof T.mimeTypes!=b&&T.mimeTypes[P]&&!T.mimeTypes[P].enabledPlugin)){x=x.replace(/^.*\s+(\S+\s+\S+$)/,"$1");AC[0]=parseInt(x.replace(/^(.*)\..*$/,"$1"),10);AC[1]=parseInt(x.replace(/^.*\.(.*)\s.*$/,"$1"),10);AC[2]=/r/.test(x)?parseInt(x.replace(/^.*r(.*)$/,"$1"),10):0}}else{if(typeof j.ActiveXObject!=b){var y=null,AB=false;try{y=new ActiveXObject(p+".7")}catch(t){try{y=new ActiveXObject(p+".6");AC=[6,0,21];y.AllowScriptAccess="always"}catch(t){if(AC[0]==6){AB=true}}if(!AB){try{y=new ActiveXObject(p)}catch(t){}}}if(!AB&&y){try{x=y.GetVariable("$version");if(x){x=x.split(" ")[1].split(",");AC=[parseInt(x[0],10),parseInt(x[1],10),parseInt(x[2],10)]}}catch(t){}}}}var AD=T.userAgent.toLowerCase(),r=T.platform.toLowerCase(),AA=/webkit/.test(AD)?parseFloat(AD.replace(/^.*webkit\/(\d+(\.\d+)?).*$/,"$1")):false,q=false,z=r?/win/.test(r):/win/.test(AD),w=r?/mac/.test(r):/mac/.test(AD);/*@cc_on q=true;@if(@_win32)z=true;@elif(@_mac)w=true;@end@*/return{w3cdom:v,pv:AC,webkit:AA,ie:q,win:z,mac:w}}();var L=function(){if(!h.w3cdom){return }f(H);if(h.ie&&h.win){try{K.write("<script id=__ie_ondomload defer=true src=//:><\/script>");J=C("__ie_ondomload");if(J){I(J,"onreadystatechange",S)}}catch(q){}}if(h.webkit&&typeof K.readyState!=b){Z=setInterval(function(){if(/loaded|complete/.test(K.readyState)){E()}},10)}if(typeof K.addEventListener!=b){K.addEventListener("DOMContentLoaded",E,null)}R(E)}();function S(){if(J.readyState=="complete"){J.parentNode.removeChild(J);E()}}function E(){if(e){return }if(h.ie&&h.win){var v=a("span");try{var u=K.getElementsByTagName("body")[0].appendChild(v);u.parentNode.removeChild(u)}catch(w){return }}e=true;if(Z){clearInterval(Z);Z=null}var q=o.length;for(var r=0;r<q;r++){o[r]()}}function f(q){if(e){q()}else{o[o.length]=q}}function R(r){if(typeof j.addEventListener!=b){j.addEventListener("load",r,false)}else{if(typeof K.addEventListener!=b){K.addEventListener("load",r,false)}else{if(typeof j.attachEvent!=b){I(j,"onload",r)}else{if(typeof j.onload=="function"){var q=j.onload;j.onload=function(){q();r()}}else{j.onload=r}}}}}function H(){var t=N.length;for(var q=0;q<t;q++){var u=N[q].id;if(h.pv[0]>0){var r=C(u);if(r){N[q].width=r.getAttribute("width")?r.getAttribute("width"):"0";N[q].height=r.getAttribute("height")?r.getAttribute("height"):"0";if(c(N[q].swfVersion)){if(h.webkit&&h.webkit<312){Y(r)}W(u,true)}else{if(N[q].expressInstall&&!A&&c("6.0.65")&&(h.win||h.mac)){k(N[q])}else{O(r)}}}}else{W(u,true)}}}function Y(t){var q=t.getElementsByTagName(Q)[0];if(q){var w=a("embed"),y=q.attributes;if(y){var v=y.length;for(var u=0;u<v;u++){if(y[u].nodeName=="DATA"){w.setAttribute("src",y[u].nodeValue)}else{w.setAttribute(y[u].nodeName,y[u].nodeValue)}}}var x=q.childNodes;if(x){var z=x.length;for(var r=0;r<z;r++){if(x[r].nodeType==1&&x[r].nodeName=="PARAM"){w.setAttribute(x[r].getAttribute("name"),x[r].getAttribute("value"))}}}t.parentNode.replaceChild(w,t)}}function k(w){A=true;var u=C(w.id);if(u){if(w.altContentId){var y=C(w.altContentId);if(y){M=y;l=w.altContentId}}else{M=G(u)}if(!(/%$/.test(w.width))&&parseInt(w.width,10)<310){w.width="310"}if(!(/%$/.test(w.height))&&parseInt(w.height,10)<137){w.height="137"}K.title=K.title.slice(0,47)+" - Flash Player Installation";var z=h.ie&&h.win?"ActiveX":"PlugIn",q=K.title,r="MMredirectURL="+j.location+"&MMplayerType="+z+"&MMdoctitle="+q,x=w.id;if(h.ie&&h.win&&u.readyState!=4){var t=a("div");x+="SWFObjectNew";t.setAttribute("id",x);u.parentNode.insertBefore(t,u);u.style.display="none";var v=function(){u.parentNode.removeChild(u)};I(j,"onload",v)}U({data:w.expressInstall,id:m,width:w.width,height:w.height},{flashvars:r},x)}}function O(t){if(h.ie&&h.win&&t.readyState!=4){var r=a("div");t.parentNode.insertBefore(r,t);r.parentNode.replaceChild(G(t),r);t.style.display="none";var q=function(){t.parentNode.removeChild(t)};I(j,"onload",q)}else{t.parentNode.replaceChild(G(t),t)}}function G(v){var u=a("div");if(h.win&&h.ie){u.innerHTML=v.innerHTML}else{var r=v.getElementsByTagName(Q)[0];if(r){var w=r.childNodes;if(w){var q=w.length;for(var t=0;t<q;t++){if(!(w[t].nodeType==1&&w[t].nodeName=="PARAM")&&!(w[t].nodeType==8)){u.appendChild(w[t].cloneNode(true))}}}}}return u}function U(AG,AE,t){var q,v=C(t);if(v){if(typeof AG.id==b){AG.id=t}if(h.ie&&h.win){var AF="";for(var AB in AG){if(AG[AB]!=Object.prototype[AB]){if(AB.toLowerCase()=="data"){AE.movie=AG[AB]}else{if(AB.toLowerCase()=="styleclass"){AF+=' class="'+AG[AB]+'"'}else{if(AB.toLowerCase()!="classid"){AF+=" "+AB+'="'+AG[AB]+'"'}}}}}var AD="";for(var AA in AE){if(AE[AA]!=Object.prototype[AA]){AD+='<param name="'+AA+'" value="'+AE[AA]+'" />'}}v.outerHTML='<object classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"'+AF+">"+AD+"</object>";i[i.length]=AG.id;q=C(AG.id)}else{if(h.webkit&&h.webkit<312){var AC=a("embed");AC.setAttribute("type",P);for(var z in AG){if(AG[z]!=Object.prototype[z]){if(z.toLowerCase()=="data"){AC.setAttribute("src",AG[z])}else{if(z.toLowerCase()=="styleclass"){AC.setAttribute("class",AG[z])}else{if(z.toLowerCase()!="classid"){AC.setAttribute(z,AG[z])}}}}}for(var y in AE){if(AE[y]!=Object.prototype[y]){if(y.toLowerCase()!="movie"){AC.setAttribute(y,AE[y])}}}v.parentNode.replaceChild(AC,v);q=AC}else{var u=a(Q);u.setAttribute("type",P);for(var x in AG){if(AG[x]!=Object.prototype[x]){if(x.toLowerCase()=="styleclass"){u.setAttribute("class",AG[x])}else{if(x.toLowerCase()!="classid"){u.setAttribute(x,AG[x])}}}}for(var w in AE){if(AE[w]!=Object.prototype[w]&&w.toLowerCase()!="movie"){F(u,w,AE[w])}}v.parentNode.replaceChild(u,v);q=u}}}return q}function F(t,q,r){var u=a("param");u.setAttribute("name",q);u.setAttribute("value",r);t.appendChild(u)}function X(r){var q=C(r);if(q&&(q.nodeName=="OBJECT"||q.nodeName=="EMBED")){if(h.ie&&h.win){if(q.readyState==4){B(r)}else{j.attachEvent("onload",function(){B(r)})}}else{q.parentNode.removeChild(q)}}}function B(t){var r=C(t);if(r){for(var q in r){if(typeof r[q]=="function"){r[q]=null}}r.parentNode.removeChild(r)}}function C(t){var q=null;try{q=K.getElementById(t)}catch(r){}return q}function a(q){return K.createElement(q)}function I(t,q,r){t.attachEvent(q,r);d[d.length]=[t,q,r]}function c(t){var r=h.pv,q=t.split(".");q[0]=parseInt(q[0],10);q[1]=parseInt(q[1],10)||0;q[2]=parseInt(q[2],10)||0;return(r[0]>q[0]||(r[0]==q[0]&&r[1]>q[1])||(r[0]==q[0]&&r[1]==q[1]&&r[2]>=q[2]))?true:false}function V(v,r){if(h.ie&&h.mac){return }var u=K.getElementsByTagName("head")[0],t=a("style");t.setAttribute("type","text/css");t.setAttribute("media","screen");if(!(h.ie&&h.win)&&typeof K.createTextNode!=b){t.appendChild(K.createTextNode(v+" {"+r+"}"))}u.appendChild(t);if(h.ie&&h.win&&typeof K.styleSheets!=b&&K.styleSheets.length>0){var q=K.styleSheets[K.styleSheets.length-1];if(typeof q.addRule==Q){q.addRule(v,r)}}}function W(t,q){var r=q?"visible":"hidden";if(e&&C(t)){C(t).style.visibility=r}else{V("#"+t,"visibility:"+r)}}function g(s){var r=/[\\\"<>\.;]/;var q=r.exec(s)!=null;return q?encodeURIComponent(s):s}var D=function(){if(h.ie&&h.win){window.attachEvent("onunload",function(){var w=d.length;for(var v=0;v<w;v++){d[v][0].detachEvent(d[v][1],d[v][2])}var t=i.length;for(var u=0;u<t;u++){X(i[u])}for(var r in h){h[r]=null}h=null;for(var q in swfobject){swfobject[q]=null}swfobject=null})}}();return{registerObject:function(u,q,t){if(!h.w3cdom||!u||!q){return }var r={};r.id=u;r.swfVersion=q;r.expressInstall=t?t:false;N[N.length]=r;W(u,false)},getObjectById:function(v){var q=null;if(h.w3cdom){var t=C(v);if(t){var u=t.getElementsByTagName(Q)[0];if(!u||(u&&typeof t.SetVariable!=b)){q=t}else{if(typeof u.SetVariable!=b){q=u}}}}return q},embedSWF:function(x,AE,AB,AD,q,w,r,z,AC){if(!h.w3cdom||!x||!AE||!AB||!AD||!q){return }AB+="";AD+="";if(c(q)){W(AE,false);var AA={};if(AC&&typeof AC===Q){for(var v in AC){if(AC[v]!=Object.prototype[v]){AA[v]=AC[v]}}}AA.data=x;AA.width=AB;AA.height=AD;var y={};if(z&&typeof z===Q){for(var u in z){if(z[u]!=Object.prototype[u]){y[u]=z[u]}}}if(r&&typeof r===Q){for(var t in r){if(r[t]!=Object.prototype[t]){if(typeof y.flashvars!=b){y.flashvars+="&"+t+"="+r[t]}else{y.flashvars=t+"="+r[t]}}}}f(function(){U(AA,y,AE);if(AA.id==AE){W(AE,true)}})}else{if(w&&!A&&c("6.0.65")&&(h.win||h.mac)){A=true;W(AE,false);f(function(){var AF={};AF.id=AF.altContentId=AE;AF.width=AB;AF.height=AD;AF.expressInstall=w;k(AF)})}}},getFlashPlayerVersion:function(){return{major:h.pv[0],minor:h.pv[1],release:h.pv[2]}},hasFlashPlayerVersion:c,createSWF:function(t,r,q){if(h.w3cdom){return U(t,r,q)}else{return undefined}},removeSWF:function(q){if(h.w3cdom){X(q)}},createCSS:function(r,q){if(h.w3cdom){V(r,q)}},addDomLoadEvent:f,addLoadEvent:R,getQueryParamValue:function(v){var u=K.location.search||K.location.hash;if(v==null){return g(u)}if(u){var t=u.substring(1).split("&");for(var r=0;r<t.length;r++){if(t[r].substring(0,t[r].indexOf("="))==v){return g(t[r].substring((t[r].indexOf("=")+1)))}}}return""},expressInstallCallback:function(){if(A&&M){var q=C(m);if(q){q.parentNode.replaceChild(M,q);if(l){W(l,true);if(h.ie&&h.win){M.style.display="block"}}M=null;l=null;A=false}}}}}();
+
+
+       
+var SWFUpload;
+if (typeof(SWFUpload) === "function") {
+       SWFUpload.onload = function () {};
+       
+       swfobject.addDomLoadEvent(function () {
+               if (typeof(SWFUpload.onload) === "function") {
+                       SWFUpload.onload.call(window);
+               }
+       });
+       
+       SWFUpload.prototype.initSettings = (function (oldInitSettings) {
+               return function () {
+                       if (typeof(oldInitSettings) === "function") {
+                               oldInitSettings.call(this);
+                       }
+
+                       this.ensureDefault = function (settingName, defaultValue) {
+                               this.settings[settingName] = (this.settings[settingName] == undefined) ? defaultValue : this.settings[settingName];
+                       };
+
+                       this.ensureDefault("minimum_flash_version", "9.0.28");
+                       this.ensureDefault("swfupload_pre_load_handler", null);
+                       this.ensureDefault("swfupload_load_failed_handler", null);
+
+                       delete this.ensureDefault;
+
+               };
+       })(SWFUpload.prototype.initSettings);
+
+
+       SWFUpload.prototype.loadFlash = function (oldLoadFlash) {
+               return function () {
+                       var hasFlash = swfobject.hasFlashPlayerVersion(this.settings.minimum_flash_version);
+                       
+                       if (hasFlash) {
+                               this.queueEvent("swfupload_pre_load_handler");
+                               if (typeof(oldLoadFlash) === "function") {
+                                       oldLoadFlash.call(this);
+                               }
+                       } else {
+                               this.queueEvent("swfupload_load_failed_handler");
+                       }
+               };
+               
+       }(SWFUpload.prototype.loadFlash);
+                       
+       SWFUpload.prototype.displayDebugInfo = function (oldDisplayDebugInfo) {
+               return function () {
+                       if (typeof(oldDisplayDebugInfo) === "function") {
+                               oldDisplayDebugInfo.call(this);
+                       }
+                       
+                       this.debug(
+                               [
+                                       "SWFUpload.SWFObject Plugin settings:", "\n",
+                                       "\t", "minimum_flash_version:                      ", this.settings.minimum_flash_version, "\n",
+                                       "\t", "swfupload_pre_load_handler assigned:     ", (typeof(this.settings.swfupload_pre_load_handler) === "function").toString(), "\n",
+                                       "\t", "swfupload_load_failed_handler assigned:     ", (typeof(this.settings.swfupload_load_failed_handler) === "function").toString(), "\n",
+                               ].join("")
+                       );
+               };      
+       }(SWFUpload.prototype.displayDebugInfo);
+}
index 969e200..e65b19c 100755 (executable)
-/**\r
- * SWFUpload: http://www.swfupload.org, http://swfupload.googlecode.com\r
- *\r
- * mmSWFUpload 1.0: Flash upload dialog - http://profandesign.se/swfupload/,  http://www.vinterwebb.se/\r
- *\r
- * SWFUpload is (c) 2006-2007 Lars Huring, Olov Nilzén and Mammon Media and is released under the MIT License:\r
- * http://www.opensource.org/licenses/mit-license.php\r
- *\r
- * SWFUpload 2 is (c) 2007-2008 Jake Roberts and is released under the MIT License:\r
- * http://www.opensource.org/licenses/mit-license.php\r
- *\r
- */\r
-\r
-\r
-/* ******************* */\r
-/* Constructor & Init  */\r
-/* ******************* */\r
-var SWFUpload;\r
-\r
-if (SWFUpload == undefined) {\r
-       SWFUpload = function (settings) {\r
-               this.initSWFUpload(settings);\r
-       };\r
-}\r
-\r
-SWFUpload.prototype.initSWFUpload = function (settings) {\r
-       try {\r
-               this.customSettings = {};       // A container where developers can place their own settings associated with this instance.\r
-               this.settings = settings;\r
-               this.eventQueue = [];\r
-               this.movieName = "SWFUpload_" + SWFUpload.movieCount++;\r
-               this.movieElement = null;\r
-\r
-\r
-               // Setup global control tracking\r
-               SWFUpload.instances[this.movieName] = this;\r
-\r
-               // Load the settings.  Load the Flash movie.\r
-               this.initSettings();\r
-               this.loadFlash();\r
-               this.displayDebugInfo();\r
-       } catch (ex) {\r
-               delete SWFUpload.instances[this.movieName];\r
-               throw ex;\r
-       }\r
-};\r
-\r
-/* *************** */\r
-/* Static Members  */\r
-/* *************** */\r
-SWFUpload.instances = {};\r
-SWFUpload.movieCount = 0;\r
-SWFUpload.version = "2.2.0 2009-03-25";\r
-SWFUpload.QUEUE_ERROR = {\r
-       QUEUE_LIMIT_EXCEEDED                    : -100,\r
-       FILE_EXCEEDS_SIZE_LIMIT                 : -110,\r
-       ZERO_BYTE_FILE                                  : -120,\r
-       INVALID_FILETYPE                                : -130\r
-};\r
-SWFUpload.UPLOAD_ERROR = {\r
-       HTTP_ERROR                                              : -200,\r
-       MISSING_UPLOAD_URL                      : -210,\r
-       IO_ERROR                                                : -220,\r
-       SECURITY_ERROR                                  : -230,\r
-       UPLOAD_LIMIT_EXCEEDED                   : -240,\r
-       UPLOAD_FAILED                                   : -250,\r
-       SPECIFIED_FILE_ID_NOT_FOUND             : -260,\r
-       FILE_VALIDATION_FAILED                  : -270,\r
-       FILE_CANCELLED                                  : -280,\r
-       UPLOAD_STOPPED                                  : -290\r
-};\r
-SWFUpload.FILE_STATUS = {\r
-       QUEUED           : -1,\r
-       IN_PROGRESS      : -2,\r
-       ERROR            : -3,\r
-       COMPLETE         : -4,\r
-       CANCELLED        : -5\r
-};\r
-SWFUpload.BUTTON_ACTION = {\r
-       SELECT_FILE  : -100,\r
-       SELECT_FILES : -110,\r
-       START_UPLOAD : -120\r
-};\r
-SWFUpload.CURSOR = {\r
-       ARROW : -1,\r
-       HAND : -2\r
-};\r
-SWFUpload.WINDOW_MODE = {\r
-       WINDOW : "window",\r
-       TRANSPARENT : "transparent",\r
-       OPAQUE : "opaque"\r
-};\r
-\r
-// Private: takes a URL, determines if it is relative and converts to an absolute URL\r
-// using the current site. Only processes the URL if it can, otherwise returns the URL untouched\r
-SWFUpload.completeURL = function(url) {\r
-       if (typeof(url) !== "string" || url.match(/^https?:\/\//i) || url.match(/^\//)) {\r
-               return url;\r
-       }\r
-       \r
-       var currentURL = window.location.protocol + "//" + window.location.hostname + (window.location.port ? ":" + window.location.port : "");\r
-       \r
-       var indexSlash = window.location.pathname.lastIndexOf("/");\r
-       if (indexSlash <= 0) {\r
-               path = "/";\r
-       } else {\r
-               path = window.location.pathname.substr(0, indexSlash) + "/";\r
-       }\r
-       \r
-       return /*currentURL +*/ path + url;\r
-       \r
-};\r
-\r
-\r
-/* ******************** */\r
-/* Instance Members  */\r
-/* ******************** */\r
-\r
-// Private: initSettings ensures that all the\r
-// settings are set, getting a default value if one was not assigned.\r
-SWFUpload.prototype.initSettings = function () {\r
-       this.ensureDefault = function (settingName, defaultValue) {\r
-               this.settings[settingName] = (this.settings[settingName] == undefined) ? defaultValue : this.settings[settingName];\r
-       };\r
-       \r
-       // Upload backend settings\r
-       this.ensureDefault("upload_url", "");\r
-       this.ensureDefault("preserve_relative_urls", false);\r
-       this.ensureDefault("file_post_name", "Filedata");\r
-       this.ensureDefault("post_params", {});\r
-       this.ensureDefault("use_query_string", false);\r
-       this.ensureDefault("requeue_on_error", false);\r
-       this.ensureDefault("http_success", []);\r
-       this.ensureDefault("assume_success_timeout", 0);\r
-       \r
-       // File Settings\r
-       this.ensureDefault("file_types", "*.*");\r
-       this.ensureDefault("file_types_description", "All Files");\r
-       this.ensureDefault("file_size_limit", 0);       // Default zero means "unlimited"\r
-       this.ensureDefault("file_upload_limit", 0);\r
-       this.ensureDefault("file_queue_limit", 0);\r
-\r
-       // Flash Settings\r
-       this.ensureDefault("flash_url", "swfupload.swf");\r
-       this.ensureDefault("prevent_swf_caching", true);\r
-       \r
-       // Button Settings\r
-       this.ensureDefault("button_image_url", "");\r
-       this.ensureDefault("button_width", 1);\r
-       this.ensureDefault("button_height", 1);\r
-       this.ensureDefault("button_text", "");\r
-       this.ensureDefault("button_text_style", "color: #000000; font-size: 16pt;");\r
-       this.ensureDefault("button_text_top_padding", 0);\r
-       this.ensureDefault("button_text_left_padding", 0);\r
-       this.ensureDefault("button_action", SWFUpload.BUTTON_ACTION.SELECT_FILES);\r
-       this.ensureDefault("button_disabled", false);\r
-       this.ensureDefault("button_placeholder_id", "");\r
-       this.ensureDefault("button_placeholder", null);\r
-       this.ensureDefault("button_cursor", SWFUpload.CURSOR.ARROW);\r
-       this.ensureDefault("button_window_mode", SWFUpload.WINDOW_MODE.WINDOW);\r
-       \r
-       // Debug Settings\r
-       this.ensureDefault("debug", false);\r
-       this.settings.debug_enabled = this.settings.debug;      // Here to maintain v2 API\r
-       \r
-       // Event Handlers\r
-       this.settings.return_upload_start_handler = this.returnUploadStart;\r
-       this.ensureDefault("swfupload_loaded_handler", null);\r
-       this.ensureDefault("file_dialog_start_handler", null);\r
-       this.ensureDefault("file_queued_handler", null);\r
-       this.ensureDefault("file_queue_error_handler", null);\r
-       this.ensureDefault("file_dialog_complete_handler", null);\r
-       \r
-       this.ensureDefault("upload_start_handler", null);\r
-       this.ensureDefault("upload_progress_handler", null);\r
-       this.ensureDefault("upload_error_handler", null);\r
-       this.ensureDefault("upload_success_handler", null);\r
-       this.ensureDefault("upload_complete_handler", null);\r
-       \r
-       this.ensureDefault("debug_handler", this.debugMessage);\r
-\r
-       this.ensureDefault("custom_settings", {});\r
-\r
-       // Other settings\r
-       this.customSettings = this.settings.custom_settings;\r
-       \r
-       // Update the flash url if needed\r
-       if (!!this.settings.prevent_swf_caching) {\r
-               this.settings.flash_url = this.settings.flash_url + (this.settings.flash_url.indexOf("?") < 0 ? "?" : "&") + "preventswfcaching=" + new Date().getTime();\r
-       }\r
-       \r
-       if (!this.settings.preserve_relative_urls) {\r
-               //this.settings.flash_url = SWFUpload.completeURL(this.settings.flash_url);     // Don't need to do this one since flash doesn't look at it\r
-               this.settings.upload_url = SWFUpload.completeURL(this.settings.upload_url);\r
-               this.settings.button_image_url = SWFUpload.completeURL(this.settings.button_image_url);\r
-       }\r
-       \r
-       delete this.ensureDefault;\r
-};\r
-\r
-// Private: loadFlash replaces the button_placeholder element with the flash movie.\r
-SWFUpload.prototype.loadFlash = function () {\r
-       var targetElement, tempParent;\r
-\r
-       // Make sure an element with the ID we are going to use doesn't already exist\r
-       if (document.getElementById(this.movieName) !== null) {\r
-               throw "ID " + this.movieName + " is already in use. The Flash Object could not be added";\r
-       }\r
-\r
-       // Get the element where we will be placing the flash movie\r
-       targetElement = document.getElementById(this.settings.button_placeholder_id) || this.settings.button_placeholder;\r
-\r
-       if (targetElement == undefined) {\r
-               throw "Could not find the placeholder element: " + this.settings.button_placeholder_id;\r
-       }\r
-\r
-       // Append the container and load the flash\r
-       tempParent = document.createElement("div");\r
-       tempParent.innerHTML = this.getFlashHTML();     // Using innerHTML is non-standard but the only sensible way to dynamically add Flash in IE (and maybe other browsers)\r
-       targetElement.parentNode.replaceChild(tempParent.firstChild, targetElement);\r
-\r
-       // Fix IE Flash/Form bug\r
-       if (window[this.movieName] == undefined) {\r
-               window[this.movieName] = this.getMovieElement();\r
-       }\r
-       \r
-};\r
-\r
-// Private: getFlashHTML generates the object tag needed to embed the flash in to the document\r
-SWFUpload.prototype.getFlashHTML = function () {\r
-       // Flash Satay object syntax: http://www.alistapart.com/articles/flashsatay\r
-       return ['<object id="', this.movieName, '" type="application/x-shockwave-flash" data="', this.settings.flash_url, '" width="', this.settings.button_width, '" height="', this.settings.button_height, '" class="swfupload">',\r
-                               '<param name="wmode" value="', this.settings.button_window_mode, '" />',\r
-                               '<param name="movie" value="', this.settings.flash_url, '" />',\r
-                               '<param name="quality" value="high" />',\r
-                               '<param name="menu" value="false" />',\r
-                               '<param name="allowScriptAccess" value="always" />',\r
-                               '<param name="flashvars" value="' + this.getFlashVars() + '" />',\r
-                               '</object>'].join("");\r
-};\r
-\r
-// Private: getFlashVars builds the parameter string that will be passed\r
-// to flash in the flashvars param.\r
-SWFUpload.prototype.getFlashVars = function () {\r
-       // Build a string from the post param object\r
-       var paramString = this.buildParamString();\r
-       var httpSuccessString = this.settings.http_success.join(",");\r
-       \r
-       // Build the parameter string\r
-       return ["movieName=", encodeURIComponent(this.movieName),\r
-                       "&amp;uploadURL=", encodeURIComponent(this.settings.upload_url),\r
-                       "&amp;useQueryString=", encodeURIComponent(this.settings.use_query_string),\r
-                       "&amp;requeueOnError=", encodeURIComponent(this.settings.requeue_on_error),\r
-                       "&amp;httpSuccess=", encodeURIComponent(httpSuccessString),\r
-                       "&amp;assumeSuccessTimeout=", encodeURIComponent(this.settings.assume_success_timeout),\r
-                       "&amp;params=", encodeURIComponent(paramString),\r
-                       "&amp;filePostName=", encodeURIComponent(this.settings.file_post_name),\r
-                       "&amp;fileTypes=", encodeURIComponent(this.settings.file_types),\r
-                       "&amp;fileTypesDescription=", encodeURIComponent(this.settings.file_types_description),\r
-                       "&amp;fileSizeLimit=", encodeURIComponent(this.settings.file_size_limit),\r
-                       "&amp;fileUploadLimit=", encodeURIComponent(this.settings.file_upload_limit),\r
-                       "&amp;fileQueueLimit=", encodeURIComponent(this.settings.file_queue_limit),\r
-                       "&amp;debugEnabled=", encodeURIComponent(this.settings.debug_enabled),\r
-                       "&amp;buttonImageURL=", encodeURIComponent(this.settings.button_image_url),\r
-                       "&amp;buttonWidth=", encodeURIComponent(this.settings.button_width),\r
-                       "&amp;buttonHeight=", encodeURIComponent(this.settings.button_height),\r
-                       "&amp;buttonText=", encodeURIComponent(this.settings.button_text),\r
-                       "&amp;buttonTextTopPadding=", encodeURIComponent(this.settings.button_text_top_padding),\r
-                       "&amp;buttonTextLeftPadding=", encodeURIComponent(this.settings.button_text_left_padding),\r
-                       "&amp;buttonTextStyle=", encodeURIComponent(this.settings.button_text_style),\r
-                       "&amp;buttonAction=", encodeURIComponent(this.settings.button_action),\r
-                       "&amp;buttonDisabled=", encodeURIComponent(this.settings.button_disabled),\r
-                       "&amp;buttonCursor=", encodeURIComponent(this.settings.button_cursor)\r
-               ].join("");\r
-};\r
-\r
-// Public: getMovieElement retrieves the DOM reference to the Flash element added by SWFUpload\r
-// The element is cached after the first lookup\r
-SWFUpload.prototype.getMovieElement = function () {\r
-       if (this.movieElement == undefined) {\r
-               this.movieElement = document.getElementById(this.movieName);\r
-       }\r
-\r
-       if (this.movieElement === null) {\r
-               throw "Could not find Flash element";\r
-       }\r
-       \r
-       return this.movieElement;\r
-};\r
-\r
-// Private: buildParamString takes the name/value pairs in the post_params setting object\r
-// and joins them up in to a string formatted "name=value&amp;name=value"\r
-SWFUpload.prototype.buildParamString = function () {\r
-       var postParams = this.settings.post_params; \r
-       var paramStringPairs = [];\r
-\r
-       if (typeof(postParams) === "object") {\r
-               for (var name in postParams) {\r
-                       if (postParams.hasOwnProperty(name)) {\r
-                               paramStringPairs.push(encodeURIComponent(name.toString()) + "=" + encodeURIComponent(postParams[name].toString()));\r
-                       }\r
-               }\r
-       }\r
-\r
-       return paramStringPairs.join("&amp;");\r
-};\r
-\r
-// Public: Used to remove a SWFUpload instance from the page. This method strives to remove\r
-// all references to the SWF, and other objects so memory is properly freed.\r
-// Returns true if everything was destroyed. Returns a false if a failure occurs leaving SWFUpload in an inconsistant state.\r
-// Credits: Major improvements provided by steffen\r
-SWFUpload.prototype.destroy = function () {\r
-       try {\r
-               // Make sure Flash is done before we try to remove it\r
-               this.cancelUpload(null, false);\r
-               \r
-\r
-               // Remove the SWFUpload DOM nodes\r
-               var movieElement = null;\r
-               movieElement = this.getMovieElement();\r
-               \r
-               if (movieElement && typeof(movieElement.CallFunction) === "unknown") { // We only want to do this in IE\r
-                       // Loop through all the movie's properties and remove all function references (DOM/JS IE 6/7 memory leak workaround)\r
-                       for (var i in movieElement) {\r
-                               try {\r
-                                       if (typeof(movieElement[i]) === "function") {\r
-                                               movieElement[i] = null;\r
-                                       }\r
-                               } catch (ex1) {}\r
-                       }\r
-\r
-                       // Remove the Movie Element from the page\r
-                       try {\r
-                               movieElement.parentNode.removeChild(movieElement);\r
-                       } catch (ex) {}\r
-               }\r
-               \r
-               // Remove IE form fix reference\r
-               window[this.movieName] = null;\r
-\r
-               // Destroy other references\r
-               SWFUpload.instances[this.movieName] = null;\r
-               delete SWFUpload.instances[this.movieName];\r
-\r
-               this.movieElement = null;\r
-               this.settings = null;\r
-               this.customSettings = null;\r
-               this.eventQueue = null;\r
-               this.movieName = null;\r
-               \r
-               \r
-               return true;\r
-       } catch (ex2) {\r
-               return false;\r
-       }\r
-};\r
-\r
-\r
-// Public: displayDebugInfo prints out settings and configuration\r
-// information about this SWFUpload instance.\r
-// This function (and any references to it) can be deleted when placing\r
-// SWFUpload in production.\r
-SWFUpload.prototype.displayDebugInfo = function () {\r
-       this.debug(\r
-               [\r
-                       "---SWFUpload Instance Info---\n",\r
-                       "Version: ", SWFUpload.version, "\n",\r
-                       "Movie Name: ", this.movieName, "\n",\r
-                       "Settings:\n",\r
-                       "\t", "upload_url:               ", this.settings.upload_url, "\n",\r
-                       "\t", "flash_url:                ", this.settings.flash_url, "\n",\r
-                       "\t", "use_query_string:         ", this.settings.use_query_string.toString(), "\n",\r
-                       "\t", "requeue_on_error:         ", this.settings.requeue_on_error.toString(), "\n",\r
-                       "\t", "http_success:             ", this.settings.http_success.join(", "), "\n",\r
-                       "\t", "assume_success_timeout:   ", this.settings.assume_success_timeout, "\n",\r
-                       "\t", "file_post_name:           ", this.settings.file_post_name, "\n",\r
-                       "\t", "post_params:              ", this.settings.post_params.toString(), "\n",\r
-                       "\t", "file_types:               ", this.settings.file_types, "\n",\r
-                       "\t", "file_types_description:   ", this.settings.file_types_description, "\n",\r
-                       "\t", "file_size_limit:          ", this.settings.file_size_limit, "\n",\r
-                       "\t", "file_upload_limit:        ", this.settings.file_upload_limit, "\n",\r
-                       "\t", "file_queue_limit:         ", this.settings.file_queue_limit, "\n",\r
-                       "\t", "debug:                    ", this.settings.debug.toString(), "\n",\r
-\r
-                       "\t", "prevent_swf_caching:      ", this.settings.prevent_swf_caching.toString(), "\n",\r
-\r
-                       "\t", "button_placeholder_id:    ", this.settings.button_placeholder_id.toString(), "\n",\r
-                       "\t", "button_placeholder:       ", (this.settings.button_placeholder ? "Set" : "Not Set"), "\n",\r
-                       "\t", "button_image_url:         ", this.settings.button_image_url.toString(), "\n",\r
-                       "\t", "button_width:             ", this.settings.button_width.toString(), "\n",\r
-                       "\t", "button_height:            ", this.settings.button_height.toString(), "\n",\r
-                       "\t", "button_text:              ", this.settings.button_text.toString(), "\n",\r
-                       "\t", "button_text_style:        ", this.settings.button_text_style.toString(), "\n",\r
-                       "\t", "button_text_top_padding:  ", this.settings.button_text_top_padding.toString(), "\n",\r
-                       "\t", "button_text_left_padding: ", this.settings.button_text_left_padding.toString(), "\n",\r
-                       "\t", "button_action:            ", this.settings.button_action.toString(), "\n",\r
-                       "\t", "button_disabled:          ", this.settings.button_disabled.toString(), "\n",\r
-\r
-                       "\t", "custom_settings:          ", this.settings.custom_settings.toString(), "\n",\r
-                       "Event Handlers:\n",\r
-                       "\t", "swfupload_loaded_handler assigned:  ", (typeof this.settings.swfupload_loaded_handler === "function").toString(), "\n",\r
-                       "\t", "file_dialog_start_handler assigned: ", (typeof this.settings.file_dialog_start_handler === "function").toString(), "\n",\r
-                       "\t", "file_queued_handler assigned:       ", (typeof this.settings.file_queued_handler === "function").toString(), "\n",\r
-                       "\t", "file_queue_error_handler assigned:  ", (typeof this.settings.file_queue_error_handler === "function").toString(), "\n",\r
-                       "\t", "upload_start_handler assigned:      ", (typeof this.settings.upload_start_handler === "function").toString(), "\n",\r
-                       "\t", "upload_progress_handler assigned:   ", (typeof this.settings.upload_progress_handler === "function").toString(), "\n",\r
-                       "\t", "upload_error_handler assigned:      ", (typeof this.settings.upload_error_handler === "function").toString(), "\n",\r
-                       "\t", "upload_success_handler assigned:    ", (typeof this.settings.upload_success_handler === "function").toString(), "\n",\r
-                       "\t", "upload_complete_handler assigned:   ", (typeof this.settings.upload_complete_handler === "function").toString(), "\n",\r
-                       "\t", "debug_handler assigned:             ", (typeof this.settings.debug_handler === "function").toString(), "\n"\r
-               ].join("")\r
-       );\r
-};\r
-\r
-/* Note: addSetting and getSetting are no longer used by SWFUpload but are included\r
-       the maintain v2 API compatibility\r
-*/\r
-// Public: (Deprecated) addSetting adds a setting value. If the value given is undefined or null then the default_value is used.\r
-SWFUpload.prototype.addSetting = function (name, value, default_value) {\r
-    if (value == undefined) {\r
-        return (this.settings[name] = default_value);\r
-    } else {\r
-        return (this.settings[name] = value);\r
-       }\r
-};\r
-\r
-// Public: (Deprecated) getSetting gets a setting. Returns an empty string if the setting was not found.\r
-SWFUpload.prototype.getSetting = function (name) {\r
-    if (this.settings[name] != undefined) {\r
-        return this.settings[name];\r
-       }\r
-\r
-    return "";\r
-};\r
-\r
-\r
-\r
-// Private: callFlash handles function calls made to the Flash element.\r
-// Calls are made with a setTimeout for some functions to work around\r
-// bugs in the ExternalInterface library.\r
-SWFUpload.prototype.callFlash = function (functionName, argumentArray) {\r
-       argumentArray = argumentArray || [];\r
-       \r
-       var movieElement = this.getMovieElement();\r
-       var returnValue, returnString;\r
-\r
-       // Flash's method if calling ExternalInterface methods (code adapted from MooTools).\r
-       try {\r
-               returnString = movieElement.CallFunction('<invoke name="' + functionName + '" returntype="javascript">' + __flash__argumentsToXML(argumentArray, 0) + '</invoke>');\r
-               returnValue = eval(returnString);\r
-       } catch (ex) {\r
-               throw "Call to " + functionName + " failed";\r
-       }\r
-       \r
-       // Unescape file post param values\r
-       if (returnValue != undefined && typeof returnValue.post === "object") {\r
-               returnValue = this.unescapeFilePostParams(returnValue);\r
-       }\r
-\r
-       return returnValue;\r
-};\r
-\r
-/* *****************************\r
-       -- Flash control methods --\r
-       Your UI should use these\r
-       to operate SWFUpload\r
-   ***************************** */\r
-\r
-// WARNING: this function does not work in Flash Player 10\r
-// Public: selectFile causes a File Selection Dialog window to appear.  This\r
-// dialog only allows 1 file to be selected.\r
-SWFUpload.prototype.selectFile = function () {\r
-       this.callFlash("SelectFile");\r
-};\r
-\r
-// WARNING: this function does not work in Flash Player 10\r
-// Public: selectFiles causes a File Selection Dialog window to appear/ This\r
-// dialog allows the user to select any number of files\r
-// Flash Bug Warning: Flash limits the number of selectable files based on the combined length of the file names.\r
-// If the selection name length is too long the dialog will fail in an unpredictable manner.  There is no work-around\r
-// for this bug.\r
-SWFUpload.prototype.selectFiles = function () {\r
-       this.callFlash("SelectFiles");\r
-};\r
-\r
-\r
-// Public: startUpload starts uploading the first file in the queue unless\r
-// the optional parameter 'fileID' specifies the ID \r
-SWFUpload.prototype.startUpload = function (fileID) {\r
-       this.callFlash("StartUpload", [fileID]);\r
-};\r
-\r
-// Public: cancelUpload cancels any queued file.  The fileID parameter may be the file ID or index.\r
-// If you do not specify a fileID the current uploading file or first file in the queue is cancelled.\r
-// If you do not want the uploadError event to trigger you can specify false for the triggerErrorEvent parameter.\r
-SWFUpload.prototype.cancelUpload = function (fileID, triggerErrorEvent) {\r
-       if (triggerErrorEvent !== false) {\r
-               triggerErrorEvent = true;\r
-       }\r
-       this.callFlash("CancelUpload", [fileID, triggerErrorEvent]);\r
-};\r
-\r
-// Public: stopUpload stops the current upload and requeues the file at the beginning of the queue.\r
-// If nothing is currently uploading then nothing happens.\r
-SWFUpload.prototype.stopUpload = function () {\r
-       this.callFlash("StopUpload");\r
-};\r
-\r
-/* ************************\r
- * Settings methods\r
- *   These methods change the SWFUpload settings.\r
- *   SWFUpload settings should not be changed directly on the settings object\r
- *   since many of the settings need to be passed to Flash in order to take\r
- *   effect.\r
- * *********************** */\r
-\r
-// Public: getStats gets the file statistics object.\r
-SWFUpload.prototype.getStats = function () {\r
-       return this.callFlash("GetStats");\r
-};\r
-\r
-// Public: setStats changes the SWFUpload statistics.  You shouldn't need to \r
-// change the statistics but you can.  Changing the statistics does not\r
-// affect SWFUpload accept for the successful_uploads count which is used\r
-// by the upload_limit setting to determine how many files the user may upload.\r
-SWFUpload.prototype.setStats = function (statsObject) {\r
-       this.callFlash("SetStats", [statsObject]);\r
-};\r
-\r
-// Public: getFile retrieves a File object by ID or Index.  If the file is\r
-// not found then 'null' is returned.\r
-SWFUpload.prototype.getFile = function (fileID) {\r
-       if (typeof(fileID) === "number") {\r
-               return this.callFlash("GetFileByIndex", [fileID]);\r
-       } else {\r
-               return this.callFlash("GetFile", [fileID]);\r
-       }\r
-};\r
-\r
-// Public: addFileParam sets a name/value pair that will be posted with the\r
-// file specified by the Files ID.  If the name already exists then the\r
-// exiting value will be overwritten.\r
-SWFUpload.prototype.addFileParam = function (fileID, name, value) {\r
-       return this.callFlash("AddFileParam", [fileID, name, value]);\r
-};\r
-\r
-// Public: removeFileParam removes a previously set (by addFileParam) name/value\r
-// pair from the specified file.\r
-SWFUpload.prototype.removeFileParam = function (fileID, name) {\r
-       this.callFlash("RemoveFileParam", [fileID, name]);\r
-};\r
-\r
-// Public: setUploadUrl changes the upload_url setting.\r
-SWFUpload.prototype.setUploadURL = function (url) {\r
-       this.settings.upload_url = url.toString();\r
-       this.callFlash("SetUploadURL", [url]);\r
-};\r
-\r
-// Public: setPostParams changes the post_params setting\r
-SWFUpload.prototype.setPostParams = function (paramsObject) {\r
-       this.settings.post_params = paramsObject;\r
-       this.callFlash("SetPostParams", [paramsObject]);\r
-};\r
-\r
-// Public: addPostParam adds post name/value pair.  Each name can have only one value.\r
-SWFUpload.prototype.addPostParam = function (name, value) {\r
-       this.settings.post_params[name] = value;\r
-       this.callFlash("SetPostParams", [this.settings.post_params]);\r
-};\r
-\r
-// Public: removePostParam deletes post name/value pair.\r
-SWFUpload.prototype.removePostParam = function (name) {\r
-       delete this.settings.post_params[name];\r
-       this.callFlash("SetPostParams", [this.settings.post_params]);\r
-};\r
-\r
-// Public: setFileTypes changes the file_types setting and the file_types_description setting\r
-SWFUpload.prototype.setFileTypes = function (types, description) {\r
-       this.settings.file_types = types;\r
-       this.settings.file_types_description = description;\r
-       this.callFlash("SetFileTypes", [types, description]);\r
-};\r
-\r
-// Public: setFileSizeLimit changes the file_size_limit setting\r
-SWFUpload.prototype.setFileSizeLimit = function (fileSizeLimit) {\r
-       this.settings.file_size_limit = fileSizeLimit;\r
-       this.callFlash("SetFileSizeLimit", [fileSizeLimit]);\r
-};\r
-\r
-// Public: setFileUploadLimit changes the file_upload_limit setting\r
-SWFUpload.prototype.setFileUploadLimit = function (fileUploadLimit) {\r
-       this.settings.file_upload_limit = fileUploadLimit;\r
-       this.callFlash("SetFileUploadLimit", [fileUploadLimit]);\r
-};\r
-\r
-// Public: setFileQueueLimit changes the file_queue_limit setting\r
-SWFUpload.prototype.setFileQueueLimit = function (fileQueueLimit) {\r
-       this.settings.file_queue_limit = fileQueueLimit;\r
-       this.callFlash("SetFileQueueLimit", [fileQueueLimit]);\r
-};\r
-\r
-// Public: setFilePostName changes the file_post_name setting\r
-SWFUpload.prototype.setFilePostName = function (filePostName) {\r
-       this.settings.file_post_name = filePostName;\r
-       this.callFlash("SetFilePostName", [filePostName]);\r
-};\r
-\r
-// Public: setUseQueryString changes the use_query_string setting\r
-SWFUpload.prototype.setUseQueryString = function (useQueryString) {\r
-       this.settings.use_query_string = useQueryString;\r
-       this.callFlash("SetUseQueryString", [useQueryString]);\r
-};\r
-\r
-// Public: setRequeueOnError changes the requeue_on_error setting\r
-SWFUpload.prototype.setRequeueOnError = function (requeueOnError) {\r
-       this.settings.requeue_on_error = requeueOnError;\r
-       this.callFlash("SetRequeueOnError", [requeueOnError]);\r
-};\r
-\r
-// Public: setHTTPSuccess changes the http_success setting\r
-SWFUpload.prototype.setHTTPSuccess = function (http_status_codes) {\r
-       if (typeof http_status_codes === "string") {\r
-               http_status_codes = http_status_codes.replace(" ", "").split(",");\r
-       }\r
-       \r
-       this.settings.http_success = http_status_codes;\r
-       this.callFlash("SetHTTPSuccess", [http_status_codes]);\r
-};\r
-\r
-// Public: setHTTPSuccess changes the http_success setting\r
-SWFUpload.prototype.setAssumeSuccessTimeout = function (timeout_seconds) {\r
-       this.settings.assume_success_timeout = timeout_seconds;\r
-       this.callFlash("SetAssumeSuccessTimeout", [timeout_seconds]);\r
-};\r
-\r
-// Public: setDebugEnabled changes the debug_enabled setting\r
-SWFUpload.prototype.setDebugEnabled = function (debugEnabled) {\r
-       this.settings.debug_enabled = debugEnabled;\r
-       this.callFlash("SetDebugEnabled", [debugEnabled]);\r
-};\r
-\r
-// Public: setButtonImageURL loads a button image sprite\r
-SWFUpload.prototype.setButtonImageURL = function (buttonImageURL) {\r
-       if (buttonImageURL == undefined) {\r
-               buttonImageURL = "";\r
-       }\r
-       \r
-       this.settings.button_image_url = buttonImageURL;\r
-       this.callFlash("SetButtonImageURL", [buttonImageURL]);\r
-};\r
-\r
-// Public: setButtonDimensions resizes the Flash Movie and button\r
-SWFUpload.prototype.setButtonDimensions = function (width, height) {\r
-       this.settings.button_width = width;\r
-       this.settings.button_height = height;\r
-       \r
-       var movie = this.getMovieElement();\r
-       if (movie != undefined) {\r
-               movie.style.width = width + "px";\r
-               movie.style.height = height + "px";\r
-       }\r
-       \r
-       this.callFlash("SetButtonDimensions", [width, height]);\r
-};\r
-// Public: setButtonText Changes the text overlaid on the button\r
-SWFUpload.prototype.setButtonText = function (html) {\r
-       this.settings.button_text = html;\r
-       this.callFlash("SetButtonText", [html]);\r
-};\r
-// Public: setButtonTextPadding changes the top and left padding of the text overlay\r
-SWFUpload.prototype.setButtonTextPadding = function (left, top) {\r
-       this.settings.button_text_top_padding = top;\r
-       this.settings.button_text_left_padding = left;\r
-       this.callFlash("SetButtonTextPadding", [left, top]);\r
-};\r
-\r
-// Public: setButtonTextStyle changes the CSS used to style the HTML/Text overlaid on the button\r
-SWFUpload.prototype.setButtonTextStyle = function (css) {\r
-       this.settings.button_text_style = css;\r
-       this.callFlash("SetButtonTextStyle", [css]);\r
-};\r
-// Public: setButtonDisabled disables/enables the button\r
-SWFUpload.prototype.setButtonDisabled = function (isDisabled) {\r
-       this.settings.button_disabled = isDisabled;\r
-       this.callFlash("SetButtonDisabled", [isDisabled]);\r
-};\r
-// Public: setButtonAction sets the action that occurs when the button is clicked\r
-SWFUpload.prototype.setButtonAction = function (buttonAction) {\r
-       this.settings.button_action = buttonAction;\r
-       this.callFlash("SetButtonAction", [buttonAction]);\r
-};\r
-\r
-// Public: setButtonCursor changes the mouse cursor displayed when hovering over the button\r
-SWFUpload.prototype.setButtonCursor = function (cursor) {\r
-       this.settings.button_cursor = cursor;\r
-       this.callFlash("SetButtonCursor", [cursor]);\r
-};\r
-\r
-/* *******************************\r
-       Flash Event Interfaces\r
-       These functions are used by Flash to trigger the various\r
-       events.\r
-       \r
-       All these functions a Private.\r
-       \r
-       Because the ExternalInterface library is buggy the event calls\r
-       are added to a queue and the queue then executed by a setTimeout.\r
-       This ensures that events are executed in a determinate order and that\r
-       the ExternalInterface bugs are avoided.\r
-******************************* */\r
-\r
-SWFUpload.prototype.queueEvent = function (handlerName, argumentArray) {\r
-       // Warning: Don't call this.debug inside here or you'll create an infinite loop\r
-       \r
-       if (argumentArray == undefined) {\r
-               argumentArray = [];\r
-       } else if (!(argumentArray instanceof Array)) {\r
-               argumentArray = [argumentArray];\r
-       }\r
-       \r
-       var self = this;\r
-       if (typeof this.settings[handlerName] === "function") {\r
-               // Queue the event\r
-               this.eventQueue.push(function () {\r
-                       this.settings[handlerName].apply(this, argumentArray);\r
-               });\r
-               \r
-               // Execute the next queued event\r
-               setTimeout(function () {\r
-                       self.executeNextEvent();\r
-               }, 0);\r
-               \r
-       } else if (this.settings[handlerName] !== null) {\r
-               throw "Event handler " + handlerName + " is unknown or is not a function";\r
-       }\r
-};\r
-\r
-// Private: Causes the next event in the queue to be executed.  Since events are queued using a setTimeout\r
-// we must queue them in order to garentee that they are executed in order.\r
-SWFUpload.prototype.executeNextEvent = function () {\r
-       // Warning: Don't call this.debug inside here or you'll create an infinite loop\r
-\r
-       var  f = this.eventQueue ? this.eventQueue.shift() : null;\r
-       if (typeof(f) === "function") {\r
-               f.apply(this);\r
-       }\r
-};\r
-\r
-// Private: unescapeFileParams is part of a workaround for a flash bug where objects passed through ExternalInterface cannot have\r
-// properties that contain characters that are not valid for JavaScript identifiers. To work around this\r
-// the Flash Component escapes the parameter names and we must unescape again before passing them along.\r
-SWFUpload.prototype.unescapeFilePostParams = function (file) {\r
-       var reg = /[$]([0-9a-f]{4})/i;\r
-       var unescapedPost = {};\r
-       var uk;\r
-\r
-       if (file != undefined) {\r
-               for (var k in file.post) {\r
-                       if (file.post.hasOwnProperty(k)) {\r
-                               uk = k;\r
-                               var match;\r
-                               while ((match = reg.exec(uk)) !== null) {\r
-                                       uk = uk.replace(match[0], String.fromCharCode(parseInt("0x" + match[1], 16)));\r
-                               }\r
-                               unescapedPost[uk] = file.post[k];\r
-                       }\r
-               }\r
-\r
-               file.post = unescapedPost;\r
-       }\r
-\r
-       return file;\r
-};\r
-\r
-// Private: Called by Flash to see if JS can call in to Flash (test if External Interface is working)\r
-SWFUpload.prototype.testExternalInterface = function () {\r
-       try {\r
-               return this.callFlash("TestExternalInterface");\r
-       } catch (ex) {\r
-               return false;\r
-       }\r
-};\r
-\r
-// Private: This event is called by Flash when it has finished loading. Don't modify this.\r
-// Use the swfupload_loaded_handler event setting to execute custom code when SWFUpload has loaded.\r
-SWFUpload.prototype.flashReady = function () {\r
-       // Check that the movie element is loaded correctly with its ExternalInterface methods defined\r
-       var movieElement = this.getMovieElement();\r
-\r
-       if (!movieElement) {\r
-               this.debug("Flash called back ready but the flash movie can't be found.");\r
-               return;\r
-       }\r
-\r
-       this.cleanUp(movieElement);\r
-       \r
-       this.queueEvent("swfupload_loaded_handler");\r
-};\r
-\r
-// Private: removes Flash added fuctions to the DOM node to prevent memory leaks in IE.\r
-// This function is called by Flash each time the ExternalInterface functions are created.\r
-SWFUpload.prototype.cleanUp = function (movieElement) {\r
-       // Pro-actively unhook all the Flash functions\r
-       try {\r
-               if (this.movieElement && typeof(movieElement.CallFunction) === "unknown") { // We only want to do this in IE\r
-                       this.debug("Removing Flash functions hooks (this should only run in IE and should prevent memory leaks)");\r
-                       for (var key in movieElement) {\r
-                               try {\r
-                                       if (typeof(movieElement[key]) === "function") {\r
-                                               movieElement[key] = null;\r
-                                       }\r
-                               } catch (ex) {\r
-                               }\r
-                       }\r
-               }\r
-       } catch (ex1) {\r
-       \r
-       }\r
-\r
-       // Fix Flashes own cleanup code so if the SWFMovie was removed from the page\r
-       // it doesn't display errors.\r
-       window["__flash__removeCallback"] = function (instance, name) {\r
-               try {\r
-                       if (instance) {\r
-                               instance[name] = null;\r
-                       }\r
-               } catch (flashEx) {\r
-               \r
-               }\r
-       };\r
-\r
-};\r
-\r
-\r
-/* This is a chance to do something before the browse window opens */\r
-SWFUpload.prototype.fileDialogStart = function () {\r
-       this.queueEvent("file_dialog_start_handler");\r
-};\r
-\r
-\r
-/* Called when a file is successfully added to the queue. */\r
-SWFUpload.prototype.fileQueued = function (file) {\r
-       file = this.unescapeFilePostParams(file);\r
-       this.queueEvent("file_queued_handler", file);\r
-};\r
-\r
-\r
-/* Handle errors that occur when an attempt to queue a file fails. */\r
-SWFUpload.prototype.fileQueueError = function (file, errorCode, message) {\r
-       file = this.unescapeFilePostParams(file);\r
-       this.queueEvent("file_queue_error_handler", [file, errorCode, message]);\r
-};\r
-\r
-/* Called after the file dialog has closed and the selected files have been queued.\r
-       You could call startUpload here if you want the queued files to begin uploading immediately. */\r
-SWFUpload.prototype.fileDialogComplete = function (numFilesSelected, numFilesQueued, numFilesInQueue) {\r
-       this.queueEvent("file_dialog_complete_handler", [numFilesSelected, numFilesQueued, numFilesInQueue]);\r
-};\r
-\r
-SWFUpload.prototype.uploadStart = function (file) {\r
-       file = this.unescapeFilePostParams(file);\r
-       this.queueEvent("return_upload_start_handler", file);\r
-};\r
-\r
-SWFUpload.prototype.returnUploadStart = function (file) {\r
-       var returnValue;\r
-       if (typeof this.settings.upload_start_handler === "function") {\r
-               file = this.unescapeFilePostParams(file);\r
-               returnValue = this.settings.upload_start_handler.call(this, file);\r
-       } else if (this.settings.upload_start_handler != undefined) {\r
-               throw "upload_start_handler must be a function";\r
-       }\r
-\r
-       // Convert undefined to true so if nothing is returned from the upload_start_handler it is\r
-       // interpretted as 'true'.\r
-       if (returnValue === undefined) {\r
-               returnValue = true;\r
-       }\r
-       \r
-       returnValue = !!returnValue;\r
-       \r
-       this.callFlash("ReturnUploadStart", [returnValue]);\r
-};\r
-\r
-\r
-\r
-SWFUpload.prototype.uploadProgress = function (file, bytesComplete, bytesTotal) {\r
-       file = this.unescapeFilePostParams(file);\r
-       this.queueEvent("upload_progress_handler", [file, bytesComplete, bytesTotal]);\r
-};\r
-\r
-SWFUpload.prototype.uploadError = function (file, errorCode, message) {\r
-       file = this.unescapeFilePostParams(file);\r
-       this.queueEvent("upload_error_handler", [file, errorCode, message]);\r
-};\r
-\r
-SWFUpload.prototype.uploadSuccess = function (file, serverData, responseReceived) {\r
-       file = this.unescapeFilePostParams(file);\r
-       this.queueEvent("upload_success_handler", [file, serverData, responseReceived]);\r
-};\r
-\r
-SWFUpload.prototype.uploadComplete = function (file) {\r
-       file = this.unescapeFilePostParams(file);\r
-       this.queueEvent("upload_complete_handler", file);\r
-};\r
-\r
-/* Called by SWFUpload JavaScript and Flash functions when debug is enabled. By default it writes messages to the\r
-   internal debug console.  You can override this event and have messages written where you want. */\r
-SWFUpload.prototype.debug = function (message) {\r
-       this.queueEvent("debug_handler", message);\r
-};\r
-\r
-\r
-/* **********************************\r
-       Debug Console\r
-       The debug console is a self contained, in page location\r
-       for debug message to be sent.  The Debug Console adds\r
-       itself to the body if necessary.\r
-\r
-       The console is automatically scrolled as messages appear.\r
-       \r
-       If you are using your own debug handler or when you deploy to production and\r
-       have debug disabled you can remove these functions to reduce the file size\r
-       and complexity.\r
-********************************** */\r
-   \r
-// Private: debugMessage is the default debug_handler.  If you want to print debug messages\r
-// call the debug() function.  When overriding the function your own function should\r
-// check to see if the debug setting is true before outputting debug information.\r
-SWFUpload.prototype.debugMessage = function (message) {\r
-       if (this.settings.debug) {\r
-               var exceptionMessage, exceptionValues = [];\r
-\r
-               // Check for an exception object and print it nicely\r
-               if (typeof message === "object" && typeof message.name === "string" && typeof message.message === "string") {\r
-                       for (var key in message) {\r
-                               if (message.hasOwnProperty(key)) {\r
-                                       exceptionValues.push(key + ": " + message[key]);\r
-                               }\r
-                       }\r
-                       exceptionMessage = exceptionValues.join("\n") || "";\r
-                       exceptionValues = exceptionMessage.split("\n");\r
-                       exceptionMessage = "EXCEPTION: " + exceptionValues.join("\nEXCEPTION: ");\r
-                       SWFUpload.Console.writeLine(exceptionMessage);\r
-               } else {\r
-                       SWFUpload.Console.writeLine(message);\r
-               }\r
-       }\r
-};\r
-\r
-SWFUpload.Console = {};\r
-SWFUpload.Console.writeLine = function (message) {\r
-       var console, documentForm;\r
-\r
-       try {\r
-               console = document.getElementById("SWFUpload_Console");\r
-\r
-               if (!console) {\r
-                       documentForm = document.createElement("form");\r
-                       document.getElementsByTagName("body")[0].appendChild(documentForm);\r
-\r
-                       console = document.createElement("textarea");\r
-                       console.id = "SWFUpload_Console";\r
-                       console.style.fontFamily = "monospace";\r
-                       console.setAttribute("wrap", "off");\r
-                       console.wrap = "off";\r
-                       console.style.overflow = "auto";\r
-                       console.style.width = "700px";\r
-                       console.style.height = "350px";\r
-                       console.style.margin = "5px";\r
-                       documentForm.appendChild(console);\r
-               }\r
-\r
-               console.value += message + "\n";\r
-\r
-               console.scrollTop = console.scrollHeight - console.clientHeight;\r
-       } catch (ex) {\r
-               alert("Exception: " + ex.name + " Message: " + ex.message);\r
-       }\r
-};\r
+/**
+ * SWFUpload: http://www.swfupload.org, http://swfupload.googlecode.com
+ *
+ * mmSWFUpload 1.0: Flash upload dialog - http://profandesign.se/swfupload/,  http://www.vinterwebb.se/
+ *
+ * SWFUpload is (c) 2006-2007 Lars Huring, Olov Nilzén and Mammon Media and is released under the MIT License:
+ * http://www.opensource.org/licenses/mit-license.php
+ *
+ * SWFUpload 2 is (c) 2007-2008 Jake Roberts and is released under the MIT License:
+ * http://www.opensource.org/licenses/mit-license.php
+ *
+ */
+
+
+/* ******************* */
+/* Constructor & Init  */
+/* ******************* */
+var SWFUpload;
+
+if (SWFUpload == undefined) {
+       SWFUpload = function (settings) {
+               this.initSWFUpload(settings);
+       };
+}
+
+SWFUpload.prototype.initSWFUpload = function (settings) {
+       try {
+               this.customSettings = {};       // A container where developers can place their own settings associated with this instance.
+               this.settings = settings;
+               this.eventQueue = [];
+               this.movieName = "SWFUpload_" + SWFUpload.movieCount++;
+               this.movieElement = null;
+
+
+               // Setup global control tracking
+               SWFUpload.instances[this.movieName] = this;
+
+               // Load the settings.  Load the Flash movie.
+               this.initSettings();
+               this.loadFlash();
+               this.displayDebugInfo();
+       } catch (ex) {
+               delete SWFUpload.instances[this.movieName];
+               throw ex;
+       }
+};
+
+/* *************** */
+/* Static Members  */
+/* *************** */
+SWFUpload.instances = {};
+SWFUpload.movieCount = 0;
+SWFUpload.version = "2.2.0 2009-03-25";
+SWFUpload.QUEUE_ERROR = {
+       QUEUE_LIMIT_EXCEEDED                    : -100,
+       FILE_EXCEEDS_SIZE_LIMIT                 : -110,
+       ZERO_BYTE_FILE                                  : -120,
+       INVALID_FILETYPE                                : -130
+};
+SWFUpload.UPLOAD_ERROR = {
+       HTTP_ERROR                                              : -200,
+       MISSING_UPLOAD_URL                      : -210,
+       IO_ERROR                                                : -220,
+       SECURITY_ERROR                                  : -230,
+       UPLOAD_LIMIT_EXCEEDED                   : -240,
+       UPLOAD_FAILED                                   : -250,
+       SPECIFIED_FILE_ID_NOT_FOUND             : -260,
+       FILE_VALIDATION_FAILED                  : -270,
+       FILE_CANCELLED                                  : -280,
+       UPLOAD_STOPPED                                  : -290
+};
+SWFUpload.FILE_STATUS = {
+       QUEUED           : -1,
+       IN_PROGRESS      : -2,
+       ERROR            : -3,
+       COMPLETE         : -4,
+       CANCELLED        : -5
+};
+SWFUpload.BUTTON_ACTION = {
+       SELECT_FILE  : -100,
+       SELECT_FILES : -110,
+       START_UPLOAD : -120
+};
+SWFUpload.CURSOR = {
+       ARROW : -1,
+       HAND : -2
+};
+SWFUpload.WINDOW_MODE = {
+       WINDOW : "window",
+       TRANSPARENT : "transparent",
+       OPAQUE : "opaque"
+};
+
+// Private: takes a URL, determines if it is relative and converts to an absolute URL
+// using the current site. Only processes the URL if it can, otherwise returns the URL untouched
+SWFUpload.completeURL = function(url) {
+       if (typeof(url) !== "string" || url.match(/^https?:\/\//i) || url.match(/^\//)) {
+               return url;
+       }
+       
+       var currentURL = window.location.protocol + "//" + window.location.hostname + (window.location.port ? ":" + window.location.port : "");
+       
+       var indexSlash = window.location.pathname.lastIndexOf("/");
+       if (indexSlash <= 0) {
+               path = "/";
+       } else {
+               path = window.location.pathname.substr(0, indexSlash) + "/";
+       }
+       
+       return /*currentURL +*/ path + url;
+       
+};
+
+
+/* ******************** */
+/* Instance Members  */
+/* ******************** */
+
+// Private: initSettings ensures that all the
+// settings are set, getting a default value if one was not assigned.
+SWFUpload.prototype.initSettings = function () {
+       this.ensureDefault = function (settingName, defaultValue) {
+               this.settings[settingName] = (this.settings[settingName] == undefined) ? defaultValue : this.settings[settingName];
+       };
+       
+       // Upload backend settings
+       this.ensureDefault("upload_url", "");
+       this.ensureDefault("preserve_relative_urls", false);
+       this.ensureDefault("file_post_name", "Filedata");
+       this.ensureDefault("post_params", {});
+       this.ensureDefault("use_query_string", false);
+       this.ensureDefault("requeue_on_error", false);
+       this.ensureDefault("http_success", []);
+       this.ensureDefault("assume_success_timeout", 0);
+       
+       // File Settings
+       this.ensureDefault("file_types", "*.*");
+       this.ensureDefault("file_types_description", "All Files");
+       this.ensureDefault("file_size_limit", 0);       // Default zero means "unlimited"
+       this.ensureDefault("file_upload_limit", 0);
+       this.ensureDefault("file_queue_limit", 0);
+
+       // Flash Settings
+       this.ensureDefault("flash_url", "swfupload.swf");
+       this.ensureDefault("prevent_swf_caching", true);
+       
+       // Button Settings
+       this.ensureDefault("button_image_url", "");
+       this.ensureDefault("button_width", 1);
+       this.ensureDefault("button_height", 1);
+       this.ensureDefault("button_text", "");
+       this.ensureDefault("button_text_style", "color: #000000; font-size: 16pt;");
+       this.ensureDefault("button_text_top_padding", 0);
+       this.ensureDefault("button_text_left_padding", 0);
+       this.ensureDefault("button_action", SWFUpload.BUTTON_ACTION.SELECT_FILES);
+       this.ensureDefault("button_disabled", false);
+       this.ensureDefault("button_placeholder_id", "");
+       this.ensureDefault("button_placeholder", null);
+       this.ensureDefault("button_cursor", SWFUpload.CURSOR.ARROW);
+       this.ensureDefault("button_window_mode", SWFUpload.WINDOW_MODE.WINDOW);
+       
+       // Debug Settings
+       this.ensureDefault("debug", false);
+       this.settings.debug_enabled = this.settings.debug;      // Here to maintain v2 API
+       
+       // Event Handlers
+       this.settings.return_upload_start_handler = this.returnUploadStart;
+       this.ensureDefault("swfupload_loaded_handler", null);
+       this.ensureDefault("file_dialog_start_handler", null);
+       this.ensureDefault("file_queued_handler", null);
+       this.ensureDefault("file_queue_error_handler", null);
+       this.ensureDefault("file_dialog_complete_handler", null);
+       
+       this.ensureDefault("upload_start_handler", null);
+       this.ensureDefault("upload_progress_handler", null);
+       this.ensureDefault("upload_error_handler", null);
+       this.ensureDefault("upload_success_handler", null);
+       this.ensureDefault("upload_complete_handler", null);
+       
+       this.ensureDefault("debug_handler", this.debugMessage);
+
+       this.ensureDefault("custom_settings", {});
+
+       // Other settings
+       this.customSettings = this.settings.custom_settings;
+       
+       // Update the flash url if needed
+       if (!!this.settings.prevent_swf_caching) {
+               this.settings.flash_url = this.settings.flash_url + (this.settings.flash_url.indexOf("?") < 0 ? "?" : "&") + "preventswfcaching=" + new Date().getTime();
+       }
+       
+       if (!this.settings.preserve_relative_urls) {
+               //this.settings.flash_url = SWFUpload.completeURL(this.settings.flash_url);     // Don't need to do this one since flash doesn't look at it
+               this.settings.upload_url = SWFUpload.completeURL(this.settings.upload_url);
+               this.settings.button_image_url = SWFUpload.completeURL(this.settings.button_image_url);
+       }
+       
+       delete this.ensureDefault;
+};
+
+// Private: loadFlash replaces the button_placeholder element with the flash movie.
+SWFUpload.prototype.loadFlash = function () {
+       var targetElement, tempParent;
+
+       // Make sure an element with the ID we are going to use doesn't already exist
+       if (document.getElementById(this.movieName) !== null) {
+               throw "ID " + this.movieName + " is already in use. The Flash Object could not be added";
+       }
+
+       // Get the element where we will be placing the flash movie
+       targetElement = document.getElementById(this.settings.button_placeholder_id) || this.settings.button_placeholder;
+
+       if (targetElement == undefined) {
+               throw "Could not find the placeholder element: " + this.settings.button_placeholder_id;
+       }
+
+       // Append the container and load the flash
+       tempParent = document.createElement("div");
+       tempParent.innerHTML = this.getFlashHTML();     // Using innerHTML is non-standard but the only sensible way to dynamically add Flash in IE (and maybe other browsers)
+       targetElement.parentNode.replaceChild(tempParent.firstChild, targetElement);
+
+       // Fix IE Flash/Form bug
+       if (window[this.movieName] == undefined) {
+               window[this.movieName] = this.getMovieElement();
+       }
+       
+};
+
+// Private: getFlashHTML generates the object tag needed to embed the flash in to the document
+SWFUpload.prototype.getFlashHTML = function () {
+       // Flash Satay object syntax: http://www.alistapart.com/articles/flashsatay
+       return ['<object id="', this.movieName, '" type="application/x-shockwave-flash" data="', this.settings.flash_url, '" width="', this.settings.button_width, '" height="', this.settings.button_height, '" class="swfupload">',
+                               '<param name="wmode" value="', this.settings.button_window_mode, '" />',
+                               '<param name="movie" value="', this.settings.flash_url, '" />',
+                               '<param name="quality" value="high" />',
+                               '<param name="menu" value="false" />',
+                               '<param name="allowScriptAccess" value="always" />',
+                               '<param name="flashvars" value="' + this.getFlashVars() + '" />',
+                               '</object>'].join("");
+};
+
+// Private: getFlashVars builds the parameter string that will be passed
+// to flash in the flashvars param.
+SWFUpload.prototype.getFlashVars = function () {
+       // Build a string from the post param object
+       var paramString = this.buildParamString();
+       var httpSuccessString = this.settings.http_success.join(",");
+       
+       // Build the parameter string
+       return ["movieName=", encodeURIComponent(this.movieName),
+                       "&amp;uploadURL=", encodeURIComponent(this.settings.upload_url),
+                       "&amp;useQueryString=", encodeURIComponent(this.settings.use_query_string),
+                       "&amp;requeueOnError=", encodeURIComponent(this.settings.requeue_on_error),
+                       "&amp;httpSuccess=", encodeURIComponent(httpSuccessString),
+                       "&amp;assumeSuccessTimeout=", encodeURIComponent(this.settings.assume_success_timeout),
+                       "&amp;params=", encodeURIComponent(paramString),
+                       "&amp;filePostName=", encodeURIComponent(this.settings.file_post_name),
+                       "&amp;fileTypes=", encodeURIComponent(this.settings.file_types),
+                       "&amp;fileTypesDescription=", encodeURIComponent(this.settings.file_types_description),
+                       "&amp;fileSizeLimit=", encodeURIComponent(this.settings.file_size_limit),
+                       "&amp;fileUploadLimit=", encodeURIComponent(this.settings.file_upload_limit),
+                       "&amp;fileQueueLimit=", encodeURIComponent(this.settings.file_queue_limit),
+                       "&amp;debugEnabled=", encodeURIComponent(this.settings.debug_enabled),
+                       "&amp;buttonImageURL=", encodeURIComponent(this.settings.button_image_url),
+                       "&amp;buttonWidth=", encodeURIComponent(this.settings.button_width),
+                       "&amp;buttonHeight=", encodeURIComponent(this.settings.button_height),
+                       "&amp;buttonText=", encodeURIComponent(this.settings.button_text),
+                       "&amp;buttonTextTopPadding=", encodeURIComponent(this.settings.button_text_top_padding),
+                       "&amp;buttonTextLeftPadding=", encodeURIComponent(this.settings.button_text_left_padding),
+                       "&amp;buttonTextStyle=", encodeURIComponent(this.settings.button_text_style),
+                       "&amp;buttonAction=", encodeURIComponent(this.settings.button_action),
+                       "&amp;buttonDisabled=", encodeURIComponent(this.settings.button_disabled),
+                       "&amp;buttonCursor=", encodeURIComponent(this.settings.button_cursor)
+               ].join("");
+};
+
+// Public: getMovieElement retrieves the DOM reference to the Flash element added by SWFUpload
+// The element is cached after the first lookup
+SWFUpload.prototype.getMovieElement = function () {
+       if (this.movieElement == undefined) {
+               this.movieElement = document.getElementById(this.movieName);
+       }
+
+       if (this.movieElement === null) {
+               throw "Could not find Flash element";
+       }
+       
+       return this.movieElement;
+};
+
+// Private: buildParamString takes the name/value pairs in the post_params setting object
+// and joins them up in to a string formatted "name=value&amp;name=value"
+SWFUpload.prototype.buildParamString = function () {
+       var postParams = this.settings.post_params; 
+       var paramStringPairs = [];
+
+       if (typeof(postParams) === "object") {
+               for (var name in postParams) {
+                       if (postParams.hasOwnProperty(name)) {
+                               paramStringPairs.push(encodeURIComponent(name.toString()) + "=" + encodeURIComponent(postParams[name].toString()));
+                       }
+               }
+       }
+
+       return paramStringPairs.join("&amp;");
+};
+
+// Public: Used to remove a SWFUpload instance from the page. This method strives to remove
+// all references to the SWF, and other objects so memory is properly freed.
+// Returns true if everything was destroyed. Returns a false if a failure occurs leaving SWFUpload in an inconsistant state.
+// Credits: Major improvements provided by steffen
+SWFUpload.prototype.destroy = function () {
+       try {
+               // Make sure Flash is done before we try to remove it
+               this.cancelUpload(null, false);
+               
+
+               // Remove the SWFUpload DOM nodes
+               var movieElement = null;
+               movieElement = this.getMovieElement();
+               
+               if (movieElement && typeof(movieElement.CallFunction) === "unknown") { // We only want to do this in IE
+                       // Loop through all the movie's properties and remove all function references (DOM/JS IE 6/7 memory leak workaround)
+                       for (var i in movieElement) {
+                               try {
+                                       if (typeof(movieElement[i]) === "function") {
+                                               movieElement[i] = null;
+                                       }
+                               } catch (ex1) {}
+                       }
+
+                       // Remove the Movie Element from the page
+                       try {
+                               movieElement.parentNode.removeChild(movieElement);
+                       } catch (ex) {}
+               }
+               
+               // Remove IE form fix reference
+               window[this.movieName] = null;
+
+               // Destroy other references
+               SWFUpload.instances[this.movieName] = null;
+               delete SWFUpload.instances[this.movieName];
+
+               this.movieElement = null;
+               this.settings = null;
+               this.customSettings = null;
+               this.eventQueue = null;
+               this.movieName = null;
+               
+               
+               return true;
+       } catch (ex2) {
+               return false;
+       }
+};
+
+
+// Public: displayDebugInfo prints out settings and configuration
+// information about this SWFUpload instance.
+// This function (and any references to it) can be deleted when placing
+// SWFUpload in production.
+SWFUpload.prototype.displayDebugInfo = function () {
+       this.debug(
+               [
+                       "---SWFUpload Instance Info---\n",
+                       "Version: ", SWFUpload.version, "\n",
+                       "Movie Name: ", this.movieName, "\n",
+                       "Settings:\n",
+                       "\t", "upload_url:               ", this.settings.upload_url, "\n",
+                       "\t", "flash_url:                ", this.settings.flash_url, "\n",
+                       "\t", "use_query_string:         ", this.settings.use_query_string.toString(), "\n",
+                       "\t", "requeue_on_error:         ", this.settings.requeue_on_error.toString(), "\n",
+                       "\t", "http_success:             ", this.settings.http_success.join(", "), "\n",
+                       "\t", "assume_success_timeout:   ", this.settings.assume_success_timeout, "\n",
+                       "\t", "file_post_name:           ", this.settings.file_post_name, "\n",
+                       "\t", "post_params:              ", this.settings.post_params.toString(), "\n",
+                       "\t", "file_types:               ", this.settings.file_types, "\n",
+                       "\t", "file_types_description:   ", this.settings.file_types_description, "\n",
+                       "\t", "file_size_limit:          ", this.settings.file_size_limit, "\n",
+                       "\t", "file_upload_limit:        ", this.settings.file_upload_limit, "\n",
+                       "\t", "file_queue_limit:         ", this.settings.file_queue_limit, "\n",
+                       "\t", "debug:                    ", this.settings.debug.toString(), "\n",
+
+                       "\t", "prevent_swf_caching:      ", this.settings.prevent_swf_caching.toString(), "\n",
+
+                       "\t", "button_placeholder_id:    ", this.settings.button_placeholder_id.toString(), "\n",
+                       "\t", "button_placeholder:       ", (this.settings.button_placeholder ? "Set" : "Not Set"), "\n",
+                       "\t", "button_image_url:         ", this.settings.button_image_url.toString(), "\n",
+                       "\t", "button_width:             ", this.settings.button_width.toString(), "\n",
+                       "\t", "button_height:            ", this.settings.button_height.toString(), "\n",
+                       "\t", "button_text:              ", this.settings.button_text.toString(), "\n",
+                       "\t", "button_text_style:        ", this.settings.button_text_style.toString(), "\n",
+                       "\t", "button_text_top_padding:  ", this.settings.button_text_top_padding.toString(), "\n",
+                       "\t", "button_text_left_padding: ", this.settings.button_text_left_padding.toString(), "\n",
+                       "\t", "button_action:            ", this.settings.button_action.toString(), "\n",
+                       "\t", "button_disabled:          ", this.settings.button_disabled.toString(), "\n",
+
+                       "\t", "custom_settings:          ", this.settings.custom_settings.toString(), "\n",
+                       "Event Handlers:\n",
+                       "\t", "swfupload_loaded_handler assigned:  ", (typeof this.settings.swfupload_loaded_handler === "function").toString(), "\n",
+                       "\t", "file_dialog_start_handler assigned: ", (typeof this.settings.file_dialog_start_handler === "function").toString(), "\n",
+                       "\t", "file_queued_handler assigned:       ", (typeof this.settings.file_queued_handler === "function").toString(), "\n",
+                       "\t", "file_queue_error_handler assigned:  ", (typeof this.settings.file_queue_error_handler === "function").toString(), "\n",
+                       "\t", "upload_start_handler assigned:      ", (typeof this.settings.upload_start_handler === "function").toString(), "\n",
+                       "\t", "upload_progress_handler assigned:   ", (typeof this.settings.upload_progress_handler === "function").toString(), "\n",
+                       "\t", "upload_error_handler assigned:      ", (typeof this.settings.upload_error_handler === "function").toString(), "\n",
+                       "\t", "upload_success_handler assigned:    ", (typeof this.settings.upload_success_handler === "function").toString(), "\n",
+                       "\t", "upload_complete_handler assigned:   ", (typeof this.settings.upload_complete_handler === "function").toString(), "\n",
+                       "\t", "debug_handler assigned:             ", (typeof this.settings.debug_handler === "function").toString(), "\n"
+               ].join("")
+       );
+};
+
+/* Note: addSetting and getSetting are no longer used by SWFUpload but are included
+       the maintain v2 API compatibility
+*/
+// Public: (Deprecated) addSetting adds a setting value. If the value given is undefined or null then the default_value is used.
+SWFUpload.prototype.addSetting = function (name, value, default_value) {
+    if (value == undefined) {
+        return (this.settings[name] = default_value);
+    } else {
+        return (this.settings[name] = value);
+       }
+};
+
+// Public: (Deprecated) getSetting gets a setting. Returns an empty string if the setting was not found.
+SWFUpload.prototype.getSetting = function (name) {
+    if (this.settings[name] != undefined) {
+        return this.settings[name];
+       }
+
+    return "";
+};
+
+
+
+// Private: callFlash handles function calls made to the Flash element.
+// Calls are made with a setTimeout for some functions to work around
+// bugs in the ExternalInterface library.
+SWFUpload.prototype.callFlash = function (functionName, argumentArray) {
+       argumentArray = argumentArray || [];
+       
+       var movieElement = this.getMovieElement();
+       var returnValue, returnString;
+
+       // Flash's method if calling ExternalInterface methods (code adapted from MooTools).
+       try {
+               returnString = movieElement.CallFunction('<invoke name="' + functionName + '" returntype="javascript">' + __flash__argumentsToXML(argumentArray, 0) + '</invoke>');
+               returnValue = eval(returnString);
+       } catch (ex) {
+               throw "Call to " + functionName + " failed";
+       }
+       
+       // Unescape file post param values
+       if (returnValue != undefined && typeof returnValue.post === "object") {
+               returnValue = this.unescapeFilePostParams(returnValue);
+       }
+
+       return returnValue;
+};
+
+/* *****************************
+       -- Flash control methods --
+       Your UI should use these
+       to operate SWFUpload
+   ***************************** */
+
+// WARNING: this function does not work in Flash Player 10
+// Public: selectFile causes a File Selection Dialog window to appear.  This
+// dialog only allows 1 file to be selected.
+SWFUpload.prototype.selectFile = function () {
+       this.callFlash("SelectFile");
+};
+
+// WARNING: this function does not work in Flash Player 10
+// Public: selectFiles causes a File Selection Dialog window to appear/ This
+// dialog allows the user to select any number of files
+// Flash Bug Warning: Flash limits the number of selectable files based on the combined length of the file names.
+// If the selection name length is too long the dialog will fail in an unpredictable manner.  There is no work-around
+// for this bug.
+SWFUpload.prototype.selectFiles = function () {
+       this.callFlash("SelectFiles");
+};
+
+
+// Public: startUpload starts uploading the first file in the queue unless
+// the optional parameter 'fileID' specifies the ID 
+SWFUpload.prototype.startUpload = function (fileID) {
+       this.callFlash("StartUpload", [fileID]);
+};
+
+// Public: cancelUpload cancels any queued file.  The fileID parameter may be the file ID or index.
+// If you do not specify a fileID the current uploading file or first file in the queue is cancelled.
+// If you do not want the uploadError event to trigger you can specify false for the triggerErrorEvent parameter.
+SWFUpload.prototype.cancelUpload = function (fileID, triggerErrorEvent) {
+       if (triggerErrorEvent !== false) {
+               triggerErrorEvent = true;
+       }
+       this.callFlash("CancelUpload", [fileID, triggerErrorEvent]);
+};
+
+// Public: stopUpload stops the current upload and requeues the file at the beginning of the queue.
+// If nothing is currently uploading then nothing happens.
+SWFUpload.prototype.stopUpload = function () {
+       this.callFlash("StopUpload");
+};
+
+/* ************************
+ * Settings methods
+ *   These methods change the SWFUpload settings.
+ *   SWFUpload settings should not be changed directly on the settings object
+ *   since many of the settings need to be passed to Flash in order to take
+ *   effect.
+ * *********************** */
+
+// Public: getStats gets the file statistics object.
+SWFUpload.prototype.getStats = function () {
+       return this.callFlash("GetStats");
+};
+
+// Public: setStats changes the SWFUpload statistics.  You shouldn't need to 
+// change the statistics but you can.  Changing the statistics does not
+// affect SWFUpload accept for the successful_uploads count which is used
+// by the upload_limit setting to determine how many files the user may upload.
+SWFUpload.prototype.setStats = function (statsObject) {
+       this.callFlash("SetStats", [statsObject]);
+};
+
+// Public: getFile retrieves a File object by ID or Index.  If the file is
+// not found then 'null' is returned.
+SWFUpload.prototype.getFile = function (fileID) {
+       if (typeof(fileID) === "number") {
+               return this.callFlash("GetFileByIndex", [fileID]);
+       } else {
+               return this.callFlash("GetFile", [fileID]);
+       }
+};
+
+// Public: addFileParam sets a name/value pair that will be posted with the
+// file specified by the Files ID.  If the name already exists then the
+// exiting value will be overwritten.
+SWFUpload.prototype.addFileParam = function (fileID, name, value) {
+       return this.callFlash("AddFileParam", [fileID, name, value]);
+};
+
+// Public: removeFileParam removes a previously set (by addFileParam) name/value
+// pair from the specified file.
+SWFUpload.prototype.removeFileParam = function (fileID, name) {
+       this.callFlash("RemoveFileParam", [fileID, name]);
+};
+
+// Public: setUploadUrl changes the upload_url setting.
+SWFUpload.prototype.setUploadURL = function (url) {
+       this.settings.upload_url = url.toString();
+       this.callFlash("SetUploadURL", [url]);
+};
+
+// Public: setPostParams changes the post_params setting
+SWFUpload.prototype.setPostParams = function (paramsObject) {
+       this.settings.post_params = paramsObject;
+       this.callFlash("SetPostParams", [paramsObject]);
+};
+
+// Public: addPostParam adds post name/value pair.  Each name can have only one value.
+SWFUpload.prototype.addPostParam = function (name, value) {
+       this.settings.post_params[name] = value;
+       this.callFlash("SetPostParams", [this.settings.post_params]);
+};
+
+// Public: removePostParam deletes post name/value pair.
+SWFUpload.prototype.removePostParam = function (name) {
+       delete this.settings.post_params[name];
+       this.callFlash("SetPostParams", [this.settings.post_params]);
+};
+
+// Public: setFileTypes changes the file_types setting and the file_types_description setting
+SWFUpload.prototype.setFileTypes = function (types, description) {
+       this.settings.file_types = types;
+       this.settings.file_types_description = description;
+       this.callFlash("SetFileTypes", [types, description]);
+};
+
+// Public: setFileSizeLimit changes the file_size_limit setting
+SWFUpload.prototype.setFileSizeLimit = function (fileSizeLimit) {
+       this.settings.file_size_limit = fileSizeLimit;
+       this.callFlash("SetFileSizeLimit", [fileSizeLimit]);
+};
+
+// Public: setFileUploadLimit changes the file_upload_limit setting
+SWFUpload.prototype.setFileUploadLimit = function (fileUploadLimit) {
+       this.settings.file_upload_limit = fileUploadLimit;
+       this.callFlash("SetFileUploadLimit", [fileUploadLimit]);
+};
+
+// Public: setFileQueueLimit changes the file_queue_limit setting
+SWFUpload.prototype.setFileQueueLimit = function (fileQueueLimit) {
+       this.settings.file_queue_limit = fileQueueLimit;
+       this.callFlash("SetFileQueueLimit", [fileQueueLimit]);
+};
+
+// Public: setFilePostName changes the file_post_name setting
+SWFUpload.prototype.setFilePostName = function (filePostName) {
+       this.settings.file_post_name = filePostName;
+       this.callFlash("SetFilePostName", [filePostName]);
+};
+
+// Public: setUseQueryString changes the use_query_string setting
+SWFUpload.prototype.setUseQueryString = function (useQueryString) {
+       this.settings.use_query_string = useQueryString;
+       this.callFlash("SetUseQueryString", [useQueryString]);
+};
+
+// Public: setRequeueOnError changes the requeue_on_error setting
+SWFUpload.prototype.setRequeueOnError = function (requeueOnError) {
+       this.settings.requeue_on_error = requeueOnError;
+       this.callFlash("SetRequeueOnError", [requeueOnError]);
+};
+
+// Public: setHTTPSuccess changes the http_success setting
+SWFUpload.prototype.setHTTPSuccess = function (http_status_codes) {
+       if (typeof http_status_codes === "string") {
+               http_status_codes = http_status_codes.replace(" ", "").split(",");
+       }
+       
+       this.settings.http_success = http_status_codes;
+       this.callFlash("SetHTTPSuccess", [http_status_codes]);
+};
+
+// Public: setHTTPSuccess changes the http_success setting
+SWFUpload.prototype.setAssumeSuccessTimeout = function (timeout_seconds) {
+       this.settings.assume_success_timeout = timeout_seconds;
+       this.callFlash("SetAssumeSuccessTimeout", [timeout_seconds]);
+};
+
+// Public: setDebugEnabled changes the debug_enabled setting
+SWFUpload.prototype.setDebugEnabled = function (debugEnabled) {
+       this.settings.debug_enabled = debugEnabled;
+       this.callFlash("SetDebugEnabled", [debugEnabled]);
+};
+
+// Public: setButtonImageURL loads a button image sprite
+SWFUpload.prototype.setButtonImageURL = function (buttonImageURL) {
+       if (buttonImageURL == undefined) {
+               buttonImageURL = "";
+       }
+       
+       this.settings.button_image_url = buttonImageURL;
+       this.callFlash("SetButtonImageURL", [buttonImageURL]);
+};
+
+// Public: setButtonDimensions resizes the Flash Movie and button
+SWFUpload.prototype.setButtonDimensions = function (width, height) {
+       this.settings.button_width = width;
+       this.settings.button_height = height;
+       
+       var movie = this.getMovieElement();
+       if (movie != undefined) {
+               movie.style.width = width + "px";
+               movie.style.height = height + "px";
+       }
+       
+       this.callFlash("SetButtonDimensions", [width, height]);
+};
+// Public: setButtonText Changes the text overlaid on the button
+SWFUpload.prototype.setButtonText = function (html) {
+       this.settings.button_text = html;
+       this.callFlash("SetButtonText", [html]);
+};
+// Public: setButtonTextPadding changes the top and left padding of the text overlay
+SWFUpload.prototype.setButtonTextPadding = function (left, top) {
+       this.settings.button_text_top_padding = top;
+       this.settings.button_text_left_padding = left;
+       this.callFlash("SetButtonTextPadding", [left, top]);
+};
+
+// Public: setButtonTextStyle changes the CSS used to style the HTML/Text overlaid on the button
+SWFUpload.prototype.setButtonTextStyle = function (css) {
+       this.settings.button_text_style = css;
+       this.callFlash("SetButtonTextStyle", [css]);
+};
+// Public: setButtonDisabled disables/enables the button
+SWFUpload.prototype.setButtonDisabled = function (isDisabled) {
+       this.settings.button_disabled = isDisabled;
+       this.callFlash("SetButtonDisabled", [isDisabled]);
+};
+// Public: setButtonAction sets the action that occurs when the button is clicked
+SWFUpload.prototype.setButtonAction = function (buttonAction) {
+       this.settings.button_action = buttonAction;
+       this.callFlash("SetButtonAction", [buttonAction]);
+};
+
+// Public: setButtonCursor changes the mouse cursor displayed when hovering over the button
+SWFUpload.prototype.setButtonCursor = function (cursor) {
+       this.settings.button_cursor = cursor;
+       this.callFlash("SetButtonCursor", [cursor]);
+};
+
+/* *******************************
+       Flash Event Interfaces
+       These functions are used by Flash to trigger the various
+       events.
+       
+       All these functions a Private.
+       
+       Because the ExternalInterface library is buggy the event calls
+       are added to a queue and the queue then executed by a setTimeout.
+       This ensures that events are executed in a determinate order and that
+       the ExternalInterface bugs are avoided.
+******************************* */
+
+SWFUpload.prototype.queueEvent = function (handlerName, argumentArray) {
+       // Warning: Don't call this.debug inside here or you'll create an infinite loop
+       
+       if (argumentArray == undefined) {
+               argumentArray = [];
+       } else if (!(argumentArray instanceof Array)) {
+               argumentArray = [argumentArray];
+       }
+       
+       var self = this;
+       if (typeof this.settings[handlerName] === "function") {
+               // Queue the event
+               this.eventQueue.push(function () {
+                       this.settings[handlerName].apply(this, argumentArray);
+               });
+               
+               // Execute the next queued event
+               setTimeout(function () {
+                       self.executeNextEvent();
+               }, 0);
+               
+       } else if (this.settings[handlerName] !== null) {
+               throw "Event handler " + handlerName + " is unknown or is not a function";
+       }
+};
+
+// Private: Causes the next event in the queue to be executed.  Since events are queued using a setTimeout
+// we must queue them in order to garentee that they are executed in order.
+SWFUpload.prototype.executeNextEvent = function () {
+       // Warning: Don't call this.debug inside here or you'll create an infinite loop
+
+       var  f = this.eventQueue ? this.eventQueue.shift() : null;
+       if (typeof(f) === "function") {
+               f.apply(this);
+       }
+};
+
+// Private: unescapeFileParams is part of a workaround for a flash bug where objects passed through ExternalInterface cannot have
+// properties that contain characters that are not valid for JavaScript identifiers. To work around this
+// the Flash Component escapes the parameter names and we must unescape again before passing them along.
+SWFUpload.prototype.unescapeFilePostParams = function (file) {
+       var reg = /[$]([0-9a-f]{4})/i;
+       var unescapedPost = {};
+       var uk;
+
+       if (file != undefined) {
+               for (var k in file.post) {
+                       if (file.post.hasOwnProperty(k)) {
+                               uk = k;
+                               var match;
+                               while ((match = reg.exec(uk)) !== null) {
+                                       uk = uk.replace(match[0], String.fromCharCode(parseInt("0x" + match[1], 16)));
+                               }
+                               unescapedPost[uk] = file.post[k];
+                       }
+               }
+
+               file.post = unescapedPost;
+       }
+
+       return file;
+};
+
+// Private: Called by Flash to see if JS can call in to Flash (test if External Interface is working)
+SWFUpload.prototype.testExternalInterface = function () {
+       try {
+               return this.callFlash("TestExternalInterface");
+       } catch (ex) {
+               return false;
+       }
+};
+
+// Private: This event is called by Flash when it has finished loading. Don't modify this.
+// Use the swfupload_loaded_handler event setting to execute custom code when SWFUpload has loaded.
+SWFUpload.prototype.flashReady = function () {
+       // Check that the movie element is loaded correctly with its ExternalInterface methods defined
+       var movieElement = this.getMovieElement();
+
+       if (!movieElement) {
+               this.debug("Flash called back ready but the flash movie can't be found.");
+               return;
+       }
+
+       this.cleanUp(movieElement);
+       
+       this.queueEvent("swfupload_loaded_handler");
+};
+
+// Private: removes Flash added fuctions to the DOM node to prevent memory leaks in IE.
+// This function is called by Flash each time the ExternalInterface functions are created.
+SWFUpload.prototype.cleanUp = function (movieElement) {
+       // Pro-actively unhook all the Flash functions
+       try {
+               if (this.movieElement && typeof(movieElement.CallFunction) === "unknown") { // We only want to do this in IE
+                       this.debug("Removing Flash functions hooks (this should only run in IE and should prevent memory leaks)");
+                       for (var key in movieElement) {
+                               try {
+                                       if (typeof(movieElement[key]) === "function") {
+                                               movieElement[key] = null;
+                                       }
+                               } catch (ex) {
+                               }
+                       }
+               }
+       } catch (ex1) {
+       
+       }
+
+       // Fix Flashes own cleanup code so if the SWFMovie was removed from the page
+       // it doesn't display errors.
+       window["__flash__removeCallback"] = function (instance, name) {
+               try {
+                       if (instance) {
+                               instance[name] = null;
+                       }
+               } catch (flashEx) {
+               
+               }
+       };
+
+};
+
+
+/* This is a chance to do something before the browse window opens */
+SWFUpload.prototype.fileDialogStart = function () {
+       this.queueEvent("file_dialog_start_handler");
+};
+
+
+/* Called when a file is successfully added to the queue. */
+SWFUpload.prototype.fileQueued = function (file) {
+       file = this.unescapeFilePostParams(file);
+       this.queueEvent("file_queued_handler", file);
+};
+
+
+/* Handle errors that occur when an attempt to queue a file fails. */
+SWFUpload.prototype.fileQueueError = function (file, errorCode, message) {
+       file = this.unescapeFilePostParams(file);
+       this.queueEvent("file_queue_error_handler", [file, errorCode, message]);
+};
+
+/* Called after the file dialog has closed and the selected files have been queued.
+       You could call startUpload here if you want the queued files to begin uploading immediately. */
+SWFUpload.prototype.fileDialogComplete = function (numFilesSelected, numFilesQueued, numFilesInQueue) {
+       this.queueEvent("file_dialog_complete_handler", [numFilesSelected, numFilesQueued, numFilesInQueue]);
+};
+
+SWFUpload.prototype.uploadStart = function (file) {
+       file = this.unescapeFilePostParams(file);
+       this.queueEvent("return_upload_start_handler", file);
+};
+
+SWFUpload.prototype.returnUploadStart = function (file) {
+       var returnValue;
+       if (typeof this.settings.upload_start_handler === "function") {
+               file = this.unescapeFilePostParams(file);
+               returnValue = this.settings.upload_start_handler.call(this, file);
+       } else if (this.settings.upload_start_handler != undefined) {
+               throw "upload_start_handler must be a function";
+       }
+
+       // Convert undefined to true so if nothing is returned from the upload_start_handler it is
+       // interpretted as 'true'.
+       if (returnValue === undefined) {
+               returnValue = true;
+       }
+       
+       returnValue = !!returnValue;
+       
+       this.callFlash("ReturnUploadStart", [returnValue]);
+};
+
+
+
+SWFUpload.prototype.uploadProgress = function (file, bytesComplete, bytesTotal) {
+       file = this.unescapeFilePostParams(file);
+       this.queueEvent("upload_progress_handler", [file, bytesComplete, bytesTotal]);
+};
+
+SWFUpload.prototype.uploadError = function (file, errorCode, message) {
+       file = this.unescapeFilePostParams(file);
+       this.queueEvent("upload_error_handler", [file, errorCode, message]);
+};
+
+SWFUpload.prototype.uploadSuccess = function (file, serverData, responseReceived) {
+       file = this.unescapeFilePostParams(file);
+       this.queueEvent("upload_success_handler", [file, serverData, responseReceived]);
+};
+
+SWFUpload.prototype.uploadComplete = function (file) {
+       file = this.unescapeFilePostParams(file);
+       this.queueEvent("upload_complete_handler", file);
+};
+
+/* Called by SWFUpload JavaScript and Flash functions when debug is enabled. By default it writes messages to the
+   internal debug console.  You can override this event and have messages written where you want. */
+SWFUpload.prototype.debug = function (message) {
+       this.queueEvent("debug_handler", message);
+};
+
+
+/* **********************************
+       Debug Console
+       The debug console is a self contained, in page location
+       for debug message to be sent.  The Debug Console adds
+       itself to the body if necessary.
+
+       The console is automatically scrolled as messages appear.
+       
+       If you are using your own debug handler or when you deploy to production and
+       have debug disabled you can remove these functions to reduce the file size
+       and complexity.
+********************************** */
+   
+// Private: debugMessage is the default debug_handler.  If you want to print debug messages
+// call the debug() function.  When overriding the function your own function should
+// check to see if the debug setting is true before outputting debug information.
+SWFUpload.prototype.debugMessage = function (message) {
+       if (this.settings.debug) {
+               var exceptionMessage, exceptionValues = [];
+
+               // Check for an exception object and print it nicely
+               if (typeof message === "object" && typeof message.name === "string" && typeof message.message === "string") {
+                       for (var key in message) {
+                               if (message.hasOwnProperty(key)) {
+                                       exceptionValues.push(key + ": " + message[key]);
+                               }
+                       }
+                       exceptionMessage = exceptionValues.join("\n") || "";
+                       exceptionValues = exceptionMessage.split("\n");
+                       exceptionMessage = "EXCEPTION: " + exceptionValues.join("\nEXCEPTION: ");
+                       SWFUpload.Console.writeLine(exceptionMessage);
+               } else {
+                       SWFUpload.Console.writeLine(message);
+               }
+       }
+};
+
+SWFUpload.Console = {};
+SWFUpload.Console.writeLine = function (message) {
+       var console, documentForm;
+
+       try {
+               console = document.getElementById("SWFUpload_Console");
+
+               if (!console) {
+                       documentForm = document.createElement("form");
+                       document.getElementsByTagName("body")[0].appendChild(documentForm);
+
+                       console = document.createElement("textarea");
+                       console.id = "SWFUpload_Console";
+                       console.style.fontFamily = "monospace";
+                       console.setAttribute("wrap", "off");
+                       console.wrap = "off";
+                       console.style.overflow = "auto";
+                       console.style.width = "700px";
+                       console.style.height = "350px";
+                       console.style.margin = "5px";
+                       documentForm.appendChild(console);
+               }
+
+               console.value += message + "\n";
+
+               console.scrollTop = console.scrollHeight - console.clientHeight;
+       } catch (ex) {
+               alert("Exception: " + ex.name + " Message: " + ex.message);
+       }
+};
index 31a2bad..fd5d6b9 100644 (file)
@@ -1,76 +1,76 @@
-<?php\r
-/***************************************************************\r
-*  Copyright notice\r
-*\r
-*  (c) 1999-2009 Kasper Skaarhoj (kasperYYYY@typo3.com)\r
-*  All rights reserved\r
-*\r
-*  This script is part of the TYPO3 project. The TYPO3 project is\r
-*  free software; you can redistribute it and/or modify\r
-*  it under the terms of the GNU General Public License as published by\r
-*  the Free Software Foundation; either version 2 of the License, or\r
-*  (at your option) any later version.\r
-*\r
-*  The GNU General Public License can be found at\r
-*  http://www.gnu.org/copyleft/gpl.html.\r
-*  A copy is found in the textfile GPL.txt and important notices to the license\r
-*  from the author is found in LICENSE.txt distributed with these scripts.\r
-*\r
-*\r
-*  This script is distributed in the hope that it will be useful,\r
-*  but WITHOUT ANY WARRANTY; without even the implied warranty of\r
-*  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\r
-*  GNU General Public License for more details.\r
-*\r
-*  This copyright notice MUST APPEAR in all copies of the script!\r
-***************************************************************/\r
-\r
-/**\r
- * Adds extra fields into 'media' flexform\r
- * \r
- * @package TYPO3\r
- * @subpackage cms \r
- */\r
\r
-class tx_cms_mediaItems implements t3lib_Singleton {\r
-\r
-       /**\r
-        * Load extra render types if they exist\r
-        *\r
-        * @param       array           $params: Existing types by reference\r
-        * @param       array           $conf: config array\r
-        */\r
-       public function customMediaRenderTypes(&$params, $conf) {\r
-               if (is_array ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['tslib/hooks/class.tx_cms_mediaitems.php']['customMediaRenderTypes'])) {\r
-                       foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['tslib/hooks/class.tx_cms_mediaitems.php']['customMediaRenderTypes'] as $classRef) {\r
-                               $hookObj = &t3lib_div::getUserObj($classRef);\r
-                               $hookObj->customMediaRenderTypes($params, $conf);\r
-                       }\r
-               }\r
-               \r
-               \r
-       }\r
-       \r
-       /**\r
-        * Load extra predefined media params if they exist\r
-        *\r
-        * @param       array           $params: Existing types by reference\r
-        * @param       array           $conf: config array\r
-        */\r
-       public function customMediaParams(&$params, $conf) {\r
-               \r
-               if (is_array ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['tslib/hooks/class.tx_cms_mediaitems.php']['customMediaParams'])) {\r
-                       foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['tslib/hooks/class.tx_cms_mediaitems.php']['customMediaParams'] as $classRef) {\r
-                               $hookObj = &t3lib_div::getUserObj($classRef);\r
-                               $hookObj->customMediaParams($params, $conf);\r
-                       }\r
-               }\r
-               \r
-               \r
-       }\r
-} \r
\r
-if (defined('TYPO3_MODE') && $TYPO3_CONF_VARS[TYPO3_MODE]['XCLASS']['tslib/hooks/class.tx_cms_mediaitems.php']) {\r
-       include_once($TYPO3_CONF_VARS[TYPO3_MODE]['XCLASS']['tslib/hooks/class.tx_cms_mediaitems.php']);\r
-}\r
-?>\r
+<?php
+/***************************************************************
+*  Copyright notice
+*
+*  (c) 1999-2009 Kasper Skaarhoj (kasperYYYY@typo3.com)
+*  All rights reserved
+*
+*  This script is part of the TYPO3 project. The TYPO3 project is
+*  free software; you can redistribute it and/or modify
+*  it under the terms of the GNU General Public License as published by
+*  the Free Software Foundation; either version 2 of the License, or
+*  (at your option) any later version.
+*
+*  The GNU General Public License can be found at
+*  http://www.gnu.org/copyleft/gpl.html.
+*  A copy is found in the textfile GPL.txt and important notices to the license
+*  from the author is found in LICENSE.txt distributed with these scripts.
+*
+*
+*  This script is distributed in the hope that it will be useful,
+*  but WITHOUT ANY WARRANTY; without even the implied warranty of
+*  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+*  GNU General Public License for more details.
+*
+*  This copyright notice MUST APPEAR in all copies of the script!
+***************************************************************/
+
+/**
+ * Adds extra fields into 'media' flexform
+ * 
+ * @package TYPO3
+ * @subpackage cms 
+ */
+class tx_cms_mediaItems implements t3lib_Singleton {
+
+       /**
+        * Load extra render types if they exist
+        *
+        * @param       array           $params: Existing types by reference
+        * @param       array           $conf: config array
+        */
+       public function customMediaRenderTypes(&$params, $conf) {
+               if (is_array ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['tslib/hooks/class.tx_cms_mediaitems.php']['customMediaRenderTypes'])) {
+                       foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['tslib/hooks/class.tx_cms_mediaitems.php']['customMediaRenderTypes'] as $classRef) {
+                               $hookObj = &t3lib_div::getUserObj($classRef);
+                               $hookObj->customMediaRenderTypes($params, $conf);
+                       }
+               }
+               
+               
+       }
+       
+       /**
+        * Load extra predefined media params if they exist
+        *
+        * @param       array           $params: Existing types by reference
+        * @param       array           $conf: config array
+        */
+       public function customMediaParams(&$params, $conf) {
+               
+               if (is_array ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['tslib/hooks/class.tx_cms_mediaitems.php']['customMediaParams'])) {
+                       foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['tslib/hooks/class.tx_cms_mediaitems.php']['customMediaParams'] as $classRef) {
+                               $hookObj = &t3lib_div::getUserObj($classRef);
+                               $hookObj->customMediaParams($params, $conf);
+                       }
+               }
+               
+               
+       }
+} 
+if (defined('TYPO3_MODE') && $TYPO3_CONF_VARS[TYPO3_MODE]['XCLASS']['tslib/hooks/class.tx_cms_mediaitems.php']) {
+       include_once($TYPO3_CONF_VARS[TYPO3_MODE]['XCLASS']['tslib/hooks/class.tx_cms_mediaitems.php']);
+}
+?>