Skip to content

课程 29 - 嵌入 HTML 内容

有时候我们希望在画布中嵌入 HTML 内容,例如 YouTube 播放器、CodeSandbox 组件、ShaderToy 等等。

创建 HTML 容器

Excalidraw 并不支持在画布中嵌入 HTML 内容,但 tldraw 支持 TLEmbedShape。它在网页中将一个 HTML 容器(含 iframe 或其他元素)和画布 <svg> 元素并排或叠加显示,而不是“完全”在单一画布内部。

HTML external content in tldraw

相机同步

课程 4 - 相机 中,我们介绍了相机的一系列重要参数:平移、旋转和缩放。现在我们需要把相机参数映射为 HTML 容器的 CSS transform,实现画布与 HTML 容器的同步。

HTML 图形

External content sources 例子中,我们可以看到 tldraw 是这样支持 HTML 内容的:

ts
class DangerousHtmlExample extends BaseBoxShapeUtil<IDangerousHtmlShape> {
    static override type = 'html' as const;

    override getDefaultProps() {
        return {
            type: 'html',
            w: 500,
            h: 300,
            html: '<div>hello</div>',
        };
    }
}

我们也增加一种可序列化图形,除了通用的属性 x/y/width/height 之外,最重要的属性就是 innerHTML 内容:

ts
export interface HtmlAttributes {
    html: string;
}
export interface HtmlSerializedNode
    extends BaseSerializeNode<'html'>,
        Partial<HtmlAttributes> {}

当位置发生改变时,需要通过 CSS Transform 同步:

ts
const { matrix } = entity.read(GlobalTransform);

$child.style.transform = `matrix(${toDomPrecision(
    matrix.m00,
)}, ${toDomPrecision(matrix.m01)}, ${toDomPrecision(
    matrix.m10,
)}, ${toDomPrecision(matrix.m11)}, ${toDomPrecision(
    matrix.m20,
)}, ${toDomPrecision(matrix.m21)})`;
$child.style.width = `${toDomPrecision(width)}px`;
$child.style.height = `${toDomPrecision(height)}px`;

粘贴 URL

课程 24 - 读取剪贴板 中,我们介绍过如何处理剪贴板中的图片和文本内容。

URL 是特殊的文本,在 tldraw 中:

  • 当 URL 被识别为外部链接时,默认处理器会抓取页面的 metadata(og:image、title、favicon、description),并把这些信息包装成一个 bookmark asset(TLBookmarkAsset)和对应的 shape,使用书签样式渲染
  • 当 URL 是可嵌入内容(例如 YouTube、Figma、Google Maps 等),此时使用 <iframe> 渲染
  • 当 URL 是图片或者视频资源时,把它作为媒体 asset(TLImageAsset / TLVideoAsset)去加载并用 ImageShapeUtil 渲染
ts
// @see https://github.com/tldraw/tldraw/blob/main/packages/tldraw/src/lib/ui/hooks/clipboard/pasteUrl.ts#L12
export async function pasteUrl() {
    return await editor.putExternalContent({
        type: 'url',
        point,
        url,
        sources,
    });
}

书签

ts
// @see https://github.com/tldraw/tldraw/blob/ef0eba14c5a8baf4f36b3659ac9af98256d3b5dd/packages/tldraw/src/lib/defaultExternalContentHandlers.ts#L249
export async function defaultHandleExternalUrlAsset() {
    let meta: {
        image: string;
        favicon: string;
        title: string;
        description: string;
    };

    const resp = await fetch(url, {
        method: 'GET',
        mode: 'no-cors',
    });
    const html = await resp.text();
    const doc = new DOMParser().parseFromString(html, 'text/html');
    meta = {
        image:
            doc.head
                .querySelector('meta[property="og:image"]')
                ?.getAttribute('content') ?? '',
        // title, favicon, description
    };

    // Create bookmark asset
}

iframe

粘贴 HTML 内容

从 VSCode 中复制的代码块是 HTML 片段:

html
<meta charset="utf-8" />
<div
    style="color: #e4e4e4;background-color: #181818;font-family: Menlo, Monaco, 'Courier New', monospace;font-weight: normal;font-size: 12px;line-height: 18px;white-space: pre;"
>
    <div><span style="color: #d6d6dd;">### iframe</span></div>
</div>

我们可以尝试从剪贴板中读取它:

ts
const html = event.clipboardData?.getData(MIME_TYPES.html); // text/html

然后就可以根据内容创建。为了获取尺寸,可以把 HTML 内容插入一个隐藏元素,然后让浏览器排版,最后读取它的 offsetWidth / offsetHeight

ts
function createHTML(
    api: ExtendedAPI,
    appState: AppState,
    html: string,
    position?: { x: number; y: number },
) {
    const { width, height } = measureHTML(html);

    updateAndSelectNodes(api, appState, [
        {
            id: uuidv4(),
            type: 'html',
            x: position?.x ?? 0,
            y: position?.y ?? 0,
            width,
            height,
            html,
        },
    ]);
}

导出成 SVG 或图片

课程 10 - 图片导入导出 中我们介绍过导出画布内容成 SVG 或者 PNG 等格式的图片。对于 HTML 内容可以使用社区内比较成熟的方案,例如 html-to-image

This library uses a feature of SVG that allows having arbitrary HTML content inside of the <foreignObject> tag.

Released under the MIT License.