Passe adereços de componente em rota privada com texto digitado e React
Estou implementando rotas autenticadas com Typescript e React usando os adereços de renderização do componente Route do React Router v4.
Rotas:
import React from 'react';
import { Switch, Route } from 'react-router-dom';
import { ROUTES } from 'utils/constants';
import HomePage from 'components/pages/Home';
import GuestLogin from 'components/pages/GuestLogin';
import ProfilePage from 'components/pages/Profile';
import NotFoundPage from 'components/pages/NotFound';
import ResetPassword from 'components/pages/ResetPassword';
import SetPassword from 'components/pages/SetPassword';
import LoginContainer from 'containers/Login';
import PrivateRoute from './PrivateRoute';
const Routes: React.FunctionComponent = () => (
<Switch>
<Route path={ROUTES.LOGIN} component={LoginContainer} exact></Route>
<PrivateRoute
path={ROUTES.HOME}
component={HomePage}
></PrivateRoute>
<Route path={ROUTES.GUEST_LOGIN} component={GuestLogin}></Route>
<Route path={ROUTES.RESET_PASSWORD} component={ResetPassword}></Route>
<Route path={ROUTES.SET_PASSWORD} component={SetPassword}></Route>
<Route path={ROUTES.PROFILE} component={ProfilePage}></Route>
<Route component={NotFoundPage}></Route>
</Switch>
);
export default Routes;
Rota privada:
import React from 'react';
import { useAppContext } from 'containers/App/AppContext';
import { RouteProps, Route, Redirect } from 'react-router-dom';
import { ROUTES } from 'utils/constants';
const PrivateRoute: React.FunctionComponent<RouteProps> = ({
component: Component,
...routeProps
}) => {
const { isSignedIn } = useAppContext();
const ComponentToRender = Component as React.ElementType;
return (
<Route
{...routeProps}
render={(props) =>
isSignedIn ? (
<ComponentToRender {...props} />
) : (
<Redirect to={ROUTES.LOGIN} />
)
}
/>
);
};
export default PrivateRoute;
O problema é que eu quero chamar o conjunto de componentes nos adereços, no entanto, toda vez que tento fazer isso, o Typescript gera o seguinte erro.
JSX element type 'Component' does not have any construct or call signatures. TS2604
Imagem do erro
A razão parece ser que o tipo de componente para Rota não é o que o Typecript espera, conforme explicado aqui: https://github.com/microsoft/TypeScript/issues/28631, portanto, acabei de criar uma cópia que tem um novo tipo (ComponentToRender).
Existe uma maneira melhor de implementar isso? Talvez substituindo o elemento do componente RouteProps?
Obrigado!
Respostas
Finalmente entendi o erro e resolvi! A chave é que o atributo do componente e a função de renderização são tratados de forma diferente. Ao usar RouteProps
, o compilador Typescript está, na verdade, definindo os props do componente como ComponentType (o tipo especificado para 'componente' de acordo com RouterProps), que não pode ser usado para a função de renderização, como aparece no arquivo index.d.ts.
export interface RouteProps {
location?: H.Location;
component?: React.ComponentType<RouteComponentProps<any>> | React.ComponentType<any>;
render?: (props: RouteComponentProps<any>) => React.ReactNode;
children?: ((props: RouteChildrenProps<any>) => React.ReactNode) | React.ReactNode;
path?: string | string[];
exact?: boolean;
sensitive?: boolean;
strict?: boolean;
}
Esse comportamento se torna aparente apenas movendo a função de renderização para o arquivo Rotas e não usando o componente PrivateRoute. O código abaixo funciona conforme o esperado, pois o compilador infere o tipo corretamente (React.ReactNode).
<Route
path={ROUTES.POINTS_HISTORY}
render={(props) =>
isSignedIn ? <PointsTransactions /> : <Redirect to={ROUTES.LOGIN} />
}
></Route>
Portanto, para resolver o problema, acabo de criar um novo tipo apenas com os parâmetros necessários para o meu caso de uso.
import React from 'react';
import { useAppContext } from 'containers/App/AppContext';
import { RouteProps, Route, Redirect } from 'react-router-dom';
import { ROUTES } from 'utils/constants';
type PrivateRouteProps = {
path: RouteProps['path'];
component: React.ElementType;
};
const PrivateRoute: React.FunctionComponent<PrivateRouteProps> = ({
component: Component,
...routeProps
}) => {
const { isSignedIn } = useAppContext();
return (
<Route
{...routeProps}
render={(props) =>
isSignedIn ? <Component /> : <Redirect to={ROUTES.LOGIN} />
}
/>
);
};
export default PrivateRoute;