: str_replace(): Passing null to parameter #2 ($replace) of type array|string is deprecated in
* @class elFinder command "upload"
* Upload files using iframe or XMLHttpRequest & FormData.
* Dialog allow to send files using drag and drop
* @author Dmitry (dio) Levashov
elFinder.prototype.commands.upload = function() {
var hover = this.fm.res('class', 'hover');
this.disableOnSearch = true;
this.updateOnSelect = false;
this.getstate = function(select) {
sel = (select || [fm.cwd().hash]);
if (!this._disabled && sel.length == 1) {
return (f && f.mime == 'directory' && f.write)? 0 : -1;
this.exec = function(data) {
getTargets = function() {
var tgts = data && (data instanceof Array)? data : null,
if (! data || data instanceof Array) {
if (! tgts && (sel = fm.selected()).length === 1 && fm.file(sel[0]).mime === 'directory') {
} else if (!tgts || tgts.length !== 1 || fm.file(tgts[0]).mime !== 'directory') {
check = targets? targets[0] : (data && data.target? data.target : null),
targetDir = check? fm.file(check) : fm.cwd(),
fmUpload = function(data) {
var cwd = fm.getUI('cwd'),
if (data && data.added && data.added[0] && ! fm.ui.notify.children('.elfinder-notify-upload').length) {
var newItem = fm.findCwdNodes(data.added);
newItem.trigger('scrolltoview');
if (targetDir.hash !== cwdHash) {
node = jQuery('<div></div>').append(
jQuery('<button type="button" class="ui-button ui-widget ui-state-default ui-corner-all elfinder-tabstop"><span class="ui-button-text">'+fm.i18n('cmdopendir')+'</span></button>')
.on('mouseenter mouseleave', function(e) {
jQuery(this).toggleClass('ui-state-hover', e.type == 'mouseenter');
}).on('click', function() {
fm.exec('open', check).done(function() {
fm.one('opendone', function() {
fm.trigger('selectfiles', {files : jQuery.map(data.added, function(f) {return f.hash;})});
fm.trigger('selectfiles', {files : jQuery.map(data.added, function(f) {return f.hash;})});
fm.toast({msg: fm.i18n(['complete', fm.i18n('cmdupload')]), extNode: node});
dfrd.notifyWith(this, Array.from(arguments));
upload = function(data) {
dialog.elfinderdialog('close');
data.target = targets[0];
getSelector = function() {
var hash = targetDir.hash,
dirs = jQuery.map(fm.files(hash), function(f) {
return (f.mime === 'directory' && f.write)? f : null;
return jQuery('<div class="elfinder-upload-dirselect elfinder-tabstop" title="' + fm.i18n('folders') + '"></div>')
.on('click', function(e) {
dirs = fm.sortFiles(dirs);
var $this = jQuery(this),
base = dialog.closest('div.ui-dialog'),
getRaw = function(f, icon) {
label : fm.escape(f.i18 || f.name),
var title = base.children('.ui-dialog-titlebar:first').find('span.elfinder-upload-target');
title.html(' - ' + fm.escape(f.i18 || f.name));
className : (targets && targets.length && f.hash === targets[0])? 'ui-state-active' : '',
iconClass : f.csscls || '',
raw = [ getRaw(targetDir, 'opendir'), '|' ];
jQuery.each(dirs, function(i, f) {
raw.push(getRaw(f, 'dir'));
fm.trigger('contextmenu', {
x: e.pageX || jQuery(this).offset().left,
y: e.pageY || jQuery(this).offset().top,
}).append('<span class="elfinder-button-icon elfinder-button-icon-dir" ></span>');
inputButton = function(type, caption) {
input = jQuery('<input type="file" ' + type + '/>')
.on('click', function() {
form.css('display', 'none').css('position', 'relative');
requestAnimationFrame(function() {
form.css('display', '').css('position', '');
.on('change', function() {
upload({input : input.get(0), type : 'files'});
.on('dragover', function(e) {
e.originalEvent.dataTransfer.dropEffect = 'copy';
form = jQuery('<form></form>').append(input).on('click', function(e) {
return jQuery('<div class="ui-button ui-widget ui-state-default ui-corner-all ui-button-text-only elfinder-tabstop elfinder-focus"><span class="ui-button-text">'+fm.i18n(caption)+'</span></div>')
.on('click', function(e) {
.on('mouseenter mouseleave', function(e) {
jQuery(this).toggleClass(hover, e.type === 'mouseenter');
dfrd = jQuery.Deferred(),
dialog, dropbox, pastebox, dropUpload, paste, dirs, spinner, uidialog;
dropUpload = function(e) {
target = e._target || null,
trf = e.dataTransfer || null,
if (trf.types && trf.types.length && jQuery.inArray('Files', trf.types) !== -1) {
else if (trf.items && trf.items.length && trf.items[0].kind) {
kind = trf.items[0].kind;
elfFrom = trf.getData('elfinderfrom');
mycwd = window.location.href + fm.cwd().hash;
if ((!target && elfFrom === mycwd) || target === mycwd) {
if (kind === 'file' && (trf.items[0].getAsEntry || trf.items[0].webkitGetAsEntry)) {
} else if (kind !== 'string' && trf.files && trf.files.length && jQuery.inArray('Text', trf.types) === -1) {
if ((data = trf.getData('text/html')) && data.match(/<(?:img|a)/i)) {
if (data = trf.getData('text')) {
} else if (trf && trf.files) {
// maybe folder uploading but this UA dose not support it
fmUpload({files : file, type : type, target : target, dropEvt : e});
errors = ['errUploadNoFiles'];
errors.push('errFolderUpload');
if (data.input || data.files) {
} else if (data.dropEvt) {
dropUpload(data.dropEvt);
var e = ev.originalEvent || ev;
var files = [], items = [];
if (e.clipboardData.items && e.clipboardData.items.length){
items = e.clipboardData.items;
for (var i=0; i < items.length; i++) {
if (e.clipboardData.items[i].kind == 'file') {
file = e.clipboardData.items[i].getAsFile();
} else if (e.clipboardData.files && e.clipboardData.files.length) {
files = e.clipboardData.files;
upload({files : files, type : 'files', clipdata : true});
var my = e.target || e.srcElement;
requestAnimationFrame(function() {
jQuery(my).find('img').each(function(i, v){
if (v.src.match(/^webkit-fake-url:\/\//)) {
// ref. https://bugs.webkit.org/show_bug.cgi?id=49141
// https://dev.ckeditor.com/ticket/13029
if (jQuery(my).find('a,img').length) {
upload({files : [ src ], type : type});
dialog = jQuery('<div class="elfinder-upload-dialog-wrapper"></div>')
.append(inputButton('multiple', 'selectForUpload'));
if (! fm.UA.Mobile && (function(input) {
return (typeof input.webkitdirectory !== 'undefined' || typeof input.directory !== 'undefined');})(document.createElement('input'))) {
dialog.append(inputButton('multiple webkitdirectory directory', 'selectFolder'));
if (targetDir.hash === cwdHash || fm.navHash2Elm(targetDir.hash).hasClass('elfinder-subtree-loaded')) {
getSelector().appendTo(dialog);
spinner = jQuery('<div class="elfinder-upload-dirselect" title="' + fm.i18n('nowLoading') + '"></div>')
.append('<span class="elfinder-button-icon elfinder-button-icon-spinner" ></span>')
fm.request({cmd : 'tree', target : targetDir.hash})
fm.one('treedone', function() {
spinner.replaceWith(getSelector());
uidialog.elfinderdialog('tabstopsInit');
dropbox = jQuery('<div class="ui-corner-all elfinder-upload-dropbox elfinder-tabstop" contenteditable="true" data-ph="'+fm.i18n('dropPasteFiles')+'"></div>')
.on('paste', function(e){
.on('mousedown click', function(){
jQuery(this).trigger('focus');
.on('mouseover', function(){
jQuery(this).addClass(hover);
.on('mouseout', function(){
jQuery(this).removeClass(hover);
.on('dragenter', function(e) {
jQuery(this).addClass(hover);
.on('dragleave', function(e) {
jQuery(this).removeClass(hover);
.on('dragover', function(e) {
e.originalEvent.dataTransfer.dropEffect = 'copy';
jQuery(this).addClass(hover);
.on('drop', function(e) {
dialog.elfinderdialog('close');
targets && (e.originalEvent._target = targets[0]);
dropUpload(e.originalEvent);
.after('<div class="elfinder-upload-dialog-or">'+fm.i18n('or')+'</div>')[0];
pastebox = jQuery('<div class="ui-corner-all elfinder-upload-dropbox" contenteditable="true">'+fm.i18n('dropFilesBrowser')+'</div>')
.on('paste drop', function(e){
.on('mousedown click', function(){
jQuery(this).trigger('focus');
.on('dragenter mouseover', function(){
jQuery(this).addClass(hover);
.on('dragleave mouseout', function(){
jQuery(this).removeClass(hover);
.after('<div class="elfinder-upload-dialog-or">'+fm.i18n('or')+'</div>')[0];
uidialog = this.fmDialog(dialog, {
title : this.title + '<span class="elfinder-upload-target">' + (targetDir? ' - ' + fm.escape(targetDir.i18 || targetDir.name) : '') + '</span>',
propagationEvents : ['mousemove', 'mouseup', 'click'],
var cm = fm.getUI('contextmenu');