mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-01-20 17:22:51 -05:00
* Fix initial build breaks from 1.67 merge (#2514) * Update yarn lock files * Update build scripts * Fix tsconfig * Build breaks * WIP * Update yarn lock files * Misc breaks * Updates to package.json * Breaks * Update yarn * Fix breaks * Breaks * Build breaks * Breaks * Breaks * Breaks * Breaks * Breaks * Missing file * Breaks * Breaks * Breaks * Breaks * Breaks * Fix several runtime breaks (#2515) * Missing files * Runtime breaks * Fix proxy ordering issue * Remove commented code * Fix breaks with opening query editor * Fix post merge break * Updates related to setup build and other breaks (#2516) * Fix bundle build issues * Update distro * Fix distro merge and update build JS files * Disable pipeline steps * Remove stats call * Update license name * Make new RPM dependencies a warning * Fix extension manager version checks * Update JS file * Fix a few runtime breaks * Fixes * Fix runtime issues * Fix build breaks * Update notebook tests (part 1) * Fix broken tests * Linting errors * Fix hygiene * Disable lint rules * Bump distro * Turn off smoke tests * Disable integration tests * Remove failing "activate" test * Remove failed test assertion * Disable other broken test * Disable query history tests * Disable extension unit tests * Disable failing tasks
173 lines
4.8 KiB
TypeScript
173 lines
4.8 KiB
TypeScript
/*---------------------------------------------------------------------------------------------
|
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
|
*--------------------------------------------------------------------------------------------*/
|
|
|
|
import * as vscode from 'vscode';
|
|
import { MarkdownEngine } from './markdownEngine';
|
|
import { githubSlugifier, Slug } from './slugify';
|
|
import { isMarkdownFile } from './util/file';
|
|
import { SkinnyTextDocument } from './workspaceContents';
|
|
|
|
export interface TocEntry {
|
|
readonly slug: Slug;
|
|
readonly text: string;
|
|
readonly level: number;
|
|
readonly line: number;
|
|
|
|
/**
|
|
* The entire range of the header section.
|
|
*
|
|
* For the doc:
|
|
*
|
|
* ```md
|
|
* # Head #
|
|
* text
|
|
* # Next head #
|
|
* ```
|
|
*
|
|
* This is the range from `# Head #` to `# Next head #`
|
|
*/
|
|
readonly sectionLocation: vscode.Location;
|
|
|
|
/**
|
|
* The range of the header declaration.
|
|
*
|
|
* For the doc:
|
|
*
|
|
* ```md
|
|
* # Head #
|
|
* text
|
|
* ```
|
|
*
|
|
* This is the range of `# Head #`
|
|
*/
|
|
readonly headerLocation: vscode.Location;
|
|
|
|
/**
|
|
* The range of the header text.
|
|
*
|
|
* For the doc:
|
|
*
|
|
* ```md
|
|
* # Head #
|
|
* text
|
|
* ```
|
|
*
|
|
* This is the range of `Head`
|
|
*/
|
|
readonly headerTextLocation: vscode.Location;
|
|
}
|
|
|
|
export class TableOfContents {
|
|
|
|
public static async create(engine: MarkdownEngine, document: SkinnyTextDocument,): Promise<TableOfContents> {
|
|
const entries = await this.buildToc(engine, document);
|
|
return new TableOfContents(entries);
|
|
}
|
|
|
|
public static async createForDocumentOrNotebook(engine: MarkdownEngine, document: SkinnyTextDocument): Promise<TableOfContents> {
|
|
if (document.uri.scheme === 'vscode-notebook-cell') {
|
|
const notebook = vscode.workspace.notebookDocuments
|
|
.find(notebook => notebook.getCells().some(cell => cell.document === document));
|
|
|
|
if (notebook) {
|
|
const entries: TocEntry[] = [];
|
|
|
|
for (const cell of notebook.getCells()) {
|
|
if (cell.kind === vscode.NotebookCellKind.Markup && isMarkdownFile(cell.document)) {
|
|
entries.push(...(await this.buildToc(engine, cell.document)));
|
|
}
|
|
}
|
|
|
|
return new TableOfContents(entries);
|
|
}
|
|
}
|
|
|
|
return this.create(engine, document);
|
|
}
|
|
|
|
private static async buildToc(engine: MarkdownEngine, document: SkinnyTextDocument): Promise<TocEntry[]> {
|
|
const toc: TocEntry[] = [];
|
|
const tokens = await engine.parse(document);
|
|
|
|
const existingSlugEntries = new Map<string, { count: number }>();
|
|
|
|
for (const heading of tokens.filter(token => token.type === 'heading_open')) {
|
|
if (!heading.map) {
|
|
continue;
|
|
}
|
|
|
|
const lineNumber = heading.map[0];
|
|
const line = document.lineAt(lineNumber);
|
|
|
|
let slug = githubSlugifier.fromHeading(line.text);
|
|
const existingSlugEntry = existingSlugEntries.get(slug.value);
|
|
if (existingSlugEntry) {
|
|
++existingSlugEntry.count;
|
|
slug = githubSlugifier.fromHeading(slug.value + '-' + existingSlugEntry.count);
|
|
} else {
|
|
existingSlugEntries.set(slug.value, { count: 0 });
|
|
}
|
|
|
|
const headerLocation = new vscode.Location(document.uri,
|
|
new vscode.Range(lineNumber, 0, lineNumber, line.text.length));
|
|
|
|
const headerTextLocation = new vscode.Location(document.uri,
|
|
new vscode.Range(lineNumber, line.text.match(/^#+\s*/)?.[0].length ?? 0, lineNumber, line.text.length - (line.text.match(/\s*#*$/)?.[0].length ?? 0)));
|
|
|
|
toc.push({
|
|
slug,
|
|
text: TableOfContents.getHeaderText(line.text),
|
|
level: TableOfContents.getHeaderLevel(heading.markup),
|
|
line: lineNumber,
|
|
sectionLocation: headerLocation, // Populated in next steps
|
|
headerLocation,
|
|
headerTextLocation
|
|
});
|
|
}
|
|
|
|
// Get full range of section
|
|
return toc.map((entry, startIndex): TocEntry => {
|
|
let end: number | undefined = undefined;
|
|
for (let i = startIndex + 1; i < toc.length; ++i) {
|
|
if (toc[i].level <= entry.level) {
|
|
end = toc[i].line - 1;
|
|
break;
|
|
}
|
|
}
|
|
const endLine = end ?? document.lineCount - 1;
|
|
return {
|
|
...entry,
|
|
sectionLocation: new vscode.Location(document.uri,
|
|
new vscode.Range(
|
|
entry.sectionLocation.range.start,
|
|
new vscode.Position(endLine, document.lineAt(endLine).text.length)))
|
|
};
|
|
});
|
|
}
|
|
|
|
private static getHeaderLevel(markup: string): number {
|
|
if (markup === '=') {
|
|
return 1;
|
|
} else if (markup === '-') {
|
|
return 2;
|
|
} else { // '#', '##', ...
|
|
return markup.length;
|
|
}
|
|
}
|
|
|
|
private static getHeaderText(header: string): string {
|
|
return header.replace(/^\s*#+\s*(.*?)(\s+#+)?$/, (_, word) => word.trim());
|
|
}
|
|
|
|
private constructor(
|
|
public readonly entries: readonly TocEntry[],
|
|
) { }
|
|
|
|
public lookup(fragment: string): TocEntry | undefined {
|
|
const slug = githubSlugifier.fromHeading(fragment);
|
|
return this.entries.find(entry => entry.slug.equals(slug));
|
|
}
|
|
}
|