2022-07-01 11:52:39 +00:00
|
|
|
/**
|
2024-01-03 10:11:33 +00:00
|
|
|
* @license
|
|
|
|
* Copyright 2022 Google Inc.
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
2022-07-01 11:52:39 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
// Copyright (c) Microsoft Corporation. All rights reserved. Licensed under the
|
|
|
|
// MIT license. See LICENSE in the project root for license information.
|
|
|
|
|
|
|
|
// Taken from
|
|
|
|
// https://github.com/microsoft/rushstack/blob/main/apps/api-documenter/src/documenters/MarkdownDocumenter.ts
|
|
|
|
// This file has been edited to morph into Docusaurus's expected inputs.
|
|
|
|
|
2023-02-15 23:09:31 +00:00
|
|
|
import * as path from 'path';
|
|
|
|
|
2023-10-12 13:04:29 +00:00
|
|
|
import type {DocumenterConfig} from '@microsoft/api-documenter/lib/documenters/DocumenterConfig.js';
|
2023-11-29 12:06:05 +00:00
|
|
|
import {CustomMarkdownEmitter as ApiFormatterMarkdownEmitter} from '@microsoft/api-documenter/lib/markdown/CustomMarkdownEmitter.js';
|
2023-10-12 13:04:29 +00:00
|
|
|
import {CustomDocNodes} from '@microsoft/api-documenter/lib/nodes/CustomDocNodeKind.js';
|
|
|
|
import {DocEmphasisSpan} from '@microsoft/api-documenter/lib/nodes/DocEmphasisSpan.js';
|
|
|
|
import {DocHeading} from '@microsoft/api-documenter/lib/nodes/DocHeading.js';
|
|
|
|
import {DocNoteBox} from '@microsoft/api-documenter/lib/nodes/DocNoteBox.js';
|
|
|
|
import {DocTable} from '@microsoft/api-documenter/lib/nodes/DocTable.js';
|
|
|
|
import {DocTableCell} from '@microsoft/api-documenter/lib/nodes/DocTableCell.js';
|
|
|
|
import {DocTableRow} from '@microsoft/api-documenter/lib/nodes/DocTableRow.js';
|
|
|
|
import {MarkdownDocumenterAccessor} from '@microsoft/api-documenter/lib/plugin/MarkdownDocumenterAccessor.js';
|
2023-02-15 23:09:31 +00:00
|
|
|
import {
|
2023-09-15 11:00:20 +00:00
|
|
|
type IMarkdownDocumenterFeatureOnBeforeWritePageArgs,
|
2023-02-15 23:09:31 +00:00
|
|
|
MarkdownDocumenterFeatureContext,
|
2023-10-12 13:04:29 +00:00
|
|
|
} from '@microsoft/api-documenter/lib/plugin/MarkdownDocumenterFeature.js';
|
|
|
|
import {PluginLoader} from '@microsoft/api-documenter/lib/plugin/PluginLoader.js';
|
|
|
|
import {Utilities} from '@microsoft/api-documenter/lib/utils/Utilities.js';
|
2022-07-01 11:52:39 +00:00
|
|
|
import {
|
|
|
|
ApiClass,
|
|
|
|
ApiDeclaredItem,
|
|
|
|
ApiDocumentedItem,
|
2023-09-15 11:00:20 +00:00
|
|
|
type ApiEnum,
|
2022-07-01 11:52:39 +00:00
|
|
|
ApiInitializerMixin,
|
|
|
|
ApiInterface,
|
2023-09-15 11:00:20 +00:00
|
|
|
type ApiItem,
|
2022-07-01 11:52:39 +00:00
|
|
|
ApiItemKind,
|
2023-09-15 11:00:20 +00:00
|
|
|
type ApiModel,
|
|
|
|
type ApiNamespace,
|
2022-07-01 11:52:39 +00:00
|
|
|
ApiOptionalMixin,
|
2023-09-15 11:00:20 +00:00
|
|
|
type ApiPackage,
|
2022-07-01 11:52:39 +00:00
|
|
|
ApiParameterListMixin,
|
|
|
|
ApiPropertyItem,
|
|
|
|
ApiProtectedMixin,
|
|
|
|
ApiReadonlyMixin,
|
|
|
|
ApiReleaseTagMixin,
|
|
|
|
ApiReturnTypeMixin,
|
|
|
|
ApiStaticMixin,
|
|
|
|
ApiTypeAlias,
|
2023-09-15 11:00:20 +00:00
|
|
|
type Excerpt,
|
|
|
|
type ExcerptToken,
|
2022-07-01 11:52:39 +00:00
|
|
|
ExcerptTokenKind,
|
2023-09-15 11:00:20 +00:00
|
|
|
type IResolveDeclarationReferenceResult,
|
2022-07-01 11:52:39 +00:00
|
|
|
ReleaseTag,
|
|
|
|
} from '@microsoft/api-extractor-model';
|
|
|
|
import {
|
2023-09-15 11:00:20 +00:00
|
|
|
type DocBlock,
|
2022-07-01 11:52:39 +00:00
|
|
|
DocCodeSpan,
|
2023-09-15 11:00:20 +00:00
|
|
|
type DocComment,
|
2022-07-01 11:52:39 +00:00
|
|
|
DocFencedCode,
|
|
|
|
DocLinkTag,
|
2023-09-15 11:00:20 +00:00
|
|
|
type DocNodeContainer,
|
2022-07-01 11:52:39 +00:00
|
|
|
DocNodeKind,
|
|
|
|
DocParagraph,
|
|
|
|
DocPlainText,
|
|
|
|
DocSection,
|
|
|
|
StandardTags,
|
|
|
|
StringBuilder,
|
2023-09-15 11:00:20 +00:00
|
|
|
type TSDocConfiguration,
|
2022-07-01 11:52:39 +00:00
|
|
|
} from '@microsoft/tsdoc';
|
|
|
|
import {
|
|
|
|
FileSystem,
|
|
|
|
NewlineKind,
|
|
|
|
PackageName,
|
|
|
|
} from '@rushstack/node-core-library';
|
|
|
|
|
|
|
|
export interface IMarkdownDocumenterOptions {
|
|
|
|
apiModel: ApiModel;
|
|
|
|
documenterConfig: DocumenterConfig | undefined;
|
|
|
|
outputFolder: string;
|
|
|
|
}
|
|
|
|
|
2023-11-29 12:06:05 +00:00
|
|
|
export class CustomMarkdownEmitter extends ApiFormatterMarkdownEmitter {
|
|
|
|
protected override getEscapedText(text: string): string {
|
|
|
|
const textWithBackslashes: string = text
|
|
|
|
.replace(/\\/g, '\\\\') // first replace the escape character
|
|
|
|
.replace(/[*#[\]_|`~]/g, x => {
|
|
|
|
return '\\' + x;
|
|
|
|
}) // then escape any special characters
|
|
|
|
.replace(/---/g, '\\-\\-\\-') // hyphens only if it's 3 or more
|
|
|
|
.replace(/&/g, '&')
|
|
|
|
.replace(/</g, '<')
|
|
|
|
.replace(/>/g, '>')
|
|
|
|
.replace(/\{/g, '{')
|
|
|
|
.replace(/\}/g, '}');
|
|
|
|
return textWithBackslashes;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-01 11:52:39 +00:00
|
|
|
/**
|
|
|
|
* Renders API documentation in the Markdown file format.
|
2022-08-12 12:15:26 +00:00
|
|
|
* For more info: https://en.wikipedia.org/wiki/Markdown
|
2022-07-01 11:52:39 +00:00
|
|
|
*/
|
|
|
|
export class MarkdownDocumenter {
|
|
|
|
private readonly _apiModel: ApiModel;
|
|
|
|
private readonly _documenterConfig: DocumenterConfig | undefined;
|
|
|
|
private readonly _tsdocConfiguration: TSDocConfiguration;
|
|
|
|
private readonly _markdownEmitter: CustomMarkdownEmitter;
|
|
|
|
private readonly _outputFolder: string;
|
|
|
|
private readonly _pluginLoader: PluginLoader;
|
|
|
|
|
|
|
|
public constructor(options: IMarkdownDocumenterOptions) {
|
|
|
|
this._apiModel = options.apiModel;
|
|
|
|
this._documenterConfig = options.documenterConfig;
|
|
|
|
this._outputFolder = options.outputFolder;
|
|
|
|
this._tsdocConfiguration = CustomDocNodes.configuration;
|
|
|
|
this._markdownEmitter = new CustomMarkdownEmitter(this._apiModel);
|
|
|
|
|
|
|
|
this._pluginLoader = new PluginLoader();
|
|
|
|
}
|
|
|
|
|
|
|
|
public generateFiles(): void {
|
|
|
|
if (this._documenterConfig) {
|
|
|
|
this._pluginLoader.load(this._documenterConfig, () => {
|
|
|
|
return new MarkdownDocumenterFeatureContext({
|
|
|
|
apiModel: this._apiModel,
|
|
|
|
outputFolder: this._outputFolder,
|
|
|
|
documenter: new MarkdownDocumenterAccessor({
|
|
|
|
getLinkForApiItem: (apiItem: ApiItem) => {
|
|
|
|
return this._getLinkFilenameForApiItem(apiItem);
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
this._deleteOldOutputFiles();
|
|
|
|
|
|
|
|
this._writeApiItemPage(this._apiModel.members[0]!);
|
|
|
|
|
|
|
|
if (this._pluginLoader.markdownDocumenterFeature) {
|
|
|
|
this._pluginLoader.markdownDocumenterFeature.onFinished({});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private _writeApiItemPage(apiItem: ApiItem): void {
|
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
const output: DocSection = new DocSection({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
});
|
|
|
|
|
|
|
|
const scopedName: string = apiItem.getScopedNameWithinPackage();
|
|
|
|
|
|
|
|
switch (apiItem.kind) {
|
|
|
|
case ApiItemKind.Class:
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({configuration, title: `${scopedName} class`})
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Enum:
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({configuration, title: `${scopedName} enum`})
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Interface:
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({configuration, title: `${scopedName} interface`})
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Constructor:
|
|
|
|
case ApiItemKind.ConstructSignature:
|
|
|
|
output.appendNode(new DocHeading({configuration, title: scopedName}));
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Method:
|
|
|
|
case ApiItemKind.MethodSignature:
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({configuration, title: `${scopedName} method`})
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Function:
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({configuration, title: `${scopedName} function`})
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Model:
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({configuration, title: `API Reference`})
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Namespace:
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({configuration, title: `${scopedName} namespace`})
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Package:
|
|
|
|
console.log(`Writing ${apiItem.displayName} package`);
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration,
|
|
|
|
title: `API Reference`,
|
|
|
|
})
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Property:
|
|
|
|
case ApiItemKind.PropertySignature:
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({configuration, title: `${scopedName} property`})
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.TypeAlias:
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({configuration, title: `${scopedName} type`})
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Variable:
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({configuration, title: `${scopedName} variable`})
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw new Error('Unsupported API item kind: ' + apiItem.kind);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ApiReleaseTagMixin.isBaseClassOf(apiItem)) {
|
|
|
|
if (apiItem.releaseTag === ReleaseTag.Beta) {
|
|
|
|
this._writeBetaWarning(output);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const decoratorBlocks: DocBlock[] = [];
|
|
|
|
|
|
|
|
if (apiItem instanceof ApiDocumentedItem) {
|
|
|
|
const tsdocComment: DocComment | undefined = apiItem.tsdocComment;
|
|
|
|
|
|
|
|
if (tsdocComment) {
|
|
|
|
decoratorBlocks.push(
|
|
|
|
...tsdocComment.customBlocks.filter(block => {
|
|
|
|
return (
|
|
|
|
block.blockTag.tagNameWithUpperCase ===
|
|
|
|
StandardTags.decorator.tagNameWithUpperCase
|
|
|
|
);
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
|
|
|
if (tsdocComment.deprecatedBlock) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocNoteBox({configuration: this._tsdocConfiguration}, [
|
|
|
|
new DocParagraph({configuration: this._tsdocConfiguration}, [
|
|
|
|
new DocPlainText({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
text: 'Warning: This API is now obsolete. ',
|
|
|
|
}),
|
|
|
|
]),
|
|
|
|
...tsdocComment.deprecatedBlock.content.nodes,
|
|
|
|
])
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
this._appendSection(output, tsdocComment.summarySection);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (apiItem instanceof ApiDeclaredItem) {
|
|
|
|
if (apiItem.excerpt.text.length > 0) {
|
|
|
|
output.appendNode(
|
2022-10-24 07:07:05 +00:00
|
|
|
new DocHeading({configuration, title: 'Signature:', level: 4})
|
2022-07-01 11:52:39 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
let code: string;
|
|
|
|
switch (apiItem.parent?.kind) {
|
|
|
|
case ApiItemKind.Class:
|
|
|
|
code = `class ${
|
|
|
|
apiItem.parent.displayName
|
|
|
|
} {${apiItem.getExcerptWithModifiers()}}`;
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Interface:
|
|
|
|
code = `interface ${
|
|
|
|
apiItem.parent.displayName
|
|
|
|
} {${apiItem.getExcerptWithModifiers()}}`;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
code = apiItem.getExcerptWithModifiers();
|
|
|
|
}
|
|
|
|
output.appendNode(
|
|
|
|
new DocFencedCode({
|
|
|
|
configuration,
|
|
|
|
code: code,
|
|
|
|
language: 'typescript',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
this._writeHeritageTypes(output, apiItem);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (decoratorBlocks.length > 0) {
|
|
|
|
output.appendNode(
|
2022-10-24 07:07:05 +00:00
|
|
|
new DocHeading({configuration, title: 'Decorators:', level: 4})
|
2022-07-01 11:52:39 +00:00
|
|
|
);
|
|
|
|
for (const decoratorBlock of decoratorBlocks) {
|
|
|
|
output.appendNodes(decoratorBlock.content.nodes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let appendRemarks = true;
|
|
|
|
switch (apiItem.kind) {
|
|
|
|
case ApiItemKind.Class:
|
|
|
|
case ApiItemKind.Interface:
|
|
|
|
case ApiItemKind.Namespace:
|
|
|
|
case ApiItemKind.Package:
|
|
|
|
this._writeRemarksSection(output, apiItem);
|
|
|
|
appendRemarks = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (apiItem.kind) {
|
|
|
|
case ApiItemKind.Class:
|
|
|
|
this._writeClassTables(output, apiItem as ApiClass);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Enum:
|
|
|
|
this._writeEnumTables(output, apiItem as ApiEnum);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Interface:
|
|
|
|
this._writeInterfaceTables(output, apiItem as ApiInterface);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Constructor:
|
|
|
|
case ApiItemKind.ConstructSignature:
|
|
|
|
case ApiItemKind.Method:
|
|
|
|
case ApiItemKind.MethodSignature:
|
|
|
|
case ApiItemKind.Function:
|
|
|
|
this._writeParameterTables(output, apiItem as ApiParameterListMixin);
|
|
|
|
this._writeThrowsSection(output, apiItem);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Namespace:
|
|
|
|
this._writePackageOrNamespaceTables(output, apiItem as ApiNamespace);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Model:
|
|
|
|
this._writeModelTable(output, apiItem as ApiModel);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Package:
|
|
|
|
this._writePackageOrNamespaceTables(output, apiItem as ApiPackage);
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Property:
|
|
|
|
case ApiItemKind.PropertySignature:
|
|
|
|
break;
|
|
|
|
case ApiItemKind.TypeAlias:
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Variable:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw new Error('Unsupported API item kind: ' + apiItem.kind);
|
|
|
|
}
|
|
|
|
|
2022-10-24 07:07:05 +00:00
|
|
|
this._writeDefaultValueSection(output, apiItem);
|
|
|
|
|
2022-07-01 11:52:39 +00:00
|
|
|
if (appendRemarks) {
|
|
|
|
this._writeRemarksSection(output, apiItem);
|
|
|
|
}
|
|
|
|
|
|
|
|
const filename: string = path.join(
|
|
|
|
this._outputFolder,
|
|
|
|
this._getFilenameForApiItem(apiItem)
|
|
|
|
);
|
|
|
|
const stringBuilder: StringBuilder = new StringBuilder();
|
|
|
|
|
|
|
|
this._markdownEmitter.emit(stringBuilder, output, {
|
|
|
|
contextApiItem: apiItem,
|
|
|
|
onGetFilenameForApiItem: (apiItemForFilename: ApiItem) => {
|
|
|
|
return this._getLinkFilenameForApiItem(apiItemForFilename);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
let pageContent: string = stringBuilder.toString();
|
|
|
|
|
|
|
|
if (this._pluginLoader.markdownDocumenterFeature) {
|
|
|
|
// Allow the plugin to customize the pageContent
|
|
|
|
const eventArgs: IMarkdownDocumenterFeatureOnBeforeWritePageArgs = {
|
|
|
|
apiItem: apiItem,
|
|
|
|
outputFilename: filename,
|
|
|
|
pageContent: pageContent,
|
|
|
|
};
|
|
|
|
this._pluginLoader.markdownDocumenterFeature.onBeforeWritePage(eventArgs);
|
|
|
|
pageContent = eventArgs.pageContent;
|
|
|
|
}
|
|
|
|
|
|
|
|
pageContent =
|
|
|
|
`---\nsidebar_label: ${this._getSidebarLabelForApiItem(apiItem)}\n---` +
|
|
|
|
pageContent;
|
|
|
|
pageContent = pageContent.replace('##', '#');
|
|
|
|
pageContent = pageContent.replace(/<!-- -->/g, '');
|
2022-08-12 12:15:26 +00:00
|
|
|
pageContent = pageContent.replace(/\\\*\\\*/g, '**');
|
2022-07-01 11:52:39 +00:00
|
|
|
pageContent = pageContent.replace(/<b>|<\/b>/g, '**');
|
|
|
|
FileSystem.writeFile(filename, pageContent, {
|
|
|
|
convertLineEndings: this._documenterConfig
|
|
|
|
? this._documenterConfig.newlineKind
|
|
|
|
: NewlineKind.CrLf,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
private _writeHeritageTypes(
|
|
|
|
output: DocSection,
|
|
|
|
apiItem: ApiDeclaredItem
|
|
|
|
): void {
|
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
|
|
|
|
if (apiItem instanceof ApiClass) {
|
|
|
|
if (apiItem.extendsType) {
|
|
|
|
const extendsParagraph: DocParagraph = new DocParagraph(
|
|
|
|
{configuration},
|
|
|
|
[
|
|
|
|
new DocEmphasisSpan({configuration, bold: true}, [
|
|
|
|
new DocPlainText({configuration, text: 'Extends: '}),
|
|
|
|
]),
|
|
|
|
]
|
|
|
|
);
|
|
|
|
this._appendExcerptWithHyperlinks(
|
|
|
|
extendsParagraph,
|
|
|
|
apiItem.extendsType.excerpt
|
|
|
|
);
|
|
|
|
output.appendNode(extendsParagraph);
|
|
|
|
}
|
|
|
|
if (apiItem.implementsTypes.length > 0) {
|
|
|
|
const extendsParagraph: DocParagraph = new DocParagraph(
|
|
|
|
{configuration},
|
|
|
|
[
|
|
|
|
new DocEmphasisSpan({configuration, bold: true}, [
|
|
|
|
new DocPlainText({configuration, text: 'Implements: '}),
|
|
|
|
]),
|
|
|
|
]
|
|
|
|
);
|
|
|
|
let needsComma = false;
|
|
|
|
for (const implementsType of apiItem.implementsTypes) {
|
|
|
|
if (needsComma) {
|
|
|
|
extendsParagraph.appendNode(
|
|
|
|
new DocPlainText({configuration, text: ', '})
|
|
|
|
);
|
|
|
|
}
|
|
|
|
this._appendExcerptWithHyperlinks(
|
|
|
|
extendsParagraph,
|
|
|
|
implementsType.excerpt
|
|
|
|
);
|
|
|
|
needsComma = true;
|
|
|
|
}
|
|
|
|
output.appendNode(extendsParagraph);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (apiItem instanceof ApiInterface) {
|
|
|
|
if (apiItem.extendsTypes.length > 0) {
|
|
|
|
const extendsParagraph: DocParagraph = new DocParagraph(
|
|
|
|
{configuration},
|
|
|
|
[
|
|
|
|
new DocEmphasisSpan({configuration, bold: true}, [
|
|
|
|
new DocPlainText({configuration, text: 'Extends: '}),
|
|
|
|
]),
|
|
|
|
]
|
|
|
|
);
|
|
|
|
let needsComma = false;
|
|
|
|
for (const extendsType of apiItem.extendsTypes) {
|
|
|
|
if (needsComma) {
|
|
|
|
extendsParagraph.appendNode(
|
|
|
|
new DocPlainText({configuration, text: ', '})
|
|
|
|
);
|
|
|
|
}
|
|
|
|
this._appendExcerptWithHyperlinks(
|
|
|
|
extendsParagraph,
|
|
|
|
extendsType.excerpt
|
|
|
|
);
|
|
|
|
needsComma = true;
|
|
|
|
}
|
|
|
|
output.appendNode(extendsParagraph);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (apiItem instanceof ApiTypeAlias) {
|
|
|
|
const refs: ExcerptToken[] = apiItem.excerptTokens.filter(token => {
|
|
|
|
return (
|
|
|
|
token.kind === ExcerptTokenKind.Reference &&
|
|
|
|
token.canonicalReference &&
|
|
|
|
this._apiModel.resolveDeclarationReference(
|
|
|
|
token.canonicalReference,
|
|
|
|
undefined
|
|
|
|
).resolvedApiItem
|
|
|
|
);
|
|
|
|
});
|
|
|
|
if (refs.length > 0) {
|
|
|
|
const referencesParagraph: DocParagraph = new DocParagraph(
|
|
|
|
{configuration},
|
|
|
|
[
|
|
|
|
new DocEmphasisSpan({configuration, bold: true}, [
|
|
|
|
new DocPlainText({configuration, text: 'References: '}),
|
|
|
|
]),
|
|
|
|
]
|
|
|
|
);
|
|
|
|
let needsComma = false;
|
2023-07-17 08:52:54 +00:00
|
|
|
const visited = new Set<string>();
|
2022-07-01 11:52:39 +00:00
|
|
|
for (const ref of refs) {
|
|
|
|
if (visited.has(ref.text)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
visited.add(ref.text);
|
|
|
|
|
|
|
|
if (needsComma) {
|
|
|
|
referencesParagraph.appendNode(
|
|
|
|
new DocPlainText({configuration, text: ', '})
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
this._appendExcerptTokenWithHyperlinks(referencesParagraph, ref);
|
|
|
|
needsComma = true;
|
|
|
|
}
|
|
|
|
output.appendNode(referencesParagraph);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-24 07:07:05 +00:00
|
|
|
private _writeDefaultValueSection(output: DocSection, apiItem: ApiItem) {
|
|
|
|
if (apiItem instanceof ApiDocumentedItem) {
|
|
|
|
const block = apiItem.tsdocComment?.customBlocks.find(block => {
|
|
|
|
return (
|
|
|
|
block.blockTag.tagNameWithUpperCase ===
|
|
|
|
StandardTags.defaultValue.tagNameWithUpperCase
|
|
|
|
);
|
|
|
|
});
|
|
|
|
if (block) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Default value:',
|
|
|
|
level: 4,
|
|
|
|
})
|
|
|
|
);
|
|
|
|
this._appendSection(output, block.content);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-01 11:52:39 +00:00
|
|
|
private _writeRemarksSection(output: DocSection, apiItem: ApiItem): void {
|
|
|
|
if (apiItem instanceof ApiDocumentedItem) {
|
|
|
|
const tsdocComment: DocComment | undefined = apiItem.tsdocComment;
|
|
|
|
|
|
|
|
if (tsdocComment) {
|
|
|
|
// Write the @remarks block
|
|
|
|
if (tsdocComment.remarksBlock) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Remarks',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
this._appendSection(output, tsdocComment.remarksBlock.content);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write the @example blocks
|
|
|
|
const exampleBlocks: DocBlock[] = tsdocComment.customBlocks.filter(
|
|
|
|
x => {
|
|
|
|
return (
|
|
|
|
x.blockTag.tagNameWithUpperCase ===
|
|
|
|
StandardTags.example.tagNameWithUpperCase
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
let exampleNumber = 1;
|
|
|
|
for (const exampleBlock of exampleBlocks) {
|
|
|
|
const heading: string =
|
|
|
|
exampleBlocks.length > 1 ? `Example ${exampleNumber}` : 'Example';
|
|
|
|
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: heading,
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
|
|
|
this._appendSection(output, exampleBlock.content);
|
|
|
|
|
|
|
|
++exampleNumber;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private _writeThrowsSection(output: DocSection, apiItem: ApiItem): void {
|
|
|
|
if (apiItem instanceof ApiDocumentedItem) {
|
|
|
|
const tsdocComment: DocComment | undefined = apiItem.tsdocComment;
|
|
|
|
|
|
|
|
if (tsdocComment) {
|
|
|
|
// Write the @throws blocks
|
|
|
|
const throwsBlocks: DocBlock[] = tsdocComment.customBlocks.filter(x => {
|
|
|
|
return (
|
|
|
|
x.blockTag.tagNameWithUpperCase ===
|
|
|
|
StandardTags.throws.tagNameWithUpperCase
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
if (throwsBlocks.length > 0) {
|
|
|
|
const heading = 'Exceptions';
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: heading,
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
|
|
|
for (const throwsBlock of throwsBlocks) {
|
|
|
|
this._appendSection(output, throwsBlock.content);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* GENERATE PAGE: MODEL
|
|
|
|
*/
|
|
|
|
private _writeModelTable(output: DocSection, apiModel: ApiModel): void {
|
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
|
|
|
|
const packagesTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
|
|
|
headerTitles: ['Package', 'Description'],
|
|
|
|
});
|
|
|
|
|
|
|
|
for (const apiMember of apiModel.members) {
|
|
|
|
const row: DocTableRow = new DocTableRow({configuration}, [
|
|
|
|
this._createTitleCell(apiMember),
|
|
|
|
this._createDescriptionCell(apiMember),
|
|
|
|
]);
|
|
|
|
|
|
|
|
switch (apiMember.kind) {
|
|
|
|
case ApiItemKind.Package:
|
|
|
|
packagesTable.addRow(row);
|
|
|
|
this._writeApiItemPage(apiMember);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (packagesTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Packages',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(packagesTable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* GENERATE PAGE: PACKAGE or NAMESPACE
|
|
|
|
*/
|
|
|
|
private _writePackageOrNamespaceTables(
|
|
|
|
output: DocSection,
|
|
|
|
apiContainer: ApiPackage | ApiNamespace
|
|
|
|
): void {
|
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
|
|
|
|
const classesTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
|
|
|
headerTitles: ['Class', 'Description'],
|
|
|
|
});
|
|
|
|
|
|
|
|
const enumerationsTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
|
|
|
headerTitles: ['Enumeration', 'Description'],
|
|
|
|
});
|
|
|
|
|
|
|
|
const functionsTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
|
|
|
headerTitles: ['Function', 'Description'],
|
|
|
|
});
|
|
|
|
|
|
|
|
const interfacesTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
|
|
|
headerTitles: ['Interface', 'Description'],
|
|
|
|
});
|
|
|
|
|
|
|
|
const namespacesTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
|
|
|
headerTitles: ['Namespace', 'Description'],
|
|
|
|
});
|
|
|
|
|
|
|
|
const variablesTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
|
|
|
headerTitles: ['Variable', 'Description'],
|
|
|
|
});
|
|
|
|
|
|
|
|
const typeAliasesTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
|
|
|
headerTitles: ['Type Alias', 'Description'],
|
|
|
|
});
|
|
|
|
|
2022-07-06 07:05:37 +00:00
|
|
|
const apiMembers: readonly ApiItem[] =
|
2022-07-01 11:52:39 +00:00
|
|
|
apiContainer.kind === ApiItemKind.Package
|
|
|
|
? (apiContainer as ApiPackage).entryPoints[0]!.members
|
|
|
|
: (apiContainer as ApiNamespace).members;
|
|
|
|
|
|
|
|
for (const apiMember of apiMembers) {
|
|
|
|
const row: DocTableRow = new DocTableRow({configuration}, [
|
|
|
|
this._createTitleCell(apiMember),
|
|
|
|
this._createDescriptionCell(apiMember),
|
|
|
|
]);
|
|
|
|
|
|
|
|
switch (apiMember.kind) {
|
|
|
|
case ApiItemKind.Class:
|
|
|
|
classesTable.addRow(row);
|
|
|
|
this._writeApiItemPage(apiMember);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ApiItemKind.Enum:
|
|
|
|
enumerationsTable.addRow(row);
|
|
|
|
this._writeApiItemPage(apiMember);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ApiItemKind.Interface:
|
|
|
|
interfacesTable.addRow(row);
|
|
|
|
this._writeApiItemPage(apiMember);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ApiItemKind.Namespace:
|
|
|
|
namespacesTable.addRow(row);
|
|
|
|
this._writeApiItemPage(apiMember);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ApiItemKind.Function:
|
|
|
|
functionsTable.addRow(row);
|
|
|
|
this._writeApiItemPage(apiMember);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ApiItemKind.TypeAlias:
|
|
|
|
typeAliasesTable.addRow(row);
|
|
|
|
this._writeApiItemPage(apiMember);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ApiItemKind.Variable:
|
|
|
|
variablesTable.addRow(row);
|
|
|
|
this._writeApiItemPage(apiMember);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (classesTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Classes',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(classesTable);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (enumerationsTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Enumerations',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(enumerationsTable);
|
|
|
|
}
|
|
|
|
if (functionsTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Functions',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(functionsTable);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (interfacesTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Interfaces',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(interfacesTable);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (namespacesTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Namespaces',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(namespacesTable);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (variablesTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Variables',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(variablesTable);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeAliasesTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Type Aliases',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(typeAliasesTable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* GENERATE PAGE: CLASS
|
|
|
|
*/
|
|
|
|
private _writeClassTables(output: DocSection, apiClass: ApiClass): void {
|
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
|
|
|
|
const eventsTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
|
|
|
headerTitles: ['Property', 'Modifiers', 'Type', 'Description'],
|
|
|
|
});
|
|
|
|
|
|
|
|
const constructorsTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
|
|
|
headerTitles: ['Constructor', 'Modifiers', 'Description'],
|
|
|
|
});
|
|
|
|
|
|
|
|
const propertiesTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
|
|
|
headerTitles: ['Property', 'Modifiers', 'Type', 'Description'],
|
|
|
|
});
|
|
|
|
|
|
|
|
const methodsTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
|
|
|
headerTitles: ['Method', 'Modifiers', 'Description'],
|
|
|
|
});
|
|
|
|
|
|
|
|
for (const apiMember of apiClass.members) {
|
|
|
|
switch (apiMember.kind) {
|
|
|
|
case ApiItemKind.Constructor: {
|
|
|
|
constructorsTable.addRow(
|
|
|
|
new DocTableRow({configuration}, [
|
|
|
|
this._createTitleCell(apiMember),
|
|
|
|
this._createModifiersCell(apiMember),
|
|
|
|
this._createDescriptionCell(apiMember),
|
|
|
|
])
|
|
|
|
);
|
|
|
|
|
|
|
|
this._writeApiItemPage(apiMember);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ApiItemKind.Method: {
|
|
|
|
methodsTable.addRow(
|
|
|
|
new DocTableRow({configuration}, [
|
|
|
|
this._createTitleCell(apiMember),
|
|
|
|
this._createModifiersCell(apiMember),
|
|
|
|
this._createDescriptionCell(apiMember),
|
|
|
|
])
|
|
|
|
);
|
|
|
|
|
|
|
|
this._writeApiItemPage(apiMember);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ApiItemKind.Property: {
|
|
|
|
if ((apiMember as ApiPropertyItem).isEventProperty) {
|
|
|
|
eventsTable.addRow(
|
|
|
|
new DocTableRow({configuration}, [
|
2023-03-28 18:02:00 +00:00
|
|
|
this._createTitleCell(apiMember, true),
|
2022-07-01 11:52:39 +00:00
|
|
|
this._createModifiersCell(apiMember),
|
|
|
|
this._createPropertyTypeCell(apiMember),
|
|
|
|
this._createDescriptionCell(apiMember),
|
|
|
|
])
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
propertiesTable.addRow(
|
|
|
|
new DocTableRow({configuration}, [
|
2023-03-28 18:02:00 +00:00
|
|
|
this._createTitleCell(apiMember, true),
|
2022-07-01 11:52:39 +00:00
|
|
|
this._createModifiersCell(apiMember),
|
|
|
|
this._createPropertyTypeCell(apiMember),
|
|
|
|
this._createDescriptionCell(apiMember),
|
|
|
|
])
|
|
|
|
);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (eventsTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Events',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(eventsTable);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (constructorsTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Constructors',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(constructorsTable);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (propertiesTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Properties',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(propertiesTable);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (methodsTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Methods',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(methodsTable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* GENERATE PAGE: ENUM
|
|
|
|
*/
|
|
|
|
private _writeEnumTables(output: DocSection, apiEnum: ApiEnum): void {
|
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
|
|
|
|
const enumMembersTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
|
|
|
headerTitles: ['Member', 'Value', 'Description'],
|
|
|
|
});
|
|
|
|
|
|
|
|
for (const apiEnumMember of apiEnum.members) {
|
|
|
|
enumMembersTable.addRow(
|
|
|
|
new DocTableRow({configuration}, [
|
|
|
|
new DocTableCell({configuration}, [
|
|
|
|
new DocParagraph({configuration}, [
|
|
|
|
new DocPlainText({
|
|
|
|
configuration,
|
|
|
|
text: Utilities.getConciseSignature(apiEnumMember),
|
|
|
|
}),
|
|
|
|
]),
|
|
|
|
]),
|
|
|
|
this._createInitializerCell(apiEnumMember),
|
|
|
|
this._createDescriptionCell(apiEnumMember),
|
|
|
|
])
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (enumMembersTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Enumeration Members',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(enumMembersTable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* GENERATE PAGE: INTERFACE
|
|
|
|
*/
|
|
|
|
private _writeInterfaceTables(
|
|
|
|
output: DocSection,
|
|
|
|
apiClass: ApiInterface
|
|
|
|
): void {
|
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
|
|
|
|
const eventsTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
|
|
|
headerTitles: ['Property', 'Modifiers', 'Type', 'Description'],
|
|
|
|
});
|
|
|
|
|
|
|
|
const propertiesTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
2022-10-24 07:07:05 +00:00
|
|
|
headerTitles: ['Property', 'Modifiers', 'Type', 'Description', 'Default'],
|
2022-07-01 11:52:39 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
const methodsTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
|
|
|
headerTitles: ['Method', 'Description'],
|
|
|
|
});
|
|
|
|
|
|
|
|
for (const apiMember of apiClass.members) {
|
|
|
|
switch (apiMember.kind) {
|
|
|
|
case ApiItemKind.ConstructSignature:
|
|
|
|
case ApiItemKind.MethodSignature: {
|
|
|
|
methodsTable.addRow(
|
|
|
|
new DocTableRow({configuration}, [
|
|
|
|
this._createTitleCell(apiMember),
|
|
|
|
this._createDescriptionCell(apiMember),
|
|
|
|
])
|
|
|
|
);
|
|
|
|
|
|
|
|
this._writeApiItemPage(apiMember);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ApiItemKind.PropertySignature: {
|
|
|
|
if ((apiMember as ApiPropertyItem).isEventProperty) {
|
|
|
|
eventsTable.addRow(
|
|
|
|
new DocTableRow({configuration}, [
|
2023-03-28 18:02:00 +00:00
|
|
|
this._createTitleCell(apiMember, true),
|
2022-07-01 11:52:39 +00:00
|
|
|
this._createModifiersCell(apiMember),
|
|
|
|
this._createPropertyTypeCell(apiMember),
|
|
|
|
this._createDescriptionCell(apiMember),
|
|
|
|
])
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
propertiesTable.addRow(
|
|
|
|
new DocTableRow({configuration}, [
|
2023-03-28 18:02:00 +00:00
|
|
|
this._createTitleCell(apiMember, true),
|
2022-07-01 11:52:39 +00:00
|
|
|
this._createModifiersCell(apiMember),
|
|
|
|
this._createPropertyTypeCell(apiMember),
|
|
|
|
this._createDescriptionCell(apiMember),
|
2022-10-24 07:07:05 +00:00
|
|
|
this._createDefaultCell(apiMember),
|
2022-07-01 11:52:39 +00:00
|
|
|
])
|
|
|
|
);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (eventsTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Events',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(eventsTable);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (propertiesTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Properties',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(propertiesTable);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (methodsTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Methods',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(methodsTable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* GENERATE PAGE: FUNCTION-LIKE
|
|
|
|
*/
|
|
|
|
private _writeParameterTables(
|
|
|
|
output: DocSection,
|
|
|
|
apiParameterListMixin: ApiParameterListMixin
|
|
|
|
): void {
|
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
|
|
|
|
const parametersTable: DocTable = new DocTable({
|
|
|
|
configuration,
|
|
|
|
headerTitles: ['Parameter', 'Type', 'Description'],
|
|
|
|
});
|
|
|
|
for (const apiParameter of apiParameterListMixin.parameters) {
|
|
|
|
const parameterDescription: DocSection = new DocSection({configuration});
|
|
|
|
|
|
|
|
if (apiParameter.isOptional) {
|
|
|
|
parameterDescription.appendNodesInParagraph([
|
|
|
|
new DocEmphasisSpan({configuration, italic: true}, [
|
|
|
|
new DocPlainText({configuration, text: '(Optional)'}),
|
|
|
|
]),
|
|
|
|
new DocPlainText({configuration, text: ' '}),
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (apiParameter.tsdocParamBlock) {
|
|
|
|
this._appendAndMergeSection(
|
|
|
|
parameterDescription,
|
|
|
|
apiParameter.tsdocParamBlock.content
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
parametersTable.addRow(
|
|
|
|
new DocTableRow({configuration}, [
|
|
|
|
new DocTableCell({configuration}, [
|
|
|
|
new DocParagraph({configuration}, [
|
|
|
|
new DocPlainText({configuration, text: apiParameter.name}),
|
|
|
|
]),
|
|
|
|
]),
|
|
|
|
new DocTableCell({configuration}, [
|
|
|
|
this._createParagraphForTypeExcerpt(
|
|
|
|
apiParameter.parameterTypeExcerpt
|
|
|
|
),
|
|
|
|
]),
|
|
|
|
new DocTableCell({configuration}, parameterDescription.nodes),
|
|
|
|
])
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parametersTable.rows.length > 0) {
|
|
|
|
output.appendNode(
|
|
|
|
new DocHeading({
|
|
|
|
configuration: this._tsdocConfiguration,
|
|
|
|
title: 'Parameters',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
output.appendNode(parametersTable);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ApiReturnTypeMixin.isBaseClassOf(apiParameterListMixin)) {
|
|
|
|
const returnTypeExcerpt: Excerpt =
|
|
|
|
apiParameterListMixin.returnTypeExcerpt;
|
|
|
|
output.appendNode(
|
|
|
|
new DocParagraph({configuration}, [
|
|
|
|
new DocEmphasisSpan({configuration, bold: true}, [
|
|
|
|
new DocPlainText({configuration, text: 'Returns:'}),
|
|
|
|
]),
|
|
|
|
])
|
|
|
|
);
|
|
|
|
|
|
|
|
output.appendNode(this._createParagraphForTypeExcerpt(returnTypeExcerpt));
|
|
|
|
|
|
|
|
if (apiParameterListMixin instanceof ApiDocumentedItem) {
|
|
|
|
if (
|
|
|
|
apiParameterListMixin.tsdocComment &&
|
|
|
|
apiParameterListMixin.tsdocComment.returnsBlock
|
|
|
|
) {
|
|
|
|
this._appendSection(
|
|
|
|
output,
|
|
|
|
apiParameterListMixin.tsdocComment.returnsBlock.content
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private _createParagraphForTypeExcerpt(excerpt: Excerpt): DocParagraph {
|
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
|
|
|
|
const paragraph: DocParagraph = new DocParagraph({configuration});
|
|
|
|
if (!excerpt.text.trim()) {
|
|
|
|
paragraph.appendNode(
|
|
|
|
new DocPlainText({configuration, text: '(not declared)'})
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
this._appendExcerptWithHyperlinks(paragraph, excerpt);
|
|
|
|
}
|
|
|
|
|
|
|
|
return paragraph;
|
|
|
|
}
|
|
|
|
|
|
|
|
private _appendExcerptWithHyperlinks(
|
|
|
|
docNodeContainer: DocNodeContainer,
|
|
|
|
excerpt: Excerpt
|
|
|
|
): void {
|
|
|
|
for (const token of excerpt.spannedTokens) {
|
|
|
|
this._appendExcerptTokenWithHyperlinks(docNodeContainer, token);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private _appendExcerptTokenWithHyperlinks(
|
|
|
|
docNodeContainer: DocNodeContainer,
|
|
|
|
token: ExcerptToken
|
|
|
|
): void {
|
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
|
|
|
|
// Markdown doesn't provide a standardized syntax for hyperlinks inside code
|
|
|
|
// spans, so we will render the type expression as DocPlainText. Instead of
|
|
|
|
// creating multiple DocParagraphs, we can simply discard any newlines and
|
|
|
|
// let the renderer do normal word-wrapping.
|
|
|
|
const unwrappedTokenText: string = token.text.replace(/[\r\n]+/g, ' ');
|
|
|
|
|
|
|
|
// If it's hyperlinkable, then append a DocLinkTag
|
|
|
|
if (token.kind === ExcerptTokenKind.Reference && token.canonicalReference) {
|
|
|
|
const apiItemResult: IResolveDeclarationReferenceResult =
|
|
|
|
this._apiModel.resolveDeclarationReference(
|
|
|
|
token.canonicalReference,
|
|
|
|
undefined
|
|
|
|
);
|
|
|
|
|
|
|
|
if (apiItemResult.resolvedApiItem) {
|
|
|
|
docNodeContainer.appendNode(
|
|
|
|
new DocLinkTag({
|
|
|
|
configuration,
|
2022-10-24 07:07:05 +00:00
|
|
|
tagName: StandardTags.link.tagName,
|
2022-07-01 11:52:39 +00:00
|
|
|
linkText: unwrappedTokenText,
|
|
|
|
urlDestination: this._getLinkFilenameForApiItem(
|
|
|
|
apiItemResult.resolvedApiItem
|
|
|
|
),
|
|
|
|
})
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise append non-hyperlinked text
|
|
|
|
docNodeContainer.appendNode(
|
|
|
|
new DocPlainText({configuration, text: unwrappedTokenText})
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-03-28 18:02:00 +00:00
|
|
|
private _createTitleCell(apiItem: ApiItem, plain = false): DocTableCell {
|
2022-07-01 11:52:39 +00:00
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
|
2023-03-28 18:02:00 +00:00
|
|
|
const text: string = Utilities.getConciseSignature(apiItem);
|
2022-07-01 11:52:39 +00:00
|
|
|
|
|
|
|
return new DocTableCell({configuration}, [
|
|
|
|
new DocParagraph({configuration}, [
|
2023-03-28 18:02:00 +00:00
|
|
|
plain
|
|
|
|
? new DocPlainText({configuration, text})
|
|
|
|
: new DocLinkTag({
|
|
|
|
configuration,
|
|
|
|
tagName: '@link',
|
|
|
|
linkText: text,
|
|
|
|
urlDestination: this._getLinkFilenameForApiItem(apiItem),
|
|
|
|
}),
|
2022-07-01 11:52:39 +00:00
|
|
|
]),
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This generates a DocTableCell for an ApiItem including the summary section
|
|
|
|
* and "(BETA)" annotation.
|
|
|
|
*
|
|
|
|
* @remarks
|
|
|
|
* We mostly assume that the input is an ApiDocumentedItem, but it's easier to
|
|
|
|
* perform this as a runtime check than to have each caller perform a type
|
|
|
|
* cast.
|
|
|
|
*/
|
|
|
|
private _createDescriptionCell(apiItem: ApiItem): DocTableCell {
|
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
|
|
|
|
const section: DocSection = new DocSection({configuration});
|
|
|
|
|
|
|
|
if (ApiReleaseTagMixin.isBaseClassOf(apiItem)) {
|
|
|
|
if (apiItem.releaseTag === ReleaseTag.Beta) {
|
|
|
|
section.appendNodesInParagraph([
|
|
|
|
new DocEmphasisSpan({configuration, bold: true, italic: true}, [
|
|
|
|
new DocPlainText({configuration, text: '(BETA)'}),
|
|
|
|
]),
|
|
|
|
new DocPlainText({configuration, text: ' '}),
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (apiItem instanceof ApiDocumentedItem) {
|
|
|
|
if (apiItem.tsdocComment !== undefined) {
|
|
|
|
this._appendAndMergeSection(
|
|
|
|
section,
|
|
|
|
apiItem.tsdocComment.summarySection
|
|
|
|
);
|
2024-04-04 07:17:26 +00:00
|
|
|
|
|
|
|
if (apiItem.tsdocComment.deprecatedBlock) {
|
|
|
|
section.appendNode(
|
|
|
|
new DocParagraph({configuration}, [
|
|
|
|
new DocEmphasisSpan({configuration, bold: true}, [
|
|
|
|
new DocPlainText({configuration, text: 'Deprecated: '}),
|
|
|
|
]),
|
|
|
|
])
|
|
|
|
);
|
|
|
|
|
|
|
|
section.appendNodes(
|
|
|
|
apiItem.tsdocComment.deprecatedBlock.content.getChildNodes()
|
|
|
|
);
|
|
|
|
}
|
2022-07-01 11:52:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return new DocTableCell({configuration}, section.nodes);
|
|
|
|
}
|
|
|
|
|
2022-10-24 07:07:05 +00:00
|
|
|
private _createDefaultCell(apiItem: ApiItem): DocTableCell {
|
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
|
|
|
|
if (apiItem instanceof ApiDocumentedItem) {
|
|
|
|
const block = apiItem.tsdocComment?.customBlocks.find(block => {
|
|
|
|
return (
|
|
|
|
block.blockTag.tagNameWithUpperCase ===
|
|
|
|
StandardTags.defaultValue.tagNameWithUpperCase
|
|
|
|
);
|
|
|
|
});
|
|
|
|
if (block !== undefined) {
|
|
|
|
return new DocTableCell({configuration}, block.content.getChildNodes());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return new DocTableCell({configuration}, []);
|
|
|
|
}
|
|
|
|
|
2022-07-01 11:52:39 +00:00
|
|
|
private _createModifiersCell(apiItem: ApiItem): DocTableCell {
|
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
|
|
|
|
const section: DocSection = new DocSection({configuration});
|
|
|
|
|
2024-04-04 07:17:26 +00:00
|
|
|
const codes = [];
|
|
|
|
|
2022-07-01 11:52:39 +00:00
|
|
|
if (ApiProtectedMixin.isBaseClassOf(apiItem)) {
|
|
|
|
if (apiItem.isProtected) {
|
2024-04-04 07:17:26 +00:00
|
|
|
codes.push('protected');
|
2022-07-01 11:52:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ApiReadonlyMixin.isBaseClassOf(apiItem)) {
|
|
|
|
if (apiItem.isReadonly) {
|
2024-04-04 07:17:26 +00:00
|
|
|
codes.push('readonly');
|
2022-07-01 11:52:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ApiStaticMixin.isBaseClassOf(apiItem)) {
|
|
|
|
if (apiItem.isStatic) {
|
2024-04-04 07:17:26 +00:00
|
|
|
codes.push('static');
|
2022-07-01 11:52:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-28 18:02:00 +00:00
|
|
|
if (ApiOptionalMixin.isBaseClassOf(apiItem)) {
|
|
|
|
if (apiItem.isOptional) {
|
2024-04-04 07:17:26 +00:00
|
|
|
codes.push('optional');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (apiItem instanceof ApiDocumentedItem) {
|
|
|
|
if (apiItem.tsdocComment?.deprecatedBlock) {
|
|
|
|
codes.push('deprecated');
|
2023-03-28 18:02:00 +00:00
|
|
|
}
|
|
|
|
}
|
2024-04-04 07:17:26 +00:00
|
|
|
if (codes.length) {
|
|
|
|
section.appendNode(
|
|
|
|
new DocParagraph({configuration}, [
|
|
|
|
new DocCodeSpan({configuration, code: codes.join(', ')}),
|
|
|
|
])
|
|
|
|
);
|
|
|
|
}
|
2023-03-28 18:02:00 +00:00
|
|
|
|
2022-07-01 11:52:39 +00:00
|
|
|
return new DocTableCell({configuration}, section.nodes);
|
|
|
|
}
|
|
|
|
|
|
|
|
private _createPropertyTypeCell(apiItem: ApiItem): DocTableCell {
|
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
|
|
|
|
const section: DocSection = new DocSection({configuration});
|
|
|
|
|
|
|
|
if (apiItem instanceof ApiPropertyItem) {
|
|
|
|
section.appendNode(
|
|
|
|
this._createParagraphForTypeExcerpt(apiItem.propertyTypeExcerpt)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return new DocTableCell({configuration}, section.nodes);
|
|
|
|
}
|
|
|
|
|
|
|
|
private _createInitializerCell(apiItem: ApiItem): DocTableCell {
|
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
|
|
|
|
const section: DocSection = new DocSection({configuration});
|
|
|
|
|
|
|
|
if (ApiInitializerMixin.isBaseClassOf(apiItem)) {
|
|
|
|
if (apiItem.initializerExcerpt) {
|
|
|
|
section.appendNodeInParagraph(
|
|
|
|
new DocCodeSpan({
|
|
|
|
configuration,
|
|
|
|
code: apiItem.initializerExcerpt.text,
|
|
|
|
})
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return new DocTableCell({configuration}, section.nodes);
|
|
|
|
}
|
|
|
|
|
|
|
|
private _writeBetaWarning(output: DocSection): void {
|
|
|
|
const configuration: TSDocConfiguration = this._tsdocConfiguration;
|
|
|
|
const betaWarning: string =
|
|
|
|
'This API is provided as a preview for developers and may change' +
|
|
|
|
' based on feedback that we receive. Do not use this API in a production environment.';
|
|
|
|
output.appendNode(
|
|
|
|
new DocNoteBox({configuration}, [
|
|
|
|
new DocParagraph({configuration}, [
|
|
|
|
new DocPlainText({configuration, text: betaWarning}),
|
|
|
|
]),
|
|
|
|
])
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
private _appendSection(output: DocSection, docSection: DocSection): void {
|
|
|
|
for (const node of docSection.nodes) {
|
|
|
|
output.appendNode(node);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private _appendAndMergeSection(
|
|
|
|
output: DocSection,
|
|
|
|
docSection: DocSection
|
|
|
|
): void {
|
|
|
|
let firstNode = true;
|
|
|
|
for (const node of docSection.nodes) {
|
|
|
|
if (firstNode) {
|
|
|
|
if (node.kind === DocNodeKind.Paragraph) {
|
|
|
|
output.appendNodesInParagraph(node.getChildNodes());
|
|
|
|
firstNode = false;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
firstNode = false;
|
|
|
|
|
|
|
|
output.appendNode(node);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private _getSidebarLabelForApiItem(apiItem: ApiItem): string {
|
|
|
|
if (apiItem.kind === ApiItemKind.Package) {
|
|
|
|
return 'API';
|
|
|
|
}
|
|
|
|
|
|
|
|
let baseName = '';
|
|
|
|
for (const hierarchyItem of apiItem.getHierarchy()) {
|
|
|
|
// For overloaded methods, add a suffix such as "MyClass.myMethod_2".
|
|
|
|
let qualifiedName: string = hierarchyItem.displayName;
|
|
|
|
if (ApiParameterListMixin.isBaseClassOf(hierarchyItem)) {
|
|
|
|
if (hierarchyItem.overloadIndex > 1) {
|
|
|
|
// Subtract one for compatibility with earlier releases of API Documenter.
|
|
|
|
qualifiedName += `_${hierarchyItem.overloadIndex - 1}`;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (hierarchyItem.kind) {
|
|
|
|
case ApiItemKind.Model:
|
|
|
|
case ApiItemKind.EntryPoint:
|
|
|
|
case ApiItemKind.EnumMember:
|
|
|
|
case ApiItemKind.Package:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
baseName += qualifiedName + '.';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return baseName.slice(0, baseName.length - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
private _getFilenameForApiItem(apiItem: ApiItem): string {
|
|
|
|
if (apiItem.kind === ApiItemKind.Package) {
|
|
|
|
return 'index.md';
|
|
|
|
}
|
|
|
|
|
|
|
|
let baseName = '';
|
|
|
|
for (const hierarchyItem of apiItem.getHierarchy()) {
|
|
|
|
// For overloaded methods, add a suffix such as "MyClass.myMethod_2".
|
|
|
|
let qualifiedName: string = Utilities.getSafeFilenameForName(
|
|
|
|
hierarchyItem.displayName
|
|
|
|
);
|
|
|
|
if (ApiParameterListMixin.isBaseClassOf(hierarchyItem)) {
|
|
|
|
if (hierarchyItem.overloadIndex > 1) {
|
|
|
|
// Subtract one for compatibility with earlier releases of API Documenter.
|
|
|
|
// (This will get revamped when we fix GitHub issue #1308)
|
|
|
|
qualifiedName += `_${hierarchyItem.overloadIndex - 1}`;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (hierarchyItem.kind) {
|
|
|
|
case ApiItemKind.Model:
|
|
|
|
case ApiItemKind.EntryPoint:
|
|
|
|
case ApiItemKind.EnumMember:
|
|
|
|
break;
|
|
|
|
case ApiItemKind.Package:
|
|
|
|
baseName = Utilities.getSafeFilenameForName(
|
|
|
|
PackageName.getUnscopedName(hierarchyItem.displayName)
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
baseName += '.' + qualifiedName;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return baseName + '.md';
|
|
|
|
}
|
|
|
|
|
|
|
|
private _getLinkFilenameForApiItem(apiItem: ApiItem): string {
|
|
|
|
return './' + this._getFilenameForApiItem(apiItem);
|
|
|
|
}
|
|
|
|
|
|
|
|
private _deleteOldOutputFiles(): void {
|
|
|
|
console.log('Deleting old output from ' + this._outputFolder);
|
|
|
|
FileSystem.ensureEmptyFolder(this._outputFolder);
|
|
|
|
}
|
|
|
|
}
|