chrome-devtools-frontend
Version:
Chrome DevTools UI
124 lines (107 loc) • 4.65 kB
text/typescript
// Copyright 2020 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import * as LitHtml from '../../lit-html/lit-html.js';
import type * as Marked from '../../../third_party/marked/marked.js';
import * as ComponentHelpers from '../../components/helpers/helpers.js';
import markdownViewStyles from './markdownView.css.js';
import {MarkdownLink, type MarkdownLinkData} from './MarkdownLink.js';
import {MarkdownImage, type MarkdownImageData} from './MarkdownImage.js';
const html = LitHtml.html;
const render = LitHtml.render;
export interface MarkdownViewData {
tokens: Marked.Marked.Token[];
}
export class MarkdownView extends HTMLElement {
static readonly litTagName = LitHtml.literal`devtools-markdown-view`;
readonly #shadow = this.attachShadow({mode: 'open'});
// TODO(crbug.com/1108699): Replace with `Marked.Marked.Token[]` once AST types are fixed upstream.
#tokenData: readonly Object[] = [];
connectedCallback(): void {
this.#shadow.adoptedStyleSheets = [markdownViewStyles];
}
set data(data: MarkdownViewData) {
this.#tokenData = data.tokens;
this.#update();
}
#update(): void {
this.#render();
}
#render(): void {
// Disabled until https://crbug.com/1079231 is fixed.
// clang-format off
render(html`
<div class='message'>
${this.#tokenData.map(renderToken)}
</div>
`, this.#shadow, {host: this});
// clang-format on
}
}
ComponentHelpers.CustomElements.defineComponent('devtools-markdown-view', MarkdownView);
declare global {
// eslint-disable-next-line @typescript-eslint/no-unused-vars
interface HTMLElementTagNameMap {
'devtools-markdown-view': MarkdownView;
}
}
// TODO(crbug.com/1108699): Fix types when they are available.
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const renderChildTokens = (token: any): string => {
return token.tokens.map(renderToken);
};
const unescape = (text: string): string => {
// Unescape will get rid of the escaping done by Marked to avoid double escaping due to escaping it also with Lit-html
// Table taken from: front_end/third_party/marked/package/src/helpers.js
const escapeReplacements = new Map<string, string>([
['&', '&'],
['<', '<'],
['>', '>'],
['"', '"'],
[''', '\''],
]);
return text.replace(/&(amp|lt|gt|quot|#39);/g, (matchedString: string) => {
const replacement = escapeReplacements.get(matchedString);
return replacement ? replacement : matchedString;
});
};
// TODO(crbug.com/1108699): Fix types when they are available.
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const renderText = (token: any): LitHtml.TemplateResult => {
if (token.tokens && token.tokens.length > 0) {
return html`${renderChildTokens(token)}`;
}
// Due to unescaping, unescaped html entities (see escapeReplacements' keys) will be rendered
// as their corresponding symbol while the rest will be rendered as verbatim.
// Marked's escape function can be found in front_end/third_party/marked/package/src/helpers.js
return html`${unescape(token.text)}`;
};
// TODO(crbug.com/1108699): Fix types when they are available.
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const tokenRenderers = new Map<string, (token: any) => LitHtml.TemplateResult>([
['paragraph', (token): LitHtml.TemplateResult => html`<p>${renderChildTokens(token)}`],
['list', (token): LitHtml.TemplateResult => html`<ul>${token.items.map(renderToken)}</ul>`],
['list_item', (token): LitHtml.TemplateResult => html`<li>${renderChildTokens(token)}`],
['text', renderText],
['codespan', (token): LitHtml.TemplateResult => html`<code>${unescape(token.text)}</code>`],
['space', (): LitHtml.TemplateResult => html``],
[
'link',
(token): LitHtml.TemplateResult => html`<${MarkdownLink.litTagName} .data=${
{key: token.href, title: token.text} as MarkdownLinkData}></${MarkdownLink.litTagName}>`,
],
[
'image',
(token): LitHtml.TemplateResult => html`<${MarkdownImage.litTagName} .data=${
{key: token.href, title: token.text} as MarkdownImageData}></${MarkdownImage.litTagName}>`,
],
]);
// TODO(crbug.com/1108699): Fix types when they are available.
// eslint-disable-next-line @typescript-eslint/no-explicit-any
export const renderToken = (token: any): LitHtml.TemplateResult => {
const renderFn = tokenRenderers.get(token.type);
if (!renderFn) {
throw new Error(`Markdown token type '${token.type}' not supported.`);
}
return renderFn(token);
};