📝 Sign Up | 🔐 Log In

← Root | ↑ Up

┌──────────────────────────────────────────────────────────┐ │ 📄 shadcn/directory/udecode/plate/(guides)/static.cn │ └──────────────────────────────────────────────────────────┘

╔══════════════════════════════════════════════════════════════════════════════════════════════╗
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║

title: 静态渲染 description: 一个最小化、记忆化、只读版本的 Plate,支持 RSC/SSR。

<PlateStatic> 是一个快速、只读的 React 组件,用于渲染 Plate 内容,针对服务器端React Server Component (RSC) 环境进行了优化。它避免了客户端编辑逻辑,并对节点渲染进行记忆化,相比在只读模式下使用 <Plate>,具有更好的性能。

它是 serializeHtml 用于 HTML 导出的核心部分,非常适合任何需要非交互式、展示性 Plate 内容视图的服务器或 RSC 环境。

主要优势

  • 服务器安全: 无浏览器 API 依赖;可在 SSR/RSC 中工作。
  • 无 Plate 编辑器开销: 排除交互功能,如选择或事件处理程序。
  • 记忆化渲染: 使用 _memo 和结构检查,仅重新渲染已更改的节点。
  • 部分重新渲染: 文档某一部分的更改不会强制完全重新渲染。
  • 轻量级: 由于省略了交互式编辑器代码,打包体积更小。

何时使用 <PlateStatic>

  • 使用 HTML 序列化 生成 HTML。
  • 在 Next.js 中显示服务器渲染的预览(特别是使用 RSC)。
  • 构建具有只读 Plate 内容的静态站点。
  • 优化性能关键的只读视图。
  • 渲染 AI 流式内容。
<Callout type="info" title="交互式 vs. 静态"> 对于交互式只读功能(如评论弹出框或选择),请在浏览器中使用标准 `<Plate>` 组件。对于纯服务器渲染的、非交互式的内容,`<PlateStatic>` 是推荐的选择。 </Callout>

Kit 使用

<Steps>

安装

启用静态渲染的最快方法是使用 BaseEditorKit,它包含了预配置的基础插件,可以与服务器端渲染无缝配合。

<ComponentSource name="editor-base-kit" />

添加 Kit

import { createSlateEditor, PlateStatic } from 'platejs';
import { BaseEditorKit } from '@/components/editor/editor-base-kit';

const editor = createSlateEditor({
  plugins: BaseEditorKit,
  value: [
    { type: 'h1', children: [{ text: '服务器渲染的标题' }] },
    { type: 'p', children: [{ text: '此内容是静态渲染的。' }] },
  ],
});

// 静态渲染
export default function MyStaticPage() {
  return <PlateStatic editor={editor} />;
}

示例

查看完整的服务器端静态渲染示例:

<ComponentSource name="slate-to-html" /> </Steps>

手动使用

<Steps>

创建 Slate 编辑器

使用 createSlateEditor 初始化一个 Slate 编辑器实例,包含所需的插件和组件。这类似于为交互式 <Plate> 组件使用 usePlateEditor

import { createSlateEditor } from 'platejs';
// 导入所需的基础插件(例如 BaseHeadingPlugin、MarkdownPlugin)
// 确保在服务器环境中不要从 /react 子路径导入。

const editor = createSlateEditor({
  plugins: [
    // 在此添加您的基础插件列表
    // 示例:BaseHeadingPlugin, MarkdownPlugin.configure({...})
  ],
  value: [ // 示例初始值
    {
      type: 'p',
      children: [{ text: '来自静态 Plate 编辑器的问候!' }],
    },
  ],
});

定义静态节点组件

如果您的交互式编辑器使用客户端组件(例如带有 use client 或事件处理程序),您必须创建静态的、服务器安全的等效组件。这些组件应该渲染纯 HTML,不包含浏览器特定的逻辑。

import React from 'react';
import type { SlateElementProps } from 'platejs';

export function ParagraphElementStatic(props: SlateElementProps) {
  return (
    <SlateElement {...props}>
      {props.children}
    </SlateElement>
  );
}

为标题、图片、链接等创建类似的静态组件。

映射插件键到静态组件

创建一个对象,将插件键或节点类型映射到相应的静态 React 组件,然后将其传递给编辑器。

import { ParagraphElementStatic } from './ui/paragraph-node-static';
import { HeadingElementStatic } from './ui/heading-node-static';
// ... 导入其他静态组件

