从零开始-文件资源管理器-24-Next.js + antd 暗黑|明亮主题切换

当前主题默认为 "dard" 暗黑模式。这次将支持"暗黑模式"与"明亮模式"之间的切换。

开发

创建一个"主题"上下文文件

explorer/src/components/change-theme/change-theme-context.tsx

typescript 复制代码
'use client'
import createCtx from '@/lib/create-ctx'
import React from 'react'

export const ChangeThemeContext = createCtx<string>('')

export const ChangeThemeProvider: React.FC<React.ProviderProps<string>> = ({ value, children }) => {
  return <ChangeThemeContext.ContextProvider value={value}>{children}</ChangeThemeContext.ContextProvider>
}

再创建一个内联 cookie 的文件

explorer/src/components/change-theme/inject-cookie.tsx

javascript 复制代码
import { ChangeThemeProvider } from '@/components/change-theme/change-theme-context'
import React from 'react'
import { cookies } from 'next/headers'

const InjectCookieChangeThemeProvider: React.FC<React.PropsWithChildren> = ({ children }) => {
  const theme = cookies().get('theme')?.value || 'dark'

  return <ChangeThemeProvider value={theme}>{children}</ChangeThemeProvider>
}

export default InjectCookieChangeThemeProvider

为 SSR 渲染时获取 cookie 内的 theme 的值,保持服务端与客户端渲染一致。避免主题切换闪屏。

该主题的改变会影响所有的组件渲染。所以将 InjectCookieChangeThemeProvider 内联 cookie 的组件插入最顶层的 layout 组件内。

explorer/src/app/layout.tsx

javascript 复制代码
...
import InjectCookieChangeThemeProvider from '@/components/change-theme/inject-cookie'
...

const RootLayout: React.FC<React.PropsWithChildren> = ({ children }) => (
  <html lang="en">
    <body className={inter.className}>
      <InjectCookieChangeThemeProvider>
        ...
      </InjectCookieChangeThemeProvider>
    </body>
  </html>
)
export default RootLayout

antd 的 ConfigProvider 上下文通过 ChangeThemeContext 上下文获取当前主题。

explorer/src/lib/antd-registry.tsx

javascript 复制代码
'use client'
...
import { ChangeThemeContext } from '@/components/change-theme/change-theme-context'
...

const AntdConfigProvider: React.FC<React.PropsWithChildren> = ({ children }) => {
  const select_theme = ChangeThemeContext.useStore()

  return (
    <ConfigProvider
      componentSize="small"
      theme={{
        algorithm: select_theme === 'dark' ? theme.darkAlgorithm : theme.defaultAlgorithm,
      }}
    >
      <Layout style={{ height: '100vh' }}>{children}</Layout>
    </ConfigProvider>
  )
}
...

创建一个改变主题的dropdown 下拉菜单,并插入页面右上角位置。

javascript 复制代码
'use client'
import React from 'react'
import { Button, Dropdown } from 'antd'
import { DarkIcon, LightIcon } from '@/components/icon/theme'
import { ChangeThemeContext } from '@/components/change-theme/change-theme-context'
import { useCookieState } from 'ahooks'

const ChangeThemeDropdown: React.FC = () => {
  const theme = ChangeThemeContext.useStore()
  const changeThemeDispatch = ChangeThemeContext.useDispatch()
  const [, changeCookies] = useCookieState('theme')

  return (
    <Dropdown
      menu={{
        selectedKeys: [theme],
        onClick: (item) => {
          changeThemeDispatch(item.key)
          changeCookies(item.key)
        },
        items: [
          { key: 'light', icon: <LightIcon />, label: '明亮模式' },
          { key: 'dark', icon: <DarkIcon />, label: '暗黑模式' },
        ],
      }}
    >
      <Button type="text" icon={theme === 'dark' ? <DarkIcon /> : <LightIcon />} />
    </Dropdown>
  )
}

export default ChangeThemeDropdown

当点击下拉菜单项时改变 ChangeThemeContext 上下文的值,并将当前改变值写入 cookie 内。

效果

到此,完成了改变主题的功能,通过将主题状态写入 cookie 内,保证服务端与客户端的渲染一致。

git-repo

yangWs29/share-explorer

相关推荐
excel6 分钟前
webpack 核心编译器 十四 节
前端
excel13 分钟前
webpack 核心编译器 十三 节
前端
腾讯TNTWeb前端团队7 小时前
helux v5 发布了,像pinia一样优雅地管理你的react状态吧
前端·javascript·react.js
范文杰11 小时前
AI 时代如何更高效开发前端组件?21st.dev 给了一种答案
前端·ai编程
拉不动的猪11 小时前
刷刷题50(常见的js数据通信与渲染问题)
前端·javascript·面试
拉不动的猪11 小时前
JS多线程Webworks中的几种实战场景演示
前端·javascript·面试
FreeCultureBoy11 小时前
macOS 命令行 原生挂载 webdav 方法
前端
uhakadotcom12 小时前
Astro 框架:快速构建内容驱动型网站的利器
前端·javascript·面试
uhakadotcom12 小时前
了解Nest.js和Next.js:如何选择合适的框架
前端·javascript·面试
uhakadotcom12 小时前
React与Next.js:基础知识及应用场景
前端·面试·github