静态渲染

一个最小化、记忆化、只读版本的 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 流式内容。

交互式 vs. 静态

对于交互式只读功能(如评论弹出框或选择),请在浏览器中使用标准 <Plate> 组件。对于纯服务器渲染的、非交互式的内容,<PlateStatic> 是推荐的选择。

Kit 使用

安装

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

添加 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} />;
}

示例

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

手动使用

创建 Slate 编辑器

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

lib/plate-static-editor.ts
import { createSlateEditor } from 'platejs';
// 导入所需的基础插件(例如 BaseHeadingPlugin、MarkdownPlugin)
// 确保在服务器环境中不要从 /react 子路径导入。
 
const editor = createSlateEditor({
  plugins: [
    // 在此添加您的基础插件列表
    // 示例:BaseHeadingPlugin, MarkdownPlugin.configure({...})
  ],
  value: [ // 示例初始值
    {
      type: 'p',
      children: [{ text: '来自静态 Plate 编辑器的问候!' }],
    },
  ],
});

定义静态节点组件

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

components/ui/paragraph-node-static.tsx
import React from 'react';
import type { SlateElementProps } from 'platejs';
 
export function ParagraphElementStatic(props: SlateElementProps) {
  return (
    <SlateElement {...props}>
      {props.children}
    </SlateElement>
  );
}

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

映射插件键到静态组件

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

components/static-components.ts
import { ParagraphElementStatic } from './ui/paragraph-node-static';
import { HeadingElementStatic } from './ui/heading-node-static';
// ... 导入其他静态组件
 
export const staticComponents = {
  p: ParagraphElementStatic,
  h1: HeadingElementStatic,
  // ... 为所有元素和叶子类型添加映射
};

渲染 <PlateStatic>

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

app/my-static-page/page.tsx (RSC 示例)
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"
    />
  );
}

值覆盖

如果您直接向 <PlateStatic> 传递 value 属性,它将覆盖 editor.children

<PlateStatic
  editor={editor}
  value={[
    { type: 'p', children: [{ text: '覆盖的内容。' }] }
  ]}
/>

记忆化详情

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

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

客户端替代方案:PlateView

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

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

app/document/page.tsx
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 的客户端组件

app/document/interactive-viewer.tsx
'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,提供事件处理功能

服务器组件兼容性

PlateView 不能在服务器组件中使用。如果您从服务器组件向客户端组件传递编辑器,将遇到序列化错误。在服务器端使用 PlateStatic,或在客户端使用 usePlateViewEditor 创建编辑器。

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> 可以直接在服务器组件中使用:

app/preview/page.tsx (RSC)
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>

lib/html-serializer.ts
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 中的内容。

下一步