| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346 | /*	Speed Plug-in		Features:		*Adds several properties to the 'file' object indicated upload speed, time left, upload time, etc.			- currentSpeed -- String indicating the upload speed, bits per second			- averageSpeed -- Overall average upload speed, bits per second			- movingAverageSpeed -- Speed over averaged over the last several measurements, bits per second			- timeRemaining -- Estimated remaining upload time in seconds			- timeElapsed -- Number of seconds passed for this upload			- percentUploaded -- Percentage of the file uploaded (0 to 100)			- sizeUploaded -- Formatted size uploaded so far, bytes				*Adds setting 'moving_average_history_size' for defining the window size used to calculate the moving average speed.				*Adds several Formatting functions for formatting that values provided on the file object.			- SWFUpload.speed.formatBPS(bps) -- outputs string formatted in the best units (Gbps, Mbps, Kbps, bps)			- SWFUpload.speed.formatTime(seconds) -- outputs string formatted in the best units (x Hr y M z S)			- SWFUpload.speed.formatSize(bytes) -- outputs string formatted in the best units (w GB x MB y KB z B )			- SWFUpload.speed.formatPercent(percent) -- outputs string formatted with a percent sign (x.xx %)			- SWFUpload.speed.formatUnits(baseNumber, divisionArray, unitLabelArray, fractionalBoolean)				- Formats a number using the division array to determine how to apply the labels in the Label Array				- factionalBoolean indicates whether the number should be returned as a single fractional number with a unit (speed)				    or as several numbers labeled with units (time)	*/var SWFUpload;if (typeof(SWFUpload) === "function") {	SWFUpload.speed = {};		SWFUpload.prototype.initSettings = (function (oldInitSettings) {		return function (userSettings) {			if (typeof(oldInitSettings) === "function") {				oldInitSettings.call(this, userSettings);			}						this.ensureDefault = function (settingName, defaultValue) {				this.settings[settingName] = (userSettings[settingName] == undefined) ? defaultValue : userSettings[settingName];			};			// List used to keep the speed stats for the files we are tracking			this.fileSpeedStats = {};			this.speedSettings = {};			this.ensureDefault("moving_average_history_size", "10");						this.speedSettings.user_file_queued_handler = this.settings.file_queued_handler;			this.speedSettings.user_file_queue_error_handler = this.settings.file_queue_error_handler;			this.speedSettings.user_upload_start_handler = this.settings.upload_start_handler;			this.speedSettings.user_upload_error_handler = this.settings.upload_error_handler;			this.speedSettings.user_upload_progress_handler = this.settings.upload_progress_handler;			this.speedSettings.user_upload_success_handler = this.settings.upload_success_handler;			this.speedSettings.user_upload_complete_handler = this.settings.upload_complete_handler;						this.settings.file_queued_handler = SWFUpload.speed.fileQueuedHandler;			this.settings.file_queue_error_handler = SWFUpload.speed.fileQueueErrorHandler;			this.settings.upload_start_handler = SWFUpload.speed.uploadStartHandler;			this.settings.upload_error_handler = SWFUpload.speed.uploadErrorHandler;			this.settings.upload_progress_handler = SWFUpload.speed.uploadProgressHandler;			this.settings.upload_success_handler = SWFUpload.speed.uploadSuccessHandler;			this.settings.upload_complete_handler = SWFUpload.speed.uploadCompleteHandler;						delete this.ensureDefault;		};	})(SWFUpload.prototype.initSettings);		SWFUpload.speed.fileQueuedHandler = function (file) {		if (typeof this.speedSettings.user_file_queued_handler === "function") {			file = SWFUpload.speed.extendFile(file);						return this.speedSettings.user_file_queued_handler.call(this, file);		}	};		SWFUpload.speed.fileQueueErrorHandler = function (file, errorCode, message) {		if (typeof this.speedSettings.user_file_queue_error_handler === "function") {			file = SWFUpload.speed.extendFile(file);						return this.speedSettings.user_file_queue_error_handler.call(this, file, errorCode, message);		}	};	SWFUpload.speed.uploadStartHandler = function (file) {		if (typeof this.speedSettings.user_upload_start_handler === "function") {			file = SWFUpload.speed.extendFile(file, this.fileSpeedStats);			return this.speedSettings.user_upload_start_handler.call(this, file);		}	};		SWFUpload.speed.uploadErrorHandler = function (file, errorCode, message) {		file = SWFUpload.speed.extendFile(file, this.fileSpeedStats);		SWFUpload.speed.removeTracking(file, this.fileSpeedStats);		if (typeof this.speedSettings.user_upload_error_handler === "function") {			return this.speedSettings.user_upload_error_handler.call(this, file, errorCode, message);		}	};	SWFUpload.speed.uploadProgressHandler = function (file, bytesComplete, bytesTotal) {		this.updateTracking(file, bytesComplete);		file = SWFUpload.speed.extendFile(file, this.fileSpeedStats);		if (typeof this.speedSettings.user_upload_progress_handler === "function") {			return this.speedSettings.user_upload_progress_handler.call(this, file, bytesComplete, bytesTotal);		}	};		SWFUpload.speed.uploadSuccessHandler = function (file, serverData) {		if (typeof this.speedSettings.user_upload_success_handler === "function") {			file = SWFUpload.speed.extendFile(file, this.fileSpeedStats);			return this.speedSettings.user_upload_success_handler.call(this, file, serverData);		}	};	SWFUpload.speed.uploadCompleteHandler = function (file) {		file = SWFUpload.speed.extendFile(file, this.fileSpeedStats);		SWFUpload.speed.removeTracking(file, this.fileSpeedStats);		if (typeof this.speedSettings.user_upload_complete_handler === "function") {			return this.speedSettings.user_upload_complete_handler.call(this, file);		}	};		// Private: extends the file object with the speed plugin values	SWFUpload.speed.extendFile = function (file, trackingList) {		var tracking;				if (!file) {			return file;		}				if (trackingList) {			tracking = trackingList[file.id];		}				if (tracking) {			file.currentSpeed = tracking.currentSpeed;			file.averageSpeed = tracking.averageSpeed;			file.movingAverageSpeed = tracking.movingAverageSpeed;			file.timeRemaining = tracking.timeRemaining;			file.timeElapsed = tracking.timeElapsed;			file.percentUploaded = tracking.percentUploaded;			file.sizeUploaded = tracking.bytesUploaded;		} else {			file.currentSpeed = 0;			file.averageSpeed = 0;			file.movingAverageSpeed = 0;			file.timeRemaining = 0;			file.timeElapsed = 0;			file.percentUploaded = 0;			file.sizeUploaded = 0;		}				return file;	};		// Private: Updates the speed tracking object, or creates it if necessary	SWFUpload.prototype.updateTracking = function (file, bytesUploaded) {		var tracking = this.fileSpeedStats[file.id];		if (!tracking) {			this.fileSpeedStats[file.id] = tracking = {};		}				// Sanity check inputs		bytesUploaded = bytesUploaded || tracking.bytesUploaded || 0;		if (bytesUploaded < 0) {			bytesUploaded = 0;		}		if (bytesUploaded > file.size) {			bytesUploaded = file.size;		}				var tickTime = (new Date()).getTime();		if (!tracking.startTime) {			tracking.startTime = (new Date()).getTime();			tracking.lastTime = tracking.startTime;			tracking.currentSpeed = 0;			tracking.averageSpeed = 0;			tracking.movingAverageSpeed = 0;			tracking.movingAverageHistory = [];			tracking.timeRemaining = 0;			tracking.timeElapsed = 0;			tracking.percentUploaded = bytesUploaded / file.size;			tracking.bytesUploaded = bytesUploaded;		} else if (tracking.startTime > tickTime) {			this.debug("When backwards in time");		} else {			// Get time and deltas			var now = (new Date()).getTime();			var lastTime = tracking.lastTime;			var deltaTime = now - lastTime;			var deltaBytes = bytesUploaded - tracking.bytesUploaded;						if (deltaBytes === 0 || deltaTime === 0) {				return tracking;			}						// Update tracking object			tracking.lastTime = now;			tracking.bytesUploaded = bytesUploaded;						// Calculate speeds			tracking.currentSpeed = (deltaBytes * 8 ) / (deltaTime / 1000);			tracking.averageSpeed = (tracking.bytesUploaded * 8) / ((now - tracking.startTime) / 1000);			// Calculate moving average			tracking.movingAverageHistory.push(tracking.currentSpeed);			if (tracking.movingAverageHistory.length > this.settings.moving_average_history_size) {				tracking.movingAverageHistory.shift();			}						tracking.movingAverageSpeed = SWFUpload.speed.calculateMovingAverage(tracking.movingAverageHistory);						// Update times			tracking.timeRemaining = (file.size - tracking.bytesUploaded) * 8 / tracking.movingAverageSpeed;			tracking.timeElapsed = (now - tracking.startTime) / 1000;						// Update percent			tracking.percentUploaded = (tracking.bytesUploaded / file.size * 100);		}				return tracking;	};	SWFUpload.speed.removeTracking = function (file, trackingList) {		try {			trackingList[file.id] = null;			delete trackingList[file.id];		} catch (ex) {		}	};		SWFUpload.speed.formatUnits = function (baseNumber, unitDivisors, unitLabels, singleFractional) {		var i, unit, unitDivisor, unitLabel;		if (baseNumber === 0) {			return "0 " + unitLabels[unitLabels.length - 1];		}				if (singleFractional) {			unit = baseNumber;			unitLabel = unitLabels.length >= unitDivisors.length ? unitLabels[unitDivisors.length - 1] : "";			for (i = 0; i < unitDivisors.length; i++) {				if (baseNumber >= unitDivisors[i]) {					unit = (baseNumber / unitDivisors[i]).toFixed(2);					unitLabel = unitLabels.length >= i ? " " + unitLabels[i] : "";					break;				}			}						return unit + unitLabel;		} else {			var formattedStrings = [];			var remainder = baseNumber;						for (i = 0; i < unitDivisors.length; i++) {				unitDivisor = unitDivisors[i];				unitLabel = unitLabels.length > i ? " " + unitLabels[i] : "";								unit = remainder / unitDivisor;				if (i < unitDivisors.length -1) {					unit = Math.floor(unit);				} else {					unit = unit.toFixed(2);				}				if (unit > 0) {					remainder = remainder % unitDivisor;										formattedStrings.push(unit + unitLabel);				}			}						return formattedStrings.join(" ");		}	};		SWFUpload.speed.formatBPS = function (baseNumber) {		var bpsUnits = [1073741824, 1048576, 1024, 1], bpsUnitLabels = ["Gbps", "Mbps", "Kbps", "bps"];		return SWFUpload.speed.formatUnits(baseNumber, bpsUnits, bpsUnitLabels, true);		};	SWFUpload.speed.formatTime = function (baseNumber) {		var timeUnits = [86400, 3600, 60, 1], timeUnitLabels = ["d", "h", "m", "s"];		return SWFUpload.speed.formatUnits(baseNumber, timeUnits, timeUnitLabels, false);		};	SWFUpload.speed.formatBytes = function (baseNumber) {		var sizeUnits = [1073741824, 1048576, 1024, 1], sizeUnitLabels = ["GB", "MB", "KB", "bytes"];		return SWFUpload.speed.formatUnits(baseNumber, sizeUnits, sizeUnitLabels, true);		};	SWFUpload.speed.formatPercent = function (baseNumber) {		return baseNumber.toFixed(2) + " %";	};		SWFUpload.speed.calculateMovingAverage = function (history) {		var vals = [], size, sum = 0.0, mean = 0.0, varianceTemp = 0.0, variance = 0.0, standardDev = 0.0;		var i;		var mSum = 0, mCount = 0;				size = history.length;				// Check for sufficient data		if (size >= 8) {			// Clone the array and Calculate sum of the values 			for (i = 0; i < size; i++) {				vals[i] = history[i];				sum += vals[i];			}			mean = sum / size;			// Calculate variance for the set			for (i = 0; i < size; i++) {				varianceTemp += Math.pow((vals[i] - mean), 2);			}			variance = varianceTemp / size;			standardDev = Math.sqrt(variance);						//Standardize the Data			for (i = 0; i < size; i++) {				vals[i] = (vals[i] - mean) / standardDev;			}			// Calculate the average excluding outliers			var deviationRange = 2.0;			for (i = 0; i < size; i++) {								if (vals[i] <= deviationRange && vals[i] >= -deviationRange) {					mCount++;					mSum += history[i];				}			}					} else {			// Calculate the average (not enough data points to remove outliers)			mCount = size;			for (i = 0; i < size; i++) {				mSum += history[i];			}		}		return mSum / mCount;	};	}
 |