export const staticComponents = {
  p: ParagraphElementStatic,
  h1: HeadingElementStatic,
  // ... 为所有元素和叶子类型添加映射
};

渲染 <PlateStatic>

使用 <PlateStatic> 组件,提供配置了组件的 editor 实例。

import { PlateStatic } from 'platejs';
import { createSlateEditor } from 'platejs';
// import { BaseHeadingPlugin, ... } from '@platejs/basic-nodes'; // 等等
import { staticComponents } from '@/components/static-components';

export default async function MyStaticPage() {
  // 示例:获取或定义编辑器值
  const initialValue = [
    { type: 'h1', children: [{ text: '服务器渲染的标题' }] },
    { type: 'p', children: [{ text: '静态渲染的内容。' }] },
  ];

  const editor = createSlateEditor({
    plugins: [/* 您的基础插件 */],
    components: staticComponents,
    value: initialValue,
  });

  return (
    <PlateStatic
      editor={editor}
      style={{ padding: 16 }}
      className="my-plate-static-content"
    />
  );
}
<Callout type="note" title="值覆盖"> 如果您直接向 `<PlateStatic>` 传递 `value` 属性,它将覆盖 `editor.children`。 ```tsx <PlateStatic editor={editor} value={[ { type: 'p', children: [{ text: '覆盖的内容。' }] } ]} /> ``` </Callout>

记忆化详情

<PlateStatic> 通过记忆化提升性能:

  • 每个 <ElementStatic><LeafStatic> 都被 React.memo 包装。
  • 引用相等性: 未更改的节点引用可防止重新渲染。
  • _memo 字段: 在元素或叶子上设置 node._memo = true(或任何稳定值)可以强制 Plate 跳过重新渲染该特定节点,即使其内容发生变化。这对于更新的细粒度控制很有用。
</Steps>

客户端替代方案:PlateView

对于需要与静态内容进行最小交互的情况,请使用 <PlateView>。该组件包装了 <PlateStatic> 并添加了客户端事件处理程序来处理用户交互,同时保持静态渲染的性能优势。

示例:具有两种静态视图的服务器组件

import { createStaticEditor } from 'platejs';
import { PlateStatic } from 'platejs';
import { BaseEditorKit } from '@/components/editor/editor-base-kit';
import { InteractiveViewer } from './interactive-viewer';

export default async function DocumentPage() {
  const content = await fetchDocument(); // 您的文档数据
  
  // 服务器端静态编辑器
  const editor = createStaticEditor({
    plugins: BaseEditorKit,
    value: content,
  });

  return (
    <div className="grid grid-cols-2 gap-4">
      {/* 纯静态渲染 - 无交互性 */}
      <div>
        <h2>静态视图(服务器渲染)</h2>
        <PlateStatic editor={editor} />
      </div>

      {/* 交互式视图 - 在客户端渲染 */}
      <div>
        <h2>交互式视图</h2>
        <InteractiveViewer value={content} />
      </div>
    </div>
  );
}

示例:使用 PlateView 的客户端组件

'use client';

import { usePlateViewEditor } from 'platejs/react';
import { PlateView } from 'platejs/react';
import { BaseEditorKit } from '@/components/editor/editor-base-kit';

export function InteractiveViewer({ value }) {
  const editor = usePlateViewEditor({
    plugins: BaseEditorKit,
    value,
  });

  return <PlateView editor={editor} />;
}

PlateView 的主要功能

  • 仅客户端: 需要 'use client' 指令
  • 添加交互性: 启用用户与内容的交互(例如文本选择、复制、未来的交互如工具提示、高亮等)
  • 最小开销: 内部仍使用 PlateStatic 进行渲染
  • usePlateViewEditor 配合使用: 创建一个为仅查看 React 组件优化的静态编辑器
  • 包含 ViewPlugin: 静态编辑器自动包含 ViewPlugin,提供事件处理功能
<Callout type="warning" title="服务器组件兼容性"> `PlateView` 不能在服务器组件中使用。如果您从服务器组件向客户端组件传递编辑器,将遇到序列化错误。在服务器端使用 `PlateStatic`,或在客户端使用 `usePlateViewEditor` 创建编辑器。 </Callout>

PlateStatic vs. PlateView vs. Plate + readOnly

