parent
79d24bfb8e
commit
0984970cd8
@ -0,0 +1,128 @@ |
||||
/** |
||||
* @copyright Copyright (c) 2023 John Molakvoæ <skjnldsv@protonmail.com> |
||||
* |
||||
* @author John Molakvoæ <skjnldsv@protonmail.com> |
||||
* |
||||
* @license AGPL-3.0-or-later |
||||
* |
||||
* This program is free software: you can redistribute it and/or modify |
||||
* it under the terms of the GNU Affero General Public License as |
||||
* published by the Free Software Foundation, either version 3 of the |
||||
* License, or (at your option) any later version. |
||||
* |
||||
* This program is distributed in the hope that it will be useful, |
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
||||
* GNU Affero General Public License for more details. |
||||
* |
||||
* You should have received a copy of the GNU Affero General Public License |
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>. |
||||
* |
||||
*/ |
||||
|
||||
import logger from '../logger' |
||||
|
||||
type DavProperty = { [key: string]: string } |
||||
|
||||
declare global { |
||||
interface Window { |
||||
OC: any; |
||||
_nc_dav_properties: string[]; |
||||
_nc_dav_namespaces: DavProperty; |
||||
} |
||||
} |
||||
|
||||
const defaultDavProperties = [ |
||||
'd:getcontentlength', |
||||
'd:getcontenttype', |
||||
'd:getetag', |
||||
'd:getlastmodified', |
||||
'd:quota-available-bytes', |
||||
'd:resourcetype', |
||||
'nc:has-preview', |
||||
'nc:is-encrypted', |
||||
'nc:mount-type', |
||||
'nc:share-attributes', |
||||
'oc:comments-unread', |
||||
'oc:favorite', |
||||
'oc:fileid', |
||||
'oc:owner-display-name', |
||||
'oc:owner-id', |
||||
'oc:permissions', |
||||
'oc:share-types', |
||||
'oc:size', |
||||
'ocs:share-permissions', |
||||
] |
||||
|
||||
const defaultDavNamespaces = { |
||||
d: 'DAV:', |
||||
nc: 'http://nextcloud.org/ns', |
||||
oc: 'http://owncloud.org/ns', |
||||
ocs: 'http://open-collaboration-services.org/ns', |
||||
} |
||||
|
||||
/** |
||||
* TODO: remove and move to @nextcloud/files |
||||
*/ |
||||
export const registerDavProperty = function(prop: string, namespace: DavProperty = { nc: 'http://nextcloud.org/ns' }): void { |
||||
if (typeof window._nc_dav_properties === 'undefined') { |
||||
window._nc_dav_properties = defaultDavProperties |
||||
window._nc_dav_namespaces = defaultDavNamespaces |
||||
} |
||||
|
||||
const namespaces = { ...window._nc_dav_namespaces, ...namespace } |
||||
|
||||
// Check duplicates
|
||||
if (window._nc_dav_properties.find(search => search === prop)) { |
||||
logger.error(`${prop} already registered`, { prop }) |
||||
return |
||||
} |
||||
|
||||
if (prop.startsWith('<') || prop.split(':').length !== 2) { |
||||
logger.error(`${prop} is not valid. See example: 'oc:fileid'`, { prop }) |
||||
return |
||||
} |
||||
|
||||
const ns = prop.split(':')[0] |
||||
if (!namespaces[ns]) { |
||||
logger.error(`${prop} namespace unknown`, { prop, namespaces }) |
||||
return |
||||
} |
||||
|
||||
window._nc_dav_properties.push(prop) |
||||
window._nc_dav_namespaces = namespaces |
||||
} |
||||
|
||||
/** |
||||
* Get the registered dav properties |
||||
*/ |
||||
export const getDavProperties = function(): string { |
||||
if (typeof window._nc_dav_properties === 'undefined') { |
||||
window._nc_dav_properties = defaultDavProperties |
||||
} |
||||
|
||||
return window._nc_dav_properties.map(prop => `<${prop} />`).join(' ') |
||||
} |
||||
|
||||
/** |
||||
* Get the registered dav namespaces |
||||
*/ |
||||
export const getDavNameSpaces = function(): string { |
||||
if (typeof window._nc_dav_namespaces === 'undefined') { |
||||
window._nc_dav_namespaces = defaultDavNamespaces |
||||
} |
||||
|
||||
return Object.keys(window._nc_dav_namespaces).map(ns => `xmlns:${ns}="${window._nc_dav_namespaces[ns]}"`).join(' ') |
||||
} |
||||
|
||||
/** |
||||
* Get the default PROPFIND request payload |
||||
*/ |
||||
export const getDefaultPropfind = function() { |
||||
return `<?xml version="1.0"?>
|
||||
<d:propfind ${getDavNameSpaces()}> |
||||
<d:prop> |
||||
${getDavProperties()} |
||||
</d:prop> |
||||
</d:propfind>` |
||||
} |
||||
@ -0,0 +1,100 @@ |
||||
/** |
||||
* @copyright Copyright (c) 2023 John Molakvoæ <skjnldsv@protonmail.com> |
||||
* |
||||
* @author John Molakvoæ <skjnldsv@protonmail.com> |
||||
* |
||||
* @license AGPL-3.0-or-later |
||||
* |
||||
* This program is free software: you can redistribute it and/or modify |
||||
* it under the terms of the GNU Affero General Public License as |
||||
* published by the Free Software Foundation, either version 3 of the |
||||
* License, or (at your option) any later version. |
||||
* |
||||
* This program is distributed in the hope that it will be useful, |
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
||||
* GNU Affero General Public License for more details. |
||||
* |
||||
* You should have received a copy of the GNU Affero General Public License |
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>. |
||||
* |
||||
*/ |
||||
import { File, Folder, parseWebdavPermissions } from '@nextcloud/files' |
||||
import { generateRemoteUrl, generateUrl } from '@nextcloud/router' |
||||
import { getClient, rootPath } from './WebdavClient' |
||||
import { getCurrentUser } from '@nextcloud/auth' |
||||
import { getDavNameSpaces, getDavProperties, getDefaultPropfind } from './DavProperties' |
||||
import type { ContentsWithRoot } from './Navigation' |
||||
import type { FileStat, ResponseDataDetailed } from 'webdav' |
||||
|
||||
const client = getClient() |
||||
|
||||
const reportPayload = `<?xml version="1.0"?>
|
||||
<oc:filter-files ${getDavNameSpaces()}> |
||||
<d:prop> |
||||
${getDavProperties()} |
||||
</d:prop> |
||||
<oc:filter-rules> |
||||
<oc:favorite>1</oc:favorite> |
||||
</oc:filter-rules> |
||||
</oc:filter-files>` |
||||
|
||||
const resultToNode = function(node: FileStat): File | Folder { |
||||
const permissions = parseWebdavPermissions(node.props?.permissions) |
||||
const owner = getCurrentUser()?.uid as string |
||||
const previewUrl = generateUrl('/core/preview?fileId={fileid}&x=32&y=32&forceIcon=0', node.props) |
||||
|
||||
const nodeData = { |
||||
id: node.props?.fileid as number || 0, |
||||
source: generateRemoteUrl('dav' + rootPath + node.filename), |
||||
mtime: new Date(node.lastmod), |
||||
mime: node.mime as string, |
||||
size: node.props?.size as number || 0, |
||||
permissions, |
||||
owner, |
||||
root: rootPath, |
||||
attributes: { |
||||
...node, |
||||
...node.props, |
||||
previewUrl, |
||||
}, |
||||
} |
||||
|
||||
delete nodeData.attributes.props |
||||
|
||||
return node.type === 'file' |
||||
? new File(nodeData) |
||||
: new Folder(nodeData) |
||||
} |
||||
|
||||
export const getContents = async (path = '/'): Promise<ContentsWithRoot> => { |
||||
const propfindPayload = getDefaultPropfind() |
||||
|
||||
// Get root folder
|
||||
let rootResponse |
||||
if (path === '/') { |
||||
rootResponse = await client.stat(path, { |
||||
details: true, |
||||
data: getDefaultPropfind(), |
||||
}) as ResponseDataDetailed<FileStat> |
||||
} |
||||
|
||||
const contentsResponse = await client.getDirectoryContents(path, { |
||||
details: true, |
||||
// Only filter favorites if we're at the root
|
||||
data: path === '/' ? reportPayload : propfindPayload, |
||||
headers: { |
||||
// Patched in WebdavClient.ts
|
||||
method: path === '/' ? 'REPORT' : 'PROPFIND', |
||||
}, |
||||
includeSelf: true, |
||||
}) as ResponseDataDetailed<FileStat[]> |
||||
|
||||
const root = rootResponse?.data || contentsResponse.data[0] |
||||
const contents = contentsResponse.data.filter(node => node.filename !== path) |
||||
|
||||
return { |
||||
folder: resultToNode(root) as Folder, |
||||
contents: contents.map(resultToNode), |
||||
} |
||||
} |
||||
@ -0,0 +1,51 @@ |
||||
/** |
||||
* @copyright Copyright (c) 2023 John Molakvoæ <skjnldsv@protonmail.com> |
||||
* |
||||
* @author John Molakvoæ <skjnldsv@protonmail.com> |
||||
* |
||||
* @license AGPL-3.0-or-later |
||||
* |
||||
* This program is free software: you can redistribute it and/or modify |
||||
* it under the terms of the GNU Affero General Public License as |
||||
* published by the Free Software Foundation, either version 3 of the |
||||
* License, or (at your option) any later version. |
||||
* |
||||
* This program is distributed in the hope that it will be useful, |
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
||||
* GNU Affero General Public License for more details. |
||||
* |
||||
* You should have received a copy of the GNU Affero General Public License |
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>. |
||||
* |
||||
*/ |
||||
import { createClient, getPatcher, RequestOptions } from 'webdav' |
||||
import { request } from '../../../../node_modules/webdav/dist/node/request.js' |
||||
import { generateRemoteUrl } from '@nextcloud/router' |
||||
import { getCurrentUser, getRequestToken } from '@nextcloud/auth' |
||||
|
||||
export const rootPath = `/files/${getCurrentUser()?.uid}` |
||||
export const defaultRootUrl = generateRemoteUrl('dav' + rootPath) |
||||
|
||||
export const getClient = (rootUrl = defaultRootUrl) => { |
||||
const client = createClient(rootUrl, { |
||||
headers: { |
||||
requesttoken: getRequestToken() || '', |
||||
}, |
||||
}) |
||||
|
||||
/** |
||||
* Allow to override the METHOD to support dav REPORT |
||||
* |
||||
* @see https://github.com/perry-mitchell/webdav-client/blob/8d9694613c978ce7404e26a401c39a41f125f87f/source/request.ts
|
||||
*/ |
||||
const patcher = getPatcher() |
||||
patcher.patch('request', (options: RequestOptions) => { |
||||
if (options.headers?.method) { |
||||
options.method = options.headers.method |
||||
delete options.headers.method |
||||
} |
||||
return request(options) |
||||
}) |
||||
return client |
||||
} |
||||
@ -0,0 +1,114 @@ |
||||
/** |
||||
* @copyright Copyright (c) 2023 John Molakvoæ <skjnldsv@protonmail.com> |
||||
* |
||||
* @author John Molakvoæ <skjnldsv@protonmail.com> |
||||
* |
||||
* @license AGPL-3.0-or-later |
||||
* |
||||
* This program is free software: you can redistribute it and/or modify |
||||
* it under the terms of the GNU Affero General Public License as |
||||
* published by the Free Software Foundation, either version 3 of the |
||||
* License, or (at your option) any later version. |
||||
* |
||||
* This program is distributed in the hope that it will be useful, |
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
||||
* GNU Affero General Public License for more details. |
||||
* |
||||
* You should have received a copy of the GNU Affero General Public License |
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>. |
||||
* |
||||
*/ |
||||
import type NavigationService from '../services/Navigation.ts' |
||||
import type { Navigation } from '../services/Navigation.ts' |
||||
import { translate as t } from '@nextcloud/l10n' |
||||
import StarSvg from '@mdi/svg/svg/star.svg?raw' |
||||
import FolderSvg from '@mdi/svg/svg/folder.svg?raw' |
||||
|
||||
import { getContents } from '../services/Favorites.ts' |
||||
import { loadState } from '@nextcloud/initial-state' |
||||
import { basename } from 'path' |
||||
import { hashCode } from '../utils/hashUtils' |
||||
import { subscribe } from '@nextcloud/event-bus' |
||||
import { Node, FileType } from '@nextcloud/files' |
||||
import logger from '../logger' |
||||
|
||||
const favoriteFolders = loadState('files', 'favoriteFolders', []) |
||||
|
||||
export default () => { |
||||
const Navigation = window.OCP.Files.Navigation as NavigationService |
||||
Navigation.register({ |
||||
id: 'favorites', |
||||
name: t('files', 'Favorites'), |
||||
caption: t('files', 'List of favorites files and folders.'), |
||||
|
||||
icon: StarSvg, |
||||
order: 5, |
||||
|
||||
columns: [], |
||||
|
||||
getContents, |
||||
} as Navigation) |
||||
|
||||
favoriteFolders.forEach((folder) => { |
||||
Navigation.register(generateFolderView(folder)) |
||||
}) |
||||
|
||||
/** |
||||
* Update favourites navigation when a new folder is added |
||||
*/ |
||||
subscribe('files:favorites:added', (node: Node) => { |
||||
if (node.type !== FileType.Folder) { |
||||
return |
||||
} |
||||
|
||||
// Sanity check
|
||||
if (node.path === null || !node.root?.startsWith('/files')) { |
||||
logger.error('Favorite folder is not within user files root', { node }) |
||||
return |
||||
} |
||||
|
||||
Navigation.register(generateFolderView(node.path)) |
||||
}) |
||||
|
||||
/** |
||||
* Remove favourites navigation when a folder is removed |
||||
*/ |
||||
subscribe('files:favorites:removed', (node: Node) => { |
||||
if (node.type !== FileType.Folder) { |
||||
return |
||||
} |
||||
|
||||
// Sanity check
|
||||
if (node.path === null || !node.root?.startsWith('/files')) { |
||||
logger.error('Favorite folder is not within user files root', { node }) |
||||
return |
||||
} |
||||
|
||||
Navigation.remove(generateIdFromPath(node.path)) |
||||
}) |
||||
} |
||||
|
||||
const generateFolderView = function(folder: string): Navigation { |
||||
return { |
||||
id: generateIdFromPath(folder), |
||||
name: basename(folder), |
||||
|
||||
icon: FolderSvg, |
||||
order: -100, // always first
|
||||
params: { |
||||
dir: folder, |
||||
view: 'favorites', |
||||
}, |
||||
|
||||
parent: 'favorites', |
||||
|
||||
columns: [], |
||||
|
||||
getContents, |
||||
} as Navigation |
||||
} |
||||
|
||||
const generateIdFromPath = function(path: string): string { |
||||
return `favorite-${hashCode(path)}` |
||||
} |
||||
Loading…
Reference in new issue