aboutsummaryrefslogtreecommitdiffstats
path: root/apps/files/src/views/folderTree.ts
diff options
context:
space:
mode:
Diffstat (limited to 'apps/files/src/views/folderTree.ts')
-rw-r--r--apps/files/src/views/folderTree.ts153
1 files changed, 153 insertions, 0 deletions
diff --git a/apps/files/src/views/folderTree.ts b/apps/files/src/views/folderTree.ts
new file mode 100644
index 00000000000..a466a838f66
--- /dev/null
+++ b/apps/files/src/views/folderTree.ts
@@ -0,0 +1,153 @@
+/**
+ * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
+ * SPDX-License-Identifier: AGPL-3.0-or-later
+ */
+
+import type { TreeNode } from '../services/FolderTree.ts'
+
+import { Folder, Node, View, getNavigation } from '@nextcloud/files'
+import { translate as t } from '@nextcloud/l10n'
+import { subscribe } from '@nextcloud/event-bus'
+import { isSamePath } from '@nextcloud/paths'
+import { loadState } from '@nextcloud/initial-state'
+
+import FolderSvg from '@mdi/svg/svg/folder.svg?raw'
+import FolderMultipleSvg from '@mdi/svg/svg/folder-multiple.svg?raw'
+
+import {
+ encodeSource,
+ folderTreeId,
+ getContents,
+ getFolderTreeNodes,
+ getFolderTreeParentId,
+ getFolderTreeViewId,
+ getSourceParent,
+ sourceRoot,
+} from '../services/FolderTree.ts'
+
+const isFolderTreeEnabled = loadState('files', 'config', { folder_tree: true }).folder_tree
+
+const Navigation = getNavigation()
+
+const registerTreeNodeView = (node: TreeNode) => {
+ Navigation.register(new View({
+ id: encodeSource(node.source),
+ parent: getSourceParent(node.source),
+
+ name: node.displayName ?? node.basename,
+
+ icon: FolderSvg,
+ order: 0, // TODO Allow undefined order for natural sort
+
+ getContents,
+
+ params: {
+ view: folderTreeId,
+ fileid: String(node.fileid), // Needed for matching exact routes
+ dir: node.path,
+ },
+ }))
+}
+
+const registerFolderView = (folder: Folder) => {
+ Navigation.register(new View({
+ id: getFolderTreeViewId(folder),
+ parent: getFolderTreeParentId(folder),
+
+ name: folder.displayname,
+
+ icon: FolderSvg,
+ order: 0, // TODO Allow undefined order for natural sort
+
+ getContents,
+
+ params: {
+ view: folderTreeId,
+ fileid: String(folder.fileid),
+ dir: folder.path,
+ },
+ }))
+}
+
+const removeFolderView = (folder: Folder) => {
+ const viewId = getFolderTreeViewId(folder)
+ Navigation.remove(viewId)
+}
+
+const removeFolderViewSource = (source: string) => {
+ const Navigation = getNavigation()
+ Navigation.remove(source)
+}
+
+const onCreateNode = (node: Node) => {
+ if (!(node instanceof Folder)) {
+ return
+ }
+ registerFolderView(node)
+}
+
+const onDeleteNode = (node: Node) => {
+ if (!(node instanceof Folder)) {
+ return
+ }
+ removeFolderView(node)
+}
+
+const onMoveNode = ({ node, oldSource }) => {
+ if (!(node instanceof Folder)) {
+ return
+ }
+ removeFolderViewSource(oldSource)
+ registerFolderView(node)
+
+ const newPath = node.source.replace(sourceRoot, '')
+ const oldPath = oldSource.replace(sourceRoot, '')
+ const childViews = Navigation.views.filter(view => {
+ if (!view.params?.dir) {
+ return false
+ }
+ if (isSamePath(view.params.dir, oldPath)) {
+ return false
+ }
+ return view.params.dir.startsWith(oldPath)
+ })
+ for (const view of childViews) {
+ // @ts-expect-error FIXME Allow setting parent
+ view.parent = getFolderTreeParentId(node)
+ // @ts-expect-error dir param is defined
+ view.params.dir = view.params.dir.replace(oldPath, newPath)
+ }
+}
+
+const registerFolderTreeRoot = () => {
+ Navigation.register(new View({
+ id: folderTreeId,
+
+ name: t('files', 'All folders'),
+ caption: t('files', 'List of your files and folders.'),
+
+ icon: FolderMultipleSvg,
+ order: 50, // Below all other views
+
+ getContents,
+ }))
+}
+
+const registerFolderTreeChildren = async () => {
+ const nodes = await getFolderTreeNodes()
+ for (const node of nodes) {
+ registerTreeNodeView(node)
+ }
+
+ subscribe('files:node:created', onCreateNode)
+ subscribe('files:node:deleted', onDeleteNode)
+ subscribe('files:node:moved', onMoveNode)
+}
+
+export const registerFolderTreeView = async () => {
+ if (!isFolderTreeEnabled) {
+ return
+ }
+ registerFolderTreeRoot()
+ await registerFolderTreeChildren()
+}