META-INF.resources.org.richfaces.fileupload.js Maven / Gradle / Ivy
The newest version!
/*
* JBoss, Home of Professional Open Source
* Copyright ${year}, Red Hat, Inc. and individual contributors
* by the @authors tag. See the copyright.txt in the distribution for a
* full listing of individual contributors.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/
(function($, rf) {
rf.ui = rf.ui || {};
/**
* Backing object for rich:fileUpload
*
* @extends RichFaces.BaseComponent
* @memberOf! RichFaces.ui
* @constructs RichFaces.ui.FileUpload
*
* @param id
* @param options
*/
rf.ui.FileUpload = function(id, options) {
this.id = id;
this.items = [];
this.submitedItems = [];
$.extend(this, options);
if (this.acceptedTypes) {
this.acceptedTypes = $.trim(this.acceptedTypes).toUpperCase().split(/\s*,\s*\.?/);
}
if (this.maxFilesQuantity) {
this.maxFilesQuantity = parseInt($.trim(this.maxFilesQuantity));
}
this.element = $(this.attachToDom());
this.form = this.element.parents("form:first");
var header = this.element.children(".rf-fu-hdr:first");
var leftButtons = header.children(".rf-fu-btns-lft:first");
this.addButton = leftButtons.children(".rf-fu-btn-add:first");
this.uploadButton = this.addButton.next();
this.clearButton = leftButtons.next().children(".rf-fu-btn-clr:first");
this.inputContainer = this.addButton.find(".rf-fu-inp-cntr:first");
this.input = this.inputContainer.children("input");
this.list = header.next();
this.element.on('dragenter', function(e) {e.stopPropagation(); e.preventDefault();});
this.element.on('dragover', function(e) {e.stopPropagation(); e.preventDefault();});
this.element.on('drop', $.proxy(this.__addItemsFromDrop, this));
this.hiddenContainer = this.list.next();
this.cleanInput = this.input.clone();
this.addProxy = $.proxy(this.__addItems, this);
this.input.on("change", this.addProxy);
this.addButton.on("mousedown", pressButton).on("mouseup", unpressButton).on("mouseout", unpressButton);
this.uploadButton.on("click", $.proxy(this.__startUpload, this)).on("mousedown", pressButton)
.on("mouseup",unpressButton).on("mouseout", unpressButton);
this.clearButton.on("click", $.proxy(this.__removeAllItems, this)).on("mousedown", pressButton)
.on("mouseup", unpressButton).on("mouseout", unpressButton);
if (this.onfilesubmit) {
rf.Event.bind(this.element, "onfilesubmit", new Function("event", this.onfilesubmit));
}
if (this.ontyperejected) {
rf.Event.bind(this.element, "ontyperejected", new Function("event", this.ontyperejected));
}
if (this.onsizerejected) {
rf.Event.bind(this.element, "onsizerejected", new Function("event", this.onsizerejected));
}
if (this.onuploadcomplete) {
rf.Event.bind(this.element, "onuploadcomplete", new Function("event", this.onuploadcomplete));
}
if (this.onclear) {
rf.Event.bind(this.element, "onclear", new Function("event", this.onclear));
}
if (this.onfileselect) {
rf.Event.bind(this.element, "onfileselect", new Function("event", this.onfileselect));
}
};
var UID = "rf_fu_uid";
var UID_ALT = "rf_fu_uid_alt";
var FAKE_PATH = "C:\\fakepath\\";
var ITEM_HTML = '';
var ITEM_STATE = {
NEW: "new",
UPLOADING: "uploading",
DONE: "done",
SIZE_EXCEEDED: "sizeExceeded",
STOPPED: "stopped",
SERVER_ERROR_PROCESS: "serverErrorProc",
SERVER_ERROR_UPLOAD: "serverErrorUp"
};
var pressButton = function(event) {
$(this).children(":first").css("background-position", "3px 3px").css("padding", "4px 4px 2px 22px");
};
var unpressButton = function(event) {
$(this).children(":first").css("background-position", "2px 2px").css("padding", "3px 5px 3px 21px");
};
rf.BaseComponent.extend(rf.ui.FileUpload);
function TypeRejectedException(fileName) {
this.name = "TypeRejectedException";
this.message = "The type of file " + fileName + " is not accepted";
this.fileName = fileName;
}
$.extend(rf.ui.FileUpload.prototype, (function () {
return {
name: "FileUpload",
doneLabel: "Done",
sizeExceededLabel: "File size is exceeded",
stoppedLabel: "",
serverErrorProcLabel: "Server error: error in processing",
serverErrorUpLabel: "Server error: upload failed",
clearLabel: "Clear",
deleteLabel: "Delete",
__addFiles : function(files) {
var context = {
acceptedFileNames: [],
rejectedFileNames: []
};
if (files) {
for (var i = 0 ; i < files.length; i++) {
this.__tryAddItem(context, files[i]);
if (this.maxFilesQuantity && this.__getTotalItemCount() >= this.maxFilesQuantity) {
this.addButton.hide();
break;
}
}
} else {
var fileName = this.input.val();
this.__tryAddItem(context, fileName);
}
if (context.rejectedFileNames.length > 0) {
rf.Event.fire(this.element, "ontyperejected", context.rejectedFileNames.join(','));
}
if (this.immediateUpload) {
this.__startUpload();
}
},
__addItems : function() {
this.__addFiles(this.input.prop("files"));
// replace input with a copy, IE 10 doesn't allow clearing just the value (this.input.val(""))
this.input.replaceWith(this.input.clone(true));
this.input = this.inputContainer.children("input");
},
__addItemsFromDrop: function(dropEvent) {
dropEvent.stopPropagation();
dropEvent.preventDefault();
if (this.maxFilesQuantity && this.__getTotalItemCount() >= this.maxFilesQuantity) {
return;
}
this.__addFiles(dropEvent.originalEvent.dataTransfer.files);
},
__tryAddItem: function(context, file) {
if (this.maxFileSize && file.size > this.maxFileSize) {
rf.Event.fire(this.element, "onsizerejected", file);
return;
}
try {
if (this.__addItem(file)) {
context.acceptedFileNames.push(file.name);
}
} catch (e) {
if (e instanceof TypeRejectedException) {
context.rejectedFileNames.push(file.name);
} else {
throw e;
}
}
},
__addItem: function(file) {
var fileName = file.name;
if (!navigator.platform.indexOf("Win")) {
fileName = fileName.match(/[^\\]*$/)[0];
} else {
if (!fileName.indexOf(FAKE_PATH)) {
fileName = fileName.substr(FAKE_PATH.length);
} else {
fileName = fileName.match(/[^\/]*$/)[0];
}
}
if (this.__accept(fileName) && (!this.noDuplicate || !this.__isFileAlreadyAdded(fileName))) {
var item = new Item(this, file);
this.list.append(item.getJQuery());
this.items.push(item);
this.__updateButtons();
rf.Event.fire(this.element, "onfileselect", fileName);
return true;
}
return false;
},
__removeItem: function(item) {
var inItems = $.inArray(item, this.items),
inSItems = $.inArray(item, this.submitedItems);
if (inItems != -1) {
this.items.splice(inItems, 1);
}
if (inSItems != -1) {
this.submitedItems.splice(inSItems, 1);
}
this.__updateButtons();
rf.Event.fire(this.element, "onclear", [item.model]);
},
__removeAllItems: function(item) {
var itemsRemoved = [];
for (var i = 0; i < this.submitedItems.length; i++) {
itemsRemoved.push(this.submitedItems[i].model);
}
for (var i = 0; i < this.items.length; i++) {
itemsRemoved.push(this.items[i].model);
}
this.list.empty();
this.items.splice(0, this.items.length);
this.submitedItems.splice(0, this.submitedItems.length);
this.__updateButtons();
rf.Event.fire(this.element, "onclear", itemsRemoved);
},
__updateButtons: function() {
if (!this.loadableItem && this.list.children(".rf-fu-itm").length) {
if (this.items.length) {
this.uploadButton.css("display", "inline-block");
} else {
this.uploadButton.hide();
}
this.clearButton.css("display", "inline-block");
} else {
this.uploadButton.hide();
this.clearButton.hide();
}
if (this.maxFilesQuantity && this.__getTotalItemCount() >= this.maxFilesQuantity) {
this.addButton.hide();
} else {
this.addButton.css("display", "inline-block");
}
},
__startUpload: function() {
if (!this.items.length) {
this.__finishUpload();
return;
}
rf.setGlobalStatusNameVariable(this.status);
this.loadableItem = this.items.shift();
this.__updateButtons();
this.loadableItem.startUploading();
},
__accept: function(fileName) {
fileName = fileName.toUpperCase();
var result = !this.acceptedTypes;
for (var i = 0; !result && i < this.acceptedTypes.length; i++) {
var extension = this.acceptedTypes[i];
if (extension === "" && fileName.indexOf(".") < 0) {
// no extension
result = true;
} else {
result = fileName.indexOf(extension, fileName.length - extension.length) !== -1;
}
}
if (!result) {
throw new TypeRejectedException(fileName);
}
return result;
},
__isFileAlreadyAdded: function(fileName) {
var result = false;
for (var i = 0; !result && i < this.items.length; i++) {
result = this.items[i].model.name == fileName;
}
result = result || (this.loadableItem && this.loadableItem.model.name == fileName);
for (var i = 0; !result && i < this.submitedItems.length; i++) {
result = this.submitedItems[i].model.name == fileName;
}
return result;
},
__getTotalItemCount : function() {
return this.__getItemCountByState(this.items, ITEM_STATE.NEW)
+ this.__getItemCountByState(this.submitedItems, ITEM_STATE.DONE);
},
__getItemCountByState : function(items) {
var statuses = {};
var s = 0;
for ( var i = 1; i < arguments.length; i++) {
statuses[arguments[i]] = true;
}
for ( var i = 0; i < items.length; i++) {
if (statuses[items[i].model.state]) {
s++;
}
}
return s;
},
__finishUpload : function() {
this.loadableItem = null;
this.__updateButtons();
var items = [];
for (var i = 0; i < this.submitedItems.length; i++) {
items.push(this.submitedItems[i].model);
}
for (var i = 0; i < this.items.length; i++) {
items.push(this.items[i].model);
}
rf.Event.fire(this.element, "onuploadcomplete", items);
}
};
})());
var Item = function(fileUpload, file) {
this.fileUpload = fileUpload;
this.model = {name: file.name, state: ITEM_STATE.NEW, file: file};
};
$.extend(Item.prototype, {
__createProgressBar: function(item, facet) {
var $pb = facet.find(".rf-pb");
if ($pb.length) { // custom progressBar
return {
pb: RichFaces.component($pb),
prepare: function() {
item.find(".rf-fu-itm-lft").append($pb.detach());
},
setValue: function(progress) {
var max = parseFloat(this.pb.maxValue), min = parseFloat(this.pb.minValue),
relativeProgress = progress*(max - min)/100 + min;
this.pb.setValue(relativeProgress);
},
cleanUp: function() {
facet.append($pb.detach());
},
show: function() {
$pb.show();
},
hide: function() {
$pb.hide();
}
}
}
var progressElement =
''
+ '';
return {
prepare: function() {
item.find(".rf-fu-itm-lft").append($(progressElement));
this.element = item.find(".progress-bar");
this.label = this.element.find("span");
},
setValue: function(progress) {
this.label.html( progress + " %" );
this.element.attr("aria-valuenow", progress);
this.element.css("width", progress + "%");
},
cleanUp: function() {
},
show: function() {
this.element.parent().show();
},
hide: function() {
this.element.parent().hide();
}
}
},
getJQuery: function() {
this.element = $(ITEM_HTML);
var leftArea = this.element.children(".rf-fu-itm-lft:first");
this.label = leftArea.children(".rf-fu-itm-lbl:first");
this.state = this.label.nextAll(".rf-fu-itm-st:first");
this.link = leftArea.next().children("a");
this.label.html(this.model.name);
this.link.html(this.fileUpload["deleteLabel"]);
this.link.on("click", $.proxy(this.removeOrStop, this));
return this.element;
},
removeOrStop: function() {
this.element.remove();
this.fileUpload.__removeItem(this);
},
startUploading: function() {
this.progressBarFacet = this.__createProgressBar(this.element, this.fileUpload.hiddenContainer);
this.progressBarFacet.prepare();
this.progressBarFacet.hide();
this.state.css("display", "block");
this.progressBarFacet.setValue(0);
this.progressBarFacet.show();
this.link.html("");
this.model.state = ITEM_STATE.UPLOADING;
this.uid = window.crypto.getRandomValues(new Uint32Array(1))[0];
var formData = new FormData(),
fileName = this.model.file.name,
viewState = this.fileUpload.form.find("input[name='javax.faces.ViewState']").val();
formData.append(this.fileUpload.form[0].id, this.fileUpload.form[0].id);
formData.append("javax.faces.ViewState", viewState);
formData.append(this.fileUpload.id, this.model.file);
var originalAction = this.fileUpload.form.attr("action"),
delimiter = originalAction.indexOf("?") == -1 ? "?" : "&",
encodedId = encodeURIComponent(this.fileUpload.id),
newAction = originalAction + delimiter + UID + "=" + this.uid +
"&javax.faces.partial.ajax=true" +
"&javax.faces.source=" + encodedId +
"&javax.faces.partial.execute=" + encodedId +
"&org.richfaces.ajax.component=" + encodedId +
"&javax.faces.ViewState=" + encodeURIComponent(viewState);
if (jsf.getClientWindow && jsf.getClientWindow()) {
newAction += "&javax.faces.ClientWindow=" + encodeURIComponent(jsf.getClientWindow());
};
var eventHandler = function(handlerCode) {
if (handlerCode) {
var safeHandlerCode = "try {" +
handlerCode +
"} catch (e) {" +
"window.RichFaces.log.error('Error in method execution: ' + e.message)" +
"}";
return new Function("event", safeHandlerCode);
}
}
this.onerror = eventHandler(this.fileUpload.onerror);
this.xhr = new XMLHttpRequest();
this.xhr.open('POST', newAction, true);
this.xhr.setRequestHeader('Faces-Request', 'partial/ajax');
this.xhr.upload.onprogress = $.proxy(function(e) {
if (e.lengthComputable) {
var progress = Math.floor((e.loaded / e.total) * 100);
this.progressBarFacet.setValue(progress);
}
}, this);
this.xhr.upload.onerror = $.proxy(function (e) {
this.fileUpload.loadableItem = null;
this.finishUploading(ITEM_STATE.SERVER_ERROR_UPLOAD);
}, this);
this.xhr.onload = $.proxy(function (e) {
switch (e.target.status) {
case 413:
responseStatus = ITEM_STATE.SIZE_EXCEEDED;
break;
case 200:
responseStatus = ITEM_STATE.DONE;
break;
default: // 500 - error in processing parts
responseStatus = ITEM_STATE.SERVER_ERROR_PROCESS;
}
var handlerFunction = function(handlerName) {
return function (event) {
var xml = $("partial-response extension#org\\.richfaces\\.extension", event.responseXML)
var handlerCode = xml.children(handlerName).text();
var handler = eventHandler(handlerCode);
if (handler) {
handler.call(this,event);
}
$("form").trigger('ajaxcomplete');
}
}
var eventsAdapter = rf.createJSFEventsAdapter({
complete: handlerFunction('complete'),
error: handlerFunction('error')
});
var responseContext = {
source: this.fileUpload.element[0], // MyFaces
sourceid: this.fileUpload.element[0], // Mojarra
element: this.fileUpload.element[0],
/* hack for MyFaces */
_mfInternal: {
_mfSourceControlId: this.fileUpload.element.attr('id')
},
onevent: eventsAdapter,
onerror: eventsAdapter
};
var onbeforedomupdate = eventHandler(this.fileUpload.onbeforedomupdate);
if (onbeforedomupdate) {
var data = {};
data.type = "event";
data.status = 'complete';
data.source = this.fileUpload.element[0];
data.responseCode = this.xhr.status;
data.responseXML = this.xhr.responseXML;
data.responseText = this.xhr.responseText
onbeforedomupdate.call(this.fileUpload, data);
}
this.fileUpload.form.trigger('ajaxbeforedomupdate');
jsf.ajax.response(this.xhr, responseContext);
this.finishUploading(responseStatus);
this.fileUpload.__startUpload();
}, this);
var onbegin = eventHandler(this.fileUpload.onbegin);
if (onbegin) {
onbegin.call(this.fileUpload, {
source: this.fileUpload.element[0],
type: 'event',
status: 'begin'
})
}
this.fileUpload.form.trigger('ajaxbegin');
this.xhr.send(formData);
rf.Event.fire(this.fileUpload.element, "onfilesubmit", this.model);
},
finishUploading: function(state) {
if (state != ITEM_STATE.DONE && this.onerror) {
this.onerror.call(this.fileUpload, {state: state, error: this.fileUpload[state + "Label"]});
}
this.state.html(this.fileUpload[state + "Label"]);
this.progressBarFacet.hide();
this.progressBarFacet.cleanUp();
this.link.html(this.fileUpload["clearLabel"]);
this.model.state = state;
this.fileUpload.submitedItems.push(this);
}
});
}(RichFaces.jQuery, window.RichFaces));
© 2015 - 2025 Weber Informatics LLC | Privacy Policy