UNPKG

jsii-docgen

Version:

generates api docs for jsii modules

159 lines 20 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.MarkdownDocument = void 0; /** * Markdown element. */ class MarkdownDocument { /** * Sanitize markdown reserved characters from external input. */ static sanitize(line) { let sanitized = line.trim(); if (line.startsWith('-')) { sanitized = sanitized.substring(1, line.length).trim(); } return sanitized; } /** * Remove newlines from markdown. */ static removeNewlines(line) { return line.replace(/\n/g, ' '); } static bold(text) { return `**${text}**`; } static pre(text) { // using <code> instead of backticks since this allows links return `<code>${text}</code>`; } static italic(text) { return `*${text}*`; } constructor(options = {}) { var _a, _b; this.options = options; this._lines = new Array(); this._sections = new Array(); this.id = (_a = options.id) !== null && _a !== void 0 ? _a : (_b = options.header) === null || _b === void 0 ? void 0 : _b.title; this.header = this.formatHeader(); } /** * Render a docs element into the markdown. */ docs(docs, language) { if (docs.summary) { this.lines(MarkdownDocument.sanitize(docs.summary)); this.lines(''); } if (docs.remarks) { this.lines(MarkdownDocument.sanitize(docs.remarks)); this.lines(''); } if (docs.links) { for (const link of docs.links) { this.quote(`[${link}](${link})`); } } if (docs.example) { if (!language) { throw new Error('language must be provided if docs.example has been specified'); } const example = new MarkdownDocument({ id: `${this.options.id}.example`, }); example.lines(MarkdownDocument.italic('Example'), ''); example.code(language.toString(), docs.example); example.lines(''); this.section(example); } } table(data) { const numColumns = data[0].length; const header = data[0]; const rows = data.slice(1); this.lines('| ' + header.map(this.escapePipes).join(' | ') + ' |'); this.lines('|' + ' --- |'.repeat(numColumns)); for (const row of rows) { this.lines('| ' + row.map(this.escapePipes).join(' | ') + ' |'); } this.lines(''); } quote(line) { this.lines(`> ${line}`); this.lines(''); } bullet(line) { this.lines(`- ${line}`); } code(language, ...snippet) { this.lines(`\`\`\`${language}`, ...snippet, '```'); this.lines(''); } lines(...lines) { this._lines.push(...lines); } split() { this.lines('---'); this.lines(''); } section(section) { this._sections.push(section); } render(headerSize = 0) { var _a, _b; const content = []; if (this.header) { if (headerSize > 6) { // headers are mapped to `h1-h6` html elements. // passed that, markdown just renders `#` signs. // lets see if and when we'll hit this limit. throw new Error('Unable to render markdown. Header limit (6) reached.'); } const heading = `${'#'.repeat(headerSize)} ${this.header}`; // temporary hack to avoid breaking Construct Hub const headerSpan = !!process.env.HEADER_SPAN; if (headerSpan) { content.push(`${heading} <span data-heading-title="${(_a = this.options.header) === null || _a === void 0 ? void 0 : _a.title}" data-heading-id="${this.id}"></span>`); } else { content.push(`${heading} <a name="${(_b = this.options.header) === null || _b === void 0 ? void 0 : _b.title}" id="${this.id}"></a>`); } content.push(''); } for (const line of this._lines) { content.push(`${line}`); } for (const section of this._sections) { content.push(section.render(headerSize + 1)); } return content.join('\n'); } formatHeader() { var _a, _b, _c, _d, _e, _f, _g; if (!((_a = this.options.header) === null || _a === void 0 ? void 0 : _a.title)) { return undefined; } let caption = this.options.header.title; if ((_c = (_b = this.options.header) === null || _b === void 0 ? void 0 : _b.pre) !== null && _c !== void 0 ? _c : false) { caption = `\`${caption}\``; } if ((_e = (_d = this.options.header) === null || _d === void 0 ? void 0 : _d.strike) !== null && _e !== void 0 ? _e : false) { caption = `~~${caption}~~`; } if ((_f = this.options.header) === null || _f === void 0 ? void 0 : _f.sup) { caption = `${caption}<sup>${(_g = this.options.header) === null || _g === void 0 ? void 0 : _g.sup}</sup>`; } return caption; } escapePipes(line) { return line.replace(/\|/g, '\\|'); } } exports.MarkdownDocument = MarkdownDocument; /** * An empty markdown element. */ MarkdownDocument.EMPTY = new MarkdownDocument(); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"markdown-doc.js","sourceRoot":"","sources":["../../../src/docgen/render/markdown-doc.ts"],"names":[],"mappings":"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","sourcesContent":["import { Language } from '../..';\nimport { DocsSchema } from '../schema';\n\n/**\n * Options for defining a markdown header.\n */\nexport interface MarkdownHeaderOptions {\n  /**\n   * Title to be displayed.\n   */\n  readonly title?: string;\n\n  /**\n   * Superscript.\n   *\n   * @default - No superscript\n   */\n  readonly sup?: string;\n\n  /**\n   * Preformat the header.\n   *\n   * @default false\n   */\n  readonly pre?: boolean;\n\n  /**\n   * Strikethough the title.\n   *\n   * @default false\n   */\n  readonly strike?: boolean;\n}\n\n/**\n * Options for defining a markdown element.\n */\nexport interface MarkdownOptions {\n  /**\n   * Markdown header.\n   *\n   * @default - No header.\n   */\n  readonly header?: MarkdownHeaderOptions;\n\n  /**\n   * Id of the element.\n   *\n   * @default - The title will be used.\n   */\n  readonly id?: string;\n}\n\n/**\n * Markdown element.\n */\nexport class MarkdownDocument {\n  /**\n   * An empty markdown element.\n   */\n  public static readonly EMPTY = new MarkdownDocument();\n\n  /**\n   * Sanitize markdown reserved characters from external input.\n   */\n  public static sanitize(line: string): string {\n    let sanitized = line.trim();\n\n    if (line.startsWith('-')) {\n      sanitized = sanitized.substring(1, line.length).trim();\n    }\n\n    return sanitized;\n  }\n\n  /**\n   * Remove newlines from markdown.\n   */\n  public static removeNewlines(line: string): string {\n    return line.replace(/\\n/g, ' ');\n  }\n\n  public static bold(text: string): string {\n    return `**${text}**`;\n  }\n\n  public static pre(text: string): string {\n    // using <code> instead of backticks since this allows links\n    return `<code>${text}</code>`;\n  }\n\n  public static italic(text: string) {\n    return `*${text}*`;\n  }\n\n  private readonly _lines = new Array<string>();\n  private readonly _sections = new Array<MarkdownDocument>();\n\n  private readonly id?: string;\n  private readonly header?: string;\n\n  constructor(private readonly options: MarkdownOptions = {}) {\n    this.id = options.id ?? options.header?.title;\n    this.header = this.formatHeader();\n  }\n\n  /**\n   * Render a docs element into the markdown.\n   */\n  public docs(docs: DocsSchema, language?: Language) {\n    if (docs.summary) {\n      this.lines(MarkdownDocument.sanitize(docs.summary));\n      this.lines('');\n    }\n    if (docs.remarks) {\n      this.lines(MarkdownDocument.sanitize(docs.remarks));\n      this.lines('');\n    }\n\n    if (docs.links) {\n      for (const link of docs.links) {\n        this.quote(`[${link}](${link})`);\n      }\n    }\n\n    if (docs.example) {\n      if (!language) {\n        throw new Error('language must be provided if docs.example has been specified');\n      }\n      const example = new MarkdownDocument({\n        id: `${this.options.id}.example`,\n      });\n      example.lines(MarkdownDocument.italic('Example'), '');\n      example.code(language.toString(), docs.example);\n      example.lines('');\n      this.section(example);\n    }\n  }\n\n  public table(data: string[][]) {\n    const numColumns = data[0].length;\n    const header = data[0];\n    const rows = data.slice(1);\n    this.lines('| ' + header.map(this.escapePipes).join(' | ') + ' |');\n    this.lines('|' + ' --- |'.repeat(numColumns));\n    for (const row of rows) {\n      this.lines('| ' + row.map(this.escapePipes).join(' | ') + ' |');\n    }\n    this.lines('');\n  }\n\n  public quote(line: string) {\n    this.lines(`> ${line}`);\n    this.lines('');\n  }\n\n  public bullet(line: string) {\n    this.lines(`- ${line}`);\n  }\n\n  public code(language: string, ...snippet: string[]) {\n    this.lines(`\\`\\`\\`${language}`, ...snippet, '```');\n    this.lines('');\n  }\n\n  public lines(...lines: string[]) {\n    this._lines.push(...lines);\n  }\n\n  public split() {\n    this.lines('---');\n    this.lines('');\n  }\n\n  public section(section: MarkdownDocument) {\n    this._sections.push(section);\n  }\n\n  public render(headerSize: number = 0): string {\n    const content: string[] = [];\n\n    if (this.header) {\n      if (headerSize > 6) {\n        // headers are mapped to `h1-h6` html elements.\n        // passed that, markdown just renders `#` signs.\n        // lets see if and when we'll hit this limit.\n        throw new Error('Unable to render markdown. Header limit (6) reached.');\n      }\n\n      const heading = `${'#'.repeat(headerSize)} ${this.header}`;\n\n      // temporary hack to avoid breaking Construct Hub\n      const headerSpan = !!process.env.HEADER_SPAN;\n      if (headerSpan) {\n        content.push(\n          `${heading} <span data-heading-title=\"${this.options.header?.title}\" data-heading-id=\"${this.id}\"></span>`,\n        );\n      } else {\n        content.push(`${heading} <a name=\"${this.options.header?.title}\" id=\"${this.id}\"></a>`);\n      }\n      content.push('');\n    }\n\n    for (const line of this._lines) {\n      content.push(`${line}`);\n    }\n\n    for (const section of this._sections) {\n      content.push(section.render(headerSize + 1));\n    }\n    return content.join('\\n');\n  }\n\n  private formatHeader(): string | undefined {\n    if (!this.options.header?.title) {\n      return undefined;\n    }\n    let caption = this.options.header.title;\n\n    if (this.options.header?.pre ?? false) {\n      caption = `\\`${caption}\\``;\n    }\n\n    if (this.options.header?.strike ?? false) {\n      caption = `~~${caption}~~`;\n    }\n\n    if (this.options.header?.sup) {\n      caption = `${caption}<sup>${this.options.header?.sup}</sup>`;\n    }\n\n    return caption;\n  }\n\n  private escapePipes(line: string): string {\n    return line.replace(/\\|/g, '\\\\|');\n  }\n}\n"]}