Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Typescript type error in component returned by withStyles() #8447

Closed
1 task done
cfilipov opened this issue Sep 28, 2017 · 55 comments
Closed
1 task done

Typescript type error in component returned by withStyles() #8447

cfilipov opened this issue Sep 28, 2017 · 55 comments

Comments

@cfilipov
Copy link

cfilipov commented Sep 28, 2017

When using withStyles() hoc in typescript, I am getting the following error when trying to use the returned component:

Type '{}' is not assignable to type 'IntrinsicAttributes & IntrinsicClassAttributes<App> & Readonly<{ children?: ReactNode; }> & Reado...'.
  Type '{}' is not assignable to type 'Readonly<WithStyles<"main">>'.
    Property 'classes' is missing in type '{}'.
  • I have searched the issues of this repository and believe that this is not a duplicate.

It appears this change to the type definition might be related to this issue.

Expected Behavior

Given the App component code below, I should be able to use the component <App /> without the type error as I did in 1.0.0-beta.10.

Current Behavior

Given the App component code below, trying to use <App /> results in the aforementioned error.

The Code

import * as React from 'react';
import { withStyles } from 'material-ui/styles';

const styles = {
    main: {
        marginTop: 48,
        padding: 10,
    },
    foo: {
        margin: 0,
    },
};

interface Props {
    message: string;
};

type ClassNames = { classes: { [className in keyof typeof styles]: string } };

class App extends React.Component<Props & ClassNames> {
    render() {
        const { classes, message } = this.props;
        return (
            <div className={classes.main}>
                <div className={classes.foo} >
                    Hello World! {message}
                </div>
            </div>
        );
    }
}

export default withStyles(styles)(App);

Context

The code worked fine in 1.0.0-beta.10, when I upgraded to 1.0.0-beta.12 I got the type error.

In the code snippet provided I used the keyof typeof styles trick so that I would not need to define a list of class names twice (I strongly dislike the repetitiveness). I have also tried other variations:

type ClassNames = WithStyles<keyof typeof styles>;

and doing it the more common way (as seen in styles.spec.tsx):

type ComponentClassNames = 'main' | 'foo';
type ClassNames = WithStyles<ComponentClassNames>;

I still get the same error.

It seems the previous type definition would return a component whose props type would be StyledComponentProps which has an optional classes property. The new definition...

<P, C extends React.ComponentClass<P & StyledComponentProps<Names>>>(
    component: C
  ): C;

...returns the same type C as the component, this means that passing ClassNames which is not marked optional propagates to the returned component. I see mentioned here the use of Partial<> which I think it an unsightly hack.

Your Environment

Tech Version
Material-UI 1.0.0-beta.12
React 15.6.1
browser Chrome 61.0.3163.100
@pelotom
Copy link
Member

pelotom commented Sep 28, 2017

@cfilipov when decorating a component class you should use StyledComponentProps instead of WithStyles, and if in strict type checking mode you need to use the non-null assertion operator ! to extract the fields of classes. This is a compromise to allow the use of withStyles as a class decorator. Class decorators in TypeScript suffer from the limitation that their return type must match the argument type. Only one option given this limitations is possible:

  1. don't support using withStyles as a class decorator
  2. require that you pass a dummy classes prop when constructing an element of the decorated component type (the previous compromise, which was arguably more annoying)
  3. require that classes be considered nullable inside the component, forcing you to use the ! operator when accessing its fields (the current compromise)

I would highly recommend that if your component has no state you use a stateless functional component, which will require fewer type annotations and be more type safe:

export default withStyles(styles)<Props>(({ classes, message }) => (
  <div className={classes.main}>
    <div className={classes.foo}>
      Hello World! {message}
    </div>
  </div>
));

@pelotom
Copy link
Member

pelotom commented Sep 28, 2017

I see mentioned here the use of Partial<> which I think it an unsightly hack.

If you read my followup to that comment you'll see that Partial<> is not required.

@cfilipov
Copy link
Author

If you read my followup to that comment you'll see that Partial<> is not required.

