import React, { ComponentType, useCallback } from 'react'; import { Route, Switch, Redirect } from 'react-router-dom'; import { connect } from 'app/functions'; import { Dispatch } from 'app/types'; import { refreshUserData } from 'app/components/accounts/actions'; import { create as createPopup } from 'app/components/ui/popup/actions'; import PasswordRequestForm from 'app/components/profile/passwordRequestForm'; import logger from 'app/services/logger'; import { browserHistory } from 'app/services/history'; import { FooterMenu } from 'app/components/footerMenu'; import { FormModel } from 'app/components/ui/form'; import { Provider } from 'app/components/profile/Context'; import { ComponentLoader } from 'app/components/ui/loader'; import styles from './profile.scss'; import Profile from 'app/pages/profile/ProfilePage'; import ChangePasswordPage from 'app/pages/profile/ChangePasswordPage'; import ChangeUsernamePage from 'app/pages/profile/ChangeUsernamePage'; import ChangeEmailPage from 'app/pages/profile/ChangeEmailPage'; import MultiFactorAuthPage from 'app/pages/profile/MultiFactorAuthPage'; interface Props { userId: number; onSubmit: (options: { form: FormModel; sendData: () => Promise }) => Promise; refreshUserData: () => Promise; } const ProfileController: ComponentType = ({ userId, onSubmit, refreshUserData }) => { const goToProfile = useCallback(async () => { await refreshUserData(); browserHistory.push('/'); }, [refreshUserData]); return (
}>
); }; export default connect( (state) => ({ userId: state.user.id!, }), { refreshUserData, onSubmit: ({ form, sendData }: { form: FormModel; sendData: () => Promise }) => (dispatch: Dispatch) => { form.beginLoading(); return sendData() .catch((resp) => { const requirePassword = resp.errors && !!resp.errors.password; // prevalidate user input, because requestPassword popup will block the // entire form from input, so it must be valid if (resp.errors) { delete resp.errors.password; if (resp.errors.email && resp.data && resp.data.canRepeatIn) { resp.errors.email = { type: resp.errors.email, payload: { msLeft: resp.data.canRepeatIn * 1000, }, }; } if (Object.keys(resp.errors).length) { form.setErrors(resp.errors); return Promise.reject(resp); } if (requirePassword) { return requestPassword(form); } } return Promise.reject(resp); }) .catch((resp) => { if (!resp || !resp.errors) { logger.warn('Unexpected profile editing error', { resp, }); } else { return Promise.reject(resp); } }) .finally(() => form.endLoading()); function requestPassword(form: FormModel) { return new Promise((resolve, reject) => { dispatch( createPopup({ Popup(props: { onClose: () => Promise }) { const onSubmit = () => { form.beginLoading(); sendData() .then(resolve) .then(props.onClose) .catch((resp) => { if (resp.errors) { form.setErrors(resp.errors); const parentFormHasErrors = Object.keys(resp.errors).filter((name) => name !== 'password') .length > 0; if (parentFormHasErrors) { // something wrong with parent form, hiding popup and show that form props.onClose(); reject(resp); logger.warn( 'Profile: can not submit password popup due to errors in source form', { resp }, ); } } else { return Promise.reject(resp); } }) .finally(() => form.endLoading()); }; return ; }, // TODO: this property should be automatically extracted from the popup's isClosable prop disableOverlayClose: true, }), ); }); } }, }, )(ProfileController);