Skip to content

Commit

Permalink
feat: add profile form validation
Browse files Browse the repository at this point in the history
  • Loading branch information
TomatoVan committed Dec 13, 2023
1 parent 0dbf323 commit 5958427
Show file tree
Hide file tree
Showing 11 changed files with 105 additions and 17 deletions.
5 changes: 5 additions & 0 deletions extractedTranslations/en/profile.json
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,12 @@
"Error": "",
"Error with getting profile data": "",
"Error with profile ": "",
"Incorrect age": "",
"Incorrect country": "",
"Incorrect user data": "",
"No data": "",
"Profile": "",
"Server error": "",
"Try to reload page": "",
"Your age": "",
"Your avatar": "",
Expand Down
6 changes: 6 additions & 0 deletions extractedTranslations/en/translation.json
Original file line number Diff line number Diff line change
@@ -1,11 +1,17 @@
{
"": "",
"Cancel": "",
"Choose country": "",
"Choose currency": "",
"Edit": "",
"Incorrect age": "",
"Incorrect country": "",
"Incorrect user data": "",
"No data": "",
"PROFILE PAGE": "",
"Profile": "",
"Save": "",
"Server error": "",
"auth_form": "",
"auth_password": "",
"auth_username": "",
Expand Down
10 changes: 8 additions & 2 deletions json-server/db.json
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,13 @@
}
],
"profile": {
"currency": "RUB",
"country": "Russia"
"currency": "USD",
"country": "Belarus",
"first": "Leon",
"lastname": "Bobrov",
"age": 22,
"city": "Yaroslavl",
"avatar": "",
"username": "hello"
}
}
8 changes: 4 additions & 4 deletions src/entities/Currency/model/types/currency.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
export enum Currency {
"RUB" = "RUB",
"EUR" = "EUR",
"USD" = "USD",
}
RUB = 'RUB',
EUR = 'EUR',
USD = 'USD',
}
3 changes: 2 additions & 1 deletion src/entities/Profile/index.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
export { Profile, ProfileSchema } from './model/types/profile';
export { Profile, ProfileSchema, ValidateProfileError } from './model/types/profile';

export { profileActions, profileReducer } from './model/slice/profileSlice';
export { fetchProfileData } from './model/services/fetchProfileData/fetchProfileData';
Expand All @@ -10,3 +10,4 @@ export { getProfileIsLoading } from './model/selectors/getProfileIsLoading/getPr
export { getProfileReadonly } from './model/selectors/getProfileReadonly/getProfileReadonly';
export { getProfileForm } from './model/selectors/getProfileForm/getProfileForm';
export { updateProfileData } from './model/services/updateProfileData/updateProfileData';
export { getProfileValidateErrors } from './model/selectors/getProfileValidateErrors/getProfileValidateErrors';
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
import { StateSchema } from 'app/providers/StoreProvider';

export const getProfileValidateErrors = (state: StateSchema) => state.profile?.validateErrors;
Original file line number Diff line number Diff line change
@@ -1,22 +1,28 @@
import { createAsyncThunk } from '@reduxjs/toolkit';
import { ThunkConfig } from 'app/providers/StoreProvider';
import { getProfileForm } from '../../selectors/getProfileForm/getProfileForm';
import { Profile } from '../../types/profile';
import { Profile, ValidateProfileError } from '../../types/profile';
import { validateProfileData } from '../validateProfileData/validateProfileData';

export const updateProfileData = createAsyncThunk<
Profile,
void,
ThunkConfig<string>
Profile,
void,
ThunkConfig<ValidateProfileError[]>
>('profile/updateProfileData', async (_, thunkAPI) => {
const { extra, rejectWithValue, getState } = thunkAPI;

const formData = getProfileForm(getState());

const errors = validateProfileData(formData);
if (errors.length) {
return rejectWithValue(errors);
}

try {
const response = await extra.api.put<Profile>('/profile', formData);

return response.data;
} catch (error) {
return rejectWithValue('error');
return rejectWithValue([ValidateProfileError.SERVER_ERROR]);
}
});
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
import { Profile, ValidateProfileError } from '../../types/profile';

