68 lines
		
	
	
		
			2.0 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			68 lines
		
	
	
		
			2.0 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
import PropTypes from 'prop-types';
 | 
						|
 | 
						|
import { __RouterContext } from 'react-router';
 | 
						|
 | 
						|
import hoistStatics from 'hoist-non-react-statics';
 | 
						|
 | 
						|
export const WithRouterPropTypes = {
 | 
						|
  match: PropTypes.object.isRequired,
 | 
						|
  location: PropTypes.object.isRequired,
 | 
						|
  history: PropTypes.object.isRequired,
 | 
						|
};
 | 
						|
 | 
						|
export const WithOptionalRouterPropTypes = {
 | 
						|
  match: PropTypes.object,
 | 
						|
  location: PropTypes.object,
 | 
						|
  history: PropTypes.object,
 | 
						|
};
 | 
						|
 | 
						|
export interface OptionalRouterProps {
 | 
						|
  ref: unknown;
 | 
						|
  wrappedComponentRef: unknown;
 | 
						|
}
 | 
						|
 | 
						|
// This is copied from https://github.com/remix-run/react-router/blob/v5.3.4/packages/react-router/modules/withRouter.js
 | 
						|
// but does not fail if called outside of a React Router context
 | 
						|
export function withOptionalRouter<
 | 
						|
  ComponentType extends React.ComponentType<OptionalRouterProps>,
 | 
						|
>(Component: ComponentType) {
 | 
						|
  const displayName = `withRouter(${Component.displayName ?? Component.name})`;
 | 
						|
  const C = (props: React.ComponentProps<ComponentType>) => {
 | 
						|
    const { wrappedComponentRef, ...remainingProps } = props;
 | 
						|
 | 
						|
    return (
 | 
						|
      <__RouterContext.Consumer>
 | 
						|
        {(context) => {
 | 
						|
          // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
 | 
						|
          if (context) {
 | 
						|
            return (
 | 
						|
              // @ts-expect-error - Dynamic covariant generic components are tough to type.
 | 
						|
              <Component
 | 
						|
                {...remainingProps}
 | 
						|
                {...context}
 | 
						|
                ref={wrappedComponentRef}
 | 
						|
              />
 | 
						|
            );
 | 
						|
          } else {
 | 
						|
            // @ts-expect-error - Dynamic covariant generic components are tough to type.
 | 
						|
            return <Component {...remainingProps} ref={wrappedComponentRef} />;
 | 
						|
          }
 | 
						|
        }}
 | 
						|
      </__RouterContext.Consumer>
 | 
						|
    );
 | 
						|
  };
 | 
						|
 | 
						|
  C.displayName = displayName;
 | 
						|
  C.WrappedComponent = Component;
 | 
						|
  C.propTypes = {
 | 
						|
    ...Component.propTypes,
 | 
						|
    wrappedComponentRef: PropTypes.oneOfType([
 | 
						|
      PropTypes.string,
 | 
						|
      PropTypes.func,
 | 
						|
      PropTypes.object,
 | 
						|
    ]),
 | 
						|
  };
 | 
						|
 | 
						|
  return hoistStatics(C, Component);
 | 
						|
}
 |