163 lines
5.9 KiB
TypeScript
163 lines
5.9 KiB
TypeScript
/*---------------------------------------------------------------------------------------------
|
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
* Licensed under the MIT License. See License.txt in the project root for license information.
|
|
*--------------------------------------------------------------------------------------------*/
|
|
|
|
import * as vscode from 'vscode';
|
|
import { Disposable } from '../util/dispose';
|
|
import { isMarkdownFile } from '../util/file';
|
|
import { Lazy, lazy } from '../util/lazy';
|
|
import MDDocumentSymbolProvider from './documentSymbolProvider';
|
|
import { SkinnyTextDocument, SkinnyTextLine } from '../tableOfContentsProvider';
|
|
|
|
export interface WorkspaceMarkdownDocumentProvider {
|
|
getAllMarkdownDocuments(): Thenable<Iterable<SkinnyTextDocument>>;
|
|
|
|
readonly onDidChangeMarkdownDocument: vscode.Event<SkinnyTextDocument>;
|
|
readonly onDidCreateMarkdownDocument: vscode.Event<SkinnyTextDocument>;
|
|
readonly onDidDeleteMarkdownDocument: vscode.Event<vscode.Uri>;
|
|
}
|
|
|
|
class VSCodeWorkspaceMarkdownDocumentProvider extends Disposable implements WorkspaceMarkdownDocumentProvider {
|
|
|
|
private readonly _onDidChangeMarkdownDocumentEmitter = this._register(new vscode.EventEmitter<SkinnyTextDocument>());
|
|
private readonly _onDidCreateMarkdownDocumentEmitter = this._register(new vscode.EventEmitter<SkinnyTextDocument>());
|
|
private readonly _onDidDeleteMarkdownDocumentEmitter = this._register(new vscode.EventEmitter<vscode.Uri>());
|
|
|
|
private _watcher: vscode.FileSystemWatcher | undefined;
|
|
|
|
async getAllMarkdownDocuments() {
|
|
const resources = await vscode.workspace.findFiles('**/*.md', '**/node_modules/**');
|
|
const docs = await Promise.all(resources.map(doc => this.getMarkdownDocument(doc)));
|
|
return docs.filter(doc => !!doc) as SkinnyTextDocument[];
|
|
}
|
|
|
|
public get onDidChangeMarkdownDocument() {
|
|
this.ensureWatcher();
|
|
return this._onDidChangeMarkdownDocumentEmitter.event;
|
|
}
|
|
|
|
public get onDidCreateMarkdownDocument() {
|
|
this.ensureWatcher();
|
|
return this._onDidCreateMarkdownDocumentEmitter.event;
|
|
}
|
|
|
|
public get onDidDeleteMarkdownDocument() {
|
|
this.ensureWatcher();
|
|
return this._onDidDeleteMarkdownDocumentEmitter.event;
|
|
}
|
|
|
|
private ensureWatcher(): void {
|
|
if (this._watcher) {
|
|
return;
|
|
}
|
|
|
|
this._watcher = this._register(vscode.workspace.createFileSystemWatcher('**/*.md'));
|
|
|
|
this._watcher.onDidChange(async resource => {
|
|
const document = await this.getMarkdownDocument(resource);
|
|
if (document) {
|
|
this._onDidChangeMarkdownDocumentEmitter.fire(document);
|
|
}
|
|
}, null, this._disposables);
|
|
|
|
this._watcher.onDidCreate(async resource => {
|
|
const document = await this.getMarkdownDocument(resource);
|
|
if (document) {
|
|
this._onDidCreateMarkdownDocumentEmitter.fire(document);
|
|
}
|
|
}, null, this._disposables);
|
|
|
|
this._watcher.onDidDelete(async resource => {
|
|
this._onDidDeleteMarkdownDocumentEmitter.fire(resource);
|
|
}, null, this._disposables);
|
|
|
|
vscode.workspace.onDidChangeTextDocument(e => {
|
|
if (isMarkdownFile(e.document)) {
|
|
this._onDidChangeMarkdownDocumentEmitter.fire(e.document);
|
|
}
|
|
}, null, this._disposables);
|
|
}
|
|
|
|
private async getMarkdownDocument(resource: vscode.Uri): Promise<SkinnyTextDocument | undefined> {
|
|
const matchingDocuments = vscode.workspace.textDocuments.filter((doc) => doc.uri.toString() === resource.toString());
|
|
if (matchingDocuments.length !== 0) {
|
|
return matchingDocuments[0];
|
|
}
|
|
|
|
const bytes = await vscode.workspace.fs.readFile(resource);
|
|
|
|
// We assume that markdown is in UTF-8
|
|
const text = Buffer.from(bytes).toString('utf-8');
|
|
|
|
const lines: SkinnyTextLine[] = [];
|
|
const parts = text.split(/(\r?\n)/);
|
|
const lineCount = Math.floor(parts.length / 2) + 1;
|
|
for (let line = 0; line < lineCount; line++) {
|
|
lines.push({
|
|
text: parts[line * 2]
|
|
});
|
|
}
|
|
|
|
return {
|
|
uri: resource,
|
|
version: 0,
|
|
lineCount: lineCount,
|
|
lineAt: (index) => {
|
|
return lines[index];
|
|
},
|
|
getText: () => {
|
|
return text;
|
|
}
|
|
};
|
|
}
|
|
}
|
|
|
|
export default class MarkdownWorkspaceSymbolProvider extends Disposable implements vscode.WorkspaceSymbolProvider {
|
|
private _symbolCache = new Map<string, Lazy<Thenable<vscode.SymbolInformation[]>>>();
|
|
private _symbolCachePopulated: boolean = false;
|
|
|
|
public constructor(
|
|
private _symbolProvider: MDDocumentSymbolProvider,
|
|
private _workspaceMarkdownDocumentProvider: WorkspaceMarkdownDocumentProvider = new VSCodeWorkspaceMarkdownDocumentProvider()
|
|
) {
|
|
super();
|
|
}
|
|
|
|
public async provideWorkspaceSymbols(query: string): Promise<vscode.SymbolInformation[]> {
|
|
if (!this._symbolCachePopulated) {
|
|
await this.populateSymbolCache();
|
|
this._symbolCachePopulated = true;
|
|
|
|
this._workspaceMarkdownDocumentProvider.onDidChangeMarkdownDocument(this.onDidChangeDocument, this, this._disposables);
|
|
this._workspaceMarkdownDocumentProvider.onDidCreateMarkdownDocument(this.onDidChangeDocument, this, this._disposables);
|
|
this._workspaceMarkdownDocumentProvider.onDidDeleteMarkdownDocument(this.onDidDeleteDocument, this, this._disposables);
|
|
}
|
|
|
|
const allSymbolsSets = await Promise.all(Array.from(this._symbolCache.values(), x => x.value));
|
|
const allSymbols = allSymbolsSets.flat();
|
|
return allSymbols.filter(symbolInformation => symbolInformation.name.toLowerCase().indexOf(query.toLowerCase()) !== -1);
|
|
}
|
|
|
|
public async populateSymbolCache(): Promise<void> {
|
|
const markdownDocumentUris = await this._workspaceMarkdownDocumentProvider.getAllMarkdownDocuments();
|
|
for (const document of markdownDocumentUris) {
|
|
this._symbolCache.set(document.uri.fsPath, this.getSymbols(document));
|
|
}
|
|
}
|
|
|
|
private getSymbols(document: SkinnyTextDocument): Lazy<Thenable<vscode.SymbolInformation[]>> {
|
|
return lazy(async () => {
|
|
return this._symbolProvider.provideDocumentSymbolInformation(document);
|
|
});
|
|
}
|
|
|
|
private onDidChangeDocument(document: SkinnyTextDocument) {
|
|
this._symbolCache.set(document.uri.fsPath, this.getSymbols(document));
|
|
}
|
|
|
|
private onDidDeleteDocument(resource: vscode.Uri) {
|
|
this._symbolCache.delete(resource.fsPath);
|
|
}
|
|
}
|