export const validateProfileData = (profile: Profile | undefined) => {
if (!profile) {
return [ValidateProfileError.NO_DATA];
}
const {
first, lastname, age, country,
} = profile;

const errors: ValidateProfileError[] = [];

if (!first || !lastname) {
errors.push(ValidateProfileError.INCORRECT_USER_DATA);
}

if (!age || !Number.isInteger(age)) {
errors.push(ValidateProfileError.INCORRECT_AGE);
}

if (!country) {
errors.push(ValidateProfileError.INCORRECT_COUNTRY);
}
return errors;
};
6 changes: 4 additions & 2 deletions src/entities/Profile/model/slice/profileSlice.ts
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ export const profileSlice = createSlice({
cancelEdit: (state) => {
state.readonly = true;
state.form = state.data;
state.validateErrors = undefined;
},
updateProfile: (state, action:PayloadAction<Profile>) => {
state.form = {
Expand Down Expand Up @@ -47,7 +48,7 @@ export const profileSlice = createSlice({
state.error = action.payload;
})
.addCase(updateProfileData.pending, (state) => {
state.error = undefined;
state.validateErrors = undefined;
state.isLoading = true;
})
.addCase(
Expand All @@ -57,11 +58,12 @@ export const profileSlice = createSlice({
state.data = action.payload;
state.form = action.payload;
state.readonly = true;
state.validateErrors = undefined;
},
)
.addCase(updateProfileData.rejected, (state, action) => {
state.isLoading = false;
state.error = action.payload;
state.validateErrors = action.payload;
});
},
});
Expand Down
9 changes: 9 additions & 0 deletions src/entities/Profile/model/types/profile.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,14 @@
import { Currency } from 'entities/Currency/model/types/currency';
import { Country } from 'entities/Country';

export enum ValidateProfileError {
INCORRECT_USER_DATA = 'INCORRECT_USER_DATA',
INCORRECT_AGE = 'INCORRECT_AGE',
INCORRECT_COUNTRY = 'INCORRECT_COUNTRY',
NO_DATA = 'NO_DATA',
SERVER_ERROR = 'SERVER_ERROR',
}

export interface Profile {
first?: string,
lastname?: string,
Expand All @@ -18,4 +26,5 @@ export interface ProfileSchema {
isLoading: boolean;
error?: string;
readonly: boolean;
validateErrors?: ValidateProfileError[];
}
31 changes: 28 additions & 3 deletions src/pages/ProfilePage/ui/ProfilePage.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -7,14 +7,20 @@ import {
import { useSelector } from 'react-redux';
import { Currency } from 'entities/Currency';
import { Country } from 'entities/Country';
import { Text, TextTheme } from 'shared/ui/Text/Text';
import { useTranslation } from 'react-i18next';
import { ProfilePageHeader } from './ProfilePageHeader/ProfilePageHeader';
import { getProfileIsLoading } from '../../../entities/Profile/model/selectors/getProfileIsLoading/getProfileIsLoading';
import { getProfileError } from '../../../entities/Profile/model/selectors/getProfileError/getProfileError';
import { useAppDispatch } from '../../../shared/lib/hook/useAppDispatch/useAppDispatch';
import {
fetchProfileData, getProfileForm, getProfileReadonly, profileActions,
fetchProfileData,
getProfileForm,
getProfileReadonly,
getProfileValidateErrors,
profileActions,
ProfileCard,
profileReducer,
profileReducer, ValidateProfileError,
} from '../../../entities/Profile';

const reducers: ReducersList = {
Expand All @@ -28,10 +34,22 @@ interface ProfilePageProps {
const ProfilePage = memo(({ className }: ProfilePageProps) => {
const dispatch = useAppDispatch();

const { t } = useTranslation('profile');

const formData = useSelector(getProfileForm);
const isLoading = useSelector(getProfileIsLoading);
const error = useSelector(getProfileError);
const readonly = useSelector(getProfileReadonly);
const validateErrors = useSelector(getProfileValidateErrors);

const validateErrorTranslates = {
[ValidateProfileError.SERVER_ERROR]: t('Server error'),
[ValidateProfileError.NO_DATA]: t('No data'),
[ValidateProfileError.INCORRECT_AGE]: t('Incorrect age'),
[ValidateProfileError.INCORRECT_USER_DATA]: t('Incorrect user data'),
[ValidateProfileError.INCORRECT_COUNTRY]: t('Incorrect country'),

};

useEffect(() => {
dispatch(fetchProfileData());
Expand All @@ -53,7 +71,7 @@ const ProfilePage = memo(({ className }: ProfilePageProps) => {

const onChangeAge = useCallback(
(value?: string) => {
const regex = /^[0-9]+$/;
const regex = /^[0-9]*$/;
if (regex.test(value || '')) {
dispatch(profileActions.updateProfile({ age: Number(value || 0) }));
}
Expand Down Expand Up @@ -100,6 +118,13 @@ const ProfilePage = memo(({ className }: ProfilePageProps) => {
<DynamicModuleLoader reducers={reducers} removeAfterUnmount>
<div className={classNames('', {}, [className])}>
<ProfilePageHeader />
{validateErrors?.length && validateErrors.map((err) => (
<Text
key={err}
theme={TextTheme.ERROR}
text={validateErrorTranslates[err]}
/>
))}
<ProfileCard
data={formData}
isLoading={isLoading}
Expand Down

0 comments on commit 5958427

Please sign in to comment.