2014-04-30 15:42:35 +00: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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
(function() {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sharing file list
|
|
|
|
*
|
|
|
|
* Contains both "shared with others" and "shared with you" modes.
|
|
|
|
*/
|
|
|
|
var FileList = function($el, options) {
|
|
|
|
this.initialize($el, options);
|
|
|
|
};
|
|
|
|
|
|
|
|
FileList.prototype = _.extend({}, OCA.Files.FileList.prototype, {
|
|
|
|
appName: 'Shares',
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the list shows the files shared with the user (true) or
|
|
|
|
* the files that the user shared with others (false).
|
|
|
|
*/
|
|
|
|
_sharedWithUser: false,
|
2014-06-04 09:10:11 +00:00
|
|
|
_linksOnly: false,
|
2014-04-30 15:42:35 +00:00
|
|
|
|
|
|
|
initialize: function($el, options) {
|
|
|
|
OCA.Files.FileList.prototype.initialize.apply(this, arguments);
|
|
|
|
if (this.initialized) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-06-04 09:10:11 +00:00
|
|
|
// TODO: consolidate both options
|
2014-04-30 15:42:35 +00:00
|
|
|
if (options && options.sharedWithUser) {
|
|
|
|
this._sharedWithUser = true;
|
|
|
|
}
|
2014-06-04 09:10:11 +00:00
|
|
|
if (options && options.linksOnly) {
|
|
|
|
this._linksOnly = true;
|
|
|
|
}
|
2014-04-30 15:42:35 +00:00
|
|
|
},
|
|
|
|
|
2014-06-03 09:04:57 +00:00
|
|
|
_renderRow: function() {
|
|
|
|
// HACK: needed to call the overridden _renderRow
|
|
|
|
// this is because at the time this class is created
|
|
|
|
// the overriding hasn't been done yet...
|
|
|
|
return OCA.Files.FileList.prototype._renderRow.apply(this, arguments);
|
|
|
|
},
|
|
|
|
|
2014-04-30 15:42:35 +00:00
|
|
|
_createRow: function(fileData) {
|
|
|
|
// TODO: hook earlier and render the whole row here
|
|
|
|
var $tr = OCA.Files.FileList.prototype._createRow.apply(this, arguments);
|
|
|
|
$tr.find('.filesize').remove();
|
2014-05-28 14:53:42 +00:00
|
|
|
$tr.find('td.date').before($tr.children('td:first'));
|
2014-04-30 15:42:35 +00:00
|
|
|
$tr.find('td.filename input:checkbox').remove();
|
2014-05-19 13:20:44 +00:00
|
|
|
$tr.attr('data-share-id', _.pluck(fileData.shares, 'id').join(','));
|
2014-05-26 10:59:44 +00:00
|
|
|
if (this._sharedWithUser) {
|
2014-06-02 16:52:31 +00:00
|
|
|
$tr.attr('data-share-owner', fileData.shareOwner);
|
2014-05-26 10:59:44 +00:00
|
|
|
}
|
2014-04-30 15:42:35 +00:00
|
|
|
return $tr;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set whether the list should contain outgoing shares
|
|
|
|
* or incoming shares.
|
|
|
|
*
|
|
|
|
* @param state true for incoming shares, false otherwise
|
|
|
|
*/
|
|
|
|
setSharedWithUser: function(state) {
|
|
|
|
this._sharedWithUser = !!state;
|
|
|
|
},
|
|
|
|
|
2014-05-19 13:20:44 +00:00
|
|
|
updateEmptyContent: function() {
|
|
|
|
var dir = this.getCurrentDirectory();
|
|
|
|
if (dir === '/') {
|
|
|
|
// root has special permissions
|
|
|
|
this.$el.find('#emptycontent').toggleClass('hidden', !this.isEmpty);
|
|
|
|
this.$el.find('#filestable thead th').toggleClass('hidden', this.isEmpty);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
OCA.Files.FileList.prototype.updateEmptyContent.apply(this, arguments);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
getDirectoryPermissions: function() {
|
2014-05-21 10:54:34 +00:00
|
|
|
return OC.PERMISSION_READ | OC.PERMISSION_DELETE;
|
|
|
|
},
|
|
|
|
|
|
|
|
updateStorageStatistics: function() {
|
|
|
|
// no op because it doesn't have
|
|
|
|
// storage info like free space / used space
|
2014-05-19 13:20:44 +00:00
|
|
|
},
|
|
|
|
|
2014-04-30 15:42:35 +00:00
|
|
|
reload: function() {
|
|
|
|
var self = this;
|
|
|
|
this.showMask();
|
|
|
|
if (this._reloadCall) {
|
|
|
|
this._reloadCall.abort();
|
|
|
|
}
|
|
|
|
this._reloadCall = $.ajax({
|
|
|
|
url: OC.linkToOCS('apps/files_sharing/api/v1') + 'shares',
|
|
|
|
/* jshint camelcase: false */
|
|
|
|
data: {
|
|
|
|
format: 'json',
|
|
|
|
shared_with_me: !!this._sharedWithUser
|
|
|
|
},
|
|
|
|
type: 'GET',
|
|
|
|
beforeSend: function(xhr) {
|
|
|
|
xhr.setRequestHeader('OCS-APIREQUEST', 'true');
|
|
|
|
},
|
|
|
|
error: function(result) {
|
|
|
|
self.reloadCallback(result);
|
|
|
|
},
|
|
|
|
success: function(result) {
|
|
|
|
self.reloadCallback(result);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
reloadCallback: function(result) {
|
|
|
|
delete this._reloadCall;
|
|
|
|
this.hideMask();
|
|
|
|
|
|
|
|
this.$el.find('#headerSharedWith').text(
|
|
|
|
t('files_sharing', this._sharedWithUser ? 'Shared by' : 'Shared with')
|
|
|
|
);
|
|
|
|
if (result.ocs && result.ocs.data) {
|
|
|
|
this.setFiles(this._makeFilesFromShares(result.ocs.data));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// TODO: error handling
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts the OCS API share response data to a file info
|
|
|
|
* list
|
|
|
|
* @param OCS API share array
|
|
|
|
* @return array of file info maps
|
|
|
|
*/
|
|
|
|
_makeFilesFromShares: function(data) {
|
2014-06-04 09:10:11 +00:00
|
|
|
/* jshint camelcase: false */
|
2014-04-30 15:42:35 +00:00
|
|
|
var self = this;
|
2014-06-04 09:10:11 +00:00
|
|
|
var files = data;
|
|
|
|
|
|
|
|
if (this._linksOnly) {
|
|
|
|
files = _.filter(data, function(share) {
|
|
|
|
return share.share_type === OC.Share.SHARE_TYPE_LINK;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2014-04-30 15:42:35 +00:00
|
|
|
// OCS API uses non-camelcased names
|
2014-06-04 09:10:11 +00:00
|
|
|
files = _.chain(files)
|
2014-05-21 10:54:34 +00:00
|
|
|
// convert share data to file data
|
2014-05-15 17:51:15 +00:00
|
|
|
.map(function(share) {
|
|
|
|
var file = {
|
|
|
|
id: share.file_source,
|
2014-05-20 11:11:06 +00:00
|
|
|
mimetype: share.mimetype
|
2014-05-15 17:51:15 +00:00
|
|
|
};
|
|
|
|
if (share.item_type === 'folder') {
|
|
|
|
file.type = 'dir';
|
2014-05-21 10:54:34 +00:00
|
|
|
file.mimetype = 'httpd/unix-directory';
|
2014-05-15 17:51:15 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
file.type = 'file';
|
|
|
|
// force preview retrieval as we don't have mime types,
|
|
|
|
// the preview endpoint will fall back to the mime type
|
|
|
|
// icon if no preview exists
|
|
|
|
file.isPreviewAvailable = true;
|
|
|
|
file.icon = true;
|
|
|
|
}
|
|
|
|
file.share = {
|
|
|
|
id: share.id,
|
|
|
|
type: share.share_type,
|
2014-05-21 10:54:34 +00:00
|
|
|
target: share.share_with,
|
|
|
|
stime: share.stime * 1000,
|
2014-05-15 17:51:15 +00:00
|
|
|
};
|
|
|
|
if (self._sharedWithUser) {
|
2014-06-02 16:52:31 +00:00
|
|
|
file.shareOwner = share.displayname_owner;
|
2014-05-15 17:51:15 +00:00
|
|
|
file.name = OC.basename(share.file_target);
|
|
|
|
file.path = OC.dirname(share.file_target);
|
2014-05-19 16:33:43 +00:00
|
|
|
file.permissions = share.permissions;
|
2014-05-15 17:51:15 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
file.share.targetDisplayName = share.share_with_displayname;
|
|
|
|
file.name = OC.basename(share.path);
|
|
|
|
file.path = OC.dirname(share.path);
|
2014-05-19 16:33:43 +00:00
|
|
|
file.permissions = OC.PERMISSION_ALL;
|
2014-05-15 17:51:15 +00:00
|
|
|
}
|
|
|
|
return file;
|
|
|
|
})
|
|
|
|
// Group all files and have a "shares" array with
|
|
|
|
// the share info for each file.
|
|
|
|
//
|
|
|
|
// This uses a hash memo to cumulate share information
|
|
|
|
// inside the same file object (by file id).
|
|
|
|
.reduce(function(memo, file) {
|
|
|
|
var data = memo[file.id];
|
2014-05-28 16:39:29 +00:00
|
|
|
var recipient = file.share.targetDisplayName;
|
2014-05-15 17:51:15 +00:00
|
|
|
if (!data) {
|
|
|
|
data = memo[file.id] = file;
|
|
|
|
data.shares = [file.share];
|
2014-05-21 10:54:34 +00:00
|
|
|
// using a hash to make them unique,
|
|
|
|
// this is only a list to be displayed
|
2014-05-28 16:39:29 +00:00
|
|
|
data.recipients = {};
|
2014-05-21 10:54:34 +00:00
|
|
|
// counter is cheaper than calling _.keys().length
|
2014-05-28 16:39:29 +00:00
|
|
|
data.recipientsCount = 0;
|
2014-05-21 10:54:34 +00:00
|
|
|
data.mtime = file.share.stime;
|
2014-05-15 17:51:15 +00:00
|
|
|
}
|
|
|
|
else {
|
2014-05-21 10:54:34 +00:00
|
|
|
// always take the most recent stime
|
|
|
|
if (file.share.stime > data.mtime) {
|
|
|
|
data.mtime = file.share.stime;
|
|
|
|
}
|
2014-05-15 17:51:15 +00:00
|
|
|
data.shares.push(file.share);
|
|
|
|
}
|
2014-05-21 10:54:34 +00:00
|
|
|
|
2014-06-02 13:59:06 +00:00
|
|
|
if (recipient) {
|
2014-05-21 10:54:34 +00:00
|
|
|
// limit counterparts for output
|
2014-06-02 13:59:06 +00:00
|
|
|
if (data.recipientsCount < 4) {
|
2014-05-28 16:39:29 +00:00
|
|
|
// only store the first ones, they will be the only ones
|
|
|
|
// displayed
|
|
|
|
data.recipients[recipient] = true;
|
|
|
|
}
|
|
|
|
data.recipientsCount++;
|
2014-05-15 17:51:15 +00:00
|
|
|
}
|
2014-05-21 10:54:34 +00:00
|
|
|
|
2014-05-15 17:51:15 +00:00
|
|
|
delete file.share;
|
|
|
|
return memo;
|
|
|
|
}, {})
|
|
|
|
// Retrieve only the values of the returned hash
|
|
|
|
.values()
|
2014-05-21 10:54:34 +00:00
|
|
|
// Clean up
|
|
|
|
.each(function(data) {
|
2014-05-28 16:39:29 +00:00
|
|
|
// convert the recipients map to a flat
|
2014-05-21 10:54:34 +00:00
|
|
|
// array of sorted names
|
2014-06-02 13:59:06 +00:00
|
|
|
data.recipients = _.keys(data.recipients);
|
2014-05-28 16:39:29 +00:00
|
|
|
data.recipientsDisplayName = OCA.Sharing.Util.formatRecipients(
|
|
|
|
data.recipients,
|
|
|
|
data.recipientsCount
|
|
|
|
);
|
|
|
|
delete data.recipientsCount;
|
2014-05-21 10:54:34 +00:00
|
|
|
})
|
2014-05-15 17:51:15 +00:00
|
|
|
// Sort by expected sort comparator
|
|
|
|
.sortBy(this._sortComparator)
|
|
|
|
// Finish the chain by getting the result
|
|
|
|
.value();
|
|
|
|
|
|
|
|
return files;
|
2014-04-30 15:42:35 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
OCA.Sharing.FileList = FileList;
|
|
|
|
})();
|