2014-01-30 09:41:04 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2014
|
|
|
|
*
|
|
|
|
* This file is licensed under the Affero General Public License version 3
|
|
|
|
* or later.
|
|
|
|
*
|
|
|
|
* See the COPYING-README file.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* global trashBinApp */
|
2014-05-08 20:06:30 +00:00
|
|
|
(function() {
|
|
|
|
|
2014-05-20 14:01:34 +00:00
|
|
|
/**
|
|
|
|
* Construct a new FileActions instance
|
|
|
|
*/
|
|
|
|
var FileActions = function() {
|
|
|
|
this.initialize();
|
|
|
|
}
|
|
|
|
FileActions.prototype = {
|
2014-05-08 20:06:30 +00:00
|
|
|
actions: {},
|
|
|
|
defaults: {},
|
|
|
|
icons: {},
|
|
|
|
currentFile: null,
|
2014-06-27 11:36:18 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* List of handlers to be notified whenever a register() or
|
|
|
|
* setDefault() was called.
|
|
|
|
*/
|
|
|
|
_updateListeners: [],
|
|
|
|
|
2014-05-20 14:01:34 +00:00
|
|
|
initialize: function() {
|
|
|
|
this.clear();
|
|
|
|
},
|
2014-06-27 11:36:18 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds an update listener to be notified whenever register()
|
|
|
|
* or setDefault() has been called.
|
|
|
|
*
|
|
|
|
* @param Function callback
|
|
|
|
*/
|
|
|
|
addUpdateListener: function(callback) {
|
|
|
|
if (!_.isFunction(callback)) {
|
|
|
|
throw 'Argument passed to FileActions.addUpdateListener must be a function';
|
|
|
|
}
|
|
|
|
this._updateListeners.push(callback);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes an update listener.
|
|
|
|
*
|
|
|
|
* @param Function callback
|
|
|
|
*/
|
|
|
|
removeUpdateListener: function(callback) {
|
|
|
|
if (!_.isFunction(callback)) {
|
|
|
|
throw 'Argument passed to FileActions.removeUpdateListener must be a function';
|
|
|
|
}
|
|
|
|
this._updateListeners = _.without(this._updateListeners, callback);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Notifies the registered update listeners
|
|
|
|
*/
|
|
|
|
_notifyUpdateListeners: function() {
|
|
|
|
for (var i = 0; i < this._updateListeners.length; i++) {
|
|
|
|
this._updateListeners[i](this);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2014-05-20 14:01:34 +00:00
|
|
|
/**
|
|
|
|
* Merges the actions from the given fileActions into
|
|
|
|
* this instance.
|
|
|
|
*
|
|
|
|
* @param fileActions instance of OCA.Files.FileActions
|
|
|
|
*/
|
|
|
|
merge: function(fileActions) {
|
|
|
|
var self = this;
|
|
|
|
// merge first level to avoid unintended overwriting
|
|
|
|
_.each(fileActions.actions, function(sourceMimeData, mime) {
|
|
|
|
var targetMimeData = self.actions[mime];
|
|
|
|
if (!targetMimeData) {
|
|
|
|
targetMimeData = {};
|
|
|
|
}
|
|
|
|
self.actions[mime] = _.extend(targetMimeData, sourceMimeData);
|
|
|
|
});
|
|
|
|
|
|
|
|
this.defaults = _.extend(this.defaults, fileActions.defaults);
|
|
|
|
this.icons = _.extend(this.icons, fileActions.icons);
|
|
|
|
},
|
2014-05-08 20:06:30 +00:00
|
|
|
register: function (mime, name, permissions, icon, action, displayName) {
|
|
|
|
if (!this.actions[mime]) {
|
|
|
|
this.actions[mime] = {};
|
2011-06-04 18:16:44 +00:00
|
|
|
}
|
2014-07-04 10:53:03 +00:00
|
|
|
this.actions[mime][name] = {
|
|
|
|
action: action,
|
|
|
|
permissions: permissions,
|
|
|
|
displayName: displayName || t('files', name)
|
|
|
|
};
|
2014-05-08 20:06:30 +00:00
|
|
|
this.icons[name] = icon;
|
2014-06-27 11:36:18 +00:00
|
|
|
this._notifyUpdateListeners();
|
2014-05-08 20:06:30 +00:00
|
|
|
},
|
|
|
|
clear: function() {
|
|
|
|
this.actions = {};
|
|
|
|
this.defaults = {};
|
|
|
|
this.icons = {};
|
|
|
|
this.currentFile = null;
|
2014-06-27 11:36:18 +00:00
|
|
|
this._updateListeners = [];
|
2014-05-08 20:06:30 +00:00
|
|
|
},
|
|
|
|
setDefault: function (mime, name) {
|
|
|
|
this.defaults[mime] = name;
|
2014-06-27 11:36:18 +00:00
|
|
|
this._notifyUpdateListeners();
|
2014-05-08 20:06:30 +00:00
|
|
|
},
|
|
|
|
get: function (mime, type, permissions) {
|
|
|
|
var actions = this.getActions(mime, type, permissions);
|
|
|
|
var filteredActions = {};
|
|
|
|
$.each(actions, function (name, action) {
|
|
|
|
filteredActions[name] = action.action;
|
|
|
|
});
|
|
|
|
return filteredActions;
|
|
|
|
},
|
|
|
|
getActions: function (mime, type, permissions) {
|
|
|
|
var actions = {};
|
|
|
|
if (this.actions.all) {
|
|
|
|
actions = $.extend(actions, this.actions.all);
|
2012-07-25 20:33:08 +00:00
|
|
|
}
|
2014-05-08 20:06:30 +00:00
|
|
|
if (type) {//type is 'dir' or 'file'
|
|
|
|
if (this.actions[type]) {
|
|
|
|
actions = $.extend(actions, this.actions[type]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (mime) {
|
|
|
|
var mimePart = mime.substr(0, mime.indexOf('/'));
|
|
|
|
if (this.actions[mimePart]) {
|
|
|
|
actions = $.extend(actions, this.actions[mimePart]);
|
|
|
|
}
|
|
|
|
if (this.actions[mime]) {
|
|
|
|
actions = $.extend(actions, this.actions[mime]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
var filteredActions = {};
|
|
|
|
$.each(actions, function (name, action) {
|
|
|
|
if (action.permissions & permissions) {
|
|
|
|
filteredActions[name] = action;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return filteredActions;
|
|
|
|
},
|
|
|
|
getDefault: function (mime, type, permissions) {
|
|
|
|
var mimePart;
|
|
|
|
if (mime) {
|
|
|
|
mimePart = mime.substr(0, mime.indexOf('/'));
|
|
|
|
}
|
|
|
|
var name = false;
|
|
|
|
if (mime && this.defaults[mime]) {
|
|
|
|
name = this.defaults[mime];
|
|
|
|
} else if (mime && this.defaults[mimePart]) {
|
|
|
|
name = this.defaults[mimePart];
|
|
|
|
} else if (type && this.defaults[type]) {
|
|
|
|
name = this.defaults[type];
|
|
|
|
} else {
|
|
|
|
name = this.defaults.all;
|
|
|
|
}
|
|
|
|
var actions = this.get(mime, type, permissions);
|
|
|
|
return actions[name];
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* Display file actions for the given element
|
|
|
|
* @param parent "td" element of the file for which to display actions
|
|
|
|
* @param triggerEvent if true, triggers the fileActionsReady on the file
|
|
|
|
* list afterwards (false by default)
|
2014-05-19 11:18:44 +00:00
|
|
|
* @param fileList OCA.Files.FileList instance on which the action is
|
|
|
|
* done, defaults to OCA.Files.App.fileList
|
2014-05-08 20:06:30 +00:00
|
|
|
*/
|
2014-05-19 11:18:44 +00:00
|
|
|
display: function (parent, triggerEvent, fileList) {
|
|
|
|
if (!fileList) {
|
|
|
|
console.warn('FileActions.display() MUST be called with a OCA.Files.FileList instance');
|
2014-06-27 11:36:18 +00:00
|
|
|
return;
|
2014-05-19 11:18:44 +00:00
|
|
|
}
|
2014-05-08 20:06:30 +00:00
|
|
|
this.currentFile = parent;
|
2014-07-01 19:32:04 +00:00
|
|
|
var $tr = parent.closest('tr');
|
2014-05-08 20:06:30 +00:00
|
|
|
var self = this;
|
|
|
|
var actions = this.getActions(this.getCurrentMimeType(), this.getCurrentType(), this.getCurrentPermissions());
|
|
|
|
var file = this.getCurrentFile();
|
|
|
|
var nameLinks;
|
2014-07-01 19:32:04 +00:00
|
|
|
if ($tr.data('renaming')) {
|
2014-05-08 20:06:30 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// recreate fileactions
|
|
|
|
nameLinks = parent.children('a.name');
|
|
|
|
nameLinks.find('.fileactions, .nametext .action').remove();
|
|
|
|
nameLinks.append('<span class="fileactions" />');
|
|
|
|
var defaultAction = this.getDefault(this.getCurrentMimeType(), this.getCurrentType(), this.getCurrentPermissions());
|
2013-09-18 11:09:47 +00:00
|
|
|
|
2014-05-08 20:06:30 +00:00
|
|
|
var actionHandler = function (event) {
|
|
|
|
event.stopPropagation();
|
|
|
|
event.preventDefault();
|
2013-01-14 19:30:39 +00:00
|
|
|
|
2014-05-08 20:06:30 +00:00
|
|
|
self.currentFile = event.data.elem;
|
2014-05-20 14:01:34 +00:00
|
|
|
// also set on global object for legacy apps
|
|
|
|
window.FileActions.currentFile = self.currentFile;
|
|
|
|
|
2014-05-08 20:06:30 +00:00
|
|
|
var file = self.getCurrentFile();
|
2014-05-20 09:44:18 +00:00
|
|
|
var $tr = $(this).closest('tr');
|
2012-11-14 15:05:24 +00:00
|
|
|
|
2014-05-19 11:18:44 +00:00
|
|
|
event.data.actionFunc(file, {
|
2014-05-20 09:44:18 +00:00
|
|
|
$file: $tr,
|
2014-06-20 10:58:04 +00:00
|
|
|
fileList: fileList,
|
2014-05-20 09:44:18 +00:00
|
|
|
fileActions: self,
|
|
|
|
dir: $tr.attr('data-path') || fileList.getCurrentDirectory()
|
2014-05-19 11:18:44 +00:00
|
|
|
});
|
2014-05-08 20:06:30 +00:00
|
|
|
};
|
2013-01-14 19:30:39 +00:00
|
|
|
|
2014-05-08 20:06:30 +00:00
|
|
|
var addAction = function (name, action, displayName) {
|
2013-01-14 19:30:39 +00:00
|
|
|
|
2014-05-08 20:06:30 +00:00
|
|
|
if ((name === 'Download' || action !== defaultAction) && name !== 'Delete') {
|
2013-01-14 19:30:39 +00:00
|
|
|
|
2014-05-08 20:06:30 +00:00
|
|
|
var img = self.icons[name],
|
|
|
|
actionText = displayName,
|
|
|
|
actionContainer = 'a.name>span.fileactions';
|
2014-03-17 19:15:10 +00:00
|
|
|
|
2014-05-08 20:06:30 +00:00
|
|
|
if (name === 'Rename') {
|
|
|
|
// rename has only an icon which appears behind
|
|
|
|
// the file name
|
|
|
|
actionText = '';
|
|
|
|
actionContainer = 'a.name span.nametext';
|
|
|
|
}
|
|
|
|
if (img.call) {
|
|
|
|
img = img(file);
|
|
|
|
}
|
|
|
|
var html = '<a href="#" class="action action-' + name.toLowerCase() + '" data-action="' + name + '">';
|
|
|
|
if (img) {
|
|
|
|
html += '<img class ="svg" src="' + img + '" />';
|
|
|
|
}
|
|
|
|
html += '<span> ' + actionText + '</span></a>';
|
2013-10-02 19:09:03 +00:00
|
|
|
|
2014-05-08 20:06:30 +00:00
|
|
|
var element = $(html);
|
|
|
|
element.data('action', name);
|
|
|
|
element.on('click', {a: null, elem: parent, actionFunc: actions[name].action}, actionHandler);
|
|
|
|
parent.find(actionContainer).append(element);
|
2013-10-02 19:09:03 +00:00
|
|
|
}
|
2014-05-08 20:06:30 +00:00
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
$.each(actions, function (name, action) {
|
|
|
|
if (name !== 'Share') {
|
|
|
|
displayName = action.displayName;
|
|
|
|
ah = action.action;
|
|
|
|
|
|
|
|
addAction(name, ah, displayName);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
if(actions.Share){
|
|
|
|
displayName = t('files', 'Share');
|
|
|
|
addAction('Share', actions.Share, displayName);
|
|
|
|
}
|
|
|
|
|
|
|
|
// remove the existing delete action
|
|
|
|
parent.parent().children().last().find('.action.delete').remove();
|
|
|
|
if (actions['Delete']) {
|
|
|
|
var img = self.icons['Delete'];
|
|
|
|
var html;
|
2012-10-28 10:51:54 +00:00
|
|
|
if (img.call) {
|
|
|
|
img = img(file);
|
2011-08-11 21:21:30 +00:00
|
|
|
}
|
2014-05-08 20:06:30 +00:00
|
|
|
if (typeof trashBinApp !== 'undefined' && trashBinApp) {
|
|
|
|
html = '<a href="#" original-title="' + t('files', 'Delete permanently') + '" class="action delete delete-icon" />';
|
|
|
|
} else {
|
2014-05-28 18:27:28 +00:00
|
|
|
html = '<a href="#" original-title="' + t('files', 'Delete') + '" class="action delete delete-icon" />';
|
2012-10-28 10:51:54 +00:00
|
|
|
}
|
|
|
|
var element = $(html);
|
2014-05-08 20:06:30 +00:00
|
|
|
element.data('action', actions['Delete']);
|
|
|
|
element.on('click', {a: null, elem: parent, actionFunc: actions['Delete'].action}, actionHandler);
|
|
|
|
parent.parent().children().last().append(element);
|
2011-07-27 21:23:46 +00:00
|
|
|
}
|
2013-01-14 19:30:39 +00:00
|
|
|
|
2014-05-08 20:06:30 +00:00
|
|
|
if (triggerEvent){
|
2014-07-01 19:32:04 +00:00
|
|
|
fileList.$fileList.trigger(jQuery.Event("fileActionsReady", {fileList: fileList, $files: $tr}));
|
2014-05-08 20:06:30 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
getCurrentFile: function () {
|
|
|
|
return this.currentFile.parent().attr('data-file');
|
|
|
|
},
|
|
|
|
getCurrentMimeType: function () {
|
|
|
|
return this.currentFile.parent().attr('data-mime');
|
|
|
|
},
|
|
|
|
getCurrentType: function () {
|
|
|
|
return this.currentFile.parent().attr('data-type');
|
|
|
|
},
|
|
|
|
getCurrentPermissions: function () {
|
|
|
|
return this.currentFile.parent().data('permissions');
|
|
|
|
},
|
2013-02-11 15:09:04 +00:00
|
|
|
|
2014-05-08 20:06:30 +00:00
|
|
|
/**
|
|
|
|
* Register the actions that are used by default for the files app.
|
|
|
|
*/
|
2014-05-19 13:20:44 +00:00
|
|
|
registerDefaultActions: function() {
|
2014-05-08 20:06:30 +00:00
|
|
|
this.register('all', 'Delete', OC.PERMISSION_DELETE, function () {
|
|
|
|
return OC.imagePath('core', 'actions/delete');
|
2014-05-19 13:20:44 +00:00
|
|
|
}, function (filename, context) {
|
2014-06-06 13:13:02 +00:00
|
|
|
context.fileList.do_delete(filename, context.dir);
|
2014-05-08 20:06:30 +00:00
|
|
|
$('.tipsy').remove();
|
|
|
|
});
|
2014-03-17 19:15:10 +00:00
|
|
|
|
2014-05-08 20:06:30 +00:00
|
|
|
// t('files', 'Rename')
|
|
|
|
this.register('all', 'Rename', OC.PERMISSION_UPDATE, function () {
|
|
|
|
return OC.imagePath('core', 'actions/rename');
|
2014-05-19 13:20:44 +00:00
|
|
|
}, function (filename, context) {
|
|
|
|
context.fileList.rename(filename);
|
2014-05-08 20:06:30 +00:00
|
|
|
});
|
2013-01-14 19:30:39 +00:00
|
|
|
|
2014-05-19 13:20:44 +00:00
|
|
|
this.register('dir', 'Open', OC.PERMISSION_READ, '', function (filename, context) {
|
2014-07-04 10:53:03 +00:00
|
|
|
var dir = context.$file.attr('data-path') || context.fileList.getCurrentDirectory();
|
2014-05-08 20:06:30 +00:00
|
|
|
if (dir !== '/') {
|
|
|
|
dir = dir + '/';
|
|
|
|
}
|
2014-05-19 13:20:44 +00:00
|
|
|
context.fileList.changeDirectory(dir + filename);
|
2014-05-08 20:06:30 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
this.setDefault('dir', 'Open');
|
2013-10-17 09:59:13 +00:00
|
|
|
|
2014-06-02 14:29:03 +00:00
|
|
|
this.register('all', 'Download', OC.PERMISSION_READ, function () {
|
2014-05-08 20:06:30 +00:00
|
|
|
return OC.imagePath('core', 'actions/download');
|
2014-05-19 13:20:44 +00:00
|
|
|
}, function (filename, context) {
|
2014-05-21 10:54:34 +00:00
|
|
|
var dir = context.dir || context.fileList.getCurrentDirectory();
|
|
|
|
var url = context.fileList.getDownloadUrl(filename, dir);
|
2014-05-08 20:06:30 +00:00
|
|
|
if (url) {
|
|
|
|
OC.redirect(url);
|
|
|
|
}
|
|
|
|
});
|
2013-10-17 09:59:13 +00:00
|
|
|
}
|
2014-05-08 20:06:30 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
OCA.Files.FileActions = FileActions;
|
|
|
|
|
2014-05-20 14:01:34 +00:00
|
|
|
// global file actions to be used by all lists
|
|
|
|
OCA.Files.fileActions = new OCA.Files.FileActions();
|
|
|
|
OCA.Files.legacyFileActions = new OCA.Files.FileActions();
|
|
|
|
|
|
|
|
// for backward compatibility
|
|
|
|
//
|
|
|
|
// legacy apps are expecting a stateful global FileActions object to register
|
|
|
|
// their actions on. Since legacy apps are very likely to break with other
|
|
|
|
// FileList views than the main one ("All files"), actions registered
|
|
|
|
// through window.FileActions will be limited to the main file list.
|
|
|
|
window.FileActions = OCA.Files.legacyFileActions;
|
|
|
|
window.FileActions.register = function (mime, name, permissions, icon, action, displayName) {
|
2014-05-21 10:54:34 +00:00
|
|
|
console.warn('FileActions.register() is deprecated, please use OCA.Files.fileActions.register() instead', arguments);
|
|
|
|
OCA.Files.FileActions.prototype.register.call(
|
|
|
|
window.FileActions, mime, name, permissions, icon, action, displayName
|
|
|
|
);
|
2014-05-20 14:01:34 +00:00
|
|
|
};
|
2014-06-27 11:36:18 +00:00
|
|
|
window.FileActions.display = function (parent, triggerEvent, fileList) {
|
|
|
|
fileList = fileList || OCA.Files.App.fileList;
|
|
|
|
console.warn('FileActions.display() is deprecated, please use OCA.Files.fileActions.register() which automatically redisplays actions', mime, name);
|
|
|
|
OCA.Files.FileActions.prototype.display.call(window.FileActions, parent, triggerEvent, fileList);
|
2014-05-20 14:01:34 +00:00
|
|
|
};
|
|
|
|
})();
|
2014-05-08 20:06:30 +00:00
|
|
|
|