2014-11-18 20:53:45 +03:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2014 Vincent Petry <pvince81@owncloud.com>
|
|
|
|
*
|
|
|
|
* This file is licensed under the Affero General Public License version 3
|
|
|
|
* or later.
|
|
|
|
*
|
|
|
|
* See the COPYING-README file.
|
|
|
|
*
|
|
|
|
*/
|
2014-12-15 14:43:16 +03:00
|
|
|
|
|
|
|
/* global Handlebars */
|
|
|
|
|
2018-06-20 23:40:10 +03:00
|
|
|
(function (OCA) {
|
2014-11-18 20:53:45 +03:00
|
|
|
|
2016-11-21 17:03:45 +03:00
|
|
|
_.extend(OC.Files.Client, {
|
2018-06-20 23:40:10 +03:00
|
|
|
PROPERTY_TAGS: '{' + OC.Files.Client.NS_OWNCLOUD + '}tags',
|
|
|
|
PROPERTY_FAVORITE: '{' + OC.Files.Client.NS_OWNCLOUD + '}favorite'
|
2016-11-21 17:03:45 +03:00
|
|
|
});
|
|
|
|
|
2014-12-15 14:43:16 +03:00
|
|
|
/**
|
2016-02-25 19:23:32 +03:00
|
|
|
* Returns the icon class for the matching state
|
2014-12-15 14:43:16 +03:00
|
|
|
*
|
|
|
|
* @param {boolean} state true if starred, false otherwise
|
2016-02-25 19:23:32 +03:00
|
|
|
* @return {string} icon class for star image
|
2014-12-15 14:43:16 +03:00
|
|
|
*/
|
2018-06-20 23:40:10 +03:00
|
|
|
function getStarIconClass (state) {
|
2016-02-25 19:23:32 +03:00
|
|
|
return state ? 'icon-starred' : 'icon-star';
|
2014-12-15 14:43:16 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Render the star icon with the given state
|
2014-12-15 19:20:41 +03:00
|
|
|
*
|
2014-12-15 14:43:16 +03:00
|
|
|
* @param {boolean} state true if starred, false otherwise
|
|
|
|
* @return {Object} jQuery object
|
|
|
|
*/
|
2018-06-20 23:40:10 +03:00
|
|
|
function renderStar (state) {
|
2018-10-01 20:23:40 +03:00
|
|
|
return OCA.Files.Templates['favorite_mark']({
|
2014-12-15 14:43:16 +03:00
|
|
|
isFavorite: state,
|
2017-09-27 13:39:59 +03:00
|
|
|
altText: state ? t('files', 'Favorited') : t('files', 'Not favorited'),
|
2016-02-25 19:23:32 +03:00
|
|
|
iconClass: getStarIconClass(state)
|
2014-12-15 14:43:16 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2014-12-15 17:50:57 +03:00
|
|
|
/**
|
2017-09-27 13:39:59 +03:00
|
|
|
* Toggle star icon on favorite mark element
|
2014-12-15 17:50:57 +03:00
|
|
|
*
|
2017-09-27 13:39:59 +03:00
|
|
|
* @param {Object} $favoriteMarkEl favorite mark element
|
2014-12-15 17:50:57 +03:00
|
|
|
* @param {boolean} state true if starred, false otherwise
|
|
|
|
*/
|
2018-06-20 23:40:10 +03:00
|
|
|
function toggleStar ($favoriteMarkEl, state) {
|
2017-09-27 13:39:59 +03:00
|
|
|
$favoriteMarkEl.removeClass('icon-star icon-starred').addClass(getStarIconClass(state));
|
|
|
|
$favoriteMarkEl.toggleClass('permanent', state);
|
2014-12-15 17:50:57 +03:00
|
|
|
}
|
|
|
|
|
2018-06-18 12:13:52 +03:00
|
|
|
/**
|
|
|
|
* Remove Item from Quickaccesslist
|
|
|
|
*
|
2018-06-20 23:40:10 +03:00
|
|
|
* @param {String} appfolder folder to be removed
|
2018-06-18 12:13:52 +03:00
|
|
|
*/
|
2018-06-20 23:40:10 +03:00
|
|
|
function removeFavoriteFromList (appfolder) {
|
|
|
|
var quickAccessList = 'sublist-favorites';
|
|
|
|
var listULElements = document.getElementById(quickAccessList);
|
2018-07-12 17:49:25 +03:00
|
|
|
if (!listULElements) {
|
|
|
|
return;
|
|
|
|
}
|
2018-06-27 11:40:32 +03:00
|
|
|
|
|
|
|
var apppath=appfolder;
|
|
|
|
if(appfolder.startsWith("//")){
|
|
|
|
apppath=appfolder.substring(1, appfolder.length);
|
|
|
|
}
|
2018-06-18 12:13:52 +03:00
|
|
|
|
2018-07-13 10:57:54 +03:00
|
|
|
$(listULElements).find('[data-dir="' + apppath + '"]').remove();
|
2018-06-18 12:13:52 +03:00
|
|
|
|
2018-06-20 23:40:10 +03:00
|
|
|
if (listULElements.childElementCount === 0) {
|
2018-07-13 10:57:54 +03:00
|
|
|
var collapsibleButton = $(listULElements).parent().find('button.collapse');
|
|
|
|
collapsibleButton.hide();
|
2018-06-27 01:35:28 +03:00
|
|
|
$("#button-collapse-parent-favorites").removeClass('collapsible');
|
2018-06-18 12:13:52 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add Item to Quickaccesslist
|
|
|
|
*
|
2018-06-20 23:40:10 +03:00
|
|
|
* @param {String} appfolder folder to be added
|
2018-06-18 12:13:52 +03:00
|
|
|
*/
|
2018-06-20 23:40:10 +03:00
|
|
|
function addFavoriteToList (appfolder) {
|
|
|
|
var quickAccessList = 'sublist-favorites';
|
|
|
|
var listULElements = document.getElementById(quickAccessList);
|
2018-07-12 17:49:25 +03:00
|
|
|
if (!listULElements) {
|
|
|
|
return;
|
|
|
|
}
|
2018-06-20 23:40:10 +03:00
|
|
|
var listLIElements = listULElements.getElementsByTagName('li');
|
|
|
|
|
|
|
|
var appName = appfolder.substring(appfolder.lastIndexOf("/") + 1, appfolder.length);
|
2018-07-13 10:57:54 +03:00
|
|
|
var apppath = appfolder;
|
2018-06-27 11:40:32 +03:00
|
|
|
|
|
|
|
if(appfolder.startsWith("//")){
|
2018-07-13 10:57:54 +03:00
|
|
|
apppath = appfolder.substring(1, appfolder.length);
|
2018-06-27 11:40:32 +03:00
|
|
|
}
|
2018-07-13 10:57:54 +03:00
|
|
|
var url = OC.generateUrl('/apps/files/?dir=' + apppath + '&view=files');
|
|
|
|
|
2018-06-20 23:40:10 +03:00
|
|
|
|
|
|
|
var innerTagA = document.createElement('A');
|
2018-06-27 11:40:32 +03:00
|
|
|
innerTagA.setAttribute("href", url);
|
2018-06-20 23:40:10 +03:00
|
|
|
innerTagA.setAttribute("class", "nav-icon-files svg");
|
|
|
|
innerTagA.innerHTML = appName;
|
|
|
|
|
|
|
|
var length = listLIElements.length + 1;
|
|
|
|
var innerTagLI = document.createElement('li');
|
2018-07-13 10:57:54 +03:00
|
|
|
innerTagLI.setAttribute("data-id", apppath.replace('/', '-'));
|
|
|
|
innerTagLI.setAttribute("data-dir", apppath);
|
|
|
|
innerTagLI.setAttribute("data-view", 'files');
|
2018-06-20 23:40:10 +03:00
|
|
|
innerTagLI.setAttribute("class", "nav-" + appName);
|
|
|
|
innerTagLI.setAttribute("folderpos", length.toString());
|
|
|
|
innerTagLI.appendChild(innerTagA);
|
|
|
|
|
2018-06-27 11:40:32 +03:00
|
|
|
$.get(OC.generateUrl("/apps/files/api/v1/quickaccess/get/NodeType"),{folderpath: apppath}, function (data, status) {
|
|
|
|
if (data === "dir") {
|
|
|
|
if (listULElements.childElementCount <= 0) {
|
|
|
|
listULElements.appendChild(innerTagLI);
|
2018-07-13 10:57:54 +03:00
|
|
|
var collapsibleButton = $(listULElements).parent().find('button.collapse');
|
|
|
|
collapsibleButton.show();
|
|
|
|
$(listULElements).parent().addClass('collapsible');
|
2018-06-27 11:40:32 +03:00
|
|
|
} else {
|
|
|
|
listLIElements[listLIElements.length - 1].after(innerTagLI);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
2018-06-18 12:13:52 +03:00
|
|
|
}
|
|
|
|
|
2014-11-18 20:53:45 +03:00
|
|
|
OCA.Files = OCA.Files || {};
|
|
|
|
|
|
|
|
/**
|
2017-09-27 13:39:59 +03:00
|
|
|
* Extends the file actions and file list to include a favorite mark icon
|
|
|
|
* and a favorite action in the file actions menu; it also adds "data-tags"
|
|
|
|
* and "data-favorite" attributes to file elements.
|
2018-10-15 15:29:52 +03:00
|
|
|
*
|
|
|
|
* @namespace OCA.Files.TagsPlugin
|
2014-11-18 20:53:45 +03:00
|
|
|
*/
|
|
|
|
OCA.Files.TagsPlugin = {
|
|
|
|
name: 'Tags',
|
|
|
|
|
|
|
|
allowedLists: [
|
|
|
|
'files',
|
2016-11-10 11:13:25 +03:00
|
|
|
'favorites',
|
|
|
|
'systemtags',
|
|
|
|
'shares.self',
|
|
|
|
'shares.others',
|
|
|
|
'shares.link'
|
2014-11-18 20:53:45 +03:00
|
|
|
],
|
|
|
|
|
2018-06-20 23:40:10 +03:00
|
|
|
_extendFileActions: function (fileActions) {
|
2014-11-18 20:53:45 +03:00
|
|
|
var self = this;
|
2017-09-27 09:07:20 +03:00
|
|
|
|
|
|
|
fileActions.registerAction({
|
|
|
|
name: 'Favorite',
|
2018-06-20 23:40:10 +03:00
|
|
|
displayName: function (context) {
|
2017-09-27 09:07:20 +03:00
|
|
|
var $file = context.$file;
|
|
|
|
var isFavorite = $file.data('favorite') === true;
|
|
|
|
|
|
|
|
if (isFavorite) {
|
|
|
|
return t('files', 'Remove from favorites');
|
|
|
|
}
|
|
|
|
|
|
|
|
// As it is currently not possible to provide a context for
|
|
|
|
// the i18n strings "Add to favorites" was used instead of
|
|
|
|
// "Favorite" to remove the ambiguity between verb and noun
|
|
|
|
// when it is translated.
|
|
|
|
return t('files', 'Add to favorites');
|
|
|
|
},
|
|
|
|
mime: 'all',
|
2017-10-30 19:09:15 +03:00
|
|
|
order: -100,
|
2017-11-02 17:01:17 +03:00
|
|
|
permissions: OC.PERMISSION_NONE,
|
2018-06-20 23:40:10 +03:00
|
|
|
iconClass: function (fileName, context) {
|
2017-09-27 09:07:20 +03:00
|
|
|
var $file = context.$file;
|
|
|
|
var isFavorite = $file.data('favorite') === true;
|
|
|
|
|
|
|
|
if (isFavorite) {
|
2017-10-30 20:17:59 +03:00
|
|
|
return 'icon-star-dark';
|
2017-09-27 09:07:20 +03:00
|
|
|
}
|
|
|
|
|
2017-10-30 20:17:59 +03:00
|
|
|
return 'icon-starred';
|
2017-09-27 09:07:20 +03:00
|
|
|
},
|
2018-06-20 23:40:10 +03:00
|
|
|
actionHandler: function (fileName, context) {
|
2017-09-27 13:39:59 +03:00
|
|
|
var $favoriteMarkEl = context.$file.find('.favorite-mark');
|
2017-09-27 09:07:20 +03:00
|
|
|
var $file = context.$file;
|
|
|
|
var fileInfo = context.fileList.files[$file.index()];
|
|
|
|
var dir = context.dir || context.fileList.getCurrentDirectory();
|
|
|
|
var tags = $file.attr('data-tags');
|
2018-06-18 12:13:52 +03:00
|
|
|
|
2017-09-27 09:07:20 +03:00
|
|
|
if (_.isUndefined(tags)) {
|
|
|
|
tags = '';
|
|
|
|
}
|
|
|
|
tags = tags.split('|');
|
|
|
|
tags = _.without(tags, '');
|
|
|
|
var isFavorite = tags.indexOf(OC.TAG_FAVORITE) >= 0;
|
|
|
|
if (isFavorite) {
|
2014-11-18 20:53:45 +03:00
|
|
|
// remove tag from list
|
|
|
|
tags = _.without(tags, OC.TAG_FAVORITE);
|
2018-06-18 12:13:52 +03:00
|
|
|
removeFavoriteFromList(dir + '/' + fileName);
|
2014-11-18 20:53:45 +03:00
|
|
|
} else {
|
|
|
|
tags.push(OC.TAG_FAVORITE);
|
2018-06-18 12:13:52 +03:00
|
|
|
addFavoriteToList(dir + '/' + fileName);
|
2014-11-18 20:53:45 +03:00
|
|
|
}
|
2015-09-01 20:29:55 +03:00
|
|
|
|
|
|
|
// pre-toggle the star
|
2017-09-27 13:39:59 +03:00
|
|
|
toggleStar($favoriteMarkEl, !isFavorite);
|
2014-12-15 17:50:57 +03:00
|
|
|
|
2015-09-01 20:29:55 +03:00
|
|
|
context.fileInfoModel.trigger('busy', context.fileInfoModel, true);
|
2015-08-14 16:54:27 +03:00
|
|
|
|
2014-11-18 20:53:45 +03:00
|
|
|
self.applyFileTags(
|
|
|
|
dir + '/' + fileName,
|
2015-01-26 14:01:07 +03:00
|
|
|
tags,
|
2017-09-27 13:39:59 +03:00
|
|
|
$favoriteMarkEl,
|
2015-01-26 14:01:07 +03:00
|
|
|
isFavorite
|
2018-06-20 23:40:10 +03:00
|
|
|
).then(function (result) {
|
2015-09-01 20:29:55 +03:00
|
|
|
context.fileInfoModel.trigger('busy', context.fileInfoModel, false);
|
2015-01-12 19:54:35 +03:00
|
|
|
// response from server should contain updated tags
|
|
|
|
var newTags = result.tags;
|
|
|
|
if (_.isUndefined(newTags)) {
|
|
|
|
newTags = tags;
|
|
|
|
}
|
2015-09-01 20:29:55 +03:00
|
|
|
context.fileInfoModel.set({
|
|
|
|
'tags': newTags,
|
|
|
|
'favorite': !isFavorite
|
|
|
|
});
|
2014-11-18 20:53:45 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2018-06-20 23:40:10 +03:00
|
|
|
_extendFileList: function (fileList) {
|
2014-11-18 20:53:45 +03:00
|
|
|
// extend row prototype
|
|
|
|
var oldCreateRow = fileList._createRow;
|
2018-06-20 23:40:10 +03:00
|
|
|
fileList._createRow = function (fileData) {
|
2014-11-18 20:53:45 +03:00
|
|
|
var $tr = oldCreateRow.apply(this, arguments);
|
2017-09-27 13:39:59 +03:00
|
|
|
var isFavorite = false;
|
2014-11-18 20:53:45 +03:00
|
|
|
if (fileData.tags) {
|
|
|
|
$tr.attr('data-tags', fileData.tags.join('|'));
|
|
|
|
if (fileData.tags.indexOf(OC.TAG_FAVORITE) >= 0) {
|
|
|
|
$tr.attr('data-favorite', true);
|
2017-09-27 13:39:59 +03:00
|
|
|
isFavorite = true;
|
2014-11-18 20:53:45 +03:00
|
|
|
}
|
|
|
|
}
|
2017-09-27 13:39:59 +03:00
|
|
|
var $icon = $(renderStar(isFavorite));
|
2017-09-28 15:42:37 +03:00
|
|
|
$tr.find('td.filename .thumbnail').append($icon);
|
2014-11-18 20:53:45 +03:00
|
|
|
return $tr;
|
|
|
|
};
|
2015-08-12 18:30:20 +03:00
|
|
|
var oldElementToFile = fileList.elementToFile;
|
2018-06-20 23:40:10 +03:00
|
|
|
fileList.elementToFile = function ($el) {
|
2015-08-12 18:30:20 +03:00
|
|
|
var fileInfo = oldElementToFile.apply(this, arguments);
|
2015-09-02 20:02:06 +03:00
|
|
|
var tags = $el.attr('data-tags');
|
|
|
|
if (_.isUndefined(tags)) {
|
|
|
|
tags = '';
|
|
|
|
}
|
|
|
|
tags = tags.split('|');
|
|
|
|
tags = _.without(tags, '');
|
|
|
|
fileInfo.tags = tags;
|
2015-08-12 18:30:20 +03:00
|
|
|
return fileInfo;
|
|
|
|
};
|
2015-11-02 16:17:49 +03:00
|
|
|
|
|
|
|
var oldGetWebdavProperties = fileList._getWebdavProperties;
|
2018-06-20 23:40:10 +03:00
|
|
|
fileList._getWebdavProperties = function () {
|
2015-11-02 16:17:49 +03:00
|
|
|
var props = oldGetWebdavProperties.apply(this, arguments);
|
2016-11-21 17:03:45 +03:00
|
|
|
props.push(OC.Files.Client.PROPERTY_TAGS);
|
|
|
|
props.push(OC.Files.Client.PROPERTY_FAVORITE);
|
2015-11-02 16:17:49 +03:00
|
|
|
return props;
|
|
|
|
};
|
|
|
|
|
2018-06-20 23:40:10 +03:00
|
|
|
fileList.filesClient.addFileInfoParser(function (response) {
|
2015-11-02 16:17:49 +03:00
|
|
|
var data = {};
|
|
|
|
var props = response.propStat[0].properties;
|
2016-11-21 17:03:45 +03:00
|
|
|
var tags = props[OC.Files.Client.PROPERTY_TAGS];
|
|
|
|
var favorite = props[OC.Files.Client.PROPERTY_FAVORITE];
|
2015-11-02 16:17:49 +03:00
|
|
|
if (tags && tags.length) {
|
2018-06-20 23:40:10 +03:00
|
|
|
tags = _.chain(tags).filter(function (xmlvalue) {
|
2016-11-21 17:03:45 +03:00
|
|
|
return (xmlvalue.namespaceURI === OC.Files.Client.NS_OWNCLOUD && xmlvalue.nodeName.split(':')[1] === 'tag');
|
2018-06-20 23:40:10 +03:00
|
|
|
}).map(function (xmlvalue) {
|
2015-11-02 16:17:49 +03:00
|
|
|
return xmlvalue.textContent || xmlvalue.text;
|
|
|
|
}).value();
|
|
|
|
}
|
|
|
|
if (tags) {
|
|
|
|
data.tags = tags;
|
|
|
|
}
|
|
|
|
if (favorite && parseInt(favorite, 10) !== 0) {
|
|
|
|
data.tags = data.tags || [];
|
|
|
|
data.tags.push(OC.TAG_FAVORITE);
|
|
|
|
}
|
|
|
|
return data;
|
|
|
|
});
|
2014-11-18 20:53:45 +03:00
|
|
|
},
|
|
|
|
|
2018-06-20 23:40:10 +03:00
|
|
|
attach: function (fileList) {
|
2014-11-18 20:53:45 +03:00
|
|
|
if (this.allowedLists.indexOf(fileList.id) < 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this._extendFileActions(fileList.fileActions);
|
|
|
|
this._extendFileList(fileList);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Replaces the given files' tags with the specified ones.
|
|
|
|
*
|
|
|
|
* @param {String} fileName path to the file or folder to tag
|
|
|
|
* @param {Array.<String>} tagNames array of tag names
|
2017-09-27 13:39:59 +03:00
|
|
|
* @param {Object} $favoriteMarkEl favorite mark element
|
2015-01-26 14:01:07 +03:00
|
|
|
* @param {boolean} isFavorite Was the item favorited before
|
2014-11-18 20:53:45 +03:00
|
|
|
*/
|
2018-06-20 23:40:10 +03:00
|
|
|
applyFileTags: function (fileName, tagNames, $favoriteMarkEl, isFavorite) {
|
2014-11-18 20:53:45 +03:00
|
|
|
var encodedPath = OC.encodePath(fileName);
|
|
|
|
while (encodedPath[0] === '/') {
|
|
|
|
encodedPath = encodedPath.substr(1);
|
|
|
|
}
|
|
|
|
return $.ajax({
|
|
|
|
url: OC.generateUrl('/apps/files/api/v1/files/') + encodedPath,
|
|
|
|
contentType: 'application/json',
|
|
|
|
data: JSON.stringify({
|
|
|
|
tags: tagNames || []
|
|
|
|
}),
|
|
|
|
dataType: 'json',
|
|
|
|
type: 'POST'
|
2018-06-20 23:40:10 +03:00
|
|
|
}).fail(function (response) {
|
2015-01-26 13:27:51 +03:00
|
|
|
var message = '';
|
|
|
|
// show message if it is available
|
2018-06-20 23:40:10 +03:00
|
|
|
if (response.responseJSON && response.responseJSON.message) {
|
2015-01-26 13:27:51 +03:00
|
|
|
message = ': ' + response.responseJSON.message;
|
|
|
|
}
|
2017-02-14 23:26:00 +03:00
|
|
|
OC.Notification.show(t('files', 'An error occurred while trying to update the tags' + message), {type: 'error'});
|
2017-09-27 13:39:59 +03:00
|
|
|
toggleStar($favoriteMarkEl, isFavorite);
|
2014-11-18 20:53:45 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
2018-06-27 11:40:32 +03:00
|
|
|
})
|
|
|
|
(OCA);
|
2014-11-18 20:53:45 +03:00
|
|
|
|
|
|
|
OC.Plugins.register('OCA.Files.FileList', OCA.Files.TagsPlugin);
|