UNPKG

@theia/navigator

Version:
222 lines (202 loc) • 8.63 kB
// ***************************************************************************** // Copyright (C) 2017 TypeFox and others. // // This program and the accompanying materials are made available under the // terms of the Eclipse Public License v. 2.0 which is available at // http://www.eclipse.org/legal/epl-2.0. // // This Source Code may also be made available under the following Secondary // Licenses when the conditions for such availability set forth in the Eclipse // Public License v. 2.0 are satisfied: GNU General Public License, version 2 // with the GNU Classpath Exception which is available at // https://www.gnu.org/software/classpath/license.html. // // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0 // ***************************************************************************** import { injectable, inject, postConstruct } from '@theia/core/shared/inversify'; import URI from '@theia/core/lib/common/uri'; import { FileNode, FileTreeModel } from '@theia/filesystem/lib/browser'; import { OpenerService, open, TreeNode, ExpandableTreeNode, CompositeTreeNode, SelectableTreeNode } from '@theia/core/lib/browser'; import { FileNavigatorTree, WorkspaceRootNode, WorkspaceNode } from './navigator-tree'; import { WorkspaceService } from '@theia/workspace/lib/browser'; import { FrontendApplicationStateService } from '@theia/core/lib/browser/frontend-application-state'; import { ProgressService } from '@theia/core/lib/common/progress-service'; import { Deferred } from '@theia/core/lib/common/promise-util'; import { Disposable } from '@theia/core/lib/common/disposable'; @injectable() export class FileNavigatorModel extends FileTreeModel { @inject(OpenerService) protected readonly openerService: OpenerService; @inject(FileNavigatorTree) protected override readonly tree: FileNavigatorTree; @inject(WorkspaceService) protected readonly workspaceService: WorkspaceService; @inject(FrontendApplicationStateService) protected readonly applicationState: FrontendApplicationStateService; @inject(ProgressService) protected readonly progressService: ProgressService; @postConstruct() protected override init(): void { super.init(); this.reportBusyProgress(); this.initializeRoot(); } protected readonly pendingBusyProgress = new Map<string, Deferred<void>>(); protected reportBusyProgress(): void { this.toDispose.push(this.onDidChangeBusy(node => { const pending = this.pendingBusyProgress.get(node.id); if (pending) { if (!node.busy) { pending.resolve(); this.pendingBusyProgress.delete(node.id); } return; } if (node.busy) { const progress = new Deferred<void>(); this.pendingBusyProgress.set(node.id, progress); this.progressService.withProgress('', 'explorer', () => progress.promise); } })); this.toDispose.push(Disposable.create(() => { for (const pending of this.pendingBusyProgress.values()) { pending.resolve(); } this.pendingBusyProgress.clear(); })); } protected async initializeRoot(): Promise<void> { await Promise.all([ this.applicationState.reachedState('initialized_layout'), this.workspaceService.roots ]); await this.updateRoot(); if (this.toDispose.disposed) { return; } this.toDispose.push(this.workspaceService.onWorkspaceChanged(() => this.updateRoot())); this.toDispose.push(this.workspaceService.onWorkspaceLocationChanged(() => this.updateRoot())); if (this.selectedNodes.length) { return; } const root = this.root; if (CompositeTreeNode.is(root) && root.children.length === 1) { const child = root.children[0]; if (SelectableTreeNode.is(child) && !child.selected && ExpandableTreeNode.is(child)) { this.selectNode(child); this.expandNode(child); } } } previewNode(node: TreeNode): void { if (FileNode.is(node)) { open(this.openerService, node.uri, { mode: 'reveal', preview: true }); } } protected override doOpenNode(node: TreeNode): void { if (node.visible === false) { return; } else if (FileNode.is(node)) { open(this.openerService, node.uri); } } override *getNodesByUri(uri: URI): IterableIterator<TreeNode> { const workspace = this.root; if (WorkspaceNode.is(workspace)) { for (const root of workspace.children) { const id = this.tree.createId(root, uri); const node = this.getNode(id); if (node) { yield node; } } } } protected async updateRoot(): Promise<void> { this.root = await this.createRoot(); } protected async createRoot(): Promise<TreeNode | undefined> { if (this.workspaceService.opened) { const stat = this.workspaceService.workspace; const isMulti = (stat) ? !stat.isDirectory : false; const workspaceNode = isMulti ? this.createMultipleRootNode() : WorkspaceNode.createRoot(); const roots = await this.workspaceService.roots; for (const root of roots) { workspaceNode.children.push( await this.tree.createWorkspaceRoot(root, workspaceNode) ); } return workspaceNode; } } /** * Create multiple root node used to display * the multiple root workspace name. * * @returns `WorkspaceNode` */ protected createMultipleRootNode(): WorkspaceNode { const workspace = this.workspaceService.workspace; let name = workspace ? workspace.resource.path.name : 'untitled'; name += ' (Workspace)'; return WorkspaceNode.createRoot(name); } /** * Move the given source file or directory to the given target directory. */ override async move(source: TreeNode, target: TreeNode): Promise<URI | undefined> { if (source.parent && WorkspaceRootNode.is(source)) { // do not support moving a root folder return undefined; } return super.move(source, target); } /** * Reveals node in the navigator by given file uri. * * @param uri uri to file which should be revealed in the navigator * @returns file tree node if the file with given uri was revealed, undefined otherwise */ async revealFile(uri: URI): Promise<TreeNode | undefined> { if (!uri.path.isAbsolute) { return undefined; } let node = this.getNodeClosestToRootByUri(uri); // success stop condition // we have to reach workspace root because expanded node could be inside collapsed one if (WorkspaceRootNode.is(node)) { if (ExpandableTreeNode.is(node)) { if (!node.expanded) { node = await this.expandNode(node); } return node; } // shouldn't happen, root node is always directory, i.e. expandable return undefined; } // fail stop condition if (uri.path.isRoot) { // file system root is reached but workspace root wasn't found, it means that // given uri is not in workspace root folder or points to not existing file. return undefined; } if (await this.revealFile(uri.parent)) { if (node === undefined) { // get node if it wasn't mounted into navigator tree before expansion node = this.getNodeClosestToRootByUri(uri); } if (ExpandableTreeNode.is(node) && !node.expanded) { node = await this.expandNode(node); } return node; } return undefined; } protected getNodeClosestToRootByUri(uri: URI): TreeNode | undefined { const nodes = [...this.getNodesByUri(uri)]; return nodes.length > 0 ? nodes.reduce((node1, node2) => // return the node closest to the workspace root node1.id.length >= node2.id.length ? node1 : node2 ) : undefined; } }