import React, { CSSProperties, MouseEventHandler, ReactElement, ReactNode } from 'react'; import { AccountsState } from 'app/components/accounts'; import { User } from 'app/components/user'; import { connect } from 'app/functions'; import { TransitionMotion, spring, PlainStyle, Style, TransitionStyle, TransitionPlainStyle } from 'react-motion'; import { Panel, PanelBody, PanelFooter, PanelHeader } from 'app/components/ui/Panel'; import { Form } from 'app/components/ui/form'; import MeasureHeight from 'app/components/MeasureHeight'; import panelStyles from 'app/components/ui/panel.scss'; import icons from 'app/components/ui/icons.scss'; import authFlow from 'app/services/authFlow'; import { Provider as AuthContextProvider } from './Context'; import { getLogin, State as AuthState } from './reducer'; import * as actions from './actions'; import helpLinks from './helpLinks.scss'; const opacitySpringConfig = { stiffness: 300, damping: 20 }; const transformSpringConfig = { stiffness: 500, damping: 50, precision: 0.5 }; const changeContextSpringConfig = { stiffness: 500, damping: 20, precision: 0.5, }; const { helpLinks: helpLinksStyles } = helpLinks; type PanelId = string; /** * Definition of relation between contexts and panels * * Each sub-array is context. Each sub-array item is panel * * This definition declares animations between panels: * - The animation between panels from different contexts will be along Y axe (height toggling) * - The animation between panels from the same context will be along X axe (sliding) * - Panel index defines the direction of X transition of both panels * (e.g. the panel with lower index will slide from left side, and with greater from right side) */ const contexts: Array> = [ ['login', 'password', 'forgotPassword', 'mfa', 'recoverPassword'], ['register', 'activation', 'resendActivation'], ['acceptRules'], ['chooseAccount', 'permissions'], ]; // eslint-disable-next-line if (process.env.NODE_ENV !== 'production') { // test panel uniquenes between contexts // TODO: it may be moved to tests in future contexts.reduce((acc, context) => { context.forEach((panel) => { if (acc[panel]) { throw new Error(`Panel ${panel} is already exists in context ${JSON.stringify(acc[panel])}`); } acc[panel] = context; }); return acc; }, {} as Record>); } type ValidationError = | string | { type: string; payload: Record; }; interface AnimationStyle extends PlainStyle { opacitySpring: number; transformSpring: number; } interface AnimationData { Title: ReactElement; Body: ReactElement; Footer: ReactElement; Links: ReactNode; hasBackButton: boolean | ((props: Props) => boolean); } interface AnimationContext extends TransitionPlainStyle { key: PanelId; style: AnimationStyle; data: AnimationData; } interface OwnProps { Title: ReactElement; Body: ReactElement; Footer: ReactElement; Links?: ReactNode; className?: string; } interface Props extends OwnProps { // context props auth: AuthState; user: User; accounts: AccountsState; clearErrors: () => void; resolve: () => void; reject: () => void; setErrors: (errors: Record) => void; } interface State { contextHeight: number; panelId: PanelId | void; prevPanelId: PanelId | void; isHeightDirty: boolean; forceHeight: 1 | 0; direction: 'X' | 'Y'; formsHeights: Record; } // TODO: completely broken for RTL languages class PanelTransition extends React.PureComponent { state: State = { contextHeight: 0, panelId: this.props.Body && (this.props.Body.type as any).panelId, isHeightDirty: false, forceHeight: 0 as const, direction: 'X' as const, prevPanelId: undefined, formsHeights: {}, }; isHeightMeasured: boolean = false; wasAutoFocused: boolean = false; body: { autoFocus: () => void; onFormSubmit: () => void; } | null = null; timerIds: Array = []; // this is a list of a probably running timeouts to clean on unmount componentDidUpdate(prevProps: Props) { const nextPanel: PanelId = this.props.Body && (this.props.Body.type as any).panelId; const prevPanel: PanelId = prevProps.Body && (prevProps.Body.type as any).panelId; if (nextPanel !== prevPanel) { const direction = this.getDirection(nextPanel, prevPanel); const forceHeight = direction === 'Y' && nextPanel !== prevPanel ? 1 : 0; this.props.clearErrors(); this.setState({ direction, panelId: nextPanel, prevPanelId: prevPanel, forceHeight, }); if (forceHeight) { this.timerIds.push( // https://stackoverflow.com/a/51040768/5184751 window.setTimeout(() => { this.setState({ forceHeight: 0 }); }, 100), ); } } } componentWillUnmount() { this.timerIds.forEach((id) => clearTimeout(id)); this.timerIds = []; } render() { const { contextHeight, forceHeight } = this.state; const { Title, Body, Footer, Links, auth, user, clearErrors, resolve, reject } = this.props; if (this.props.children) { return this.props.children; } const { panelId, hasGoBack, }: { panelId: PanelId; hasGoBack: boolean; } = Body.type as any; const formHeight = this.state.formsHeights[panelId] || 0; // a hack to disable height animation on first render const { isHeightMeasured } = this; this.isHeightMeasured = isHeightMeasured || formHeight > 0; return ( {(items) => { const panels = items.filter(({ key }) => key !== 'common'); const [common] = items.filter(({ key }) => key === 'common'); const contentHeight = { overflow: 'hidden', height: forceHeight ? common.style.switchContextHeightSpring : 'auto', }; this.tryToAutoFocus(panels.length); const bodyHeight = { position: 'relative' as const, height: `${common.style.heightSpring}px`, }; return (
{panels.map((config) => this.getHeader(config))}
{panels.map((config) => this.getBody(config))}
{panels.map((config) => this.getFooter(config))}
{panels.map((config) => this.getLinks(config))}
); }}
); } onFormSubmit = (): void => { this.props.clearErrors(); this.body?.onFormSubmit(); }; onFormInvalid = (errors: Record): void => this.props.setErrors(errors); willEnter = (config: TransitionStyle): PlainStyle => { const transform = this.getTransformForPanel(config.key); return { transformSpring: transform, opacitySpring: 1, }; }; willLeave = (config: TransitionStyle): Style => { const transform = this.getTransformForPanel(config.key); return { transformSpring: spring(transform, transformSpringConfig), opacitySpring: spring(0, opacitySpringConfig), }; }; getTransformForPanel(key: PanelId): number { const { panelId, prevPanelId } = this.state; const fromLeft = -1; const fromRight = 1; const currentContext = contexts.find((context) => context.includes(key)); if (!currentContext) { throw new Error(`Can not find settings for ${key} panel`); } let sign = prevPanelId && panelId && currentContext.indexOf(panelId) > currentContext.indexOf(prevPanelId) ? fromRight : fromLeft; if (prevPanelId === key) { sign *= -1; } return sign * 100; } getDirection(next: PanelId, prev: PanelId): 'X' | 'Y' { const context = contexts.find((item) => item.includes(prev)); if (!context) { throw new Error(`Can not find context for transition ${prev} -> ${next}`); } return context.includes(next) ? 'X' : 'Y'; } onUpdateHeight = (height: number, key: PanelId): void => { this.setState({ formsHeights: { ...this.state.formsHeights, [key]: height, }, }); }; onUpdateContextHeight = (height: number): void => { this.setState({ contextHeight: height, }); }; onGoBack: MouseEventHandler = (event): void => { event.preventDefault(); authFlow.goBack(); }; /** * Tries to auto focus form fields after transition end * * @param {number} length number of panels transitioned */ tryToAutoFocus(length: number): void { if (!this.body) { return; } if (length === 1) { if (!this.wasAutoFocused) { this.body?.autoFocus(); } this.wasAutoFocused = true; } else if (this.wasAutoFocused) { this.wasAutoFocused = false; } } shouldMeasureHeight(): string { const { user, accounts, auth } = this.props; const { isHeightDirty } = this.state; const errorString = Object.values(auth.error || {}).reduce((acc, item) => { if (typeof item === 'string') { return acc + item; } return acc + item.type; }, '') as string; return [errorString, isHeightDirty, user.lang, accounts.available.length].join(''); } getHeader({ key, style, data }: TransitionPlainStyle): ReactElement { const { Title } = data as AnimationData; const { transformSpring } = style as unknown as AnimationStyle; let { hasBackButton } = data; if (typeof hasBackButton === 'function') { hasBackButton = hasBackButton(this.props); } const transitionStyle = { ...this.getDefaultTransitionStyles(key, style as unknown as AnimationStyle), opacity: 1, // reset default }; const scrollStyle = this.translate(transformSpring, 'Y'); const sideScrollStyle = { position: 'relative' as const, zIndex: 2, ...this.translate(-Math.abs(transformSpring)), }; const backButton = ( ); return (
{hasBackButton ? backButton : null}
{Title}
); } getBody({ key, style, data }: TransitionPlainStyle): ReactElement { const { Body } = data as AnimationData; const { transformSpring } = style as unknown as AnimationStyle; const { direction } = this.state; let transform = this.translate(transformSpring, direction); let verticalOrigin = 'top'; if (direction === 'Y') { verticalOrigin = 'bottom'; transform = {}; } const transitionStyle: CSSProperties = { ...this.getDefaultTransitionStyles(key, style as unknown as AnimationStyle), top: 'auto', // reset default [verticalOrigin]: 0, ...transform, }; return ( this.onUpdateHeight(height, key)} > {React.cloneElement(Body, { // @ts-ignore ref: (body) => { this.body = body; }, })} ); } getFooter({ key, style, data }: TransitionPlainStyle): ReactElement { const { Footer } = data as AnimationData; const transitionStyle = this.getDefaultTransitionStyles(key, style as unknown as AnimationStyle); return (
{Footer}
); } getLinks({ key, style, data }: TransitionPlainStyle): ReactElement { const { Links } = data as AnimationData; const transitionStyle = this.getDefaultTransitionStyles(key, style as unknown as AnimationStyle); return (
{Links}
); } getDefaultTransitionStyles( key: string, { opacitySpring }: Readonly, ): { position: 'absolute'; top: number; left: number; width: string; opacity: number; pointerEvents: 'none' | 'auto'; } { return { position: 'absolute', top: 0, left: 0, width: '100%', opacity: opacitySpring, pointerEvents: key === this.state.panelId ? 'auto' : 'none', }; } translate(value: number, direction: 'X' | 'Y' = 'X', unit: '%' | 'px' = '%'): CSSProperties { return { WebkitTransform: `translate${direction}(${value}${unit})`, transform: `translate${direction}(${value}${unit})`, }; } requestRedraw = (): Promise => new Promise((resolve) => this.setState({ isHeightDirty: true }, () => { this.setState({ isHeightDirty: false }); // wait till transition end this.timerIds.push(setTimeout(resolve, 200)); }), ); } export default connect( (state) => { const login = getLogin(state); let user = { ...state.user, }; if (login) { user = { ...user, isGuest: true, email: '', username: '', }; if (/[@.]/.test(login)) { user.email = login; } else { user.username = login; } } return { user, accounts: state.accounts, // need this, to re-render height auth: state.auth, resolve: authFlow.resolve.bind(authFlow), reject: authFlow.reject.bind(authFlow), }; }, { clearErrors: actions.clearErrors, setErrors: actions.setErrors, }, )(PanelTransition);