| 方面 | <PlateStatic> | <PlateView> | <Plate> + readOnly | | --------------------- | ----------------------------------------------------- | ------------------------------------------------------ | ------------------------------------------------------ | | 环境 | 服务器/客户端(SSR/RSC 安全) | 仅客户端 | 仅客户端 | | 交互性 | 无 | 最小(选择、复制、工具栏等) | 完整的交互功能(仅浏览器) | | 浏览器 API | 不使用 | 最小(事件处理程序) | 完全使用 | | 性能 | 最佳 - 仅静态 HTML | 良好 - 静态渲染 + 事件委托 | 较重 - 完整的编辑器内部结构 | | 打包体积 | 最小 | 小 | 最大 | | 使用场景 | 服务器渲染、HTML 导出 | 需要基本交互的客户端内容 | 需要所有功能的完整只读编辑器 | | 推荐 | 无任何交互的 SSR/RSC | 需要轻量级交互的客户端内容 | 具有复杂交互需求的客户端 |

RSC/SSR 示例

在 Next.js App Router(或类似的 RSC 环境)中,<PlateStatic> 可以直接在服务器组件中使用:

import { PlateStatic } from 'platejs';
import { createSlateEditor } from 'platejs';
// 示例基础插件(确保非 /react 导入)
// import { BaseHeadingPlugin } from '@platejs/basic-nodes';
import { staticComponents } from '@/components/static-components'; // 您的静态组件映射

export default async function Page() {
  // 在服务器端获取或定义内容
  const serverContent = [
    { type: 'h1', children: [{ text: '在服务器上渲染! 🎉' }] },
    { type: 'p', children: [{ text: '此内容是静态的且在服务器端渲染。' }] },
  ];

  const editor = createSlateEditor({
    // plugins: [BaseHeadingPlugin, /* ...其他基础插件 */],
    plugins: [], // 添加您的基础插件
    components: staticComponents,
    value: serverContent,
  });

  return (
    <PlateStatic
      editor={editor}
      className="my-static-preview-container"
    />
  );
}

这会在服务器上将内容渲染为 HTML,而无需为 PlateStatic 本身提供客户端 JavaScript 包。

serializeHtml 配合使用

要生成完整的 HTML 字符串(例如用于电子邮件、PDF 或外部系统),请使用 serializeHtml。它在内部使用 <PlateStatic>

import { createSlateEditor, serializeHtml } from 'platejs';
import { staticComponents } from '@/components/static-components';
// import { BaseHeadingPlugin, ... } from '@platejs/basic-nodes';

async function getDocumentAsHtml(value: any[]) {
  const editor = createSlateEditor({
    plugins: [/* ...您的基础插件... */],
    components: staticComponents,
    value,
  });

  const html = await serializeHtml(editor, {
    // editorComponent: PlateStatic, // 可选:默认为 PlateStatic
    props: { className: 'prose max-w-none' }, // 示例:向根 div 传递属性
  });

  return html;
}

// 使用示例:
// const mySlateValue = [ { type: 'h1', children: [{ text: '我的文档' }] } ];
// getDocumentAsHtml(mySlateValue).then(console.log);

有关更多详细信息,请参阅 HTML 序列化指南

API 参考

<PlateStatic> 属性

import type React from 'react';
import type { Descendant } from 'slate';
import type { PlateEditor } from 'platejs/core'; // 根据您的设置调整导入

interface PlateStaticProps extends React.HTMLAttributes<HTMLDivElement> {
  /**
   * Plate 编辑器实例,通过 `createSlateEditor` 创建。
   * 必须包含与要渲染的内容相关的插件和组件。
   */
  editor: PlateEditor;

  /**
   * 可选的 Plate `Value`(`Descendant` 节点数组)。
   * 如果提供,将用于渲染而不是 `editor.children`。
   */
  value?: Descendant[];

  /** 根 `div` 元素的内联 CSS 样式。 */
  style?: React.CSSProperties;

  // 其他 HTMLDivElement 属性如 `className`、`id` 等也受支持。
}
  • editor:使用 createSlateEditor 创建的 PlateEditor 实例,包括组件配置。
  • value:可选。如果提供,此 Descendant 节点数组将被渲染,覆盖当前在 editor.children 中的内容。

下一步

║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
║
╚══════════════════════════════════════════════════════════════════════════════════════════════╝

← Root | ↑ Up