使用高级 Chakra UI 组件更快地构建 💎

了解更多
跳到内容
文档在线演练指南博客
赞助商

在 Iframe 中使用 Chakra UI

在 Iframe 中安装和使用 Chakra UI 的指南

Iframe 对于在独立上下文中隔离样式和逻辑非常有用。例如,你可能想在专门的沙盒中展示 Chakra 组件。

模板

使用以下模板快速开始

安装

所需的最低 Node 版本为 Node.20.x

1

安装依赖

npm i @chakra-ui/react @emotion/react @emotion/cache react-frame-component

使用的额外包有

  • react-frame-component 用于轻松创建 iframe
  • @emotion/cache 用于为样式创建自定义插入点
2

添加代码片段

代码片段是预构建的组件,可以帮助你更快地构建 UI。使用 @chakra-ui/cli 可以将代码片段添加到你的项目中。

npx @chakra-ui/cli snippet add
3

更新 tsconfig

如果你正在使用 TypeScript,需要更新 tsconfig 文件中的 compilerOptions 以包含以下选项

{
  "compilerOptions": {
    "module": "ESNext",
    "moduleResolution": "Bundler",
    "skipLibCheck": true,
    "paths": {
      "@/*": ["./src/*"]
    }
  }
}
4

设置 Iframe

创建一个 components/ui/iframe-provider.tsx 文件,使用 react-frame-component 包来设置 iframe。

components/ui/iframe-provider.tsx

import {
  ChakraProvider,
  EnvironmentProvider,
  defaultSystem,
} from "@chakra-ui/react"
import createCache from "@emotion/cache"
import { CacheProvider } from "@emotion/react"
import Iframe, { FrameContextConsumer } from "react-frame-component"

function memoize<T extends object, R>(func: (arg: T) => R): (arg: T) => R {
  const cache = new WeakMap<T, R>()
  return (arg: T) => {
    if (cache.has(arg)) return cache.get(arg)!
    const ret = func(arg)
    cache.set(arg, ret)
    return ret
  }
}

const createCacheFn = memoize((container: HTMLElement) =>
  createCache({ container, key: "frame" }),
)

export const IframeProvider = (props: React.PropsWithChildren) => {
  const { children } = props
  return (
    <Iframe>
      <FrameContextConsumer>
        {(frame) => {
          const head = frame.document?.head
          if (!head) return null
          return (
            <CacheProvider value={createCacheFn(head)}>
              <EnvironmentProvider value={() => head.ownerDocument}>
                <ChakraProvider value={defaultSystem}>
                  {children}
                </ChakraProvider>
              </EnvironmentProvider>
            </CacheProvider>
          )
        }}
      </FrameContextConsumer>
    </Iframe>
  )
}
5

设置 Provider

在应用程序的根目录,使用 components/ui/provider 组件中生成的 Provider 组件包裹你的应用程序。

此 Provider 包含以下内容

  • 来自 @chakra-ui/reactChakraProvider,用于样式系统
  • 来自 next-themesThemeProvider,用于颜色模式
import { Provider } from "@/components/ui/provider"
import React from "react"
import ReactDOM from "react-dom/client"
import App from "./App"

ReactDOM.createRoot(document.getElementById("root")!).render(
  <React.StrictMode>
    <Provider>
      <App />
    </Provider>
  </React.StrictMode>,
)
6

使用 IframeProvider

在应用程序中的任何组件,使用 IframeProvider 组件包裹它,以便在 iframe 中渲染。

src/App.tsx

import { Button, Container, Heading, Stack } from "@chakra-ui/react"
import { IframeProvider } from "./components/ui/iframe-provider"

function App() {
  return (
    <Container py="8">
      <Heading mb="5">Outside Iframe</Heading>

      <IframeProvider>
        <Stack p="6" align="flex-start" border="1px solid red">
          <Heading>Inside Iframe</Heading>
          <Button>Click me</Button>
        </Stack>
      </IframeProvider>
    </Container>
  )
}

export default App

自定义

如果你使用 createSystem 函数创建了自定义主题,请确保将其传递给 IframeProviderProvider 组件,以确保在 iframe 中使用它。

例如,假设你创建了一个自定义主题

export const system = createSystem(defaultConfig, {
  theme: { colors: {} },
})

然后,将其传递给 IframeProviderProvider 组件

<ChakraProvider value={system}>{/* ... */}</ChakraProvider>

上一页

Shadow DOM

下一页

概览