github.com/insionng/yougam@v0.0.0-20170714101924-2bc18d833463/public/libs/qiniu-js-sdk-master/demo/js/plupload/moxie.js (about)

     1  /**
     2   * mOxie - multi-runtime File API & XMLHttpRequest L2 Polyfill
     3   * v1.2.0
     4   *
     5   * Copyright 2013, Moxiecode Systems AB
     6   * Released under GPL License.
     7   *
     8   * License: http://www.plupload.com/license
     9   * Contributing: http://www.plupload.com/contributing
    10   *
    11   * Date: 2014-01-16
    12   */
    13  /**
    14   * Compiled inline version. (Library mode)
    15   */
    16  
    17  /*jshint smarttabs:true, undef:true, latedef:true, curly:true, bitwise:true, camelcase:true */
    18  /*globals $code */
    19  
    20  (function(exports, undefined) {
    21  	"use strict";
    22  
    23  	var modules = {};
    24  
    25  	function require(ids, callback) {
    26  		var module, defs = [];
    27  
    28  		for (var i = 0; i < ids.length; ++i) {
    29  			module = modules[ids[i]] || resolve(ids[i]);
    30  			if (!module) {
    31  				throw 'module definition dependecy not found: ' + ids[i];
    32  			}
    33  
    34  			defs.push(module);
    35  		}
    36  
    37  		callback.apply(null, defs);
    38  	}
    39  
    40  	function define(id, dependencies, definition) {
    41  		if (typeof id !== 'string') {
    42  			throw 'invalid module definition, module id must be defined and be a string';
    43  		}
    44  
    45  		if (dependencies === undefined) {
    46  			throw 'invalid module definition, dependencies must be specified';
    47  		}
    48  
    49  		if (definition === undefined) {
    50  			throw 'invalid module definition, definition function must be specified';
    51  		}
    52  
    53  		require(dependencies, function() {
    54  			modules[id] = definition.apply(null, arguments);
    55  		});
    56  	}
    57  
    58  	function defined(id) {
    59  		return !!modules[id];
    60  	}
    61  
    62  	function resolve(id) {
    63  		var target = exports;
    64  		var fragments = id.split(/[.\/]/);
    65  
    66  		for (var fi = 0; fi < fragments.length; ++fi) {
    67  			if (!target[fragments[fi]]) {
    68  				return;
    69  			}
    70  
    71  			target = target[fragments[fi]];
    72  		}
    73  
    74  		return target;
    75  	}
    76  
    77  	function expose(ids) {
    78  		for (var i = 0; i < ids.length; i++) {
    79  			var target = exports;
    80  			var id = ids[i];
    81  			var fragments = id.split(/[.\/]/);
    82  
    83  			for (var fi = 0; fi < fragments.length - 1; ++fi) {
    84  				if (target[fragments[fi]] === undefined) {
    85  					target[fragments[fi]] = {};
    86  				}
    87  
    88  				target = target[fragments[fi]];
    89  			}
    90  
    91  			target[fragments[fragments.length - 1]] = modules[id];
    92  		}
    93  	}
    94  
    95  // Included from: src/javascript/core/utils/Basic.js
    96  
    97  /**
    98   * Basic.js
    99   *
   100   * Copyright 2013, Moxiecode Systems AB
   101   * Released under GPL License.
   102   *
   103   * License: http://www.plupload.com/license
   104   * Contributing: http://www.plupload.com/contributing
   105   */
   106  
   107  define('moxie/core/utils/Basic', [], function() {
   108  	/**
   109  	Gets the true type of the built-in object (better version of typeof).
   110  	@author Angus Croll (http://javascriptweblog.wordpress.com/)
   111  
   112  	@method typeOf
   113  	@for Utils
   114  	@static
   115  	@param {Object} o Object to check.
   116  	@return {String} Object [[Class]]
   117  	*/
   118  	var typeOf = function(o) {
   119  		var undef;
   120  
   121  		if (o === undef) {
   122  			return 'undefined';
   123  		} else if (o === null) {
   124  			return 'null';
   125  		} else if (o.nodeType) {
   126  			return 'node';
   127  		}
   128  
   129  		// the snippet below is awesome, however it fails to detect null, undefined and arguments types in IE lte 8
   130  		return ({}).toString.call(o).match(/\s([a-z|A-Z]+)/)[1].toLowerCase();
   131  	};
   132  		
   133  	/**
   134  	Extends the specified object with another object.
   135  
   136  	@method extend
   137  	@static
   138  	@param {Object} target Object to extend.
   139  	@param {Object} [obj]* Multiple objects to extend with.
   140  	@return {Object} Same as target, the extended object.
   141  	*/
   142  	var extend = function(target) {
   143  		var undef;
   144  
   145  		each(arguments, function(arg, i) {
   146  			if (i > 0) {
   147  				each(arg, function(value, key) {
   148  					if (value !== undef) {
   149  						if (typeOf(target[key]) === typeOf(value) && !!~inArray(typeOf(value), ['array', 'object'])) {
   150  							extend(target[key], value);
   151  						} else {
   152  							target[key] = value;
   153  						}
   154  					}
   155  				});
   156  			}
   157  		});
   158  		return target;
   159  	};
   160  		
   161  	/**
   162  	Executes the callback function for each item in array/object. If you return false in the
   163  	callback it will break the loop.
   164  
   165  	@method each
   166  	@static
   167  	@param {Object} obj Object to iterate.
   168  	@param {function} callback Callback function to execute for each item.
   169  	*/
   170  	var each = function(obj, callback) {
   171  		var length, key, i, undef;
   172  
   173  		if (obj) {
   174  			try {
   175  				length = obj.length;
   176  			} catch(ex) {
   177  				length = undef;
   178  			}
   179  
   180  			if (length === undef) {
   181  				// Loop object items
   182  				for (key in obj) {
   183  					if (obj.hasOwnProperty(key)) {
   184  						if (callback(obj[key], key) === false) {
   185  							return;
   186  						}
   187  					}
   188  				}
   189  			} else {
   190  				// Loop array items
   191  				for (i = 0; i < length; i++) {
   192  					if (callback(obj[i], i) === false) {
   193  						return;
   194  					}
   195  				}
   196  			}
   197  		}
   198  	};
   199  
   200  	/**
   201  	Checks if object is empty.
   202  	
   203  	@method isEmptyObj
   204  	@static
   205  	@param {Object} o Object to check.
   206  	@return {Boolean}
   207  	*/
   208  	var isEmptyObj = function(obj) {
   209  		var prop;
   210  
   211  		if (!obj || typeOf(obj) !== 'object') {
   212  			return true;
   213  		}
   214  
   215  		for (prop in obj) {
   216  			return false;
   217  		}
   218  
   219  		return true;
   220  	};
   221  
   222  	/**
   223  	Recieve an array of functions (usually async) to call in sequence, each  function
   224  	receives a callback as first argument that it should call, when it completes. Finally,
   225  	after everything is complete, main callback is called. Passing truthy value to the
   226  	callback as a first argument will interrupt the sequence and invoke main callback
   227  	immediately.
   228  
   229  	@method inSeries
   230  	@static
   231  	@param {Array} queue Array of functions to call in sequence
   232  	@param {Function} cb Main callback that is called in the end, or in case of erro
   233  	*/
   234  	var inSeries = function(queue, cb) {
   235  		var i = 0, length = queue.length;
   236  
   237  		if (typeOf(cb) !== 'function') {
   238  			cb = function() {};
   239  		}
   240  
   241  		if (!queue || !queue.length) {
   242  			cb();
   243  		}
   244  
   245  		function callNext(i) {
   246  			if (typeOf(queue[i]) === 'function') {
   247  				queue[i](function(error) {
   248  					/*jshint expr:true */
   249  					++i < length && !error ? callNext(i) : cb(error);
   250  				});
   251  			}
   252  		}
   253  		callNext(i);
   254  	};
   255  
   256  
   257  	/**
   258  	Recieve an array of functions (usually async) to call in parallel, each  function
   259  	receives a callback as first argument that it should call, when it completes. After 
   260  	everything is complete, main callback is called. Passing truthy value to the
   261  	callback as a first argument will interrupt the process and invoke main callback
   262  	immediately.
   263  
   264  	@method inParallel
   265  	@static
   266  	@param {Array} queue Array of functions to call in sequence
   267  	@param {Function} cb Main callback that is called in the end, or in case of erro
   268  	*/
   269  	var inParallel = function(queue, cb) {
   270  		var count = 0, num = queue.length, cbArgs = new Array(num);
   271  
   272  		each(queue, function(fn, i) {
   273  			fn(function(error) {
   274  				if (error) {
   275  					return cb(error);
   276  				}
   277  				
   278  				var args = [].slice.call(arguments);
   279  				args.shift(); // strip error - undefined or not
   280  
   281  				cbArgs[i] = args;
   282  				count++;
   283  
   284  				if (count === num) {
   285  					cbArgs.unshift(null);
   286  					cb.apply(this, cbArgs);
   287  				} 
   288  			});
   289  		});
   290  	};
   291  	
   292  	
   293  	/**
   294  	Find an element in array and return it's index if present, otherwise return -1.
   295  	
   296  	@method inArray
   297  	@static
   298  	@param {Mixed} needle Element to find
   299  	@param {Array} array
   300  	@return {Int} Index of the element, or -1 if not found
   301  	*/
   302  	var inArray = function(needle, array) {
   303  		if (array) {
   304  			if (Array.prototype.indexOf) {
   305  				return Array.prototype.indexOf.call(array, needle);
   306  			}
   307  		
   308  			for (var i = 0, length = array.length; i < length; i++) {
   309  				if (array[i] === needle) {
   310  					return i;
   311  				}
   312  			}
   313  		}
   314  		return -1;
   315  	};
   316  
   317  
   318  	/**
   319  	Returns elements of first array if they are not present in second. And false - otherwise.
   320  
   321  	@private
   322  	@method arrayDiff
   323  	@param {Array} needles
   324  	@param {Array} array
   325  	@return {Array|Boolean}
   326  	*/
   327  	var arrayDiff = function(needles, array) {
   328  		var diff = [];
   329  
   330  		if (typeOf(needles) !== 'array') {
   331  			needles = [needles];
   332  		}
   333  
   334  		if (typeOf(array) !== 'array') {
   335  			array = [array];
   336  		}
   337  
   338  		for (var i in needles) {
   339  			if (inArray(needles[i], array) === -1) {
   340  				diff.push(needles[i]);
   341  			}	
   342  		}
   343  		return diff.length ? diff : false;
   344  	};
   345  
   346  
   347  	/**
   348  	Find intersection of two arrays.
   349  
   350  	@private
   351  	@method arrayIntersect
   352  	@param {Array} array1
   353  	@param {Array} array2
   354  	@return {Array} Intersection of two arrays or null if there is none
   355  	*/
   356  	var arrayIntersect = function(array1, array2) {
   357  		var result = [];
   358  		each(array1, function(item) {
   359  			if (inArray(item, array2) !== -1) {
   360  				result.push(item);
   361  			}
   362  		});
   363  		return result.length ? result : null;
   364  	};
   365  	
   366  	
   367  	/**
   368  	Forces anything into an array.
   369  	
   370  	@method toArray
   371  	@static
   372  	@param {Object} obj Object with length field.
   373  	@return {Array} Array object containing all items.
   374  	*/
   375  	var toArray = function(obj) {
   376  		var i, arr = [];
   377  
   378  		for (i = 0; i < obj.length; i++) {
   379  			arr[i] = obj[i];
   380  		}
   381  
   382  		return arr;
   383  	};
   384  	
   385  			
   386  	/**
   387  	Generates an unique ID. This is 99.99% unique since it takes the current time and 5 random numbers.
   388  	The only way a user would be able to get the same ID is if the two persons at the same exact milisecond manages
   389  	to get 5 the same random numbers between 0-65535 it also uses a counter so each call will be guaranteed to be page unique.
   390  	It's more probable for the earth to be hit with an ansteriod. Y
   391  	
   392  	@method guid
   393  	@static
   394  	@param {String} prefix to prepend (by default 'o' will be prepended).
   395  	@method guid
   396  	@return {String} Virtually unique id.
   397  	*/
   398  	var guid = (function() {
   399  		var counter = 0;
   400  		
   401  		return function(prefix) {
   402  			var guid = new Date().getTime().toString(32), i;
   403  
   404  			for (i = 0; i < 5; i++) {
   405  				guid += Math.floor(Math.random() * 65535).toString(32);
   406  			}
   407  			
   408  			return (prefix || 'o_') + guid + (counter++).toString(32);
   409  		};
   410  	}());
   411  	
   412  
   413  	/**
   414  	Trims white spaces around the string
   415  	
   416  	@method trim
   417  	@static
   418  	@param {String} str
   419  	@return {String}
   420  	*/
   421  	var trim = function(str) {
   422  		if (!str) {
   423  			return str;
   424  		}
   425  		return String.prototype.trim ? String.prototype.trim.call(str) : str.toString().replace(/^\s*/, '').replace(/\s*$/, '');
   426  	};
   427  
   428  
   429  	/**
   430  	Parses the specified size string into a byte value. For example 10kb becomes 10240.
   431  	
   432  	@method parseSizeStr
   433  	@static
   434  	@param {String/Number} size String to parse or number to just pass through.
   435  	@return {Number} Size in bytes.
   436  	*/
   437  	var parseSizeStr = function(size) {
   438  		if (typeof(size) !== 'string') {
   439  			return size;
   440  		}
   441  		
   442  		var muls = {
   443  				t: 1099511627776,
   444  				g: 1073741824,
   445  				m: 1048576,
   446  				k: 1024
   447  			},
   448  			mul;
   449  
   450  		size = /^([0-9]+)([mgk]?)$/.exec(size.toLowerCase().replace(/[^0-9mkg]/g, ''));
   451  		mul = size[2];
   452  		size = +size[1];
   453  		
   454  		if (muls.hasOwnProperty(mul)) {
   455  			size *= muls[mul];
   456  		}
   457  		return size;
   458  	};
   459  	
   460  
   461  	return {
   462  		guid: guid,
   463  		typeOf: typeOf,
   464  		extend: extend,
   465  		each: each,
   466  		isEmptyObj: isEmptyObj,
   467  		inSeries: inSeries,
   468  		inParallel: inParallel,
   469  		inArray: inArray,
   470  		arrayDiff: arrayDiff,
   471  		arrayIntersect: arrayIntersect,
   472  		toArray: toArray,
   473  		trim: trim,
   474  		parseSizeStr: parseSizeStr
   475  	};
   476  });
   477  
   478  // Included from: src/javascript/core/I18n.js
   479  
   480  /**
   481   * I18n.js
   482   *
   483   * Copyright 2013, Moxiecode Systems AB
   484   * Released under GPL License.
   485   *
   486   * License: http://www.plupload.com/license
   487   * Contributing: http://www.plupload.com/contributing
   488   */
   489  
   490  define("moxie/core/I18n", [
   491  	"moxie/core/utils/Basic"
   492  ], function(Basic) {
   493  	var i18n = {};
   494  
   495  	return {
   496  		/**
   497  		 * Extends the language pack object with new items.
   498  		 *
   499  		 * @param {Object} pack Language pack items to add.
   500  		 * @return {Object} Extended language pack object.
   501  		 */
   502  		addI18n: function(pack) {
   503  			return Basic.extend(i18n, pack);
   504  		},
   505  
   506  		/**
   507  		 * Translates the specified string by checking for the english string in the language pack lookup.
   508  		 *
   509  		 * @param {String} str String to look for.
   510  		 * @return {String} Translated string or the input string if it wasn't found.
   511  		 */
   512  		translate: function(str) {
   513  			return i18n[str] || str;
   514  		},
   515  
   516  		/**
   517  		 * Shortcut for translate function
   518  		 *
   519  		 * @param {String} str String to look for.
   520  		 * @return {String} Translated string or the input string if it wasn't found.
   521  		 */
   522  		_: function(str) {
   523  			return this.translate(str);
   524  		},
   525  
   526  		/**
   527  		 * Pseudo sprintf implementation - simple way to replace tokens with specified values.
   528  		 *
   529  		 * @param {String} str String with tokens
   530  		 * @return {String} String with replaced tokens
   531  		 */
   532  		sprintf: function(str) {
   533  			var args = [].slice.call(arguments, 1);
   534  
   535  			return str.replace(/%[a-z]/g, function() {
   536  				var value = args.shift();
   537  				return Basic.typeOf(value) !== 'undefined' ? value : '';
   538  			});
   539  		}
   540  	};
   541  });
   542  
   543  // Included from: src/javascript/core/utils/Mime.js
   544  
   545  /**
   546   * Mime.js
   547   *
   548   * Copyright 2013, Moxiecode Systems AB
   549   * Released under GPL License.
   550   *
   551   * License: http://www.plupload.com/license
   552   * Contributing: http://www.plupload.com/contributing
   553   */
   554  
   555  define("moxie/core/utils/Mime", [
   556  	"moxie/core/utils/Basic",
   557  	"moxie/core/I18n"
   558  ], function(Basic, I18n) {
   559  	
   560  	var mimeData = "" +
   561  		"application/msword,doc dot," +
   562  		"application/pdf,pdf," +
   563  		"application/pgp-signature,pgp," +
   564  		"application/postscript,ps ai eps," +
   565  		"application/rtf,rtf," +
   566  		"application/vnd.ms-excel,xls xlb," +
   567  		"application/vnd.ms-powerpoint,ppt pps pot," +
   568  		"application/zip,zip," +
   569  		"application/x-shockwave-flash,swf swfl," +
   570  		"application/vnd.openxmlformats-officedocument.wordprocessingml.document,docx," +
   571  		"application/vnd.openxmlformats-officedocument.wordprocessingml.template,dotx," +
   572  		"application/vnd.openxmlformats-officedocument.spreadsheetml.sheet,xlsx," +
   573  		"application/vnd.openxmlformats-officedocument.presentationml.presentation,pptx," +
   574  		"application/vnd.openxmlformats-officedocument.presentationml.template,potx," +
   575  		"application/vnd.openxmlformats-officedocument.presentationml.slideshow,ppsx," +
   576  		"application/x-javascript,js," +
   577  		"application/json,json," +
   578  		"audio/mpeg,mp3 mpga mpega mp2," +
   579  		"audio/x-wav,wav," +
   580  		"audio/x-m4a,m4a," +
   581  		"audio/ogg,oga ogg," +
   582  		"audio/aiff,aiff aif," +
   583  		"audio/flac,flac," +
   584  		"audio/aac,aac," +
   585  		"audio/ac3,ac3," +
   586  		"audio/x-ms-wma,wma," +
   587  		"image/bmp,bmp," +
   588  		"image/gif,gif," +
   589  		"image/jpeg,jpg jpeg jpe," +
   590  		"image/photoshop,psd," +
   591  		"image/png,png," +
   592  		"image/svg+xml,svg svgz," +
   593  		"image/tiff,tiff tif," +
   594  		"text/plain,asc txt text diff log," +
   595  		"text/html,htm html xhtml," +
   596  		"text/css,css," +
   597  		"text/csv,csv," +
   598  		"text/rtf,rtf," +
   599  		"video/mpeg,mpeg mpg mpe m2v," +
   600  		"video/quicktime,qt mov," +
   601  		"video/mp4,mp4," +
   602  		"video/x-m4v,m4v," +
   603  		"video/x-flv,flv," +
   604  		"video/x-ms-wmv,wmv," +
   605  		"video/avi,avi," +
   606  		"video/webm,webm," +
   607  		"video/3gpp,3gpp 3gp," +
   608  		"video/3gpp2,3g2," +
   609  		"video/vnd.rn-realvideo,rv," +
   610  		"video/ogg,ogv," + 
   611  		"video/x-matroska,mkv," +
   612  		"application/vnd.oasis.opendocument.formula-template,otf," +
   613  		"application/octet-stream,exe";
   614  	
   615  	
   616  	var Mime = {
   617  
   618  		mimes: {},
   619  
   620  		extensions: {},
   621  
   622  		// Parses the default mime types string into a mimes and extensions lookup maps
   623  		addMimeType: function (mimeData) {
   624  			var items = mimeData.split(/,/), i, ii, ext;
   625  			
   626  			for (i = 0; i < items.length; i += 2) {
   627  				ext = items[i + 1].split(/ /);
   628  
   629  				// extension to mime lookup
   630  				for (ii = 0; ii < ext.length; ii++) {
   631  					this.mimes[ext[ii]] = items[i];
   632  				}
   633  				// mime to extension lookup
   634  				this.extensions[items[i]] = ext;
   635  			}
   636  		},
   637  
   638  
   639  		extList2mimes: function (filters, addMissingExtensions) {
   640  			var self = this, ext, i, ii, type, mimes = [];
   641  			
   642  			// convert extensions to mime types list
   643  			for (i = 0; i < filters.length; i++) {
   644  				ext = filters[i].extensions.split(/\s*,\s*/);
   645  
   646  				for (ii = 0; ii < ext.length; ii++) {
   647  					
   648  					// if there's an asterisk in the list, then accept attribute is not required
   649  					if (ext[ii] === '*') {
   650  						return [];
   651  					}
   652  					
   653  					type = self.mimes[ext[ii]];
   654  					if (!type) {
   655  						if (addMissingExtensions && /^\w+$/.test(ext[ii])) {
   656  							mimes.push('.' + ext[ii]);
   657  						} else {
   658  							return []; // accept all
   659  						}
   660  					} else if (Basic.inArray(type, mimes) === -1) {
   661  						mimes.push(type);
   662  					}
   663  				}
   664  			}
   665  			return mimes;
   666  		},
   667  
   668  
   669  		mimes2exts: function(mimes) {
   670  			var self = this, exts = [];
   671  			
   672  			Basic.each(mimes, function(mime) {
   673  				if (mime === '*') {
   674  					exts = [];
   675  					return false;
   676  				}
   677  
   678  				// check if this thing looks like mime type
   679  				var m = mime.match(/^(\w+)\/(\*|\w+)$/);
   680  				if (m) {
   681  					if (m[2] === '*') { 
   682  						// wildcard mime type detected
   683  						Basic.each(self.extensions, function(arr, mime) {
   684  							if ((new RegExp('^' + m[1] + '/')).test(mime)) {
   685  								[].push.apply(exts, self.extensions[mime]);
   686  							}
   687  						});
   688  					} else if (self.extensions[mime]) {
   689  						[].push.apply(exts, self.extensions[mime]);
   690  					}
   691  				}
   692  			});
   693  			return exts;
   694  		},
   695  
   696  
   697  		mimes2extList: function(mimes) {
   698  			var accept = [], exts = [];
   699  
   700  			if (Basic.typeOf(mimes) === 'string') {
   701  				mimes = Basic.trim(mimes).split(/\s*,\s*/);
   702  			}
   703  
   704  			exts = this.mimes2exts(mimes);
   705  			
   706  			accept.push({
   707  				title: I18n.translate('Files'),
   708  				extensions: exts.length ? exts.join(',') : '*'
   709  			});
   710  			
   711  			// save original mimes string
   712  			accept.mimes = mimes;
   713  
   714  			return accept;
   715  		},
   716  
   717  
   718  		getFileExtension: function(fileName) {
   719  			var matches = fileName && fileName.match(/\.([^.]+)$/);
   720  			if (matches) {
   721  				return matches[1].toLowerCase();
   722  			}
   723  			return '';
   724  		},
   725  
   726  		getFileMime: function(fileName) {
   727  			return this.mimes[this.getFileExtension(fileName)] || '';
   728  		}
   729  	};
   730  
   731  	Mime.addMimeType(mimeData);
   732  
   733  	return Mime;
   734  });
   735  
   736  // Included from: src/javascript/core/utils/Env.js
   737  
   738  /**
   739   * Env.js
   740   *
   741   * Copyright 2013, Moxiecode Systems AB
   742   * Released under GPL License.
   743   *
   744   * License: http://www.plupload.com/license
   745   * Contributing: http://www.plupload.com/contributing
   746   */
   747  
   748  define("moxie/core/utils/Env", [
   749  	"moxie/core/utils/Basic"
   750  ], function(Basic) {
   751  	
   752  	// UAParser.js v0.6.2
   753  	// Lightweight JavaScript-based User-Agent string parser
   754  	// https://github.com/faisalman/ua-parser-js
   755  	//
   756  	// Copyright © 2012-2013 Faisalman <fyzlman@gmail.com>
   757  	// Dual licensed under GPLv2 & MIT
   758  
   759  	var UAParser = (function (undefined) {
   760  
   761  	    //////////////
   762  	    // Constants
   763  	    /////////////
   764  
   765  
   766  	    var EMPTY       = '',
   767  	        UNKNOWN     = '?',
   768  	        FUNC_TYPE   = 'function',
   769  	        UNDEF_TYPE  = 'undefined',
   770  	        OBJ_TYPE    = 'object',
   771  	        MAJOR       = 'major',
   772  	        MODEL       = 'model',
   773  	        NAME        = 'name',
   774  	        TYPE        = 'type',
   775  	        VENDOR      = 'vendor',
   776  	        VERSION     = 'version',
   777  	        ARCHITECTURE= 'architecture',
   778  	        CONSOLE     = 'console',
   779  	        MOBILE      = 'mobile',
   780  	        TABLET      = 'tablet';
   781  
   782  
   783  	    ///////////
   784  	    // Helper
   785  	    //////////
   786  
   787  
   788  	    var util = {
   789  	        has : function (str1, str2) {
   790  	            return str2.toLowerCase().indexOf(str1.toLowerCase()) !== -1;
   791  	        },
   792  	        lowerize : function (str) {
   793  	            return str.toLowerCase();
   794  	        }
   795  	    };
   796  
   797  
   798  	    ///////////////
   799  	    // Map helper
   800  	    //////////////
   801  
   802  
   803  	    var mapper = {
   804  
   805  	        rgx : function () {
   806  
   807  	            // loop through all regexes maps
   808  	            for (var result, i = 0, j, k, p, q, matches, match, args = arguments; i < args.length; i += 2) {
   809  
   810  	                var regex = args[i],       // even sequence (0,2,4,..)
   811  	                    props = args[i + 1];   // odd sequence (1,3,5,..)
   812  
   813  	                // construct object barebones
   814  	                if (typeof(result) === UNDEF_TYPE) {
   815  	                    result = {};
   816  	                    for (p in props) {
   817  	                        q = props[p];
   818  	                        if (typeof(q) === OBJ_TYPE) {
   819  	                            result[q[0]] = undefined;
   820  	                        } else {
   821  	                            result[q] = undefined;
   822  	                        }
   823  	                    }
   824  	                }
   825  
   826  	                // try matching uastring with regexes
   827  	                for (j = k = 0; j < regex.length; j++) {
   828  	                    matches = regex[j].exec(this.getUA());
   829  	                    if (!!matches) {
   830  	                        for (p = 0; p < props.length; p++) {
   831  	                            match = matches[++k];
   832  	                            q = props[p];
   833  	                            // check if given property is actually array
   834  	                            if (typeof(q) === OBJ_TYPE && q.length > 0) {
   835  	                                if (q.length == 2) {
   836  	                                    if (typeof(q[1]) == FUNC_TYPE) {
   837  	                                        // assign modified match
   838  	                                        result[q[0]] = q[1].call(this, match);
   839  	                                    } else {
   840  	                                        // assign given value, ignore regex match
   841  	                                        result[q[0]] = q[1];
   842  	                                    }
   843  	                                } else if (q.length == 3) {
   844  	                                    // check whether function or regex
   845  	                                    if (typeof(q[1]) === FUNC_TYPE && !(q[1].exec && q[1].test)) {
   846  	                                        // call function (usually string mapper)
   847  	                                        result[q[0]] = match ? q[1].call(this, match, q[2]) : undefined;
   848  	                                    } else {
   849  	                                        // sanitize match using given regex
   850  	                                        result[q[0]] = match ? match.replace(q[1], q[2]) : undefined;
   851  	                                    }
   852  	                                } else if (q.length == 4) {
   853  	                                        result[q[0]] = match ? q[3].call(this, match.replace(q[1], q[2])) : undefined;
   854  	                                }
   855  	                            } else {
   856  	                                result[q] = match ? match : undefined;
   857  	                            }
   858  	                        }
   859  	                        break;
   860  	                    }
   861  	                }
   862  
   863  	                if(!!matches) break; // break the loop immediately if match found
   864  	            }
   865  	            return result;
   866  	        },
   867  
   868  	        str : function (str, map) {
   869  
   870  	            for (var i in map) {
   871  	                // check if array
   872  	                if (typeof(map[i]) === OBJ_TYPE && map[i].length > 0) {
   873  	                    for (var j = 0; j < map[i].length; j++) {
   874  	                        if (util.has(map[i][j], str)) {
   875  	                            return (i === UNKNOWN) ? undefined : i;
   876  	                        }
   877  	                    }
   878  	                } else if (util.has(map[i], str)) {
   879  	                    return (i === UNKNOWN) ? undefined : i;
   880  	                }
   881  	            }
   882  	            return str;
   883  	        }
   884  	    };
   885  
   886  
   887  	    ///////////////
   888  	    // String map
   889  	    //////////////
   890  
   891  
   892  	    var maps = {
   893  
   894  	        browser : {
   895  	            oldsafari : {
   896  	                major : {
   897  	                    '1' : ['/8', '/1', '/3'],
   898  	                    '2' : '/4',
   899  	                    '?' : '/'
   900  	                },
   901  	                version : {
   902  	                    '1.0'   : '/8',
   903  	                    '1.2'   : '/1',
   904  	                    '1.3'   : '/3',
   905  	                    '2.0'   : '/412',
   906  	                    '2.0.2' : '/416',
   907  	                    '2.0.3' : '/417',
   908  	                    '2.0.4' : '/419',
   909  	                    '?'     : '/'
   910  	                }
   911  	            }
   912  	        },
   913  
   914  	        device : {
   915  	            sprint : {
   916  	                model : {
   917  	                    'Evo Shift 4G' : '7373KT'
   918  	                },
   919  	                vendor : {
   920  	                    'HTC'       : 'APA',
   921  	                    'Sprint'    : 'Sprint'
   922  	                }
   923  	            }
   924  	        },
   925  
   926  	        os : {
   927  	            windows : {
   928  	                version : {
   929  	                    'ME'        : '4.90',
   930  	                    'NT 3.11'   : 'NT3.51',
   931  	                    'NT 4.0'    : 'NT4.0',
   932  	                    '2000'      : 'NT 5.0',
   933  	                    'XP'        : ['NT 5.1', 'NT 5.2'],
   934  	                    'Vista'     : 'NT 6.0',
   935  	                    '7'         : 'NT 6.1',
   936  	                    '8'         : 'NT 6.2',
   937  	                    '8.1'       : 'NT 6.3',
   938  	                    'RT'        : 'ARM'
   939  	                }
   940  	            }
   941  	        }
   942  	    };
   943  
   944  
   945  	    //////////////
   946  	    // Regex map
   947  	    /////////////
   948  
   949  
   950  	    var regexes = {
   951  
   952  	        browser : [[
   953  
   954  	            // Presto based
   955  	            /(opera\smini)\/((\d+)?[\w\.-]+)/i,                                 // Opera Mini
   956  	            /(opera\s[mobiletab]+).+version\/((\d+)?[\w\.-]+)/i,                // Opera Mobi/Tablet
   957  	            /(opera).+version\/((\d+)?[\w\.]+)/i,                               // Opera > 9.80
   958  	            /(opera)[\/\s]+((\d+)?[\w\.]+)/i                                    // Opera < 9.80
   959  	            
   960  	            ], [NAME, VERSION, MAJOR], [
   961  
   962  	            /\s(opr)\/((\d+)?[\w\.]+)/i                                         // Opera Webkit
   963  	            ], [[NAME, 'Opera'], VERSION, MAJOR], [
   964  
   965  	            // Mixed
   966  	            /(kindle)\/((\d+)?[\w\.]+)/i,                                       // Kindle
   967  	            /(lunascape|maxthon|netfront|jasmine|blazer)[\/\s]?((\d+)?[\w\.]+)*/i,
   968  	                                                                                // Lunascape/Maxthon/Netfront/Jasmine/Blazer
   969  
   970  	            // Trident based
   971  	            /(avant\s|iemobile|slim|baidu)(?:browser)?[\/\s]?((\d+)?[\w\.]*)/i,
   972  	                                                                                // Avant/IEMobile/SlimBrowser/Baidu
   973  	            /(?:ms|\()(ie)\s((\d+)?[\w\.]+)/i,                                  // Internet Explorer
   974  
   975  	            // Webkit/KHTML based
   976  	            /(rekonq)((?:\/)[\w\.]+)*/i,                                        // Rekonq
   977  	            /(chromium|flock|rockmelt|midori|epiphany|silk|skyfire|ovibrowser|bolt|iron)\/((\d+)?[\w\.-]+)/i
   978  	                                                                                // Chromium/Flock/RockMelt/Midori/Epiphany/Silk/Skyfire/Bolt/Iron
   979  	            ], [NAME, VERSION, MAJOR], [
   980  
   981  	            /(trident).+rv[:\s]((\d+)?[\w\.]+).+like\sgecko/i                   // IE11
   982  	            ], [[NAME, 'IE'], VERSION, MAJOR], [
   983  
   984  	            /(yabrowser)\/((\d+)?[\w\.]+)/i                                     // Yandex
   985  	            ], [[NAME, 'Yandex'], VERSION, MAJOR], [
   986  
   987  	            /(comodo_dragon)\/((\d+)?[\w\.]+)/i                                 // Comodo Dragon
   988  	            ], [[NAME, /_/g, ' '], VERSION, MAJOR], [
   989  
   990  	            /(chrome|omniweb|arora|[tizenoka]{5}\s?browser)\/v?((\d+)?[\w\.]+)/i
   991  	                                                                                // Chrome/OmniWeb/Arora/Tizen/Nokia
   992  	            ], [NAME, VERSION, MAJOR], [
   993  
   994  	            /(dolfin)\/((\d+)?[\w\.]+)/i                                        // Dolphin
   995  	            ], [[NAME, 'Dolphin'], VERSION, MAJOR], [
   996  
   997  	            /((?:android.+)crmo|crios)\/((\d+)?[\w\.]+)/i                       // Chrome for Android/iOS
   998  	            ], [[NAME, 'Chrome'], VERSION, MAJOR], [
   999  
  1000  	            /((?:android.+))version\/((\d+)?[\w\.]+)\smobile\ssafari/i          // Android Browser
  1001  	            ], [[NAME, 'Android Browser'], VERSION, MAJOR], [
  1002  
  1003  	            /version\/((\d+)?[\w\.]+).+?mobile\/\w+\s(safari)/i                 // Mobile Safari
  1004  	            ], [VERSION, MAJOR, [NAME, 'Mobile Safari']], [
  1005  
  1006  	            /version\/((\d+)?[\w\.]+).+?(mobile\s?safari|safari)/i              // Safari & Safari Mobile
  1007  	            ], [VERSION, MAJOR, NAME], [
  1008  
  1009  	            /webkit.+?(mobile\s?safari|safari)((\/[\w\.]+))/i                   // Safari < 3.0
  1010  	            ], [NAME, [MAJOR, mapper.str, maps.browser.oldsafari.major], [VERSION, mapper.str, maps.browser.oldsafari.version]], [
  1011  
  1012  	            /(konqueror)\/((\d+)?[\w\.]+)/i,                                    // Konqueror
  1013  	            /(webkit|khtml)\/((\d+)?[\w\.]+)/i
  1014  	            ], [NAME, VERSION, MAJOR], [
  1015  
  1016  	            // Gecko based
  1017  	            /(navigator|netscape)\/((\d+)?[\w\.-]+)/i                           // Netscape
  1018  	            ], [[NAME, 'Netscape'], VERSION, MAJOR], [
  1019  	            /(swiftfox)/i,                                                      // Swiftfox
  1020  	            /(icedragon|iceweasel|camino|chimera|fennec|maemo\sbrowser|minimo|conkeror)[\/\s]?((\d+)?[\w\.\+]+)/i,
  1021  	                                                                                // IceDragon/Iceweasel/Camino/Chimera/Fennec/Maemo/Minimo/Conkeror
  1022  	            /(firefox|seamonkey|k-meleon|icecat|iceape|firebird|phoenix)\/((\d+)?[\w\.-]+)/i,
  1023  	                                                                                // Firefox/SeaMonkey/K-Meleon/IceCat/IceApe/Firebird/Phoenix
  1024  	            /(mozilla)\/((\d+)?[\w\.]+).+rv\:.+gecko\/\d+/i,                    // Mozilla
  1025  
  1026  	            // Other
  1027  	            /(uc\s?browser|polaris|lynx|dillo|icab|doris|amaya|w3m|netsurf|qqbrowser)[\/\s]?((\d+)?[\w\.]+)/i,
  1028  	                                                                                // UCBrowser/Polaris/Lynx/Dillo/iCab/Doris/Amaya/w3m/NetSurf/QQBrowser
  1029  	            /(links)\s\(((\d+)?[\w\.]+)/i,                                      // Links
  1030  	            /(gobrowser)\/?((\d+)?[\w\.]+)*/i,                                  // GoBrowser
  1031  	            /(ice\s?browser)\/v?((\d+)?[\w\._]+)/i,                             // ICE Browser
  1032  	            /(mosaic)[\/\s]((\d+)?[\w\.]+)/i                                    // Mosaic
  1033  	            ], [NAME, VERSION, MAJOR]
  1034  	        ],
  1035  
  1036  	        engine : [[
  1037  
  1038  	            /(presto)\/([\w\.]+)/i,                                             // Presto
  1039  	            /(webkit|trident|netfront|netsurf|amaya|lynx|w3m)\/([\w\.]+)/i,     // WebKit/Trident/NetFront/NetSurf/Amaya/Lynx/w3m
  1040  	            /(khtml|tasman|links)[\/\s]\(?([\w\.]+)/i,                          // KHTML/Tasman/Links
  1041  	            /(icab)[\/\s]([23]\.[\d\.]+)/i                                      // iCab
  1042  	            ], [NAME, VERSION], [
  1043  
  1044  	            /rv\:([\w\.]+).*(gecko)/i                                           // Gecko
  1045  	            ], [VERSION, NAME]
  1046  	        ],
  1047  
  1048  	        os : [[
  1049  
  1050  	            // Windows based
  1051  	            /(windows)\snt\s6\.2;\s(arm)/i,                                     // Windows RT
  1052  	            /(windows\sphone(?:\sos)*|windows\smobile|windows)[\s\/]?([ntce\d\.\s]+\w)/i
  1053  	            ], [NAME, [VERSION, mapper.str, maps.os.windows.version]], [
  1054  	            /(win(?=3|9|n)|win\s9x\s)([nt\d\.]+)/i
  1055  	            ], [[NAME, 'Windows'], [VERSION, mapper.str, maps.os.windows.version]], [
  1056  
  1057  	            // Mobile/Embedded OS
  1058  	            /\((bb)(10);/i                                                      // BlackBerry 10
  1059  	            ], [[NAME, 'BlackBerry'], VERSION], [
  1060  	            /(blackberry)\w*\/?([\w\.]+)*/i,                                    // Blackberry
  1061  	            /(tizen)\/([\w\.]+)/i,                                              // Tizen
  1062  	            /(android|webos|palm\os|qnx|bada|rim\stablet\sos|meego)[\/\s-]?([\w\.]+)*/i
  1063  	                                                                                // Android/WebOS/Palm/QNX/Bada/RIM/MeeGo
  1064  	            ], [NAME, VERSION], [
  1065  	            /(symbian\s?os|symbos|s60(?=;))[\/\s-]?([\w\.]+)*/i                 // Symbian
  1066  	            ], [[NAME, 'Symbian'], VERSION],[
  1067  	            /mozilla.+\(mobile;.+gecko.+firefox/i                               // Firefox OS
  1068  	            ], [[NAME, 'Firefox OS'], VERSION], [
  1069  
  1070  	            // Console
  1071  	            /(nintendo|playstation)\s([wids3portablevu]+)/i,                    // Nintendo/Playstation
  1072  
  1073  	            // GNU/Linux based
  1074  	            /(mint)[\/\s\(]?(\w+)*/i,                                           // Mint
  1075  	            /(joli|[kxln]?ubuntu|debian|[open]*suse|gentoo|arch|slackware|fedora|mandriva|centos|pclinuxos|redhat|zenwalk)[\/\s-]?([\w\.-]+)*/i,
  1076  	                                                                                // Joli/Ubuntu/Debian/SUSE/Gentoo/Arch/Slackware
  1077  	                                                                                // Fedora/Mandriva/CentOS/PCLinuxOS/RedHat/Zenwalk
  1078  	            /(hurd|linux)\s?([\w\.]+)*/i,                                       // Hurd/Linux
  1079  	            /(gnu)\s?([\w\.]+)*/i                                               // GNU
  1080  	            ], [NAME, VERSION], [
  1081  
  1082  	            /(cros)\s[\w]+\s([\w\.]+\w)/i                                       // Chromium OS
  1083  	            ], [[NAME, 'Chromium OS'], VERSION],[
  1084  
  1085  	            // Solaris
  1086  	            /(sunos)\s?([\w\.]+\d)*/i                                           // Solaris
  1087  	            ], [[NAME, 'Solaris'], VERSION], [
  1088  
  1089  	            // BSD based
  1090  	            /\s([frentopc-]{0,4}bsd|dragonfly)\s?([\w\.]+)*/i                   // FreeBSD/NetBSD/OpenBSD/PC-BSD/DragonFly
  1091  	            ], [NAME, VERSION],[
  1092  
  1093  	            /(ip[honead]+)(?:.*os\s*([\w]+)*\slike\smac|;\sopera)/i             // iOS
  1094  	            ], [[NAME, 'iOS'], [VERSION, /_/g, '.']], [
  1095  
  1096  	            /(mac\sos\sx)\s?([\w\s\.]+\w)*/i                                    // Mac OS
  1097  	            ], [NAME, [VERSION, /_/g, '.']], [
  1098  
  1099  	            // Other
  1100  	            /(haiku)\s(\w+)/i,                                                  // Haiku
  1101  	            /(aix)\s((\d)(?=\.|\)|\s)[\w\.]*)*/i,                               // AIX
  1102  	            /(macintosh|mac(?=_powerpc)|plan\s9|minix|beos|os\/2|amigaos|morphos|risc\sos)/i,
  1103  	                                                                                // Plan9/Minix/BeOS/OS2/AmigaOS/MorphOS/RISCOS
  1104  	            /(unix)\s?([\w\.]+)*/i                                              // UNIX
  1105  	            ], [NAME, VERSION]
  1106  	        ]
  1107  	    };
  1108  
  1109  
  1110  	    /////////////////
  1111  	    // Constructor
  1112  	    ////////////////
  1113  
  1114  
  1115  	    var UAParser = function (uastring) {
  1116  
  1117  	        var ua = uastring || ((window && window.navigator && window.navigator.userAgent) ? window.navigator.userAgent : EMPTY);
  1118  
  1119  	        this.getBrowser = function () {
  1120  	            return mapper.rgx.apply(this, regexes.browser);
  1121  	        };
  1122  	        this.getEngine = function () {
  1123  	            return mapper.rgx.apply(this, regexes.engine);
  1124  	        };
  1125  	        this.getOS = function () {
  1126  	            return mapper.rgx.apply(this, regexes.os);
  1127  	        };
  1128  	        this.getResult = function() {
  1129  	            return {
  1130  	                ua      : this.getUA(),
  1131  	                browser : this.getBrowser(),
  1132  	                engine  : this.getEngine(),
  1133  	                os      : this.getOS()
  1134  	            };
  1135  	        };
  1136  	        this.getUA = function () {
  1137  	            return ua;
  1138  	        };
  1139  	        this.setUA = function (uastring) {
  1140  	            ua = uastring;
  1141  	            return this;
  1142  	        };
  1143  	        this.setUA(ua);
  1144  	    };
  1145  
  1146  	    return new UAParser().getResult();
  1147  	})();
  1148  
  1149  
  1150  	function version_compare(v1, v2, operator) {
  1151  	  // From: http://phpjs.org/functions
  1152  	  // +      original by: Philippe Jausions (http://pear.php.net/user/jausions)
  1153  	  // +      original by: Aidan Lister (http://aidanlister.com/)
  1154  	  // + reimplemented by: Kankrelune (http://www.webfaktory.info/)
  1155  	  // +      improved by: Brett Zamir (http://brett-zamir.me)
  1156  	  // +      improved by: Scott Baker
  1157  	  // +      improved by: Theriault
  1158  	  // *        example 1: version_compare('8.2.5rc', '8.2.5a');
  1159  	  // *        returns 1: 1
  1160  	  // *        example 2: version_compare('8.2.50', '8.2.52', '<');
  1161  	  // *        returns 2: true
  1162  	  // *        example 3: version_compare('5.3.0-dev', '5.3.0');
  1163  	  // *        returns 3: -1
  1164  	  // *        example 4: version_compare('4.1.0.52','4.01.0.51');
  1165  	  // *        returns 4: 1
  1166  
  1167  	  // Important: compare must be initialized at 0.
  1168  	  var i = 0,
  1169  	    x = 0,
  1170  	    compare = 0,
  1171  	    // vm maps textual PHP versions to negatives so they're less than 0.
  1172  	    // PHP currently defines these as CASE-SENSITIVE. It is important to
  1173  	    // leave these as negatives so that they can come before numerical versions
  1174  	    // and as if no letters were there to begin with.
  1175  	    // (1alpha is < 1 and < 1.1 but > 1dev1)
  1176  	    // If a non-numerical value can't be mapped to this table, it receives
  1177  	    // -7 as its value.
  1178  	    vm = {
  1179  	      'dev': -6,
  1180  	      'alpha': -5,
  1181  	      'a': -5,
  1182  	      'beta': -4,
  1183  	      'b': -4,
  1184  	      'RC': -3,
  1185  	      'rc': -3,
  1186  	      '#': -2,
  1187  	      'p': 1,
  1188  	      'pl': 1
  1189  	    },
  1190  	    // This function will be called to prepare each version argument.
  1191  	    // It replaces every _, -, and + with a dot.
  1192  	    // It surrounds any nonsequence of numbers/dots with dots.
  1193  	    // It replaces sequences of dots with a single dot.
  1194  	    //    version_compare('4..0', '4.0') == 0
  1195  	    // Important: A string of 0 length needs to be converted into a value
  1196  	    // even less than an unexisting value in vm (-7), hence [-8].
  1197  	    // It's also important to not strip spaces because of this.
  1198  	    //   version_compare('', ' ') == 1
  1199  	    prepVersion = function (v) {
  1200  	      v = ('' + v).replace(/[_\-+]/g, '.');
  1201  	      v = v.replace(/([^.\d]+)/g, '.$1.').replace(/\.{2,}/g, '.');
  1202  	      return (!v.length ? [-8] : v.split('.'));
  1203  	    },
  1204  	    // This converts a version component to a number.
  1205  	    // Empty component becomes 0.
  1206  	    // Non-numerical component becomes a negative number.
  1207  	    // Numerical component becomes itself as an integer.
  1208  	    numVersion = function (v) {
  1209  	      return !v ? 0 : (isNaN(v) ? vm[v] || -7 : parseInt(v, 10));
  1210  	    };
  1211  
  1212  	  v1 = prepVersion(v1);
  1213  	  v2 = prepVersion(v2);
  1214  	  x = Math.max(v1.length, v2.length);
  1215  	  for (i = 0; i < x; i++) {
  1216  	    if (v1[i] == v2[i]) {
  1217  	      continue;
  1218  	    }
  1219  	    v1[i] = numVersion(v1[i]);
  1220  	    v2[i] = numVersion(v2[i]);
  1221  	    if (v1[i] < v2[i]) {
  1222  	      compare = -1;
  1223  	      break;
  1224  	    } else if (v1[i] > v2[i]) {
  1225  	      compare = 1;
  1226  	      break;
  1227  	    }
  1228  	  }
  1229  	  if (!operator) {
  1230  	    return compare;
  1231  	  }
  1232  
  1233  	  // Important: operator is CASE-SENSITIVE.
  1234  	  // "No operator" seems to be treated as "<."
  1235  	  // Any other values seem to make the function return null.
  1236  	  switch (operator) {
  1237  	  case '>':
  1238  	  case 'gt':
  1239  	    return (compare > 0);
  1240  	  case '>=':
  1241  	  case 'ge':
  1242  	    return (compare >= 0);
  1243  	  case '<=':
  1244  	  case 'le':
  1245  	    return (compare <= 0);
  1246  	  case '==':
  1247  	  case '=':
  1248  	  case 'eq':
  1249  	    return (compare === 0);
  1250  	  case '<>':
  1251  	  case '!=':
  1252  	  case 'ne':
  1253  	    return (compare !== 0);
  1254  	  case '':
  1255  	  case '<':
  1256  	  case 'lt':
  1257  	    return (compare < 0);
  1258  	  default:
  1259  	    return null;
  1260  	  }
  1261  	}
  1262  
  1263  
  1264  	var can = (function() {
  1265  		var caps = {
  1266  				define_property: (function() {
  1267  					/* // currently too much extra code required, not exactly worth it
  1268  					try { // as of IE8, getters/setters are supported only on DOM elements
  1269  						var obj = {};
  1270  						if (Object.defineProperty) {
  1271  							Object.defineProperty(obj, 'prop', {
  1272  								enumerable: true,
  1273  								configurable: true
  1274  							});
  1275  							return true;
  1276  						}
  1277  					} catch(ex) {}
  1278  
  1279  					if (Object.prototype.__defineGetter__ && Object.prototype.__defineSetter__) {
  1280  						return true;
  1281  					}*/
  1282  					return false;
  1283  				}()),
  1284  
  1285  				create_canvas: (function() {
  1286  					// On the S60 and BB Storm, getContext exists, but always returns undefined
  1287  					// so we actually have to call getContext() to verify
  1288  					// github.com/Modernizr/Modernizr/issues/issue/97/
  1289  					var el = document.createElement('canvas');
  1290  					return !!(el.getContext && el.getContext('2d'));
  1291  				}()),
  1292  
  1293  				return_response_type: function(responseType) {
  1294  					try {
  1295  						if (Basic.inArray(responseType, ['', 'text', 'document']) !== -1) {
  1296  							return true;
  1297  						} else if (window.XMLHttpRequest) {
  1298  							var xhr = new XMLHttpRequest();
  1299  							xhr.open('get', '/'); // otherwise Gecko throws an exception
  1300  							if ('responseType' in xhr) {
  1301  								xhr.responseType = responseType;
  1302  								// as of 23.0.1271.64, Chrome switched from throwing exception to merely logging it to the console (why? o why?)
  1303  								if (xhr.responseType !== responseType) {
  1304  									return false;
  1305  								}
  1306  								return true;
  1307  							}
  1308  						}
  1309  					} catch (ex) {}
  1310  					return false;
  1311  				},
  1312  
  1313  				// ideas for this heavily come from Modernizr (http://modernizr.com/)
  1314  				use_data_uri: (function() {
  1315  					var du = new Image();
  1316  
  1317  					du.onload = function() {
  1318  						caps.use_data_uri = (du.width === 1 && du.height === 1);
  1319  					};
  1320  					
  1321  					setTimeout(function() {
  1322  						du.src = "";
  1323  					}, 1);
  1324  					return false;
  1325  				}()),
  1326  
  1327  				use_data_uri_over32kb: function() { // IE8
  1328  					return caps.use_data_uri && (Env.browser !== 'IE' || Env.version >= 9);
  1329  				},
  1330  
  1331  				use_data_uri_of: function(bytes) {
  1332  					return (caps.use_data_uri && bytes < 33000 || caps.use_data_uri_over32kb());
  1333  				},
  1334  
  1335  				use_fileinput: function() {
  1336  					var el = document.createElement('input');
  1337  					el.setAttribute('type', 'file');
  1338  					return !el.disabled;
  1339  				}
  1340  			};
  1341  
  1342  		return function(cap) {
  1343  			var args = [].slice.call(arguments);
  1344  			args.shift(); // shift of cap
  1345  			return Basic.typeOf(caps[cap]) === 'function' ? caps[cap].apply(this, args) : !!caps[cap];
  1346  		};
  1347  	}());
  1348  
  1349  
  1350  	var Env = {
  1351  		can: can,
  1352  		
  1353  		browser: UAParser.browser.name,
  1354  		version: parseFloat(UAParser.browser.major),
  1355  		os: UAParser.os.name, // everybody intuitively types it in a lowercase for some reason
  1356  		osVersion: UAParser.os.version,
  1357  
  1358  		verComp: version_compare,
  1359  		
  1360  		swf_url: "../flash/Moxie.swf",
  1361  		xap_url: "../silverlight/Moxie.xap",
  1362  		global_event_dispatcher: "moxie.core.EventTarget.instance.dispatchEvent"
  1363  	};
  1364  
  1365  	// for backward compatibility
  1366  	// @deprecated Use `Env.os` instead
  1367  	Env.OS = Env.os;
  1368  
  1369  	return Env;
  1370  });
  1371  
  1372  // Included from: src/javascript/core/utils/Dom.js
  1373  
  1374  /**
  1375   * Dom.js
  1376   *
  1377   * Copyright 2013, Moxiecode Systems AB
  1378   * Released under GPL License.
  1379   *
  1380   * License: http://www.plupload.com/license
  1381   * Contributing: http://www.plupload.com/contributing
  1382   */
  1383  
  1384  define('moxie/core/utils/Dom', ['moxie/core/utils/Env'], function(Env) {
  1385  
  1386  	/**
  1387  	Get DOM Element by it's id.
  1388  
  1389  	@method get
  1390  	@for Utils
  1391  	@param {String} id Identifier of the DOM Element
  1392  	@return {DOMElement}
  1393  	*/
  1394  	var get = function(id) {
  1395  		if (typeof id !== 'string') {
  1396  			return id;
  1397  		}
  1398  		return document.getElementById(id);
  1399  	};
  1400  
  1401  	/**
  1402  	Checks if specified DOM element has specified class.
  1403  
  1404  	@method hasClass
  1405  	@static
  1406  	@param {Object} obj DOM element like object to add handler to.
  1407  	@param {String} name Class name
  1408  	*/
  1409  	var hasClass = function(obj, name) {
  1410  		if (!obj.className) {
  1411  			return false;
  1412  		}
  1413  
  1414  		var regExp = new RegExp("(^|\\s+)"+name+"(\\s+|$)");
  1415  		return regExp.test(obj.className);
  1416  	};
  1417  
  1418  	/**
  1419  	Adds specified className to specified DOM element.
  1420  
  1421  	@method addClass
  1422  	@static
  1423  	@param {Object} obj DOM element like object to add handler to.
  1424  	@param {String} name Class name
  1425  	*/
  1426  	var addClass = function(obj, name) {
  1427  		if (!hasClass(obj, name)) {
  1428  			obj.className = !obj.className ? name : obj.className.replace(/\s+$/, '') + ' ' + name;
  1429  		}
  1430  	};
  1431  
  1432  	/**
  1433  	Removes specified className from specified DOM element.
  1434  
  1435  	@method removeClass
  1436  	@static
  1437  	@param {Object} obj DOM element like object to add handler to.
  1438  	@param {String} name Class name
  1439  	*/
  1440  	var removeClass = function(obj, name) {
  1441  		if (obj.className) {
  1442  			var regExp = new RegExp("(^|\\s+)"+name+"(\\s+|$)");
  1443  			obj.className = obj.className.replace(regExp, function($0, $1, $2) {
  1444  				return $1 === ' ' && $2 === ' ' ? ' ' : '';
  1445  			});
  1446  		}
  1447  	};
  1448  
  1449  	/**
  1450  	Returns a given computed style of a DOM element.
  1451  
  1452  	@method getStyle
  1453  	@static
  1454  	@param {Object} obj DOM element like object.
  1455  	@param {String} name Style you want to get from the DOM element
  1456  	*/
  1457  	var getStyle = function(obj, name) {
  1458  		if (obj.currentStyle) {
  1459  			return obj.currentStyle[name];
  1460  		} else if (window.getComputedStyle) {
  1461  			return window.getComputedStyle(obj, null)[name];
  1462  		}
  1463  	};
  1464  
  1465  
  1466  	/**
  1467  	Returns the absolute x, y position of an Element. The position will be returned in a object with x, y fields.
  1468  
  1469  	@method getPos
  1470  	@static
  1471  	@param {Element} node HTML element or element id to get x, y position from.
  1472  	@param {Element} root Optional root element to stop calculations at.
  1473  	@return {object} Absolute position of the specified element object with x, y fields.
  1474  	*/
  1475  	var getPos = function(node, root) {
  1476  		var x = 0, y = 0, parent, doc = document, nodeRect, rootRect;
  1477  
  1478  		node = node;
  1479  		root = root || doc.body;
  1480  
  1481  		// Returns the x, y cordinate for an element on IE 6 and IE 7
  1482  		function getIEPos(node) {
  1483  			var bodyElm, rect, x = 0, y = 0;
  1484  
  1485  			if (node) {
  1486  				rect = node.getBoundingClientRect();
  1487  				bodyElm = doc.compatMode === "CSS1Compat" ? doc.documentElement : doc.body;
  1488  				x = rect.left + bodyElm.scrollLeft;
  1489  				y = rect.top + bodyElm.scrollTop;
  1490  			}
  1491  
  1492  			return {
  1493  				x : x,
  1494  				y : y
  1495  			};
  1496  		}
  1497  
  1498  		// Use getBoundingClientRect on IE 6 and IE 7 but not on IE 8 in standards mode
  1499  		if (node && node.getBoundingClientRect && Env.browser === 'IE' && (!doc.documentMode || doc.documentMode < 8)) {
  1500  			nodeRect = getIEPos(node);
  1501  			rootRect = getIEPos(root);
  1502  
  1503  			return {
  1504  				x : nodeRect.x - rootRect.x,
  1505  				y : nodeRect.y - rootRect.y
  1506  			};
  1507  		}
  1508  
  1509  		parent = node;
  1510  		while (parent && parent != root && parent.nodeType) {
  1511  			x += parent.offsetLeft || 0;
  1512  			y += parent.offsetTop || 0;
  1513  			parent = parent.offsetParent;
  1514  		}
  1515  
  1516  		parent = node.parentNode;
  1517  		while (parent && parent != root && parent.nodeType) {
  1518  			x -= parent.scrollLeft || 0;
  1519  			y -= parent.scrollTop || 0;
  1520  			parent = parent.parentNode;
  1521  		}
  1522  
  1523  		return {
  1524  			x : x,
  1525  			y : y
  1526  		};
  1527  	};
  1528  
  1529  	/**
  1530  	Returns the size of the specified node in pixels.
  1531  
  1532  	@method getSize
  1533  	@static
  1534  	@param {Node} node Node to get the size of.
  1535  	@return {Object} Object with a w and h property.
  1536  	*/
  1537  	var getSize = function(node) {
  1538  		return {
  1539  			w : node.offsetWidth || node.clientWidth,
  1540  			h : node.offsetHeight || node.clientHeight
  1541  		};
  1542  	};
  1543  
  1544  	return {
  1545  		get: get,
  1546  		hasClass: hasClass,
  1547  		addClass: addClass,
  1548  		removeClass: removeClass,
  1549  		getStyle: getStyle,
  1550  		getPos: getPos,
  1551  		getSize: getSize
  1552  	};
  1553  });
  1554  
  1555  // Included from: src/javascript/core/Exceptions.js
  1556  
  1557  /**
  1558   * Exceptions.js
  1559   *
  1560   * Copyright 2013, Moxiecode Systems AB
  1561   * Released under GPL License.
  1562   *
  1563   * License: http://www.plupload.com/license
  1564   * Contributing: http://www.plupload.com/contributing
  1565   */
  1566  
  1567  define('moxie/core/Exceptions', [
  1568  	'moxie/core/utils/Basic'
  1569  ], function(Basic) {
  1570  	function _findKey(obj, value) {
  1571  		var key;
  1572  		for (key in obj) {
  1573  			if (obj[key] === value) {
  1574  				return key;
  1575  			}
  1576  		}
  1577  		return null;
  1578  	}
  1579  
  1580  	return {
  1581  		RuntimeError: (function() {
  1582  			var namecodes = {
  1583  				NOT_INIT_ERR: 1,
  1584  				NOT_SUPPORTED_ERR: 9,
  1585  				JS_ERR: 4
  1586  			};
  1587  
  1588  			function RuntimeError(code) {
  1589  				this.code = code;
  1590  				this.name = _findKey(namecodes, code);
  1591  				this.message = this.name + ": RuntimeError " + this.code;
  1592  			}
  1593  			
  1594  			Basic.extend(RuntimeError, namecodes);
  1595  			RuntimeError.prototype = Error.prototype;
  1596  			return RuntimeError;
  1597  		}()),
  1598  		
  1599  		OperationNotAllowedException: (function() {
  1600  			
  1601  			function OperationNotAllowedException(code) {
  1602  				this.code = code;
  1603  				this.name = 'OperationNotAllowedException';
  1604  			}
  1605  			
  1606  			Basic.extend(OperationNotAllowedException, {
  1607  				NOT_ALLOWED_ERR: 1
  1608  			});
  1609  			
  1610  			OperationNotAllowedException.prototype = Error.prototype;
  1611  			
  1612  			return OperationNotAllowedException;
  1613  		}()),
  1614  
  1615  		ImageError: (function() {
  1616  			var namecodes = {
  1617  				WRONG_FORMAT: 1,
  1618  				MAX_RESOLUTION_ERR: 2
  1619  			};
  1620  
  1621  			function ImageError(code) {
  1622  				this.code = code;
  1623  				this.name = _findKey(namecodes, code);
  1624  				this.message = this.name + ": ImageError " + this.code;
  1625  			}
  1626  			
  1627  			Basic.extend(ImageError, namecodes);
  1628  			ImageError.prototype = Error.prototype;
  1629  
  1630  			return ImageError;
  1631  		}()),
  1632  
  1633  		FileException: (function() {
  1634  			var namecodes = {
  1635  				NOT_FOUND_ERR: 1,
  1636  				SECURITY_ERR: 2,
  1637  				ABORT_ERR: 3,
  1638  				NOT_READABLE_ERR: 4,
  1639  				ENCODING_ERR: 5,
  1640  				NO_MODIFICATION_ALLOWED_ERR: 6,
  1641  				INVALID_STATE_ERR: 7,
  1642  				SYNTAX_ERR: 8
  1643  			};
  1644  
  1645  			function FileException(code) {
  1646  				this.code = code;
  1647  				this.name = _findKey(namecodes, code);
  1648  				this.message = this.name + ": FileException " + this.code;
  1649  			}
  1650  			
  1651  			Basic.extend(FileException, namecodes);
  1652  			FileException.prototype = Error.prototype;
  1653  			return FileException;
  1654  		}()),
  1655  		
  1656  		DOMException: (function() {
  1657  			var namecodes = {
  1658  				INDEX_SIZE_ERR: 1,
  1659  				DOMSTRING_SIZE_ERR: 2,
  1660  				HIERARCHY_REQUEST_ERR: 3,
  1661  				WRONG_DOCUMENT_ERR: 4,
  1662  				INVALID_CHARACTER_ERR: 5,
  1663  				NO_DATA_ALLOWED_ERR: 6,
  1664  				NO_MODIFICATION_ALLOWED_ERR: 7,
  1665  				NOT_FOUND_ERR: 8,
  1666  				NOT_SUPPORTED_ERR: 9,
  1667  				INUSE_ATTRIBUTE_ERR: 10,
  1668  				INVALID_STATE_ERR: 11,
  1669  				SYNTAX_ERR: 12,
  1670  				INVALID_MODIFICATION_ERR: 13,
  1671  				NAMESPACE_ERR: 14,
  1672  				INVALID_ACCESS_ERR: 15,
  1673  				VALIDATION_ERR: 16,
  1674  				TYPE_MISMATCH_ERR: 17,
  1675  				SECURITY_ERR: 18,
  1676  				NETWORK_ERR: 19,
  1677  				ABORT_ERR: 20,
  1678  				URL_MISMATCH_ERR: 21,
  1679  				QUOTA_EXCEEDED_ERR: 22,
  1680  				TIMEOUT_ERR: 23,
  1681  				INVALID_NODE_TYPE_ERR: 24,
  1682  				DATA_CLONE_ERR: 25
  1683  			};
  1684  
  1685  			function DOMException(code) {
  1686  				this.code = code;
  1687  				this.name = _findKey(namecodes, code);
  1688  				this.message = this.name + ": DOMException " + this.code;
  1689  			}
  1690  			
  1691  			Basic.extend(DOMException, namecodes);
  1692  			DOMException.prototype = Error.prototype;
  1693  			return DOMException;
  1694  		}()),
  1695  		
  1696  		EventException: (function() {
  1697  			function EventException(code) {
  1698  				this.code = code;
  1699  				this.name = 'EventException';
  1700  			}
  1701  			
  1702  			Basic.extend(EventException, {
  1703  				UNSPECIFIED_EVENT_TYPE_ERR: 0
  1704  			});
  1705  			
  1706  			EventException.prototype = Error.prototype;
  1707  			
  1708  			return EventException;
  1709  		}())
  1710  	};
  1711  });
  1712  
  1713  // Included from: src/javascript/core/EventTarget.js
  1714  
  1715  /**
  1716   * EventTarget.js
  1717   *
  1718   * Copyright 2013, Moxiecode Systems AB
  1719   * Released under GPL License.
  1720   *
  1721   * License: http://www.plupload.com/license
  1722   * Contributing: http://www.plupload.com/contributing
  1723   */
  1724  
  1725  define('moxie/core/EventTarget', [
  1726  	'moxie/core/Exceptions',
  1727  	'moxie/core/utils/Basic'
  1728  ], function(x, Basic) {
  1729  	/**
  1730  	Parent object for all event dispatching components and objects
  1731  
  1732  	@class EventTarget
  1733  	@constructor EventTarget
  1734  	*/
  1735  	function EventTarget() {
  1736  		// hash of event listeners by object uid
  1737  		var eventpool = {};
  1738  				
  1739  		Basic.extend(this, {
  1740  			
  1741  			/**
  1742  			Unique id of the event dispatcher, usually overriden by children
  1743  
  1744  			@property uid
  1745  			@type String
  1746  			*/
  1747  			uid: null,
  1748  			
  1749  			/**
  1750  			Can be called from within a child  in order to acquire uniqie id in automated manner
  1751  
  1752  			@method init
  1753  			*/
  1754  			init: function() {
  1755  				if (!this.uid) {
  1756  					this.uid = Basic.guid('uid_');
  1757  				}
  1758  			},
  1759  
  1760  			/**
  1761  			Register a handler to a specific event dispatched by the object
  1762  
  1763  			@method addEventListener
  1764  			@param {String} type Type or basically a name of the event to subscribe to
  1765  			@param {Function} fn Callback function that will be called when event happens
  1766  			@param {Number} [priority=0] Priority of the event handler - handlers with higher priorities will be called first
  1767  			@param {Object} [scope=this] A scope to invoke event handler in
  1768  			*/
  1769  			addEventListener: function(type, fn, priority, scope) {
  1770  				var self = this, list;
  1771  				
  1772  				type = Basic.trim(type);
  1773  				
  1774  				if (/\s/.test(type)) {
  1775  					// multiple event types were passed for one handler
  1776  					Basic.each(type.split(/\s+/), function(type) {
  1777  						self.addEventListener(type, fn, priority, scope);
  1778  					});
  1779  					return;
  1780  				}
  1781  				
  1782  				type = type.toLowerCase();
  1783  				priority = parseInt(priority, 10) || 0;
  1784  				
  1785  				list = eventpool[this.uid] && eventpool[this.uid][type] || [];
  1786  				list.push({fn : fn, priority : priority, scope : scope || this});
  1787  				
  1788  				if (!eventpool[this.uid]) {
  1789  					eventpool[this.uid] = {};
  1790  				}
  1791  				eventpool[this.uid][type] = list;
  1792  			},
  1793  			
  1794  			/**
  1795  			Check if any handlers were registered to the specified event
  1796  
  1797  			@method hasEventListener
  1798  			@param {String} type Type or basically a name of the event to check
  1799  			@return {Mixed} Returns a handler if it was found and false, if - not
  1800  			*/
  1801  			hasEventListener: function(type) {
  1802  				return type ? !!(eventpool[this.uid] && eventpool[this.uid][type]) : !!eventpool[this.uid];
  1803  			},
  1804  			
  1805  			/**
  1806  			Unregister the handler from the event, or if former was not specified - unregister all handlers
  1807  
  1808  			@method removeEventListener
  1809  			@param {String} type Type or basically a name of the event
  1810  			@param {Function} [fn] Handler to unregister
  1811  			*/
  1812  			removeEventListener: function(type, fn) {
  1813  				type = type.toLowerCase();
  1814  	
  1815  				var list = eventpool[this.uid] && eventpool[this.uid][type], i;
  1816  	
  1817  				if (list) {
  1818  					if (fn) {
  1819  						for (i = list.length - 1; i >= 0; i--) {
  1820  							if (list[i].fn === fn) {
  1821  								list.splice(i, 1);
  1822  								break;
  1823  							}
  1824  						}
  1825  					} else {
  1826  						list = [];
  1827  					}
  1828  	
  1829  					// delete event list if it has become empty
  1830  					if (!list.length) {
  1831  						delete eventpool[this.uid][type];
  1832  						
  1833  						// and object specific entry in a hash if it has no more listeners attached
  1834  						if (Basic.isEmptyObj(eventpool[this.uid])) {
  1835  							delete eventpool[this.uid];
  1836  						}
  1837  					}
  1838  				}
  1839  			},
  1840  			
  1841  			/**
  1842  			Remove all event handlers from the object
  1843  
  1844  			@method removeAllEventListeners
  1845  			*/
  1846  			removeAllEventListeners: function() {
  1847  				if (eventpool[this.uid]) {
  1848  					delete eventpool[this.uid];
  1849  				}
  1850  			},
  1851  			
  1852  			/**
  1853  			Dispatch the event
  1854  
  1855  			@method dispatchEvent
  1856  			@param {String/Object} Type of event or event object to dispatch
  1857  			@param {Mixed} [...] Variable number of arguments to be passed to a handlers
  1858  			@return {Boolean} true by default and false if any handler returned false
  1859  			*/
  1860  			dispatchEvent: function(type) {
  1861  				var uid, list, args, tmpEvt, evt = {}, result = true, undef;
  1862  				
  1863  				if (Basic.typeOf(type) !== 'string') {
  1864  					// we can't use original object directly (because of Silverlight)
  1865  					tmpEvt = type;
  1866  
  1867  					if (Basic.typeOf(tmpEvt.type) === 'string') {
  1868  						type = tmpEvt.type;
  1869  
  1870  						if (tmpEvt.total !== undef && tmpEvt.loaded !== undef) { // progress event
  1871  							evt.total = tmpEvt.total;
  1872  							evt.loaded = tmpEvt.loaded;
  1873  						}
  1874  						evt.async = tmpEvt.async || false;
  1875  					} else {
  1876  						throw new x.EventException(x.EventException.UNSPECIFIED_EVENT_TYPE_ERR);
  1877  					}
  1878  				}
  1879  				
  1880  				// check if event is meant to be dispatched on an object having specific uid
  1881  				if (type.indexOf('::') !== -1) {
  1882  					(function(arr) {
  1883  						uid = arr[0];
  1884  						type = arr[1];
  1885  					}(type.split('::')));
  1886  				} else {
  1887  					uid = this.uid;
  1888  				}
  1889  				
  1890  				type = type.toLowerCase();
  1891  								
  1892  				list = eventpool[uid] && eventpool[uid][type];
  1893  
  1894  				if (list) {
  1895  					// sort event list by prority
  1896  					list.sort(function(a, b) { return b.priority - a.priority; });
  1897  					
  1898  					args = [].slice.call(arguments);
  1899  					
  1900  					// first argument will be pseudo-event object
  1901  					args.shift();
  1902  					evt.type = type;
  1903  					args.unshift(evt);
  1904  
  1905  					// Dispatch event to all listeners
  1906  					var queue = [];
  1907  					Basic.each(list, function(handler) {
  1908  						// explicitly set the target, otherwise events fired from shims do not get it
  1909  						args[0].target = handler.scope;
  1910  						// if event is marked as async, detach the handler
  1911  						if (evt.async) {
  1912  							queue.push(function(cb) {
  1913  								setTimeout(function() {
  1914  									cb(handler.fn.apply(handler.scope, args) === false);
  1915  								}, 1);
  1916  							});
  1917  						} else {
  1918  							queue.push(function(cb) {
  1919  								cb(handler.fn.apply(handler.scope, args) === false); // if handler returns false stop propagation
  1920  							});
  1921  						}
  1922  					});
  1923  					if (queue.length) {
  1924  						Basic.inSeries(queue, function(err) {
  1925  							result = !err;
  1926  						});
  1927  					}
  1928  				}
  1929  				return result;
  1930  			},
  1931  			
  1932  			/**
  1933  			Alias for addEventListener
  1934  
  1935  			@method bind
  1936  			@protected
  1937  			*/
  1938  			bind: function() {
  1939  				this.addEventListener.apply(this, arguments);
  1940  			},
  1941  			
  1942  			/**
  1943  			Alias for removeEventListener
  1944  
  1945  			@method unbind
  1946  			@protected
  1947  			*/
  1948  			unbind: function() {
  1949  				this.removeEventListener.apply(this, arguments);
  1950  			},
  1951  			
  1952  			/**
  1953  			Alias for removeAllEventListeners
  1954  
  1955  			@method unbindAll
  1956  			@protected
  1957  			*/
  1958  			unbindAll: function() {
  1959  				this.removeAllEventListeners.apply(this, arguments);
  1960  			},
  1961  			
  1962  			/**
  1963  			Alias for dispatchEvent
  1964  
  1965  			@method trigger
  1966  			@protected
  1967  			*/
  1968  			trigger: function() {
  1969  				return this.dispatchEvent.apply(this, arguments);
  1970  			},
  1971  			
  1972  			
  1973  			/**
  1974  			Converts properties of on[event] type to corresponding event handlers,
  1975  			is used to avoid extra hassle around the process of calling them back
  1976  
  1977  			@method convertEventPropsToHandlers
  1978  			@private
  1979  			*/
  1980  			convertEventPropsToHandlers: function(handlers) {
  1981  				var h;
  1982  						
  1983  				if (Basic.typeOf(handlers) !== 'array') {
  1984  					handlers = [handlers];
  1985  				}
  1986  
  1987  				for (var i = 0; i < handlers.length; i++) {
  1988  					h = 'on' + handlers[i];
  1989  					
  1990  					if (Basic.typeOf(this[h]) === 'function') {
  1991  						this.addEventListener(handlers[i], this[h]);
  1992  					} else if (Basic.typeOf(this[h]) === 'undefined') {
  1993  						this[h] = null; // object must have defined event properties, even if it doesn't make use of them
  1994  					}
  1995  				}
  1996  			}
  1997  			
  1998  		});
  1999  	}
  2000  
  2001  	EventTarget.instance = new EventTarget(); 
  2002  
  2003  	return EventTarget;
  2004  });
  2005  
  2006  // Included from: src/javascript/core/utils/Encode.js
  2007  
  2008  /**
  2009   * Encode.js
  2010   *
  2011   * Copyright 2013, Moxiecode Systems AB
  2012   * Released under GPL License.
  2013   *
  2014   * License: http://www.plupload.com/license
  2015   * Contributing: http://www.plupload.com/contributing
  2016   */
  2017  
  2018  define('moxie/core/utils/Encode', [], function() {
  2019  
  2020  	/**
  2021  	Encode string with UTF-8
  2022  
  2023  	@method utf8_encode
  2024  	@for Utils
  2025  	@static
  2026  	@param {String} str String to encode
  2027  	@return {String} UTF-8 encoded string
  2028  	*/
  2029  	var utf8_encode = function(str) {
  2030  		return unescape(encodeURIComponent(str));
  2031  	};
  2032  	
  2033  	/**
  2034  	Decode UTF-8 encoded string
  2035  
  2036  	@method utf8_decode
  2037  	@static
  2038  	@param {String} str String to decode
  2039  	@return {String} Decoded string
  2040  	*/
  2041  	var utf8_decode = function(str_data) {
  2042  		return decodeURIComponent(escape(str_data));
  2043  	};
  2044  	
  2045  	/**
  2046  	Decode Base64 encoded string (uses browser's default method if available),
  2047  	from: https://raw.github.com/kvz/phpjs/master/functions/url/base64_decode.js
  2048  
  2049  	@method atob
  2050  	@static
  2051  	@param {String} data String to decode
  2052  	@return {String} Decoded string
  2053  	*/
  2054  	var atob = function(data, utf8) {
  2055  		if (typeof(window.atob) === 'function') {
  2056  			return utf8 ? utf8_decode(window.atob(data)) : window.atob(data);
  2057  		}
  2058  
  2059  		// http://kevin.vanzonneveld.net
  2060  		// +   original by: Tyler Akins (http://rumkin.com)
  2061  		// +   improved by: Thunder.m
  2062  		// +      input by: Aman Gupta
  2063  		// +   improved by: Kevin van Zonneveld (http://kevin.vanzonneveld.net)
  2064  		// +   bugfixed by: Onno Marsman
  2065  		// +   bugfixed by: Pellentesque Malesuada
  2066  		// +   improved by: Kevin van Zonneveld (http://kevin.vanzonneveld.net)
  2067  		// +      input by: Brett Zamir (http://brett-zamir.me)
  2068  		// +   bugfixed by: Kevin van Zonneveld (http://kevin.vanzonneveld.net)
  2069  		// *     example 1: base64_decode('S2V2aW4gdmFuIFpvbm5ldmVsZA==');
  2070  		// *     returns 1: 'Kevin van Zonneveld'
  2071  		// mozilla has this native
  2072  		// - but breaks in 2.0.0.12!
  2073  		//if (typeof this.window.atob == 'function') {
  2074  		//    return atob(data);
  2075  		//}
  2076  		var b64 = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";
  2077  		var o1, o2, o3, h1, h2, h3, h4, bits, i = 0,
  2078  			ac = 0,
  2079  			dec = "",
  2080  			tmp_arr = [];
  2081  
  2082  		if (!data) {
  2083  			return data;
  2084  		}
  2085  
  2086  		data += '';
  2087  
  2088  		do { // unpack four hexets into three octets using index points in b64
  2089  			h1 = b64.indexOf(data.charAt(i++));
  2090  			h2 = b64.indexOf(data.charAt(i++));
  2091  			h3 = b64.indexOf(data.charAt(i++));
  2092  			h4 = b64.indexOf(data.charAt(i++));
  2093  
  2094  			bits = h1 << 18 | h2 << 12 | h3 << 6 | h4;
  2095  
  2096  			o1 = bits >> 16 & 0xff;
  2097  			o2 = bits >> 8 & 0xff;
  2098  			o3 = bits & 0xff;
  2099  
  2100  			if (h3 == 64) {
  2101  				tmp_arr[ac++] = String.fromCharCode(o1);
  2102  			} else if (h4 == 64) {
  2103  				tmp_arr[ac++] = String.fromCharCode(o1, o2);
  2104  			} else {
  2105  				tmp_arr[ac++] = String.fromCharCode(o1, o2, o3);
  2106  			}
  2107  		} while (i < data.length);
  2108  
  2109  		dec = tmp_arr.join('');
  2110  
  2111  		return utf8 ? utf8_decode(dec) : dec;
  2112  	};
  2113  	
  2114  	/**
  2115  	Base64 encode string (uses browser's default method if available),
  2116  	from: https://raw.github.com/kvz/phpjs/master/functions/url/base64_encode.js
  2117  
  2118  	@method btoa
  2119  	@static
  2120  	@param {String} data String to encode
  2121  	@return {String} Base64 encoded string
  2122  	*/
  2123  	var btoa = function(data, utf8) {
  2124  		if (utf8) {
  2125  			utf8_encode(data);
  2126  		}
  2127  
  2128  		if (typeof(window.btoa) === 'function') {
  2129  			return window.btoa(data);
  2130  		}
  2131  
  2132  		// http://kevin.vanzonneveld.net
  2133  		// +   original by: Tyler Akins (http://rumkin.com)
  2134  		// +   improved by: Bayron Guevara
  2135  		// +   improved by: Thunder.m
  2136  		// +   improved by: Kevin van Zonneveld (http://kevin.vanzonneveld.net)
  2137  		// +   bugfixed by: Pellentesque Malesuada
  2138  		// +   improved by: Kevin van Zonneveld (http://kevin.vanzonneveld.net)
  2139  		// +   improved by: Rafał Kukawski (http://kukawski.pl)
  2140  		// *     example 1: base64_encode('Kevin van Zonneveld');
  2141  		// *     returns 1: 'S2V2aW4gdmFuIFpvbm5ldmVsZA=='
  2142  		// mozilla has this native
  2143  		// - but breaks in 2.0.0.12!
  2144  		var b64 = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";
  2145  		var o1, o2, o3, h1, h2, h3, h4, bits, i = 0,
  2146  			ac = 0,
  2147  			enc = "",
  2148  			tmp_arr = [];
  2149  
  2150  		if (!data) {
  2151  			return data;
  2152  		}
  2153  
  2154  		do { // pack three octets into four hexets
  2155  			o1 = data.charCodeAt(i++);
  2156  			o2 = data.charCodeAt(i++);
  2157  			o3 = data.charCodeAt(i++);
  2158  
  2159  			bits = o1 << 16 | o2 << 8 | o3;
  2160  
  2161  			h1 = bits >> 18 & 0x3f;
  2162  			h2 = bits >> 12 & 0x3f;
  2163  			h3 = bits >> 6 & 0x3f;
  2164  			h4 = bits & 0x3f;
  2165  
  2166  			// use hexets to index into b64, and append result to encoded string
  2167  			tmp_arr[ac++] = b64.charAt(h1) + b64.charAt(h2) + b64.charAt(h3) + b64.charAt(h4);
  2168  		} while (i < data.length);
  2169  
  2170  		enc = tmp_arr.join('');
  2171  
  2172  		var r = data.length % 3;
  2173  
  2174  		return (r ? enc.slice(0, r - 3) : enc) + '==='.slice(r || 3);
  2175  	};
  2176  
  2177  
  2178  	return {
  2179  		utf8_encode: utf8_encode,
  2180  		utf8_decode: utf8_decode,
  2181  		atob: atob,
  2182  		btoa: btoa
  2183  	};
  2184  });
  2185  
  2186  // Included from: src/javascript/runtime/Runtime.js
  2187  
  2188  /**
  2189   * Runtime.js
  2190   *
  2191   * Copyright 2013, Moxiecode Systems AB
  2192   * Released under GPL License.
  2193   *
  2194   * License: http://www.plupload.com/license
  2195   * Contributing: http://www.plupload.com/contributing
  2196   */
  2197  
  2198  define('moxie/runtime/Runtime', [
  2199  	"moxie/core/utils/Basic",
  2200  	"moxie/core/utils/Dom",
  2201  	"moxie/core/EventTarget"
  2202  ], function(Basic, Dom, EventTarget) {
  2203  	var runtimeConstructors = {}, runtimes = {};
  2204  
  2205  	/**
  2206  	Common set of methods and properties for every runtime instance
  2207  
  2208  	@class Runtime
  2209  
  2210  	@param {Object} options
  2211  	@param {String} type Sanitized name of the runtime
  2212  	@param {Object} [caps] Set of capabilities that differentiate specified runtime
  2213  	@param {Object} [modeCaps] Set of capabilities that do require specific operational mode
  2214  	@param {String} [preferredMode='browser'] Preferred operational mode to choose if no required capabilities were requested
  2215  	*/
  2216  	function Runtime(options, type, caps, modeCaps, preferredMode) {
  2217  		/**
  2218  		Dispatched when runtime is initialized and ready.
  2219  		Results in RuntimeInit on a connected component.
  2220  
  2221  		@event Init
  2222  		*/
  2223  
  2224  		/**
  2225  		Dispatched when runtime fails to initialize.
  2226  		Results in RuntimeError on a connected component.
  2227  
  2228  		@event Error
  2229  		*/
  2230  
  2231  		var self = this
  2232  		, _shim
  2233  		, _uid = Basic.guid(type + '_')
  2234  		, defaultMode = preferredMode || 'browser'
  2235  		;
  2236  
  2237  		options = options || {};
  2238  
  2239  		// register runtime in private hash
  2240  		runtimes[_uid] = this;
  2241  
  2242  		/**
  2243  		Default set of capabilities, which can be redifined later by specific runtime
  2244  
  2245  		@private
  2246  		@property caps
  2247  		@type Object
  2248  		*/
  2249  		caps = Basic.extend({
  2250  			// Runtime can: 
  2251  			// provide access to raw binary data of the file
  2252  			access_binary: false,
  2253  			// provide access to raw binary data of the image (image extension is optional) 
  2254  			access_image_binary: false,
  2255  			// display binary data as thumbs for example
  2256  			display_media: false,
  2257  			// make cross-domain requests
  2258  			do_cors: false,
  2259  			// accept files dragged and dropped from the desktop
  2260  			drag_and_drop: false,
  2261  			// filter files in selection dialog by their extensions
  2262  			filter_by_extension: true,
  2263  			// resize image (and manipulate it raw data of any file in general)
  2264  			resize_image: false,
  2265  			// periodically report how many bytes of total in the file were uploaded (loaded)
  2266  			report_upload_progress: false,
  2267  			// provide access to the headers of http response 
  2268  			return_response_headers: false,
  2269  			// support response of specific type, which should be passed as an argument
  2270  			// e.g. runtime.can('return_response_type', 'blob')
  2271  			return_response_type: false,
  2272  			// return http status code of the response
  2273  			return_status_code: true,
  2274  			// send custom http header with the request
  2275  			send_custom_headers: false,
  2276  			// pick up the files from a dialog
  2277  			select_file: false,
  2278  			// select whole folder in file browse dialog
  2279  			select_folder: false,
  2280  			// select multiple files at once in file browse dialog
  2281  			select_multiple: true,
  2282  			// send raw binary data, that is generated after image resizing or manipulation of other kind
  2283  			send_binary_string: false,
  2284  			// send cookies with http request and therefore retain session
  2285  			send_browser_cookies: true,
  2286  			// send data formatted as multipart/form-data
  2287  			send_multipart: true,
  2288  			// slice the file or blob to smaller parts
  2289  			slice_blob: false,
  2290  			// upload file without preloading it to memory, stream it out directly from disk
  2291  			stream_upload: false,
  2292  			// programmatically trigger file browse dialog
  2293  			summon_file_dialog: false,
  2294  			// upload file of specific size, size should be passed as argument
  2295  			// e.g. runtime.can('upload_filesize', '500mb')
  2296  			upload_filesize: true,
  2297  			// initiate http request with specific http method, method should be passed as argument
  2298  			// e.g. runtime.can('use_http_method', 'put')
  2299  			use_http_method: true
  2300  		}, caps);
  2301  			
  2302  	
  2303  		// default to the mode that is compatible with preferred caps
  2304  		if (options.preferred_caps) {
  2305  			defaultMode = Runtime.getMode(modeCaps, options.preferred_caps, defaultMode);
  2306  		}
  2307  		
  2308  		// small extension factory here (is meant to be extended with actual extensions constructors)
  2309  		_shim = (function() {
  2310  			var objpool = {};
  2311  			return {
  2312  				exec: function(uid, comp, fn, args) {
  2313  					if (_shim[comp]) {
  2314  						if (!objpool[uid]) {
  2315  							objpool[uid] = {
  2316  								context: this,
  2317  								instance: new _shim[comp]()
  2318  							};
  2319  						}
  2320  						if (objpool[uid].instance[fn]) {
  2321  							return objpool[uid].instance[fn].apply(this, args);
  2322  						}
  2323  					}
  2324  				},
  2325  
  2326  				removeInstance: function(uid) {
  2327  					delete objpool[uid];
  2328  				},
  2329  
  2330  				removeAllInstances: function() {
  2331  					var self = this;
  2332  					Basic.each(objpool, function(obj, uid) {
  2333  						if (Basic.typeOf(obj.instance.destroy) === 'function') {
  2334  							obj.instance.destroy.call(obj.context);
  2335  						}
  2336  						self.removeInstance(uid);
  2337  					});
  2338  				}
  2339  			};
  2340  		}());
  2341  
  2342  
  2343  		// public methods
  2344  		Basic.extend(this, {
  2345  			/**
  2346  			Specifies whether runtime instance was initialized or not
  2347  
  2348  			@property initialized
  2349  			@type {Boolean}
  2350  			@default false
  2351  			*/
  2352  			initialized: false, // shims require this flag to stop initialization retries
  2353  
  2354  			/**
  2355  			Unique ID of the runtime
  2356  
  2357  			@property uid
  2358  			@type {String}
  2359  			*/
  2360  			uid: _uid,
  2361  
  2362  			/**
  2363  			Runtime type (e.g. flash, html5, etc)
  2364  
  2365  			@property type
  2366  			@type {String}
  2367  			*/
  2368  			type: type,
  2369  
  2370  			/**
  2371  			Runtime (not native one) may operate in browser or client mode.
  2372  
  2373  			@property mode
  2374  			@private
  2375  			@type {String|Boolean} current mode or false, if none possible
  2376  			*/
  2377  			mode: Runtime.getMode(modeCaps, (options.required_caps), defaultMode),
  2378  
  2379  			/**
  2380  			id of the DOM container for the runtime (if available)
  2381  
  2382  			@property shimid
  2383  			@type {String}
  2384  			*/
  2385  			shimid: _uid + '_container',
  2386  
  2387  			/**
  2388  			Number of connected clients. If equal to zero, runtime can be destroyed
  2389  
  2390  			@property clients
  2391  			@type {Number}
  2392  			*/
  2393  			clients: 0,
  2394  
  2395  			/**
  2396  			Runtime initialization options
  2397  
  2398  			@property options
  2399  			@type {Object}
  2400  			*/
  2401  			options: options,
  2402  
  2403  			/**
  2404  			Checks if the runtime has specific capability
  2405  
  2406  			@method can
  2407  			@param {String} cap Name of capability to check
  2408  			@param {Mixed} [value] If passed, capability should somehow correlate to the value
  2409  			@param {Object} [refCaps] Set of capabilities to check the specified cap against (defaults to internal set)
  2410  			@return {Boolean} true if runtime has such capability and false, if - not
  2411  			*/
  2412  			can: function(cap, value) {
  2413  				var refCaps = arguments[2] || caps;
  2414  
  2415  				// if cap var is a comma-separated list of caps, convert it to object (key/value)
  2416  				if (Basic.typeOf(cap) === 'string' && Basic.typeOf(value) === 'undefined') {
  2417  					cap = Runtime.parseCaps(cap);
  2418  				}
  2419  
  2420  				if (Basic.typeOf(cap) === 'object') {
  2421  					for (var key in cap) {
  2422  						if (!this.can(key, cap[key], refCaps)) {
  2423  							return false;
  2424  						}
  2425  					}
  2426  					return true;
  2427  				}
  2428  
  2429  				// check the individual cap
  2430  				if (Basic.typeOf(refCaps[cap]) === 'function') {
  2431  					return refCaps[cap].call(this, value);
  2432  				} else {
  2433  					return (value === refCaps[cap]);
  2434  				}
  2435  			},
  2436  
  2437  			/**
  2438  			Returns container for the runtime as DOM element
  2439  
  2440  			@method getShimContainer
  2441  			@return {DOMElement}
  2442  			*/
  2443  			getShimContainer: function() {
  2444  				var container, shimContainer = Dom.get(this.shimid);
  2445  
  2446  				// if no container for shim, create one
  2447  				if (!shimContainer) {
  2448  					container = this.options.container ? Dom.get(this.options.container) : document.body;
  2449  
  2450  					// create shim container and insert it at an absolute position into the outer container
  2451  					shimContainer = document.createElement('div');
  2452  					shimContainer.id = this.shimid;
  2453  					shimContainer.className = 'moxie-shim moxie-shim-' + this.type;
  2454  
  2455  					Basic.extend(shimContainer.style, {
  2456  						position: 'absolute',
  2457  						top: '0px',
  2458  						left: '0px',
  2459  						width: '1px',
  2460  						height: '1px',
  2461  						overflow: 'hidden'
  2462  					});
  2463  
  2464  					container.appendChild(shimContainer);
  2465  					container = null;
  2466  				}
  2467  
  2468  				return shimContainer;
  2469  			},
  2470  
  2471  			/**
  2472  			Returns runtime as DOM element (if appropriate)
  2473  
  2474  			@method getShim
  2475  			@return {DOMElement}
  2476  			*/
  2477  			getShim: function() {
  2478  				return _shim;
  2479  			},
  2480  
  2481  			/**
  2482  			Invokes a method within the runtime itself (might differ across the runtimes)
  2483  
  2484  			@method shimExec
  2485  			@param {Mixed} []
  2486  			@protected
  2487  			@return {Mixed} Depends on the action and component
  2488  			*/
  2489  			shimExec: function(component, action) {
  2490  				var args = [].slice.call(arguments, 2);
  2491  				return self.getShim().exec.call(this, this.uid, component, action, args);
  2492  			},
  2493  
  2494  			/**
  2495  			Operaional interface that is used by components to invoke specific actions on the runtime
  2496  			(is invoked in the scope of component)
  2497  
  2498  			@method exec
  2499  			@param {Mixed} []*
  2500  			@protected
  2501  			@return {Mixed} Depends on the action and component
  2502  			*/
  2503  			exec: function(component, action) { // this is called in the context of component, not runtime
  2504  				var args = [].slice.call(arguments, 2);
  2505  
  2506  				if (self[component] && self[component][action]) {
  2507  					return self[component][action].apply(this, args);
  2508  				}
  2509  				return self.shimExec.apply(this, arguments);
  2510  			},
  2511  
  2512  			/**
  2513  			Destroys the runtime (removes all events and deletes DOM structures)
  2514  
  2515  			@method destroy
  2516  			*/
  2517  			destroy: function() {
  2518  				if (!self) {
  2519  					return; // obviously already destroyed
  2520  				}
  2521  
  2522  				var shimContainer = Dom.get(this.shimid);
  2523  				if (shimContainer) {
  2524  					shimContainer.parentNode.removeChild(shimContainer);
  2525  				}
  2526  
  2527  				if (_shim) {
  2528  					_shim.removeAllInstances();
  2529  				}
  2530  
  2531  				this.unbindAll();
  2532  				delete runtimes[this.uid];
  2533  				this.uid = null; // mark this runtime as destroyed
  2534  				_uid = self = _shim = shimContainer = null;
  2535  			}
  2536  		});
  2537  
  2538  		// once we got the mode, test against all caps
  2539  		if (this.mode && options.required_caps && !this.can(options.required_caps)) {
  2540  			this.mode = false;
  2541  		}	
  2542  	}
  2543  
  2544  
  2545  	/**
  2546  	Default order to try different runtime types
  2547  
  2548  	@property order
  2549  	@type String
  2550  	@static
  2551  	*/
  2552  	Runtime.order = 'html5,flash,silverlight,html4';
  2553  
  2554  
  2555  	/**
  2556  	Retrieves runtime from private hash by it's uid
  2557  
  2558  	@method getRuntime
  2559  	@private
  2560  	@static
  2561  	@param {String} uid Unique identifier of the runtime
  2562  	@return {Runtime|Boolean} Returns runtime, if it exists and false, if - not
  2563  	*/
  2564  	Runtime.getRuntime = function(uid) {
  2565  		return runtimes[uid] ? runtimes[uid] : false;
  2566  	};
  2567  
  2568  
  2569  	/**
  2570  	Register constructor for the Runtime of new (or perhaps modified) type
  2571  
  2572  	@method addConstructor
  2573  	@static
  2574  	@param {String} type Runtime type (e.g. flash, html5, etc)
  2575  	@param {Function} construct Constructor for the Runtime type
  2576  	*/
  2577  	Runtime.addConstructor = function(type, constructor) {
  2578  		constructor.prototype = EventTarget.instance;
  2579  		runtimeConstructors[type] = constructor;
  2580  	};
  2581  
  2582  
  2583  	/**
  2584  	Get the constructor for the specified type.
  2585  
  2586  	method getConstructor
  2587  	@static
  2588  	@param {String} type Runtime type (e.g. flash, html5, etc)
  2589  	@return {Function} Constructor for the Runtime type
  2590  	*/
  2591  	Runtime.getConstructor = function(type) {
  2592  		return runtimeConstructors[type] || null;
  2593  	};
  2594  
  2595  
  2596  	/**
  2597  	Get info about the runtime (uid, type, capabilities)
  2598  
  2599  	@method getInfo
  2600  	@static
  2601  	@param {String} uid Unique identifier of the runtime
  2602  	@return {Mixed} Info object or null if runtime doesn't exist
  2603  	*/
  2604  	Runtime.getInfo = function(uid) {
  2605  		var runtime = Runtime.getRuntime(uid);
  2606  
  2607  		if (runtime) {
  2608  			return {
  2609  				uid: runtime.uid,
  2610  				type: runtime.type,
  2611  				mode: runtime.mode,
  2612  				can: function() {
  2613  					return runtime.can.apply(runtime, arguments);
  2614  				}
  2615  			};
  2616  		}
  2617  		return null;
  2618  	};
  2619  
  2620  
  2621  	/**
  2622  	Convert caps represented by a comma-separated string to the object representation.
  2623  
  2624  	@method parseCaps
  2625  	@static
  2626  	@param {String} capStr Comma-separated list of capabilities
  2627  	@return {Object}
  2628  	*/
  2629  	Runtime.parseCaps = function(capStr) {
  2630  		var capObj = {};
  2631  
  2632  		if (Basic.typeOf(capStr) !== 'string') {
  2633  			return capStr || {};
  2634  		}
  2635  
  2636  		Basic.each(capStr.split(','), function(key) {
  2637  			capObj[key] = true; // we assume it to be - true
  2638  		});
  2639  
  2640  		return capObj;
  2641  	};
  2642  
  2643  	/**
  2644  	Test the specified runtime for specific capabilities.
  2645  
  2646  	@method can
  2647  	@static
  2648  	@param {String} type Runtime type (e.g. flash, html5, etc)
  2649  	@param {String|Object} caps Set of capabilities to check
  2650  	@return {Boolean} Result of the test
  2651  	*/
  2652  	Runtime.can = function(type, caps) {
  2653  		var runtime
  2654  		, constructor = Runtime.getConstructor(type)
  2655  		, mode
  2656  		;
  2657  		if (constructor) {
  2658  			runtime = new constructor({
  2659  				required_caps: caps
  2660  			});
  2661  			mode = runtime.mode;
  2662  			runtime.destroy();
  2663  			return !!mode;
  2664  		}
  2665  		return false;
  2666  	};
  2667  
  2668  
  2669  	/**
  2670  	Figure out a runtime that supports specified capabilities.
  2671  
  2672  	@method thatCan
  2673  	@static
  2674  	@param {String|Object} caps Set of capabilities to check
  2675  	@param {String} [runtimeOrder] Comma-separated list of runtimes to check against
  2676  	@return {String} Usable runtime identifier or null
  2677  	*/
  2678  	Runtime.thatCan = function(caps, runtimeOrder) {
  2679  		var types = (runtimeOrder || Runtime.order).split(/\s*,\s*/);
  2680  		for (var i in types) {
  2681  			if (Runtime.can(types[i], caps)) {
  2682  				return types[i];
  2683  			}
  2684  		}
  2685  		return null;
  2686  	};
  2687  
  2688  
  2689  	/**
  2690  	Figure out an operational mode for the specified set of capabilities.
  2691  
  2692  	@method getMode
  2693  	@static
  2694  	@param {Object} modeCaps Set of capabilities that depend on particular runtime mode
  2695  	@param {Object} [requiredCaps] Supplied set of capabilities to find operational mode for
  2696  	@param {String|Boolean} [defaultMode='browser'] Default mode to use 
  2697  	@return {String|Boolean} Compatible operational mode
  2698  	*/
  2699  	Runtime.getMode = function(modeCaps, requiredCaps, defaultMode) {
  2700  		var mode = null;
  2701  
  2702  		if (Basic.typeOf(defaultMode) === 'undefined') { // only if not specified
  2703  			defaultMode = 'browser';
  2704  		}
  2705  
  2706  		if (requiredCaps && !Basic.isEmptyObj(modeCaps)) {
  2707  			// loop over required caps and check if they do require the same mode
  2708  			Basic.each(requiredCaps, function(value, cap) {
  2709  				if (modeCaps.hasOwnProperty(cap)) {
  2710  					var capMode = modeCaps[cap](value);
  2711  
  2712  					// make sure we always have an array
  2713  					if (typeof(capMode) === 'string') {
  2714  						capMode = [capMode];
  2715  					}
  2716  					
  2717  					if (!mode) {
  2718  						mode = capMode;
  2719  					} else if (!(mode = Basic.arrayIntersect(mode, capMode))) {
  2720  						// if cap requires conflicting mode - runtime cannot fulfill required caps
  2721  						return (mode = false);
  2722  					}
  2723  				}
  2724  			});
  2725  
  2726  			if (mode) {
  2727  				return Basic.inArray(defaultMode, mode) !== -1 ? defaultMode : mode[0];
  2728  			} else if (mode === false) {
  2729  				return false;
  2730  			}
  2731  		}
  2732  		return defaultMode; 
  2733  	};
  2734  
  2735  
  2736  	/**
  2737  	Capability check that always returns true
  2738  
  2739  	@private
  2740  	@static
  2741  	@return {True}
  2742  	*/
  2743  	Runtime.capTrue = function() {
  2744  		return true;
  2745  	};
  2746  
  2747  	/**
  2748  	Capability check that always returns false
  2749  
  2750  	@private
  2751  	@static
  2752  	@return {False}
  2753  	*/
  2754  	Runtime.capFalse = function() {
  2755  		return false;
  2756  	};
  2757  
  2758  	/**
  2759  	Evaluate the expression to boolean value and create a function that always returns it.
  2760  
  2761  	@private
  2762  	@static
  2763  	@param {Mixed} expr Expression to evaluate
  2764  	@return {Function} Function returning the result of evaluation
  2765  	*/
  2766  	Runtime.capTest = function(expr) {
  2767  		return function() {
  2768  			return !!expr;
  2769  		};
  2770  	};
  2771  
  2772  	return Runtime;
  2773  });
  2774  
  2775  // Included from: src/javascript/runtime/RuntimeClient.js
  2776  
  2777  /**
  2778   * RuntimeClient.js
  2779   *
  2780   * Copyright 2013, Moxiecode Systems AB
  2781   * Released under GPL License.
  2782   *
  2783   * License: http://www.plupload.com/license
  2784   * Contributing: http://www.plupload.com/contributing
  2785   */
  2786  
  2787  define('moxie/runtime/RuntimeClient', [
  2788  	'moxie/core/Exceptions',
  2789  	'moxie/core/utils/Basic',
  2790  	'moxie/runtime/Runtime'
  2791  ], function(x, Basic, Runtime) {
  2792  	/**
  2793  	Set of methods and properties, required by a component to acquire ability to connect to a runtime
  2794  
  2795  	@class RuntimeClient
  2796  	*/
  2797  	return function RuntimeClient() {
  2798  		var runtime;
  2799  
  2800  		Basic.extend(this, {
  2801  			/**
  2802  			Connects to the runtime specified by the options. Will either connect to existing runtime or create a new one.
  2803  			Increments number of clients connected to the specified runtime.
  2804  
  2805  			@method connectRuntime
  2806  			@param {Mixed} options Can be a runtme uid or a set of key-value pairs defining requirements and pre-requisites
  2807  			*/
  2808  			connectRuntime: function(options) {
  2809  				var comp = this, ruid;
  2810  
  2811  				function initialize(items) {
  2812  					var type, constructor;
  2813  
  2814  					// if we ran out of runtimes
  2815  					if (!items.length) {
  2816  						comp.trigger('RuntimeError', new x.RuntimeError(x.RuntimeError.NOT_INIT_ERR));
  2817  						runtime = null;
  2818  						return;
  2819  					}
  2820  
  2821  					type = items.shift();
  2822  					constructor = Runtime.getConstructor(type);
  2823  					if (!constructor) {
  2824  						initialize(items);
  2825  						return;
  2826  					}
  2827  
  2828  					// try initializing the runtime
  2829  					runtime = new constructor(options);
  2830  
  2831  					runtime.bind('Init', function() {
  2832  						// mark runtime as initialized
  2833  						runtime.initialized = true;
  2834  
  2835  						// jailbreak ...
  2836  						setTimeout(function() {
  2837  							runtime.clients++;
  2838  							// this will be triggered on component
  2839  							comp.trigger('RuntimeInit', runtime);
  2840  						}, 1);
  2841  					});
  2842  
  2843  					runtime.bind('Error', function() {
  2844  						runtime.destroy(); // runtime cannot destroy itself from inside at a right moment, thus we do it here
  2845  						initialize(items);
  2846  					});
  2847  
  2848  					/*runtime.bind('Exception', function() { });*/
  2849  
  2850  					// check if runtime managed to pick-up operational mode
  2851  					if (!runtime.mode) {
  2852  						runtime.trigger('Error');
  2853  						return;
  2854  					}
  2855  
  2856  					runtime.init();
  2857  				}
  2858  
  2859  				// check if a particular runtime was requested
  2860  				if (Basic.typeOf(options) === 'string') {
  2861  					ruid = options;
  2862  				} else if (Basic.typeOf(options.ruid) === 'string') {
  2863  					ruid = options.ruid;
  2864  				}
  2865  
  2866  				if (ruid) {
  2867  					runtime = Runtime.getRuntime(ruid);
  2868  					if (runtime) {
  2869  						runtime.clients++;
  2870  						return runtime;
  2871  					} else {
  2872  						// there should be a runtime and there's none - weird case
  2873  						throw new x.RuntimeError(x.RuntimeError.NOT_INIT_ERR);
  2874  					}
  2875  				}
  2876  
  2877  				// initialize a fresh one, that fits runtime list and required features best
  2878  				initialize((options.runtime_order || Runtime.order).split(/\s*,\s*/));
  2879  			},
  2880  
  2881  			/**
  2882  			Returns the runtime to which the client is currently connected.
  2883  
  2884  			@method getRuntime
  2885  			@return {Runtime} Runtime or null if client is not connected
  2886  			*/
  2887  			getRuntime: function() {
  2888  				if (runtime && runtime.uid) {
  2889  					return runtime;
  2890  				}
  2891  				runtime = null; // make sure we do not leave zombies rambling around
  2892  				return null;
  2893  			},
  2894  
  2895  			/**
  2896  			Disconnects from the runtime. Decrements number of clients connected to the specified runtime.
  2897  
  2898  			@method disconnectRuntime
  2899  			*/
  2900  			disconnectRuntime: function() {
  2901  				if (runtime && --runtime.clients <= 0) {
  2902  					runtime.destroy();
  2903  					runtime = null;
  2904  				}
  2905  			}
  2906  
  2907  		});
  2908  	};
  2909  
  2910  
  2911  });
  2912  
  2913  // Included from: src/javascript/file/Blob.js
  2914  
  2915  /**
  2916   * Blob.js
  2917   *
  2918   * Copyright 2013, Moxiecode Systems AB
  2919   * Released under GPL License.
  2920   *
  2921   * License: http://www.plupload.com/license
  2922   * Contributing: http://www.plupload.com/contributing
  2923   */
  2924  
  2925  define('moxie/file/Blob', [
  2926  	'moxie/core/utils/Basic',
  2927  	'moxie/core/utils/Encode',
  2928  	'moxie/runtime/RuntimeClient'
  2929  ], function(Basic, Encode, RuntimeClient) {
  2930  	
  2931  	var blobpool = {};
  2932  
  2933  	/**
  2934  	@class Blob
  2935  	@constructor
  2936  	@param {String} ruid Unique id of the runtime, to which this blob belongs to
  2937  	@param {Object} blob Object "Native" blob object, as it is represented in the runtime
  2938  	*/
  2939  	function Blob(ruid, blob) {
  2940  
  2941  		function _sliceDetached(start, end, type) {
  2942  			var blob, data = blobpool[this.uid];
  2943  
  2944  			if (Basic.typeOf(data) !== 'string' || !data.length) {
  2945  				return null; // or throw exception
  2946  			}
  2947  
  2948  			blob = new Blob(null, {
  2949  				type: type,
  2950  				size: end - start
  2951  			});
  2952  			blob.detach(data.substr(start, blob.size));
  2953  
  2954  			return blob;
  2955  		}
  2956  
  2957  		RuntimeClient.call(this);
  2958  
  2959  		if (ruid) {	
  2960  			this.connectRuntime(ruid);
  2961  		}
  2962  
  2963  		if (!blob) {
  2964  			blob = {};
  2965  		} else if (Basic.typeOf(blob) === 'string') { // dataUrl or binary string
  2966  			blob = { data: blob };
  2967  		}
  2968  
  2969  		Basic.extend(this, {
  2970  			
  2971  			/**
  2972  			Unique id of the component
  2973  
  2974  			@property uid
  2975  			@type {String}
  2976  			*/
  2977  			uid: blob.uid || Basic.guid('uid_'),
  2978  			
  2979  			/**
  2980  			Unique id of the connected runtime, if falsy, then runtime will have to be initialized 
  2981  			before this Blob can be used, modified or sent
  2982  
  2983  			@property ruid
  2984  			@type {String}
  2985  			*/
  2986  			ruid: ruid,
  2987  	
  2988  			/**
  2989  			Size of blob
  2990  
  2991  			@property size
  2992  			@type {Number}
  2993  			@default 0
  2994  			*/
  2995  			size: blob.size || 0,
  2996  			
  2997  			/**
  2998  			Mime type of blob
  2999  
  3000  			@property type
  3001  			@type {String}
  3002  			@default ''
  3003  			*/
  3004  			type: blob.type || '',
  3005  			
  3006  			/**
  3007  			@method slice
  3008  			@param {Number} [start=0]
  3009  			*/
  3010  			slice: function(start, end, type) {		
  3011  				if (this.isDetached()) {
  3012  					return _sliceDetached.apply(this, arguments);
  3013  				}
  3014  				return this.getRuntime().exec.call(this, 'Blob', 'slice', this.getSource(), start, end, type);
  3015  			},
  3016  
  3017  			/**
  3018  			Returns "native" blob object (as it is represented in connected runtime) or null if not found
  3019  
  3020  			@method getSource
  3021  			@return {Blob} Returns "native" blob object or null if not found
  3022  			*/
  3023  			getSource: function() {
  3024  				if (!blobpool[this.uid]) {
  3025  					return null;	
  3026  				}
  3027  				return blobpool[this.uid];
  3028  			},
  3029  
  3030  			/** 
  3031  			Detaches blob from any runtime that it depends on and initialize with standalone value
  3032  
  3033  			@method detach
  3034  			@protected
  3035  			@param {DOMString} [data=''] Standalone value
  3036  			*/
  3037  			detach: function(data) {
  3038  				if (this.ruid) {
  3039  					this.getRuntime().exec.call(this, 'Blob', 'destroy', blobpool[this.uid]);
  3040  					this.disconnectRuntime();
  3041  					this.ruid = null;
  3042  				}
  3043  
  3044  				data = data || '';
  3045  
  3046  				// if dataUrl, convert to binary string
  3047  				var matches = data.match(/^data:([^;]*);base64,/);
  3048  				if (matches) {
  3049  					this.type = matches[1];
  3050  					data = Encode.atob(data.substring(data.indexOf('base64,') + 7));
  3051  				}
  3052  
  3053  				this.size = data.length;
  3054  
  3055  				blobpool[this.uid] = data;
  3056  			},
  3057  
  3058  			/**
  3059  			Checks if blob is standalone (detached of any runtime)
  3060  			
  3061  			@method isDetached
  3062  			@protected
  3063  			@return {Boolean}
  3064  			*/
  3065  			isDetached: function() {
  3066  				return !this.ruid && Basic.typeOf(blobpool[this.uid]) === 'string';
  3067  			},
  3068  			
  3069  			/** 
  3070  			Destroy Blob and free any resources it was using
  3071  
  3072  			@method destroy
  3073  			*/
  3074  			destroy: function() {
  3075  				this.detach();
  3076  				delete blobpool[this.uid];
  3077  			}
  3078  		});
  3079  
  3080  		
  3081  		if (blob.data) {
  3082  			this.detach(blob.data); // auto-detach if payload has been passed
  3083  		} else {
  3084  			blobpool[this.uid] = blob;	
  3085  		}
  3086  	}
  3087  	
  3088  	return Blob;
  3089  });
  3090  
  3091  // Included from: src/javascript/file/File.js
  3092  
  3093  /**
  3094   * File.js
  3095   *
  3096   * Copyright 2013, Moxiecode Systems AB
  3097   * Released under GPL License.
  3098   *
  3099   * License: http://www.plupload.com/license
  3100   * Contributing: http://www.plupload.com/contributing
  3101   */
  3102  
  3103  define('moxie/file/File', [
  3104  	'moxie/core/utils/Basic',
  3105  	'moxie/core/utils/Mime',
  3106  	'moxie/file/Blob'
  3107  ], function(Basic, Mime, Blob) {
  3108  	/**
  3109  	@class File
  3110  	@extends Blob
  3111  	@constructor
  3112  	@param {String} ruid Unique id of the runtime, to which this blob belongs to
  3113  	@param {Object} file Object "Native" file object, as it is represented in the runtime
  3114  	*/
  3115  	function File(ruid, file) {
  3116  		var name, type;
  3117  
  3118  		if (!file) { // avoid extra errors in case we overlooked something
  3119  			file = {};
  3120  		}
  3121  
  3122  		// figure out the type
  3123  		if (file.type && file.type !== '') {
  3124  			type = file.type;
  3125  		} else {
  3126  			type = Mime.getFileMime(file.name);
  3127  		}
  3128  
  3129  		// sanitize file name or generate new one
  3130  		if (file.name) {
  3131  			name = file.name.replace(/\\/g, '/');
  3132  			name = name.substr(name.lastIndexOf('/') + 1);
  3133  		} else {
  3134  			var prefix = type.split('/')[0];
  3135  			name = Basic.guid((prefix !== '' ? prefix : 'file') + '_');
  3136  			
  3137  			if (Mime.extensions[type]) {
  3138  				name += '.' + Mime.extensions[type][0]; // append proper extension if possible
  3139  			}
  3140  		}
  3141  
  3142  		Blob.apply(this, arguments);
  3143  		
  3144  		Basic.extend(this, {
  3145  			/**
  3146  			File mime type
  3147  
  3148  			@property type
  3149  			@type {String}
  3150  			@default ''
  3151  			*/
  3152  			type: type || '',
  3153  
  3154  			/**
  3155  			File name
  3156  
  3157  			@property name
  3158  			@type {String}
  3159  			@default UID
  3160  			*/
  3161  			name: name || Basic.guid('file_'),
  3162  			
  3163  			/**
  3164  			Date of last modification
  3165  
  3166  			@property lastModifiedDate
  3167  			@type {String}
  3168  			@default now
  3169  			*/
  3170  			lastModifiedDate: file.lastModifiedDate || (new Date()).toLocaleString() // Thu Aug 23 2012 19:40:00 GMT+0400 (GET)
  3171  		});
  3172  	}
  3173  
  3174  	File.prototype = Blob.prototype;
  3175  
  3176  	return File;
  3177  });
  3178  
  3179  // Included from: src/javascript/file/FileInput.js
  3180  
  3181  /**
  3182   * FileInput.js
  3183   *
  3184   * Copyright 2013, Moxiecode Systems AB
  3185   * Released under GPL License.
  3186   *
  3187   * License: http://www.plupload.com/license
  3188   * Contributing: http://www.plupload.com/contributing
  3189   */
  3190  
  3191  define('moxie/file/FileInput', [
  3192  	'moxie/core/utils/Basic',
  3193  	'moxie/core/utils/Mime',
  3194  	'moxie/core/utils/Dom',
  3195  	'moxie/core/Exceptions',
  3196  	'moxie/core/EventTarget',
  3197  	'moxie/core/I18n',
  3198  	'moxie/file/File',
  3199  	'moxie/runtime/Runtime',
  3200  	'moxie/runtime/RuntimeClient'
  3201  ], function(Basic, Mime, Dom, x, EventTarget, I18n, File, Runtime, RuntimeClient) {
  3202  	/**
  3203  	Provides a convenient way to create cross-browser file-picker. Generates file selection dialog on click,
  3204  	converts selected files to _File_ objects, to be used in conjunction with _Image_, preloaded in memory
  3205  	with _FileReader_ or uploaded to a server through _XMLHttpRequest_.
  3206  
  3207  	@class FileInput
  3208  	@constructor
  3209  	@extends EventTarget
  3210  	@uses RuntimeClient
  3211  	@param {Object|String|DOMElement} options If options is string or node, argument is considered as _browse\_button_.
  3212  		@param {String|DOMElement} options.browse_button DOM Element to turn into file picker.
  3213  		@param {Array} [options.accept] Array of mime types to accept. By default accepts all.
  3214  		@param {String} [options.file='file'] Name of the file field (not the filename).
  3215  		@param {Boolean} [options.multiple=false] Enable selection of multiple files.
  3216  		@param {Boolean} [options.directory=false] Turn file input into the folder input (cannot be both at the same time).
  3217  		@param {String|DOMElement} [options.container] DOM Element to use as a container for file-picker. Defaults to parentNode 
  3218  		for _browse\_button_.
  3219  		@param {Object|String} [options.required_caps] Set of required capabilities, that chosen runtime must support.
  3220  
  3221  	@example
  3222  		<div id="container">
  3223  			<a id="file-picker" href="javascript:;">Browse...</a>
  3224  		</div>
  3225  
  3226  		<script>
  3227  			var fileInput = new mOxie.FileInput({
  3228  				browse_button: 'file-picker', // or document.getElementById('file-picker')
  3229  				container: 'container',
  3230  				accept: [
  3231  					{title: "Image files", extensions: "jpg,gif,png"} // accept only images
  3232  				],
  3233  				multiple: true // allow multiple file selection
  3234  			});
  3235  
  3236  			fileInput.onchange = function(e) {
  3237  				// do something to files array
  3238  				console.info(e.target.files); // or this.files or fileInput.files
  3239  			};
  3240  
  3241  			fileInput.init(); // initialize
  3242  		</script>
  3243  	*/
  3244  	var dispatches = [
  3245  		/**
  3246  		Dispatched when runtime is connected and file-picker is ready to be used.
  3247  
  3248  		@event ready
  3249  		@param {Object} event
  3250  		*/
  3251  		'ready',
  3252  
  3253  		/**
  3254  		Dispatched right after [ready](#event_ready) event, and whenever [refresh()](#method_refresh) is invoked. 
  3255  		Check [corresponding documentation entry](#method_refresh) for more info.
  3256  
  3257  		@event refresh
  3258  		@param {Object} event
  3259  		*/
  3260  
  3261  		/**
  3262  		Dispatched when selection of files in the dialog is complete.
  3263  
  3264  		@event change
  3265  		@param {Object} event
  3266  		*/
  3267  		'change',
  3268  
  3269  		'cancel', // TODO: might be useful
  3270  
  3271  		/**
  3272  		Dispatched when mouse cursor enters file-picker area. Can be used to style element
  3273  		accordingly.
  3274  
  3275  		@event mouseenter
  3276  		@param {Object} event
  3277  		*/
  3278  		'mouseenter',
  3279  
  3280  		/**
  3281  		Dispatched when mouse cursor leaves file-picker area. Can be used to style element
  3282  		accordingly.
  3283  
  3284  		@event mouseleave
  3285  		@param {Object} event
  3286  		*/
  3287  		'mouseleave',
  3288  
  3289  		/**
  3290  		Dispatched when functional mouse button is pressed on top of file-picker area.
  3291  
  3292  		@event mousedown
  3293  		@param {Object} event
  3294  		*/
  3295  		'mousedown',
  3296  
  3297  		/**
  3298  		Dispatched when functional mouse button is released on top of file-picker area.
  3299  
  3300  		@event mouseup
  3301  		@param {Object} event
  3302  		*/
  3303  		'mouseup'
  3304  	];
  3305  
  3306  	function FileInput(options) {
  3307  		var self = this,
  3308  			container, browseButton, defaults;
  3309  
  3310  		// if flat argument passed it should be browse_button id
  3311  		if (Basic.inArray(Basic.typeOf(options), ['string', 'node']) !== -1) {
  3312  			options = { browse_button : options };
  3313  		}
  3314  
  3315  		// this will help us to find proper default container
  3316  		browseButton = Dom.get(options.browse_button);
  3317  		if (!browseButton) {
  3318  			// browse button is required
  3319  			throw new x.DOMException(x.DOMException.NOT_FOUND_ERR);
  3320  		}
  3321  
  3322  		// figure out the options
  3323  		defaults = {
  3324  			accept: [{
  3325  				title: I18n.translate('All Files'),
  3326  				extensions: '*'
  3327  			}],
  3328  			name: 'file',
  3329  			multiple: false,
  3330  			required_caps: false,
  3331  			container: browseButton.parentNode || document.body
  3332  		};
  3333  		
  3334  		options = Basic.extend({}, defaults, options);
  3335  
  3336  		// convert to object representation
  3337  		if (typeof(options.required_caps) === 'string') {
  3338  			options.required_caps = Runtime.parseCaps(options.required_caps);
  3339  		}
  3340  					
  3341  		// normalize accept option (could be list of mime types or array of title/extensions pairs)
  3342  		if (typeof(options.accept) === 'string') {
  3343  			options.accept = Mime.mimes2extList(options.accept);
  3344  		}
  3345  
  3346  		container = Dom.get(options.container);
  3347  		// make sure we have container
  3348  		if (!container) {
  3349  			container = document.body;
  3350  		}
  3351  
  3352  		// make container relative, if it's not
  3353  		if (Dom.getStyle(container, 'position') === 'static') {
  3354  			container.style.position = 'relative';
  3355  		}
  3356  
  3357  		container = browseButton = null; // IE
  3358  						
  3359  		RuntimeClient.call(self);
  3360  		
  3361  		Basic.extend(self, {
  3362  			/**
  3363  			Unique id of the component
  3364  
  3365  			@property uid
  3366  			@protected
  3367  			@readOnly
  3368  			@type {String}
  3369  			@default UID
  3370  			*/
  3371  			uid: Basic.guid('uid_'),
  3372  			
  3373  			/**
  3374  			Unique id of the connected runtime, if any.
  3375  
  3376  			@property ruid
  3377  			@protected
  3378  			@type {String}
  3379  			*/
  3380  			ruid: null,
  3381  
  3382  			/**
  3383  			Unique id of the runtime container. Useful to get hold of it for various manipulations.
  3384  
  3385  			@property shimid
  3386  			@protected
  3387  			@type {String}
  3388  			*/
  3389  			shimid: null,
  3390  			
  3391  			/**
  3392  			Array of selected mOxie.File objects
  3393  
  3394  			@property files
  3395  			@type {Array}
  3396  			@default null
  3397  			*/
  3398  			files: null,
  3399  
  3400  			/**
  3401  			Initializes the file-picker, connects it to runtime and dispatches event ready when done.
  3402  
  3403  			@method init
  3404  			*/
  3405  			init: function() {
  3406  				self.convertEventPropsToHandlers(dispatches);
  3407  
  3408  				self.bind('RuntimeInit', function(e, runtime) {
  3409  					self.ruid = runtime.uid;
  3410  					self.shimid = runtime.shimid;
  3411  
  3412  					self.bind("Ready", function() {
  3413  						self.trigger("Refresh");
  3414  					}, 999);
  3415  
  3416  					self.bind("Change", function() {
  3417  						var files = runtime.exec.call(self, 'FileInput', 'getFiles');
  3418  
  3419  						self.files = [];
  3420  
  3421  						Basic.each(files, function(file) {
  3422  							// ignore empty files (IE10 for example hangs if you try to send them via XHR)
  3423  							if (file.size === 0) {
  3424  								return true; 
  3425  							}
  3426  							self.files.push(new File(self.ruid, file));
  3427  						});
  3428  					}, 999);
  3429  
  3430  					// re-position and resize shim container
  3431  					self.bind('Refresh', function() {
  3432  						var pos, size, browseButton, shimContainer;
  3433  						
  3434  						browseButton = Dom.get(options.browse_button);
  3435  						shimContainer = Dom.get(runtime.shimid); // do not use runtime.getShimContainer(), since it will create container if it doesn't exist
  3436  
  3437  						if (browseButton) {
  3438  							pos = Dom.getPos(browseButton, Dom.get(options.container));
  3439  							size = Dom.getSize(browseButton);
  3440  
  3441  							if (shimContainer) {
  3442  								Basic.extend(shimContainer.style, {
  3443  									top     : pos.y + 'px',
  3444  									left    : pos.x + 'px',
  3445  									width   : size.w + 'px',
  3446  									height  : size.h + 'px'
  3447  								});
  3448  							}
  3449  						}
  3450  						shimContainer = browseButton = null;
  3451  					});
  3452  					
  3453  					runtime.exec.call(self, 'FileInput', 'init', options);
  3454  				});
  3455  
  3456  				// runtime needs: options.required_features, options.runtime_order and options.container
  3457  				self.connectRuntime(Basic.extend({}, options, {
  3458  					required_caps: {
  3459  						select_file: true
  3460  					}
  3461  				}));
  3462  			},
  3463  
  3464  			/**
  3465  			Disables file-picker element, so that it doesn't react to mouse clicks.
  3466  
  3467  			@method disable
  3468  			@param {Boolean} [state=true] Disable component if - true, enable if - false
  3469  			*/
  3470  			disable: function(state) {
  3471  				var runtime = this.getRuntime();
  3472  				if (runtime) {
  3473  					runtime.exec.call(this, 'FileInput', 'disable', Basic.typeOf(state) === 'undefined' ? true : state);
  3474  				}
  3475  			},
  3476  
  3477  
  3478  			/**
  3479  			Reposition and resize dialog trigger to match the position and size of browse_button element.
  3480  
  3481  			@method refresh
  3482  			*/
  3483  			refresh: function() {
  3484  				self.trigger("Refresh");
  3485  			},
  3486  
  3487  
  3488  			/**
  3489  			Destroy component.
  3490  
  3491  			@method destroy
  3492  			*/
  3493  			destroy: function() {
  3494  				var runtime = this.getRuntime();
  3495  				if (runtime) {
  3496  					runtime.exec.call(this, 'FileInput', 'destroy');
  3497  					this.disconnectRuntime();
  3498  				}
  3499  
  3500  				if (Basic.typeOf(this.files) === 'array') {
  3501  					// no sense in leaving associated files behind
  3502  					Basic.each(this.files, function(file) {
  3503  						file.destroy();
  3504  					});
  3505  				} 
  3506  				this.files = null;
  3507  			}
  3508  		});
  3509  	}
  3510  
  3511  	FileInput.prototype = EventTarget.instance;
  3512  
  3513  	return FileInput;
  3514  });
  3515  
  3516  // Included from: src/javascript/file/FileDrop.js
  3517  
  3518  /**
  3519   * FileDrop.js
  3520   *
  3521   * Copyright 2013, Moxiecode Systems AB
  3522   * Released under GPL License.
  3523   *
  3524   * License: http://www.plupload.com/license
  3525   * Contributing: http://www.plupload.com/contributing
  3526   */
  3527  
  3528  define('moxie/file/FileDrop', [
  3529  	'moxie/core/I18n',
  3530  	'moxie/core/utils/Dom',
  3531  	'moxie/core/Exceptions',
  3532  	'moxie/core/utils/Basic',
  3533  	'moxie/file/File',
  3534  	'moxie/runtime/RuntimeClient',
  3535  	'moxie/core/EventTarget',
  3536  	'moxie/core/utils/Mime'
  3537  ], function(I18n, Dom, x, Basic, File, RuntimeClient, EventTarget, Mime) {
  3538  	/**
  3539  	Turn arbitrary DOM element to a drop zone accepting files. Converts selected files to _File_ objects, to be used 
  3540  	in conjunction with _Image_, preloaded in memory with _FileReader_ or uploaded to a server through 
  3541  	_XMLHttpRequest_.
  3542  
  3543  	@example
  3544  		<div id="drop_zone">
  3545  			Drop files here
  3546  		</div>
  3547  		<br />
  3548  		<div id="filelist"></div>
  3549  
  3550  		<script type="text/javascript">
  3551  			var fileDrop = new mOxie.FileDrop('drop_zone'), fileList = mOxie.get('filelist');
  3552  
  3553  			fileDrop.ondrop = function() {
  3554  				mOxie.each(this.files, function(file) {
  3555  					fileList.innerHTML += '<div>' + file.name + '</div>';
  3556  				});
  3557  			};
  3558  
  3559  			fileDrop.init();
  3560  		</script>
  3561  
  3562  	@class FileDrop
  3563  	@constructor
  3564  	@extends EventTarget
  3565  	@uses RuntimeClient
  3566  	@param {Object|String} options If options has typeof string, argument is considered as options.drop_zone
  3567  		@param {String|DOMElement} options.drop_zone DOM Element to turn into a drop zone
  3568  		@param {Array} [options.accept] Array of mime types to accept. By default accepts all
  3569  		@param {Object|String} [options.required_caps] Set of required capabilities, that chosen runtime must support
  3570  	*/
  3571  	var dispatches = [
  3572  		/**
  3573  		Dispatched when runtime is connected and drop zone is ready to accept files.
  3574  
  3575  		@event ready
  3576  		@param {Object} event
  3577  		*/
  3578  		'ready', 
  3579  
  3580  		/**
  3581  		Dispatched when dragging cursor enters the drop zone.
  3582  
  3583  		@event dragenter
  3584  		@param {Object} event
  3585  		*/
  3586  		'dragenter',
  3587  
  3588  		/**
  3589  		Dispatched when dragging cursor leaves the drop zone.
  3590  
  3591  		@event dragleave
  3592  		@param {Object} event
  3593  		*/
  3594  		'dragleave', 
  3595  
  3596  		/**
  3597  		Dispatched when file is dropped onto the drop zone.
  3598  
  3599  		@event drop
  3600  		@param {Object} event
  3601  		*/
  3602  		'drop', 
  3603  
  3604  		/**
  3605  		Dispatched if error occurs.
  3606  
  3607  		@event error
  3608  		@param {Object} event
  3609  		*/
  3610  		'error'
  3611  	];
  3612  
  3613  	function FileDrop(options) {
  3614  		var self = this, defaults;
  3615  
  3616  		// if flat argument passed it should be drop_zone id
  3617  		if (typeof(options) === 'string') {
  3618  			options = { drop_zone : options };
  3619  		}
  3620  
  3621  		// figure out the options
  3622  		defaults = {
  3623  			accept: [{
  3624  				title: I18n.translate('All Files'),
  3625  				extensions: '*'
  3626  			}],
  3627  			required_caps: {
  3628  				drag_and_drop: true
  3629  			}
  3630  		};
  3631  		
  3632  		options = typeof(options) === 'object' ? Basic.extend({}, defaults, options) : defaults;
  3633  
  3634  		// this will help us to find proper default container
  3635  		options.container = Dom.get(options.drop_zone) || document.body;
  3636  
  3637  		// make container relative, if it is not
  3638  		if (Dom.getStyle(options.container, 'position') === 'static') {
  3639  			options.container.style.position = 'relative';
  3640  		}
  3641  					
  3642  		// normalize accept option (could be list of mime types or array of title/extensions pairs)
  3643  		if (typeof(options.accept) === 'string') {
  3644  			options.accept = Mime.mimes2extList(options.accept);
  3645  		}
  3646  
  3647  		RuntimeClient.call(self);
  3648  
  3649  		Basic.extend(self, {
  3650  			uid: Basic.guid('uid_'),
  3651  
  3652  			ruid: null,
  3653  
  3654  			files: null,
  3655  
  3656  			init: function() {
  3657  	
  3658  				self.convertEventPropsToHandlers(dispatches);
  3659  		
  3660  				self.bind('RuntimeInit', function(e, runtime) {
  3661  					self.ruid = runtime.uid;
  3662  
  3663  					self.bind("Drop", function() {
  3664  						var files = runtime.exec.call(self, 'FileDrop', 'getFiles');
  3665  
  3666  						self.files = [];
  3667  
  3668  						Basic.each(files, function(file) {
  3669  							self.files.push(new File(self.ruid, file));
  3670  						});
  3671  					}, 999);
  3672  
  3673  					runtime.exec.call(self, 'FileDrop', 'init', options);
  3674  
  3675  					self.dispatchEvent('ready');
  3676  				});
  3677  							
  3678  				// runtime needs: options.required_features, options.runtime_order and options.container
  3679  				self.connectRuntime(options); // throws RuntimeError
  3680  			},
  3681  
  3682  			destroy: function() {
  3683  				var runtime = this.getRuntime();
  3684  				if (runtime) {
  3685  					runtime.exec.call(this, 'FileDrop', 'destroy');
  3686  					this.disconnectRuntime();
  3687  				}
  3688  				this.files = null;
  3689  			}
  3690  		});
  3691  	}
  3692  
  3693  	FileDrop.prototype = EventTarget.instance;
  3694  
  3695  	return FileDrop;
  3696  });
  3697  
  3698  // Included from: src/javascript/runtime/RuntimeTarget.js
  3699  
  3700  /**
  3701   * RuntimeTarget.js
  3702   *
  3703   * Copyright 2013, Moxiecode Systems AB
  3704   * Released under GPL License.
  3705   *
  3706   * License: http://www.plupload.com/license
  3707   * Contributing: http://www.plupload.com/contributing
  3708   */
  3709  
  3710  define('moxie/runtime/RuntimeTarget', [
  3711  	'moxie/core/utils/Basic',
  3712  	'moxie/runtime/RuntimeClient',
  3713  	"moxie/core/EventTarget"
  3714  ], function(Basic, RuntimeClient, EventTarget) {
  3715  	/**
  3716  	Instance of this class can be used as a target for the events dispatched by shims,
  3717  	when allowing them onto components is for either reason inappropriate
  3718  
  3719  	@class RuntimeTarget
  3720  	@constructor
  3721  	@protected
  3722  	@extends EventTarget
  3723  	*/
  3724  	function RuntimeTarget() {
  3725  		this.uid = Basic.guid('uid_');
  3726  		
  3727  		RuntimeClient.call(this);
  3728  
  3729  		this.destroy = function() {
  3730  			this.disconnectRuntime();
  3731  			this.unbindAll();
  3732  		};
  3733  	}
  3734  
  3735  	RuntimeTarget.prototype = EventTarget.instance;
  3736  
  3737  	return RuntimeTarget;
  3738  });
  3739  
  3740  // Included from: src/javascript/file/FileReader.js
  3741  
  3742  /**
  3743   * FileReader.js
  3744   *
  3745   * Copyright 2013, Moxiecode Systems AB
  3746   * Released under GPL License.
  3747   *
  3748   * License: http://www.plupload.com/license
  3749   * Contributing: http://www.plupload.com/contributing
  3750   */
  3751  
  3752  define('moxie/file/FileReader', [
  3753  	'moxie/core/utils/Basic',
  3754  	'moxie/core/utils/Encode',
  3755  	'moxie/core/Exceptions',
  3756  	'moxie/core/EventTarget',
  3757  	'moxie/file/Blob',
  3758  	'moxie/file/File',
  3759  	'moxie/runtime/RuntimeTarget'
  3760  ], function(Basic, Encode, x, EventTarget, Blob, File, RuntimeTarget) {
  3761  	/**
  3762  	Utility for preloading o.Blob/o.File objects in memory. By design closely follows [W3C FileReader](http://www.w3.org/TR/FileAPI/#dfn-filereader)
  3763  	interface. Where possible uses native FileReader, where - not falls back to shims.
  3764  
  3765  	@class FileReader
  3766  	@constructor FileReader
  3767  	@extends EventTarget
  3768  	@uses RuntimeClient
  3769  	*/
  3770  	var dispatches = [
  3771  
  3772  		/** 
  3773  		Dispatched when the read starts.
  3774  
  3775  		@event loadstart
  3776  		@param {Object} event
  3777  		*/
  3778  		'loadstart', 
  3779  
  3780  		/** 
  3781  		Dispatched while reading (and decoding) blob, and reporting partial Blob data (progess.loaded/progress.total).
  3782  
  3783  		@event progress
  3784  		@param {Object} event
  3785  		*/
  3786  		'progress', 
  3787  
  3788  		/** 
  3789  		Dispatched when the read has successfully completed.
  3790  
  3791  		@event load
  3792  		@param {Object} event
  3793  		*/
  3794  		'load', 
  3795  
  3796  		/** 
  3797  		Dispatched when the read has been aborted. For instance, by invoking the abort() method.
  3798  
  3799  		@event abort
  3800  		@param {Object} event
  3801  		*/
  3802  		'abort', 
  3803  
  3804  		/** 
  3805  		Dispatched when the read has failed.
  3806  
  3807  		@event error
  3808  		@param {Object} event
  3809  		*/
  3810  		'error', 
  3811  
  3812  		/** 
  3813  		Dispatched when the request has completed (either in success or failure).
  3814  
  3815  		@event loadend
  3816  		@param {Object} event
  3817  		*/
  3818  		'loadend'
  3819  	];
  3820  	
  3821  	function FileReader() {
  3822  		var self = this, _fr;
  3823  				
  3824  		Basic.extend(this, {
  3825  			/**
  3826  			UID of the component instance.
  3827  
  3828  			@property uid
  3829  			@type {String}
  3830  			*/
  3831  			uid: Basic.guid('uid_'),
  3832  
  3833  			/**
  3834  			Contains current state of FileReader object. Can take values of FileReader.EMPTY, FileReader.LOADING
  3835  			and FileReader.DONE.
  3836  
  3837  			@property readyState
  3838  			@type {Number}
  3839  			@default FileReader.EMPTY
  3840  			*/
  3841  			readyState: FileReader.EMPTY,
  3842  			
  3843  			/**
  3844  			Result of the successful read operation.
  3845  
  3846  			@property result
  3847  			@type {String}
  3848  			*/
  3849  			result: null,
  3850  			
  3851  			/**
  3852  			Stores the error of failed asynchronous read operation.
  3853  
  3854  			@property error
  3855  			@type {DOMError}
  3856  			*/
  3857  			error: null,
  3858  			
  3859  			/**
  3860  			Initiates reading of File/Blob object contents to binary string.
  3861  
  3862  			@method readAsBinaryString
  3863  			@param {Blob|File} blob Object to preload
  3864  			*/
  3865  			readAsBinaryString: function(blob) {
  3866  				_read.call(this, 'readAsBinaryString', blob);
  3867  			},
  3868  			
  3869  			/**
  3870  			Initiates reading of File/Blob object contents to dataURL string.
  3871  
  3872  			@method readAsDataURL
  3873  			@param {Blob|File} blob Object to preload
  3874  			*/
  3875  			readAsDataURL: function(blob) {
  3876  				_read.call(this, 'readAsDataURL', blob);
  3877  			},
  3878  			
  3879  			/**
  3880  			Initiates reading of File/Blob object contents to string.
  3881  
  3882  			@method readAsText
  3883  			@param {Blob|File} blob Object to preload
  3884  			*/
  3885  			readAsText: function(blob) {
  3886  				_read.call(this, 'readAsText', blob);
  3887  			},
  3888  			
  3889  			/**
  3890  			Aborts preloading process.
  3891  
  3892  			@method abort
  3893  			*/
  3894  			abort: function() {
  3895  				this.result = null;
  3896  				
  3897  				if (Basic.inArray(this.readyState, [FileReader.EMPTY, FileReader.DONE]) !== -1) {
  3898  					return;
  3899  				} else if (this.readyState === FileReader.LOADING) {
  3900  					this.readyState = FileReader.DONE;
  3901  				}
  3902  
  3903  				if (_fr) {
  3904  					_fr.getRuntime().exec.call(this, 'FileReader', 'abort');
  3905  				}
  3906  				
  3907  				this.trigger('abort');
  3908  				this.trigger('loadend');
  3909  			},
  3910  
  3911  			/**
  3912  			Destroy component and release resources.
  3913  
  3914  			@method destroy
  3915  			*/
  3916  			destroy: function() {
  3917  				this.abort();
  3918  
  3919  				if (_fr) {
  3920  					_fr.getRuntime().exec.call(this, 'FileReader', 'destroy');
  3921  					_fr.disconnectRuntime();
  3922  				}
  3923  
  3924  				self = _fr = null;
  3925  			}
  3926  		});
  3927  		
  3928  		
  3929  		function _read(op, blob) {
  3930  			_fr = new RuntimeTarget();
  3931  
  3932  			function error(err) {
  3933  				self.readyState = FileReader.DONE;
  3934  				self.error = err;
  3935  				self.trigger('error');
  3936  				loadEnd();
  3937  			}
  3938  
  3939  			function loadEnd() {
  3940  				_fr.destroy();
  3941  				_fr = null;
  3942  				self.trigger('loadend');
  3943  			}
  3944  
  3945  			function exec(runtime) {
  3946  				_fr.bind('Error', function(e, err) {
  3947  					error(err);
  3948  				});
  3949  
  3950  				_fr.bind('Progress', function(e) {
  3951  					self.result = runtime.exec.call(_fr, 'FileReader', 'getResult');
  3952  					self.trigger(e);
  3953  				});
  3954  				
  3955  				_fr.bind('Load', function(e) {
  3956  					self.readyState = FileReader.DONE;
  3957  					self.result = runtime.exec.call(_fr, 'FileReader', 'getResult');
  3958  					self.trigger(e);
  3959  					loadEnd();
  3960  				});
  3961  
  3962  				runtime.exec.call(_fr, 'FileReader', 'read', op, blob);
  3963  			}
  3964  
  3965  			this.convertEventPropsToHandlers(dispatches);
  3966  
  3967  			if (this.readyState === FileReader.LOADING) {
  3968  				return error(new x.DOMException(x.DOMException.INVALID_STATE_ERR));
  3969  			}
  3970  
  3971  			this.readyState = FileReader.LOADING;
  3972  			this.trigger('loadstart');
  3973  
  3974  			// if source is o.Blob/o.File
  3975  			if (blob instanceof Blob) {
  3976  				if (blob.isDetached()) {
  3977  					var src = blob.getSource();
  3978  					switch (op) {
  3979  						case 'readAsText':
  3980  						case 'readAsBinaryString':
  3981  							this.result = src;
  3982  							break;
  3983  						case 'readAsDataURL':
  3984  							this.result = 'data:' + blob.type + ';base64,' + Encode.btoa(src);
  3985  							break;
  3986  					}
  3987  					this.readyState = FileReader.DONE;
  3988  					this.trigger('load');
  3989  					loadEnd();
  3990  				} else {
  3991  					exec(_fr.connectRuntime(blob.ruid));
  3992  				}
  3993  			} else {
  3994  				error(new x.DOMException(x.DOMException.NOT_FOUND_ERR));
  3995  			}
  3996  		}
  3997  	}
  3998  	
  3999  	/**
  4000  	Initial FileReader state
  4001  
  4002  	@property EMPTY
  4003  	@type {Number}
  4004  	@final
  4005  	@static
  4006  	@default 0
  4007  	*/
  4008  	FileReader.EMPTY = 0;
  4009  
  4010  	/**
  4011  	FileReader switches to this state when it is preloading the source
  4012  
  4013  	@property LOADING
  4014  	@type {Number}
  4015  	@final
  4016  	@static
  4017  	@default 1
  4018  	*/
  4019  	FileReader.LOADING = 1;
  4020  
  4021  	/**
  4022  	Preloading is complete, this is a final state
  4023  
  4024  	@property DONE
  4025  	@type {Number}
  4026  	@final
  4027  	@static
  4028  	@default 2
  4029  	*/
  4030  	FileReader.DONE = 2;
  4031  
  4032  	FileReader.prototype = EventTarget.instance;
  4033  
  4034  	return FileReader;
  4035  });
  4036  
  4037  // Included from: src/javascript/core/utils/Url.js
  4038  
  4039  /**
  4040   * Url.js
  4041   *
  4042   * Copyright 2013, Moxiecode Systems AB
  4043   * Released under GPL License.
  4044   *
  4045   * License: http://www.plupload.com/license
  4046   * Contributing: http://www.plupload.com/contributing
  4047   */
  4048  
  4049  define('moxie/core/utils/Url', [], function() {
  4050  	/**
  4051  	Parse url into separate components and fill in absent parts with parts from current url,
  4052  	based on https://raw.github.com/kvz/phpjs/master/functions/url/parse_url.js
  4053  
  4054  	@method parseUrl
  4055  	@for Utils
  4056  	@static
  4057  	@param {String} url Url to parse (defaults to empty string if undefined)
  4058  	@return {Object} Hash containing extracted uri components
  4059  	*/
  4060  	var parseUrl = function(url, currentUrl) {
  4061  		var key = ['source', 'scheme', 'authority', 'userInfo', 'user', 'pass', 'host', 'port', 'relative', 'path', 'directory', 'file', 'query', 'fragment']
  4062  		, i = key.length
  4063  		, ports = {
  4064  			http: 80,
  4065  			https: 443
  4066  		}
  4067  		, uri = {}
  4068  		, regex = /^(?:([^:\/?#]+):)?(?:\/\/()(?:(?:()(?:([^:@]*):?([^:@]*))?@)?([^:\/?#]*)(?::(\d*))?))?()(?:(()(?:(?:[^?#\/]*\/)*)()(?:[^?#]*))(?:\\?([^#]*))?(?:#(.*))?)/
  4069  		, m = regex.exec(url || '')
  4070  		;
  4071  					
  4072  		while (i--) {
  4073  			if (m[i]) {
  4074  				uri[key[i]] = m[i];
  4075  			}
  4076  		}
  4077  
  4078  		// when url is relative, we set the origin and the path ourselves
  4079  		if (!uri.scheme) {
  4080  			// come up with defaults
  4081  			if (!currentUrl || typeof(currentUrl) === 'string') {
  4082  				currentUrl = parseUrl(currentUrl || document.location.href);
  4083  			}
  4084  
  4085  			uri.scheme = currentUrl.scheme;
  4086  			uri.host = currentUrl.host;
  4087  			uri.port = currentUrl.port;
  4088  
  4089  			var path = '';
  4090  			// for urls without trailing slash we need to figure out the path
  4091  			if (/^[^\/]/.test(uri.path)) {
  4092  				path = currentUrl.path;
  4093  				// if path ends with a filename, strip it
  4094  				if (!/(\/|\/[^\.]+)$/.test(path)) {
  4095  					path = path.replace(/\/[^\/]+$/, '/');
  4096  				} else {
  4097  					path += '/';
  4098  				}
  4099  			}
  4100  			uri.path = path + (uri.path || ''); // site may reside at domain.com or domain.com/subdir
  4101  		}
  4102  
  4103  		if (!uri.port) {
  4104  			uri.port = ports[uri.scheme] || 80;
  4105  		} 
  4106  		
  4107  		uri.port = parseInt(uri.port, 10);
  4108  
  4109  		if (!uri.path) {
  4110  			uri.path = "/";
  4111  		}
  4112  
  4113  		delete uri.source;
  4114  
  4115  		return uri;
  4116  	};
  4117  
  4118  	/**
  4119  	Resolve url - among other things will turn relative url to absolute
  4120  
  4121  	@method resolveUrl
  4122  	@static
  4123  	@param {String} url Either absolute or relative
  4124  	@return {String} Resolved, absolute url
  4125  	*/
  4126  	var resolveUrl = function(url) {
  4127  		var ports = { // we ignore default ports
  4128  			http: 80,
  4129  			https: 443
  4130  		}
  4131  		, urlp = parseUrl(url)
  4132  		;
  4133  
  4134  		return urlp.scheme + '://' + urlp.host + (urlp.port !== ports[urlp.scheme] ? ':' + urlp.port : '') + urlp.path + (urlp.query ? urlp.query : '');
  4135  	};
  4136  
  4137  	/**
  4138  	Check if specified url has the same origin as the current document
  4139  
  4140  	@method hasSameOrigin
  4141  	@param {String|Object} url
  4142  	@return {Boolean}
  4143  	*/
  4144  	var hasSameOrigin = function(url) {
  4145  		function origin(url) {
  4146  			return [url.scheme, url.host, url.port].join('/');
  4147  		}
  4148  			
  4149  		if (typeof url === 'string') {
  4150  			url = parseUrl(url);
  4151  		}	
  4152  		
  4153  		return origin(parseUrl()) === origin(url);
  4154  	};
  4155  
  4156  	return {
  4157  		parseUrl: parseUrl,
  4158  		resolveUrl: resolveUrl,
  4159  		hasSameOrigin: hasSameOrigin
  4160  	};
  4161  });
  4162  
  4163  // Included from: src/javascript/file/FileReaderSync.js
  4164  
  4165  /**
  4166   * FileReaderSync.js
  4167   *
  4168   * Copyright 2013, Moxiecode Systems AB
  4169   * Released under GPL License.
  4170   *
  4171   * License: http://www.plupload.com/license
  4172   * Contributing: http://www.plupload.com/contributing
  4173   */
  4174  
  4175  define('moxie/file/FileReaderSync', [
  4176  	'moxie/core/utils/Basic',
  4177  	'moxie/runtime/RuntimeClient',
  4178  	'moxie/core/utils/Encode'
  4179  ], function(Basic, RuntimeClient, Encode) {
  4180  	/**
  4181  	Synchronous FileReader implementation. Something like this is available in WebWorkers environment, here
  4182  	it can be used to read only preloaded blobs/files and only below certain size (not yet sure what that'd be,
  4183  	but probably < 1mb). Not meant to be used directly by user.
  4184  
  4185  	@class FileReaderSync
  4186  	@private
  4187  	@constructor
  4188  	*/
  4189  	return function() {
  4190  		RuntimeClient.call(this);
  4191  
  4192  		Basic.extend(this, {
  4193  			uid: Basic.guid('uid_'),
  4194  
  4195  			readAsBinaryString: function(blob) {
  4196  				return _read.call(this, 'readAsBinaryString', blob);
  4197  			},
  4198  			
  4199  			readAsDataURL: function(blob) {
  4200  				return _read.call(this, 'readAsDataURL', blob);
  4201  			},
  4202  			
  4203  			/*readAsArrayBuffer: function(blob) {
  4204  				return _read.call(this, 'readAsArrayBuffer', blob);
  4205  			},*/
  4206  			
  4207  			readAsText: function(blob) {
  4208  				return _read.call(this, 'readAsText', blob);
  4209  			}
  4210  		});
  4211  
  4212  		function _read(op, blob) {
  4213  			if (blob.isDetached()) {
  4214  				var src = blob.getSource();
  4215  				switch (op) {
  4216  					case 'readAsBinaryString':
  4217  						return src;
  4218  					case 'readAsDataURL':
  4219  						return 'data:' + blob.type + ';base64,' + Encode.btoa(src);
  4220  					case 'readAsText':
  4221  						var txt = '';
  4222  						for (var i = 0, length = src.length; i < length; i++) {
  4223  							txt += String.fromCharCode(src[i]);
  4224  						}
  4225  						return txt;
  4226  				}
  4227  			} else {
  4228  				var result = this.connectRuntime(blob.ruid).exec.call(this, 'FileReaderSync', 'read', op, blob);
  4229  				this.disconnectRuntime();
  4230  				return result;
  4231  			}
  4232  		}
  4233  	};
  4234  });
  4235  
  4236  // Included from: src/javascript/xhr/FormData.js
  4237  
  4238  /**
  4239   * FormData.js
  4240   *
  4241   * Copyright 2013, Moxiecode Systems AB
  4242   * Released under GPL License.
  4243   *
  4244   * License: http://www.plupload.com/license
  4245   * Contributing: http://www.plupload.com/contributing
  4246   */
  4247  
  4248  define("moxie/xhr/FormData", [
  4249  	"moxie/core/Exceptions",
  4250  	"moxie/core/utils/Basic",
  4251  	"moxie/file/Blob"
  4252  ], function(x, Basic, Blob) {
  4253  	/**
  4254  	FormData
  4255  
  4256  	@class FormData
  4257  	@constructor
  4258  	*/
  4259  	function FormData() {
  4260  		var _blob, _fields = [];
  4261  
  4262  		Basic.extend(this, {
  4263  			/**
  4264  			Append another key-value pair to the FormData object
  4265  
  4266  			@method append
  4267  			@param {String} name Name for the new field
  4268  			@param {String|Blob|Array|Object} value Value for the field
  4269  			*/
  4270  			append: function(name, value) {
  4271  				var self = this, valueType = Basic.typeOf(value);
  4272  
  4273  				// according to specs value might be either Blob or String
  4274  				if (value instanceof Blob) {
  4275  					_blob = {
  4276  						name: name,
  4277  						value: value // unfortunately we can only send single Blob in one FormData
  4278  					};
  4279  				} else if ('array' === valueType) {
  4280  					name += '[]';
  4281  
  4282  					Basic.each(value, function(value) {
  4283  						self.append(name, value);
  4284  					});
  4285  				} else if ('object' === valueType) {
  4286  					Basic.each(value, function(value, key) {
  4287  						self.append(name + '[' + key + ']', value);
  4288  					});
  4289  				} else if ('null' === valueType || 'undefined' === valueType || 'number' === valueType && isNaN(value)) {
  4290  					self.append(name, "false");
  4291  				} else {
  4292  					_fields.push({
  4293  						name: name,
  4294  						value: value.toString()
  4295  					});
  4296  				}
  4297  			},
  4298  
  4299  			/**
  4300  			Checks if FormData contains Blob.
  4301  
  4302  			@method hasBlob
  4303  			@return {Boolean}
  4304  			*/
  4305  			hasBlob: function() {
  4306  				return !!this.getBlob();
  4307  			},
  4308  
  4309  			/**
  4310  			Retrieves blob.
  4311  
  4312  			@method getBlob
  4313  			@return {Object} Either Blob if found or null
  4314  			*/
  4315  			getBlob: function() {
  4316  				return _blob && _blob.value || null;
  4317  			},
  4318  
  4319  			/**
  4320  			Retrieves blob field name.
  4321  
  4322  			@method getBlobName
  4323  			@return {String} Either Blob field name or null
  4324  			*/
  4325  			getBlobName: function() {
  4326  				return _blob && _blob.name || null;
  4327  			},
  4328  
  4329  			/**
  4330  			Loop over the fields in FormData and invoke the callback for each of them.
  4331  
  4332  			@method each
  4333  			@param {Function} cb Callback to call for each field
  4334  			*/
  4335  			each: function(cb) {
  4336  				Basic.each(_fields, function(field) {
  4337  					cb(field.value, field.name);
  4338  				});
  4339  
  4340  				if (_blob) {
  4341  					cb(_blob.value, _blob.name);
  4342  				}
  4343  			},
  4344  
  4345  			destroy: function() {
  4346  				_blob = null;
  4347  				_fields = [];
  4348  			}
  4349  		});
  4350  	}
  4351  
  4352  	return FormData;
  4353  });
  4354  
  4355  // Included from: src/javascript/xhr/XMLHttpRequest.js
  4356  
  4357  /**
  4358   * XMLHttpRequest.js
  4359   *
  4360   * Copyright 2013, Moxiecode Systems AB
  4361   * Released under GPL License.
  4362   *
  4363   * License: http://www.plupload.com/license
  4364   * Contributing: http://www.plupload.com/contributing
  4365   */
  4366  
  4367  define("moxie/xhr/XMLHttpRequest", [
  4368  	"moxie/core/utils/Basic",
  4369  	"moxie/core/Exceptions",
  4370  	"moxie/core/EventTarget",
  4371  	"moxie/core/utils/Encode",
  4372  	"moxie/core/utils/Url",
  4373  	"moxie/runtime/Runtime",
  4374  	"moxie/runtime/RuntimeTarget",
  4375  	"moxie/file/Blob",
  4376  	"moxie/file/FileReaderSync",
  4377  	"moxie/xhr/FormData",
  4378  	"moxie/core/utils/Env",
  4379  	"moxie/core/utils/Mime"
  4380  ], function(Basic, x, EventTarget, Encode, Url, Runtime, RuntimeTarget, Blob, FileReaderSync, FormData, Env, Mime) {
  4381  
  4382  	var httpCode = {
  4383  		100: 'Continue',
  4384  		101: 'Switching Protocols',
  4385  		102: 'Processing',
  4386  
  4387  		200: 'OK',
  4388  		201: 'Created',
  4389  		202: 'Accepted',
  4390  		203: 'Non-Authoritative Information',
  4391  		204: 'No Content',
  4392  		205: 'Reset Content',
  4393  		206: 'Partial Content',
  4394  		207: 'Multi-Status',
  4395  		226: 'IM Used',
  4396  
  4397  		300: 'Multiple Choices',
  4398  		301: 'Moved Permanently',
  4399  		302: 'Found',
  4400  		303: 'See Other',
  4401  		304: 'Not Modified',
  4402  		305: 'Use Proxy',
  4403  		306: 'Reserved',
  4404  		307: 'Temporary Redirect',
  4405  
  4406  		400: 'Bad Request',
  4407  		401: 'Unauthorized',
  4408  		402: 'Payment Required',
  4409  		403: 'Forbidden',
  4410  		404: 'Not Found',
  4411  		405: 'Method Not Allowed',
  4412  		406: 'Not Acceptable',
  4413  		407: 'Proxy Authentication Required',
  4414  		408: 'Request Timeout',
  4415  		409: 'Conflict',
  4416  		410: 'Gone',
  4417  		411: 'Length Required',
  4418  		412: 'Precondition Failed',
  4419  		413: 'Request Entity Too Large',
  4420  		414: 'Request-URI Too Long',
  4421  		415: 'Unsupported Media Type',
  4422  		416: 'Requested Range Not Satisfiable',
  4423  		417: 'Expectation Failed',
  4424  		422: 'Unprocessable Entity',
  4425  		423: 'Locked',
  4426  		424: 'Failed Dependency',
  4427  		426: 'Upgrade Required',
  4428  
  4429  		500: 'Internal Server Error',
  4430  		501: 'Not Implemented',
  4431  		502: 'Bad Gateway',
  4432  		503: 'Service Unavailable',
  4433  		504: 'Gateway Timeout',
  4434  		505: 'HTTP Version Not Supported',
  4435  		506: 'Variant Also Negotiates',
  4436  		507: 'Insufficient Storage',
  4437  		510: 'Not Extended'
  4438  	};
  4439  
  4440  	function XMLHttpRequestUpload() {
  4441  		this.uid = Basic.guid('uid_');
  4442  	}
  4443  	
  4444  	XMLHttpRequestUpload.prototype = EventTarget.instance;
  4445  
  4446  	/**
  4447  	Implementation of XMLHttpRequest
  4448  
  4449  	@class XMLHttpRequest
  4450  	@constructor
  4451  	@uses RuntimeClient
  4452  	@extends EventTarget
  4453  	*/
  4454  	var dispatches = ['loadstart', 'progress', 'abort', 'error', 'load', 'timeout', 'loadend']; // & readystatechange (for historical reasons)
  4455  	
  4456  	var NATIVE = 1, RUNTIME = 2;
  4457  					
  4458  	function XMLHttpRequest() {
  4459  		var self = this,
  4460  			// this (together with _p() @see below) is here to gracefully upgrade to setter/getter syntax where possible
  4461  			props = {
  4462  				/**
  4463  				The amount of milliseconds a request can take before being terminated. Initially zero. Zero means there is no timeout.
  4464  
  4465  				@property timeout
  4466  				@type Number
  4467  				@default 0
  4468  				*/
  4469  				timeout: 0,
  4470  
  4471  				/**
  4472  				Current state, can take following values:
  4473  				UNSENT (numeric value 0)
  4474  				The object has been constructed.
  4475  
  4476  				OPENED (numeric value 1)
  4477  				The open() method has been successfully invoked. During this state request headers can be set using setRequestHeader() and the request can be made using the send() method.
  4478  
  4479  				HEADERS_RECEIVED (numeric value 2)
  4480  				All redirects (if any) have been followed and all HTTP headers of the final response have been received. Several response members of the object are now available.
  4481  
  4482  				LOADING (numeric value 3)
  4483  				The response entity body is being received.
  4484  
  4485  				DONE (numeric value 4)
  4486  
  4487  				@property readyState
  4488  				@type Number
  4489  				@default 0 (UNSENT)
  4490  				*/
  4491  				readyState: XMLHttpRequest.UNSENT,
  4492  
  4493  				/**
  4494  				True when user credentials are to be included in a cross-origin request. False when they are to be excluded
  4495  				in a cross-origin request and when cookies are to be ignored in its response. Initially false.
  4496  
  4497  				@property withCredentials
  4498  				@type Boolean
  4499  				@default false
  4500  				*/
  4501  				withCredentials: false,
  4502  
  4503  				/**
  4504  				Returns the HTTP status code.
  4505  
  4506  				@property status
  4507  				@type Number
  4508  				@default 0
  4509  				*/
  4510  				status: 0,
  4511  
  4512  				/**
  4513  				Returns the HTTP status text.
  4514  
  4515  				@property statusText
  4516  				@type String
  4517  				*/
  4518  				statusText: "",
  4519  
  4520  				/**
  4521  				Returns the response type. Can be set to change the response type. Values are:
  4522  				the empty string (default), "arraybuffer", "blob", "document", "json", and "text".
  4523  				
  4524  				@property responseType
  4525  				@type String
  4526  				*/
  4527  				responseType: "",
  4528  
  4529  				/**
  4530  				Returns the document response entity body.
  4531  				
  4532  				Throws an "InvalidStateError" exception if responseType is not the empty string or "document".
  4533  
  4534  				@property responseXML
  4535  				@type Document
  4536  				*/
  4537  				responseXML: null,
  4538  
  4539  				/**
  4540  				Returns the text response entity body.
  4541  				
  4542  				Throws an "InvalidStateError" exception if responseType is not the empty string or "text".
  4543  
  4544  				@property responseText
  4545  				@type String
  4546  				*/
  4547  				responseText: null,
  4548  
  4549  				/**
  4550  				Returns the response entity body (http://www.w3.org/TR/XMLHttpRequest/#response-entity-body).
  4551  				Can become: ArrayBuffer, Blob, Document, JSON, Text
  4552  				
  4553  				@property response
  4554  				@type Mixed
  4555  				*/
  4556  				response: null
  4557  			},
  4558  
  4559  			_async = true,
  4560  			_url,
  4561  			_method,
  4562  			_headers = {},
  4563  			_user,
  4564  			_password,
  4565  			_encoding = null,
  4566  			_mimeType = null,
  4567  
  4568  			// flags
  4569  			_sync_flag = false,
  4570  			_send_flag = false,
  4571  			_upload_events_flag = false,
  4572  			_upload_complete_flag = false,
  4573  			_error_flag = false,
  4574  			_same_origin_flag = false,
  4575  
  4576  			// times
  4577  			_start_time,
  4578  			_timeoutset_time,
  4579  
  4580  			_finalMime = null,
  4581  			_finalCharset = null,
  4582  
  4583  			_options = {},
  4584  			_xhr,
  4585  			_responseHeaders = '',
  4586  			_responseHeadersBag
  4587  			;
  4588  
  4589  		
  4590  		Basic.extend(this, props, {
  4591  			/**
  4592  			Unique id of the component
  4593  
  4594  			@property uid
  4595  			@type String
  4596  			*/
  4597  			uid: Basic.guid('uid_'),
  4598  			
  4599  			/**
  4600  			Target for Upload events
  4601  
  4602  			@property upload
  4603  			@type XMLHttpRequestUpload
  4604  			*/
  4605  			upload: new XMLHttpRequestUpload(),
  4606  			
  4607  
  4608  			/**
  4609  			Sets the request method, request URL, synchronous flag, request username, and request password.
  4610  
  4611  			Throws a "SyntaxError" exception if one of the following is true:
  4612  
  4613  			method is not a valid HTTP method.
  4614  			url cannot be resolved.
  4615  			url contains the "user:password" format in the userinfo production.
  4616  			Throws a "SecurityError" exception if method is a case-insensitive match for CONNECT, TRACE or TRACK.
  4617  
  4618  			Throws an "InvalidAccessError" exception if one of the following is true:
  4619  
  4620  			Either user or password is passed as argument and the origin of url does not match the XMLHttpRequest origin.
  4621  			There is an associated XMLHttpRequest document and either the timeout attribute is not zero,
  4622  			the withCredentials attribute is true, or the responseType attribute is not the empty string.
  4623  
  4624  
  4625  			@method open
  4626  			@param {String} method HTTP method to use on request
  4627  			@param {String} url URL to request
  4628  			@param {Boolean} [async=true] If false request will be done in synchronous manner. Asynchronous by default.
  4629  			@param {String} [user] Username to use in HTTP authentication process on server-side
  4630  			@param {String} [password] Password to use in HTTP authentication process on server-side
  4631  			*/
  4632  			open: function(method, url, async, user, password) {
  4633  				var urlp;
  4634  				
  4635  				// first two arguments are required
  4636  				if (!method || !url) {
  4637  					throw new x.DOMException(x.DOMException.SYNTAX_ERR);
  4638  				}
  4639  				
  4640  				// 2 - check if any code point in method is higher than U+00FF or after deflating method it does not match the method
  4641  				if (/[\u0100-\uffff]/.test(method) || Encode.utf8_encode(method) !== method) {
  4642  					throw new x.DOMException(x.DOMException.SYNTAX_ERR);
  4643  				}
  4644  
  4645  				// 3
  4646  				if (!!~Basic.inArray(method.toUpperCase(), ['CONNECT', 'DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT', 'TRACE', 'TRACK'])) {
  4647  					_method = method.toUpperCase();
  4648  				}
  4649  				
  4650  				
  4651  				// 4 - allowing these methods poses a security risk
  4652  				if (!!~Basic.inArray(_method, ['CONNECT', 'TRACE', 'TRACK'])) {
  4653  					throw new x.DOMException(x.DOMException.SECURITY_ERR);
  4654  				}
  4655  
  4656  				// 5
  4657  				url = Encode.utf8_encode(url);
  4658  				
  4659  				// 6 - Resolve url relative to the XMLHttpRequest base URL. If the algorithm returns an error, throw a "SyntaxError".
  4660  				urlp = Url.parseUrl(url);
  4661  
  4662  				_same_origin_flag = Url.hasSameOrigin(urlp);
  4663  																
  4664  				// 7 - manually build up absolute url
  4665  				_url = Url.resolveUrl(url);
  4666  		
  4667  				// 9-10, 12-13
  4668  				if ((user || password) && !_same_origin_flag) {
  4669  					throw new x.DOMException(x.DOMException.INVALID_ACCESS_ERR);
  4670  				}
  4671  
  4672  				_user = user || urlp.user;
  4673  				_password = password || urlp.pass;
  4674  				
  4675  				// 11
  4676  				_async = async || true;
  4677  				
  4678  				if (_async === false && (_p('timeout') || _p('withCredentials') || _p('responseType') !== "")) {
  4679  					throw new x.DOMException(x.DOMException.INVALID_ACCESS_ERR);
  4680  				}
  4681  				
  4682  				// 14 - terminate abort()
  4683  				
  4684  				// 15 - terminate send()
  4685  
  4686  				// 18
  4687  				_sync_flag = !_async;
  4688  				_send_flag = false;
  4689  				_headers = {};
  4690  				_reset.call(this);
  4691  
  4692  				// 19
  4693  				_p('readyState', XMLHttpRequest.OPENED);
  4694  				
  4695  				// 20
  4696  				this.convertEventPropsToHandlers(['readystatechange']); // unify event handlers
  4697  				this.dispatchEvent('readystatechange');
  4698  			},
  4699  			
  4700  			/**
  4701  			Appends an header to the list of author request headers, or if header is already
  4702  			in the list of author request headers, combines its value with value.
  4703  
  4704  			Throws an "InvalidStateError" exception if the state is not OPENED or if the send() flag is set.
  4705  			Throws a "SyntaxError" exception if header is not a valid HTTP header field name or if value
  4706  			is not a valid HTTP header field value.
  4707  			
  4708  			@method setRequestHeader
  4709  			@param {String} header
  4710  			@param {String|Number} value
  4711  			*/
  4712  			setRequestHeader: function(header, value) {
  4713  				var uaHeaders = [ // these headers are controlled by the user agent
  4714  						"accept-charset",
  4715  						"accept-encoding",
  4716  						"access-control-request-headers",
  4717  						"access-control-request-method",
  4718  						"connection",
  4719  						"content-length",
  4720  						"cookie",
  4721  						"cookie2",
  4722  						"content-transfer-encoding",
  4723  						"date",
  4724  						"expect",
  4725  						"host",
  4726  						"keep-alive",
  4727  						"origin",
  4728  						"referer",
  4729  						"te",
  4730  						"trailer",
  4731  						"transfer-encoding",
  4732  						"upgrade",
  4733  						"user-agent",
  4734  						"via"
  4735  					];
  4736  				
  4737  				// 1-2
  4738  				if (_p('readyState') !== XMLHttpRequest.OPENED || _send_flag) {
  4739  					throw new x.DOMException(x.DOMException.INVALID_STATE_ERR);
  4740  				}
  4741  
  4742  				// 3
  4743  				if (/[\u0100-\uffff]/.test(header) || Encode.utf8_encode(header) !== header) {
  4744  					throw new x.DOMException(x.DOMException.SYNTAX_ERR);
  4745  				}
  4746  
  4747  				// 4
  4748  				/* this step is seemingly bypassed in browsers, probably to allow various unicode characters in header values
  4749  				if (/[\u0100-\uffff]/.test(value) || Encode.utf8_encode(value) !== value) {
  4750  					throw new x.DOMException(x.DOMException.SYNTAX_ERR);
  4751  				}*/
  4752  
  4753  				header = Basic.trim(header).toLowerCase();
  4754  				
  4755  				// setting of proxy-* and sec-* headers is prohibited by spec
  4756  				if (!!~Basic.inArray(header, uaHeaders) || /^(proxy\-|sec\-)/.test(header)) {
  4757  					return false;
  4758  				}
  4759  
  4760  				// camelize
  4761  				// browsers lowercase header names (at least for custom ones)
  4762  				// header = header.replace(/\b\w/g, function($1) { return $1.toUpperCase(); });
  4763  				
  4764  				if (!_headers[header]) {
  4765  					_headers[header] = value;
  4766  				} else {
  4767  					// http://tools.ietf.org/html/rfc2616#section-4.2 (last paragraph)
  4768  					_headers[header] += ', ' + value;
  4769  				}
  4770  				return true;
  4771  			},
  4772  
  4773  			/**
  4774  			Returns all headers from the response, with the exception of those whose field name is Set-Cookie or Set-Cookie2.
  4775  
  4776  			@method getAllResponseHeaders
  4777  			@return {String} reponse headers or empty string
  4778  			*/
  4779  			getAllResponseHeaders: function() {
  4780  				return _responseHeaders || '';
  4781  			},
  4782  
  4783  			/**
  4784  			Returns the header field value from the response of which the field name matches header, 
  4785  			unless the field name is Set-Cookie or Set-Cookie2.
  4786  
  4787  			@method getResponseHeader
  4788  			@param {String} header
  4789  			@return {String} value(s) for the specified header or null
  4790  			*/
  4791  			getResponseHeader: function(header) {
  4792  				header = header.toLowerCase();
  4793  
  4794  				if (_error_flag || !!~Basic.inArray(header, ['set-cookie', 'set-cookie2'])) {
  4795  					return null;
  4796  				}
  4797  
  4798  				if (_responseHeaders && _responseHeaders !== '') {
  4799  					// if we didn't parse response headers until now, do it and keep for later
  4800  					if (!_responseHeadersBag) {
  4801  						_responseHeadersBag = {};
  4802  						Basic.each(_responseHeaders.split(/\r\n/), function(line) {
  4803  							var pair = line.split(/:\s+/);
  4804  							if (pair.length === 2) { // last line might be empty, omit
  4805  								pair[0] = Basic.trim(pair[0]); // just in case
  4806  								_responseHeadersBag[pair[0].toLowerCase()] = { // simply to retain header name in original form
  4807  									header: pair[0],
  4808  									value: Basic.trim(pair[1])
  4809  								};
  4810  							}
  4811  						});
  4812  					}
  4813  					if (_responseHeadersBag.hasOwnProperty(header)) {
  4814  						return _responseHeadersBag[header].header + ': ' + _responseHeadersBag[header].value;
  4815  					}
  4816  				}
  4817  				return null;
  4818  			},
  4819  			
  4820  			/**
  4821  			Sets the Content-Type header for the response to mime.
  4822  			Throws an "InvalidStateError" exception if the state is LOADING or DONE.
  4823  			Throws a "SyntaxError" exception if mime is not a valid media type.
  4824  
  4825  			@method overrideMimeType
  4826  			@param String mime Mime type to set
  4827  			*/
  4828  			overrideMimeType: function(mime) {
  4829  				var matches, charset;
  4830  			
  4831  				// 1
  4832  				if (!!~Basic.inArray(_p('readyState'), [XMLHttpRequest.LOADING, XMLHttpRequest.DONE])) {
  4833  					throw new x.DOMException(x.DOMException.INVALID_STATE_ERR);
  4834  				}
  4835  
  4836  				// 2
  4837  				mime = Basic.trim(mime.toLowerCase());
  4838  
  4839  				if (/;/.test(mime) && (matches = mime.match(/^([^;]+)(?:;\scharset\=)?(.*)$/))) {
  4840  					mime = matches[1];
  4841  					if (matches[2]) {
  4842  						charset = matches[2];
  4843  					}
  4844  				}
  4845  
  4846  				if (!Mime.mimes[mime]) {
  4847  					throw new x.DOMException(x.DOMException.SYNTAX_ERR);
  4848  				}
  4849  
  4850  				// 3-4
  4851  				_finalMime = mime;
  4852  				_finalCharset = charset;
  4853  			},
  4854  			
  4855  			/**
  4856  			Initiates the request. The optional argument provides the request entity body.
  4857  			The argument is ignored if request method is GET or HEAD.
  4858  
  4859  			Throws an "InvalidStateError" exception if the state is not OPENED or if the send() flag is set.
  4860  
  4861  			@method send
  4862  			@param {Blob|Document|String|FormData} [data] Request entity body
  4863  			@param {Object} [options] Set of requirements and pre-requisities for runtime initialization
  4864  			*/
  4865  			send: function(data, options) {					
  4866  				if (Basic.typeOf(options) === 'string') {
  4867  					_options = { ruid: options };
  4868  				} else if (!options) {
  4869  					_options = {};
  4870  				} else {
  4871  					_options = options;
  4872  				}
  4873  													
  4874  				this.convertEventPropsToHandlers(dispatches);
  4875  				this.upload.convertEventPropsToHandlers(dispatches);
  4876  															
  4877  				// 1-2
  4878  				if (this.readyState !== XMLHttpRequest.OPENED || _send_flag) {
  4879  					throw new x.DOMException(x.DOMException.INVALID_STATE_ERR);
  4880  				}
  4881  				
  4882  				// 3					
  4883  				// sending Blob
  4884  				if (data instanceof Blob) {
  4885  					_options.ruid = data.ruid;
  4886  					_mimeType = data.type || 'application/octet-stream';
  4887  				}
  4888  				
  4889  				// FormData
  4890  				else if (data instanceof FormData) {
  4891  					if (data.hasBlob()) {
  4892  						var blob = data.getBlob();
  4893  						_options.ruid = blob.ruid;
  4894  						_mimeType = blob.type || 'application/octet-stream';
  4895  					}
  4896  				}
  4897  				
  4898  				// DOMString
  4899  				else if (typeof data === 'string') {
  4900  					_encoding = 'UTF-8';
  4901  					_mimeType = 'text/plain;charset=UTF-8';
  4902  					
  4903  					// data should be converted to Unicode and encoded as UTF-8
  4904  					data = Encode.utf8_encode(data);
  4905  				}
  4906  
  4907  				// if withCredentials not set, but requested, set it automatically
  4908  				if (!this.withCredentials) {
  4909  					this.withCredentials = (_options.required_caps && _options.required_caps.send_browser_cookies) && !_same_origin_flag;
  4910  				}
  4911  
  4912  				// 4 - storage mutex
  4913  				// 5
  4914  				_upload_events_flag = (!_sync_flag && this.upload.hasEventListener()); // DSAP
  4915  				// 6
  4916  				_error_flag = false;
  4917  				// 7
  4918  				_upload_complete_flag = !data;
  4919  				// 8 - Asynchronous steps
  4920  				if (!_sync_flag) {
  4921  					// 8.1
  4922  					_send_flag = true;
  4923  					// 8.2
  4924  					// this.dispatchEvent('loadstart'); // will be dispatched either by native or runtime xhr
  4925  					// 8.3
  4926  					//if (!_upload_complete_flag) {
  4927  						// this.upload.dispatchEvent('loadstart');	// will be dispatched either by native or runtime xhr
  4928  					//}
  4929  				}
  4930  				// 8.5 - Return the send() method call, but continue running the steps in this algorithm.
  4931  				_doXHR.call(this, data);
  4932  			},
  4933  			
  4934  			/**
  4935  			Cancels any network activity.
  4936  			
  4937  			@method abort
  4938  			*/
  4939  			abort: function() {
  4940  				_error_flag = true;
  4941  				_sync_flag = false;
  4942  
  4943  				if (!~Basic.inArray(_p('readyState'), [XMLHttpRequest.UNSENT, XMLHttpRequest.OPENED, XMLHttpRequest.DONE])) {
  4944  					_p('readyState', XMLHttpRequest.DONE);
  4945  					_send_flag = false;
  4946  
  4947  					if (_xhr) {
  4948  						_xhr.getRuntime().exec.call(_xhr, 'XMLHttpRequest', 'abort', _upload_complete_flag);
  4949  					} else {
  4950  						throw new x.DOMException(x.DOMException.INVALID_STATE_ERR);
  4951  					}
  4952  
  4953  					_upload_complete_flag = true;
  4954  				} else {
  4955  					_p('readyState', XMLHttpRequest.UNSENT);
  4956  				}
  4957  			},
  4958  
  4959  			destroy: function() {
  4960  				if (_xhr) {
  4961  					if (Basic.typeOf(_xhr.destroy) === 'function') {
  4962  						_xhr.destroy();
  4963  					}
  4964  					_xhr = null;
  4965  				}
  4966  
  4967  				this.unbindAll();
  4968  
  4969  				if (this.upload) {
  4970  					this.upload.unbindAll();
  4971  					this.upload = null;
  4972  				}
  4973  			}
  4974  		});
  4975  
  4976  		/* this is nice, but maybe too lengthy
  4977  
  4978  		// if supported by JS version, set getters/setters for specific properties
  4979  		o.defineProperty(this, 'readyState', {
  4980  			configurable: false,
  4981  
  4982  			get: function() {
  4983  				return _p('readyState');
  4984  			}
  4985  		});
  4986  
  4987  		o.defineProperty(this, 'timeout', {
  4988  			configurable: false,
  4989  
  4990  			get: function() {
  4991  				return _p('timeout');
  4992  			},
  4993  
  4994  			set: function(value) {
  4995  
  4996  				if (_sync_flag) {
  4997  					throw new x.DOMException(x.DOMException.INVALID_ACCESS_ERR);
  4998  				}
  4999  
  5000  				// timeout still should be measured relative to the start time of request
  5001  				_timeoutset_time = (new Date).getTime();
  5002  
  5003  				_p('timeout', value);
  5004  			}
  5005  		});
  5006  
  5007  		// the withCredentials attribute has no effect when fetching same-origin resources
  5008  		o.defineProperty(this, 'withCredentials', {
  5009  			configurable: false,
  5010  
  5011  			get: function() {
  5012  				return _p('withCredentials');
  5013  			},
  5014  
  5015  			set: function(value) {
  5016  				// 1-2
  5017  				if (!~o.inArray(_p('readyState'), [XMLHttpRequest.UNSENT, XMLHttpRequest.OPENED]) || _send_flag) {
  5018  					throw new x.DOMException(x.DOMException.INVALID_STATE_ERR);
  5019  				}
  5020  
  5021  				// 3-4
  5022  				if (_anonymous_flag || _sync_flag) {
  5023  					throw new x.DOMException(x.DOMException.INVALID_ACCESS_ERR);
  5024  				}
  5025  
  5026  				// 5
  5027  				_p('withCredentials', value);
  5028  			}
  5029  		});
  5030  
  5031  		o.defineProperty(this, 'status', {
  5032  			configurable: false,
  5033  
  5034  			get: function() {
  5035  				return _p('status');
  5036  			}
  5037  		});
  5038  
  5039  		o.defineProperty(this, 'statusText', {
  5040  			configurable: false,
  5041  
  5042  			get: function() {
  5043  				return _p('statusText');
  5044  			}
  5045  		});
  5046  
  5047  		o.defineProperty(this, 'responseType', {
  5048  			configurable: false,
  5049  
  5050  			get: function() {
  5051  				return _p('responseType');
  5052  			},
  5053  
  5054  			set: function(value) {
  5055  				// 1
  5056  				if (!!~o.inArray(_p('readyState'), [XMLHttpRequest.LOADING, XMLHttpRequest.DONE])) {
  5057  					throw new x.DOMException(x.DOMException.INVALID_STATE_ERR);
  5058  				}
  5059  
  5060  				// 2
  5061  				if (_sync_flag) {
  5062  					throw new x.DOMException(x.DOMException.INVALID_ACCESS_ERR);
  5063  				}
  5064  
  5065  				// 3
  5066  				_p('responseType', value.toLowerCase());
  5067  			}
  5068  		});
  5069  
  5070  		o.defineProperty(this, 'responseText', {
  5071  			configurable: false,
  5072  
  5073  			get: function() {
  5074  				// 1
  5075  				if (!~o.inArray(_p('responseType'), ['', 'text'])) {
  5076  					throw new x.DOMException(x.DOMException.INVALID_STATE_ERR);
  5077  				}
  5078  
  5079  				// 2-3
  5080  				if (_p('readyState') !== XMLHttpRequest.DONE && _p('readyState') !== XMLHttpRequest.LOADING || _error_flag) {
  5081  					throw new x.DOMException(x.DOMException.INVALID_STATE_ERR);
  5082  				}
  5083  
  5084  				return _p('responseText');
  5085  			}
  5086  		});
  5087  
  5088  		o.defineProperty(this, 'responseXML', {
  5089  			configurable: false,
  5090  
  5091  			get: function() {
  5092  				// 1
  5093  				if (!~o.inArray(_p('responseType'), ['', 'document'])) {
  5094  					throw new x.DOMException(x.DOMException.INVALID_STATE_ERR);
  5095  				}
  5096  
  5097  				// 2-3
  5098  				if (_p('readyState') !== XMLHttpRequest.DONE || _error_flag) {
  5099  					throw new x.DOMException(x.DOMException.INVALID_STATE_ERR);
  5100  				}
  5101  
  5102  				return _p('responseXML');
  5103  			}
  5104  		});
  5105  
  5106  		o.defineProperty(this, 'response', {
  5107  			configurable: false,
  5108  
  5109  			get: function() {
  5110  				if (!!~o.inArray(_p('responseType'), ['', 'text'])) {
  5111  					if (_p('readyState') !== XMLHttpRequest.DONE && _p('readyState') !== XMLHttpRequest.LOADING || _error_flag) {
  5112  						return '';
  5113  					}
  5114  				}
  5115  
  5116  				if (_p('readyState') !== XMLHttpRequest.DONE || _error_flag) {
  5117  					return null;
  5118  				}
  5119  
  5120  				return _p('response');
  5121  			}
  5122  		});
  5123  
  5124  		*/
  5125  
  5126  		function _p(prop, value) {
  5127  			if (!props.hasOwnProperty(prop)) {
  5128  				return;
  5129  			}
  5130  			if (arguments.length === 1) { // get
  5131  				return Env.can('define_property') ? props[prop] : self[prop];
  5132  			} else { // set
  5133  				if (Env.can('define_property')) {
  5134  					props[prop] = value;
  5135  				} else {
  5136  					self[prop] = value;
  5137  				}
  5138  			}
  5139  		}
  5140  		
  5141  		/*
  5142  		function _toASCII(str, AllowUnassigned, UseSTD3ASCIIRules) {
  5143  			// TODO: http://tools.ietf.org/html/rfc3490#section-4.1
  5144  			return str.toLowerCase();
  5145  		}
  5146  		*/
  5147  		
  5148  		
  5149  		function _doXHR(data) {
  5150  			var self = this;
  5151  			
  5152  			_start_time = new Date().getTime();
  5153  
  5154  			_xhr = new RuntimeTarget();
  5155  
  5156  			function loadEnd() {
  5157  				_xhr.destroy();
  5158  				_xhr = null;
  5159  				self.dispatchEvent('loadend');
  5160  				self = null;
  5161  			}
  5162  
  5163  			function exec(runtime) {
  5164  				_xhr.bind('LoadStart', function(e) {
  5165  					_p('readyState', XMLHttpRequest.LOADING);
  5166  					self.dispatchEvent('readystatechange');
  5167  
  5168  					self.dispatchEvent(e);
  5169  					
  5170  					if (_upload_events_flag) {
  5171  						self.upload.dispatchEvent(e);
  5172  					}
  5173  				});
  5174  				
  5175  				_xhr.bind('Progress', function(e) {
  5176  					if (_p('readyState') !== XMLHttpRequest.LOADING) {
  5177  						_p('readyState', XMLHttpRequest.LOADING); // LoadStart unreliable (in Flash for example)
  5178  						self.dispatchEvent('readystatechange');
  5179  					}
  5180  					self.dispatchEvent(e);
  5181  				});
  5182  				
  5183  				_xhr.bind('UploadProgress', function(e) {
  5184  					if (_upload_events_flag) {
  5185  						self.upload.dispatchEvent({
  5186  							type: 'progress',
  5187  							lengthComputable: false,
  5188  							total: e.total,
  5189  							loaded: e.loaded
  5190  						});
  5191  					}
  5192  				});
  5193  				
  5194  				_xhr.bind('Load', function(e) {
  5195  					_p('readyState', XMLHttpRequest.DONE);
  5196  					_p('status', Number(runtime.exec.call(_xhr, 'XMLHttpRequest', 'getStatus') || 0));
  5197  					_p('statusText', httpCode[_p('status')] || "");
  5198  					
  5199  					_p('response', runtime.exec.call(_xhr, 'XMLHttpRequest', 'getResponse', _p('responseType')));
  5200  
  5201  					if (!!~Basic.inArray(_p('responseType'), ['text', ''])) {
  5202  						_p('responseText', _p('response'));
  5203  					} else if (_p('responseType') === 'document') {
  5204  						_p('responseXML', _p('response'));
  5205  					}
  5206  
  5207  					_responseHeaders = runtime.exec.call(_xhr, 'XMLHttpRequest', 'getAllResponseHeaders');
  5208  
  5209  					self.dispatchEvent('readystatechange');
  5210  					
  5211  					if (_p('status') > 0) { // status 0 usually means that server is unreachable
  5212  						if (_upload_events_flag) {
  5213  							self.upload.dispatchEvent(e);
  5214  						}
  5215  						self.dispatchEvent(e);
  5216  					} else {
  5217  						_error_flag = true;
  5218  						self.dispatchEvent('error');
  5219  					}
  5220  					loadEnd();
  5221  				});
  5222  
  5223  				_xhr.bind('Abort', function(e) {
  5224  					self.dispatchEvent(e);
  5225  					loadEnd();
  5226  				});
  5227  				
  5228  				_xhr.bind('Error', function(e) {
  5229  					_error_flag = true;
  5230  					_p('readyState', XMLHttpRequest.DONE);
  5231  					self.dispatchEvent('readystatechange');
  5232  					_upload_complete_flag = true;
  5233  					self.dispatchEvent(e);
  5234  					loadEnd();
  5235  				});
  5236  
  5237  				runtime.exec.call(_xhr, 'XMLHttpRequest', 'send', {
  5238  					url: _url,
  5239  					method: _method,
  5240  					async: _async,
  5241  					user: _user,
  5242  					password: _password,
  5243  					headers: _headers,
  5244  					mimeType: _mimeType,
  5245  					encoding: _encoding,
  5246  					responseType: self.responseType,
  5247  					withCredentials: self.withCredentials,
  5248  					options: _options
  5249  				}, data);
  5250  			}
  5251  
  5252  			// clarify our requirements
  5253  			if (typeof(_options.required_caps) === 'string') {
  5254  				_options.required_caps = Runtime.parseCaps(_options.required_caps);
  5255  			}
  5256  
  5257  			_options.required_caps = Basic.extend({}, _options.required_caps, {
  5258  				return_response_type: self.responseType
  5259  			});
  5260  
  5261  			if (data instanceof FormData) {
  5262  				_options.required_caps.send_multipart = true;
  5263  			}
  5264  
  5265  			if (!_same_origin_flag) {
  5266  				_options.required_caps.do_cors = true;
  5267  			}
  5268  			
  5269  
  5270  			if (_options.ruid) { // we do not need to wait if we can connect directly
  5271  				exec(_xhr.connectRuntime(_options));
  5272  			} else {
  5273  				_xhr.bind('RuntimeInit', function(e, runtime) {
  5274  					exec(runtime);
  5275  				});
  5276  				_xhr.bind('RuntimeError', function(e, err) {
  5277  					self.dispatchEvent('RuntimeError', err);
  5278  				});
  5279  				_xhr.connectRuntime(_options);
  5280  			}
  5281  		}
  5282  	
  5283  		
  5284  		function _reset() {
  5285  			_p('responseText', "");
  5286  			_p('responseXML', null);
  5287  			_p('response', null);
  5288  			_p('status', 0);
  5289  			_p('statusText', "");
  5290  			_start_time = _timeoutset_time = null;
  5291  		}
  5292  	}
  5293  
  5294  	XMLHttpRequest.UNSENT = 0;
  5295  	XMLHttpRequest.OPENED = 1;
  5296  	XMLHttpRequest.HEADERS_RECEIVED = 2;
  5297  	XMLHttpRequest.LOADING = 3;
  5298  	XMLHttpRequest.DONE = 4;
  5299  	
  5300  	XMLHttpRequest.prototype = EventTarget.instance;
  5301  
  5302  	return XMLHttpRequest;
  5303  });
  5304  
  5305  // Included from: src/javascript/runtime/Transporter.js
  5306  
  5307  /**
  5308   * Transporter.js
  5309   *
  5310   * Copyright 2013, Moxiecode Systems AB
  5311   * Released under GPL License.
  5312   *
  5313   * License: http://www.plupload.com/license
  5314   * Contributing: http://www.plupload.com/contributing
  5315   */
  5316  
  5317  define("moxie/runtime/Transporter", [
  5318  	"moxie/core/utils/Basic",
  5319  	"moxie/core/utils/Encode",
  5320  	"moxie/runtime/RuntimeClient",
  5321  	"moxie/core/EventTarget"
  5322  ], function(Basic, Encode, RuntimeClient, EventTarget) {
  5323  	function Transporter() {
  5324  		var mod, _runtime, _data, _size, _pos, _chunk_size;
  5325  
  5326  		RuntimeClient.call(this);
  5327  
  5328  		Basic.extend(this, {
  5329  			uid: Basic.guid('uid_'),
  5330  
  5331  			state: Transporter.IDLE,
  5332  
  5333  			result: null,
  5334  
  5335  			transport: function(data, type, options) {
  5336  				var self = this;
  5337  
  5338  				options = Basic.extend({
  5339  					chunk_size: 204798
  5340  				}, options);
  5341  
  5342  				// should divide by three, base64 requires this
  5343  				if ((mod = options.chunk_size % 3)) {
  5344  					options.chunk_size += 3 - mod;
  5345  				}
  5346  
  5347  				_chunk_size = options.chunk_size;
  5348  
  5349  				_reset.call(this);
  5350  				_data = data;
  5351  				_size = data.length;
  5352  
  5353  				if (Basic.typeOf(options) === 'string' || options.ruid) {
  5354  					_run.call(self, type, this.connectRuntime(options));
  5355  				} else {
  5356  					// we require this to run only once
  5357  					var cb = function(e, runtime) {
  5358  						self.unbind("RuntimeInit", cb);
  5359  						_run.call(self, type, runtime);
  5360  					};
  5361  					this.bind("RuntimeInit", cb);
  5362  					this.connectRuntime(options);
  5363  				}
  5364  			},
  5365  
  5366  			abort: function() {
  5367  				var self = this;
  5368  
  5369  				self.state = Transporter.IDLE;
  5370  				if (_runtime) {
  5371  					_runtime.exec.call(self, 'Transporter', 'clear');
  5372  					self.trigger("TransportingAborted");
  5373  				}
  5374  
  5375  				_reset.call(self);
  5376  			},
  5377  
  5378  
  5379  			destroy: function() {
  5380  				this.unbindAll();
  5381  				_runtime = null;
  5382  				this.disconnectRuntime();
  5383  				_reset.call(this);
  5384  			}
  5385  		});
  5386  
  5387  		function _reset() {
  5388  			_size = _pos = 0;
  5389  			_data = this.result = null;
  5390  		}
  5391  
  5392  		function _run(type, runtime) {
  5393  			var self = this;
  5394  
  5395  			_runtime = runtime;
  5396  
  5397  			//self.unbind("RuntimeInit");
  5398  
  5399  			self.bind("TransportingProgress", function(e) {
  5400  				_pos = e.loaded;
  5401  
  5402  				if (_pos < _size && Basic.inArray(self.state, [Transporter.IDLE, Transporter.DONE]) === -1) {
  5403  					_transport.call(self);
  5404  				}
  5405  			}, 999);
  5406  
  5407  			self.bind("TransportingComplete", function() {
  5408  				_pos = _size;
  5409  				self.state = Transporter.DONE;
  5410  				_data = null; // clean a bit
  5411  				self.result = _runtime.exec.call(self, 'Transporter', 'getAsBlob', type || '');
  5412  			}, 999);
  5413  
  5414  			self.state = Transporter.BUSY;
  5415  			self.trigger("TransportingStarted");
  5416  			_transport.call(self);
  5417  		}
  5418  
  5419  		function _transport() {
  5420  			var self = this,
  5421  				chunk,
  5422  				bytesLeft = _size - _pos;
  5423  
  5424  			if (_chunk_size > bytesLeft) {
  5425  				_chunk_size = bytesLeft;
  5426  			}
  5427  
  5428  			chunk = Encode.btoa(_data.substr(_pos, _chunk_size));
  5429  			_runtime.exec.call(self, 'Transporter', 'receive', chunk, _size);
  5430  		}
  5431  	}
  5432  
  5433  	Transporter.IDLE = 0;
  5434  	Transporter.BUSY = 1;
  5435  	Transporter.DONE = 2;
  5436  
  5437  	Transporter.prototype = EventTarget.instance;
  5438  
  5439  	return Transporter;
  5440  });
  5441  
  5442  // Included from: src/javascript/image/Image.js
  5443  
  5444  /**
  5445   * Image.js
  5446   *
  5447   * Copyright 2013, Moxiecode Systems AB
  5448   * Released under GPL License.
  5449   *
  5450   * License: http://www.plupload.com/license
  5451   * Contributing: http://www.plupload.com/contributing
  5452   */
  5453  
  5454  define("moxie/image/Image", [
  5455  	"moxie/core/utils/Basic",
  5456  	"moxie/core/utils/Dom",
  5457  	"moxie/core/Exceptions",
  5458  	"moxie/file/FileReaderSync",
  5459  	"moxie/xhr/XMLHttpRequest",
  5460  	"moxie/runtime/Runtime",
  5461  	"moxie/runtime/RuntimeClient",
  5462  	"moxie/runtime/Transporter",
  5463  	"moxie/core/utils/Env",
  5464  	"moxie/core/EventTarget",
  5465  	"moxie/file/Blob",
  5466  	"moxie/file/File",
  5467  	"moxie/core/utils/Encode"
  5468  ], function(Basic, Dom, x, FileReaderSync, XMLHttpRequest, Runtime, RuntimeClient, Transporter, Env, EventTarget, Blob, File, Encode) {
  5469  	/**
  5470  	Image preloading and manipulation utility. Additionally it provides access to image meta info (Exif, GPS) and raw binary data.
  5471  
  5472  	@class Image
  5473  	@constructor
  5474  	@extends EventTarget
  5475  	*/
  5476  	var dispatches = [
  5477  		'progress',
  5478  
  5479  		/**
  5480  		Dispatched when loading is complete.
  5481  
  5482  		@event load
  5483  		@param {Object} event
  5484  		*/
  5485  		'load',
  5486  
  5487  		'error',
  5488  
  5489  		/**
  5490  		Dispatched when resize operation is complete.
  5491  		
  5492  		@event resize
  5493  		@param {Object} event
  5494  		*/
  5495  		'resize',
  5496  
  5497  		/**
  5498  		Dispatched when visual representation of the image is successfully embedded
  5499  		into the corresponsing container.
  5500  
  5501  		@event embedded
  5502  		@param {Object} event
  5503  		*/
  5504  		'embedded'
  5505  	];
  5506  
  5507  	function Image() {
  5508  		RuntimeClient.call(this);
  5509  
  5510  		Basic.extend(this, {
  5511  			/**
  5512  			Unique id of the component
  5513  
  5514  			@property uid
  5515  			@type {String}
  5516  			*/
  5517  			uid: Basic.guid('uid_'),
  5518  
  5519  			/**
  5520  			Unique id of the connected runtime, if any.
  5521  
  5522  			@property ruid
  5523  			@type {String}
  5524  			*/
  5525  			ruid: null,
  5526  
  5527  			/**
  5528  			Name of the file, that was used to create an image, if available. If not equals to empty string.
  5529  
  5530  			@property name
  5531  			@type {String}
  5532  			@default ""
  5533  			*/
  5534  			name: "",
  5535  
  5536  			/**
  5537  			Size of the image in bytes. Actual value is set only after image is preloaded.
  5538  
  5539  			@property size
  5540  			@type {Number}
  5541  			@default 0
  5542  			*/
  5543  			size: 0,
  5544  
  5545  			/**
  5546  			Width of the image. Actual value is set only after image is preloaded.
  5547  
  5548  			@property width
  5549  			@type {Number}
  5550  			@default 0
  5551  			*/
  5552  			width: 0,
  5553  
  5554  			/**
  5555  			Height of the image. Actual value is set only after image is preloaded.
  5556  
  5557  			@property height
  5558  			@type {Number}
  5559  			@default 0
  5560  			*/
  5561  			height: 0,
  5562  
  5563  			/**
  5564  			Mime type of the image. Currently only image/jpeg and image/png are supported. Actual value is set only after image is preloaded.
  5565  
  5566  			@property type
  5567  			@type {String}
  5568  			@default ""
  5569  			*/
  5570  			type: "",
  5571  
  5572  			/**
  5573  			Holds meta info (Exif, GPS). Is populated only for image/jpeg. Actual value is set only after image is preloaded.
  5574  
  5575  			@property meta
  5576  			@type {Object}
  5577  			@default {}
  5578  			*/
  5579  			meta: {},
  5580  
  5581  			/**
  5582  			Alias for load method, that takes another mOxie.Image object as a source (see load).
  5583  
  5584  			@method clone
  5585  			@param {Image} src Source for the image
  5586  			@param {Boolean} [exact=false] Whether to activate in-depth clone mode
  5587  			*/
  5588  			clone: function() {
  5589  				this.load.apply(this, arguments);
  5590  			},
  5591  
  5592  			/**
  5593  			Loads image from various sources. Currently the source for new image can be: mOxie.Image, mOxie.Blob/mOxie.File, 
  5594  			native Blob/File, dataUrl or URL. Depending on the type of the source, arguments - differ. When source is URL, 
  5595  			Image will be downloaded from remote destination and loaded in memory.
  5596  
  5597  			@example
  5598  				var img = new mOxie.Image();
  5599  				img.onload = function() {
  5600  					var blob = img.getAsBlob();
  5601  					
  5602  					var formData = new mOxie.FormData();
  5603  					formData.append('file', blob);
  5604  
  5605  					var xhr = new mOxie.XMLHttpRequest();
  5606  					xhr.onload = function() {
  5607  						// upload complete
  5608  					};
  5609  					xhr.open('post', 'upload.php');
  5610  					xhr.send(formData);
  5611  				};
  5612  				img.load("http://www.moxiecode.com/images/mox-logo.jpg"); // notice file extension (.jpg)
  5613  			
  5614  
  5615  			@method load
  5616  			@param {Image|Blob|File|String} src Source for the image
  5617  			@param {Boolean|Object} [mixed]
  5618  			*/
  5619  			load: function() {
  5620  				// this is here because to bind properly we need an uid first, which is created above
  5621  				this.bind('Load Resize', function() {
  5622  					_updateInfo.call(this);
  5623  				}, 999);
  5624  
  5625  				this.convertEventPropsToHandlers(dispatches);
  5626  
  5627  				_load.apply(this, arguments);
  5628  			},
  5629  
  5630  			/**
  5631  			Downsizes the image to fit the specified width/height. If crop is supplied, image will be cropped to exact dimensions.
  5632  
  5633  			@method downsize
  5634  			@param {Number} width Resulting width
  5635  			@param {Number} [height=width] Resulting height (optional, if not supplied will default to width)
  5636  			@param {Boolean} [crop=false] Whether to crop the image to exact dimensions
  5637  			@param {Boolean} [preserveHeaders=true] Whether to preserve meta headers (on JPEGs after resize)
  5638  			*/
  5639  			downsize: function(width, height, crop, preserveHeaders) {
  5640  				try {
  5641  					if (!this.size) { // only preloaded image objects can be used as source
  5642  						throw new x.DOMException(x.DOMException.INVALID_STATE_ERR);
  5643  					}
  5644  
  5645  					// no way to reliably intercept the crash due to high resolution, so we simply avoid it
  5646  					if (this.width > Image.MAX_RESIZE_WIDTH || this.height > Image.MAX_RESIZE_HEIGHT) {
  5647  						throw new x.ImageError(x.ImageError.MAX_RESOLUTION_ERR);
  5648  					}
  5649  
  5650  					if (!width && !height || Basic.typeOf(crop) === 'undefined') {
  5651  						crop = false;
  5652  					}
  5653  
  5654  					width = width || this.width;
  5655  					height = height || this.height;
  5656  
  5657  					preserveHeaders = (Basic.typeOf(preserveHeaders) === 'undefined' ? true : !!preserveHeaders);
  5658  
  5659  					this.getRuntime().exec.call(this, 'Image', 'downsize', width, height, crop, preserveHeaders);
  5660  				} catch(ex) {
  5661  					// for now simply trigger error event
  5662  					this.trigger('error', ex);
  5663  				}
  5664  			},
  5665  
  5666  			/**
  5667  			Alias for downsize(width, height, true). (see downsize)
  5668  			
  5669  			@method crop
  5670  			@param {Number} width Resulting width
  5671  			@param {Number} [height=width] Resulting height (optional, if not supplied will default to width)
  5672  			@param {Boolean} [preserveHeaders=true] Whether to preserve meta headers (on JPEGs after resize)
  5673  			*/
  5674  			crop: function(width, height, preserveHeaders) {
  5675  				this.downsize(width, height, true, preserveHeaders);
  5676  			},
  5677  
  5678  			getAsCanvas: function() {
  5679  				if (!Env.can('create_canvas')) {
  5680  					throw new x.RuntimeError(x.RuntimeError.NOT_SUPPORTED_ERR);
  5681  				}
  5682  
  5683  				var runtime = this.connectRuntime(this.ruid);
  5684  				return runtime.exec.call(this, 'Image', 'getAsCanvas');
  5685  			},
  5686  
  5687  			/**
  5688  			Retrieves image in it's current state as mOxie.Blob object. Cannot be run on empty or image in progress (throws
  5689  			DOMException.INVALID_STATE_ERR).
  5690  
  5691  			@method getAsBlob
  5692  			@param {String} [type="image/jpeg"] Mime type of resulting blob. Can either be image/jpeg or image/png
  5693  			@param {Number} [quality=90] Applicable only together with mime type image/jpeg
  5694  			@return {Blob} Image as Blob
  5695  			*/
  5696  			getAsBlob: function(type, quality) {
  5697  				if (!this.size) {
  5698  					throw new x.DOMException(x.DOMException.INVALID_STATE_ERR);
  5699  				}
  5700  
  5701  				if (!type) {
  5702  					type = 'image/jpeg';
  5703  				}
  5704  
  5705  				if (type === 'image/jpeg' && !quality) {
  5706  					quality = 90;
  5707  				}
  5708  
  5709  				return this.getRuntime().exec.call(this, 'Image', 'getAsBlob', type, quality);
  5710  			},
  5711  
  5712  			/**
  5713  			Retrieves image in it's current state as dataURL string. Cannot be run on empty or image in progress (throws
  5714  			DOMException.INVALID_STATE_ERR).
  5715  
  5716  			@method getAsDataURL
  5717  			@param {String} [type="image/jpeg"] Mime type of resulting blob. Can either be image/jpeg or image/png
  5718  			@param {Number} [quality=90] Applicable only together with mime type image/jpeg
  5719  			@return {String} Image as dataURL string
  5720  			*/
  5721  			getAsDataURL: function(type, quality) {
  5722  				if (!this.size) {
  5723  					throw new x.DOMException(x.DOMException.INVALID_STATE_ERR);
  5724  				}
  5725  				return this.getRuntime().exec.call(this, 'Image', 'getAsDataURL', type, quality);
  5726  			},
  5727  
  5728  			/**
  5729  			Retrieves image in it's current state as binary string. Cannot be run on empty or image in progress (throws
  5730  			DOMException.INVALID_STATE_ERR).
  5731  
  5732  			@method getAsBinaryString
  5733  			@param {String} [type="image/jpeg"] Mime type of resulting blob. Can either be image/jpeg or image/png
  5734  			@param {Number} [quality=90] Applicable only together with mime type image/jpeg
  5735  			@return {String} Image as binary string
  5736  			*/
  5737  			getAsBinaryString: function(type, quality) {
  5738  				var dataUrl = this.getAsDataURL(type, quality);
  5739  				return Encode.atob(dataUrl.substring(dataUrl.indexOf('base64,') + 7));
  5740  			},
  5741  
  5742  			/**
  5743  			Embeds the image, or better to say, it's visual representation into the specified node. Depending on the runtime
  5744  			in use, might be a canvas, or image (actual ) element or shim object (Flash or SilverLight - very rare, used for
  5745  			legacy browsers that do not have canvas or proper dataURI support).
  5746  
  5747  			@method embed
  5748  			@param {DOMElement} el DOM element to insert the image object into
  5749  			@param {Object} options Set of key/value pairs controlling the mime type, dimensions and cropping factor of resulting
  5750  			representation
  5751  			*/
  5752  			embed: function(el) {
  5753  				var self = this
  5754  				, imgCopy
  5755  				, type, quality, crop
  5756  				, options = arguments[1] || {}
  5757  				, width = this.width
  5758  				, height = this.height
  5759  				, runtime // this has to be outside of all the closures to contain proper runtime
  5760  				;
  5761  
  5762  				function onResize() {
  5763  					// if possible, embed a canvas element directly
  5764  					if (Env.can('create_canvas')) {
  5765  						var canvas = imgCopy.getAsCanvas();
  5766  						if (canvas) {
  5767  							el.appendChild(canvas);
  5768  							canvas = null;
  5769  							imgCopy.destroy();
  5770  							self.trigger('embedded');
  5771  							return;
  5772  						}
  5773  					}
  5774  
  5775  					var dataUrl = imgCopy.getAsDataURL(type, quality);
  5776  					if (!dataUrl) {
  5777  						throw new x.ImageError(x.ImageError.WRONG_FORMAT);
  5778  					}
  5779  
  5780  					if (Env.can('use_data_uri_of', dataUrl.length)) {
  5781  						el.innerHTML = '<img src="' + dataUrl + '" width="' + imgCopy.width + '" height="' + imgCopy.height + '" />';
  5782  						imgCopy.destroy();
  5783  						self.trigger('embedded');
  5784  					} else {
  5785  						var tr = new Transporter();
  5786  
  5787  						tr.bind("TransportingComplete", function() {
  5788  							runtime = self.connectRuntime(this.result.ruid);
  5789  
  5790  							self.bind("Embedded", function() {
  5791  								// position and size properly
  5792  								Basic.extend(runtime.getShimContainer().style, {
  5793  									//position: 'relative',
  5794  									top: '0px',
  5795  									left: '0px',
  5796  									width: imgCopy.width + 'px',
  5797  									height: imgCopy.height + 'px'
  5798  								});
  5799  
  5800  								// some shims (Flash/SilverLight) reinitialize, if parent element is hidden, reordered or it's
  5801  								// position type changes (in Gecko), but since we basically need this only in IEs 6/7 and
  5802  								// sometimes 8 and they do not have this problem, we can comment this for now
  5803  								/*tr.bind("RuntimeInit", function(e, runtime) {
  5804  									tr.destroy();
  5805  									runtime.destroy();
  5806  									onResize.call(self); // re-feed our image data
  5807  								});*/
  5808  
  5809  								runtime = null;
  5810  							}, 999);
  5811  
  5812  							runtime.exec.call(self, "ImageView", "display", this.result.uid, width, height);
  5813  							imgCopy.destroy();
  5814  						});
  5815  
  5816  						tr.transport(Encode.atob(dataUrl.substring(dataUrl.indexOf('base64,') + 7)), type, Basic.extend({}, options, {
  5817  							required_caps: {
  5818  								display_media: true
  5819  							},
  5820  							runtime_order: 'flash,silverlight',
  5821  							container: el
  5822  						}));
  5823  					}
  5824  				}
  5825  
  5826  				try {
  5827  					if (!(el = Dom.get(el))) {
  5828  						throw new x.DOMException(x.DOMException.INVALID_NODE_TYPE_ERR);
  5829  					}
  5830  
  5831  					if (!this.size) { // only preloaded image objects can be used as source
  5832  						throw new x.DOMException(x.DOMException.INVALID_STATE_ERR);
  5833  					}
  5834  
  5835  					if (this.width > Image.MAX_RESIZE_WIDTH || this.height > Image.MAX_RESIZE_HEIGHT) {
  5836  						throw new x.ImageError(x.ImageError.MAX_RESOLUTION_ERR);
  5837  					}
  5838  
  5839  					type = options.type || this.type || 'image/jpeg';
  5840  					quality = options.quality || 90;
  5841  					crop = Basic.typeOf(options.crop) !== 'undefined' ? options.crop : false;
  5842  
  5843  					// figure out dimensions for the thumb
  5844  					if (options.width) {
  5845  						width = options.width;
  5846  						height = options.height || width;
  5847  					} else {
  5848  						// if container element has > 0 dimensions, take them
  5849  						var dimensions = Dom.getSize(el);
  5850  						if (dimensions.w && dimensions.h) { // both should be > 0
  5851  							width = dimensions.w;
  5852  							height = dimensions.h;
  5853  						}
  5854  					}
  5855  
  5856  					imgCopy = new Image();
  5857  
  5858  					imgCopy.bind("Resize", function() {
  5859  						onResize.call(self);
  5860  					});
  5861  
  5862  					imgCopy.bind("Load", function() {
  5863  						imgCopy.downsize(width, height, crop, false);
  5864  					});
  5865  
  5866  					imgCopy.clone(this, false);
  5867  
  5868  					return imgCopy;
  5869  				} catch(ex) {
  5870  					// for now simply trigger error event
  5871  					this.trigger('error', ex);
  5872  				}
  5873  			},
  5874  
  5875  			/**
  5876  			Properly destroys the image and frees resources in use. If any. Recommended way to dispose mOxie.Image object.
  5877  
  5878  			@method destroy
  5879  			*/
  5880  			destroy: function() {
  5881  				if (this.ruid) {
  5882  					this.getRuntime().exec.call(this, 'Image', 'destroy');
  5883  					this.disconnectRuntime();
  5884  				}
  5885  				this.unbindAll();
  5886  			}
  5887  		});
  5888  
  5889  
  5890  		function _updateInfo(info) {
  5891  			if (!info) {
  5892  				info = this.getRuntime().exec.call(this, 'Image', 'getInfo');
  5893  			}
  5894  
  5895  			this.size = info.size;
  5896  			this.width = info.width;
  5897  			this.height = info.height;
  5898  			this.type = info.type;
  5899  			this.meta = info.meta;
  5900  
  5901  			// update file name, only if empty
  5902  			if (this.name === '') {
  5903  				this.name = info.name;
  5904  			}
  5905  		}
  5906  		
  5907  
  5908  		function _load(src) {
  5909  			var srcType = Basic.typeOf(src);
  5910  
  5911  			try {
  5912  				// if source is Image
  5913  				if (src instanceof Image) {
  5914  					if (!src.size) { // only preloaded image objects can be used as source
  5915  						throw new x.DOMException(x.DOMException.INVALID_STATE_ERR);
  5916  					}
  5917  					_loadFromImage.apply(this, arguments);
  5918  				}
  5919  				// if source is o.Blob/o.File
  5920  				else if (src instanceof Blob) {
  5921  					if (!~Basic.inArray(src.type, ['image/jpeg', 'image/png'])) {
  5922  						throw new x.ImageError(x.ImageError.WRONG_FORMAT);
  5923  					}
  5924  					_loadFromBlob.apply(this, arguments);
  5925  				}
  5926  				// if native blob/file
  5927  				else if (Basic.inArray(srcType, ['blob', 'file']) !== -1) {
  5928  					_load.call(this, new File(null, src), arguments[1]);
  5929  				}
  5930  				// if String
  5931  				else if (srcType === 'string') {
  5932  					// if dataUrl String
  5933  					if (/^data:[^;]*;base64,/.test(src)) {
  5934  						_load.call(this, new Blob(null, { data: src }), arguments[1]);
  5935  					}
  5936  					// else assume Url, either relative or absolute
  5937  					else {
  5938  						_loadFromUrl.apply(this, arguments);
  5939  					}
  5940  				}
  5941  				// if source seems to be an img node
  5942  				else if (srcType === 'node' && src.nodeName.toLowerCase() === 'img') {
  5943  					_load.call(this, src.src, arguments[1]);
  5944  				}
  5945  				else {
  5946  					throw new x.DOMException(x.DOMException.TYPE_MISMATCH_ERR);
  5947  				}
  5948  			} catch(ex) {
  5949  				// for now simply trigger error event
  5950  				this.trigger('error', ex);
  5951  			}
  5952  		}
  5953  
  5954  
  5955  		function _loadFromImage(img, exact) {
  5956  			var runtime = this.connectRuntime(img.ruid);
  5957  			this.ruid = runtime.uid;
  5958  			runtime.exec.call(this, 'Image', 'loadFromImage', img, (Basic.typeOf(exact) === 'undefined' ? true : exact));
  5959  		}
  5960  
  5961  
  5962  		function _loadFromBlob(blob, options) {
  5963  			var self = this;
  5964  
  5965  			self.name = blob.name || '';
  5966  
  5967  			function exec(runtime) {
  5968  				self.ruid = runtime.uid;
  5969  				runtime.exec.call(self, 'Image', 'loadFromBlob', blob);
  5970  			}
  5971  
  5972  			if (blob.isDetached()) {
  5973  				this.bind('RuntimeInit', function(e, runtime) {
  5974  					exec(runtime);
  5975  				});
  5976  
  5977  				// convert to object representation
  5978  				if (options && typeof(options.required_caps) === 'string') {
  5979  					options.required_caps = Runtime.parseCaps(options.required_caps);
  5980  				}
  5981  
  5982  				this.connectRuntime(Basic.extend({
  5983  					required_caps: {
  5984  						access_image_binary: true,
  5985  						resize_image: true
  5986  					}
  5987  				}, options));
  5988  			} else {
  5989  				exec(this.connectRuntime(blob.ruid));
  5990  			}
  5991  		}
  5992  
  5993  
  5994  		function _loadFromUrl(url, options) {
  5995  			var self = this, xhr;
  5996  
  5997  			xhr = new XMLHttpRequest();
  5998  
  5999  			xhr.open('get', url);
  6000  			xhr.responseType = 'blob';
  6001  
  6002  			xhr.onprogress = function(e) {
  6003  				self.trigger(e);
  6004  			};
  6005  
  6006  			xhr.onload = function() {
  6007  				_loadFromBlob.call(self, xhr.response, true);
  6008  			};
  6009  
  6010  			xhr.onerror = function(e) {
  6011  				self.trigger(e);
  6012  			};
  6013  
  6014  			xhr.onloadend = function() {
  6015  				xhr.destroy();
  6016  			};
  6017  
  6018  			xhr.bind('RuntimeError', function(e, err) {
  6019  				self.trigger('RuntimeError', err);
  6020  			});
  6021  
  6022  			xhr.send(null, options);
  6023  		}
  6024  	}
  6025  
  6026  	// virtual world will crash on you if image has a resolution higher than this:
  6027  	Image.MAX_RESIZE_WIDTH = 6500;
  6028  	Image.MAX_RESIZE_HEIGHT = 6500; 
  6029  
  6030  	Image.prototype = EventTarget.instance;
  6031  
  6032  	return Image;
  6033  });
  6034  
  6035  // Included from: src/javascript/runtime/html5/Runtime.js
  6036  
  6037  /**
  6038   * Runtime.js
  6039   *
  6040   * Copyright 2013, Moxiecode Systems AB
  6041   * Released under GPL License.
  6042   *
  6043   * License: http://www.plupload.com/license
  6044   * Contributing: http://www.plupload.com/contributing
  6045   */
  6046  
  6047  /*global File:true */
  6048  
  6049  /**
  6050  Defines constructor for HTML5 runtime.
  6051  
  6052  @class moxie/runtime/html5/Runtime
  6053  @private
  6054  */
  6055  define("moxie/runtime/html5/Runtime", [
  6056  	"moxie/core/utils/Basic",
  6057  	"moxie/core/Exceptions",
  6058  	"moxie/runtime/Runtime",
  6059  	"moxie/core/utils/Env"
  6060  ], function(Basic, x, Runtime, Env) {
  6061  	
  6062  	var type = "html5", extensions = {};
  6063  	
  6064  	function Html5Runtime(options) {
  6065  		var I = this
  6066  		, Test = Runtime.capTest
  6067  		, True = Runtime.capTrue
  6068  		;
  6069  
  6070  		var caps = Basic.extend({
  6071  				access_binary: Test(window.FileReader || window.File && window.File.getAsDataURL),
  6072  				access_image_binary: function() {
  6073  					return I.can('access_binary') && !!extensions.Image;
  6074  				},
  6075  				display_media: Test(Env.can('create_canvas') || Env.can('use_data_uri_over32kb')),
  6076  				do_cors: Test(window.XMLHttpRequest && 'withCredentials' in new XMLHttpRequest()),
  6077  				drag_and_drop: Test(function() {
  6078  					// this comes directly from Modernizr: http://www.modernizr.com/
  6079  					var div = document.createElement('div');
  6080  					// IE has support for drag and drop since version 5, but doesn't support dropping files from desktop
  6081  					return (('draggable' in div) || ('ondragstart' in div && 'ondrop' in div)) && (Env.browser !== 'IE' || Env.version > 9);
  6082  				}()),
  6083  				filter_by_extension: Test(function() { // if you know how to feature-detect this, please suggest
  6084  					return (Env.browser === 'Chrome' && Env.version >= 28) || (Env.browser === 'IE' && Env.version >= 10);
  6085  				}()),
  6086  				return_response_headers: True,
  6087  				return_response_type: function(responseType) {
  6088  					if (responseType === 'json' && !!window.JSON) { // we can fake this one even if it's not supported
  6089  						return true;
  6090  					} 
  6091  					return Env.can('return_response_type', responseType);
  6092  				},
  6093  				return_status_code: True,
  6094  				report_upload_progress: Test(window.XMLHttpRequest && new XMLHttpRequest().upload),
  6095  				resize_image: function() {
  6096  					return I.can('access_binary') && Env.can('create_canvas');
  6097  				},
  6098  				select_file: function() {
  6099  					return Env.can('use_fileinput') && window.File;
  6100  				},
  6101  				select_folder: function() {
  6102  					return I.can('select_file') && Env.browser === 'Chrome' && Env.version >= 21;
  6103  				},
  6104  				select_multiple: function() {
  6105  					// it is buggy on Safari Windows and iOS
  6106  					return I.can('select_file') && 
  6107  						!(Env.browser === 'Safari' && Env.os === 'Windows') && 
  6108  						!(Env.os === 'iOS' && Env.verComp(Env.osVersion, "7.0.4", '<'));
  6109  				},
  6110  				send_binary_string: Test(window.XMLHttpRequest && (new XMLHttpRequest().sendAsBinary || (window.Uint8Array && window.ArrayBuffer))),
  6111  				send_custom_headers: Test(window.XMLHttpRequest),
  6112  				send_multipart: function() {
  6113  					return !!(window.XMLHttpRequest && new XMLHttpRequest().upload && window.FormData) || I.can('send_binary_string');
  6114  				},
  6115  				slice_blob: Test(window.File && (File.prototype.mozSlice || File.prototype.webkitSlice || File.prototype.slice)),
  6116  				stream_upload: function(){
  6117  					return I.can('slice_blob') && I.can('send_multipart');
  6118  				},
  6119  				summon_file_dialog: Test(function() { // yeah... some dirty sniffing here...
  6120  					return (Env.browser === 'Firefox' && Env.version >= 4) ||
  6121  						(Env.browser === 'Opera' && Env.version >= 12) ||
  6122  						(Env.browser === 'IE' && Env.version >= 10) ||
  6123  						!!~Basic.inArray(Env.browser, ['Chrome', 'Safari']);
  6124  				}()),
  6125  				upload_filesize: True
  6126  			}, 
  6127  			arguments[2]
  6128  		);
  6129  
  6130  		Runtime.call(this, options, (arguments[1] || type), caps);
  6131  
  6132  
  6133  		Basic.extend(this, {
  6134  
  6135  			init : function() {
  6136  				this.trigger("Init");
  6137  			},
  6138  
  6139  			destroy: (function(destroy) { // extend default destroy method
  6140  				return function() {
  6141  					destroy.call(I);
  6142  					destroy = I = null;
  6143  				};
  6144  			}(this.destroy))
  6145  		});
  6146  
  6147  		Basic.extend(this.getShim(), extensions);
  6148  	}
  6149  
  6150  	Runtime.addConstructor(type, Html5Runtime);
  6151  
  6152  	return extensions;
  6153  });
  6154  
  6155  // Included from: src/javascript/runtime/html5/file/Blob.js
  6156  
  6157  /**
  6158   * Blob.js
  6159   *
  6160   * Copyright 2013, Moxiecode Systems AB
  6161   * Released under GPL License.
  6162   *
  6163   * License: http://www.plupload.com/license
  6164   * Contributing: http://www.plupload.com/contributing
  6165   */
  6166  
  6167  /**
  6168  @class moxie/runtime/html5/file/Blob
  6169  @private
  6170  */
  6171  define("moxie/runtime/html5/file/Blob", [
  6172  	"moxie/runtime/html5/Runtime",
  6173  	"moxie/file/Blob"
  6174  ], function(extensions, Blob) {
  6175  
  6176  	function HTML5Blob() {
  6177  		function w3cBlobSlice(blob, start, end) {
  6178  			var blobSlice;
  6179  
  6180  			if (window.File.prototype.slice) {
  6181  				try {
  6182  					blob.slice();	// depricated version will throw WRONG_ARGUMENTS_ERR exception
  6183  					return blob.slice(start, end);
  6184  				} catch (e) {
  6185  					// depricated slice method
  6186  					return blob.slice(start, end - start);
  6187  				}
  6188  			// slice method got prefixed: https://bugzilla.mozilla.org/show_bug.cgi?id=649672
  6189  			} else if ((blobSlice = window.File.prototype.webkitSlice || window.File.prototype.mozSlice)) {
  6190  				return blobSlice.call(blob, start, end);
  6191  			} else {
  6192  				return null; // or throw some exception
  6193  			}
  6194  		}
  6195  
  6196  		this.slice = function() {
  6197  			return new Blob(this.getRuntime().uid, w3cBlobSlice.apply(this, arguments));
  6198  		};
  6199  	}
  6200  
  6201  	return (extensions.Blob = HTML5Blob);
  6202  });
  6203  
  6204  // Included from: src/javascript/core/utils/Events.js
  6205  
  6206  /**
  6207   * Events.js
  6208   *
  6209   * Copyright 2013, Moxiecode Systems AB
  6210   * Released under GPL License.
  6211   *
  6212   * License: http://www.plupload.com/license
  6213   * Contributing: http://www.plupload.com/contributing
  6214   */
  6215  
  6216  define('moxie/core/utils/Events', [
  6217  	'moxie/core/utils/Basic'
  6218  ], function(Basic) {
  6219  	var eventhash = {}, uid = 'moxie_' + Basic.guid();
  6220  	
  6221  	// IE W3C like event funcs
  6222  	function preventDefault() {
  6223  		this.returnValue = false;
  6224  	}
  6225  
  6226  	function stopPropagation() {
  6227  		this.cancelBubble = true;
  6228  	}
  6229  
  6230  	/**
  6231  	Adds an event handler to the specified object and store reference to the handler
  6232  	in objects internal Plupload registry (@see removeEvent).
  6233  	
  6234  	@method addEvent
  6235  	@for Utils
  6236  	@static
  6237  	@param {Object} obj DOM element like object to add handler to.
  6238  	@param {String} name Name to add event listener to.
  6239  	@param {Function} callback Function to call when event occurs.
  6240  	@param {String} [key] that might be used to add specifity to the event record.
  6241  	*/
  6242  	var addEvent = function(obj, name, callback, key) {
  6243  		var func, events;
  6244  					
  6245  		name = name.toLowerCase();
  6246  
  6247  		// Add event listener
  6248  		if (obj.addEventListener) {
  6249  			func = callback;
  6250  			
  6251  			obj.addEventListener(name, func, false);
  6252  		} else if (obj.attachEvent) {
  6253  			func = function() {
  6254  				var evt = window.event;
  6255  
  6256  				if (!evt.target) {
  6257  					evt.target = evt.srcElement;
  6258  				}
  6259  
  6260  				evt.preventDefault = preventDefault;
  6261  				evt.stopPropagation = stopPropagation;
  6262  
  6263  				callback(evt);
  6264  			};
  6265  
  6266  			obj.attachEvent('on' + name, func);
  6267  		}
  6268  		
  6269  		// Log event handler to objects internal mOxie registry
  6270  		if (!obj[uid]) {
  6271  			obj[uid] = Basic.guid();
  6272  		}
  6273  		
  6274  		if (!eventhash.hasOwnProperty(obj[uid])) {
  6275  			eventhash[obj[uid]] = {};
  6276  		}
  6277  		
  6278  		events = eventhash[obj[uid]];
  6279  		
  6280  		if (!events.hasOwnProperty(name)) {
  6281  			events[name] = [];
  6282  		}
  6283  				
  6284  		events[name].push({
  6285  			func: func,
  6286  			orig: callback, // store original callback for IE
  6287  			key: key
  6288  		});
  6289  	};
  6290  	
  6291  	
  6292  	/**
  6293  	Remove event handler from the specified object. If third argument (callback)
  6294  	is not specified remove all events with the specified name.
  6295  	
  6296  	@method removeEvent
  6297  	@static
  6298  	@param {Object} obj DOM element to remove event listener(s) from.
  6299  	@param {String} name Name of event listener to remove.
  6300  	@param {Function|String} [callback] might be a callback or unique key to match.
  6301  	*/
  6302  	var removeEvent = function(obj, name, callback) {
  6303  		var type, undef;
  6304  		
  6305  		name = name.toLowerCase();
  6306  		
  6307  		if (obj[uid] && eventhash[obj[uid]] && eventhash[obj[uid]][name]) {
  6308  			type = eventhash[obj[uid]][name];
  6309  		} else {
  6310  			return;
  6311  		}
  6312  			
  6313  		for (var i = type.length - 1; i >= 0; i--) {
  6314  			// undefined or not, key should match
  6315  			if (type[i].orig === callback || type[i].key === callback) {
  6316  				if (obj.removeEventListener) {
  6317  					obj.removeEventListener(name, type[i].func, false);
  6318  				} else if (obj.detachEvent) {
  6319  					obj.detachEvent('on'+name, type[i].func);
  6320  				}
  6321  				
  6322  				type[i].orig = null;
  6323  				type[i].func = null;
  6324  				type.splice(i, 1);
  6325  				
  6326  				// If callback was passed we are done here, otherwise proceed
  6327  				if (callback !== undef) {
  6328  					break;
  6329  				}
  6330  			}
  6331  		}
  6332  		
  6333  		// If event array got empty, remove it
  6334  		if (!type.length) {
  6335  			delete eventhash[obj[uid]][name];
  6336  		}
  6337  		
  6338  		// If mOxie registry has become empty, remove it
  6339  		if (Basic.isEmptyObj(eventhash[obj[uid]])) {
  6340  			delete eventhash[obj[uid]];
  6341  			
  6342  			// IE doesn't let you remove DOM object property with - delete
  6343  			try {
  6344  				delete obj[uid];
  6345  			} catch(e) {
  6346  				obj[uid] = undef;
  6347  			}
  6348  		}
  6349  	};
  6350  	
  6351  	
  6352  	/**
  6353  	Remove all kind of events from the specified object
  6354  	
  6355  	@method removeAllEvents
  6356  	@static
  6357  	@param {Object} obj DOM element to remove event listeners from.
  6358  	@param {String} [key] unique key to match, when removing events.
  6359  	*/
  6360  	var removeAllEvents = function(obj, key) {		
  6361  		if (!obj || !obj[uid]) {
  6362  			return;
  6363  		}
  6364  		
  6365  		Basic.each(eventhash[obj[uid]], function(events, name) {
  6366  			removeEvent(obj, name, key);
  6367  		});
  6368  	};
  6369  
  6370  	return {
  6371  		addEvent: addEvent,
  6372  		removeEvent: removeEvent,
  6373  		removeAllEvents: removeAllEvents
  6374  	};
  6375  });
  6376  
  6377  // Included from: src/javascript/runtime/html5/file/FileInput.js
  6378  
  6379  /**
  6380   * FileInput.js
  6381   *
  6382   * Copyright 2013, Moxiecode Systems AB
  6383   * Released under GPL License.
  6384   *
  6385   * License: http://www.plupload.com/license
  6386   * Contributing: http://www.plupload.com/contributing
  6387   */
  6388  
  6389  /**
  6390  @class moxie/runtime/html5/file/FileInput
  6391  @private
  6392  */
  6393  define("moxie/runtime/html5/file/FileInput", [
  6394  	"moxie/runtime/html5/Runtime",
  6395  	"moxie/core/utils/Basic",
  6396  	"moxie/core/utils/Dom",
  6397  	"moxie/core/utils/Events",
  6398  	"moxie/core/utils/Mime",
  6399  	"moxie/core/utils/Env"
  6400  ], function(extensions, Basic, Dom, Events, Mime, Env) {
  6401  	
  6402  	function FileInput() {
  6403  		var _files = [], _options;
  6404  
  6405  		Basic.extend(this, {
  6406  			init: function(options) {
  6407  				var comp = this, I = comp.getRuntime(), input, shimContainer, mimes, browseButton, zIndex, top;
  6408  
  6409  				_options = options;
  6410  				_files = [];
  6411  
  6412  				// figure out accept string
  6413  				mimes = _options.accept.mimes || Mime.extList2mimes(_options.accept, I.can('filter_by_extension'));
  6414  
  6415  				shimContainer = I.getShimContainer();
  6416  
  6417  				shimContainer.innerHTML = '<input id="' + I.uid +'" type="file" style="font-size:999px;opacity:0;"' +
  6418  					(_options.multiple && I.can('select_multiple') ? 'multiple' : '') + 
  6419  					(_options.directory && I.can('select_folder') ? 'webkitdirectory directory' : '') + // Chrome 11+
  6420  					(mimes ? ' accept="' + mimes.join(',') + '"' : '') + ' />';
  6421  
  6422  				input = Dom.get(I.uid);
  6423  
  6424  				// prepare file input to be placed underneath the browse_button element
  6425  				Basic.extend(input.style, {
  6426  					position: 'absolute',
  6427  					top: 0,
  6428  					left: 0,
  6429  					width: '100%',
  6430  					height: '100%'
  6431  				});
  6432  
  6433  
  6434  				browseButton = Dom.get(_options.browse_button);
  6435  
  6436  				// Route click event to the input[type=file] element for browsers that support such behavior
  6437  				if (I.can('summon_file_dialog')) {
  6438  					if (Dom.getStyle(browseButton, 'position') === 'static') {
  6439  						browseButton.style.position = 'relative';
  6440  					}
  6441  
  6442  					zIndex = parseInt(Dom.getStyle(browseButton, 'z-index'), 10) || 1;
  6443  
  6444  					browseButton.style.zIndex = zIndex;
  6445  					shimContainer.style.zIndex = zIndex - 1;
  6446  
  6447  					Events.addEvent(browseButton, 'click', function(e) {
  6448  						var input = Dom.get(I.uid);
  6449  						if (input && !input.disabled) { // for some reason FF (up to 8.0.1 so far) lets to click disabled input[type=file]
  6450  							input.click();
  6451  						}
  6452  						e.preventDefault();
  6453  					}, comp.uid);
  6454  				}
  6455  
  6456  				/* Since we have to place input[type=file] on top of the browse_button for some browsers,
  6457  				browse_button loses interactivity, so we restore it here */
  6458  				top = I.can('summon_file_dialog') ? browseButton : shimContainer;
  6459  
  6460  				Events.addEvent(top, 'mouseover', function() {
  6461  					comp.trigger('mouseenter');
  6462  				}, comp.uid);
  6463  
  6464  				Events.addEvent(top, 'mouseout', function() {
  6465  					comp.trigger('mouseleave');
  6466  				}, comp.uid);
  6467  
  6468  				Events.addEvent(top, 'mousedown', function() {
  6469  					comp.trigger('mousedown');
  6470  				}, comp.uid);
  6471  
  6472  				Events.addEvent(Dom.get(_options.container), 'mouseup', function() {
  6473  					comp.trigger('mouseup');
  6474  				}, comp.uid);
  6475  
  6476  
  6477  				input.onchange = function onChange() { // there should be only one handler for this
  6478  					_files = [];
  6479  
  6480  					if (_options.directory) {
  6481  						// folders are represented by dots, filter them out (Chrome 11+)
  6482  						Basic.each(this.files, function(file) {
  6483  							if (file.name !== ".") { // if it doesn't looks like a folder
  6484  								_files.push(file);
  6485  							}
  6486  						});
  6487  					} else {
  6488  						_files = [].slice.call(this.files);
  6489  					}
  6490  
  6491  					// clearing the value enables the user to select the same file again if they want to
  6492  					if (Env.browser !== 'IE') {
  6493  						this.value = '';
  6494  					} else {
  6495  						// in IE input[type="file"] is read-only so the only way to reset it is to re-insert it
  6496  						var clone = this.cloneNode(true);
  6497  						this.parentNode.replaceChild(clone, this);
  6498  						clone.onchange = onChange;
  6499  					}
  6500  					comp.trigger('change');
  6501  				};
  6502  
  6503  				// ready event is perfectly asynchronous
  6504  				comp.trigger({
  6505  					type: 'ready',
  6506  					async: true
  6507  				});
  6508  
  6509  				shimContainer = null;
  6510  			},
  6511  
  6512  			getFiles: function() {
  6513  				return _files;
  6514  			},
  6515  
  6516  			disable: function(state) {
  6517  				var I = this.getRuntime(), input;
  6518  
  6519  				if ((input = Dom.get(I.uid))) {
  6520  					input.disabled = !!state;
  6521  				}
  6522  			},
  6523  
  6524  			destroy: function() {
  6525  				var I = this.getRuntime()
  6526  				, shim = I.getShim()
  6527  				, shimContainer = I.getShimContainer()
  6528  				;
  6529  				
  6530  				Events.removeAllEvents(shimContainer, this.uid);
  6531  				Events.removeAllEvents(_options && Dom.get(_options.container), this.uid);
  6532  				Events.removeAllEvents(_options && Dom.get(_options.browse_button), this.uid);
  6533  				
  6534  				if (shimContainer) {
  6535  					shimContainer.innerHTML = '';
  6536  				}
  6537  
  6538  				shim.removeInstance(this.uid);
  6539  
  6540  				_files = _options = shimContainer = shim = null;
  6541  			}
  6542  		});
  6543  	}
  6544  
  6545  	return (extensions.FileInput = FileInput);
  6546  });
  6547  
  6548  // Included from: src/javascript/runtime/html5/file/FileDrop.js
  6549  
  6550  /**
  6551   * FileDrop.js
  6552   *
  6553   * Copyright 2013, Moxiecode Systems AB
  6554   * Released under GPL License.
  6555   *
  6556   * License: http://www.plupload.com/license
  6557   * Contributing: http://www.plupload.com/contributing
  6558   */
  6559  
  6560  /**
  6561  @class moxie/runtime/html5/file/FileDrop
  6562  @private
  6563  */
  6564  define("moxie/runtime/html5/file/FileDrop", [
  6565  	"moxie/runtime/html5/Runtime",
  6566  	"moxie/core/utils/Basic",
  6567  	"moxie/core/utils/Dom",
  6568  	"moxie/core/utils/Events",
  6569  	"moxie/core/utils/Mime"
  6570  ], function(extensions, Basic, Dom, Events, Mime) {
  6571  	
  6572  	function FileDrop() {
  6573  		var _files = [], _allowedExts = [], _options;
  6574  
  6575  		Basic.extend(this, {
  6576  			init: function(options) {
  6577  				var comp = this, dropZone;
  6578  
  6579  				_options = options;
  6580  				_allowedExts = _extractExts(_options.accept);
  6581  				dropZone = _options.container;
  6582  
  6583  				Events.addEvent(dropZone, 'dragover', function(e) {
  6584  					e.preventDefault();
  6585  					e.stopPropagation();
  6586  					e.dataTransfer.dropEffect = 'copy';
  6587  				}, comp.uid);
  6588  
  6589  				Events.addEvent(dropZone, 'drop', function(e) {
  6590  					e.preventDefault();
  6591  					e.stopPropagation();
  6592  
  6593  					_files = [];
  6594  
  6595  					// Chrome 21+ accepts folders via Drag'n'Drop
  6596  					if (e.dataTransfer.items && e.dataTransfer.items[0].webkitGetAsEntry) {
  6597  						_readItems(e.dataTransfer.items, function() {
  6598  							comp.trigger("drop");
  6599  						});
  6600  					} else {
  6601  						Basic.each(e.dataTransfer.files, function(file) {
  6602  							if (_isAcceptable(file)) {
  6603  								_files.push(file);
  6604  							}
  6605  						});
  6606  						comp.trigger("drop");
  6607  					}
  6608  				}, comp.uid);
  6609  
  6610  				Events.addEvent(dropZone, 'dragenter', function(e) {
  6611  					e.preventDefault();
  6612  					e.stopPropagation();
  6613  					comp.trigger("dragenter");
  6614  				}, comp.uid);
  6615  
  6616  				Events.addEvent(dropZone, 'dragleave', function(e) {
  6617  					e.preventDefault();
  6618  					e.stopPropagation();
  6619  					comp.trigger("dragleave");
  6620  				}, comp.uid);
  6621  			},
  6622  
  6623  			getFiles: function() {
  6624  				return _files;
  6625  			},
  6626  
  6627  			destroy: function() {
  6628  				Events.removeAllEvents(_options && Dom.get(_options.container), this.uid);
  6629  				_files = _allowedExts = _options = null;
  6630  			}
  6631  		});
  6632  
  6633  		
  6634  		function _extractExts(accept) {
  6635  			var exts = [];
  6636  			for (var i = 0; i < accept.length; i++) {
  6637  				[].push.apply(exts, accept[i].extensions.split(/\s*,\s*/));
  6638  			}
  6639  			return Basic.inArray('*', exts) === -1 ? exts : [];
  6640  		}
  6641  
  6642  
  6643  		function _isAcceptable(file) {
  6644  			var ext = Mime.getFileExtension(file.name);
  6645  			return !ext || !_allowedExts.length || Basic.inArray(ext, _allowedExts) !== -1;
  6646  		}
  6647  
  6648  
  6649  		function _readItems(items, cb) {
  6650  			var entries = [];
  6651  			Basic.each(items, function(item) {
  6652  				var entry = item.webkitGetAsEntry();
  6653  				// Address #998 (https://code.google.com/p/chromium/issues/detail?id=332579)
  6654  				if (entry) {
  6655  					// file() fails on OSX when the filename contains a special character (e.g. umlaut): see #61
  6656  					if (entry.isFile) {
  6657  						var file = item.getAsFile();
  6658  						if (_isAcceptable(file)) {
  6659  							_files.push(file);
  6660  						}
  6661  					} else {
  6662  						entries.push(entry);
  6663  					}
  6664  				}
  6665  			});
  6666  
  6667  			if (entries.length) {
  6668  				_readEntries(entries, cb);
  6669  			} else {
  6670  				cb();
  6671  			}
  6672  		}
  6673  
  6674  
  6675  		function _readEntries(entries, cb) {
  6676  			var queue = [];
  6677  			Basic.each(entries, function(entry) {
  6678  				queue.push(function(cbcb) {
  6679  					_readEntry(entry, cbcb);
  6680  				});
  6681  			});
  6682  			Basic.inSeries(queue, function() {
  6683  				cb();
  6684  			});
  6685  		}
  6686  
  6687  		function _readEntry(entry, cb) {
  6688  			if (entry.isFile) {
  6689  				entry.file(function(file) {
  6690  					if (_isAcceptable(file)) {
  6691  						_files.push(file);
  6692  					}
  6693  					cb();
  6694  				}, function() {
  6695  					// fire an error event maybe
  6696  					cb();
  6697  				});
  6698  			} else if (entry.isDirectory) {
  6699  				_readDirEntry(entry, cb);
  6700  			} else {
  6701  				cb(); // not file, not directory? what then?..
  6702  			}
  6703  		}
  6704  
  6705  		function _readDirEntry(dirEntry, cb) {
  6706  			var entries = [], dirReader = dirEntry.createReader();
  6707  
  6708  			// keep quering recursively till no more entries
  6709  			function getEntries(cbcb) {
  6710  				dirReader.readEntries(function(moreEntries) {
  6711  					if (moreEntries.length) {
  6712  						[].push.apply(entries, moreEntries);
  6713  						getEntries(cbcb);
  6714  					} else {
  6715  						cbcb();
  6716  					}
  6717  				}, cbcb);
  6718  			}
  6719  
  6720  			// ...and you thought FileReader was crazy...
  6721  			getEntries(function() {
  6722  				_readEntries(entries, cb);
  6723  			}); 
  6724  		}
  6725  	}
  6726  
  6727  	return (extensions.FileDrop = FileDrop);
  6728  });
  6729  
  6730  // Included from: src/javascript/runtime/html5/file/FileReader.js
  6731  
  6732  /**
  6733   * FileReader.js
  6734   *
  6735   * Copyright 2013, Moxiecode Systems AB
  6736   * Released under GPL License.
  6737   *
  6738   * License: http://www.plupload.com/license
  6739   * Contributing: http://www.plupload.com/contributing
  6740   */
  6741  
  6742  /**
  6743  @class moxie/runtime/html5/file/FileReader
  6744  @private
  6745  */
  6746  define("moxie/runtime/html5/file/FileReader", [
  6747  	"moxie/runtime/html5/Runtime",
  6748  	"moxie/core/utils/Encode",
  6749  	"moxie/core/utils/Basic"
  6750  ], function(extensions, Encode, Basic) {
  6751  	
  6752  	function FileReader() {
  6753  		var _fr, _convertToBinary = false;
  6754  
  6755  		Basic.extend(this, {
  6756  
  6757  			read: function(op, blob) {
  6758  				var target = this;
  6759  
  6760  				_fr = new window.FileReader();
  6761  
  6762  				_fr.addEventListener('progress', function(e) {
  6763  					target.trigger(e);
  6764  				});
  6765  
  6766  				_fr.addEventListener('load', function(e) {
  6767  					target.trigger(e);
  6768  				});
  6769  
  6770  				_fr.addEventListener('error', function(e) {
  6771  					target.trigger(e, _fr.error);
  6772  				});
  6773  
  6774  				_fr.addEventListener('loadend', function() {
  6775  					_fr = null;
  6776  				});
  6777  
  6778  				if (Basic.typeOf(_fr[op]) === 'function') {
  6779  					_convertToBinary = false;
  6780  					_fr[op](blob.getSource());
  6781  				} else if (op === 'readAsBinaryString') { // readAsBinaryString is depricated in general and never existed in IE10+
  6782  					_convertToBinary = true;
  6783  					_fr.readAsDataURL(blob.getSource());
  6784  				}
  6785  			},
  6786  
  6787  			getResult: function() {
  6788  				return _fr && _fr.result ? (_convertToBinary ? _toBinary(_fr.result) : _fr.result) : null;
  6789  			},
  6790  
  6791  			abort: function() {
  6792  				if (_fr) {
  6793  					_fr.abort();
  6794  				}
  6795  			},
  6796  
  6797  			destroy: function() {
  6798  				_fr = null;
  6799  			}
  6800  		});
  6801  
  6802  		function _toBinary(str) {
  6803  			return Encode.atob(str.substring(str.indexOf('base64,') + 7));
  6804  		}
  6805  	}
  6806  
  6807  	return (extensions.FileReader = FileReader);
  6808  });
  6809  
  6810  // Included from: src/javascript/runtime/html5/xhr/XMLHttpRequest.js
  6811  
  6812  /**
  6813   * XMLHttpRequest.js
  6814   *
  6815   * Copyright 2013, Moxiecode Systems AB
  6816   * Released under GPL License.
  6817   *
  6818   * License: http://www.plupload.com/license
  6819   * Contributing: http://www.plupload.com/contributing
  6820   */
  6821  
  6822  /*global ActiveXObject:true */
  6823  
  6824  /**
  6825  @class moxie/runtime/html5/xhr/XMLHttpRequest
  6826  @private
  6827  */
  6828  define("moxie/runtime/html5/xhr/XMLHttpRequest", [
  6829  	"moxie/runtime/html5/Runtime",
  6830  	"moxie/core/utils/Basic",
  6831  	"moxie/core/utils/Mime",
  6832  	"moxie/core/utils/Url",
  6833  	"moxie/file/File",
  6834  	"moxie/file/Blob",
  6835  	"moxie/xhr/FormData",
  6836  	"moxie/core/Exceptions",
  6837  	"moxie/core/utils/Env"
  6838  ], function(extensions, Basic, Mime, Url, File, Blob, FormData, x, Env) {
  6839  	
  6840  	function XMLHttpRequest() {
  6841  		var self = this
  6842  		, _xhr
  6843  		, _filename
  6844  		;
  6845  
  6846  		Basic.extend(this, {
  6847  			send: function(meta, data) {
  6848  				var target = this
  6849  				, isGecko2_5_6 = (Env.browser === 'Mozilla' && Env.version >= 4 && Env.version < 7)
  6850  				, isAndroidBrowser = Env.browser === 'Android Browser'
  6851  				, mustSendAsBinary = false
  6852  				;
  6853  
  6854  				// extract file name
  6855  				_filename = meta.url.replace(/^.+?\/([\w\-\.]+)$/, '$1').toLowerCase();
  6856  
  6857  				_xhr = _getNativeXHR();
  6858  				_xhr.open(meta.method, meta.url, meta.async, meta.user, meta.password);
  6859  
  6860  
  6861  				// prepare data to be sent
  6862  				if (data instanceof Blob) {
  6863  					if (data.isDetached()) {
  6864  						mustSendAsBinary = true;
  6865  					}
  6866  					data = data.getSource();
  6867  				} else if (data instanceof FormData) {
  6868  
  6869  					if (data.hasBlob()) {
  6870  						if (data.getBlob().isDetached()) {
  6871  							data = _prepareMultipart.call(target, data); // _xhr must be instantiated and be in OPENED state
  6872  							mustSendAsBinary = true;
  6873  						} else if ((isGecko2_5_6 || isAndroidBrowser) && Basic.typeOf(data.getBlob().getSource()) === 'blob' && window.FileReader) {
  6874  							// Gecko 2/5/6 can't send blob in FormData: https://bugzilla.mozilla.org/show_bug.cgi?id=649150
  6875  							// Android browsers (default one and Dolphin) seem to have the same issue, see: #613
  6876  							_preloadAndSend.call(target, meta, data);
  6877  							return; // _preloadAndSend will reinvoke send() with transmutated FormData =%D
  6878  						}	
  6879  					}
  6880  
  6881  					// transfer fields to real FormData
  6882  					if (data instanceof FormData) { // if still a FormData, e.g. not mangled by _prepareMultipart()
  6883  						var fd = new window.FormData();
  6884  						data.each(function(value, name) {
  6885  							if (value instanceof Blob) {
  6886  								fd.append(name, value.getSource());
  6887  							} else {
  6888  								fd.append(name, value);
  6889  							}
  6890  						});
  6891  						data = fd;
  6892  					}
  6893  				}
  6894  
  6895  
  6896  				// if XHR L2
  6897  				if (_xhr.upload) {
  6898  					if (meta.withCredentials) {
  6899  						_xhr.withCredentials = true;
  6900  					}
  6901  
  6902  					_xhr.addEventListener('load', function(e) {
  6903  						target.trigger(e);
  6904  					});
  6905  
  6906  					_xhr.addEventListener('error', function(e) {
  6907  						target.trigger(e);
  6908  					});
  6909  
  6910  					// additionally listen to progress events
  6911  					_xhr.addEventListener('progress', function(e) {
  6912  						target.trigger(e);
  6913  					});
  6914  
  6915  					_xhr.upload.addEventListener('progress', function(e) {
  6916  						target.trigger({
  6917  							type: 'UploadProgress',
  6918  							loaded: e.loaded,
  6919  							total: e.total
  6920  						});
  6921  					});
  6922  				// ... otherwise simulate XHR L2
  6923  				} else {
  6924  					_xhr.onreadystatechange = function onReadyStateChange() {
  6925  						
  6926  						// fake Level 2 events
  6927  						switch (_xhr.readyState) {
  6928  							
  6929  							case 1: // XMLHttpRequest.OPENED
  6930  								// readystatechanged is fired twice for OPENED state (in IE and Mozilla) - neu
  6931  								break;
  6932  							
  6933  							// looks like HEADERS_RECEIVED (state 2) is not reported in Opera (or it's old versions) - neu
  6934  							case 2: // XMLHttpRequest.HEADERS_RECEIVED
  6935  								break;
  6936  								
  6937  							case 3: // XMLHttpRequest.LOADING 
  6938  								// try to fire progress event for not XHR L2
  6939  								var total, loaded;
  6940  								
  6941  								try {
  6942  									if (Url.hasSameOrigin(meta.url)) { // Content-Length not accessible for cross-domain on some browsers
  6943  										total = _xhr.getResponseHeader('Content-Length') || 0; // old Safari throws an exception here
  6944  									}
  6945  
  6946  									if (_xhr.responseText) { // responseText was introduced in IE7
  6947  										loaded = _xhr.responseText.length;
  6948  									}
  6949  								} catch(ex) {
  6950  									total = loaded = 0;
  6951  								}
  6952  
  6953  								target.trigger({
  6954  									type: 'progress',
  6955  									lengthComputable: !!total,
  6956  									total: parseInt(total, 10),
  6957  									loaded: loaded
  6958  								});
  6959  								break;
  6960  								
  6961  							case 4: // XMLHttpRequest.DONE
  6962  								// release readystatechange handler (mostly for IE)
  6963  								_xhr.onreadystatechange = function() {};
  6964  
  6965  								// usually status 0 is returned when server is unreachable, but FF also fails to status 0 for 408 timeout
  6966  								if (_xhr.status === 0) {
  6967  									target.trigger('error');
  6968  								} else {
  6969  									target.trigger('load');
  6970  								}							
  6971  								break;
  6972  						}
  6973  					};
  6974  				}
  6975  				
  6976  
  6977  				// set request headers
  6978  				if (!Basic.isEmptyObj(meta.headers)) {
  6979  					Basic.each(meta.headers, function(value, header) {
  6980  						_xhr.setRequestHeader(header, value);
  6981  					});
  6982  				}
  6983  
  6984  				// request response type
  6985  				if ("" !== meta.responseType && 'responseType' in _xhr) {
  6986  					if ('json' === meta.responseType && !Env.can('return_response_type', 'json')) { // we can fake this one
  6987  						_xhr.responseType = 'text';
  6988  					} else {
  6989  						_xhr.responseType = meta.responseType;
  6990  					}
  6991  				}
  6992  
  6993  				// send ...
  6994  				if (!mustSendAsBinary) {
  6995  					_xhr.send(data);
  6996  				} else {
  6997  					if (_xhr.sendAsBinary) { // Gecko
  6998  						_xhr.sendAsBinary(data);
  6999  					} else { // other browsers having support for typed arrays
  7000  						(function() {
  7001  							// mimic Gecko's sendAsBinary
  7002  							var ui8a = new Uint8Array(data.length);
  7003  							for (var i = 0; i < data.length; i++) {
  7004  								ui8a[i] = (data.charCodeAt(i) & 0xff);
  7005  							}
  7006  							_xhr.send(ui8a.buffer);
  7007  						}());
  7008  					}
  7009  				}
  7010  
  7011  				target.trigger('loadstart');
  7012  			},
  7013  
  7014  			getStatus: function() {
  7015  				// according to W3C spec it should return 0 for readyState < 3, but instead it throws an exception
  7016  				try {
  7017  					if (_xhr) {
  7018  						return _xhr.status;
  7019  					}
  7020  				} catch(ex) {}
  7021  				return 0;
  7022  			},
  7023  
  7024  			getResponse: function(responseType) {
  7025  				var I = this.getRuntime();
  7026  
  7027  				try {
  7028  					switch (responseType) {
  7029  						case 'blob':
  7030  							var file = new File(I.uid, _xhr.response);
  7031  							
  7032  							// try to extract file name from content-disposition if possible (might be - not, if CORS for example)	
  7033  							var disposition = _xhr.getResponseHeader('Content-Disposition');
  7034  							if (disposition) {
  7035  								// extract filename from response header if available
  7036  								var match = disposition.match(/filename=([\'\"'])([^\1]+)\1/);
  7037  								if (match) {
  7038  									_filename = match[2];
  7039  								}
  7040  							}
  7041  							file.name = _filename;
  7042  
  7043  							// pre-webkit Opera doesn't set type property on the blob response
  7044  							if (!file.type) {
  7045  								file.type = Mime.getFileMime(_filename);
  7046  							}
  7047  							return file;
  7048  
  7049  						case 'json':
  7050  							if (!Env.can('return_response_type', 'json')) {
  7051  								return _xhr.status === 200 && !!window.JSON ? JSON.parse(_xhr.responseText) : null;
  7052  							}
  7053  							return _xhr.response;
  7054  
  7055  						case 'document':
  7056  							return _getDocument(_xhr);
  7057  
  7058  						default:
  7059  							return _xhr.responseText !== '' ? _xhr.responseText : null; // against the specs, but for consistency across the runtimes
  7060  					}
  7061  				} catch(ex) {
  7062  					return null;
  7063  				}				
  7064  			},
  7065  
  7066  			getAllResponseHeaders: function() {
  7067  				try {
  7068  					return _xhr.getAllResponseHeaders();
  7069  				} catch(ex) {}
  7070  				return '';
  7071  			},
  7072  
  7073  			abort: function() {
  7074  				if (_xhr) {
  7075  					_xhr.abort();
  7076  				}
  7077  			},
  7078  
  7079  			destroy: function() {
  7080  				self = _filename = null;
  7081  			}
  7082  		});
  7083  
  7084  
  7085  		// here we go... ugly fix for ugly bug
  7086  		function _preloadAndSend(meta, data) {
  7087  			var target = this, blob, fr;
  7088  				
  7089  			// get original blob
  7090  			blob = data.getBlob().getSource();
  7091  			
  7092  			// preload blob in memory to be sent as binary string
  7093  			fr = new window.FileReader();
  7094  			fr.onload = function() {
  7095  				// overwrite original blob
  7096  				data.append(data.getBlobName(), new Blob(null, {
  7097  					type: blob.type,
  7098  					data: fr.result
  7099  				}));
  7100  				// invoke send operation again
  7101  				self.send.call(target, meta, data);
  7102  			};
  7103  			fr.readAsBinaryString(blob);
  7104  		}
  7105  
  7106  		
  7107  		function _getNativeXHR() {
  7108  			if (window.XMLHttpRequest && !(Env.browser === 'IE' && Env.version < 8)) { // IE7 has native XHR but it's buggy
  7109  				return new window.XMLHttpRequest();
  7110  			} else {
  7111  				return (function() {
  7112  					var progIDs = ['Msxml2.XMLHTTP.6.0', 'Microsoft.XMLHTTP']; // if 6.0 available, use it, otherwise failback to default 3.0
  7113  					for (var i = 0; i < progIDs.length; i++) {
  7114  						try {
  7115  							return new ActiveXObject(progIDs[i]);
  7116  						} catch (ex) {}
  7117  					}
  7118  				})();
  7119  			}
  7120  		}
  7121  		
  7122  		// @credits Sergey Ilinsky	(http://www.ilinsky.com/)
  7123  		function _getDocument(xhr) {
  7124  			var rXML = xhr.responseXML;
  7125  			var rText = xhr.responseText;
  7126  			
  7127  			// Try parsing responseText (@see: http://www.ilinsky.com/articles/XMLHttpRequest/#bugs-ie-responseXML-content-type)
  7128  			if (Env.browser === 'IE' && rText && rXML && !rXML.documentElement && /[^\/]+\/[^\+]+\+xml/.test(xhr.getResponseHeader("Content-Type"))) {
  7129  				rXML = new window.ActiveXObject("Microsoft.XMLDOM");
  7130  				rXML.async = false;
  7131  				rXML.validateOnParse = false;
  7132  				rXML.loadXML(rText);
  7133  			}
  7134  	
  7135  			// Check if there is no error in document
  7136  			if (rXML) {
  7137  				if ((Env.browser === 'IE' && rXML.parseError !== 0) || !rXML.documentElement || rXML.documentElement.tagName === "parsererror") {
  7138  					return null;
  7139  				}
  7140  			}
  7141  			return rXML;
  7142  		}
  7143  
  7144  
  7145  		function _prepareMultipart(fd) {
  7146  			var boundary = '----moxieboundary' + new Date().getTime()
  7147  			, dashdash = '--'
  7148  			, crlf = '\r\n'
  7149  			, multipart = ''
  7150  			, I = this.getRuntime()
  7151  			;
  7152  
  7153  			if (!I.can('send_binary_string')) {
  7154  				throw new x.RuntimeError(x.RuntimeError.NOT_SUPPORTED_ERR);
  7155  			}
  7156  
  7157  			_xhr.setRequestHeader('Content-Type', 'multipart/form-data; boundary=' + boundary);
  7158  
  7159  			// append multipart parameters
  7160  			fd.each(function(value, name) {
  7161  				// Firefox 3.6 failed to convert multibyte characters to UTF-8 in sendAsBinary(), 
  7162  				// so we try it here ourselves with: unescape(encodeURIComponent(value))
  7163  				if (value instanceof Blob) {
  7164  					// Build RFC2388 blob
  7165  					multipart += dashdash + boundary + crlf +
  7166  						'Content-Disposition: form-data; name="' + name + '"; filename="' + unescape(encodeURIComponent(value.name || 'blob')) + '"' + crlf +
  7167  						'Content-Type: ' + (value.type || 'application/octet-stream') + crlf + crlf +
  7168  						value.getSource() + crlf;
  7169  				} else {
  7170  					multipart += dashdash + boundary + crlf +
  7171  						'Content-Disposition: form-data; name="' + name + '"' + crlf + crlf +
  7172  						unescape(encodeURIComponent(value)) + crlf;
  7173  				}
  7174  			});
  7175  
  7176  			multipart += dashdash + boundary + dashdash + crlf;
  7177  
  7178  			return multipart;
  7179  		}
  7180  	}
  7181  
  7182  	return (extensions.XMLHttpRequest = XMLHttpRequest);
  7183  });
  7184  
  7185  // Included from: src/javascript/runtime/html5/utils/BinaryReader.js
  7186  
  7187  /**
  7188   * BinaryReader.js
  7189   *
  7190   * Copyright 2013, Moxiecode Systems AB
  7191   * Released under GPL License.
  7192   *
  7193   * License: http://www.plupload.com/license
  7194   * Contributing: http://www.plupload.com/contributing
  7195   */
  7196  
  7197  /**
  7198  @class moxie/runtime/html5/utils/BinaryReader
  7199  @private
  7200  */
  7201  define("moxie/runtime/html5/utils/BinaryReader", [], function() {
  7202  	return function() {
  7203  		var II = false, bin;
  7204  
  7205  		// Private functions
  7206  		function read(idx, size) {
  7207  			var mv = II ? 0 : -8 * (size - 1), sum = 0, i;
  7208  
  7209  			for (i = 0; i < size; i++) {
  7210  				sum |= (bin.charCodeAt(idx + i) << Math.abs(mv + i*8));
  7211  			}
  7212  
  7213  			return sum;
  7214  		}
  7215  
  7216  		function putstr(segment, idx, length) {
  7217  			length = arguments.length === 3 ? length : bin.length - idx - 1;
  7218  			bin = bin.substr(0, idx) + segment + bin.substr(length + idx);
  7219  		}
  7220  
  7221  		function write(idx, num, size) {
  7222  			var str = '', mv = II ? 0 : -8 * (size - 1), i;
  7223  
  7224  			for (i = 0; i < size; i++) {
  7225  				str += String.fromCharCode((num >> Math.abs(mv + i*8)) & 255);
  7226  			}
  7227  
  7228  			putstr(str, idx, size);
  7229  		}
  7230  
  7231  		// Public functions
  7232  		return {
  7233  			II: function(order) {
  7234  				if (order === undefined) {
  7235  					return II;
  7236  				} else {
  7237  					II = order;
  7238  				}
  7239  			},
  7240  
  7241  			init: function(binData) {
  7242  				II = false;
  7243  				bin = binData;
  7244  			},
  7245  
  7246  			SEGMENT: function(idx, length, segment) {
  7247  				switch (arguments.length) {
  7248  					case 1:
  7249  						return bin.substr(idx, bin.length - idx - 1);
  7250  					case 2:
  7251  						return bin.substr(idx, length);
  7252  					case 3:
  7253  						putstr(segment, idx, length);
  7254  						break;
  7255  					default: return bin;
  7256  				}
  7257  			},
  7258  
  7259  			BYTE: function(idx) {
  7260  				return read(idx, 1);
  7261  			},
  7262  
  7263  			SHORT: function(idx) {
  7264  				return read(idx, 2);
  7265  			},
  7266  
  7267  			LONG: function(idx, num) {
  7268  				if (num === undefined) {
  7269  					return read(idx, 4);
  7270  				} else {
  7271  					write(idx, num, 4);
  7272  				}
  7273  			},
  7274  
  7275  			SLONG: function(idx) { // 2's complement notation
  7276  				var num = read(idx, 4);
  7277  
  7278  				return (num > 2147483647 ? num - 4294967296 : num);
  7279  			},
  7280  
  7281  			STRING: function(idx, size) {
  7282  				var str = '';
  7283  
  7284  				for (size += idx; idx < size; idx++) {
  7285  					str += String.fromCharCode(read(idx, 1));
  7286  				}
  7287  
  7288  				return str;
  7289  			}
  7290  		};
  7291  	};
  7292  });
  7293  
  7294  // Included from: src/javascript/runtime/html5/image/JPEGHeaders.js
  7295  
  7296  /**
  7297   * JPEGHeaders.js
  7298   *
  7299   * Copyright 2013, Moxiecode Systems AB
  7300   * Released under GPL License.
  7301   *
  7302   * License: http://www.plupload.com/license
  7303   * Contributing: http://www.plupload.com/contributing
  7304   */
  7305   
  7306  /**
  7307  @class moxie/runtime/html5/image/JPEGHeaders
  7308  @private
  7309  */
  7310  define("moxie/runtime/html5/image/JPEGHeaders", [
  7311  	"moxie/runtime/html5/utils/BinaryReader"
  7312  ], function(BinaryReader) {
  7313  	
  7314  	return function JPEGHeaders(data) {
  7315  		var headers = [], read, idx, marker, length = 0;
  7316  
  7317  		read = new BinaryReader();
  7318  		read.init(data);
  7319  
  7320  		// Check if data is jpeg
  7321  		if (read.SHORT(0) !== 0xFFD8) {
  7322  			return;
  7323  		}
  7324  
  7325  		idx = 2;
  7326  
  7327  		while (idx <= data.length) {
  7328  			marker = read.SHORT(idx);
  7329  
  7330  			// omit RST (restart) markers
  7331  			if (marker >= 0xFFD0 && marker <= 0xFFD7) {
  7332  				idx += 2;
  7333  				continue;
  7334  			}
  7335  
  7336  			// no headers allowed after SOS marker
  7337  			if (marker === 0xFFDA || marker === 0xFFD9) {
  7338  				break;
  7339  			}
  7340  
  7341  			length = read.SHORT(idx + 2) + 2;
  7342  
  7343  			// APPn marker detected
  7344  			if (marker >= 0xFFE1 && marker <= 0xFFEF) {
  7345  				headers.push({
  7346  					hex: marker,
  7347  					name: 'APP' + (marker & 0x000F),
  7348  					start: idx,
  7349  					length: length,
  7350  					segment: read.SEGMENT(idx, length)
  7351  				});
  7352  			}
  7353  
  7354  			idx += length;
  7355  		}
  7356  
  7357  		read.init(null); // free memory
  7358  
  7359  		return {
  7360  			headers: headers,
  7361  
  7362  			restore: function(data) {
  7363  				var max, i;
  7364  
  7365  				read.init(data);
  7366  
  7367  				idx = read.SHORT(2) == 0xFFE0 ? 4 + read.SHORT(4) : 2;
  7368  
  7369  				for (i = 0, max = headers.length; i < max; i++) {
  7370  					read.SEGMENT(idx, 0, headers[i].segment);
  7371  					idx += headers[i].length;
  7372  				}
  7373  
  7374  				data = read.SEGMENT();
  7375  				read.init(null);
  7376  				return data;
  7377  			},
  7378  
  7379  			strip: function(data) {
  7380  				var headers, jpegHeaders, i;
  7381  
  7382  				jpegHeaders = new JPEGHeaders(data);
  7383  				headers = jpegHeaders.headers;
  7384  				jpegHeaders.purge();
  7385  
  7386  				read.init(data);
  7387  
  7388  				i = headers.length;
  7389  				while (i--) {
  7390  					read.SEGMENT(headers[i].start, headers[i].length, '');
  7391  				}
  7392  				
  7393  				data = read.SEGMENT();
  7394  				read.init(null);
  7395  				return data;
  7396  			},
  7397  
  7398  			get: function(name) {
  7399  				var array = [];
  7400  
  7401  				for (var i = 0, max = headers.length; i < max; i++) {
  7402  					if (headers[i].name === name.toUpperCase()) {
  7403  						array.push(headers[i].segment);
  7404  					}
  7405  				}
  7406  				return array;
  7407  			},
  7408  
  7409  			set: function(name, segment) {
  7410  				var array = [], i, ii, max;
  7411  
  7412  				if (typeof(segment) === 'string') {
  7413  					array.push(segment);
  7414  				} else {
  7415  					array = segment;
  7416  				}
  7417  
  7418  				for (i = ii = 0, max = headers.length; i < max; i++) {
  7419  					if (headers[i].name === name.toUpperCase()) {
  7420  						headers[i].segment = array[ii];
  7421  						headers[i].length = array[ii].length;
  7422  						ii++;
  7423  					}
  7424  					if (ii >= array.length) {
  7425  						break;
  7426  					}
  7427  				}
  7428  			},
  7429  
  7430  			purge: function() {
  7431  				headers = [];
  7432  				read.init(null);
  7433  				read = null;
  7434  			}
  7435  		};
  7436  	};
  7437  });
  7438  
  7439  // Included from: src/javascript/runtime/html5/image/ExifParser.js
  7440  
  7441  /**
  7442   * ExifParser.js
  7443   *
  7444   * Copyright 2013, Moxiecode Systems AB
  7445   * Released under GPL License.
  7446   *
  7447   * License: http://www.plupload.com/license
  7448   * Contributing: http://www.plupload.com/contributing
  7449   */
  7450  
  7451  /**
  7452  @class moxie/runtime/html5/image/ExifParser
  7453  @private
  7454  */
  7455  define("moxie/runtime/html5/image/ExifParser", [
  7456  	"moxie/core/utils/Basic",
  7457  	"moxie/runtime/html5/utils/BinaryReader"
  7458  ], function(Basic, BinaryReader) {
  7459  	
  7460  	return function ExifParser() {
  7461  		// Private ExifParser fields
  7462  		var data, tags, Tiff, offsets = {}, tagDescs;
  7463  
  7464  		data = new BinaryReader();
  7465  
  7466  		tags = {
  7467  			tiff : {
  7468  				/*
  7469  				The image orientation viewed in terms of rows and columns.
  7470  
  7471  				1 = The 0th row is at the visual top of the image, and the 0th column is the visual left-hand side.
  7472  				2 = The 0th row is at the visual top of the image, and the 0th column is the visual right-hand side.
  7473  				3 = The 0th row is at the visual bottom of the image, and the 0th column is the visual right-hand side.
  7474  				4 = The 0th row is at the visual bottom of the image, and the 0th column is the visual left-hand side.
  7475  				5 = The 0th row is the visual left-hand side of the image, and the 0th column is the visual top.
  7476  				6 = The 0th row is the visual right-hand side of the image, and the 0th column is the visual top.
  7477  				7 = The 0th row is the visual right-hand side of the image, and the 0th column is the visual bottom.
  7478  				8 = The 0th row is the visual left-hand side of the image, and the 0th column is the visual bottom.
  7479  				*/
  7480  				0x0112: 'Orientation',
  7481  				0x010E: 'ImageDescription',
  7482  				0x010F: 'Make',
  7483  				0x0110: 'Model',
  7484  				0x0131: 'Software',
  7485  				0x8769: 'ExifIFDPointer',
  7486  				0x8825:	'GPSInfoIFDPointer'
  7487  			},
  7488  			exif : {
  7489  				0x9000: 'ExifVersion',
  7490  				0xA001: 'ColorSpace',
  7491  				0xA002: 'PixelXDimension',
  7492  				0xA003: 'PixelYDimension',
  7493  				0x9003: 'DateTimeOriginal',
  7494  				0x829A: 'ExposureTime',
  7495  				0x829D: 'FNumber',
  7496  				0x8827: 'ISOSpeedRatings',
  7497  				0x9201: 'ShutterSpeedValue',
  7498  				0x9202: 'ApertureValue'	,
  7499  				0x9207: 'MeteringMode',
  7500  				0x9208: 'LightSource',
  7501  				0x9209: 'Flash',
  7502  				0x920A: 'FocalLength',
  7503  				0xA402: 'ExposureMode',
  7504  				0xA403: 'WhiteBalance',
  7505  				0xA406: 'SceneCaptureType',
  7506  				0xA404: 'DigitalZoomRatio',
  7507  				0xA408: 'Contrast',
  7508  				0xA409: 'Saturation',
  7509  				0xA40A: 'Sharpness'
  7510  			},
  7511  			gps : {
  7512  				0x0000: 'GPSVersionID',
  7513  				0x0001: 'GPSLatitudeRef',
  7514  				0x0002: 'GPSLatitude',
  7515  				0x0003: 'GPSLongitudeRef',
  7516  				0x0004: 'GPSLongitude'
  7517  			}
  7518  		};
  7519  
  7520  		tagDescs = {
  7521  			'ColorSpace': {
  7522  				1: 'sRGB',
  7523  				0: 'Uncalibrated'
  7524  			},
  7525  
  7526  			'MeteringMode': {
  7527  				0: 'Unknown',
  7528  				1: 'Average',
  7529  				2: 'CenterWeightedAverage',
  7530  				3: 'Spot',
  7531  				4: 'MultiSpot',
  7532  				5: 'Pattern',
  7533  				6: 'Partial',
  7534  				255: 'Other'
  7535  			},
  7536  
  7537  			'LightSource': {
  7538  				1: 'Daylight',
  7539  				2: 'Fliorescent',
  7540  				3: 'Tungsten',
  7541  				4: 'Flash',
  7542  				9: 'Fine weather',
  7543  				10: 'Cloudy weather',
  7544  				11: 'Shade',
  7545  				12: 'Daylight fluorescent (D 5700 - 7100K)',
  7546  				13: 'Day white fluorescent (N 4600 -5400K)',
  7547  				14: 'Cool white fluorescent (W 3900 - 4500K)',
  7548  				15: 'White fluorescent (WW 3200 - 3700K)',
  7549  				17: 'Standard light A',
  7550  				18: 'Standard light B',
  7551  				19: 'Standard light C',
  7552  				20: 'D55',
  7553  				21: 'D65',
  7554  				22: 'D75',
  7555  				23: 'D50',
  7556  				24: 'ISO studio tungsten',
  7557  				255: 'Other'
  7558  			},
  7559  
  7560  			'Flash': {
  7561  				0x0000: 'Flash did not fire.',
  7562  				0x0001: 'Flash fired.',
  7563  				0x0005: 'Strobe return light not detected.',
  7564  				0x0007: 'Strobe return light detected.',
  7565  				0x0009: 'Flash fired, compulsory flash mode',
  7566  				0x000D: 'Flash fired, compulsory flash mode, return light not detected',
  7567  				0x000F: 'Flash fired, compulsory flash mode, return light detected',
  7568  				0x0010: 'Flash did not fire, compulsory flash mode',
  7569  				0x0018: 'Flash did not fire, auto mode',
  7570  				0x0019: 'Flash fired, auto mode',
  7571  				0x001D: 'Flash fired, auto mode, return light not detected',
  7572  				0x001F: 'Flash fired, auto mode, return light detected',
  7573  				0x0020: 'No flash function',
  7574  				0x0041: 'Flash fired, red-eye reduction mode',
  7575  				0x0045: 'Flash fired, red-eye reduction mode, return light not detected',
  7576  				0x0047: 'Flash fired, red-eye reduction mode, return light detected',
  7577  				0x0049: 'Flash fired, compulsory flash mode, red-eye reduction mode',
  7578  				0x004D: 'Flash fired, compulsory flash mode, red-eye reduction mode, return light not detected',
  7579  				0x004F: 'Flash fired, compulsory flash mode, red-eye reduction mode, return light detected',
  7580  				0x0059: 'Flash fired, auto mode, red-eye reduction mode',
  7581  				0x005D: 'Flash fired, auto mode, return light not detected, red-eye reduction mode',
  7582  				0x005F: 'Flash fired, auto mode, return light detected, red-eye reduction mode'
  7583  			},
  7584  
  7585  			'ExposureMode': {
  7586  				0: 'Auto exposure',
  7587  				1: 'Manual exposure',
  7588  				2: 'Auto bracket'
  7589  			},
  7590  
  7591  			'WhiteBalance': {
  7592  				0: 'Auto white balance',
  7593  				1: 'Manual white balance'
  7594  			},
  7595  
  7596  			'SceneCaptureType': {
  7597  				0: 'Standard',
  7598  				1: 'Landscape',
  7599  				2: 'Portrait',
  7600  				3: 'Night scene'
  7601  			},
  7602  
  7603  			'Contrast': {
  7604  				0: 'Normal',
  7605  				1: 'Soft',
  7606  				2: 'Hard'
  7607  			},
  7608  
  7609  			'Saturation': {
  7610  				0: 'Normal',
  7611  				1: 'Low saturation',
  7612  				2: 'High saturation'
  7613  			},
  7614  
  7615  			'Sharpness': {
  7616  				0: 'Normal',
  7617  				1: 'Soft',
  7618  				2: 'Hard'
  7619  			},
  7620  
  7621  			// GPS related
  7622  			'GPSLatitudeRef': {
  7623  				N: 'North latitude',
  7624  				S: 'South latitude'
  7625  			},
  7626  
  7627  			'GPSLongitudeRef': {
  7628  				E: 'East longitude',
  7629  				W: 'West longitude'
  7630  			}
  7631  		};
  7632  
  7633  		function extractTags(IFD_offset, tags2extract) {
  7634  			var length = data.SHORT(IFD_offset), i, ii,
  7635  				tag, type, count, tagOffset, offset, value, values = [], hash = {};
  7636  
  7637  			for (i = 0; i < length; i++) {
  7638  				// Set binary reader pointer to beginning of the next tag
  7639  				offset = tagOffset = IFD_offset + 12 * i + 2;
  7640  
  7641  				tag = tags2extract[data.SHORT(offset)];
  7642  
  7643  				if (tag === undefined) {
  7644  					continue; // Not the tag we requested
  7645  				}
  7646  
  7647  				type = data.SHORT(offset+=2);
  7648  				count = data.LONG(offset+=2);
  7649  
  7650  				offset += 4;
  7651  				values = [];
  7652  
  7653  				switch (type) {
  7654  					case 1: // BYTE
  7655  					case 7: // UNDEFINED
  7656  						if (count > 4) {
  7657  							offset = data.LONG(offset) + offsets.tiffHeader;
  7658  						}
  7659  
  7660  						for (ii = 0; ii < count; ii++) {
  7661  							values[ii] = data.BYTE(offset + ii);
  7662  						}
  7663  
  7664  						break;
  7665  
  7666  					case 2: // STRING
  7667  						if (count > 4) {
  7668  							offset = data.LONG(offset) + offsets.tiffHeader;
  7669  						}
  7670  
  7671  						hash[tag] = data.STRING(offset, count - 1);
  7672  
  7673  						continue;
  7674  
  7675  					case 3: // SHORT
  7676  						if (count > 2) {
  7677  							offset = data.LONG(offset) + offsets.tiffHeader;
  7678  						}
  7679  
  7680  						for (ii = 0; ii < count; ii++) {
  7681  							values[ii] = data.SHORT(offset + ii*2);
  7682  						}
  7683  
  7684  						break;
  7685  
  7686  					case 4: // LONG
  7687  						if (count > 1) {
  7688  							offset = data.LONG(offset) + offsets.tiffHeader;
  7689  						}
  7690  
  7691  						for (ii = 0; ii < count; ii++) {
  7692  							values[ii] = data.LONG(offset + ii*4);
  7693  						}
  7694  
  7695  						break;
  7696  
  7697  					case 5: // RATIONAL
  7698  						offset = data.LONG(offset) + offsets.tiffHeader;
  7699  
  7700  						for (ii = 0; ii < count; ii++) {
  7701  							values[ii] = data.LONG(offset + ii*4) / data.LONG(offset + ii*4 + 4);
  7702  						}
  7703  
  7704  						break;
  7705  
  7706  					case 9: // SLONG
  7707  						offset = data.LONG(offset) + offsets.tiffHeader;
  7708  
  7709  						for (ii = 0; ii < count; ii++) {
  7710  							values[ii] = data.SLONG(offset + ii*4);
  7711  						}
  7712  
  7713  						break;
  7714  
  7715  					case 10: // SRATIONAL
  7716  						offset = data.LONG(offset) + offsets.tiffHeader;
  7717  
  7718  						for (ii = 0; ii < count; ii++) {
  7719  							values[ii] = data.SLONG(offset + ii*4) / data.SLONG(offset + ii*4 + 4);
  7720  						}
  7721  
  7722  						break;
  7723  
  7724  					default:
  7725  						continue;
  7726  				}
  7727  
  7728  				value = (count == 1 ? values[0] : values);
  7729  
  7730  				if (tagDescs.hasOwnProperty(tag) && typeof value != 'object') {
  7731  					hash[tag] = tagDescs[tag][value];
  7732  				} else {
  7733  					hash[tag] = value;
  7734  				}
  7735  			}
  7736  
  7737  			return hash;
  7738  		}
  7739  
  7740  		function getIFDOffsets() {
  7741  			var idx = offsets.tiffHeader;
  7742  
  7743  			// Set read order of multi-byte data
  7744  			data.II(data.SHORT(idx) == 0x4949);
  7745  
  7746  			// Check if always present bytes are indeed present
  7747  			if (data.SHORT(idx+=2) !== 0x002A) {
  7748  				return false;
  7749  			}
  7750  
  7751  			offsets.IFD0 = offsets.tiffHeader + data.LONG(idx += 2);
  7752  			Tiff = extractTags(offsets.IFD0, tags.tiff);
  7753  
  7754  			if ('ExifIFDPointer' in Tiff) {
  7755  				offsets.exifIFD = offsets.tiffHeader + Tiff.ExifIFDPointer;
  7756  				delete Tiff.ExifIFDPointer;
  7757  			}
  7758  
  7759  			if ('GPSInfoIFDPointer' in Tiff) {
  7760  				offsets.gpsIFD = offsets.tiffHeader + Tiff.GPSInfoIFDPointer;
  7761  				delete Tiff.GPSInfoIFDPointer;
  7762  			}
  7763  			return true;
  7764  		}
  7765  
  7766  		// At the moment only setting of simple (LONG) values, that do not require offset recalculation, is supported
  7767  		function setTag(ifd, tag, value) {
  7768  			var offset, length, tagOffset, valueOffset = 0;
  7769  
  7770  			// If tag name passed translate into hex key
  7771  			if (typeof(tag) === 'string') {
  7772  				var tmpTags = tags[ifd.toLowerCase()];
  7773  				for (var hex in tmpTags) {
  7774  					if (tmpTags[hex] === tag) {
  7775  						tag = hex;
  7776  						break;
  7777  					}
  7778  				}
  7779  			}
  7780  			offset = offsets[ifd.toLowerCase() + 'IFD'];
  7781  			length = data.SHORT(offset);
  7782  
  7783  			for (var i = 0; i < length; i++) {
  7784  				tagOffset = offset + 12 * i + 2;
  7785  
  7786  				if (data.SHORT(tagOffset) == tag) {
  7787  					valueOffset = tagOffset + 8;
  7788  					break;
  7789  				}
  7790  			}
  7791  
  7792  			if (!valueOffset) {
  7793  				return false;
  7794  			}
  7795  
  7796  			data.LONG(valueOffset, value);
  7797  			return true;
  7798  		}
  7799  
  7800  
  7801  		// Public functions
  7802  		return {
  7803  			init: function(segment) {
  7804  				// Reset internal data
  7805  				offsets = {
  7806  					tiffHeader: 10
  7807  				};
  7808  
  7809  				if (segment === undefined || !segment.length) {
  7810  					return false;
  7811  				}
  7812  
  7813  				data.init(segment);
  7814  
  7815  				// Check if that's APP1 and that it has EXIF
  7816  				if (data.SHORT(0) === 0xFFE1 && data.STRING(4, 5).toUpperCase() === "EXIF\0") {
  7817  					return getIFDOffsets();
  7818  				}
  7819  				return false;
  7820  			},
  7821  
  7822  			TIFF: function() {
  7823  				return Tiff;
  7824  			},
  7825  
  7826  			EXIF: function() {
  7827  				var Exif;
  7828  
  7829  				// Populate EXIF hash
  7830  				Exif = extractTags(offsets.exifIFD, tags.exif);
  7831  
  7832  				// Fix formatting of some tags
  7833  				if (Exif.ExifVersion && Basic.typeOf(Exif.ExifVersion) === 'array') {
  7834  					for (var i = 0, exifVersion = ''; i < Exif.ExifVersion.length; i++) {
  7835  						exifVersion += String.fromCharCode(Exif.ExifVersion[i]);
  7836  					}
  7837  					Exif.ExifVersion = exifVersion;
  7838  				}
  7839  
  7840  				return Exif;
  7841  			},
  7842  
  7843  			GPS: function() {
  7844  				var GPS;
  7845  
  7846  				GPS = extractTags(offsets.gpsIFD, tags.gps);
  7847  
  7848  				// iOS devices (and probably some others) do not put in GPSVersionID tag (why?..)
  7849  				if (GPS.GPSVersionID && Basic.typeOf(GPS.GPSVersionID) === 'array') {
  7850  					GPS.GPSVersionID = GPS.GPSVersionID.join('.');
  7851  				}
  7852  
  7853  				return GPS;
  7854  			},
  7855  
  7856  			setExif: function(tag, value) {
  7857  				// Right now only setting of width/height is possible
  7858  				if (tag !== 'PixelXDimension' && tag !== 'PixelYDimension') {return false;}
  7859  
  7860  				return setTag('exif', tag, value);
  7861  			},
  7862  
  7863  
  7864  			getBinary: function() {
  7865  				return data.SEGMENT();
  7866  			},
  7867  
  7868  			purge: function() {
  7869  				data.init(null);
  7870  				data = Tiff = null;
  7871  				offsets = {};
  7872  			}
  7873  		};
  7874  	};
  7875  });
  7876  
  7877  // Included from: src/javascript/runtime/html5/image/JPEG.js
  7878  
  7879  /**
  7880   * JPEG.js
  7881   *
  7882   * Copyright 2013, Moxiecode Systems AB
  7883   * Released under GPL License.
  7884   *
  7885   * License: http://www.plupload.com/license
  7886   * Contributing: http://www.plupload.com/contributing
  7887   */
  7888  
  7889  /**
  7890  @class moxie/runtime/html5/image/JPEG
  7891  @private
  7892  */
  7893  define("moxie/runtime/html5/image/JPEG", [
  7894  	"moxie/core/utils/Basic",
  7895  	"moxie/core/Exceptions",
  7896  	"moxie/runtime/html5/image/JPEGHeaders",
  7897  	"moxie/runtime/html5/utils/BinaryReader",
  7898  	"moxie/runtime/html5/image/ExifParser"
  7899  ], function(Basic, x, JPEGHeaders, BinaryReader, ExifParser) {
  7900  	
  7901  	function JPEG(binstr) {
  7902  		var _binstr, _br, _hm, _ep, _info, hasExif;
  7903  
  7904  		function _getDimensions() {
  7905  			var idx = 0, marker, length;
  7906  
  7907  			// examine all through the end, since some images might have very large APP segments
  7908  			while (idx <= _binstr.length) {
  7909  				marker = _br.SHORT(idx += 2);
  7910  
  7911  				if (marker >= 0xFFC0 && marker <= 0xFFC3) { // SOFn
  7912  					idx += 5; // marker (2 bytes) + length (2 bytes) + Sample precision (1 byte)
  7913  					return {
  7914  						height: _br.SHORT(idx),
  7915  						width: _br.SHORT(idx += 2)
  7916  					};
  7917  				}
  7918  				length = _br.SHORT(idx += 2);
  7919  				idx += length - 2;
  7920  			}
  7921  			return null;
  7922  		}
  7923  
  7924  		_binstr = binstr;
  7925  
  7926  		_br = new BinaryReader();
  7927  		_br.init(_binstr);
  7928  
  7929  		// check if it is jpeg
  7930  		if (_br.SHORT(0) !== 0xFFD8) {
  7931  			throw new x.ImageError(x.ImageError.WRONG_FORMAT);
  7932  		}
  7933  
  7934  		// backup headers
  7935  		_hm = new JPEGHeaders(binstr);
  7936  
  7937  		// extract exif info
  7938  		_ep = new ExifParser();
  7939  		hasExif = !!_ep.init(_hm.get('app1')[0]);
  7940  
  7941  		// get dimensions
  7942  		_info = _getDimensions.call(this);
  7943  
  7944  		Basic.extend(this, {
  7945  			type: 'image/jpeg',
  7946  
  7947  			size: _binstr.length,
  7948  
  7949  			width: _info && _info.width || 0,
  7950  
  7951  			height: _info && _info.height || 0,
  7952  
  7953  			setExif: function(tag, value) {
  7954  				if (!hasExif) {
  7955  					return false; // or throw an exception
  7956  				}
  7957  
  7958  				if (Basic.typeOf(tag) === 'object') {
  7959  					Basic.each(tag, function(value, tag) {
  7960  						_ep.setExif(tag, value);
  7961  					});
  7962  				} else {
  7963  					_ep.setExif(tag, value);
  7964  				}
  7965  
  7966  				// update internal headers
  7967  				_hm.set('app1', _ep.getBinary());
  7968  			},
  7969  
  7970  			writeHeaders: function() {
  7971  				if (!arguments.length) {
  7972  					// if no arguments passed, update headers internally
  7973  					return (_binstr = _hm.restore(_binstr));
  7974  				}
  7975  				return _hm.restore(arguments[0]);
  7976  			},
  7977  
  7978  			stripHeaders: function(binstr) {
  7979  				return _hm.strip(binstr);
  7980  			},
  7981  
  7982  			purge: function() {
  7983  				_purge.call(this);
  7984  			}
  7985  		});
  7986  
  7987  		if (hasExif) {
  7988  			this.meta = {
  7989  				tiff: _ep.TIFF(),
  7990  				exif: _ep.EXIF(),
  7991  				gps: _ep.GPS()
  7992  			};
  7993  		}
  7994  
  7995  		function _purge() {
  7996  			if (!_ep || !_hm || !_br) { 
  7997  				return; // ignore any repeating purge requests
  7998  			}
  7999  			_ep.purge();
  8000  			_hm.purge();
  8001  			_br.init(null);
  8002  			_binstr = _info = _hm = _ep = _br = null;
  8003  		}
  8004  	}
  8005  
  8006  	return JPEG;
  8007  });
  8008  
  8009  // Included from: src/javascript/runtime/html5/image/PNG.js
  8010  
  8011  /**
  8012   * PNG.js
  8013   *
  8014   * Copyright 2013, Moxiecode Systems AB
  8015   * Released under GPL License.
  8016   *
  8017   * License: http://www.plupload.com/license
  8018   * Contributing: http://www.plupload.com/contributing
  8019   */
  8020  
  8021  /**
  8022  @class moxie/runtime/html5/image/PNG
  8023  @private
  8024  */
  8025  define("moxie/runtime/html5/image/PNG", [
  8026  	"moxie/core/Exceptions",
  8027  	"moxie/core/utils/Basic",
  8028  	"moxie/runtime/html5/utils/BinaryReader"
  8029  ], function(x, Basic, BinaryReader) {
  8030  	
  8031  	function PNG(binstr) {
  8032  		var _binstr, _br, _hm, _ep, _info;
  8033  
  8034  		_binstr = binstr;
  8035  
  8036  		_br = new BinaryReader();
  8037  		_br.init(_binstr);
  8038  
  8039  		// check if it's png
  8040  		(function() {
  8041  			var idx = 0, i = 0
  8042  			, signature = [0x8950, 0x4E47, 0x0D0A, 0x1A0A]
  8043  			;
  8044  
  8045  			for (i = 0; i < signature.length; i++, idx += 2) {
  8046  				if (signature[i] != _br.SHORT(idx)) {
  8047  					throw new x.ImageError(x.ImageError.WRONG_FORMAT);
  8048  				}
  8049  			}
  8050  		}());
  8051  
  8052  		function _getDimensions() {
  8053  			var chunk, idx;
  8054  
  8055  			chunk = _getChunkAt.call(this, 8);
  8056  
  8057  			if (chunk.type == 'IHDR') {
  8058  				idx = chunk.start;
  8059  				return {
  8060  					width: _br.LONG(idx),
  8061  					height: _br.LONG(idx += 4)
  8062  				};
  8063  			}
  8064  			return null;
  8065  		}
  8066  
  8067  		function _purge() {
  8068  			if (!_br) {
  8069  				return; // ignore any repeating purge requests
  8070  			}
  8071  			_br.init(null);
  8072  			_binstr = _info = _hm = _ep = _br = null;
  8073  		}
  8074  
  8075  		_info = _getDimensions.call(this);
  8076  
  8077  		Basic.extend(this, {
  8078  			type: 'image/png',
  8079  
  8080  			size: _binstr.length,
  8081  
  8082  			width: _info.width,
  8083  
  8084  			height: _info.height,
  8085  
  8086  			purge: function() {
  8087  				_purge.call(this);
  8088  			}
  8089  		});
  8090  
  8091  		// for PNG we can safely trigger purge automatically, as we do not keep any data for later
  8092  		_purge.call(this);
  8093  
  8094  		function _getChunkAt(idx) {
  8095  			var length, type, start, CRC;
  8096  
  8097  			length = _br.LONG(idx);
  8098  			type = _br.STRING(idx += 4, 4);
  8099  			start = idx += 4;
  8100  			CRC = _br.LONG(idx + length);
  8101  
  8102  			return {
  8103  				length: length,
  8104  				type: type,
  8105  				start: start,
  8106  				CRC: CRC
  8107  			};
  8108  		}
  8109  	}
  8110  
  8111  	return PNG;
  8112  });
  8113  
  8114  // Included from: src/javascript/runtime/html5/image/ImageInfo.js
  8115  
  8116  /**
  8117   * ImageInfo.js
  8118   *
  8119   * Copyright 2013, Moxiecode Systems AB
  8120   * Released under GPL License.
  8121   *
  8122   * License: http://www.plupload.com/license
  8123   * Contributing: http://www.plupload.com/contributing
  8124   */
  8125  
  8126  /**
  8127  @class moxie/runtime/html5/image/ImageInfo
  8128  @private
  8129  */
  8130  define("moxie/runtime/html5/image/ImageInfo", [
  8131  	"moxie/core/utils/Basic",
  8132  	"moxie/core/Exceptions",
  8133  	"moxie/runtime/html5/image/JPEG",
  8134  	"moxie/runtime/html5/image/PNG"
  8135  ], function(Basic, x, JPEG, PNG) {
  8136  	/**
  8137  	Optional image investigation tool for HTML5 runtime. Provides the following features:
  8138  	- ability to distinguish image type (JPEG or PNG) by signature
  8139  	- ability to extract image width/height directly from it's internals, without preloading in memory (fast)
  8140  	- ability to extract APP headers from JPEGs (Exif, GPS, etc)
  8141  	- ability to replace width/height tags in extracted JPEG headers
  8142  	- ability to restore APP headers, that were for example stripped during image manipulation
  8143  
  8144  	@class ImageInfo
  8145  	@constructor
  8146  	@param {String} binstr Image source as binary string
  8147  	*/
  8148  	return function(binstr) {
  8149  		var _cs = [JPEG, PNG], _img;
  8150  
  8151  		// figure out the format, throw: ImageError.WRONG_FORMAT if not supported
  8152  		_img = (function() {
  8153  			for (var i = 0; i < _cs.length; i++) {
  8154  				try {
  8155  					return new _cs[i](binstr);
  8156  				} catch (ex) {
  8157  					// console.info(ex);
  8158  				}
  8159  			}
  8160  			throw new x.ImageError(x.ImageError.WRONG_FORMAT);
  8161  		}());
  8162  
  8163  		Basic.extend(this, {
  8164  			/**
  8165  			Image Mime Type extracted from it's depths
  8166  
  8167  			@property type
  8168  			@type {String}
  8169  			@default ''
  8170  			*/
  8171  			type: '',
  8172  
  8173  			/**
  8174  			Image size in bytes
  8175  
  8176  			@property size
  8177  			@type {Number}
  8178  			@default 0
  8179  			*/
  8180  			size: 0,
  8181  
  8182  			/**
  8183  			Image width extracted from image source
  8184  
  8185  			@property width
  8186  			@type {Number}
  8187  			@default 0
  8188  			*/
  8189  			width: 0,
  8190  
  8191  			/**
  8192  			Image height extracted from image source
  8193  
  8194  			@property height
  8195  			@type {Number}
  8196  			@default 0
  8197  			*/
  8198  			height: 0,
  8199  
  8200  			/**
  8201  			Sets Exif tag. Currently applicable only for width and height tags. Obviously works only with JPEGs.
  8202  
  8203  			@method setExif
  8204  			@param {String} tag Tag to set
  8205  			@param {Mixed} value Value to assign to the tag
  8206  			*/
  8207  			setExif: function() {},
  8208  
  8209  			/**
  8210  			Restores headers to the source.
  8211  
  8212  			@method writeHeaders
  8213  			@param {String} data Image source as binary string
  8214  			@return {String} Updated binary string
  8215  			*/
  8216  			writeHeaders: function(data) {
  8217  				return data;
  8218  			},
  8219  
  8220  			/**
  8221  			Strip all headers from the source.
  8222  
  8223  			@method stripHeaders
  8224  			@param {String} data Image source as binary string
  8225  			@return {String} Updated binary string
  8226  			*/
  8227  			stripHeaders: function(data) {
  8228  				return data;
  8229  			},
  8230  
  8231  			/**
  8232  			Dispose resources.
  8233  
  8234  			@method purge
  8235  			*/
  8236  			purge: function() {}
  8237  		});
  8238  
  8239  		Basic.extend(this, _img);
  8240  
  8241  		this.purge = function() {
  8242  			_img.purge();
  8243  			_img = null;
  8244  		};
  8245  	};
  8246  });
  8247  
  8248  // Included from: src/javascript/runtime/html5/image/MegaPixel.js
  8249  
  8250  /**
  8251  (The MIT License)
  8252  
  8253  Copyright (c) 2012 Shinichi Tomita <shinichi.tomita@gmail.com>;
  8254  
  8255  Permission is hereby granted, free of charge, to any person obtaining
  8256  a copy of this software and associated documentation files (the
  8257  'Software'), to deal in the Software without restriction, including
  8258  without limitation the rights to use, copy, modify, merge, publish,
  8259  distribute, sublicense, and/or sell copies of the Software, and to
  8260  permit persons to whom the Software is furnished to do so, subject to
  8261  the following conditions:
  8262  
  8263  The above copyright notice and this permission notice shall be
  8264  included in all copies or substantial portions of the Software.
  8265  
  8266  THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,
  8267  EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  8268  MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
  8269  IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
  8270  CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
  8271  TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
  8272  SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  8273  */
  8274  
  8275  /**
  8276   * Mega pixel image rendering library for iOS6 Safari
  8277   *
  8278   * Fixes iOS6 Safari's image file rendering issue for large size image (over mega-pixel),
  8279   * which causes unexpected subsampling when drawing it in canvas.
  8280   * By using this library, you can safely render the image with proper stretching.
  8281   *
  8282   * Copyright (c) 2012 Shinichi Tomita <shinichi.tomita@gmail.com>
  8283   * Released under the MIT license
  8284   */
  8285  
  8286  /**
  8287  @class moxie/runtime/html5/image/MegaPixel
  8288  @private
  8289  */
  8290  define("moxie/runtime/html5/image/MegaPixel", [], function() {
  8291  
  8292  	/**
  8293  	 * Rendering image element (with resizing) into the canvas element
  8294  	 */
  8295  	function renderImageToCanvas(img, canvas, options) {
  8296  		var iw = img.naturalWidth, ih = img.naturalHeight;
  8297  		var width = options.width, height = options.height;
  8298  		var x = options.x || 0, y = options.y || 0;
  8299  		var ctx = canvas.getContext('2d');
  8300  		if (detectSubsampling(img)) {
  8301  			iw /= 2;
  8302  			ih /= 2;
  8303  		}
  8304  		var d = 1024; // size of tiling canvas
  8305  		var tmpCanvas = document.createElement('canvas');
  8306  		tmpCanvas.width = tmpCanvas.height = d;
  8307  		var tmpCtx = tmpCanvas.getContext('2d');
  8308  		var vertSquashRatio = detectVerticalSquash(img, iw, ih);
  8309  		var sy = 0;
  8310  		while (sy < ih) {
  8311  			var sh = sy + d > ih ? ih - sy : d;
  8312  			var sx = 0;
  8313  			while (sx < iw) {
  8314  				var sw = sx + d > iw ? iw - sx : d;
  8315  				tmpCtx.clearRect(0, 0, d, d);
  8316  				tmpCtx.drawImage(img, -sx, -sy);
  8317  				var dx = (sx * width / iw + x) << 0;
  8318  				var dw = Math.ceil(sw * width / iw);
  8319  				var dy = (sy * height / ih / vertSquashRatio + y) << 0;
  8320  				var dh = Math.ceil(sh * height / ih / vertSquashRatio);
  8321  				ctx.drawImage(tmpCanvas, 0, 0, sw, sh, dx, dy, dw, dh);
  8322  				sx += d;
  8323  			}
  8324  			sy += d;
  8325  		}
  8326  		tmpCanvas = tmpCtx = null;
  8327  	}
  8328  
  8329  	/**
  8330  	 * Detect subsampling in loaded image.
  8331  	 * In iOS, larger images than 2M pixels may be subsampled in rendering.
  8332  	 */
  8333  	function detectSubsampling(img) {
  8334  		var iw = img.naturalWidth, ih = img.naturalHeight;
  8335  		if (iw * ih > 1024 * 1024) { // subsampling may happen over megapixel image
  8336  			var canvas = document.createElement('canvas');
  8337  			canvas.width = canvas.height = 1;
  8338  			var ctx = canvas.getContext('2d');
  8339  			ctx.drawImage(img, -iw + 1, 0);
  8340  			// subsampled image becomes half smaller in rendering size.
  8341  			// check alpha channel value to confirm image is covering edge pixel or not.
  8342  			// if alpha value is 0 image is not covering, hence subsampled.
  8343  			return ctx.getImageData(0, 0, 1, 1).data[3] === 0;
  8344  		} else {
  8345  			return false;
  8346  		}
  8347  	}
  8348  
  8349  
  8350  	/**
  8351  	 * Detecting vertical squash in loaded image.
  8352  	 * Fixes a bug which squash image vertically while drawing into canvas for some images.
  8353  	 */
  8354  	function detectVerticalSquash(img, iw, ih) {
  8355  		var canvas = document.createElement('canvas');
  8356  		canvas.width = 1;
  8357  		canvas.height = ih;
  8358  		var ctx = canvas.getContext('2d');
  8359  		ctx.drawImage(img, 0, 0);
  8360  		var data = ctx.getImageData(0, 0, 1, ih).data;
  8361  		// search image edge pixel position in case it is squashed vertically.
  8362  		var sy = 0;
  8363  		var ey = ih;
  8364  		var py = ih;
  8365  		while (py > sy) {
  8366  			var alpha = data[(py - 1) * 4 + 3];
  8367  			if (alpha === 0) {
  8368  				ey = py;
  8369  			} else {
  8370  			sy = py;
  8371  			}
  8372  			py = (ey + sy) >> 1;
  8373  		}
  8374  		canvas = null;
  8375  		var ratio = (py / ih);
  8376  		return (ratio === 0) ? 1 : ratio;
  8377  	}
  8378  
  8379  	return {
  8380  		isSubsampled: detectSubsampling,
  8381  		renderTo: renderImageToCanvas
  8382  	};
  8383  });
  8384  
  8385  // Included from: src/javascript/runtime/html5/image/Image.js
  8386  
  8387  /**
  8388   * Image.js
  8389   *
  8390   * Copyright 2013, Moxiecode Systems AB
  8391   * Released under GPL License.
  8392   *
  8393   * License: http://www.plupload.com/license
  8394   * Contributing: http://www.plupload.com/contributing
  8395   */
  8396  
  8397  /**
  8398  @class moxie/runtime/html5/image/Image
  8399  @private
  8400  */
  8401  define("moxie/runtime/html5/image/Image", [
  8402  	"moxie/runtime/html5/Runtime",
  8403  	"moxie/core/utils/Basic",
  8404  	"moxie/core/Exceptions",
  8405  	"moxie/core/utils/Encode",
  8406  	"moxie/file/File",
  8407  	"moxie/runtime/html5/image/ImageInfo",
  8408  	"moxie/runtime/html5/image/MegaPixel",
  8409  	"moxie/core/utils/Mime",
  8410  	"moxie/core/utils/Env"
  8411  ], function(extensions, Basic, x, Encode, File, ImageInfo, MegaPixel, Mime, Env) {
  8412  	
  8413  	function HTML5Image() {
  8414  		var me = this
  8415  		, _img, _imgInfo, _canvas, _binStr, _blob
  8416  		, _modified = false // is set true whenever image is modified
  8417  		, _preserveHeaders = true
  8418  		;
  8419  
  8420  		Basic.extend(this, {
  8421  			loadFromBlob: function(blob) {
  8422  				var comp = this, I = comp.getRuntime()
  8423  				, asBinary = arguments.length > 1 ? arguments[1] : true
  8424  				;
  8425  
  8426  				if (!I.can('access_binary')) {
  8427  					throw new x.RuntimeError(x.RuntimeError.NOT_SUPPORTED_ERR);
  8428  				}
  8429  
  8430  				_blob = blob;
  8431  
  8432  				if (blob.isDetached()) {
  8433  					_binStr = blob.getSource();
  8434  					_preload.call(this, _binStr);
  8435  					return;
  8436  				} else {
  8437  					_readAsDataUrl.call(this, blob.getSource(), function(dataUrl) {
  8438  						if (asBinary) {
  8439  							_binStr = _toBinary(dataUrl);
  8440  						}
  8441  						_preload.call(comp, dataUrl);
  8442  					});
  8443  				}
  8444  			},
  8445  
  8446  			loadFromImage: function(img, exact) {
  8447  				this.meta = img.meta;
  8448  
  8449  				_blob = new File(null, {
  8450  					name: img.name,
  8451  					size: img.size,
  8452  					type: img.type
  8453  				});
  8454  
  8455  				_preload.call(this, exact ? (_binStr = img.getAsBinaryString()) : img.getAsDataURL());
  8456  			},
  8457  
  8458  			getInfo: function() {
  8459  				var I = this.getRuntime(), info;
  8460  
  8461  				if (!_imgInfo && _binStr && I.can('access_image_binary')) {
  8462  					_imgInfo = new ImageInfo(_binStr);
  8463  				}
  8464  
  8465  				info = {
  8466  					width: _getImg().width || 0,
  8467  					height: _getImg().height || 0,
  8468  					type: _blob.type || Mime.getFileMime(_blob.name),
  8469  					size: _binStr && _binStr.length || _blob.size || 0,
  8470  					name: _blob.name || '',
  8471  					meta: _imgInfo && _imgInfo.meta || this.meta || {}
  8472  				};
  8473  
  8474  				return info;
  8475  			},
  8476  
  8477  			downsize: function() {
  8478  				_downsize.apply(this, arguments);
  8479  			},
  8480  
  8481  			getAsCanvas: function() {
  8482  				if (_canvas) {
  8483  					_canvas.id = this.uid + '_canvas';
  8484  				}
  8485  				return _canvas;
  8486  			},
  8487  
  8488  			getAsBlob: function(type, quality) {
  8489  				if (type !== this.type) {
  8490  					// if different mime type requested prepare image for conversion
  8491  					_downsize.call(this, this.width, this.height, false);
  8492  				}
  8493  				return new File(null, {
  8494  					name: _blob.name || '',
  8495  					type: type,
  8496  					data: me.getAsBinaryString.call(this, type, quality)
  8497  				});
  8498  			},
  8499  
  8500  			getAsDataURL: function(type) {
  8501  				var quality = arguments[1] || 90;
  8502  
  8503  				// if image has not been modified, return the source right away
  8504  				if (!_modified) {
  8505  					return _img.src;
  8506  				}
  8507  
  8508  				if ('image/jpeg' !== type) {
  8509  					return _canvas.toDataURL('image/png');
  8510  				} else {
  8511  					try {
  8512  						// older Geckos used to result in an exception on quality argument
  8513  						return _canvas.toDataURL('image/jpeg', quality/100);
  8514  					} catch (ex) {
  8515  						return _canvas.toDataURL('image/jpeg');
  8516  					}
  8517  				}
  8518  			},
  8519  
  8520  			getAsBinaryString: function(type, quality) {
  8521  				// if image has not been modified, return the source right away
  8522  				if (!_modified) {
  8523  					// if image was not loaded from binary string
  8524  					if (!_binStr) {
  8525  						_binStr = _toBinary(me.getAsDataURL(type, quality));
  8526  					}
  8527  					return _binStr;
  8528  				}
  8529  
  8530  				if ('image/jpeg' !== type) {
  8531  					_binStr = _toBinary(me.getAsDataURL(type, quality));
  8532  				} else {
  8533  					var dataUrl;
  8534  
  8535  					// if jpeg
  8536  					if (!quality) {
  8537  						quality = 90;
  8538  					}
  8539  
  8540  					try {
  8541  						// older Geckos used to result in an exception on quality argument
  8542  						dataUrl = _canvas.toDataURL('image/jpeg', quality/100);
  8543  					} catch (ex) {
  8544  						dataUrl = _canvas.toDataURL('image/jpeg');
  8545  					}
  8546  
  8547  					_binStr = _toBinary(dataUrl);
  8548  
  8549  					if (_imgInfo) {
  8550  						_binStr = _imgInfo.stripHeaders(_binStr);
  8551  
  8552  						if (_preserveHeaders) {
  8553  							// update dimensions info in exif
  8554  							if (_imgInfo.meta && _imgInfo.meta.exif) {
  8555  								_imgInfo.setExif({
  8556  									PixelXDimension: this.width,
  8557  									PixelYDimension: this.height
  8558  								});
  8559  							}
  8560  
  8561  							// re-inject the headers
  8562  							_binStr = _imgInfo.writeHeaders(_binStr);
  8563  						}
  8564  
  8565  						// will be re-created from fresh on next getInfo call
  8566  						_imgInfo.purge();
  8567  						_imgInfo = null;
  8568  					}
  8569  				}
  8570  
  8571  				_modified = false;
  8572  
  8573  				return _binStr;
  8574  			},
  8575  
  8576  			destroy: function() {
  8577  				me = null;
  8578  				_purge.call(this);
  8579  				this.getRuntime().getShim().removeInstance(this.uid);
  8580  			}
  8581  		});
  8582  
  8583  
  8584  		function _getImg() {
  8585  			if (!_canvas && !_img) {
  8586  				throw new x.ImageError(x.DOMException.INVALID_STATE_ERR);
  8587  			}
  8588  			return _canvas || _img;
  8589  		}
  8590  
  8591  
  8592  		function _toBinary(str) {
  8593  			return Encode.atob(str.substring(str.indexOf('base64,') + 7));
  8594  		}
  8595  
  8596  
  8597  		function _toDataUrl(str, type) {
  8598  			return 'data:' + (type || '') + ';base64,' + Encode.btoa(str);
  8599  		}
  8600  
  8601  
  8602  		function _preload(str) {
  8603  			var comp = this;
  8604  
  8605  			_img = new Image();
  8606  			_img.onerror = function() {
  8607  				_purge.call(this);
  8608  				comp.trigger('error', new x.ImageError(x.ImageError.WRONG_FORMAT));
  8609  			};
  8610  			_img.onload = function() {
  8611  				comp.trigger('load');
  8612  			};
  8613  
  8614  			_img.src = /^data:[^;]*;base64,/.test(str) ? str : _toDataUrl(str, _blob.type);
  8615  		}
  8616  
  8617  
  8618  		function _readAsDataUrl(file, callback) {
  8619  			var comp = this, fr;
  8620  
  8621  			// use FileReader if it's available
  8622  			if (window.FileReader) {
  8623  				fr = new FileReader();
  8624  				fr.onload = function() {
  8625  					callback(this.result);
  8626  				};
  8627  				fr.onerror = function() {
  8628  					comp.trigger('error', new x.FileException(x.FileException.NOT_READABLE_ERR));
  8629  				};
  8630  				fr.readAsDataURL(file);
  8631  			} else {
  8632  				return callback(file.getAsDataURL());
  8633  			}
  8634  		}
  8635  
  8636  		function _downsize(width, height, crop, preserveHeaders) {
  8637  			var self = this
  8638  			, scale
  8639  			, mathFn
  8640  			, x = 0
  8641  			, y = 0
  8642  			, img
  8643  			, destWidth
  8644  			, destHeight
  8645  			, orientation
  8646  			;
  8647  
  8648  			_preserveHeaders = preserveHeaders; // we will need to check this on export (see getAsBinaryString())
  8649  
  8650  			// take into account orientation tag
  8651  			orientation = (this.meta && this.meta.tiff && this.meta.tiff.Orientation) || 1;
  8652  
  8653  			if (Basic.inArray(orientation, [5,6,7,8]) !== -1) { // values that require 90 degree rotation
  8654  				// swap dimensions
  8655  				var tmp = width;
  8656  				width = height;
  8657  				height = tmp;
  8658  			}
  8659  
  8660  			img = _getImg();
  8661  
  8662  			// unify dimensions
  8663  			mathFn = !crop ? Math.min : Math.max;
  8664  			scale = mathFn(width/img.width, height/img.height);
  8665  		
  8666  			// we only downsize here
  8667  			if (scale > 1 && (!crop || preserveHeaders)) { // when cropping one of dimensions may still exceed max, so process it anyway
  8668  				this.trigger('Resize');
  8669  				return;
  8670  			}
  8671  
  8672  			// prepare canvas if necessary
  8673  			if (!_canvas) {
  8674  				_canvas = document.createElement("canvas");
  8675  			}
  8676  
  8677  			// calculate dimensions of proportionally resized image
  8678  			destWidth = Math.round(img.width * scale);	
  8679  			destHeight = Math.round(img.height * scale);
  8680  
  8681  
  8682  			// scale image and canvas
  8683  			if (crop) {
  8684  				_canvas.width = width;
  8685  				_canvas.height = height;
  8686  
  8687  				// if dimensions of the resulting image still larger than canvas, center it
  8688  				if (destWidth > width) {
  8689  					x = Math.round((destWidth - width) / 2);
  8690  				}
  8691  
  8692  				if (destHeight > height) {
  8693  					y = Math.round((destHeight - height) / 2);
  8694  				}
  8695  			} else {
  8696  				_canvas.width = destWidth;
  8697  				_canvas.height = destHeight;
  8698  			}
  8699  
  8700  			// rotate if required, according to orientation tag
  8701  			if (!_preserveHeaders) {
  8702  				_rotateToOrientaion(_canvas.width, _canvas.height, orientation);
  8703  			}
  8704  
  8705  			_drawToCanvas.call(this, img, _canvas, -x, -y, destWidth, destHeight);
  8706  
  8707  			this.width = _canvas.width;
  8708  			this.height = _canvas.height;
  8709  
  8710  			_modified = true;
  8711  			self.trigger('Resize');
  8712  		}
  8713  
  8714  
  8715  		function _drawToCanvas(img, canvas, x, y, w, h) {
  8716  			if (Env.OS === 'iOS') { 
  8717  				// avoid squish bug in iOS6
  8718  				MegaPixel.renderTo(img, canvas, { width: w, height: h, x: x, y: y });
  8719  			} else {
  8720  				var ctx = canvas.getContext('2d');
  8721  				ctx.drawImage(img, x, y, w, h);
  8722  			}
  8723  		}
  8724  
  8725  
  8726  		/**
  8727  		* Transform canvas coordination according to specified frame size and orientation
  8728  		* Orientation value is from EXIF tag
  8729  		* @author Shinichi Tomita <shinichi.tomita@gmail.com>
  8730  		*/
  8731  		function _rotateToOrientaion(width, height, orientation) {
  8732  			switch (orientation) {
  8733  				case 5:
  8734  				case 6:
  8735  				case 7:
  8736  				case 8:
  8737  					_canvas.width = height;
  8738  					_canvas.height = width;
  8739  					break;
  8740  				default:
  8741  					_canvas.width = width;
  8742  					_canvas.height = height;
  8743  			}
  8744  
  8745  			/**
  8746  			1 = The 0th row is at the visual top of the image, and the 0th column is the visual left-hand side.
  8747  			2 = The 0th row is at the visual top of the image, and the 0th column is the visual right-hand side.
  8748  			3 = The 0th row is at the visual bottom of the image, and the 0th column is the visual right-hand side.
  8749  			4 = The 0th row is at the visual bottom of the image, and the 0th column is the visual left-hand side.
  8750  			5 = The 0th row is the visual left-hand side of the image, and the 0th column is the visual top.
  8751  			6 = The 0th row is the visual right-hand side of the image, and the 0th column is the visual top.
  8752  			7 = The 0th row is the visual right-hand side of the image, and the 0th column is the visual bottom.
  8753  			8 = The 0th row is the visual left-hand side of the image, and the 0th column is the visual bottom.
  8754  			*/
  8755  
  8756  			var ctx = _canvas.getContext('2d');
  8757  			switch (orientation) {
  8758  				case 2:
  8759  					// horizontal flip
  8760  					ctx.translate(width, 0);
  8761  					ctx.scale(-1, 1);
  8762  					break;
  8763  				case 3:
  8764  					// 180 rotate left
  8765  					ctx.translate(width, height);
  8766  					ctx.rotate(Math.PI);
  8767  					break;
  8768  				case 4:
  8769  					// vertical flip
  8770  					ctx.translate(0, height);
  8771  					ctx.scale(1, -1);
  8772  					break;
  8773  				case 5:
  8774  					// vertical flip + 90 rotate right
  8775  					ctx.rotate(0.5 * Math.PI);
  8776  					ctx.scale(1, -1);
  8777  					break;
  8778  				case 6:
  8779  					// 90 rotate right
  8780  					ctx.rotate(0.5 * Math.PI);
  8781  					ctx.translate(0, -height);
  8782  					break;
  8783  				case 7:
  8784  					// horizontal flip + 90 rotate right
  8785  					ctx.rotate(0.5 * Math.PI);
  8786  					ctx.translate(width, -height);
  8787  					ctx.scale(-1, 1);
  8788  					break;
  8789  				case 8:
  8790  					// 90 rotate left
  8791  					ctx.rotate(-0.5 * Math.PI);
  8792  					ctx.translate(-width, 0);
  8793  					break;
  8794  			}
  8795  		}
  8796  
  8797  
  8798  		function _purge() {
  8799  			if (_imgInfo) {
  8800  				_imgInfo.purge();
  8801  				_imgInfo = null;
  8802  			}
  8803  			_binStr = _img = _canvas = _blob = null;
  8804  			_modified = false;
  8805  		}
  8806  	}
  8807  
  8808  	return (extensions.Image = HTML5Image);
  8809  });
  8810  
  8811  // Included from: src/javascript/runtime/flash/Runtime.js
  8812  
  8813  /**
  8814   * Runtime.js
  8815   *
  8816   * Copyright 2013, Moxiecode Systems AB
  8817   * Released under GPL License.
  8818   *
  8819   * License: http://www.plupload.com/license
  8820   * Contributing: http://www.plupload.com/contributing
  8821   */
  8822  
  8823  /*global ActiveXObject:true */
  8824  
  8825  /**
  8826  Defines constructor for Flash runtime.
  8827  
  8828  @class moxie/runtime/flash/Runtime
  8829  @private
  8830  */
  8831  define("moxie/runtime/flash/Runtime", [
  8832  	"moxie/core/utils/Basic",
  8833  	"moxie/core/utils/Env",
  8834  	"moxie/core/utils/Dom",
  8835  	"moxie/core/Exceptions",
  8836  	"moxie/runtime/Runtime"
  8837  ], function(Basic, Env, Dom, x, Runtime) {
  8838  	
  8839  	var type = 'flash', extensions = {};
  8840  
  8841  	/**
  8842  	Get the version of the Flash Player
  8843  
  8844  	@method getShimVersion
  8845  	@private
  8846  	@return {Number} Flash Player version
  8847  	*/
  8848  	function getShimVersion() {
  8849  		var version;
  8850  
  8851  		try {
  8852  			version = navigator.plugins['Shockwave Flash'];
  8853  			version = version.description;
  8854  		} catch (e1) {
  8855  			try {
  8856  				version = new ActiveXObject('ShockwaveFlash.ShockwaveFlash').GetVariable('$version');
  8857  			} catch (e2) {
  8858  				version = '0.0';
  8859  			}
  8860  		}
  8861  		version = version.match(/\d+/g);
  8862  		return parseFloat(version[0] + '.' + version[1]);
  8863  	}
  8864  
  8865  	/**
  8866  	Constructor for the Flash Runtime
  8867  
  8868  	@class FlashRuntime
  8869  	@extends Runtime
  8870  	*/
  8871  	function FlashRuntime(options) {
  8872  		var I = this, initTimer;
  8873  
  8874  		options = Basic.extend({ swf_url: Env.swf_url }, options);
  8875  
  8876  		Runtime.call(this, options, type, {
  8877  			access_binary: function(value) {
  8878  				return value && I.mode === 'browser';
  8879  			},
  8880  			access_image_binary: function(value) {
  8881  				return value && I.mode === 'browser';
  8882  			},
  8883  			display_media: Runtime.capTrue,
  8884  			do_cors: Runtime.capTrue,
  8885  			drag_and_drop: false,
  8886  			report_upload_progress: function() {
  8887  				return I.mode === 'client';
  8888  			},
  8889  			resize_image: Runtime.capTrue,
  8890  			return_response_headers: false,
  8891  			return_response_type: function(responseType) {
  8892  				if (responseType === 'json' && !!window.JSON) {
  8893  					return true;
  8894  				} 
  8895  				return !Basic.arrayDiff(responseType, ['', 'text', 'document']) || I.mode === 'browser';
  8896  			},
  8897  			return_status_code: function(code) {
  8898  				return I.mode === 'browser' || !Basic.arrayDiff(code, [200, 404]);
  8899  			},
  8900  			select_file: Runtime.capTrue,
  8901  			select_multiple: Runtime.capTrue,
  8902  			send_binary_string: function(value) {
  8903  				return value && I.mode === 'browser';
  8904  			},
  8905  			send_browser_cookies: function(value) {
  8906  				return value && I.mode === 'browser';
  8907  			},
  8908  			send_custom_headers: function(value) {
  8909  				return value && I.mode === 'browser';
  8910  			},
  8911  			send_multipart: Runtime.capTrue,
  8912  			slice_blob: Runtime.capTrue,
  8913  			stream_upload: function(value) {
  8914  				return value && I.mode === 'browser';
  8915  			},
  8916  			summon_file_dialog: false,
  8917  			upload_filesize: function(size) {
  8918  				return Basic.parseSizeStr(size) <= 2097152 || I.mode === 'client';
  8919  			},
  8920  			use_http_method: function(methods) {
  8921  				return !Basic.arrayDiff(methods, ['GET', 'POST']);
  8922  			}
  8923  		}, { 
  8924  			// capabilities that require specific mode
  8925  			access_binary: function(value) {
  8926  				return value ? 'browser' : 'client';
  8927  			},
  8928  			access_image_binary: function(value) {
  8929  				return value ? 'browser' : 'client';
  8930  			},
  8931  			report_upload_progress: function(value) {
  8932  				return value ? 'browser' : 'client';
  8933  			},
  8934  			return_response_type: function(responseType) {
  8935  				return Basic.arrayDiff(responseType, ['', 'text', 'json', 'document']) ? 'browser' : ['client', 'browser'];
  8936  			},
  8937  			return_status_code: function(code) {
  8938  				return Basic.arrayDiff(code, [200, 404]) ? 'browser' : ['client', 'browser'];
  8939  			},
  8940  			send_binary_string: function(value) {
  8941  				return value ? 'browser' : 'client';
  8942  			},
  8943  			send_browser_cookies: function(value) {
  8944  				return value ? 'browser' : 'client';
  8945  			},
  8946  			send_custom_headers: function(value) {
  8947  				return value ? 'browser' : 'client';
  8948  			},
  8949  			stream_upload: function(value) {
  8950  				return value ? 'client' : 'browser';
  8951  			},
  8952  			upload_filesize: function(size) {
  8953  				return Basic.parseSizeStr(size) >= 2097152 ? 'client' : 'browser';
  8954  			}
  8955  		}, 'client');
  8956  
  8957  
  8958  		// minimal requirement for Flash Player version
  8959  		if (getShimVersion() < 10) {
  8960  			this.mode = false; // with falsy mode, runtime won't operable, no matter what the mode was before
  8961  		}
  8962  
  8963  
  8964  		Basic.extend(this, {
  8965  
  8966  			getShim: function() {
  8967  				return Dom.get(this.uid);
  8968  			},
  8969  
  8970  			shimExec: function(component, action) {
  8971  				var args = [].slice.call(arguments, 2);
  8972  				return I.getShim().exec(this.uid, component, action, args);
  8973  			},
  8974  
  8975  			init: function() {
  8976  				var html, el, container;
  8977  
  8978  				container = this.getShimContainer();
  8979  
  8980  				// if not the minimal height, shims are not initialized in older browsers (e.g FF3.6, IE6,7,8, Safari 4.0,5.0, etc)
  8981  				Basic.extend(container.style, {
  8982  					position: 'absolute',
  8983  					top: '-8px',
  8984  					left: '-8px',
  8985  					width: '9px',
  8986  					height: '9px',
  8987  					overflow: 'hidden'
  8988  				});
  8989  
  8990  				// insert flash object
  8991  				html = '<object id="' + this.uid + '" type="application/x-shockwave-flash" data="' +  options.swf_url + '" ';
  8992  
  8993  				if (Env.browser === 'IE') {
  8994  					html += 'classid="clsid:d27cdb6e-ae6d-11cf-96b8-444553540000" ';
  8995  				}
  8996  
  8997  				html += 'width="100%" height="100%" style="outline:0">'  +
  8998  					'<param name="movie" value="' + options.swf_url + '" />' +
  8999  					'<param name="flashvars" value="uid=' + escape(this.uid) + '&target=' + Env.global_event_dispatcher + '" />' +
  9000  					'<param name="wmode" value="transparent" />' +
  9001  					'<param name="allowscriptaccess" value="always" />' +
  9002  				'</object>';
  9003  
  9004  				if (Env.browser === 'IE') {
  9005  					el = document.createElement('div');
  9006  					container.appendChild(el);
  9007  					el.outerHTML = html;
  9008  					el = container = null; // just in case
  9009  				} else {
  9010  					container.innerHTML = html;
  9011  				}
  9012  
  9013  				// Init is dispatched by the shim
  9014  				initTimer = setTimeout(function() {
  9015  					if (I && !I.initialized) { // runtime might be already destroyed by this moment
  9016  						I.trigger("Error", new x.RuntimeError(x.RuntimeError.NOT_INIT_ERR));
  9017  					}
  9018  				}, 5000);
  9019  			},
  9020  
  9021  			destroy: (function(destroy) { // extend default destroy method
  9022  				return function() {
  9023  					destroy.call(I);
  9024  					clearTimeout(initTimer); // initialization check might be still onwait
  9025  					options = initTimer = destroy = I = null;
  9026  				};
  9027  			}(this.destroy))
  9028  
  9029  		}, extensions);
  9030  	}
  9031  
  9032  	Runtime.addConstructor(type, FlashRuntime);
  9033  
  9034  	return extensions;
  9035  });
  9036  
  9037  // Included from: src/javascript/runtime/flash/file/Blob.js
  9038  
  9039  /**
  9040   * Blob.js
  9041   *
  9042   * Copyright 2013, Moxiecode Systems AB
  9043   * Released under GPL License.
  9044   *
  9045   * License: http://www.plupload.com/license
  9046   * Contributing: http://www.plupload.com/contributing
  9047   */
  9048  
  9049  /**
  9050  @class moxie/runtime/flash/file/Blob
  9051  @private
  9052  */
  9053  define("moxie/runtime/flash/file/Blob", [
  9054  	"moxie/runtime/flash/Runtime",
  9055  	"moxie/file/Blob"
  9056  ], function(extensions, Blob) {
  9057  
  9058  	var FlashBlob = {
  9059  		slice: function(blob, start, end, type) {
  9060  			var self = this.getRuntime();
  9061  
  9062  			if (start < 0) {
  9063  				start = Math.max(blob.size + start, 0);
  9064  			} else if (start > 0) {
  9065  				start = Math.min(start, blob.size);
  9066  			}
  9067  
  9068  			if (end < 0) {
  9069  				end = Math.max(blob.size + end, 0);
  9070  			} else if (end > 0) {
  9071  				end = Math.min(end, blob.size);
  9072  			}
  9073  
  9074  			blob = self.shimExec.call(this, 'Blob', 'slice', start, end, type || '');
  9075  
  9076  			if (blob) {
  9077  				blob = new Blob(self.uid, blob);
  9078  			}
  9079  			return blob;
  9080  		}
  9081  	};
  9082  
  9083  	return (extensions.Blob = FlashBlob);
  9084  });
  9085  
  9086  // Included from: src/javascript/runtime/flash/file/FileInput.js
  9087  
  9088  /**
  9089   * FileInput.js
  9090   *
  9091   * Copyright 2013, Moxiecode Systems AB
  9092   * Released under GPL License.
  9093   *
  9094   * License: http://www.plupload.com/license
  9095   * Contributing: http://www.plupload.com/contributing
  9096   */
  9097  
  9098  /**
  9099  @class moxie/runtime/flash/file/FileInput
  9100  @private
  9101  */
  9102  define("moxie/runtime/flash/file/FileInput", [
  9103  	"moxie/runtime/flash/Runtime"
  9104  ], function(extensions) {
  9105  	
  9106  	var FileInput = {		
  9107  		init: function(options) {
  9108  			this.getRuntime().shimExec.call(this, 'FileInput', 'init', {
  9109  				name: options.name,
  9110  				accept: options.accept,
  9111  				multiple: options.multiple
  9112  			});
  9113  			this.trigger('ready');
  9114  		}
  9115  	};
  9116  
  9117  	return (extensions.FileInput = FileInput);
  9118  });
  9119  
  9120  // Included from: src/javascript/runtime/flash/file/FileReader.js
  9121  
  9122  /**
  9123   * FileReader.js
  9124   *
  9125   * Copyright 2013, Moxiecode Systems AB
  9126   * Released under GPL License.
  9127   *
  9128   * License: http://www.plupload.com/license
  9129   * Contributing: http://www.plupload.com/contributing
  9130   */
  9131  
  9132  /**
  9133  @class moxie/runtime/flash/file/FileReader
  9134  @private
  9135  */
  9136  define("moxie/runtime/flash/file/FileReader", [
  9137  	"moxie/runtime/flash/Runtime",
  9138  	"moxie/core/utils/Encode"
  9139  ], function(extensions, Encode) {
  9140  
  9141  	var _result = '';
  9142  
  9143  	function _formatData(data, op) {
  9144  		switch (op) {
  9145  			case 'readAsText':
  9146  				return Encode.atob(data, 'utf8');
  9147  			case 'readAsBinaryString':
  9148  				return Encode.atob(data);
  9149  			case 'readAsDataURL':
  9150  				return data;
  9151  		}
  9152  		return null;
  9153  	}
  9154  
  9155  	var FileReader = {
  9156  		read: function(op, blob) {
  9157  			var target = this, self = target.getRuntime();
  9158  
  9159  			// special prefix for DataURL read mode
  9160  			if (op === 'readAsDataURL') {
  9161  				_result = 'data:' + (blob.type || '') + ';base64,';
  9162  			}
  9163  
  9164  			target.bind('Progress', function(e, data) {
  9165  				if (data) {
  9166  					_result += _formatData(data, op);
  9167  				}
  9168  			});
  9169  
  9170  			return self.shimExec.call(this, 'FileReader', 'readAsBase64', blob.uid);
  9171  		},
  9172  
  9173  		getResult: function() {
  9174  			return _result;
  9175  		},
  9176  
  9177  		destroy: function() {
  9178  			_result = null;
  9179  		}
  9180  	};
  9181  
  9182  	return (extensions.FileReader = FileReader);
  9183  });
  9184  
  9185  // Included from: src/javascript/runtime/flash/file/FileReaderSync.js
  9186  
  9187  /**
  9188   * FileReaderSync.js
  9189   *
  9190   * Copyright 2013, Moxiecode Systems AB
  9191   * Released under GPL License.
  9192   *
  9193   * License: http://www.plupload.com/license
  9194   * Contributing: http://www.plupload.com/contributing
  9195   */
  9196  
  9197  /**
  9198  @class moxie/runtime/flash/file/FileReaderSync
  9199  @private
  9200  */
  9201  define("moxie/runtime/flash/file/FileReaderSync", [
  9202  	"moxie/runtime/flash/Runtime",
  9203  	"moxie/core/utils/Encode"
  9204  ], function(extensions, Encode) {
  9205  	
  9206  	function _formatData(data, op) {
  9207  		switch (op) {
  9208  			case 'readAsText':
  9209  				return Encode.atob(data, 'utf8');
  9210  			case 'readAsBinaryString':
  9211  				return Encode.atob(data);
  9212  			case 'readAsDataURL':
  9213  				return data;
  9214  		}
  9215  		return null;
  9216  	}
  9217  
  9218  	var FileReaderSync = {
  9219  		read: function(op, blob) {
  9220  			var result, self = this.getRuntime();
  9221  
  9222  			result = self.shimExec.call(this, 'FileReaderSync', 'readAsBase64', blob.uid);
  9223  			if (!result) {
  9224  				return null; // or throw ex
  9225  			}
  9226  
  9227  			// special prefix for DataURL read mode
  9228  			if (op === 'readAsDataURL') {
  9229  				result = 'data:' + (blob.type || '') + ';base64,' + result;
  9230  			}
  9231  
  9232  			return _formatData(result, op, blob.type);
  9233  		}
  9234  	};
  9235  
  9236  	return (extensions.FileReaderSync = FileReaderSync);
  9237  });
  9238  
  9239  // Included from: src/javascript/runtime/flash/xhr/XMLHttpRequest.js
  9240  
  9241  /**
  9242   * XMLHttpRequest.js
  9243   *
  9244   * Copyright 2013, Moxiecode Systems AB
  9245   * Released under GPL License.
  9246   *
  9247   * License: http://www.plupload.com/license
  9248   * Contributing: http://www.plupload.com/contributing
  9249   */
  9250  
  9251  /**
  9252  @class moxie/runtime/flash/xhr/XMLHttpRequest
  9253  @private
  9254  */
  9255  define("moxie/runtime/flash/xhr/XMLHttpRequest", [
  9256  	"moxie/runtime/flash/Runtime",
  9257  	"moxie/core/utils/Basic",
  9258  	"moxie/file/Blob",
  9259  	"moxie/file/File",
  9260  	"moxie/file/FileReaderSync",
  9261  	"moxie/xhr/FormData",
  9262  	"moxie/runtime/Transporter"
  9263  ], function(extensions, Basic, Blob, File, FileReaderSync, FormData, Transporter) {
  9264  	
  9265  	var XMLHttpRequest = {
  9266  
  9267  		send: function(meta, data) {
  9268  			var target = this, self = target.getRuntime();
  9269  
  9270  			function send() {
  9271  				meta.transport = self.mode;
  9272  				self.shimExec.call(target, 'XMLHttpRequest', 'send', meta, data);
  9273  			}
  9274  
  9275  
  9276  			function appendBlob(name, blob) {
  9277  				self.shimExec.call(target, 'XMLHttpRequest', 'appendBlob', name, blob.uid);
  9278  				data = null;
  9279  				send();
  9280  			}
  9281  
  9282  
  9283  			function attachBlob(blob, cb) {
  9284  				var tr = new Transporter();
  9285  
  9286  				tr.bind("TransportingComplete", function() {
  9287  					cb(this.result);
  9288  				});
  9289  
  9290  				tr.transport(blob.getSource(), blob.type, {
  9291  					ruid: self.uid
  9292  				});
  9293  			}
  9294  
  9295  			// copy over the headers if any
  9296  			if (!Basic.isEmptyObj(meta.headers)) {
  9297  				Basic.each(meta.headers, function(value, header) {
  9298  					self.shimExec.call(target, 'XMLHttpRequest', 'setRequestHeader', header, value.toString()); // Silverlight doesn't accept integers into the arguments of type object
  9299  				});
  9300  			}
  9301  
  9302  			// transfer over multipart params and blob itself
  9303  			if (data instanceof FormData) {
  9304  				var blobField;
  9305  				data.each(function(value, name) {
  9306  					if (value instanceof Blob) {
  9307  						blobField = name;
  9308  					} else {
  9309  						self.shimExec.call(target, 'XMLHttpRequest', 'append', name, value);
  9310  					}
  9311  				});
  9312  
  9313  				if (!data.hasBlob()) {
  9314  					data = null;
  9315  					send();
  9316  				} else {
  9317  					var blob = data.getBlob();
  9318  					if (blob.isDetached()) {
  9319  						attachBlob(blob, function(attachedBlob) {
  9320  							blob.destroy();
  9321  							appendBlob(blobField, attachedBlob);		
  9322  						});
  9323  					} else {
  9324  						appendBlob(blobField, blob);
  9325  					}
  9326  				}
  9327  			} else if (data instanceof Blob) {
  9328  				if (data.isDetached()) {
  9329  					attachBlob(data, function(attachedBlob) {
  9330  						data.destroy();
  9331  						data = attachedBlob.uid;
  9332  						send();
  9333  					});
  9334  				} else {
  9335  					data = data.uid;
  9336  					send();
  9337  				}
  9338  			} else {
  9339  				send();
  9340  			}
  9341  		},
  9342  
  9343  		getResponse: function(responseType) {
  9344  			var frs, blob, self = this.getRuntime();
  9345  
  9346  			blob = self.shimExec.call(this, 'XMLHttpRequest', 'getResponseAsBlob');
  9347  
  9348  			if (blob) {
  9349  				blob = new File(self.uid, blob);
  9350  
  9351  				if ('blob' === responseType) {
  9352  					return blob;
  9353  				}
  9354  
  9355  				try { 
  9356  					frs = new FileReaderSync();
  9357  
  9358  					if (!!~Basic.inArray(responseType, ["", "text"])) {
  9359  						return frs.readAsText(blob);
  9360  					} else if ('json' === responseType && !!window.JSON) {
  9361  						return JSON.parse(frs.readAsText(blob));
  9362  					}
  9363  				} finally {
  9364  					blob.destroy();
  9365  				}
  9366  			}
  9367  			return null;
  9368  		},
  9369  
  9370  		abort: function(upload_complete_flag) {
  9371  			var self = this.getRuntime();
  9372  
  9373  			self.shimExec.call(this, 'XMLHttpRequest', 'abort');
  9374  
  9375  			this.dispatchEvent('readystatechange');
  9376  			// this.dispatchEvent('progress');
  9377  			this.dispatchEvent('abort');
  9378  
  9379  			//if (!upload_complete_flag) {
  9380  				// this.dispatchEvent('uploadprogress');
  9381  			//}
  9382  		}
  9383  	};
  9384  
  9385  	return (extensions.XMLHttpRequest = XMLHttpRequest);
  9386  });
  9387  
  9388  // Included from: src/javascript/runtime/flash/runtime/Transporter.js
  9389  
  9390  /**
  9391   * Transporter.js
  9392   *
  9393   * Copyright 2013, Moxiecode Systems AB
  9394   * Released under GPL License.
  9395   *
  9396   * License: http://www.plupload.com/license
  9397   * Contributing: http://www.plupload.com/contributing
  9398   */
  9399  
  9400  /**
  9401  @class moxie/runtime/flash/runtime/Transporter
  9402  @private
  9403  */
  9404  define("moxie/runtime/flash/runtime/Transporter", [
  9405  	"moxie/runtime/flash/Runtime",
  9406  	"moxie/file/Blob"
  9407  ], function(extensions, Blob) {
  9408  
  9409  	var Transporter = {
  9410  		getAsBlob: function(type) {
  9411  			var self = this.getRuntime()
  9412  			, blob = self.shimExec.call(this, 'Transporter', 'getAsBlob', type)
  9413  			;
  9414  			if (blob) {
  9415  				return new Blob(self.uid, blob);
  9416  			}
  9417  			return null;
  9418  		}
  9419  	};
  9420  
  9421  	return (extensions.Transporter = Transporter);
  9422  });
  9423  
  9424  // Included from: src/javascript/runtime/flash/image/Image.js
  9425  
  9426  /**
  9427   * Image.js
  9428   *
  9429   * Copyright 2013, Moxiecode Systems AB
  9430   * Released under GPL License.
  9431   *
  9432   * License: http://www.plupload.com/license
  9433   * Contributing: http://www.plupload.com/contributing
  9434   */
  9435  
  9436  /**
  9437  @class moxie/runtime/flash/image/Image
  9438  @private
  9439  */
  9440  define("moxie/runtime/flash/image/Image", [
  9441  	"moxie/runtime/flash/Runtime",
  9442  	"moxie/core/utils/Basic",
  9443  	"moxie/runtime/Transporter",
  9444  	"moxie/file/Blob",
  9445  	"moxie/file/FileReaderSync"
  9446  ], function(extensions, Basic, Transporter, Blob, FileReaderSync) {
  9447  	
  9448  	var Image = {
  9449  		loadFromBlob: function(blob) {
  9450  			var comp = this, self = comp.getRuntime();
  9451  
  9452  			function exec(srcBlob) {
  9453  				self.shimExec.call(comp, 'Image', 'loadFromBlob', srcBlob.uid);
  9454  				comp = self = null;
  9455  			}
  9456  
  9457  			if (blob.isDetached()) { // binary string
  9458  				var tr = new Transporter();
  9459  				tr.bind("TransportingComplete", function() {
  9460  					exec(tr.result.getSource());
  9461  				});
  9462  				tr.transport(blob.getSource(), blob.type, { ruid: self.uid });
  9463  			} else {
  9464  				exec(blob.getSource());
  9465  			}
  9466  		},
  9467  
  9468  		loadFromImage: function(img) {
  9469  			var self = this.getRuntime();
  9470  			return self.shimExec.call(this, 'Image', 'loadFromImage', img.uid);
  9471  		},
  9472  
  9473  		getAsBlob: function(type, quality) {
  9474  			var self = this.getRuntime()
  9475  			, blob = self.shimExec.call(this, 'Image', 'getAsBlob', type, quality)
  9476  			;
  9477  			if (blob) {
  9478  				return new Blob(self.uid, blob);
  9479  			}
  9480  			return null;
  9481  		},
  9482  
  9483  		getAsDataURL: function() {
  9484  			var self = this.getRuntime()
  9485  			, blob = self.Image.getAsBlob.apply(this, arguments)
  9486  			, frs
  9487  			;
  9488  			if (!blob) {
  9489  				return null;
  9490  			}
  9491  			frs = new FileReaderSync();
  9492  			return frs.readAsDataURL(blob);
  9493  		}
  9494  	};
  9495  
  9496  	return (extensions.Image = Image);
  9497  });
  9498  
  9499  // Included from: src/javascript/runtime/silverlight/Runtime.js
  9500  
  9501  /**
  9502   * RunTime.js
  9503   *
  9504   * Copyright 2013, Moxiecode Systems AB
  9505   * Released under GPL License.
  9506   *
  9507   * License: http://www.plupload.com/license
  9508   * Contributing: http://www.plupload.com/contributing
  9509   */
  9510  
  9511  /*global ActiveXObject:true */
  9512  
  9513  /**
  9514  Defines constructor for Silverlight runtime.
  9515  
  9516  @class moxie/runtime/silverlight/Runtime
  9517  @private
  9518  */
  9519  define("moxie/runtime/silverlight/Runtime", [
  9520  	"moxie/core/utils/Basic",
  9521  	"moxie/core/utils/Env",
  9522  	"moxie/core/utils/Dom",
  9523  	"moxie/core/Exceptions",
  9524  	"moxie/runtime/Runtime"
  9525  ], function(Basic, Env, Dom, x, Runtime) {
  9526  	
  9527  	var type = "silverlight", extensions = {};
  9528  
  9529  	function isInstalled(version) {
  9530  		var isVersionSupported = false, control = null, actualVer,
  9531  			actualVerArray, reqVerArray, requiredVersionPart, actualVersionPart, index = 0;
  9532  
  9533  		try {
  9534  			try {
  9535  				control = new ActiveXObject('AgControl.AgControl');
  9536  
  9537  				if (control.IsVersionSupported(version)) {
  9538  					isVersionSupported = true;
  9539  				}
  9540  
  9541  				control = null;
  9542  			} catch (e) {
  9543  				var plugin = navigator.plugins["Silverlight Plug-In"];
  9544  
  9545  				if (plugin) {
  9546  					actualVer = plugin.description;
  9547  
  9548  					if (actualVer === "1.0.30226.2") {
  9549  						actualVer = "2.0.30226.2";
  9550  					}
  9551  
  9552  					actualVerArray = actualVer.split(".");
  9553  
  9554  					while (actualVerArray.length > 3) {
  9555  						actualVerArray.pop();
  9556  					}
  9557  
  9558  					while ( actualVerArray.length < 4) {
  9559  						actualVerArray.push(0);
  9560  					}
  9561  
  9562  					reqVerArray = version.split(".");
  9563  
  9564  					while (reqVerArray.length > 4) {
  9565  						reqVerArray.pop();
  9566  					}
  9567  
  9568  					do {
  9569  						requiredVersionPart = parseInt(reqVerArray[index], 10);
  9570  						actualVersionPart = parseInt(actualVerArray[index], 10);
  9571  						index++;
  9572  					} while (index < reqVerArray.length && requiredVersionPart === actualVersionPart);
  9573  
  9574  					if (requiredVersionPart <= actualVersionPart && !isNaN(requiredVersionPart)) {
  9575  						isVersionSupported = true;
  9576  					}
  9577  				}
  9578  			}
  9579  		} catch (e2) {
  9580  			isVersionSupported = false;
  9581  		}
  9582  
  9583  		return isVersionSupported;
  9584  	}
  9585  
  9586  	/**
  9587  	Constructor for the Silverlight Runtime
  9588  
  9589  	@class SilverlightRuntime
  9590  	@extends Runtime
  9591  	*/
  9592  	function SilverlightRuntime(options) {
  9593  		var I = this, initTimer;
  9594  
  9595  		options = Basic.extend({ xap_url: Env.xap_url }, options);
  9596  
  9597  		Runtime.call(this, options, type, {
  9598  			access_binary: Runtime.capTrue,
  9599  			access_image_binary: Runtime.capTrue,
  9600  			display_media: Runtime.capTrue,
  9601  			do_cors: Runtime.capTrue,
  9602  			drag_and_drop: false,
  9603  			report_upload_progress: Runtime.capTrue,
  9604  			resize_image: Runtime.capTrue,
  9605  			return_response_headers: function(value) {
  9606  				return value && I.mode === 'client';
  9607  			},
  9608  			return_response_type: function(responseType) {
  9609  				if (responseType !== 'json') {
  9610  					return true;
  9611  				} else {
  9612  					return !!window.JSON;
  9613  				}
  9614  			},
  9615  			return_status_code: function(code) {
  9616  				return I.mode === 'client' || !Basic.arrayDiff(code, [200, 404]);
  9617  			},
  9618  			select_file: Runtime.capTrue,
  9619  			select_multiple: Runtime.capTrue,
  9620  			send_binary_string: Runtime.capTrue,
  9621  			send_browser_cookies: function(value) {
  9622  				return value && I.mode === 'browser';
  9623  			},
  9624  			send_custom_headers: function(value) {
  9625  				return value && I.mode === 'client';
  9626  			},
  9627  			send_multipart: Runtime.capTrue,
  9628  			slice_blob: Runtime.capTrue,
  9629  			stream_upload: true,
  9630  			summon_file_dialog: false,
  9631  			upload_filesize: Runtime.capTrue,
  9632  			use_http_method: function(methods) {
  9633  				return I.mode === 'client' || !Basic.arrayDiff(methods, ['GET', 'POST']);
  9634  			}
  9635  		}, { 
  9636  			// capabilities that require specific mode
  9637  			return_response_headers: function(value) {
  9638  				return value ? 'client' : 'browser';
  9639  			},
  9640  			return_status_code: function(code) {
  9641  				return Basic.arrayDiff(code, [200, 404]) ? 'client' : ['client', 'browser'];
  9642  			},
  9643  			send_browser_cookies: function(value) {
  9644  				return value ? 'browser' : 'client';
  9645  			},
  9646  			send_custom_headers: function(value) {
  9647  				return value ? 'client' : 'browser';
  9648  			},
  9649  			use_http_method: function(methods) {
  9650  				return Basic.arrayDiff(methods, ['GET', 'POST']) ? 'client' : ['client', 'browser'];
  9651  			}
  9652  		});
  9653  
  9654  
  9655  		// minimal requirement
  9656  		if (!isInstalled('2.0.31005.0') || Env.browser === 'Opera') {
  9657  			this.mode = false;
  9658  		}
  9659  
  9660  
  9661  		Basic.extend(this, {
  9662  			getShim: function() {
  9663  				return Dom.get(this.uid).content.Moxie;
  9664  			},
  9665  
  9666  			shimExec: function(component, action) {
  9667  				var args = [].slice.call(arguments, 2);
  9668  				return I.getShim().exec(this.uid, component, action, args);
  9669  			},
  9670  
  9671  			init : function() {
  9672  				var container;
  9673  
  9674  				container = this.getShimContainer();
  9675  
  9676  				container.innerHTML = '<object id="' + this.uid + '" data="data:application/x-silverlight," type="application/x-silverlight-2" width="100%" height="100%" style="outline:none;">' +
  9677  					'<param name="source" value="' + options.xap_url + '"/>' +
  9678  					'<param name="background" value="Transparent"/>' +
  9679  					'<param name="windowless" value="true"/>' +
  9680  					'<param name="enablehtmlaccess" value="true"/>' +
  9681  					'<param name="initParams" value="uid=' + this.uid + ',target=' + Env.global_event_dispatcher + '"/>' +
  9682  				'</object>';
  9683  
  9684  				// Init is dispatched by the shim
  9685  				initTimer = setTimeout(function() {
  9686  					if (I && !I.initialized) { // runtime might be already destroyed by this moment
  9687  						I.trigger("Error", new x.RuntimeError(x.RuntimeError.NOT_INIT_ERR));
  9688  					}
  9689  				}, Env.OS !== 'Windows'? 10000 : 5000); // give it more time to initialize in non Windows OS (like Mac)
  9690  			},
  9691  
  9692  			destroy: (function(destroy) { // extend default destroy method
  9693  				return function() {
  9694  					destroy.call(I);
  9695  					clearTimeout(initTimer); // initialization check might be still onwait
  9696  					options = initTimer = destroy = I = null;
  9697  				};
  9698  			}(this.destroy))
  9699  
  9700  		}, extensions);
  9701  	}
  9702  
  9703  	Runtime.addConstructor(type, SilverlightRuntime); 
  9704  
  9705  	return extensions;
  9706  });
  9707  
  9708  // Included from: src/javascript/runtime/silverlight/file/Blob.js
  9709  
  9710  /**
  9711   * Blob.js
  9712   *
  9713   * Copyright 2013, Moxiecode Systems AB
  9714   * Released under GPL License.
  9715   *
  9716   * License: http://www.plupload.com/license
  9717   * Contributing: http://www.plupload.com/contributing
  9718   */
  9719  
  9720  /**
  9721  @class moxie/runtime/silverlight/file/Blob
  9722  @private
  9723  */
  9724  define("moxie/runtime/silverlight/file/Blob", [
  9725  	"moxie/runtime/silverlight/Runtime",
  9726  	"moxie/core/utils/Basic",
  9727  	"moxie/runtime/flash/file/Blob"
  9728  ], function(extensions, Basic, Blob) {
  9729  	return (extensions.Blob = Basic.extend({}, Blob));
  9730  });
  9731  
  9732  // Included from: src/javascript/runtime/silverlight/file/FileInput.js
  9733  
  9734  /**
  9735   * FileInput.js
  9736   *
  9737   * Copyright 2013, Moxiecode Systems AB
  9738   * Released under GPL License.
  9739   *
  9740   * License: http://www.plupload.com/license
  9741   * Contributing: http://www.plupload.com/contributing
  9742   */
  9743  
  9744  /**
  9745  @class moxie/runtime/silverlight/file/FileInput
  9746  @private
  9747  */
  9748  define("moxie/runtime/silverlight/file/FileInput", [
  9749  	"moxie/runtime/silverlight/Runtime"
  9750  ], function(extensions) {
  9751  	
  9752  	var FileInput = {
  9753  		init: function(options) {
  9754  
  9755  			function toFilters(accept) {
  9756  				var filter = '';
  9757  				for (var i = 0; i < accept.length; i++) {
  9758  					filter += (filter !== '' ? '|' : '') + accept[i].title + " | *." + accept[i].extensions.replace(/,/g, ';*.');
  9759  				}
  9760  				return filter;
  9761  			}
  9762  			
  9763  			this.getRuntime().shimExec.call(this, 'FileInput', 'init', toFilters(options.accept), options.name, options.multiple);
  9764  			this.trigger('ready');
  9765  		}
  9766  	};
  9767  
  9768  	return (extensions.FileInput = FileInput);
  9769  });
  9770  
  9771  // Included from: src/javascript/runtime/silverlight/file/FileDrop.js
  9772  
  9773  /**
  9774   * FileDrop.js
  9775   *
  9776   * Copyright 2013, Moxiecode Systems AB
  9777   * Released under GPL License.
  9778   *
  9779   * License: http://www.plupload.com/license
  9780   * Contributing: http://www.plupload.com/contributing
  9781   */
  9782  
  9783  /**
  9784  @class moxie/runtime/silverlight/file/FileDrop
  9785  @private
  9786  */
  9787  define("moxie/runtime/silverlight/file/FileDrop", [
  9788  	"moxie/runtime/silverlight/Runtime",
  9789  	"moxie/core/utils/Dom", 
  9790  	"moxie/core/utils/Events"
  9791  ], function(extensions, Dom, Events) {
  9792  
  9793  	// not exactly useful, since works only in safari (...crickets...)
  9794  	var FileDrop = {
  9795  		init: function() {
  9796  			var comp = this, self = comp.getRuntime(), dropZone;
  9797  
  9798  			dropZone = self.getShimContainer();
  9799  
  9800  			Events.addEvent(dropZone, 'dragover', function(e) {
  9801  				e.preventDefault();
  9802  				e.stopPropagation();
  9803  				e.dataTransfer.dropEffect = 'copy';
  9804  			}, comp.uid);
  9805  
  9806  			Events.addEvent(dropZone, 'dragenter', function(e) {
  9807  				e.preventDefault();
  9808  				var flag = Dom.get(self.uid).dragEnter(e);
  9809  				// If handled, then stop propagation of event in DOM
  9810  				if (flag) {
  9811  					e.stopPropagation();
  9812  				}
  9813  			}, comp.uid);
  9814  
  9815  			Events.addEvent(dropZone, 'drop', function(e) {
  9816  				e.preventDefault();
  9817  				var flag = Dom.get(self.uid).dragDrop(e);
  9818  				// If handled, then stop propagation of event in DOM
  9819  				if (flag) {
  9820  					e.stopPropagation();
  9821  				}
  9822  			}, comp.uid);
  9823  
  9824  			return self.shimExec.call(this, 'FileDrop', 'init');
  9825  		}
  9826  	};
  9827  
  9828  	return (extensions.FileDrop = FileDrop);
  9829  });
  9830  
  9831  // Included from: src/javascript/runtime/silverlight/file/FileReader.js
  9832  
  9833  /**
  9834   * FileReader.js
  9835   *
  9836   * Copyright 2013, Moxiecode Systems AB
  9837   * Released under GPL License.
  9838   *
  9839   * License: http://www.plupload.com/license
  9840   * Contributing: http://www.plupload.com/contributing
  9841   */
  9842  
  9843  /**
  9844  @class moxie/runtime/silverlight/file/FileReader
  9845  @private
  9846  */
  9847  define("moxie/runtime/silverlight/file/FileReader", [
  9848  	"moxie/runtime/silverlight/Runtime",
  9849  	"moxie/core/utils/Basic",
  9850  	"moxie/runtime/flash/file/FileReader"
  9851  ], function(extensions, Basic, FileReader) {
  9852  	return (extensions.FileReader = Basic.extend({}, FileReader));
  9853  });
  9854  
  9855  // Included from: src/javascript/runtime/silverlight/file/FileReaderSync.js
  9856  
  9857  /**
  9858   * FileReaderSync.js
  9859   *
  9860   * Copyright 2013, Moxiecode Systems AB
  9861   * Released under GPL License.
  9862   *
  9863   * License: http://www.plupload.com/license
  9864   * Contributing: http://www.plupload.com/contributing
  9865   */
  9866  
  9867  /**
  9868  @class moxie/runtime/silverlight/file/FileReaderSync
  9869  @private
  9870  */
  9871  define("moxie/runtime/silverlight/file/FileReaderSync", [
  9872  	"moxie/runtime/silverlight/Runtime",
  9873  	"moxie/core/utils/Basic",
  9874  	"moxie/runtime/flash/file/FileReaderSync"
  9875  ], function(extensions, Basic, FileReaderSync) {
  9876  	return (extensions.FileReaderSync = Basic.extend({}, FileReaderSync));
  9877  });
  9878  
  9879  // Included from: src/javascript/runtime/silverlight/xhr/XMLHttpRequest.js
  9880  
  9881  /**
  9882   * XMLHttpRequest.js
  9883   *
  9884   * Copyright 2013, Moxiecode Systems AB
  9885   * Released under GPL License.
  9886   *
  9887   * License: http://www.plupload.com/license
  9888   * Contributing: http://www.plupload.com/contributing
  9889   */
  9890  
  9891  /**
  9892  @class moxie/runtime/silverlight/xhr/XMLHttpRequest
  9893  @private
  9894  */
  9895  define("moxie/runtime/silverlight/xhr/XMLHttpRequest", [
  9896  	"moxie/runtime/silverlight/Runtime",
  9897  	"moxie/core/utils/Basic",
  9898  	"moxie/runtime/flash/xhr/XMLHttpRequest"
  9899  ], function(extensions, Basic, XMLHttpRequest) {
  9900  	return (extensions.XMLHttpRequest = Basic.extend({}, XMLHttpRequest));
  9901  });
  9902  
  9903  // Included from: src/javascript/runtime/silverlight/runtime/Transporter.js
  9904  
  9905  /**
  9906   * Transporter.js
  9907   *
  9908   * Copyright 2013, Moxiecode Systems AB
  9909   * Released under GPL License.
  9910   *
  9911   * License: http://www.plupload.com/license
  9912   * Contributing: http://www.plupload.com/contributing
  9913   */
  9914  
  9915  /**
  9916  @class moxie/runtime/silverlight/runtime/Transporter
  9917  @private
  9918  */
  9919  define("moxie/runtime/silverlight/runtime/Transporter", [
  9920  	"moxie/runtime/silverlight/Runtime",
  9921  	"moxie/core/utils/Basic",
  9922  	"moxie/runtime/flash/runtime/Transporter"
  9923  ], function(extensions, Basic, Transporter) {
  9924  	return (extensions.Transporter = Basic.extend({}, Transporter));
  9925  });
  9926  
  9927  // Included from: src/javascript/runtime/silverlight/image/Image.js
  9928  
  9929  /**
  9930   * Image.js
  9931   *
  9932   * Copyright 2013, Moxiecode Systems AB
  9933   * Released under GPL License.
  9934   *
  9935   * License: http://www.plupload.com/license
  9936   * Contributing: http://www.plupload.com/contributing
  9937   */
  9938   
  9939  /**
  9940  @class moxie/runtime/silverlight/image/Image
  9941  @private
  9942  */
  9943  define("moxie/runtime/silverlight/image/Image", [
  9944  	"moxie/runtime/silverlight/Runtime",
  9945  	"moxie/core/utils/Basic",
  9946  	"moxie/runtime/flash/image/Image"
  9947  ], function(extensions, Basic, Image) {
  9948  	return (extensions.Image = Basic.extend({}, Image, {
  9949  
  9950  		getInfo: function() {
  9951  			var self = this.getRuntime()
  9952  			, grps = ['tiff', 'exif', 'gps']
  9953  			, info = { meta: {} }
  9954  			, rawInfo = self.shimExec.call(this, 'Image', 'getInfo')
  9955  			;
  9956  
  9957  			if (rawInfo.meta) {
  9958  				Basic.each(grps, function(grp) {
  9959  					var meta = rawInfo.meta[grp]
  9960  					, tag
  9961  					, i
  9962  					, length
  9963  					, value
  9964  					;
  9965  					if (meta && meta.keys) {
  9966  						info.meta[grp] = {};
  9967  						for (i = 0, length = meta.keys.length; i < length; i++) {
  9968  							tag = meta.keys[i];
  9969  							value = meta[tag];
  9970  							if (value) {
  9971  								// convert numbers
  9972  								if (/^(\d|[1-9]\d+)$/.test(value)) { // integer (make sure doesn't start with zero)
  9973  									value = parseInt(value, 10);
  9974  								} else if (/^\d*\.\d+$/.test(value)) { // double
  9975  									value = parseFloat(value);
  9976  								}
  9977  								info.meta[grp][tag] = value;
  9978  							}
  9979  						}
  9980  					}
  9981  				});
  9982  			}
  9983  
  9984  			info.width = parseInt(rawInfo.width, 10);
  9985  			info.height = parseInt(rawInfo.height, 10);
  9986  			info.size = parseInt(rawInfo.size, 10);
  9987  			info.type = rawInfo.type;
  9988  			info.name = rawInfo.name;
  9989  
  9990  			return info;
  9991  		}
  9992  	}));
  9993  });
  9994  
  9995  // Included from: src/javascript/runtime/html4/Runtime.js
  9996  
  9997  /**
  9998   * Runtime.js
  9999   *
 10000   * Copyright 2013, Moxiecode Systems AB
 10001   * Released under GPL License.
 10002   *
 10003   * License: http://www.plupload.com/license
 10004   * Contributing: http://www.plupload.com/contributing
 10005   */
 10006  
 10007  /*global File:true */
 10008  
 10009  /**
 10010  Defines constructor for HTML4 runtime.
 10011  
 10012  @class moxie/runtime/html4/Runtime
 10013  @private
 10014  */
 10015  define("moxie/runtime/html4/Runtime", [
 10016  	"moxie/core/utils/Basic",
 10017  	"moxie/core/Exceptions",
 10018  	"moxie/runtime/Runtime",
 10019  	"moxie/core/utils/Env"
 10020  ], function(Basic, x, Runtime, Env) {
 10021  	
 10022  	var type = 'html4', extensions = {};
 10023  
 10024  	function Html4Runtime(options) {
 10025  		var I = this
 10026  		, Test = Runtime.capTest
 10027  		, True = Runtime.capTrue
 10028  		;
 10029  
 10030  		Runtime.call(this, options, type, {
 10031  			access_binary: Test(window.FileReader || window.File && File.getAsDataURL),
 10032  			access_image_binary: false,
 10033  			display_media: Test(extensions.Image && (Env.can('create_canvas') || Env.can('use_data_uri_over32kb'))),
 10034  			do_cors: false,
 10035  			drag_and_drop: false,
 10036  			filter_by_extension: Test(function() { // if you know how to feature-detect this, please suggest
 10037  				return (Env.browser === 'Chrome' && Env.version >= 28) || (Env.browser === 'IE' && Env.version >= 10);
 10038  			}()),
 10039  			resize_image: function() {
 10040  				return extensions.Image && I.can('access_binary') && Env.can('create_canvas');
 10041  			},
 10042  			report_upload_progress: false,
 10043  			return_response_headers: false,
 10044  			return_response_type: function(responseType) {
 10045  				if (responseType === 'json' && !!window.JSON) {
 10046  					return true;
 10047  				} 
 10048  				return !!~Basic.inArray(responseType, ['text', 'document', '']);
 10049  			},
 10050  			return_status_code: function(code) {
 10051  				return !Basic.arrayDiff(code, [200, 404]);
 10052  			},
 10053  			select_file: function() {
 10054  				return Env.can('use_fileinput');
 10055  			},
 10056  			select_multiple: false,
 10057  			send_binary_string: false,
 10058  			send_custom_headers: false,
 10059  			send_multipart: true,
 10060  			slice_blob: false,
 10061  			stream_upload: function() {
 10062  				return I.can('select_file');
 10063  			},
 10064  			summon_file_dialog: Test(function() { // yeah... some dirty sniffing here...
 10065  				return (Env.browser === 'Firefox' && Env.version >= 4) ||
 10066  					(Env.browser === 'Opera' && Env.version >= 12) ||
 10067  					!!~Basic.inArray(Env.browser, ['Chrome', 'Safari']);
 10068  			}()),
 10069  			upload_filesize: True,
 10070  			use_http_method: function(methods) {
 10071  				return !Basic.arrayDiff(methods, ['GET', 'POST']);
 10072  			}
 10073  		});
 10074  
 10075  
 10076  		Basic.extend(this, {
 10077  			init : function() {
 10078  				this.trigger("Init");
 10079  			},
 10080  
 10081  			destroy: (function(destroy) { // extend default destroy method
 10082  				return function() {
 10083  					destroy.call(I);
 10084  					destroy = I = null;
 10085  				};
 10086  			}(this.destroy))
 10087  		});
 10088  
 10089  		Basic.extend(this.getShim(), extensions);
 10090  	}
 10091  
 10092  	Runtime.addConstructor(type, Html4Runtime);
 10093  
 10094  	return extensions;
 10095  });
 10096  
 10097  // Included from: src/javascript/runtime/html4/file/FileInput.js
 10098  
 10099  /**
 10100   * FileInput.js
 10101   *
 10102   * Copyright 2013, Moxiecode Systems AB
 10103   * Released under GPL License.
 10104   *
 10105   * License: http://www.plupload.com/license
 10106   * Contributing: http://www.plupload.com/contributing
 10107   */
 10108  
 10109  /**
 10110  @class moxie/runtime/html4/file/FileInput
 10111  @private
 10112  */
 10113  define("moxie/runtime/html4/file/FileInput", [
 10114  	"moxie/runtime/html4/Runtime",
 10115  	"moxie/core/utils/Basic",
 10116  	"moxie/core/utils/Dom",
 10117  	"moxie/core/utils/Events",
 10118  	"moxie/core/utils/Mime",
 10119  	"moxie/core/utils/Env"
 10120  ], function(extensions, Basic, Dom, Events, Mime, Env) {
 10121  	
 10122  	function FileInput() {
 10123  		var _uid, _files = [], _mimes = [], _options;
 10124  
 10125  		function addInput() {
 10126  			var comp = this, I = comp.getRuntime(), shimContainer, browseButton, currForm, form, input, uid;
 10127  
 10128  			uid = Basic.guid('uid_');
 10129  
 10130  			shimContainer = I.getShimContainer(); // we get new ref everytime to avoid memory leaks in IE
 10131  
 10132  			if (_uid) { // move previous form out of the view
 10133  				currForm = Dom.get(_uid + '_form');
 10134  				if (currForm) {
 10135  					Basic.extend(currForm.style, { top: '100%' });
 10136  				}
 10137  			}
 10138  
 10139  			// build form in DOM, since innerHTML version not able to submit file for some reason
 10140  			form = document.createElement('form');
 10141  			form.setAttribute('id', uid + '_form');
 10142  			form.setAttribute('method', 'post');
 10143  			form.setAttribute('enctype', 'multipart/form-data');
 10144  			form.setAttribute('encoding', 'multipart/form-data');
 10145  
 10146  			Basic.extend(form.style, {
 10147  				overflow: 'hidden',
 10148  				position: 'absolute',
 10149  				top: 0,
 10150  				left: 0,
 10151  				width: '100%',
 10152  				height: '100%'
 10153  			});
 10154  
 10155  			input = document.createElement('input');
 10156  			input.setAttribute('id', uid);
 10157  			input.setAttribute('type', 'file');
 10158  			input.setAttribute('name', _options.name || 'Filedata');
 10159  			input.setAttribute('accept', _mimes.join(','));
 10160  
 10161  			Basic.extend(input.style, {
 10162  				fontSize: '999px',
 10163  				opacity: 0
 10164  			});
 10165  
 10166  			form.appendChild(input);
 10167  			shimContainer.appendChild(form);
 10168  
 10169  			// prepare file input to be placed underneath the browse_button element
 10170  			Basic.extend(input.style, {
 10171  				position: 'absolute',
 10172  				top: 0,
 10173  				left: 0,
 10174  				width: '100%',
 10175  				height: '100%'
 10176  			});
 10177  
 10178  			if (Env.browser === 'IE' && Env.version < 10) {
 10179  				Basic.extend(input.style, {
 10180  					filter : "progid:DXImageTransform.Microsoft.Alpha(opacity=0)"
 10181  				});
 10182  			}
 10183  
 10184  			input.onchange = function() { // there should be only one handler for this
 10185  				var file;
 10186  
 10187  				if (!this.value) {
 10188  					return;
 10189  				}
 10190  
 10191  				if (this.files) {
 10192  					file = this.files[0];
 10193  				} else {
 10194  					file = {
 10195  						name: this.value
 10196  					};
 10197  				}
 10198  
 10199  				_files = [file];
 10200  
 10201  				this.onchange = function() {}; // clear event handler
 10202  				addInput.call(comp);
 10203  
 10204  				// after file is initialized as o.File, we need to update form and input ids
 10205  				comp.bind('change', function onChange() {
 10206  					var input = Dom.get(uid), form = Dom.get(uid + '_form'), file;
 10207  
 10208  					comp.unbind('change', onChange);
 10209  
 10210  					if (comp.files.length && input && form) {
 10211  						file = comp.files[0];
 10212  
 10213  						input.setAttribute('id', file.uid);
 10214  						form.setAttribute('id', file.uid + '_form');
 10215  
 10216  						// set upload target
 10217  						form.setAttribute('target', file.uid + '_iframe');
 10218  					}
 10219  					input = form = null;
 10220  				}, 998);
 10221  
 10222  				input = form = null;
 10223  				comp.trigger('change');
 10224  			};
 10225  
 10226  
 10227  			// route click event to the input
 10228  			if (I.can('summon_file_dialog')) {
 10229  				browseButton = Dom.get(_options.browse_button);
 10230  				Events.removeEvent(browseButton, 'click', comp.uid);
 10231  				Events.addEvent(browseButton, 'click', function(e) {
 10232  					if (input && !input.disabled) { // for some reason FF (up to 8.0.1 so far) lets to click disabled input[type=file]
 10233  						input.click();
 10234  					}
 10235  					e.preventDefault();
 10236  				}, comp.uid);
 10237  			}
 10238  
 10239  			_uid = uid;
 10240  
 10241  			shimContainer = currForm = browseButton = null;
 10242  		}
 10243  
 10244  		Basic.extend(this, {
 10245  			init: function(options) {
 10246  				var comp = this, I = comp.getRuntime(), shimContainer;
 10247  
 10248  				// figure out accept string
 10249  				_options = options;
 10250  				_mimes = options.accept.mimes || Mime.extList2mimes(options.accept, I.can('filter_by_extension'));
 10251  
 10252  				shimContainer = I.getShimContainer();
 10253  
 10254  				(function() {
 10255  					var browseButton, zIndex, top;
 10256  
 10257  					browseButton = Dom.get(options.browse_button);
 10258  
 10259  					// Route click event to the input[type=file] element for browsers that support such behavior
 10260  					if (I.can('summon_file_dialog')) {
 10261  						if (Dom.getStyle(browseButton, 'position') === 'static') {
 10262  							browseButton.style.position = 'relative';
 10263  						}
 10264  
 10265  						zIndex = parseInt(Dom.getStyle(browseButton, 'z-index'), 10) || 1;
 10266  
 10267  						browseButton.style.zIndex = zIndex;
 10268  						shimContainer.style.zIndex = zIndex - 1;
 10269  					}
 10270  
 10271  					/* Since we have to place input[type=file] on top of the browse_button for some browsers,
 10272  					browse_button loses interactivity, so we restore it here */
 10273  					top = I.can('summon_file_dialog') ? browseButton : shimContainer;
 10274  
 10275  					Events.addEvent(top, 'mouseover', function() {
 10276  						comp.trigger('mouseenter');
 10277  					}, comp.uid);
 10278  
 10279  					Events.addEvent(top, 'mouseout', function() {
 10280  						comp.trigger('mouseleave');
 10281  					}, comp.uid);
 10282  
 10283  					Events.addEvent(top, 'mousedown', function() {
 10284  						comp.trigger('mousedown');
 10285  					}, comp.uid);
 10286  
 10287  					Events.addEvent(Dom.get(options.container), 'mouseup', function() {
 10288  						comp.trigger('mouseup');
 10289  					}, comp.uid);
 10290  
 10291  					browseButton = null;
 10292  				}());
 10293  
 10294  				addInput.call(this);
 10295  
 10296  				shimContainer = null;
 10297  
 10298  				// trigger ready event asynchronously
 10299  				comp.trigger({
 10300  					type: 'ready',
 10301  					async: true
 10302  				});
 10303  			},
 10304  
 10305  			getFiles: function() {
 10306  				return _files;
 10307  			},
 10308  
 10309  			disable: function(state) {
 10310  				var input;
 10311  
 10312  				if ((input = Dom.get(_uid))) {
 10313  					input.disabled = !!state;
 10314  				}
 10315  			},
 10316  
 10317  			destroy: function() {
 10318  				var I = this.getRuntime()
 10319  				, shim = I.getShim()
 10320  				, shimContainer = I.getShimContainer()
 10321  				;
 10322  				
 10323  				Events.removeAllEvents(shimContainer, this.uid);
 10324  				Events.removeAllEvents(_options && Dom.get(_options.container), this.uid);
 10325  				Events.removeAllEvents(_options && Dom.get(_options.browse_button), this.uid);
 10326  				
 10327  				if (shimContainer) {
 10328  					shimContainer.innerHTML = '';
 10329  				}
 10330  
 10331  				shim.removeInstance(this.uid);
 10332  
 10333  				_uid = _files = _mimes = _options = shimContainer = shim = null;
 10334  			}
 10335  		});
 10336  	}
 10337  
 10338  	return (extensions.FileInput = FileInput);
 10339  });
 10340  
 10341  // Included from: src/javascript/runtime/html4/file/FileReader.js
 10342  
 10343  /**
 10344   * FileReader.js
 10345   *
 10346   * Copyright 2013, Moxiecode Systems AB
 10347   * Released under GPL License.
 10348   *
 10349   * License: http://www.plupload.com/license
 10350   * Contributing: http://www.plupload.com/contributing
 10351   */
 10352  
 10353  /**
 10354  @class moxie/runtime/html4/file/FileReader
 10355  @private
 10356  */
 10357  define("moxie/runtime/html4/file/FileReader", [
 10358  	"moxie/runtime/html4/Runtime",
 10359  	"moxie/runtime/html5/file/FileReader"
 10360  ], function(extensions, FileReader) {
 10361  	return (extensions.FileReader = FileReader);
 10362  });
 10363  
 10364  // Included from: src/javascript/runtime/html4/xhr/XMLHttpRequest.js
 10365  
 10366  /**
 10367   * XMLHttpRequest.js
 10368   *
 10369   * Copyright 2013, Moxiecode Systems AB
 10370   * Released under GPL License.
 10371   *
 10372   * License: http://www.plupload.com/license
 10373   * Contributing: http://www.plupload.com/contributing
 10374   */
 10375  
 10376  /**
 10377  @class moxie/runtime/html4/xhr/XMLHttpRequest
 10378  @private
 10379  */
 10380  define("moxie/runtime/html4/xhr/XMLHttpRequest", [
 10381  	"moxie/runtime/html4/Runtime",
 10382  	"moxie/core/utils/Basic",
 10383  	"moxie/core/utils/Dom",
 10384  	"moxie/core/utils/Url",
 10385  	"moxie/core/Exceptions",
 10386  	"moxie/core/utils/Events",
 10387  	"moxie/file/Blob",
 10388  	"moxie/xhr/FormData"
 10389  ], function(extensions, Basic, Dom, Url, x, Events, Blob, FormData) {
 10390  	
 10391  	function XMLHttpRequest() {
 10392  		var _status, _response, _iframe;
 10393  
 10394  		function cleanup(cb) {
 10395  			var target = this, uid, form, inputs, i, hasFile = false;
 10396  
 10397  			if (!_iframe) {
 10398  				return;
 10399  			}
 10400  
 10401  			uid = _iframe.id.replace(/_iframe$/, '');
 10402  
 10403  			form = Dom.get(uid + '_form');
 10404  			if (form) {
 10405  				inputs = form.getElementsByTagName('input');
 10406  				i = inputs.length;
 10407  
 10408  				while (i--) {
 10409  					switch (inputs[i].getAttribute('type')) {
 10410  						case 'hidden':
 10411  							inputs[i].parentNode.removeChild(inputs[i]);
 10412  							break;
 10413  						case 'file':
 10414  							hasFile = true; // flag the case for later
 10415  							break;
 10416  					}
 10417  				}
 10418  				inputs = [];
 10419  
 10420  				if (!hasFile) { // we need to keep the form for sake of possible retries
 10421  					form.parentNode.removeChild(form);
 10422  				}
 10423  				form = null;
 10424  			}
 10425  
 10426  			// without timeout, request is marked as canceled (in console)
 10427  			setTimeout(function() {
 10428  				Events.removeEvent(_iframe, 'load', target.uid);
 10429  				if (_iframe.parentNode) { // #382
 10430  					_iframe.parentNode.removeChild(_iframe);
 10431  				}
 10432  
 10433  				// check if shim container has any other children, if - not, remove it as well
 10434  				var shimContainer = target.getRuntime().getShimContainer();
 10435  				if (!shimContainer.children.length) {
 10436  					shimContainer.parentNode.removeChild(shimContainer);
 10437  				}
 10438  
 10439  				shimContainer = _iframe = null;
 10440  				cb();
 10441  			}, 1);
 10442  		}
 10443  
 10444  		Basic.extend(this, {
 10445  			send: function(meta, data) {
 10446  				var target = this, I = target.getRuntime(), uid, form, input, blob;
 10447  
 10448  				_status = _response = null;
 10449  
 10450  				function createIframe() {
 10451  					var container = I.getShimContainer() || document.body
 10452  					, temp = document.createElement('div')
 10453  					;
 10454  
 10455  					// IE 6 won't be able to set the name using setAttribute or iframe.name
 10456  					temp.innerHTML = '<iframe id="' + uid + '_iframe" name="' + uid + '_iframe" src="javascript:&quot;&quot;" style="display:none"></iframe>';
 10457  					_iframe = temp.firstChild;
 10458  					container.appendChild(_iframe);
 10459  
 10460  					/* _iframe.onreadystatechange = function() {
 10461  						console.info(_iframe.readyState);
 10462  					};*/
 10463  
 10464  					Events.addEvent(_iframe, 'load', function() { // _iframe.onload doesn't work in IE lte 8
 10465  						var el;
 10466  
 10467  						try {
 10468  							el = _iframe.contentWindow.document || _iframe.contentDocument || window.frames[_iframe.id].document;
 10469  
 10470  							// try to detect some standard error pages
 10471  							if (/^4(0[0-9]|1[0-7]|2[2346])\s/.test(el.title)) { // test if title starts with 4xx HTTP error
 10472  								_status = el.title.replace(/^(\d+).*$/, '$1');
 10473  							} else {
 10474  								_status = 200;
 10475  								// get result
 10476  								_response = Basic.trim(el.body.innerHTML);
 10477  
 10478  								// we need to fire these at least once
 10479  								target.trigger({
 10480  									type: 'progress',
 10481  									loaded: _response.length,
 10482  									total: _response.length
 10483  								});
 10484  
 10485  								if (blob) { // if we were uploading a file
 10486  									target.trigger({
 10487  										type: 'uploadprogress',
 10488  										loaded: blob.size || 1025,
 10489  										total: blob.size || 1025
 10490  									});
 10491  								}
 10492  							}
 10493  						} catch (ex) {
 10494  							if (Url.hasSameOrigin(meta.url)) {
 10495  								// if response is sent with error code, iframe in IE gets redirected to res://ieframe.dll/http_x.htm
 10496  								// which obviously results to cross domain error (wtf?)
 10497  								_status = 404;
 10498  							} else {
 10499  								cleanup.call(target, function() {
 10500  									target.trigger('error');
 10501  								});
 10502  								return;
 10503  							}
 10504  						}	
 10505  					
 10506  						cleanup.call(target, function() {
 10507  							target.trigger('load');
 10508  						});
 10509  					}, target.uid);
 10510  				} // end createIframe
 10511  
 10512  				// prepare data to be sent and convert if required
 10513  				if (data instanceof FormData && data.hasBlob()) {
 10514  					blob = data.getBlob();
 10515  					uid = blob.uid;
 10516  					input = Dom.get(uid);
 10517  					form = Dom.get(uid + '_form');
 10518  					if (!form) {
 10519  						throw new x.DOMException(x.DOMException.NOT_FOUND_ERR);
 10520  					}
 10521  				} else {
 10522  					uid = Basic.guid('uid_');
 10523  
 10524  					form = document.createElement('form');
 10525  					form.setAttribute('id', uid + '_form');
 10526  					form.setAttribute('method', meta.method);
 10527  					form.setAttribute('enctype', 'multipart/form-data');
 10528  					form.setAttribute('encoding', 'multipart/form-data');
 10529  					form.setAttribute('target', uid + '_iframe');
 10530  
 10531  					I.getShimContainer().appendChild(form);
 10532  				}
 10533  
 10534  				if (data instanceof FormData) {
 10535  					data.each(function(value, name) {
 10536  						if (value instanceof Blob) {
 10537  							if (input) {
 10538  								input.setAttribute('name', name);
 10539  							}
 10540  						} else {
 10541  							var hidden = document.createElement('input');
 10542  
 10543  							Basic.extend(hidden, {
 10544  								type : 'hidden',
 10545  								name : name,
 10546  								value : value
 10547  							});
 10548  
 10549  							// make sure that input[type="file"], if it's there, comes last
 10550  							if (input) {
 10551  								form.insertBefore(hidden, input);
 10552  							} else {
 10553  								form.appendChild(hidden);
 10554  							}
 10555  						}
 10556  					});
 10557  				}
 10558  
 10559  				// set destination url
 10560  				form.setAttribute("action", meta.url);
 10561  
 10562  				createIframe();
 10563  				form.submit();
 10564  				target.trigger('loadstart');
 10565  			},
 10566  
 10567  			getStatus: function() {
 10568  				return _status;
 10569  			},
 10570  
 10571  			getResponse: function(responseType) {
 10572  				if ('json' === responseType) {
 10573  					// strip off <pre>..</pre> tags that might be enclosing the response
 10574  					if (Basic.typeOf(_response) === 'string' && !!window.JSON) {
 10575  						try {
 10576  							return JSON.parse(_response.replace(/^\s*<pre[^>]*>/, '').replace(/<\/pre>\s*$/, ''));
 10577  						} catch (ex) {
 10578  							return null;
 10579  						}
 10580  					} 
 10581  				} else if ('document' === responseType) {
 10582  
 10583  				}
 10584  				return _response;
 10585  			},
 10586  
 10587  			abort: function() {
 10588  				var target = this;
 10589  
 10590  				if (_iframe && _iframe.contentWindow) {
 10591  					if (_iframe.contentWindow.stop) { // FireFox/Safari/Chrome
 10592  						_iframe.contentWindow.stop();
 10593  					} else if (_iframe.contentWindow.document.execCommand) { // IE
 10594  						_iframe.contentWindow.document.execCommand('Stop');
 10595  					} else {
 10596  						_iframe.src = "about:blank";
 10597  					}
 10598  				}
 10599  
 10600  				cleanup.call(this, function() {
 10601  					// target.dispatchEvent('readystatechange');
 10602  					target.dispatchEvent('abort');
 10603  				});
 10604  			}
 10605  		});
 10606  	}
 10607  
 10608  	return (extensions.XMLHttpRequest = XMLHttpRequest);
 10609  });
 10610  
 10611  // Included from: src/javascript/runtime/html4/image/Image.js
 10612  
 10613  /**
 10614   * Image.js
 10615   *
 10616   * Copyright 2013, Moxiecode Systems AB
 10617   * Released under GPL License.
 10618   *
 10619   * License: http://www.plupload.com/license
 10620   * Contributing: http://www.plupload.com/contributing
 10621   */
 10622  
 10623  /**
 10624  @class moxie/runtime/html4/image/Image
 10625  @private
 10626  */
 10627  define("moxie/runtime/html4/image/Image", [
 10628  	"moxie/runtime/html4/Runtime",
 10629  	"moxie/runtime/html5/image/Image"
 10630  ], function(extensions, Image) {
 10631  	return (extensions.Image = Image);
 10632  });
 10633  
 10634  expose(["moxie/core/utils/Basic","moxie/core/I18n","moxie/core/utils/Mime","moxie/core/utils/Env","moxie/core/utils/Dom","moxie/core/Exceptions","moxie/core/EventTarget","moxie/core/utils/Encode","moxie/runtime/Runtime","moxie/runtime/RuntimeClient","moxie/file/Blob","moxie/file/File","moxie/file/FileInput","moxie/file/FileDrop","moxie/runtime/RuntimeTarget","moxie/file/FileReader","moxie/core/utils/Url","moxie/file/FileReaderSync","moxie/xhr/FormData","moxie/xhr/XMLHttpRequest","moxie/runtime/Transporter","moxie/image/Image","moxie/core/utils/Events"]);
 10635  })(this);/**
 10636   * o.js
 10637   *
 10638   * Copyright 2013, Moxiecode Systems AB
 10639   * Released under GPL License.
 10640   *
 10641   * License: http://www.plupload.com/license
 10642   * Contributing: http://www.plupload.com/contributing
 10643   */
 10644  
 10645  /*global moxie:true */
 10646  
 10647  /**
 10648  Globally exposed namespace with the most frequently used public classes and handy methods.
 10649  
 10650  @class o
 10651  @static
 10652  @private
 10653  */
 10654  (function() {
 10655  	"use strict";
 10656  
 10657  	var o = {}, inArray = moxie.core.utils.Basic.inArray;
 10658  
 10659  	// directly add some public classes
 10660  	// (we do it dynamically here, since for custom builds we cannot know beforehand what modules were included)
 10661  	(function addAlias(ns) {
 10662  		var name, itemType;
 10663  		for (name in ns) {
 10664  			itemType = typeof(ns[name]);
 10665  			if (itemType === 'object' && !~inArray(name, ['Exceptions', 'Env', 'Mime'])) {
 10666  				addAlias(ns[name]);
 10667  			} else if (itemType === 'function') {
 10668  				o[name] = ns[name];
 10669  			}
 10670  		}
 10671  	})(window.moxie);
 10672  
 10673  	// add some manually
 10674  	o.Env = window.moxie.core.utils.Env;
 10675  	o.Mime = window.moxie.core.utils.Mime;
 10676  	o.Exceptions = window.moxie.core.Exceptions;
 10677  
 10678  	// expose globally
 10679  	window.mOxie = o;
 10680  	if (!window.o) {
 10681  		window.o = o;
 10682  	}
 10683  	return o;
 10684  })();