/** * SPDX-FileCopyrightText: 2016-2024 Nextcloud GmbH and Nextcloud contributors * SPDX-FileCopyrightText: 2011-2016 ownCloud, Inc. * SPDX-License-Identifier: AGPL-3.0-or-later */ /* eslint-disable */ import escapeHTML from 'escape-html' import { Type as ShareTypes } from '@nextcloud/sharing' import { getCapabilities } from '@nextcloud/capabilities' (function() { _.extend(OC.Files.Client, { PROPERTY_SHARE_TYPES: '{' + OC.Files.Client.NS_OWNCLOUD + '}share-types', PROPERTY_OWNER_ID: '{' + OC.Files.Client.NS_OWNCLOUD + '}owner-id', PROPERTY_OWNER_DISPLAY_NAME: '{' + OC.Files.Client.NS_OWNCLOUD + '}owner-display-name' }) if (!OCA.Sharing) { OCA.Sharing = {} } /** * @namespace */ OCA.Sharing.Util = { /** * Regular expression for splitting parts of remote share owners: * "user@example.com/" * "user@example.com/path/to/owncloud" * "user@anotherexample.com@example.com/path/to/owncloud */ _REMOTE_OWNER_REGEXP: new RegExp('^(([^@]*)@(([^@^/\\s]*)@)?)((https://)?[^[\\s/]*)([/](.*))?$'), /** * Initialize the sharing plugin. * * Registers the "Share" file action and adds additional * DOM attributes for the sharing file info. * * @param {OCA.Files.FileList} fileList file list to be extended */ attach: function(fileList) { // core sharing is disabled/not loaded if (!getCapabilities().files_sharing?.api_enabled) { return } if (fileList.id === 'trashbin' || fileList.id === 'files.public') { return } var fileActions = fileList.fileActions var oldCreateRow = fileList._createRow fileList._createRow = function(fileData) { var tr = oldCreateRow.apply(this, arguments) var sharePermissions = OCA.Sharing.Util.getSharePermissions(fileData) if (fileData.permissions === 0) { // no permission, disabling sidebar delete fileActions.actions.all.Comment delete fileActions.actions.all.Details delete fileActions.actions.all.Goto } if (_.isFunction(fileData.canDownload) && !fileData.canDownload()) { delete fileActions.actions.all.Download if ((fileData.permissions & OC.PERMISSION_UPDATE) === 0) { // neither move nor copy is allowed, remove the action completely delete fileActions.actions.all.MoveCopy } } tr.attr('data-share-permissions', sharePermissions) tr.attr('data-share-attributes', JSON.stringify(fileData.shareAttributes)) if (fileData.shareOwner) { tr.attr('data-share-owner', fileData.shareOwner) tr.attr('data-share-owner-id', fileData.shareOwnerId) // user should always be able to rename a mount point if (fileData.mountType === 'shared-root') { tr.attr('data-permissions', fileData.permissions | OC.PERMISSION_UPDATE) } } if (fileData.recipientData && !_.isEmpty(fileData.recipientData)) { tr.attr('data-share-recipient-data', JSON.stringify(fileData.recipientData)) } if (fileData.shareTypes) { tr.attr('data-share-types', fileData.shareTypes.join(',')) } return tr } var oldElementToFile = fileList.elementToFile fileList.elementToFile = function($el) { var fileInfo = oldElementToFile.apply(this, arguments) fileInfo.shareAttributes = JSON.parse($el.attr('data-share-attributes') || '[]') fileInfo.sharePermissions = $el.attr('data-share-permissions') || undefined fileInfo.shareOwner = $el.attr('data-share-owner') || undefined fileInfo.shareOwnerId = $el.attr('data-share-owner-id') || undefined if ($el.attr('data-share-types')) { fileInfo.shareTypes = $el.attr('data-share-types').split(',') } if ($el.attr('data-expiration')) { var expirationTimestamp = parseInt($el.attr('data-expiration')) fileInfo.shares = [] fileInfo.shares.push({ expiration: expirationTimestamp }) } return fileInfo } var oldGetWebdavProperties = fileList._getWebdavProperties fileList._getWebdavProperties = function() { var props = oldGetWebdavProperties.apply(this, arguments) props.push(OC.Files.Client.PROPERTY_OWNER_ID) props.push(OC.Files.Client.PROPERTY_OWNER_DISPLAY_NAME) props.push(OC.Files.Client.PROPERTY_SHARE_TYPES) return props } fileList.filesClient.addFileInfoParser(function(response) { var data = {} var props = response.propStat[0].properties var permissionsProp = props[OC.Files.Client.PROPERTY_PERMISSIONS] if (permissionsProp && permissionsProp.indexOf('S') >= 0) { data.shareOwner = props[OC.Files.Client.PROPERTY_OWNER_DISPLAY_NAME] data.shareOwnerId = props[OC.Files.Client.PROPERTY_OWNER_ID] } var shareTypesProp = props[OC.Files.Client.PROPERTY_SHARE_TYPES] if (shareTypesProp) { data.shareTypes = _.chain(shareTypesProp).filter(function(xmlvalue) { return (xmlvalue.namespaceURI === OC.Files.Client.NS_OWNCLOUD && xmlvalue.nodeName.split(':')[1] === 'share-type') }).map(function(xmlvalue) { return parseInt(xmlvalue.textContent || xmlvalue.text, 10) }).value() } return data }) // use delegate to catch the case with multiple file lists fileList.$el.on('fileActionsReady', function(ev) { var $files = ev.$files _.each($files, function(file) { var $tr = $(file) var shareTypesStr = $tr.attr('data-share-types') || '' var shareOwner = $tr.attr('data-share-owner') if (shareTypesStr || shareOwner) { var hasLink = false var hasShares = false _.each(shareTypesStr.split(',') || [], function(shareTypeStr) { let shareType = parseInt(shareTypeStr, 10) if (shareType === ShareTypes.SHARE_TYPE_LINK) { hasLink = true } else if (shareType === ShareTypes.SHARE_TYPE_EMAIL) { hasLink = true } else if (shareType === ShareTypes.SHARE_TYPE_USER) { hasShares = true } else if (shareType === ShareTypes.SHARE_TYPE_GROUP) { hasShares = true } else if (shareType === ShareTypes.SHARE_TYPE_REMOTE) { hasShares = true } else if (shareType === ShareTypes.SHARE_TYPE_REMOTE_GROUP) { hasShares = true } else if (shareType === ShareTypes.SHARE_TYPE_CIRCLE) { hasShares = true } else if (shareType === ShareTypes.SHARE_TYPE_ROOM) { hasShares = true } else if (shareType === ShareTypes.SHARE_TYPE_DECK) { hasShares = true } else if (shareType === ShareTypes.SHARE_TYPE_SCIENCEMESH) { hasShares = true } }) OCA.Sharing.Util._updateFileActionIcon($tr, hasShares, hasLink) } }) }) fileList.$el.on('changeDirectory', function() { OCA.Sharing.sharesLoaded = false }) fileActions.registerAction({ name: 'Share', displayName: function(context) { if (context && context.$file) { var shareType = parseInt(context.$file.data('share-types'), 10) var shareOwner = context.$file.data('share-owner-id') if (shareType >= 0 || shareOwner) { return t('files_sharing', 'Shared') } } return t('files_sharing', 'Share') }, altText: t('files_sharing', 'Share'), mime: 'all', order: -150, permissions: OC.PERMISSION_ALL, iconClass: function(fileName, context) { var shareType = parseInt(context.$file.data('share-types'), 10) if (shareType === ShareTypes.SHARE_TYPE_EMAIL || shareType === ShareTypes.SHARE_TYPE_LINK) { return 'icon-public' } return 'icon-shared' }, icon: function(fileName, context) { var shareOwner = context.$file.data('share-owner-id') if (shareOwner) { return OC.generateUrl(`/avatar/${shareOwner}/32`) } }, type: OCA.Files.FileActions.TYPE_INLINE, actionHandler: function(fileName, context) { // details view disabled in some share lists if (!fileList._detailsView) { return } // do not open sidebar if permission is set and equal to 0 var permissions = parseInt(context.$file.data('share-permissions'), 10) if (isNaN(permissions) || permissions > 0) { fileList.showDetailsView(fileName, 'sharing') } }, render: function(actionSpec, isDefault, context) { var permissions = parseInt(context.$file.data('permissions'), 10) // if no share permissions but share owner exists, still show the link if ((permissions & OC.PERMISSION_SHARE) !== 0 || context.$file.attr('data-share-owner')) { return fileActions._defaultRenderAction.call(fileActions, actionSpec, isDefault, context) } // don't render anything return null } }) // register share breadcrumbs component var breadCrumbSharingDetailView = new OCA.Sharing.ShareBreadCrumbView() fileList.registerBreadCrumbDetailView(breadCrumbSharingDetailView) }, /** * Update file list data attributes */ _updateFileListDataAttributes: function(fileList, $tr, shareModel) { // files app current cannot show recipients on load, so we don't update the // icon when changed for consistency if (fileList.id === 'files') { return } var recipients = _.pluck(shareModel.get('shares'), 'share_with_displayname') // note: we only update the data attribute because updateIcon() if (recipients.length) { var recipientData = _.mapObject(shareModel.get('shares'), function(share) { return { shareWith: share.share_with, shareWithDisplayName: share.share_with_displayname } }) $tr.attr('data-share-recipient-data', JSON.stringify(recipientData)) } else { $tr.removeAttr('data-share-recipient-data') } }, /** * Update the file action share icon for the given file * * @param $tr file element of the file to update * @param {boolean} hasUserShares true if a user share exists * @param {boolean} hasLinkShares true if a link share exists * * @returns {boolean} true if the icon was set, false otherwise */ _updateFileActionIcon: function($tr, hasUserShares, hasLinkShares) { // if the statuses are loaded already, use them for the icon // (needed when scrolling to the next page) if (hasUserShares || hasLinkShares || $tr.attr('data-share-recipient-data') || $tr.attr('data-share-owner')) { OCA.Sharing.Util._markFileAsShared($tr, true, hasLinkShares) return true } return false }, /** * Marks/unmarks a given file as shared by changing its action icon * and folder icon. * * @param $tr file element to mark as shared * @param hasShares whether shares are available * @param hasLink whether link share is available */ _markFileAsShared: function($tr, hasShares, hasLink) { var action = $tr.find('.fileactions .action[data-action="Share"]') var type = $tr.data('type') var icon = action.find('.icon') var message, recipients, avatars var ownerId = $tr.attr('data-share-owner-id') var owner = $tr.attr('data-share-owner') var mountType = $tr.attr('data-mounttype') var shareFolderIcon var iconClass = 'icon-shared' action.removeClass('shared-style') // update folder icon var isEncrypted = $tr.attr('data-e2eencrypted') if (type === 'dir' && isEncrypted === 'true') { shareFolderIcon = OC.MimeType.getIconUrl('dir-encrypted') $tr.attr('data-icon', shareFolderIcon) } else if (type === 'dir' && (hasShares || hasLink || ownerId)) { if (typeof mountType !== 'undefined' && mountType !== 'shared-root' && mountType !== 'shared') { shareFolderIcon = OC.MimeType.getIconUrl('dir-' + mountType) } else if (hasLink) { shareFolderIcon = OC.MimeType.getIconUrl('dir-public') } else { shareFolderIcon = OC.MimeType.getIconUrl('dir-shared') } $tr.find('.filename .thumbnail').css('background-image', 'url(' + shareFolderIcon + ')') $tr.attr('data-icon', shareFolderIcon) } else if (type === 'dir') { // FIXME: duplicate of FileList._createRow logic for external folder, // need to refactor the icon logic into a single code path eventually if (mountType && mountType.indexOf('external') === 0) { shareFolderIcon = OC.MimeType.getIconUrl('dir-external') $tr.attr('data-icon', shareFolderIcon) } else { shareFolderIcon = OC.MimeType.getIconUrl('dir') // back to default $tr.removeAttr('data-icon') } $tr.find('.filename .thumbnail').css('background-image', 'url(' + shareFolderIcon + ')') } // update share action text / icon if (hasShares || ownerId) { recipients = $tr.data('share-recipient-data') action.addClass('shared-style') avatars = '' + t('files_sharing', 'Shared') + '' // even if reshared, only show "Shared by" if (ownerId) { message = t('files_sharing', 'Shared by') avatars = OCA.Sharing.Util._formatRemoteShare(ownerId, owner, message) } else if (recipients) { avatars = OCA.Sharing.Util._formatShareList(recipients) } action.html(avatars).prepend(icon) if (ownerId || recipients) { var avatarElement = action.find('.avatar') avatarElement.each(function() { $(this).avatar($(this).data('username'), 32) }) } } else { action.html('' + t('files_sharing', 'Shared') + '').prepend(icon) } if (hasLink) { iconClass = 'icon-public' } icon.removeClass('icon-shared icon-public').addClass(iconClass) }, /** * Format a remote address * * @param {String} shareWith userid, full remote share, or whatever * @param {String} shareWithDisplayName * @param {String} message * @returns {String} HTML code to display */ _formatRemoteShare: function(shareWith, shareWithDisplayName, message) { var parts = OCA.Sharing.Util._REMOTE_OWNER_REGEXP.exec(shareWith) if (!parts || !parts[7]) { // display avatar of the user var avatar = '' var hidden = '' + message + ' ' + escapeHTML(shareWithDisplayName) + ' ' return avatar + hidden } var userName = parts[2] var userDomain = parts[4] var server = parts[5] var protocol = parts[6] var serverPath = parts[8] ? parts[7] : ''; // no trailing slash on root var tooltip = message + ' ' + userName if (userDomain) { tooltip += '@' + userDomain } if (server) { tooltip += '@' + server.replace(protocol, '') + serverPath } var html = '' html += '' + escapeHTML(userName) + '' if (userDomain) { html += '@' + escapeHTML(userDomain) + '' } html += ' ' return html }, /** * Loop over all recipients in the list and format them using * all kind of fancy magic. * * @param {Object} recipients array of all the recipients * @returns {String[]} modified list of recipients */ _formatShareList: function(recipients) { var _parent = this recipients = _.toArray(recipients) recipients.sort(function(a, b) { return a.shareWithDisplayName.localeCompare(b.shareWithDisplayName) }) return $.map(recipients, function(recipient) { return _parent._formatRemoteShare(recipient.shareWith, recipient.shareWithDisplayName, t('files_sharing', 'Shared with')) }) }, /** * Marks/unmarks a given file as shared by changing its action icon * and folder icon. * * @param $tr file element to mark as shared * @param hasShares whether shares are available * @param hasLink whether link share is available */ markFileAsShared: function($tr, hasShares, hasLink) { var action = $tr.find('.fileactions .action[data-action="Share"]') var type = $tr.data('type') var icon = action.find('.icon') var message, recipients, avatars var ownerId = $tr.attr('data-share-owner-id') var owner = $tr.attr('data-share-owner') var mountType = $tr.attr('data-mounttype') var shareFolderIcon var iconClass = 'icon-shared' action.removeClass('shared-style') // update folder icon if (type === 'dir' && (hasShares || hasLink || ownerId)) { if (typeof mountType !== 'undefined' && mountType !== 'shared-root' && mountType !== 'shared') { shareFolderIcon = OC.MimeType.getIconUrl('dir-' + mountType) } else if (hasLink) { shareFolderIcon = OC.MimeType.getIconUrl('dir-public') } else { shareFolderIcon = OC.MimeType.getIconUrl('dir-shared') } $tr.find('.filename .thumbnail').css('background-image', 'url(' + shareFolderIcon + ')') $tr.attr('data-icon', shareFolderIcon) } else if (type === 'dir') { var isEncrypted = $tr.attr('data-e2eencrypted') // FIXME: duplicate of FileList._createRow logic for external folder, // need to refactor the icon logic into a single code path eventually if (isEncrypted === 'true') { shareFolderIcon = OC.MimeType.getIconUrl('dir-encrypted') $tr.attr('data-icon', shareFolderIcon) } else if (mountType && mountType.indexOf('external') === 0) { shareFolderIcon = OC.MimeType.getIconUrl('dir-external') $tr.attr('data-icon', shareFolderIcon) } else { shareFolderIcon = OC.MimeType.getIconUrl('dir') // back to default $tr.removeAttr('data-icon') } $tr.find('.filename .thumbnail').css('background-image', 'url(' + shareFolderIcon + ')') } // update share action text / icon if (hasShares || ownerId) { recipients = $tr.data('share-recipient-data') action.addClass('shared-style') avatars = '' + t('files_sharing', 'Shared') + '' // even if reshared, only show "Shared by" if (ownerId) { message = t('files_sharing', 'Shared by') avatars = this._formatRemoteShare(ownerId, owner, message) } else if (recipients) { avatars = this._formatShareList(recipients) } action.html(avatars).prepend(icon) if (ownerId || recipients) { var avatarElement = action.find('.avatar') avatarElement.each(function() { $(this).avatar($(this).data('username'), 32) }) } } else { action.html('' + t('files_sharing', 'Shared') + '').prepend(icon) } if (hasLink) { iconClass = 'icon-public' } icon.removeClass('icon-shared icon-public').addClass(iconClass) }, /** * @param {Array} fileData * @returns {String} */ getSharePermissions: function(fileData) { return fileData.sharePermissions } } })() OC.Plugins.register('OCA.Files.FileList', OCA.Sharing.Util)