I saw your comment regarding Partial<>. Using StyledComponentProps essentially amounts to the same thing (it's using Partial<> in that definition). My gripe with that is now class names must be accessed with ! (as you mentioned). I think passing a dummy classes prop or requiring the use of ! are both poor compromises.

To clarify, I guess the root issue I am having here is the choice to introduce a regression to the hoc in an effort to work around a limitation of an experimental feature of typescript. I admit to some bias since I don't care for decorators, whereas others clearly do, but the tradeoff here doesn't make sense to me.

@pelotom
Copy link
Member

pelotom commented Sep 28, 2017

@cfilipov my first refactor of the withStyles typings was to choose option (1), i.e. make the typings completely correct for both class and function components, at the expense of using withStyles as a class decorator. I then got feedback that using the decorator syntax was a popular request, so I switched to option (3). I'm happy to revisit that decision; I too would prefer type safety to supporting decorators in their current semifunctional state.

@cfilipov
Copy link
Author

Yeah, I understand the desire to support a popular request like this. I want to use decorators too, but I keep running into many issues with them every time I do (outside of mui) so I personally decided to not use them until they are ready.

It sounds like you share my concern and I don't have much more to add so hopefully other people can provide feedback in this direction to help influence it.

@marcusjwhelan
Copy link

I switched to beta.13 from beta.10 to see if anything had changed and Yes this is a real issue. To throw my 2 cents in here, for me, decorators are experimental. They obviously could be changed in the future. And until then I would fully support the 100% accurate way. I would much rather have coherent type safety to hacking my types to make things work.

@pelotom
Copy link
Member

pelotom commented Oct 5, 2017

#8550 looks like further evidence that people are confused by this, and we should consider not supporting @withStyles() as a decorator (in TypeScript).

@pelotom
Copy link
Member

pelotom commented Oct 5, 2017

This is what it would look like to "decorate" a class if we made the typings correct:

type NonStyleProps = {
  text: string
};

const styles = {
  root: {
    backgroundColor: 'red'
  }
};

const DecoratedComponent = withStyles(styles)(
  class extends React.Component<NonStyleProps & WithStyles<'root'>> {
    render() {
      return (
        <div className={this.props.classes.root}>
          {this.props.text}
        </div>
      );
    }
  }
);

@marcusjwhelan
Copy link

@pelotom I have yet to move forward from beta.10 because of this issue. I made comment about the styling of a class with Redux's connect method before. I think its relatively easy and robust. in #8059 the third comment is myself with the types.

@wcandillon
Copy link
Contributor

wcandillon commented Oct 5, 2017

@pelotom Thank so much for educating us on this issue. I am a big user of TS decorators but in that instance, I would be happy for withStyles to drop its decorating support in order to have better type safety.

@oliviertassinari
Copy link
Member

we should consider not supporting @withStyles() as a decorator (in TypeScript).

@pelotom I'm personally in favor of this change. @sebald What do you want to do here?

@pelotom
Copy link
Member

pelotom commented Oct 5, 2017

It's a simple change; I went ahead and opened a PR in case you want to go with it 🙂

@marcusjwhelan
Copy link

@pelotom would the typings still work if I used them like this?

interface IStyles {
    // styles interface
}
interface IHeaderInfoStyles {
    classes: any;
}
interface IHeaderInfoProps {
    // properties resulting from dispatches
}
interface IHeaderInfoInjectedProps {
   // props injected from parent if any
}
interface IHeaderInfoDispatchProps {
    // dispatches
}
interface IHeaderInfoState {
    // your state for the class
}

type HeaderInfoProps = IHeaderInfoProps & IHeaderInfoInjectedProps & IHeaderInfoDispatchProps & IHeaderInfoStyles;

class HeaderInfo extends Component<HeaderInfoProps, IHeaderInfoState> {

export const HeaderInfo_container = withStyles<IHeaderInfoInjectedProps, IStyles>(styles)(
    connect<IHeaderInfoProps, IHeaderInfoDispatchProps, (IHeaderInfoInjectedProps & IHeaderInfoStyles)>(mapStateToProps, mapDispatchToProps)(HeaderInfo),
);

@pelotom
Copy link
Member

pelotom commented Oct 5, 2017

@marcusjwhelan withStyles no longer takes 2 type parameters, and you shouldn't have to provide a type parameter for styles (it can be inferred from styles). You should instead be able to write

withStyles(styles)<NonStyleProps>(...);

If you give the types for mapStateToProps and mapDispatchToProps I might be able to show you what this would look like for your example.

@sebald
Copy link
Member

sebald commented Oct 6, 2017

tl;dr; Let's do the changes and see what the backlash is, I guess 👷🏻


@oliviertassinari @pelotom ¯_(ツ)_/¯ I do not use decorators, so personally, I really don't care about this change, because I am not affected. But it seems like a lot of people do care about this "feature". That's why we added it in the first place. That's IMHO what has priority here.

I am very happy with the changes of withStyles, but when you take a look at other more functional libraries, like ramda or recompose, the typings aren't that strict, neither are they super type-safe. Often times you have to pass in a generic, which represents the return value of a function. Not pretty but it will work for a 99.9% of the users.

Why can't we bring back the old typings that worked for the people using decorators? Because it did have two generics, we may be able to have both typings side by side.

Also, I am a bit confused what peoples issues with decorators are (regarding "not working"). Libs like Angular and https://nestjs.com/ make heavy use of them after all. In our case, people could just add WithStyles to the props and are fine.

@pelotom
Copy link
Member

pelotom commented Oct 6, 2017

@sebald

Why can't we bring back the old typings that worked for the people using decorators? Because it did have two generics, we may be able to have both typings side by side.

I'm not sure what you mean. There was never any typing that allowed people to use decorators painlessly. The first implementation suffered from #8267, which was that you couldn't construct an element of a decorated component without passing a dummy classes prop, e.g. <StyledComp classes={{} as any} />. The second (current) implementation suffers from the converse problem, that classes is seen as potentially undefined within the component class. If you want to use TypeScript decorators in their current form, those are your only 2 options; pick your poison.

The third way is using the correct type, so that classes is defined within the class but not required to be passed as a prop. To have both those conditions, you need to give up decorators.

@sebald
Copy link
Member

sebald commented Oct 6, 2017

@pelotom Yes, sorry. You're right. Really is not my day... 🤐 So lets merge!

@pelotom
Copy link
Member

pelotom commented Oct 6, 2017

@sebald

Also, I am a bit confused what peoples issues with decorators are (regarding "not working"). Libs like Angular and https://nestjs.com/ make heavy use of them after all.

I'm not sure how they're used in Angular, but there are certainly use cases where decorators can be used painlessly; basically if the decorator doesn't need to change the type of the class it's decorating, they work fine. That's not the situation we have here; withStyles needs to change the type of the component it decorates.

@sebald
Copy link
Member

sebald commented Oct 6, 2017

@pelotom Yes, exactly. It's just the mutation that is bad. Actually, the way TS currently implements decorators might even be good for the Angular users, since the decorators AFAIK set up contracts with the framework, like "register this class as a component" or "add meta data so I can use this in the DI" ... god even writing about this makes me feel 🤢

@marcusjwhelan
Copy link

@pelotom the reason I have the types the way I do is it provided Type Safety for my components. Currently the types have no type safety when it comes to components. The injectedProps in my example are the types that are required by the component to work. The types for connect from react-redux need to be that of Props which come from mapStateToProps and DispatchProps which come from the mapDispatchToProps.

At the very end there needs to be the injectedProps so that my parent component knows that I need to inject them or my project will not compile. (this is what I want). Is this change going to migrate back to this kind of typeSafety?

@pelotom
Copy link
Member

pelotom commented Oct 6, 2017

@marcusjwhelan again, if you can provide a full (but minimal) example that was compiling in beta 10 I can show you how to migrate. AFAIK the new version should be every bit as expressive and more type safe than before.

@wcandillon
Copy link
Contributor

wcandillon commented Oct 8, 2017

@pelotom Silly question, is there a way I could get notified once a new release is done?

@oliviertassinari
Copy link
Member

@wcandillon Follow us on Twitter.

@marcusjwhelan
Copy link

marcusjwhelan commented Oct 9, 2017

@pelotom I did post an example above... Why would you need to see any more than that? You can assume the properties are a,b,c,d,e. The only thing is the injected props need to be emitted as a requirement.

Edit

I figured it out.

@isman-usoh
Copy link

I solved using recompose

example

import { StyleRules, Theme, withStyles } from "material-ui/styles";
import * as React from "react";
import { compose } from "recompose";

interface IProps {
    classes?: any; // <-- add question mark
    text: string;
}

const styles = (theme: Theme): StyleRules => ({
    root: {

    },
});

@compose(
    withStyles(styles),
)
export class MyComponent extends React.Component<IProps> {
    render() {
        const { classes, text } = this.props;
        return (
            <div className={classes.root}>
                {text}
            </div>
        );
    }
}

@svachmic
Copy link
Contributor

svachmic commented Nov 9, 2017

I was fighting with this issue (and #8704), with no clear result over the past few days. Then I took the advice from @pelotom:

you should use StyledComponentProps instead of WithStyles

and searched on GitHub for similar ways to solve this. And I found ONE working example 😂. It's a good one, though, and it did solve my problem - separate container and component with TypeScript being a happy cookie: mentioned example here (Note: In my case, the component mapping is in a separate container file, but the idea is the same.).

If anybody thinks this is a bad solution, I'm open to any changes and ideas. RIght now, I'm just glad my code stopped complaining.

@lookfirst
Copy link
Contributor

type Styles = 'foo';
const styles: StyleRulesCallback<Styles> = (theme: Theme) => ({
	foo: {
		position: 'relative',
	}
});

interface Props {
  something: string;
}

class Sidebar extends React.Component<Props & WithStyles<Styles>> {
	render() {
		const { classes, something } = this.props;

		return (
                    <div className={classes.foo}>{something}</div>
		);
	}
}

export default withStyles(styles)<Props>(Sidebar);

@otroboe
Copy link

otroboe commented Apr 24, 2018

I don't want to create a new issue, but I've tried anything I saw in documentation, example, and passed issues, even with recompose, but I can't make my component working when I add some properties to it.
And the resources I found are mostly with older versions of TS, MUI or even React.

Here's my component:

import React from 'react';
import AppBar from 'material-ui/AppBar';
import { withStyles, WithStyles, StyleRulesCallback } from 'material-ui/styles';

const styles: StyleRulesCallback<'positionFixed'> = () => ({
  positionFixed: {
    top: 'auto',
    bottom: 0,
  },
});

const decorate = withStyles(styles);

interface Props {
   someProp: string;
};

export const BottomNav = decorate<Props>(
  class extends React.Component<Props & WithStyles<'positionFixed'>, {}> {
    render() {
      return (
        <AppBar />
      );
    }
  }
);

export default BottomNav;

And the error is:

TS2322: Type '{}' is not assignable to type 'IntrinsicAttributes & Props & StyledComponentProps<"positionFixed"> & { children?: ReactNode; }'.
  Type '{}' is not assignable to type 'Props'.
    Property 'someProp' is missing in type '{}'.

I'm quite a beginner with TS, but I find the documentation page, and the example quite confusing and/or not complete.

If you guys have any idea, thank you ;-)

@pelotom
Copy link
Member

pelotom commented Apr 24, 2018

@otroboe are you leaving out the code that actually got flagged with the error? Does it look like this?

<BottomNav />

If so, the problem is you need to provide the someProp prop (which is required according to your definition of Props):

<BottomNav someProp="foo" />

@otroboe
Copy link

otroboe commented Apr 24, 2018

Shame on me... Oh, shame on me.
I was so focus on the TS integration, I forgot to look around and take some steps back.

Thanks a lot @pelotom 😄

@lookfirst
Copy link
Contributor

@otroboe Also remove the string duplication...

type Styles = 'positionFixed';

Wish that one was easier though...

@otroboe
Copy link

otroboe commented Apr 24, 2018

Yes I did that too, thanks 👍

@nishmeht7
Copy link

I just faced this same problem, but it turns out that it occurs only if I have initialized the styles object in the same file as my class or function. Alternately, If I am importing the styles from another file, I don't get this error.

Any clue why this happens?

@pelotom
Copy link
Member

pelotom commented Aug 23, 2018

@nishmeht7 can you post a self-contained snippet?

@nishmeht7
Copy link

@pelotom I just worked on making one, and it works fine in my sandbox environment. I'm currently working on a large app and am still on mui version 1.2.2, while my sandbox env version is the latest one. So I'm guessing once I upgrade versions I won't be able to reproduce my error.

@HenrikBechmann
Copy link

HenrikBechmann commented Sep 10, 2018

I followed the example of a basic form from https://material-ui.com/demos/selects/ but got complaints that theme doesn't have root (using latest typescript and material-ui), and no class was being applied to the form element. I tried following the discussion above, but it seems inconclusive. And indeed the inherited classes list was missing the generated class name for form. If I added the generated classname manually (found in withStyles with console.log(theme) in dev tools inspect elements, all fine, so apparently the class was being generated correctly. Wasn't getting passed through withStyles to the form element though. Quite confusing.

So I've reverted back to styles for now until this gets sorted:

<form style = {{display:'flex',flexWrap:'wrap'}} autoComplete="off">

@lookfirst
Copy link
Contributor

@HenrikBechmann Have you tried following the styles documentation for typescript? In the past, it has been quite helpful for me. https://material-ui.com/guides/typescript/

@HenrikBechmann
Copy link

Thanks @lookfirst! I looked (though not first :-) ) at that doc, and used

export default withStyles({
  root: {
    display: 'flex',
    flexWrap: 'wrap',
  },
})(BaseForm)

(passing the object rather than the function)

... which both avoided the typescript errors, and enabled the injection of the generated class.

Hopefully the other tips there will help with more complex constructs.

@HenrikBechmann
Copy link

HenrikBechmann commented Sep 11, 2018

I also confirmed that the more complex structure for the styles function works (injects a generated className into form):

import React from 'react'

import { withStyles, createStyles, Theme } from '@material-ui/core/styles'

/*
    patterned after first demo https://material-ui.com/demos/selects/ for 3.03
    use Typsecript fixes from here: https://material-ui.com/guides/typescript/
*/

const styles = (theme:Theme) => createStyles({
  root: {
    display: 'flex',
    flexWrap: 'wrap',
  },
})

class BaseForm extends React.Component<any,any> {

    render() {
        const { classes } = this.props

        return (
            <form className = {classes.root} autoComplete="off">
                {this.props.children}
            </form>
        )
    }
}

export default withStyles(styles)(BaseForm)

@chrislambe
Copy link
Contributor

chrislambe commented Sep 24, 2018

Edit: @eps1lon has pointed out that this is unnecessary with the use of WithStyles!

I've had some success using ReturnType<T> to generate the ClassKey for me:

import * as React from 'react';

import withStyles, {
  StyledComponentProps, 
  StyleRulesCallback,
} from '@material-ui/core/styles/withStyles';

import { Theme } from '@material-ui/core/styles/createMuiTheme';

const overrideStyles = (theme: Theme) => ({
  root: {
    display: 'flex',
    flexDirection: 'column',
  },
});

type Props = StyledComponentProps<keyof ReturnType<typeof overrideStyles>>;

class MyComponent extends React.PureComponent<Props> {
  render() {
    return <div className={this.props.classes.root}></div>;
  }
}

export default withStyles(overrideStyles as StyleRulesCallback, {withTheme: true})(MyComponent);

Using keyof ReturnType<typeof styleOverrides> as the StyledComponentProps ClassKey will get you the keys from the object returned by overrideStyles and save you from having to keep a list of those keys manually. The only glitch I've noticed is a type error if I don't cast overrideStyles as StyleRulesCallback in the withStyles call. I'm not 100% sure why. I think it's withStyles not understanding what overrideStyles is for some reason.

To clarify that rather elaborate type, typeof styleOverrides resolves to the function that returns the style object. ReturnType<T> will get you the style object itself. keyof will get you the keys from the style object.

@eps1lon
Copy link
Member

eps1lon commented Sep 24, 2018

@chrislambe You should checkout the typescript guide. You shouldn't need to use ReturnType etc. createStyles and WithStyles should be sufficient as helpers.

@chrislambe
Copy link
Contributor

@eps1lon Oh hey, very cool! Thanks!

@HenrikBechmann
Copy link

fwiw I'm liking the createStyles/withStyles pair more and more as I use them. Promotes tidy code, deals with all my styling/typescript issues, and if I want local conditional css I just create local style attributes, which of course take precedence over classes.

Nice!!

@valoricDe
Copy link

valoricDe commented Oct 22, 2018

Following the typescript guide with @material-ui/[email protected] I get Test does not have required attribute classes on:

import React from 'react'
import { Theme, WithStyles, withStyles, createStyles } from '@material-ui/core/styles'

const styles = (theme: Theme) => createStyles({
  root: {
    color: theme.palette.action.active
  },
})

interface Props extends WithStyles<typeof styles> {
  asd: boolean
}

class TestComponent extends React.Component<Props> {

  render() {
    const { classes } = this.props

    return (
      <div className={classes.root}>
      </div>
    )
  }
}

const Test = withStyles(styles)(TestComponent)

const a = () => <Test asd={true}/>

@eps1lon
Copy link
Member

eps1lon commented Oct 26, 2018

@aaronlifton2
Please follow Augmenting your props using WithStyles

@TrejGun
Copy link
Contributor

TrejGun commented Feb 20, 2019

@valoricDe Have you solved yout issue?

@valoricDe
Copy link

valoricDe commented Feb 21, 2019

@TrejGun I just checked. With a functional component and @material-ui/[email protected] I do not have this problem

@yehudamakarov
Copy link

yehudamakarov commented Mar 11, 2019

I don't really understand.
Following the docs here: https://material-ui.com/guides/typescript/#augmenting-your-props-using-withstyles
seems to just result in the problem of this original issue. Where when you use the component somewhere, typescript demans you to pass the classes property as a prop instead of realizing it is going to be injected by withStyles.

I've been reading these issues for a few hours now and I'm not really getting it. Any help would be so so nice at this point.

Regarding this suggestion above

@cfilipov when decorating a component class you should use StyledComponentProps instead of WithStyles, and if in strict type checking mode you need to use the non-null assertion operator ! to extract the fields of classes. This is a compromise to allow the use of withStyles as a class decorator. Class decorators in TypeScript suffer from the limitation that their return type must match the argument type. Only one option given this limitations is possible:

  1. don't support using withStyles as a class decorator
  2. require that you pass a dummy classes prop when constructing an element of the decorated component type (the previous compromise, which was arguably more annoying)
  3. require that classes be considered nullable inside the component, forcing you to use the ! operator when accessing its fields (the current compromise)

I would highly recommend that if your component has no state you use a stateless functional component, which will require fewer type annotations and be more type safe:

export default withStyles(styles)<Props>(({ classes, message }) => (
  <div className={classes.main}>
    <div className={classes.foo}>
      Hello World! {message}
    </div>
  </div>
));

How do I found out how to use StyledComponentProps? It seems I just pass strings of the keys defined in the styles object.

But the docs tell us to do something that simply won't work? what am I missing? I woiuld like to use https://material-ui.com/guides/typescript/#augmenting-your-props-using-withstyles...

Is this possible?

@valoricDe, How did you do the functional component that didn't have that issue

@TrejGun I just checked. With a functional component and @material-ui/[email protected] I do not have this problem

I'm trying something like this:

import PropTypes from "prop-types";
import { connect } from "react-redux";
import { Grid, FormControlLabel, Theme, createStyles, withStyles, Radio, WithStyles } from "@material-ui/core";
import Amount from "./Amount";
import { onPastDueFormFieldChange } from "../../store/actions/selectPaymentAmountActions";

const styles = (theme: Theme) =>
	createStyles({
		amount: {
			alignSelf: "center",
		},
	});

interface OwnProps extends WithStyles<typeof styles> {}

interface StateProps {
	pastDue?: number;
	pastDueOrTotalOrOther: string;
}

interface DispatchProps {
	onPastDueFormFieldChange: OnPastDueFormFieldChange;
}

type Props = StateProps & DispatchProps & OwnProps;

const PastDueFormField = withStyles(styles)(
	({ classes, pastDue, pastDueOrTotalOrOther, onPastDueFormFieldChange }: Props) => (
		<Grid spacing={0} container item xs={12}>
			<Grid item xs={6}>
				<FormControlLabel
					value="pastDue"
					checked={pastDueOrTotalOrOther === "pastDue"}
					onChange={onPastDueFormFieldChange}
					label="Past Due:"
					control={<Radio color="primary" />}
				/>
			</Grid>
			<Grid xs={6} item className={classes.amount}>
				<Amount amount={pastDue} />
			</Grid>
		</Grid>
	),
);

const mapStateToProps = (state: RootState): StateProps => ({
	pastDue: state.customerData.balanceDue.pastDue,
	pastDueOrTotalOrOther: state.customerPaymentsForm.pastDueTotalOrOther,
});

export default connect<StateProps, DispatchProps, OwnProps, RootState>(
	mapStateToProps,
	{ onPastDueFormFieldChange },
)(PastDueFormField);

When I use this component I have this error:

Property 'classes' is missing in type '{}' but required in type 'Readonly<PropsWithChildren<Pick<Pick<Props, "pastDue" | "pastDueOrTotalOrOther" | "onPastDueFormFieldChange"> & StyledComponentProps<"amount">, "classes" | "innerRef"> & OwnProps>>'

@eps1lon
Copy link
Member

eps1lon commented Mar 11, 2019

@yehudamakarov Try you code without react-redux first and add connect when everything works as expected. It's incredibly hard to get a good overview when what prop is injected.

When encountering these issues I'll

  1. type my component props first
  2. check if everything is required as expected i.e. get missing props
  3. apply hoc
  4. see if typescript recognized every injected prop
  5. repeat 3 until all hocs are applied.

It promotes cleaner code. Especially with regards to order of operations. You currently mix withStyles and connect without a separation of concerns.

@yehudamakarov
Copy link

yehudamakarov commented Mar 11, 2019

Thanks so much Sebastian.
I solved my problem by simply not passing generic arguments to connect. I took out the <StateProps ...> chunk.

I think those generic arguments were messing with my OwnProps interface that extends WithStyles<>.

Since I’m passing everything to the component anyway the type checking I get from my Props is sufficient. Not sure why id need the connect<> generics.

Thanks!

@pchodak
Copy link

pchodak commented Aug 9, 2019

This is what it would look like to "decorate" a class if we made the typings correct:

type NonStyleProps = {
  text: string
};

const styles = {
  root: {
    backgroundColor: 'red'
  }
};

const DecoratedComponent = withStyles(styles)(
  class extends React.Component<NonStyleProps & WithStyles<'root'>> {
    render() {
      return (
        <div className={this.props.classes.root}>
          {this.props.text}
        </div>
      );
    }
  }
);

And how to add themes (useTheme()) inside like:

const decorate = withStyles((theme: Theme) => ({
  root: {
    display: "flex"
  },
  appBar: {
    zIndex: theme.zIndex.drawer + 1,
    transition: theme.transitions.create(["width", "margin"], {
      easing: theme.transitions.easing.sharp,
      duration: theme.transitions.duration.leavingScreen
    })
  }
})

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests