: str_replace(): Passing null to parameter #2 ($replace) of type array|string is deprecated in
* elFinder command prototype
* @author Dmitry (dio) Levashov
elFinder.prototype.command = function(fm) {
* Command name, same as class name
* Command icon class name with out 'elfinder-button-icon-'
* Use this.name if it is empty
* Short command description
* Linked(Child) commands name
* They are loaded together when tthis command is loaded.
* this.state = -1; // command disabled
* this.state = 0; // command enabled
* this.state = 1; // command active (for example "fullscreen" command while elfinder in fullscreen mode)
* If true, command can not be disabled by connector.
this.alwaysEnabled = false;
* Do not change dirctory on removed current work directory
this.noChangeDirOnRemovedCwd = false;
* If true, this means command was disabled by connector.
* If true, this command is disabled on serach results
this.disableOnSearch = false;
* Call update() when event select fired
this.updateOnSelect = true;
* Sync toolbar button title on change
this.syncTitleOnChange = false;
* Keep display of the context menu when command execution
this.keepContextmenu = false;
* elFinder events defaults handlers.
* Inside handlers "this" is current command object
enable : function() { this.update(void(0), this.value); },
disable : function() { this.update(-1, this.value); },
'open reload load sync' : function() {
this._disabled = !(this.alwaysEnabled || this.fm.isCommandEnabled(this.name));
this.update(void(0), this.value);
* elFinder events handlers.
* Inside handlers "this" is current command object
this.options = {ui : 'button'};
* Callback functions on `change` event
* bind events and shortcuts
this.setup = function(name, opts) {
setCallback = function(s) {
var cb = s.callback || function(e) {
fm.exec(self.name, void(0), {
s.callback = function(e) {
var enabled, checks = {};
if (fm.searchStatus.state < 2) {
enabled = fm.isCommandEnabled(self.name);
jQuery.each(fm.selected(), function(i, h) {
if (fm.optionsByHashes[h]) {
jQuery.each(fm.volOptions, function(id) {
if (!checks[id] && h.indexOf(id) === 0) {
jQuery.each(checks, function(h) {
enabled = fm.isCommandEnabled(self.name, h);
this.title = fm.messages['cmd'+name] ? fm.i18n('cmd'+name)
: ((this.extendsCmd && fm.messages['cmd'+this.extendsCmd]) ? fm.i18n('cmd'+this.extendsCmd) : name);
this.options = Object.assign({}, this.options, opts);
this.dialogClass = 'elfinder-dialog-' + name;
if (typeof opts.shortcuts === 'function') {
sc = opts.shortcuts(this.fm, this.shortcuts);
} else if (Array.isArray(opts.shortcuts)) {
this.shortcuts = sc || [];
if (this.updateOnSelect) {
this._handlers.select = function() { this.update(void(0), this.value); };
jQuery.each(Object.assign({}, self._handlers, self.handlers), function(cmd, handler) {
fm.bind(cmd, jQuery.proxy(handler, self));
for (i = 0; i < this.shortcuts.length; i++) {
!s.description && (s.description = this.title);
if (this.disableOnSearch) {
fm.bind('search searchend', function() {
self._disabled = this.type === 'search'? true : ! (this.alwaysEnabled || fm.isCommandEnabled(name));
self.update(void(0), self.value);
* Command specific init stuffs
this.init = function() {};
* @param Array target files hashes
* @param Array|Object command value
* @return jQuery.Deferred
this.exec = function(files, opts) {
return jQuery.Deferred().reject();
this.getUndo = function(opts, resData) {
* Return true if command disabled.
this.disabled = function() {
* Return true if command enabled.
this.enabled = function() {
* Return true if command active.
this.active = function() {
* Return current command state.
* Must be overloaded in most commands
this.getstate = function() {
* Update command state/value
* and rize 'change' event if smth changed
* @param Number new state or undefined to auto update state
this.update = function(s, v) {
if (this._disabled && this.fm.searchStatus === 0) {
this.state = s !== void(0) ? s : this.getstate();
if (state != this.state || value != this.value) {
* Bind handler / fire 'change' event.
* @param Function|undefined event callback
this.change = function(c) {
if (typeof(c) === 'function') {
for (i = 0; i < this.listeners.length; i++) {
cmd(this.state, this.value);
this.fm.debug('error', e);
* With argument check given files hashes and return list of existed files hashes.
* Without argument return selected files hashes.
* @param Array|String|void hashes
this.hashes = function(hashes) {
? jQuery.grep(Array.isArray(hashes) ? hashes : [hashes], function(hash) { return fm.file(hash) ? true : false; })
* Return only existed files from given fils hashes | selected files
* @param Array|String|void hashes
this.files = function(hashes) {
? jQuery.map(Array.isArray(hashes) ? hashes : [hashes], function(hash) { return fm.file(hash) || null; })
* @param String|DOMElement content
* @return Object jQuery element object
this.fmDialog = function(content, options) {
options.cssClass += ' ' + this.dialogClass;
options.cssClass = this.dialogClass;
return this.fm.dialog(content, options);