From 746c64bd52af3eb967f9d407d2891be7ebec52eb Mon Sep 17 00:00:00 2001 From: sebastienlorber Date: Fri, 22 Apr 2022 19:05:08 +0200 Subject: [PATCH 1/6] Handle custom- navbar item validation --- .../src/__tests__/validateThemeConfig.test.ts | 38 +++++++++++++++++++ .../src/validateThemeConfig.ts | 27 +++++++++++-- website/docusaurus.config.js | 6 +++ 3 files changed, 67 insertions(+), 4 deletions(-) diff --git a/packages/docusaurus-theme-classic/src/__tests__/validateThemeConfig.test.ts b/packages/docusaurus-theme-classic/src/__tests__/validateThemeConfig.test.ts index 7ac6a7eda81e..ac986bd0dfda 100644 --- a/packages/docusaurus-theme-classic/src/__tests__/validateThemeConfig.test.ts +++ b/packages/docusaurus-theme-classic/src/__tests__/validateThemeConfig.test.ts @@ -227,6 +227,44 @@ describe('themeConfig', () => { }); }); + it('accept "custom-" prefixed custom navbar item type', () => { + const config = { + navbar: { + items: [ + { + type: 'custom-x', + position: 'left', + xyz: 42, + }, + { + label: 'Dropdown with custom item', + position: 'right', + items: [ + { + label: 'Facebook', + href: 'https://.facebook.com/', + target: '_self', + }, + { + type: 'custom-y', + any: new Date(), + prop: 42, + isAccepted: true, + }, + ], + }, + ], + }, + }; + expect(testValidateThemeConfig(config)).toEqual({ + ...DEFAULT_CONFIG, + navbar: { + ...DEFAULT_CONFIG.navbar, + ...config.navbar, + }, + }); + }); + it('rejects unknown navbar item type', () => { const config = { navbar: { diff --git a/packages/docusaurus-theme-classic/src/validateThemeConfig.ts b/packages/docusaurus-theme-classic/src/validateThemeConfig.ts index 9a5cd826cc43..cb77452daf2e 100644 --- a/packages/docusaurus-theme-classic/src/validateThemeConfig.ts +++ b/packages/docusaurus-theme-classic/src/validateThemeConfig.ts @@ -99,11 +99,22 @@ const HtmlNavbarItemSchema = Joi.object({ value: Joi.string().required(), }); -const itemWithType = (type: string | undefined) => { +// A temporary workaround to allow users to add custom navbar items +// See https://github.com/facebook/docusaurus/issues/7227 +const CustomNavbarItemRegexp = /custom-.*/; +const CustomNavbarItemSchema = Joi.object({ + type: Joi.string().regex(CustomNavbarItemRegexp).required(), +}).unknown(); + +const itemWithType = (type: string | RegExp | undefined) => { // Because equal(undefined) is not supported :/ - const typeSchema = type - ? Joi.string().required().equal(type) - : Joi.string().forbidden(); + const typeSchema = + // eslint-disable-next-line no-nested-ternary + type instanceof RegExp + ? Joi.string().required().regex(type) + : type + ? Joi.string().required().equal(type) + : Joi.string().forbidden(); return Joi.object({ type: typeSchema, }) @@ -135,6 +146,10 @@ const DropdownSubitemSchema = Joi.object({ is: itemWithType('html'), then: HtmlNavbarItemSchema, }, + { + is: itemWithType(CustomNavbarItemRegexp), + then: CustomNavbarItemSchema, + }, { is: Joi.alternatives().try( itemWithType('dropdown'), @@ -210,6 +225,10 @@ const NavbarItemSchema = Joi.object({ is: itemWithType('html'), then: HtmlNavbarItemSchema, }, + { + is: itemWithType(CustomNavbarItemRegexp), + then: CustomNavbarItemSchema, + }, { is: itemWithType(undefined), then: Joi.object().when('.', { diff --git a/website/docusaurus.config.js b/website/docusaurus.config.js index df57e09ac88c..b8b8f8d4a098 100644 --- a/website/docusaurus.config.js +++ b/website/docusaurus.config.js @@ -409,12 +409,18 @@ const config = { position: 'left', activeBaseRegex: `/community/`, }, + // This item links to a draft doc: only displayed in dev { type: 'doc', docId: 'test-draft', label: 'Tests', docsPluginId: 'docs-tests', }, + // Custom item for dogfooding: only displayed in /tests/ routes + { + type: 'custom-dogfood-navbar-item', + content: '😉', + }, // Right { type: 'docsVersionDropdown', From 1d18f9edf40d880f136dc6273cb8525b3874402c Mon Sep 17 00:00:00 2001 From: sebastienlorber Date: Fri, 22 Apr 2022 19:54:28 +0200 Subject: [PATCH 2/6] Extract NavbarItem/ComponentTypes + add dogfood example --- .../src/theme-classic.d.ts | 29 ++++++++++++ .../src/theme/NavbarItem/ComponentTypes.tsx | 32 +++++++++++++ .../src/theme/NavbarItem/index.tsx | 47 ++++--------------- .../src/theme/NavbarItem/ComponentTypes.tsx | 39 +++++++++++++++ 4 files changed, 108 insertions(+), 39 deletions(-) create mode 100644 packages/docusaurus-theme-classic/src/theme/NavbarItem/ComponentTypes.tsx create mode 100644 website/src/theme/NavbarItem/ComponentTypes.tsx diff --git a/packages/docusaurus-theme-classic/src/theme-classic.d.ts b/packages/docusaurus-theme-classic/src/theme-classic.d.ts index eec106ba5e1b..814b73cc502f 100644 --- a/packages/docusaurus-theme-classic/src/theme-classic.d.ts +++ b/packages/docusaurus-theme-classic/src/theme-classic.d.ts @@ -893,6 +893,35 @@ declare module '@theme/NavbarItem/HtmlNavbarItem' { export default function HtmlNavbarItem(props: Props): JSX.Element; } +declare module '@theme/NavbarItem/ComponentTypes' { + import type DefaultNavbarItem from '@theme/NavbarItem/DefaultNavbarItem'; + import type DropdownNavbarItem from '@theme/NavbarItem/DropdownNavbarItem'; + import type LocaleDropdownNavbarItem from '@theme/NavbarItem/LocaleDropdownNavbarItem'; + import type SearchNavbarItem from '@theme/NavbarItem/SearchNavbarItem'; + import type HtmlNavbarItem from '@theme/NavbarItem/HtmlNavbarItem'; + import type DocNavbarItem from '@theme/NavbarItem/DocNavbarItem'; + import type DocSidebarNavbarItem from '@theme/NavbarItem/DocSidebarNavbarItem'; + import type DocsVersionNavbarItem from '@theme/NavbarItem/DocsVersionNavbarItem'; + import type DocsVersionDropdownNavbarItem from '@theme/NavbarItem/DocsVersionDropdownNavbarItem'; + + export type ComponentTypesObject = { + readonly default: typeof DefaultNavbarItem; + readonly localeDropdown: LocaleDropdownNavbarItem; + readonly search: SearchNavbarItem; + readonly dropdown: DropdownNavbarItem; + readonly html: HtmlNavbarItem; + readonly doc: DocNavbarItem; + readonly docSidebar: DocSidebarNavbarItem; + readonly docsVersion: DocsVersionNavbarItem; + readonly docsVersionDropdown: DocsVersionDropdownNavbarItem; + // eslint-disable-next-line @typescript-eslint/no-explicit-any + [customComponentType: string]: ComponentType; + }; + + const ComponentTypes: ComponentTypesObject; + export default ComponentTypes; +} + declare module '@theme/NavbarItem' { import type {ComponentProps} from 'react'; import type {Props as DefaultNavbarItemProps} from '@theme/NavbarItem/DefaultNavbarItem'; diff --git a/packages/docusaurus-theme-classic/src/theme/NavbarItem/ComponentTypes.tsx b/packages/docusaurus-theme-classic/src/theme/NavbarItem/ComponentTypes.tsx new file mode 100644 index 000000000000..a06832505558 --- /dev/null +++ b/packages/docusaurus-theme-classic/src/theme/NavbarItem/ComponentTypes.tsx @@ -0,0 +1,32 @@ +/** + * Copyright (c) Facebook, Inc. and its affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + */ + +import DefaultNavbarItem from '@theme/NavbarItem/DefaultNavbarItem'; +import DropdownNavbarItem from '@theme/NavbarItem/DropdownNavbarItem'; +import LocaleDropdownNavbarItem from '@theme/NavbarItem/LocaleDropdownNavbarItem'; +import SearchNavbarItem from '@theme/NavbarItem/SearchNavbarItem'; +import HtmlNavbarItem from '@theme/NavbarItem/HtmlNavbarItem'; +import DocNavbarItem from '@theme/NavbarItem/DocNavbarItem'; +import DocSidebarNavbarItem from '@theme/NavbarItem/DocSidebarNavbarItem'; +import DocsVersionNavbarItem from '@theme/NavbarItem/DocsVersionNavbarItem'; +import DocsVersionDropdownNavbarItem from '@theme/NavbarItem/DocsVersionDropdownNavbarItem'; + +import type {ComponentTypesObject} from '@theme/NavbarItem/ComponentTypes'; + +const ComponentTypes: ComponentTypesObject = { + default: DefaultNavbarItem, + localeDropdown: LocaleDropdownNavbarItem, + search: SearchNavbarItem, + dropdown: DropdownNavbarItem, + html: HtmlNavbarItem, + doc: DocNavbarItem, + docSidebar: DocSidebarNavbarItem, + docsVersion: DocsVersionNavbarItem, + docsVersionDropdown: DocsVersionDropdownNavbarItem, +}; + +export default ComponentTypes; diff --git a/packages/docusaurus-theme-classic/src/theme/NavbarItem/index.tsx b/packages/docusaurus-theme-classic/src/theme/NavbarItem/index.tsx index 7c3c9b07b56f..101fb7fea953 100644 --- a/packages/docusaurus-theme-classic/src/theme/NavbarItem/index.tsx +++ b/packages/docusaurus-theme-classic/src/theme/NavbarItem/index.tsx @@ -6,57 +6,26 @@ */ import React from 'react'; -import DefaultNavbarItem from '@theme/NavbarItem/DefaultNavbarItem'; -import DropdownNavbarItem, { - type Props as DropdownNavbarItemProps, -} from '@theme/NavbarItem/DropdownNavbarItem'; -import LocaleDropdownNavbarItem from '@theme/NavbarItem/LocaleDropdownNavbarItem'; -import SearchNavbarItem from '@theme/NavbarItem/SearchNavbarItem'; -import HtmlNavbarItem from '@theme/NavbarItem/HtmlNavbarItem'; +import {type Props as DropdownNavbarItemProps} from '@theme/NavbarItem/DropdownNavbarItem'; import type {Types, Props} from '@theme/NavbarItem'; -const NavbarItemComponents: { - // Not really worth typing, as we pass all props down immediately - // eslint-disable-next-line @typescript-eslint/no-explicit-any - [type in Exclude]: () => (props: any) => JSX.Element; -} = { - default: () => DefaultNavbarItem, - localeDropdown: () => LocaleDropdownNavbarItem, - search: () => SearchNavbarItem, - dropdown: () => DropdownNavbarItem, - html: () => HtmlNavbarItem, +import ComponentTypes from '@theme/NavbarItem/ComponentTypes'; - // Need to lazy load these items as we don't know for sure the docs plugin is - // loaded. See https://github.com/facebook/docusaurus/issues/3360 - /* eslint-disable @typescript-eslint/no-var-requires, global-require */ - docsVersion: () => require('@theme/NavbarItem/DocsVersionNavbarItem').default, - docsVersionDropdown: () => - require('@theme/NavbarItem/DocsVersionDropdownNavbarItem').default, - doc: () => require('@theme/NavbarItem/DocNavbarItem').default, - docSidebar: () => require('@theme/NavbarItem/DocSidebarNavbarItem').default, - /* eslint-enable @typescript-eslint/no-var-requires, global-require */ -} as const; - -type NavbarItemComponentType = keyof typeof NavbarItemComponents; - -const getNavbarItemComponent = (type: NavbarItemComponentType) => { - const navbarItemComponentFn = NavbarItemComponents[type]; - if (!navbarItemComponentFn) { +const getNavbarItemComponent = (type: NonNullable) => { + const component = ComponentTypes[type]; + if (!component) { throw new Error(`No NavbarItem component found for type "${type}".`); } - return navbarItemComponentFn(); + return component; }; -function getComponentType( - type: Types, - isDropdown: boolean, -): NavbarItemComponentType { +function getComponentType(type: Types, isDropdown: boolean) { // Backward compatibility: navbar item with no type set // but containing dropdown items should use the type "dropdown" if (!type || type === 'default') { return isDropdown ? 'dropdown' : 'default'; } - return type as NavbarItemComponentType; + return type as NonNullable; } export default function NavbarItem({type, ...props}: Props): JSX.Element { diff --git a/website/src/theme/NavbarItem/ComponentTypes.tsx b/website/src/theme/NavbarItem/ComponentTypes.tsx new file mode 100644 index 000000000000..d800ba11f80f --- /dev/null +++ b/website/src/theme/NavbarItem/ComponentTypes.tsx @@ -0,0 +1,39 @@ +/** + * Copyright (c) Facebook, Inc. and its affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + */ + +import React from 'react'; +import ComponentTypes from '@theme-original/NavbarItem/ComponentTypes'; +import {useLocation} from '@docusaurus/router'; + +// used to dogfood custom navbar elements are possible +// see https://github.com/facebook/docusaurus/issues/7227 +function CustomDogfoodNavbarItem(props: { + content: string; + mobile?: boolean; +}): JSX.Element | null { + const {pathname} = useLocation(); + const shouldRender = pathname.includes('/tests/'); + if (!shouldRender) { + return null; + } + return ( + + ); +} + +export default { + ...ComponentTypes, + 'custom-dogfood-navbar-item': CustomDogfoodNavbarItem, +}; From aadc0b583c210137619951dc7200adfe8eb3bb01 Mon Sep 17 00:00:00 2001 From: sebastienlorber Date: Fri, 22 Apr 2022 20:07:16 +0200 Subject: [PATCH 3/6] extract CustomDogfoodNavbarItem --- .../src/theme-classic.d.ts | 18 +++++----- .../NavbarItems/CustomDogfoodNavbarItem.tsx | 33 +++++++++++++++++++ .../src/theme/NavbarItem/ComponentTypes.tsx | 27 +-------------- 3 files changed, 44 insertions(+), 34 deletions(-) create mode 100644 website/src/components/NavbarItems/CustomDogfoodNavbarItem.tsx diff --git a/packages/docusaurus-theme-classic/src/theme-classic.d.ts b/packages/docusaurus-theme-classic/src/theme-classic.d.ts index 814b73cc502f..a16f02a0933a 100644 --- a/packages/docusaurus-theme-classic/src/theme-classic.d.ts +++ b/packages/docusaurus-theme-classic/src/theme-classic.d.ts @@ -894,6 +894,8 @@ declare module '@theme/NavbarItem/HtmlNavbarItem' { } declare module '@theme/NavbarItem/ComponentTypes' { + import type {ComponentType} from 'react'; + import type DefaultNavbarItem from '@theme/NavbarItem/DefaultNavbarItem'; import type DropdownNavbarItem from '@theme/NavbarItem/DropdownNavbarItem'; import type LocaleDropdownNavbarItem from '@theme/NavbarItem/LocaleDropdownNavbarItem'; @@ -906,14 +908,14 @@ declare module '@theme/NavbarItem/ComponentTypes' { export type ComponentTypesObject = { readonly default: typeof DefaultNavbarItem; - readonly localeDropdown: LocaleDropdownNavbarItem; - readonly search: SearchNavbarItem; - readonly dropdown: DropdownNavbarItem; - readonly html: HtmlNavbarItem; - readonly doc: DocNavbarItem; - readonly docSidebar: DocSidebarNavbarItem; - readonly docsVersion: DocsVersionNavbarItem; - readonly docsVersionDropdown: DocsVersionDropdownNavbarItem; + readonly localeDropdown: typeof LocaleDropdownNavbarItem; + readonly search: typeof SearchNavbarItem; + readonly dropdown: typeof DropdownNavbarItem; + readonly html: typeof HtmlNavbarItem; + readonly doc: typeof DocNavbarItem; + readonly docSidebar: typeof DocSidebarNavbarItem; + readonly docsVersion: typeof DocsVersionNavbarItem; + readonly docsVersionDropdown: typeof DocsVersionDropdownNavbarItem; // eslint-disable-next-line @typescript-eslint/no-explicit-any [customComponentType: string]: ComponentType; }; diff --git a/website/src/components/NavbarItems/CustomDogfoodNavbarItem.tsx b/website/src/components/NavbarItems/CustomDogfoodNavbarItem.tsx new file mode 100644 index 000000000000..71eb6a2a6a07 --- /dev/null +++ b/website/src/components/NavbarItems/CustomDogfoodNavbarItem.tsx @@ -0,0 +1,33 @@ +/** + * Copyright (c) Facebook, Inc. and its affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + */ + +import React from 'react'; +import {useLocation} from '@docusaurus/router'; + +// used to dogfood custom navbar elements are possible +// see https://github.com/facebook/docusaurus/issues/7227 +export default function CustomDogfoodNavbarItem(props: { + content: string; + mobile?: boolean; +}): JSX.Element | null { + const {pathname} = useLocation(); + const shouldRender = pathname.includes('/tests/'); + if (!shouldRender) { + return null; + } + return ( + + ); +} diff --git a/website/src/theme/NavbarItem/ComponentTypes.tsx b/website/src/theme/NavbarItem/ComponentTypes.tsx index d800ba11f80f..85937ca85adf 100644 --- a/website/src/theme/NavbarItem/ComponentTypes.tsx +++ b/website/src/theme/NavbarItem/ComponentTypes.tsx @@ -5,33 +5,8 @@ * LICENSE file in the root directory of this source tree. */ -import React from 'react'; import ComponentTypes from '@theme-original/NavbarItem/ComponentTypes'; -import {useLocation} from '@docusaurus/router'; - -// used to dogfood custom navbar elements are possible -// see https://github.com/facebook/docusaurus/issues/7227 -function CustomDogfoodNavbarItem(props: { - content: string; - mobile?: boolean; -}): JSX.Element | null { - const {pathname} = useLocation(); - const shouldRender = pathname.includes('/tests/'); - if (!shouldRender) { - return null; - } - return ( - - ); -} +import CustomDogfoodNavbarItem from '@site/src/components/NavbarItems/CustomDogfoodNavbarItem'; export default { ...ComponentTypes, From 70ecad4e35536751f9a77da4f98fe2a8e1070657 Mon Sep 17 00:00:00 2001 From: sebastienlorber Date: Wed, 27 Apr 2022 18:17:51 +0200 Subject: [PATCH 4/6] add TS ignore --- packages/docusaurus-theme-classic/src/theme/NavbarItem/index.tsx | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/docusaurus-theme-classic/src/theme/NavbarItem/index.tsx b/packages/docusaurus-theme-classic/src/theme/NavbarItem/index.tsx index 101fb7fea953..d8e109fa48d5 100644 --- a/packages/docusaurus-theme-classic/src/theme/NavbarItem/index.tsx +++ b/packages/docusaurus-theme-classic/src/theme/NavbarItem/index.tsx @@ -34,5 +34,6 @@ export default function NavbarItem({type, ...props}: Props): JSX.Element { (props as DropdownNavbarItemProps).items !== undefined, ); const NavbarItemComponent = getNavbarItemComponent(componentType); + // @ts-expect-error: how to type this? return ; } From e29be37ef5e74a4b062ef16bcac47c0507860533 Mon Sep 17 00:00:00 2001 From: sebastienlorber Date: Fri, 29 Apr 2022 10:51:14 +0200 Subject: [PATCH 5/6] add swizzle config for NavbarItem/ComponentTypes --- packages/docusaurus-theme-classic/src/getSwizzleConfig.ts | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/packages/docusaurus-theme-classic/src/getSwizzleConfig.ts b/packages/docusaurus-theme-classic/src/getSwizzleConfig.ts index 9e73ef9a17e4..0c34618bdc32 100644 --- a/packages/docusaurus-theme-classic/src/getSwizzleConfig.ts +++ b/packages/docusaurus-theme-classic/src/getSwizzleConfig.ts @@ -206,6 +206,14 @@ export default function getSwizzleConfig(): SwizzleConfig { description: 'A component wrapping all MDX content and providing the MDXComponents to the MDX context', }, + 'NavbarItem/ComponentTypes': { + actions: { + eject: 'safe', + wrap: 'forbidden', + }, + description: + 'The Navbar item components mapping. Can be ejected to add custom navbar item types. See https://github.com/facebook/docusaurus/issues/7227.', + }, // TODO should probably not even appear here 'NavbarItem/utils': { actions: { From 2a05d0b663a2283b49ca5a26e670c69bee826863 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9bastien=20Lorber?= Date: Fri, 29 Apr 2022 11:10:25 +0200 Subject: [PATCH 6/6] Update packages/docusaurus-theme-classic/src/theme/NavbarItem/index.tsx Co-authored-by: Joshua Chen --- .../docusaurus-theme-classic/src/theme/NavbarItem/index.tsx | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/packages/docusaurus-theme-classic/src/theme/NavbarItem/index.tsx b/packages/docusaurus-theme-classic/src/theme/NavbarItem/index.tsx index d8e109fa48d5..7ae0d4990730 100644 --- a/packages/docusaurus-theme-classic/src/theme/NavbarItem/index.tsx +++ b/packages/docusaurus-theme-classic/src/theme/NavbarItem/index.tsx @@ -34,6 +34,5 @@ export default function NavbarItem({type, ...props}: Props): JSX.Element { (props as DropdownNavbarItemProps).items !== undefined, ); const NavbarItemComponent = getNavbarItemComponent(componentType); - // @ts-expect-error: how to type this? - return ; + return ; }