Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Support opening files from 'include' directive by adding DocumentLinkProvider #99

Merged
merged 4 commits into from
Apr 30, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
46 changes: 46 additions & 0 deletions src/documentLinkProvider.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
import * as vscode from "vscode";
import { DocumentLink, Range, TextDocument } from "vscode";
import { getWorkspaceRootUri } from "./utils";

export default class DocumentLinkProvider implements vscode.DocumentLinkProvider {
public provideDocumentLinks(document: TextDocument): vscode.DocumentLink[] {
const text = document.getText();

const workspaceRootUri = getWorkspaceRootUri();
if (workspaceRootUri === undefined) {
console.error('Invalid workspace root path.');
return [];
}

const links: DocumentLink[] = [];
for (const match of text.matchAll(/include\s+"([^"]+.(?:beancount|bean))"/g)) {
const includedFileName = match[1];

const range = this.create_range(document, match);
if (range === undefined) {
// Nothing matched.
console.error("nothing matched");
continue;
}

const link = new DocumentLink(range, vscode.Uri.joinPath(workspaceRootUri, includedFileName));
link.tooltip = "Follow link";
links.push(link);
}

return links;
}

private create_range(document: TextDocument, match: RegExpMatchArray): Range | undefined {
if (match.index === undefined) {
// nothing matched.
return;
}

// extend the range by 1 in both sides to include the quote characters.
const start = document.positionAt(match.index + match[0].length - match[1].length - 2);
const end = document.positionAt(match.index + match[0].length + 1);
const r = new Range(start, end);
return r;
}
}
22 changes: 15 additions & 7 deletions src/extension.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,9 +7,11 @@ import { Completer } from "./completer";
import { Formatter } from "./formatter";
import { runCmd } from "./utils";
import { SymbolProvider } from "./symbolProvider";
import DocumentLinkProvider from "./documentLinkProvider";

export function activate(context: vscode.ExtensionContext) {
const extension = new Extension(context);
const beancountDocumentSelector = { scheme: "file", language: "beancount" };

vscode.commands.registerCommand("beancount.runFava", () =>
extension.favaManager.openFava(true)
Expand All @@ -25,7 +27,7 @@ export function activate(context: vscode.ExtensionContext) {

context.subscriptions.push(
vscode.languages.registerCompletionItemProvider(
{ scheme: "file", language: "beancount" },
beancountDocumentSelector,
extension.completer,
"2",
"#",
Expand All @@ -35,13 +37,13 @@ export function activate(context: vscode.ExtensionContext) {
);
context.subscriptions.push(
vscode.languages.registerHoverProvider(
{ scheme: "file", language: "beancount" },
beancountDocumentSelector,
extension.completer
)
);
context.subscriptions.push(
vscode.languages.registerCodeActionsProvider(
{ scheme: "file", language: "beancount" },
beancountDocumentSelector,
extension.actionProvider
)
);
Expand Down Expand Up @@ -83,13 +85,17 @@ export function activate(context: vscode.ExtensionContext) {

context.subscriptions.push(
vscode.languages.registerDocumentSymbolProvider(
{ scheme: "file", language: "beancount" },
beancountDocumentSelector,
extension.symbolProvider
)
);
));
context.subscriptions.push(
vscode.languages.registerDocumentLinkProvider(
beancountDocumentSelector,
extension.documentLinkProvider
));
}

export function deactivate() {}
export function deactivate() { }

export class Extension {
completer: Completer;
Expand All @@ -101,6 +107,7 @@ export class Extension {
flagWarnings: FlagWarnings;
context: vscode.ExtensionContext;
symbolProvider: SymbolProvider;
documentLinkProvider: DocumentLinkProvider;

constructor(context: vscode.ExtensionContext) {
this.context = context;
Expand All @@ -114,6 +121,7 @@ export class Extension {
this.flagWarnings =
vscode.workspace.getConfiguration("beancount")["flagWarnings"];
this.symbolProvider = new SymbolProvider();
this.documentLinkProvider = new DocumentLinkProvider();
}

resolveToAbsolutePath(path: string) {
Expand Down
16 changes: 16 additions & 0 deletions src/utils.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import { TextDocument, Uri, window, workspace } from 'vscode';
import { spawn, SpawnOptionsWithoutStdio } from "child_process";

export function runCmd(
Expand Down Expand Up @@ -31,3 +32,18 @@ export function pushIfEmpty<T>(array: T[], defaultValue: T): T[] {
}
return array;
}

export function getWorkspaceRootUri(): Uri | undefined {
const document = getCurrentTextDocument();
if (document) {
const fileUri = document.uri;
const workspaceFolder = workspace.getWorkspaceFolder(fileUri);
if (workspaceFolder) {
return workspaceFolder.uri;
}
}
}

export function getCurrentTextDocument(): TextDocument | undefined {
return window.activeTextEditor?.document;
}