From 9775deb5090447ebed54d0ebd7f7cdca9a63bd3d Mon Sep 17 00:00:00 2001 From: hasparus Date: Fri, 18 Mar 2022 22:11:11 +0100 Subject: [PATCH] chore(components): fix themed() --- package.json | 1 + packages/mdx/src/index.tsx | 53 +++++++++++++++++++++++-------------- packages/mdx/test/index.tsx | 17 +++--------- 3 files changed, 37 insertions(+), 34 deletions(-) diff --git a/package.json b/package.json index 3859e6777..fd7876b5b 100644 --- a/package.json +++ b/package.json @@ -13,6 +13,7 @@ "typecheck:tests": "tsc --noEmit -P ./tsconfig.test.json", "logo": "yarn workspace docs logo", "postinstall": "preconstruct dev", + "dev": "preconstruct dev", "version:bump": "lerna version", "version:bump-next": "lerna version prerelease --preid alpha --exact", "release": "yarn clean && yarn build && yarn shipit", diff --git a/packages/mdx/src/index.tsx b/packages/mdx/src/index.tsx index e5f3445c5..e2ac68e73 100644 --- a/packages/mdx/src/index.tsx +++ b/packages/mdx/src/index.tsx @@ -7,7 +7,6 @@ import { ReactNode, DetailedHTMLProps, HTMLAttributes, - ElementType, ComponentProps, useMemo, } from 'react' @@ -86,21 +85,8 @@ const alias = (n: ThemedComponentName): keyof JSX.IntrinsicElements => isAlias(n) ? aliases[n] : n export const themed = - (key: ThemedComponentName | (string & {})) => - ({ theme, ...rest }: ThemedProps) => { - const extraStyles: CSSObject = {} - - if (key === 'th' || key === 'td') { - const { align } = rest as DetailedHTMLProps< - React.ThHTMLAttributes, - HTMLTableHeaderCellElement - > - - if (align !== 'char') extraStyles.textAlign = align - } - - return css({ ...get(theme, `styles.${key}`), ...extraStyles })(theme) - } + (key: ThemedComponentName | (string & {})) => (theme: Theme) => + css(get(theme, `styles.${key}`))(theme) // opt out of typechecking whenever `as` prop is used interface AnyComponentProps extends JSX.IntrinsicAttributes { @@ -109,8 +95,11 @@ interface AnyComponentProps extends JSX.IntrinsicAttributes { export interface ThemedComponent { ( - props: Name extends keyof JSX.IntrinsicElements ? ComponentProps : {} + props: (Name extends keyof JSX.IntrinsicElements + ? ComponentProps + : {}) & { css?: CSSObject } ): JSX.Element + displayName: string } export type ThemedComponentsDict = { @@ -127,14 +116,30 @@ const createThemedComponent = ( ): ThemedComponent => { const variantStyles = themed(variant) - return (props) => { + const component: ThemedComponent = (props) => { + const extraStyles: { textAlign?: 'left' | 'right' | 'center' | 'justify' } = + {} + + if (name === 'th' || name === 'td') { + const { align } = props as DetailedHTMLProps< + React.ThHTMLAttributes, + HTMLTableHeaderCellElement + > + + if (align !== 'char') extraStyles.textAlign = align + } + const css = (props as any)['css'] return jsx(name, { ...props, - css: css ? [css, variantStyles] : variantStyles, + css: [props.css, variantStyles, extraStyles].filter(Boolean), }) } + + component.displayName = `Themed(${name})` + + return component } interface ThemedDivProps @@ -167,10 +172,18 @@ const createComponents = (comps: MDXProviderComponents) => { // todo: test this behaviour componentKeys.forEach((key) => { const componentAtKey = comps[key] + if (componentAtKey) { - next[key] = (props) => jsx(componentAtKey, { ...props, css: themed(key) }) + const component: ThemedComponent = (props) => { + return jsx(componentAtKey, { ...props, css: themed(key) }) + } + + component.displayName = "MdxComponents('" + key + "')" + + next[key] = component } }) + return next } diff --git a/packages/mdx/test/index.tsx b/packages/mdx/test/index.tsx index 1501a5032..c88cdd56e 100644 --- a/packages/mdx/test/index.tsx +++ b/packages/mdx/test/index.tsx @@ -17,8 +17,10 @@ expect.extend(matchers) test('styles React components', () => { const Beep = (props: React.ComponentPropsWithoutRef<'h2'>) => ( + // eslint-disable-next-line jsx-a11y/heading-has-content

) + const Inner = (props: React.ComponentPropsWithoutRef) => mdx('Beep', props) @@ -45,22 +47,9 @@ test('styles React components', () => { expect(json).toHaveStyleRule('color', 'tomato') }) -test('cleans up style props', () => { - const json = renderJSON( - // @ts-expect-error - - Hello - - )! - expect(json.props.id).toBe('test') - expect(json.props.mx).not.toBeDefined() -}) - test('themed extracts styles from the theme', () => { expect( - themed('footer')({ - theme: { styles: { footer: { background: 'skyblue' } } }, - }) + themed('footer')({ styles: { footer: { background: 'skyblue' } } }) ).toStrictEqual({ background: 'skyblue' }) })