From 3055efddf8f9fb14b148fda466dcb4eb9affc525 Mon Sep 17 00:00:00 2001 From: Anton Golub Date: Sun, 12 Jun 2022 11:22:00 +0300 Subject: [PATCH] Type changes around `StyledOptions`/`FilteringStyledOptions` and `shouldForwardProp` (#2333) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * chore(style): let StyledOptions generic argument be optional * chore(native): let StyledOptions generic argument be optional * test(types): add tests for StyledOptions iface * feat(types): add keys assertion for StyledOptions generic * fix(styled): add StyledOptions type to CreateStyledComponent call signature fixes https://github.com/emotion-js/emotion/pull/2333#discussion_r609965699 * test(styled): check type StyledOpts for composite component * test(styled): revert erroneous changes * test(styled): clarify StyledOptsBroken test case * test: refactor `styledOptions` generics tests * test: add more tests for StyledOptions * test: linting * test: tweak ups * feat: expose FilteringStyledOptions * fix: expost `FilteringStyledOptions` from native, tweak up some tests, add some comments * chore: use consistent default props type for both StyledOptions and FilteringStyledOptions * chore: snap #3 * styled.shouldForwardProp: require prop to be a string * Add PropertyKey->string changesets * Remove redundant `compilerOptions` * make `shouldForwardProp` always contravariant * Type fixes * Tweak `FilteringStyledOptions` and `StyledOptions` interfaces * Change `Props` constraint * Tweak type-level tests around `shouldForwardProp` * add changesets Co-authored-by: Sam Magura Co-authored-by: Mateusz BurzyƄski --- .changeset/good-cars-roll.md | 10 +++ .changeset/purple-ladybugs-think.md | 8 +++ .changeset/tall-flies-smoke.md | 8 +++ .changeset/thirty-boats-hang.md | 6 ++ packages/is-prop-valid/types/index.d.ts | 2 +- packages/native/types/base.d.ts | 16 +++-- packages/native/types/index.d.ts | 1 + packages/native/types/tests.tsx | 26 +++++++- packages/styled/types/base.d.ts | 21 +++--- packages/styled/types/index.d.ts | 1 + packages/styled/types/tests.tsx | 86 ++++++++++++++++++++++++- 11 files changed, 166 insertions(+), 19 deletions(-) create mode 100644 .changeset/good-cars-roll.md create mode 100644 .changeset/purple-ladybugs-think.md create mode 100644 .changeset/tall-flies-smoke.md create mode 100644 .changeset/thirty-boats-hang.md diff --git a/.changeset/good-cars-roll.md b/.changeset/good-cars-roll.md new file mode 100644 index 000000000..6707f28e2 --- /dev/null +++ b/.changeset/good-cars-roll.md @@ -0,0 +1,10 @@ +--- +'@emotion/native': patch +'@emotion/styled': patch +--- + +pr: #2759 +author: @srmagura +author: @Andarist + +Change the argument of the `shouldForwardProp` option of `styled` from `PropertyKey` to `string` in the TypeScript definitions. diff --git a/.changeset/purple-ladybugs-think.md b/.changeset/purple-ladybugs-think.md new file mode 100644 index 000000000..148f64d7e --- /dev/null +++ b/.changeset/purple-ladybugs-think.md @@ -0,0 +1,8 @@ +--- +'@emotion/is-prop-valid': patch +--- + +pr: #2759 +author: @srmagura + +Change the type of the argument to `isPropValid` from `PropertyKey` to `string` in the TypeScript definitions. diff --git a/.changeset/tall-flies-smoke.md b/.changeset/tall-flies-smoke.md new file mode 100644 index 000000000..cdb24cc23 --- /dev/null +++ b/.changeset/tall-flies-smoke.md @@ -0,0 +1,8 @@ +--- +'@emotion/native': patch +'@emotion/styled': patch +--- + +author: @Andarist + +`shouldForwardProp` has been changed from being a bivariant method to a contravariant function - it improves the type-safety for those that type this option. diff --git a/.changeset/thirty-boats-hang.md b/.changeset/thirty-boats-hang.md new file mode 100644 index 000000000..8b34829c5 --- /dev/null +++ b/.changeset/thirty-boats-hang.md @@ -0,0 +1,6 @@ +--- +'@emotion/native': patch +'@emotion/styled': patch +--- + +`FilteringStyledOptions` and `StyledOptions` types no longer require a type argument for the `Props` generic. diff --git a/packages/is-prop-valid/types/index.d.ts b/packages/is-prop-valid/types/index.d.ts index 2ca137567..e7d00bfdb 100644 --- a/packages/is-prop-valid/types/index.d.ts +++ b/packages/is-prop-valid/types/index.d.ts @@ -1,5 +1,5 @@ // Definitions by: Junyoung Clare Jang // TypeScript Version: 2.1 -declare function isPropValid(string: PropertyKey): boolean +declare function isPropValid(prop: string): boolean export default isPropValid diff --git a/packages/native/types/base.d.ts b/packages/native/types/base.d.ts index 82aae23be..a14f22bad 100644 --- a/packages/native/types/base.d.ts +++ b/packages/native/types/base.d.ts @@ -60,14 +60,14 @@ export type Interpolation< /** Same as StyledOptions but shouldForwardProp must be a type guard */ export interface FilteringStyledOptions< - Props, - ForwardedProps extends keyof Props = keyof Props + Props = Record, + ForwardedProps extends keyof Props & string = keyof Props & string > { - shouldForwardProp?(propName: PropertyKey): propName is ForwardedProps + shouldForwardProp?: (propName: string) => propName is ForwardedProps } -export interface StyledOptions { - shouldForwardProp?(propName: PropertyKey): boolean +export interface StyledOptions> { + shouldForwardProp?: (propName: string) => boolean } /** @@ -146,7 +146,8 @@ export interface CreateStyledComponent< export interface CreateStyled { < C extends React.ComponentClass>, - ForwardedProps extends keyof React.ComponentProps = keyof React.ComponentProps + ForwardedProps extends keyof React.ComponentProps & + string = keyof React.ComponentProps & string >( component: C, options: FilteringStyledOptions, ForwardedProps> @@ -175,7 +176,8 @@ export interface CreateStyled { < C extends React.ComponentType>, - ForwardedProps extends keyof React.ComponentProps = keyof React.ComponentProps + ForwardedProps extends keyof React.ComponentProps & + string = keyof React.ComponentProps & string >( component: C, options: FilteringStyledOptions, ForwardedProps> diff --git a/packages/native/types/index.d.ts b/packages/native/types/index.d.ts index 4e6bb9a12..975f2a7bf 100644 --- a/packages/native/types/index.d.ts +++ b/packages/native/types/index.d.ts @@ -18,6 +18,7 @@ export { ArrayInterpolation, CreateStyledComponent, CSSInterpolation, + FilteringStyledOptions, FunctionInterpolation, Interpolation, InterpolationPrimitive, diff --git a/packages/native/types/tests.tsx b/packages/native/types/tests.tsx index cbec7284c..002865633 100644 --- a/packages/native/types/tests.tsx +++ b/packages/native/types/tests.tsx @@ -7,7 +7,12 @@ import { TextStyle, View } from 'react-native' -import styled, { css, ReactNativeStyle } from '@emotion/native' +import styled, { + css, + ReactNativeStyle, + StyledOptions, + FilteringStyledOptions +} from '@emotion/native' declare module '@emotion/react' { // tslint:disable-next-line: strict-export-declare-modifiers @@ -166,3 +171,22 @@ export const ImageFullWidthContained = styled.Image` ; ; } + +{ + // Props forwarding through StyledOptions and FilteringStyledOptions + + styled(View, { shouldForwardProp: (prop: string) => true })({}) + // $ExpectError + styled(View, { shouldForwardProp: (prop: 'testID') => true })({}) + + styled(View, { + shouldForwardProp: (prop: string): prop is 'testID' => true + })({}) + styled(View, { + // $ExpectError + shouldForwardProp: (prop: 'testID'): prop is 'testID' => true + })({}) + + // $ExpectError + styled(View, { shouldForwardProp: (prop: 'foo') => true })({}) +} diff --git a/packages/styled/types/base.d.ts b/packages/styled/types/base.d.ts index c39200c75..05efbaea4 100644 --- a/packages/styled/types/base.d.ts +++ b/packages/styled/types/base.d.ts @@ -3,7 +3,7 @@ import * as React from 'react' import { ComponentSelector, Interpolation } from '@emotion/serialize' -import { PropsOf, DistributiveOmit, Theme } from '@emotion/react' +import { PropsOf, Theme } from '@emotion/react' export { ArrayInterpolation, @@ -15,17 +15,17 @@ export { ComponentSelector, Interpolation } /** Same as StyledOptions but shouldForwardProp must be a type guard */ export interface FilteringStyledOptions< - Props, - ForwardedProps extends keyof Props = keyof Props + Props = Record, + ForwardedProps extends keyof Props & string = keyof Props & string > { label?: string - shouldForwardProp?(propName: PropertyKey): propName is ForwardedProps + shouldForwardProp?: (propName: string) => propName is ForwardedProps target?: string } -export interface StyledOptions { +export interface StyledOptions> { label?: string - shouldForwardProp?(propName: PropertyKey): boolean + shouldForwardProp?: (propName: string) => boolean target?: string } @@ -118,7 +118,8 @@ export interface CreateStyledComponent< export interface CreateStyled { < C extends React.ComponentClass>, - ForwardedProps extends keyof React.ComponentProps = keyof React.ComponentProps + ForwardedProps extends keyof React.ComponentProps & + string = keyof React.ComponentProps & string >( component: C, options: FilteringStyledOptions, ForwardedProps> @@ -147,7 +148,8 @@ export interface CreateStyled { < C extends React.ComponentType>, - ForwardedProps extends keyof React.ComponentProps = keyof React.ComponentProps + ForwardedProps extends keyof React.ComponentProps & + string = keyof React.ComponentProps & string >( component: C, options: FilteringStyledOptions, ForwardedProps> @@ -168,7 +170,8 @@ export interface CreateStyled { < Tag extends keyof JSX.IntrinsicElements, - ForwardedProps extends keyof JSX.IntrinsicElements[Tag] = keyof JSX.IntrinsicElements[Tag] + ForwardedProps extends keyof JSX.IntrinsicElements[Tag] & + string = keyof JSX.IntrinsicElements[Tag] & string >( tag: Tag, options: FilteringStyledOptions diff --git a/packages/styled/types/index.d.ts b/packages/styled/types/index.d.ts index 286a2d176..883a79ea0 100644 --- a/packages/styled/types/index.d.ts +++ b/packages/styled/types/index.d.ts @@ -12,6 +12,7 @@ export { Interpolation, StyledComponent, StyledOptions, + FilteringStyledOptions, CreateStyledComponent } from './base' diff --git a/packages/styled/types/tests.tsx b/packages/styled/types/tests.tsx index 45d88699f..97c17a073 100644 --- a/packages/styled/types/tests.tsx +++ b/packages/styled/types/tests.tsx @@ -1,5 +1,5 @@ import * as React from 'react' -import styled from '@emotion/styled' +import styled, { StyledOptions, FilteringStyledOptions } from '@emotion/styled' // This file uses the same Theme declaration from tests-base.tsx @@ -217,3 +217,87 @@ const Input5 = styled.input` // $ExpectError ; } + +{ + // Props forwarding through StyledOptions and FilteringStyledOptions + + const fc: React.FC<{ foo: string }> = () => null + + // we can't accept a "known" prop here because we need to include `AdditionalProps` and those aren't available yet + // `Props` represent the actual props of a component while `AdditionalProps` represent props used only for styling purposes + // $ExpectError + styled(fc, { shouldForwardProp: (prop: 'foo') => true })({}) + + styled(fc, { shouldForwardProp: (prop: string) => true })({}) + + // $ExpectError + styled(fc, { shouldForwardProp: (prop: 'bar') => true })({}) + // $ExpectError + styled(fc, { shouldForwardProp: (prop: 'foo') => true })({}) + + // $ExpectError + const shouldForwardProp1: StyledOptions['shouldForwardProp'] = ( + prop: 'unknown' + ) => true + styled(fc, { shouldForwardProp: shouldForwardProp1 })({}) + + // $ExpectError + styled(fc, { shouldForwardProp: (prop: 'unknown') => true })({}) + + // $ExpectError + const shouldForwardProp2: StyledOptions<{ + foo: string + }>['shouldForwardProp'] = (prop: 'unknown') => true + + styled(fc, { shouldForwardProp: (prop: string): prop is 'foo' => true })({}) + // $ExpectError + styled(fc, { shouldForwardProp: (prop: 'foo'): prop is 'foo' => true })({}) + + const shouldForwardProp3: FilteringStyledOptions['shouldForwardProp'] = ( + prop: string + ): prop is 'foo' => true + + // $ExpectError + const shouldForwardProp4: FilteringStyledOptions['shouldForwardProp'] = ( + prop: 'foo' + ): prop is 'foo' => true + + const shouldForwardProp5: FilteringStyledOptions<{ + foo: string + }>['shouldForwardProp'] = (prop: string): prop is 'foo' => true + // $ExpectError + const shouldForwardProp6: FilteringStyledOptions<{ + foo: string + }>['shouldForwardProp'] = (prop: 'foo'): prop is 'foo' => true + + // $ExpectError + const shouldForwardProp7: FilteringStyledOptions<{ + foo: string + }>['shouldForwardProp'] = (prop: 'unknown'): prop is 'unknown' => true + + const shouldForwardProp8: FilteringStyledOptions< + { foo: string; bar: string }, + 'foo' + >['shouldForwardProp'] = (prop: string): prop is 'foo' => true + + // $ExpectError + const shouldForwardProp9: FilteringStyledOptions< + { foo: string; bar: string }, + 'foo' + >['shouldForwardProp'] = (prop: 'foo' | 'bar'): prop is 'bar' => true + + styled('div', { + shouldForwardProp: (prop: string) => true + })({}) + + // $ExpectError + styled('div', { shouldForwardProp: (prop: 'color') => true })({}) + + styled('div', { + // $ExpectError + shouldForwardProp: (prop: 'color'): prop is 'color' => true + })({}) + + // $ExpectError + styled('div', { shouldForwardProp: (prop: 'foo') => true })({}) +}