Skip to content

Instantly share code, notes, and snippets.

@weixiyen
Created June 22, 2010 17:52
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save weixiyen/448814 to your computer and use it in GitHub Desktop.
Save weixiyen/448814 to your computer and use it in GitHub Desktop.
(function($){
var opts = {},
default_opts = {
url: '',
refresh: 1000,
paramname: 'userfile',
maxfiles: 25,
maxfilesize: 1, // MBs
data: {},
drop: empty,
dragEnter: empty,
dragOver: empty,
dragLeave: empty,
docEnter: empty,
docOver: empty,
docLeave: empty,
error: function(err, filename){alert(err);},
uploadStarted: empty,
uploadFinished: empty,
progressUpdated: empty,
speedUpdated: empty
},
errors = ["BrowserNotSupported", "TooManyFiles", "FileTooLarge"],
doc_leave_timer,
stop_loop = false;
$.fn.filedrop = function(options) {
opts = $.extend( {}, default_opts, options );
this.get(0).addEventListener("drop", drop, true);
this.bind('dragenter', dragEnter).bind('dragover', dragOver).bind('dragleave', dragLeave);
document.addEventListener("drop", docDrop, true);
$(document).bind('dragenter', docEnter).bind('dragover', docOver).bind('dragleave', docLeave);
};
function drop(e) {
opts.drop(e);
upload(e.dataTransfer.files);
e.preventDefault();
return false;
}
function upload(files) {
stop_loop = false;
if (!files) {
opts.error(errors[0]);
return false;
}
var len = files.length;
if (len > opts.maxfiles) {
opts.error(errors[1]);
return false;
}
var i = 0;
function up(i){
if (stop_loop) return false;
try {
if (i === len) return;
var reader = new FileReader(),
max_file_size = 1048576 * opts.maxfilesize;
reader.index = i,
reader.file = files[i],
reader.len = len;
if (reader.file.size > max_file_size) {
opts.error(errors[2], reader.file);
return false;
}
reader.addEventListener("loadend", send, false);
reader.readAsBinaryString(files[i]);
} catch(err) {
opts.error(errors[0]);
return false;
}
}
function send(e) {
var xhr = new XMLHttpRequest(),
upload = xhr.upload,
file = e.target.file,
index = e.target.index,
start_time = new Date().getTime(),
boundary = '------multipartformboundary' + (new Date).getTime(),
builder = getBuilder(file.name, e.target.result, boundary);
upload.index = index;
upload.file = file;
upload.downloadStartTime = start_time;
upload.currentStart = start_time;
upload.currentProgress = 0;
upload.startData = 0;
upload.addEventListener("progress", progress, false);
xhr.open("POST", opts.url, true);
xhr.setRequestHeader('content-type', 'multipart/form-data; boundary='
+ boundary);
xhr.sendAsBinary(builder);
opts.uploadStarted(index, file, e.target.len);
xhr.onload = function() {
if (xhr.responseText) {
var now = new Date().getTime(),
timeDiff = now - start_time,
result = opts.uploadFinished(index, file, eval( '[' + xhr.responseText + ']' ), timeDiff);
if (result === false) stop_loop = true;
up(index + 1);
}
};
}
up(i);
}
function getBuilder(filename, filedata, boundary) {
var dashdash = '--',
crlf = '\r\n',
builder = '';
$.each(opts.data, function(i, val) {
if (typeof val === 'function') val = val();
builder += dashdash;
builder += boundary;
builder += crlf;
builder += 'Content-Disposition: form-data; name="'+i+'"';
builder += crlf;
builder += val;
builder += crlf;
});
builder += dashdash;
builder += boundary;
builder += crlf;
builder += 'Content-Disposition: form-data; name="'+opts.paramname+'"';
builder += '; filename="' + filename + '"';
builder += crlf;
builder += 'Content-Type: application/octet-stream';
builder += crlf;
builder += crlf;
builder += filedata;
builder += crlf;
builder += dashdash;
builder += boundary;
builder += crlf;
builder += dashdash;
builder += boundary;
builder += dashdash;
builder += crlf;
return builder;
}
function progress(e) {
if (e.lengthComputable) {
var percentage = Math.round((e.loaded * 100) / e.total);
if (this.currentProgress != percentage) {
this.currentProgress = percentage;
opts.progressUpdated(this.index, this.file, this.currentProgress);
var elapsed = new Date().getTime();
var diffTime = elapsed - this.currentStart;
if (diffTime >= opts.refresh) {
var diffData = e.loaded - this.startData;
var speed = diffData / diffTime; // KB per second
opts.speedUpdated(this.index, this.file, speed);
this.startData = e.loaded;
this.currentStart = elapsed;
}
}
}
}
/*
function upload(files) {
if (!files) {
opts.error(errors[0]);
return false;
}
var len = files.length;
if (len > opts.maxfiles) {
opts.error(errors[1]);
return false;
}
for (var i=0; i<len; i++) {
if (stop_loop) return false;
try {
if (i === len) return;
var reader = new FileReader(),
max_file_size = 1048576 * opts.maxfilesize;
reader.index = i,
reader.file = files[i],
reader.len = len;
if (reader.file.size > max_file_size) {
opts.error(errors[2], reader.file);
return false;
}
reader.addEventListener("loadend", send, false);
reader.readAsBinaryString(files[i]);
} catch(err) {
opts.error(errors[0]);
return false;
}
}
function send(e) {
var xhr = new XMLHttpRequest(),
upload = xhr.upload,
file = e.target.file,
index = e.target.index,
start_time = new Date().getTime(),
boundary = '------multipartformboundary' + (new Date).getTime(),
builder = getBuilder(file.name, e.target.result, boundary);
upload.index = index;
upload.file = file;
upload.downloadStartTime = start_time;
upload.currentStart = start_time;
upload.currentProgress = 0;
upload.startData = 0;
upload.addEventListener("progress", progress, false);
xhr.open("POST", opts.url, true);
xhr.setRequestHeader('content-type', 'multipart/form-data; boundary='
+ boundary);
xhr.sendAsBinary(builder);
opts.uploadStarted(index, file, e.target.len);
xhr.onload = function() {
if (xhr.responseText) {
var now = new Date().getTime(),
timeDiff = now - start_time,
result = opts.uploadFinished(index, file, eval( '[' + xhr.responseText + ']' ), timeDiff);
if (result === false) stop_loop = true;
}
};
}
}
*/
function dragEnter(e) {
clearTimeout(doc_leave_timer);
e.preventDefault();
opts.dragEnter(e);
}
function dragOver(e) {
clearTimeout(doc_leave_timer);
e.preventDefault();
opts.docOver(e);
opts.dragOver(e);
}
function dragLeave(e) {
clearTimeout(doc_leave_timer);
opts.dragLeave(e);
e.stopPropagation();
}
function docDrop(e) {
e.preventDefault();
opts.docLeave(e);
return false;
}
function docEnter(e) {
clearTimeout(doc_leave_timer);
e.preventDefault();
opts.docEnter(e);
return false;
}
function docOver(e) {
clearTimeout(doc_leave_timer);
e.preventDefault();
opts.docOver(e);
return false;
}
function docLeave(e) {
doc_leave_timer = setTimeout(function(){
opts.docLeave(e);
}, 200);
}
function empty(){}
})(jQuery);
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment