2024-01-19 04:47:01 +08:00
|
|
|
<script lang="ts">
|
|
|
|
import "./prism.css";
|
|
|
|
|
|
|
|
import Prism from "prismjs";
|
|
|
|
import "prismjs/components/prism-python";
|
|
|
|
import "prismjs/components/prism-typescript";
|
|
|
|
|
|
|
|
interface Param {
|
|
|
|
type: string | null;
|
|
|
|
description: string;
|
|
|
|
default: string | null;
|
|
|
|
name?: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
export let docs: Record<string, Param>;
|
|
|
|
export let lang: "python" | "typescript" = "python";
|
|
|
|
export let linkify: string[] = [];
|
2024-07-11 08:58:28 +08:00
|
|
|
export let header: string | null;
|
|
|
|
|
|
|
|
let component_root: HTMLElement;
|
2024-01-19 04:47:01 +08:00
|
|
|
let _docs: Param[];
|
2024-07-11 08:58:28 +08:00
|
|
|
let all_open = false;
|
2024-01-19 04:47:01 +08:00
|
|
|
|
|
|
|
$: {
|
|
|
|
setTimeout(() => {
|
|
|
|
_docs = highlight_code(docs, lang);
|
|
|
|
}, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
function highlight(code: string, lang: "python" | "typescript"): string {
|
|
|
|
let highlighted = Prism.highlight(code, Prism.languages[lang], lang);
|
|
|
|
|
|
|
|
for (const link of linkify) {
|
|
|
|
highlighted = highlighted.replace(
|
|
|
|
new RegExp(link, "g"),
|
|
|
|
`<a href="#h-${link.toLocaleLowerCase()}">${link}</a>`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return highlighted;
|
|
|
|
}
|
|
|
|
|
|
|
|
function highlight_code(
|
|
|
|
_docs: typeof docs,
|
|
|
|
lang: "python" | "typescript"
|
|
|
|
): Param[] {
|
2024-07-17 06:53:04 +08:00
|
|
|
if (!_docs) {
|
|
|
|
return [];
|
|
|
|
}
|
2024-01-19 04:47:01 +08:00
|
|
|
return Object.entries(_docs).map(
|
|
|
|
([name, { type, description, default: _default }]) => {
|
|
|
|
let highlighted_type = type ? highlight(type, lang) : null;
|
|
|
|
|
|
|
|
return {
|
|
|
|
name: name,
|
|
|
|
type: highlighted_type,
|
|
|
|
description: description,
|
|
|
|
default: _default ? highlight(_default, lang) : null
|
|
|
|
};
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
2024-07-11 08:58:28 +08:00
|
|
|
|
|
|
|
function toggle_all(): void {
|
|
|
|
all_open = !all_open;
|
|
|
|
const details = component_root.querySelectorAll(".param");
|
|
|
|
details.forEach((detail) => {
|
|
|
|
if (detail instanceof HTMLDetailsElement) {
|
|
|
|
detail.open = all_open;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2024-01-19 04:47:01 +08:00
|
|
|
</script>
|
|
|
|
|
2024-07-11 08:58:28 +08:00
|
|
|
<div class="wrap" bind:this={component_root}>
|
|
|
|
{#if header !== null}
|
|
|
|
<div class="header">
|
|
|
|
<span class="title">{header}</span>
|
|
|
|
<button
|
|
|
|
class="toggle-all"
|
|
|
|
on:click={toggle_all}
|
|
|
|
title={all_open ? "Close All" : "Open All"}
|
|
|
|
>
|
|
|
|
▼
|
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
{/if}
|
2024-01-19 04:47:01 +08:00
|
|
|
{#if _docs}
|
2024-07-11 08:58:28 +08:00
|
|
|
{#each _docs as { type, description, default: _default, name } (name)}
|
|
|
|
<details class="param md">
|
|
|
|
<summary class="type">
|
|
|
|
<pre class="language-{lang}"><code
|
2024-01-19 04:47:01 +08:00
|
|
|
>{name}{#if type}: {@html type}{/if}</code
|
|
|
|
></pre>
|
2024-07-11 08:58:28 +08:00
|
|
|
</summary>
|
|
|
|
{#if _default}
|
|
|
|
<div class="default" class:last={!description}>
|
|
|
|
<span style:padding-right={"4px"}>default</span>
|
|
|
|
<code>= {@html _default}</code>
|
|
|
|
</div>
|
2024-01-19 04:47:01 +08:00
|
|
|
{/if}
|
2024-07-11 08:58:28 +08:00
|
|
|
{#if description}
|
|
|
|
<div class="description"><p>{description}</p></div>
|
|
|
|
{/if}
|
|
|
|
</details>
|
2024-01-19 04:47:01 +08:00
|
|
|
{/each}
|
|
|
|
{/if}
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<style>
|
2024-07-11 08:58:28 +08:00
|
|
|
.header {
|
|
|
|
display: flex;
|
|
|
|
justify-content: space-between;
|
|
|
|
align-items: center;
|
|
|
|
padding: 0.7rem 1rem;
|
|
|
|
border-bottom: 1px solid var(--table-border-color);
|
|
|
|
}
|
|
|
|
|
|
|
|
.title {
|
|
|
|
font-size: var(--scale-0);
|
|
|
|
font-weight: 600;
|
|
|
|
color: var(--body-text-color);
|
|
|
|
}
|
|
|
|
|
|
|
|
.toggle-all {
|
|
|
|
background: none;
|
|
|
|
border: none;
|
|
|
|
cursor: pointer;
|
|
|
|
padding: 0;
|
|
|
|
color: var(--body-text-color);
|
|
|
|
font-size: 0.7em;
|
|
|
|
line-height: 1;
|
|
|
|
opacity: 0.7;
|
|
|
|
transition:
|
|
|
|
opacity 0.2s ease,
|
|
|
|
transform 0.3s ease;
|
|
|
|
}
|
|
|
|
|
|
|
|
.toggle-all:hover {
|
|
|
|
opacity: 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
:global(.wrap[data-all-open="true"]) .toggle-all {
|
|
|
|
transform: rotate(180deg);
|
|
|
|
}
|
|
|
|
|
2024-01-19 04:47:01 +08:00
|
|
|
.default :global(pre),
|
|
|
|
.default :global(.highlight) {
|
|
|
|
display: inline-block;
|
|
|
|
}
|
|
|
|
|
|
|
|
.wrap :global(pre),
|
|
|
|
.wrap :global(.highlight) {
|
2024-01-26 05:11:38 +08:00
|
|
|
margin: 0 !important;
|
2024-01-19 04:47:01 +08:00
|
|
|
background: transparent !important;
|
|
|
|
font-family: var(--font-mono);
|
|
|
|
font-weight: 400;
|
2024-01-26 05:11:38 +08:00
|
|
|
padding: 0 !important;
|
2024-01-19 04:47:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
.wrap :global(pre a) {
|
|
|
|
color: var(--link-text-color-hover);
|
|
|
|
text-decoration: underline;
|
|
|
|
}
|
|
|
|
|
|
|
|
.wrap :global(pre a:hover) {
|
|
|
|
color: var(--link-text-color-hover);
|
|
|
|
}
|
|
|
|
|
|
|
|
.default > span {
|
|
|
|
text-transform: uppercase;
|
|
|
|
font-size: 0.7rem;
|
|
|
|
font-weight: 600;
|
|
|
|
}
|
|
|
|
|
|
|
|
.default > code {
|
|
|
|
border: none;
|
|
|
|
}
|
|
|
|
code {
|
|
|
|
background: none;
|
|
|
|
font-family: var(--font-mono);
|
|
|
|
}
|
|
|
|
|
|
|
|
.wrap {
|
|
|
|
padding: 0rem;
|
|
|
|
border-radius: 5px;
|
|
|
|
border: 1px solid #eee;
|
|
|
|
overflow: hidden;
|
|
|
|
position: relative;
|
|
|
|
margin: 0;
|
|
|
|
box-shadow: var(--block-shadow);
|
|
|
|
border-width: var(--block-border-width);
|
|
|
|
border-color: var(--block-border-color);
|
|
|
|
border-radius: var(--block-radius);
|
|
|
|
width: 100%;
|
|
|
|
line-height: var(--line-sm);
|
|
|
|
color: var(--body-text-color);
|
|
|
|
}
|
|
|
|
|
|
|
|
.type {
|
|
|
|
position: relative;
|
|
|
|
padding: 0.7rem 1rem;
|
2024-01-26 05:11:38 +08:00
|
|
|
background: var(--table-odd-background-fill);
|
|
|
|
border-bottom: 0px solid var(--table-border-color);
|
2024-07-11 08:58:28 +08:00
|
|
|
list-style: none;
|
2024-01-19 04:47:01 +08:00
|
|
|
}
|
|
|
|
|
2024-07-11 08:58:28 +08:00
|
|
|
.type::after {
|
|
|
|
content: "▼";
|
2024-01-19 04:47:01 +08:00
|
|
|
position: absolute;
|
2024-07-11 08:58:28 +08:00
|
|
|
top: 50%;
|
2024-01-19 04:47:01 +08:00
|
|
|
right: 15px;
|
2024-07-11 08:58:28 +08:00
|
|
|
transform: translateY(-50%);
|
|
|
|
transition: transform 0.3s ease;
|
|
|
|
font-size: 0.7em;
|
|
|
|
opacity: 0.7;
|
2024-01-19 04:47:01 +08:00
|
|
|
}
|
|
|
|
|
2024-07-11 08:58:28 +08:00
|
|
|
details[open] .type::after {
|
|
|
|
transform: translateY(-50%) rotate(180deg);
|
2024-01-19 04:47:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
.default {
|
|
|
|
padding: 0.2rem 1rem 0.3rem 1rem;
|
2024-01-26 05:11:38 +08:00
|
|
|
border-bottom: 1px solid var(--table-border-color);
|
|
|
|
background: var(--block-background-fill);
|
2024-01-19 04:47:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
.default.last {
|
|
|
|
border-bottom: none;
|
|
|
|
}
|
|
|
|
|
|
|
|
.description {
|
|
|
|
padding: 0.7rem 1rem;
|
|
|
|
font-size: var(--scale-00);
|
|
|
|
font-family: var(--font-sans);
|
2024-01-26 05:11:38 +08:00
|
|
|
background: var(--block-background-fill);
|
2024-01-19 04:47:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
.param {
|
2024-01-26 05:11:38 +08:00
|
|
|
border-bottom: 1px solid var(--table-border-color);
|
|
|
|
}
|
|
|
|
|
|
|
|
.param:last-child {
|
|
|
|
border-bottom: none;
|
2024-01-19 04:47:01 +08:00
|
|
|
}
|
|
|
|
|
2024-07-11 08:58:28 +08:00
|
|
|
details[open] .type {
|
2024-01-19 04:47:01 +08:00
|
|
|
border-bottom-width: 1px;
|
|
|
|
}
|
|
|
|
|
|
|
|
.param.md code {
|
|
|
|
background: none;
|
|
|
|
}
|
2024-07-11 08:58:28 +08:00
|
|
|
|
|
|
|
details > summary {
|
|
|
|
cursor: pointer;
|
|
|
|
}
|
|
|
|
|
|
|
|
details > summary::-webkit-details-marker {
|
|
|
|
display: none;
|
|
|
|
}
|
2024-01-19 04:47:01 +08:00
|
|
|
</style>
|