UNPKG

dumi-theme-antd

Version:

Ant Design 5.0 官网风格类似的 dumi2 主题插件

193 lines 27 kB
import _extends from "@babel/runtime/helpers/esm/extends"; import _objectSpread from "@babel/runtime/helpers/esm/objectSpread2"; import _slicedToArray from "@babel/runtime/helpers/esm/slicedToArray"; import _defineProperty from "@babel/runtime/helpers/esm/defineProperty"; /* eslint-disable import/no-unresolved */ import { MenuOutlined } from '@ant-design/icons'; import { ClassNames, css } from '@emotion/react'; import { Col, Popover, Row, Select } from 'antd'; import classNames from 'classnames'; import { useLocale, useLocation } from 'dumi'; import DumiSearchBar from 'dumi/theme-default/slots/SearchBar'; import React, { useCallback, useContext, useEffect, useState } from 'react'; import LangSwitch from "dumi/theme/slots/LangSwitch"; import RtlSwitch from "dumi/theme/slots/RtlSwitch"; import useAdditionalThemeConfig from "../../hooks/useAdditionalThemeConfig"; import useSiteToken from "../../hooks/useSiteToken"; import SiteContext from "dumi/theme/slots/SiteContext"; import HeaderExtra from "./HeaderExtral"; import Logo from "./Logo"; import More from "./More"; import Navigation from "./Navigation"; import { jsx as ___EmotionJSX } from "@emotion/react"; var RESPONSIVE_XS = 1120; var RESPONSIVE_SM = 1200; var colPropsHome = [{ flex: 'none' }, { flex: 'auto' }]; var _colProps = [{ xxl: 4, xl: 5, lg: 6, md: 6, sm: 24, xs: 24 }, { xxl: 20, xl: 19, lg: 18, md: 18, sm: 0, xs: 0 }]; var useStyle = function useStyle() { var _useSiteToken = useSiteToken(), token = _useSiteToken.token; var searchIconColor = '#ced4d9'; return { header: /*#__PURE__*/css("position:sticky;top:0;z-index:99;max-width:100%;background:", token.colorBgContainer, ";box-shadow:", token.boxShadowTertiary, ";@media only screen and (max-width: ", token.mobileMaxWidth, "px){text-align:center;}.nav-search-wrapper{display:flex;flex:auto;}.dumi-default-search-bar{border-inline-start:1px solid rgba(0, 0, 0, 0.06);>svg{width:14px;fill:", searchIconColor, ";}>input{height:22px;border:0;&:focus{box-shadow:none;}&::placeholder{color:", searchIconColor, ";}}.dumi-default-search-shortcut{color:", searchIconColor, ";background-color:rgba(150, 150, 150, 0.06);border-color:rgba(100, 100, 100, 0.2);border-radius:4px;}.dumi-default-search-popover{inset-inline-start:11px;inset-inline-end:unset;&::before{inset-inline-start:100px;inset-inline-end:unset;}}}" + (process.env.NODE_ENV === "production" ? "" : ";label:header;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["index.tsx"],"names":[],"mappings":"AA4De","file":"index.tsx","sourcesContent":["/* eslint-disable import/no-unresolved */\nimport { MenuOutlined } from '@ant-design/icons';\nimport { ClassNames, css } from '@emotion/react';\nimport { Col, Popover, Row, Select } from 'antd';\nimport classNames from 'classnames';\nimport { useLocale, useLocation } from 'dumi';\nimport DumiSearchBar from 'dumi/theme-default/slots/SearchBar';\nimport React, { useCallback, useContext, useEffect, useState, type FC } from 'react';\nimport LangSwitch from 'dumi/theme/slots/LangSwitch';\nimport RtlSwitch from 'dumi/theme/slots/RtlSwitch';\nimport useAdditionalThemeConfig from '../../hooks/useAdditionalThemeConfig';\nimport useSiteToken from '../../hooks/useSiteToken';\nimport type { SiteContextProps } from '../SiteContext';\nimport SiteContext from '../SiteContext';\nimport HeaderExtra from './HeaderExtral';\nimport Logo from './Logo';\nimport More from './More';\nimport Navigation from './Navigation';\n\ninterface HeaderState {\n  windowWidth: number;\n  menuVisible: boolean;\n}\nexport type IResponsive = null | 'narrow' | 'crowded';\n\nconst RESPONSIVE_XS = 1120;\nconst RESPONSIVE_SM = 1200;\n\nconst colPropsHome = [\n  {\n    flex: 'none'\n  },\n  {\n    flex: 'auto'\n  }\n];\nconst _colProps = [\n  {\n    xxl: 4,\n    xl: 5,\n    lg: 6,\n    md: 6,\n    sm: 24,\n    xs: 24\n  },\n  {\n    xxl: 20,\n    xl: 19,\n    lg: 18,\n    md: 18,\n    sm: 0,\n    xs: 0\n  }\n];\n\nconst useStyle = () => {\n  const { token } = useSiteToken();\n  const searchIconColor = '#ced4d9';\n\n  return {\n    header: css`\n      position: sticky;\n      top: 0;\n      z-index: 99;\n      max-width: 100%;\n      background: ${token.colorBgContainer};\n      box-shadow: ${token.boxShadowTertiary};\n\n      @media only screen and (max-width: ${token.mobileMaxWidth}px) {\n        text-align: center;\n      }\n\n      .nav-search-wrapper {\n        display: flex;\n        flex: auto;\n      }\n\n      .dumi-default-search-bar {\n        border-inline-start: 1px solid rgba(0, 0, 0, 0.06);\n\n        > svg {\n          width: 14px;\n          fill: ${searchIconColor};\n        }\n\n        > input {\n          height: 22px;\n          border: 0;\n\n          &:focus {\n            box-shadow: none;\n          }\n\n          &::placeholder {\n            color: ${searchIconColor};\n          }\n        }\n\n        .dumi-default-search-shortcut {\n          color: ${searchIconColor};\n          background-color: rgba(150, 150, 150, 0.06);\n          border-color: rgba(100, 100, 100, 0.2);\n          border-radius: 4px;\n        }\n\n        .dumi-default-search-popover {\n          inset-inline-start: 11px;\n          inset-inline-end: unset;\n\n          &::before {\n            inset-inline-start: 100px;\n            inset-inline-end: unset;\n          }\n        }\n      }\n    `,\n    menuRow: css`\n      display: flex;\n      align-items: center;\n      margin: 0;\n\n      > * {\n        flex: none;\n        margin: 0 12px 0 0;\n\n        &:last-child {\n          margin-inline-end: 40px;\n        }\n      }\n\n      ${token.antCls}-row-rtl & {\n        > * {\n          &:last-child {\n            margin-right: 12px;\n            margin-left: 40px;\n          }\n        }\n      }\n    `,\n    popoverMenu: {\n      width: 300,\n\n      [`${token.antCls}-popover-inner-content`]: {\n        padding: 0\n      }\n    }\n  };\n};\n\nconst Header: FC = () => {\n  const { isMobile } = useContext<SiteContextProps>(SiteContext);\n  const [headerState, setHeaderState] = useState<HeaderState>({\n    windowWidth: 1400,\n    menuVisible: false\n  });\n  const location = useLocation();\n  const { docVersions } = useAdditionalThemeConfig();\n\n  const onWindowResize = useCallback(() => {\n    if (typeof window === 'undefined') {\n      return;\n    }\n    setHeaderState((prev) => ({\n      ...prev,\n      windowWidth: window.innerWidth\n    }));\n  }, []);\n  const handleHideMenu = useCallback(() => {\n    setHeaderState((prev) => ({\n      ...prev,\n      menuVisible: false\n    }));\n  }, []);\n  const onMenuVisibleChange = useCallback((visible: boolean) => {\n    setHeaderState((prev) => ({\n      ...prev,\n      menuVisible: visible\n    }));\n  }, []);\n\n  const handleVersionChange = useCallback((url: string) => {\n    if (typeof window === 'undefined') {\n      return;\n    }\n    window.location.href = url;\n  }, []);\n\n  useEffect(() => {\n    handleHideMenu();\n  }, [location, handleHideMenu]);\n\n  useEffect(() => {\n    if (typeof window === 'undefined') {\n      return () => {};\n    }\n    onWindowResize();\n    window.addEventListener('resize', onWindowResize);\n    return () => {\n      window.removeEventListener('resize', onWindowResize);\n    };\n  }, [onWindowResize]);\n\n  const { pathname } = location;\n  const locale = useLocale();\n  const suffix: string = (locale as any).suffex || '';\n  const isHome = ['', `index${suffix}`].includes(pathname);\n  const { windowWidth, menuVisible } = headerState;\n  const style = useStyle();\n  const headerClassName = classNames({\n    clearfix: true,\n    'home-header': isHome\n  });\n  let responsive: IResponsive = null;\n\n  if (windowWidth < RESPONSIVE_XS) {\n    responsive = 'crowded';\n  } else if (windowWidth < RESPONSIVE_SM) {\n    responsive = 'narrow';\n  }\n\n  const navigationNode = <Navigation key=\"nav\" isMobile={isMobile} responsive={responsive} />;\n  const versionOptions = Object.keys(docVersions ?? {}).map((version) => ({\n    value: docVersions?.[version],\n    label: version\n  }));\n  let menu: (React.ReactElement | null)[] = [\n    navigationNode,\n    versionOptions.length > 0 ? (\n      <Select\n        key=\"version\"\n        size=\"small\"\n        defaultValue={versionOptions[0]?.value}\n        onChange={handleVersionChange}\n        popupMatchSelectWidth={false}\n        getPopupContainer={(trigger) => trigger.parentNode}\n        options={versionOptions}\n      />\n    ) : null,\n    <More key=\"more\" />,\n    <LangSwitch key={new Date().getTime()} />,\n    <RtlSwitch key=\"direction\" />,\n    <HeaderExtra key=\"header-Extra\" />\n  ];\n  if (windowWidth < RESPONSIVE_XS) {\n    menu = [navigationNode];\n  }\n\n  const colProps = isHome ? colPropsHome : _colProps;\n\n  return (\n    <header css={style.header} className={headerClassName}>\n      {isMobile && (\n        <ClassNames>\n          {({ css: cssFn }) => (\n            <Popover\n              overlayClassName={cssFn(style.popoverMenu)}\n              placement=\"bottomRight\"\n              content={menu}\n              trigger=\"click\"\n              open={menuVisible}\n              arrow\n              onOpenChange={onMenuVisibleChange}\n            >\n              <MenuOutlined className=\"nav-phone-icon\" />\n            </Popover>\n          )}\n        </ClassNames>\n      )}\n      <Row\n        style={{\n          height: 64\n        }}\n      >\n        <Col {...colProps[0]}>\n          <Logo />\n        </Col>\n        <Col {...colProps[1]} css={style.menuRow}>\n          <div className=\"nav-search-wrapper\">\n            <DumiSearchBar />\n          </div>\n          {!isMobile && menu}\n        </Col>\n      </Row>\n    </header>\n  );\n};\n\nexport default Header;\n"]} */"), menuRow: /*#__PURE__*/css("display:flex;align-items:center;margin:0;>*{flex:none;margin:0 12px 0 0;&:last-child{margin-inline-end:40px;}}", token.antCls, "-row-rtl &{>*{&:last-child{margin-right:12px;margin-left:40px;}}}" + (process.env.NODE_ENV === "production" ? "" : ";label:menuRow;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["index.tsx"],"names":[],"mappings":"AAoHgB","file":"index.tsx","sourcesContent":["/* eslint-disable import/no-unresolved */\nimport { MenuOutlined } from '@ant-design/icons';\nimport { ClassNames, css } from '@emotion/react';\nimport { Col, Popover, Row, Select } from 'antd';\nimport classNames from 'classnames';\nimport { useLocale, useLocation } from 'dumi';\nimport DumiSearchBar from 'dumi/theme-default/slots/SearchBar';\nimport React, { useCallback, useContext, useEffect, useState, type FC } from 'react';\nimport LangSwitch from 'dumi/theme/slots/LangSwitch';\nimport RtlSwitch from 'dumi/theme/slots/RtlSwitch';\nimport useAdditionalThemeConfig from '../../hooks/useAdditionalThemeConfig';\nimport useSiteToken from '../../hooks/useSiteToken';\nimport type { SiteContextProps } from '../SiteContext';\nimport SiteContext from '../SiteContext';\nimport HeaderExtra from './HeaderExtral';\nimport Logo from './Logo';\nimport More from './More';\nimport Navigation from './Navigation';\n\ninterface HeaderState {\n  windowWidth: number;\n  menuVisible: boolean;\n}\nexport type IResponsive = null | 'narrow' | 'crowded';\n\nconst RESPONSIVE_XS = 1120;\nconst RESPONSIVE_SM = 1200;\n\nconst colPropsHome = [\n  {\n    flex: 'none'\n  },\n  {\n    flex: 'auto'\n  }\n];\nconst _colProps = [\n  {\n    xxl: 4,\n    xl: 5,\n    lg: 6,\n    md: 6,\n    sm: 24,\n    xs: 24\n  },\n  {\n    xxl: 20,\n    xl: 19,\n    lg: 18,\n    md: 18,\n    sm: 0,\n    xs: 0\n  }\n];\n\nconst useStyle = () => {\n  const { token } = useSiteToken();\n  const searchIconColor = '#ced4d9';\n\n  return {\n    header: css`\n      position: sticky;\n      top: 0;\n      z-index: 99;\n      max-width: 100%;\n      background: ${token.colorBgContainer};\n      box-shadow: ${token.boxShadowTertiary};\n\n      @media only screen and (max-width: ${token.mobileMaxWidth}px) {\n        text-align: center;\n      }\n\n      .nav-search-wrapper {\n        display: flex;\n        flex: auto;\n      }\n\n      .dumi-default-search-bar {\n        border-inline-start: 1px solid rgba(0, 0, 0, 0.06);\n\n        > svg {\n          width: 14px;\n          fill: ${searchIconColor};\n        }\n\n        > input {\n          height: 22px;\n          border: 0;\n\n          &:focus {\n            box-shadow: none;\n          }\n\n          &::placeholder {\n            color: ${searchIconColor};\n          }\n        }\n\n        .dumi-default-search-shortcut {\n          color: ${searchIconColor};\n          background-color: rgba(150, 150, 150, 0.06);\n          border-color: rgba(100, 100, 100, 0.2);\n          border-radius: 4px;\n        }\n\n        .dumi-default-search-popover {\n          inset-inline-start: 11px;\n          inset-inline-end: unset;\n\n          &::before {\n            inset-inline-start: 100px;\n            inset-inline-end: unset;\n          }\n        }\n      }\n    `,\n    menuRow: css`\n      display: flex;\n      align-items: center;\n      margin: 0;\n\n      > * {\n        flex: none;\n        margin: 0 12px 0 0;\n\n        &:last-child {\n          margin-inline-end: 40px;\n        }\n      }\n\n      ${token.antCls}-row-rtl & {\n        > * {\n          &:last-child {\n            margin-right: 12px;\n            margin-left: 40px;\n          }\n        }\n      }\n    `,\n    popoverMenu: {\n      width: 300,\n\n      [`${token.antCls}-popover-inner-content`]: {\n        padding: 0\n      }\n    }\n  };\n};\n\nconst Header: FC = () => {\n  const { isMobile } = useContext<SiteContextProps>(SiteContext);\n  const [headerState, setHeaderState] = useState<HeaderState>({\n    windowWidth: 1400,\n    menuVisible: false\n  });\n  const location = useLocation();\n  const { docVersions } = useAdditionalThemeConfig();\n\n  const onWindowResize = useCallback(() => {\n    if (typeof window === 'undefined') {\n      return;\n    }\n    setHeaderState((prev) => ({\n      ...prev,\n      windowWidth: window.innerWidth\n    }));\n  }, []);\n  const handleHideMenu = useCallback(() => {\n    setHeaderState((prev) => ({\n      ...prev,\n      menuVisible: false\n    }));\n  }, []);\n  const onMenuVisibleChange = useCallback((visible: boolean) => {\n    setHeaderState((prev) => ({\n      ...prev,\n      menuVisible: visible\n    }));\n  }, []);\n\n  const handleVersionChange = useCallback((url: string) => {\n    if (typeof window === 'undefined') {\n      return;\n    }\n    window.location.href = url;\n  }, []);\n\n  useEffect(() => {\n    handleHideMenu();\n  }, [location, handleHideMenu]);\n\n  useEffect(() => {\n    if (typeof window === 'undefined') {\n      return () => {};\n    }\n    onWindowResize();\n    window.addEventListener('resize', onWindowResize);\n    return () => {\n      window.removeEventListener('resize', onWindowResize);\n    };\n  }, [onWindowResize]);\n\n  const { pathname } = location;\n  const locale = useLocale();\n  const suffix: string = (locale as any).suffex || '';\n  const isHome = ['', `index${suffix}`].includes(pathname);\n  const { windowWidth, menuVisible } = headerState;\n  const style = useStyle();\n  const headerClassName = classNames({\n    clearfix: true,\n    'home-header': isHome\n  });\n  let responsive: IResponsive = null;\n\n  if (windowWidth < RESPONSIVE_XS) {\n    responsive = 'crowded';\n  } else if (windowWidth < RESPONSIVE_SM) {\n    responsive = 'narrow';\n  }\n\n  const navigationNode = <Navigation key=\"nav\" isMobile={isMobile} responsive={responsive} />;\n  const versionOptions = Object.keys(docVersions ?? {}).map((version) => ({\n    value: docVersions?.[version],\n    label: version\n  }));\n  let menu: (React.ReactElement | null)[] = [\n    navigationNode,\n    versionOptions.length > 0 ? (\n      <Select\n        key=\"version\"\n        size=\"small\"\n        defaultValue={versionOptions[0]?.value}\n        onChange={handleVersionChange}\n        popupMatchSelectWidth={false}\n        getPopupContainer={(trigger) => trigger.parentNode}\n        options={versionOptions}\n      />\n    ) : null,\n    <More key=\"more\" />,\n    <LangSwitch key={new Date().getTime()} />,\n    <RtlSwitch key=\"direction\" />,\n    <HeaderExtra key=\"header-Extra\" />\n  ];\n  if (windowWidth < RESPONSIVE_XS) {\n    menu = [navigationNode];\n  }\n\n  const colProps = isHome ? colPropsHome : _colProps;\n\n  return (\n    <header css={style.header} className={headerClassName}>\n      {isMobile && (\n        <ClassNames>\n          {({ css: cssFn }) => (\n            <Popover\n              overlayClassName={cssFn(style.popoverMenu)}\n              placement=\"bottomRight\"\n              content={menu}\n              trigger=\"click\"\n              open={menuVisible}\n              arrow\n              onOpenChange={onMenuVisibleChange}\n            >\n              <MenuOutlined className=\"nav-phone-icon\" />\n            </Popover>\n          )}\n        </ClassNames>\n      )}\n      <Row\n        style={{\n          height: 64\n        }}\n      >\n        <Col {...colProps[0]}>\n          <Logo />\n        </Col>\n        <Col {...colProps[1]} css={style.menuRow}>\n          <div className=\"nav-search-wrapper\">\n            <DumiSearchBar />\n          </div>\n          {!isMobile && menu}\n        </Col>\n      </Row>\n    </header>\n  );\n};\n\nexport default Header;\n"]} */"), popoverMenu: _defineProperty({ width: 300 }, "".concat(token.antCls, "-popover-inner-content"), { padding: 0 }) }; }; var Header = function Header() { var _versionOptions$; var _useContext = useContext(SiteContext), isMobile = _useContext.isMobile; var _useState = useState({ windowWidth: 1400, menuVisible: false }), _useState2 = _slicedToArray(_useState, 2), headerState = _useState2[0], setHeaderState = _useState2[1]; var location = useLocation(); var _useAdditionalThemeCo = useAdditionalThemeConfig(), docVersions = _useAdditionalThemeCo.docVersions; var onWindowResize = useCallback(function () { if (typeof window === 'undefined') { return; } setHeaderState(function (prev) { return _objectSpread(_objectSpread({}, prev), {}, { windowWidth: window.innerWidth }); }); }, []); var handleHideMenu = useCallback(function () { setHeaderState(function (prev) { return _objectSpread(_objectSpread({}, prev), {}, { menuVisible: false }); }); }, []); var onMenuVisibleChange = useCallback(function (visible) { setHeaderState(function (prev) { return _objectSpread(_objectSpread({}, prev), {}, { menuVisible: visible }); }); }, []); var handleVersionChange = useCallback(function (url) { if (typeof window === 'undefined') { return; } window.location.href = url; }, []); useEffect(function () { handleHideMenu(); }, [location, handleHideMenu]); useEffect(function () { if (typeof window === 'undefined') { return function () {}; } onWindowResize(); window.addEventListener('resize', onWindowResize); return function () { window.removeEventListener('resize', onWindowResize); }; }, [onWindowResize]); var pathname = location.pathname; var locale = useLocale(); var suffix = locale.suffex || ''; var isHome = ['', "index".concat(suffix)].includes(pathname); var windowWidth = headerState.windowWidth, menuVisible = headerState.menuVisible; var style = useStyle(); var headerClassName = classNames({ clearfix: true, 'home-header': isHome }); var responsive = null; if (windowWidth < RESPONSIVE_XS) { responsive = 'crowded'; } else if (windowWidth < RESPONSIVE_SM) { responsive = 'narrow'; } var navigationNode = ___EmotionJSX(Navigation, { key: "nav", isMobile: isMobile, responsive: responsive }); var versionOptions = Object.keys(docVersions !== null && docVersions !== void 0 ? docVersions : {}).map(function (version) { return { value: docVersions === null || docVersions === void 0 ? void 0 : docVersions[version], label: version }; }); var menu = [navigationNode, versionOptions.length > 0 ? ___EmotionJSX(Select, { key: "version", size: "small", defaultValue: (_versionOptions$ = versionOptions[0]) === null || _versionOptions$ === void 0 ? void 0 : _versionOptions$.value, onChange: handleVersionChange, popupMatchSelectWidth: false, getPopupContainer: function getPopupContainer(trigger) { return trigger.parentNode; }, options: versionOptions }) : null, ___EmotionJSX(More, { key: "more" }), ___EmotionJSX(LangSwitch, { key: new Date().getTime() }), ___EmotionJSX(RtlSwitch, { key: "direction" }), ___EmotionJSX(HeaderExtra, { key: "header-Extra" })]; if (windowWidth < RESPONSIVE_XS) { menu = [navigationNode]; } var colProps = isHome ? colPropsHome : _colProps; return ___EmotionJSX("header", { css: style.header, className: headerClassName }, isMobile && ___EmotionJSX(ClassNames, null, function (_ref) { var cssFn = _ref.css; return ___EmotionJSX(Popover, { overlayClassName: cssFn(style.popoverMenu), placement: "bottomRight", content: menu, trigger: "click", open: menuVisible, arrow: true, onOpenChange: onMenuVisibleChange }, ___EmotionJSX(MenuOutlined, { className: "nav-phone-icon" })); }), ___EmotionJSX(Row, { style: { height: 64 } }, ___EmotionJSX(Col, colProps[0], ___EmotionJSX(Logo, null)), ___EmotionJSX(Col, _extends({}, colProps[1], { css: style.menuRow }), ___EmotionJSX("div", { className: "nav-search-wrapper" }, ___EmotionJSX(DumiSearchBar, null)), !isMobile && menu))); }; export default Header;