fear: change requests to RTK query #78
| @ -25,10 +25,16 @@ | |||||||
|   "dry-wash.arm.master.table.header.phone": "Телефон", |   "dry-wash.arm.master.table.header.phone": "Телефон", | ||||||
|   "dry-wash.arm.master.table.header.actions": "Действия", |   "dry-wash.arm.master.table.header.actions": "Действия", | ||||||
|   "dry-wash.arm.master.table.actionsMenu.delete": "Удалить мастера", |   "dry-wash.arm.master.table.actionsMenu.delete": "Удалить мастера", | ||||||
|  |   "dry-wash.arm.master.table.actionsMenu.toast.success": "Мастер удалён", | ||||||
|  |   "dry-wash.arm.master.table.actionsMenu.toast.error.title": "Ошибка!", | ||||||
|  |   "dry-wash.arm.master.table.actionsMenu.toast.error.description": "Не удалось удалить мастера. Попробуйте ещё раз.", | ||||||
|   "dry-wash.arm.master.schedule.empty": "Свободен", |   "dry-wash.arm.master.schedule.empty": "Свободен", | ||||||
|   "dry-wash.arm.master.editable.aria.cancel": "Отменить изменения", |   "dry-wash.arm.master.editable.aria.cancel": "Отменить изменения", | ||||||
|   "dry-wash.arm.master.editable.aria.save": "Сохранить изменения", |   "dry-wash.arm.master.editable.aria.save": "Сохранить изменения", | ||||||
|   "dry-wash.arm.master.editable.aria.edit": "Редактировать", |   "dry-wash.arm.master.editable.aria.edit": "Редактировать", | ||||||
|  |   "dry-wash.arm.master.editable.toast.success": "Успешно!", | ||||||
|  |   "dry-wash.arm.master.editable.toast.error.description": "Не удалось обновить данные", | ||||||
|  |   "dry-wash.arm.master.editable.toast.error.title": "Ошибка!", | ||||||
|   "dry-wash.arm.master.drawer.title": "Добавить нового мастера", |   "dry-wash.arm.master.drawer.title": "Добавить нового мастера", | ||||||
|   "dry-wash.arm.master.drawer.inputName.label": "ФИО", |   "dry-wash.arm.master.drawer.inputName.label": "ФИО", | ||||||
|   "dry-wash.arm.master.drawer.inputName.placeholder": "Введите ФИО", |   "dry-wash.arm.master.drawer.inputName.placeholder": "Введите ФИО", | ||||||
|  | |||||||
| @ -1,20 +1,51 @@ | |||||||
| import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react'; | import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react'; | ||||||
| import { getConfigValue } from '@brojs/cli'; | import { getConfigValue } from '@brojs/cli'; | ||||||
|  | import dayjs from 'dayjs'; | ||||||
| 
 | 
 | ||||||
| import { Master } from '../../models/api/master'; | import { Master, OrderArm } from '../../models/api'; | ||||||
| 
 | 
 | ||||||
| import { extractBodyFromResponse } from './utils'; | import { extractBodyFromResponse } from './utils'; | ||||||
| 
 | 
 | ||||||
|  | export type UpdateMasterPayload = Required<Pick<Master, 'id'>> & | ||||||
|  |   Partial<Omit<Master, 'id'>>; | ||||||
|  | 
 | ||||||
|  | type UpdateOrderProps = Required<Pick<OrderArm, 'id'>> & | ||||||
|  |   Partial<Pick<OrderArm, 'status' | 'notes'>> & { | ||||||
|  |     master?: string; | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
| export const api = createApi({ | export const api = createApi({ | ||||||
|   reducerPath: 'api', |   reducerPath: 'api', | ||||||
|   baseQuery: fetchBaseQuery({ baseUrl: getConfigValue('dry-wash.api') }), |   baseQuery: fetchBaseQuery({ baseUrl: getConfigValue('dry-wash.api') }), | ||||||
|   tagTypes: ['Masters'], |   tagTypes: ['Masters', 'Orders'], | ||||||
|   endpoints: (builder) => ({ |   endpoints: (builder) => ({ | ||||||
|     getMasters: builder.query<Master[], void>({ |     getMasters: builder.query<Master[], void>({ | ||||||
|       query: () => ({ url: '/arm/masters' }), |       query: () => ({ url: '/arm/masters' }), | ||||||
|       transformResponse: extractBodyFromResponse<Master[]>, |       transformResponse: extractBodyFromResponse<Master[]>, | ||||||
|       providesTags: ['Masters'], |       providesTags: ['Masters'], | ||||||
|     }), |     }), | ||||||
|  |     updateOrders: builder.mutation<void, UpdateOrderProps>({ | ||||||
|  |       query: ({ id, status, notes, master }) => ({ | ||||||
|  |         url: `/order/${id}`, | ||||||
|  |         method: 'PATCH', | ||||||
|  |         body: { status, notes, master }, | ||||||
|  |       }), | ||||||
|  |       invalidatesTags: ['Orders'], | ||||||
|  |     }), | ||||||
|  |     getOrders: builder.query<OrderArm[], { date: Date }>({ | ||||||
|  |       query: ({ date }) => { | ||||||
|  |         const startDate = dayjs(date).startOf('day').toISOString(); | ||||||
|  |         const endDate = dayjs(date).endOf('day').toISOString(); | ||||||
|  |         return { | ||||||
|  |           url: '/arm/orders', | ||||||
|  |           method: 'POST', | ||||||
|  |           body: { startDate, endDate }, | ||||||
|  |         }; | ||||||
|  |       }, | ||||||
|  |       transformResponse: extractBodyFromResponse<OrderArm[]>, | ||||||
|  |       providesTags: ['Orders'], | ||||||
|  |     }), | ||||||
|  | 
 | ||||||
|     addMaster: builder.mutation<void, Pick<Master, 'name' | 'phone'>>({ |     addMaster: builder.mutation<void, Pick<Master, 'name' | 'phone'>>({ | ||||||
|       query: (master) => ({ |       query: (master) => ({ | ||||||
|         url: '/arm/masters', |         url: '/arm/masters', | ||||||
| @ -23,5 +54,29 @@ export const api = createApi({ | |||||||
|       }), |       }), | ||||||
|       invalidatesTags: ['Masters'], |       invalidatesTags: ['Masters'], | ||||||
|     }), |     }), | ||||||
|  |     deleteMaster: builder.mutation<void, { id: string }>({ | ||||||
|  |       query: ({ id }) => ({ | ||||||
|  |         url: `/arm/masters/${id}`, | ||||||
|  |         method: 'DELETE', | ||||||
|  |       }), | ||||||
|  |       invalidatesTags: ['Masters'], | ||||||
|  |     }), | ||||||
|  |     updateMaster: builder.mutation<void, UpdateMasterPayload>({ | ||||||
|  |       query: ({ id, name, phone }) => ({ | ||||||
|  |         url: `/arm/masters/${id}`, | ||||||
|  |         method: 'PATCH', | ||||||
|  |         body: { name, phone }, | ||||||
|  |       }), | ||||||
|  |       invalidatesTags: ['Masters'], | ||||||
|  |     }), | ||||||
|   }), |   }), | ||||||
| }); | }); | ||||||
|  | 
 | ||||||
|  | export const { | ||||||
|  |   useGetMastersQuery, | ||||||
|  |   useAddMasterMutation, | ||||||
|  |   useDeleteMasterMutation, | ||||||
|  |   useUpdateMasterMutation, | ||||||
|  |   useGetOrdersQuery, | ||||||
|  |   useUpdateOrdersMutation, | ||||||
|  | } = api; | ||||||
|  | |||||||
							
								
								
									
										139
									
								
								src/api/arm.ts
									
									
									
									
									
								
							
							
						
						
									
										139
									
								
								src/api/arm.ts
									
									
									
									
									
								
							| @ -1,139 +0,0 @@ | |||||||
| import { getConfigValue } from '@brojs/cli'; |  | ||||||
| import dayjs from 'dayjs'; |  | ||||||
| 
 |  | ||||||
| enum ArmEndpoints { |  | ||||||
|   ORDERS = '/arm/orders', |  | ||||||
|   MASTERS = '/arm/masters', |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| const armService = () => { |  | ||||||
|   const endpoint = getConfigValue('dry-wash.api'); |  | ||||||
| 
 |  | ||||||
|   const fetchOrders = async ({ date }: { date: Date }) => { |  | ||||||
|     const startDate = dayjs(date).startOf('day').toISOString(); |  | ||||||
|     const endDate = dayjs(date).endOf('day').toISOString(); |  | ||||||
| 
 |  | ||||||
|     const response = await fetch(`${endpoint}${ArmEndpoints.ORDERS}`, { |  | ||||||
|       method: 'POST', |  | ||||||
|       headers: { |  | ||||||
|         'Content-Type': 'application/json', |  | ||||||
|       }, |  | ||||||
|       body: JSON.stringify({ startDate, endDate }), |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     if (!response.ok) { |  | ||||||
|       throw new Error(`Failed to fetch orders: ${response.status}`); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return await response.json(); |  | ||||||
|   }; |  | ||||||
| 
 |  | ||||||
|   const fetchMasters = async () => { |  | ||||||
|     const response = await fetch(`${endpoint}${ArmEndpoints.MASTERS}`); |  | ||||||
| 
 |  | ||||||
|     if (!response.ok) { |  | ||||||
|       throw new Error(`Failed to fetch masters: ${response.status}`); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return await response.json(); |  | ||||||
|   }; |  | ||||||
| 
 |  | ||||||
|   const addMaster = async ({ |  | ||||||
|     name, |  | ||||||
|     phone, |  | ||||||
|   }: { |  | ||||||
|     name: string; |  | ||||||
|     phone: string; |  | ||||||
|   }) => { |  | ||||||
|     const response = await fetch(`${endpoint}${ArmEndpoints.MASTERS}`, { |  | ||||||
|       method: 'POST', |  | ||||||
|       headers: { |  | ||||||
|         'Content-Type': 'application/json', |  | ||||||
|       }, |  | ||||||
|       body: JSON.stringify({ name, phone }), |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     if (!response.ok) { |  | ||||||
|       throw new Error(`Failed to fetch masters: ${response.status}`); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return await response.json(); |  | ||||||
|   }; |  | ||||||
| 
 |  | ||||||
|   const deleteMaster = async ({ id }: { id: string }) => { |  | ||||||
|     const response = await fetch(`${endpoint}${ArmEndpoints.MASTERS}/${id}`, { |  | ||||||
|       method: 'DELETE', |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     if (!response.ok) { |  | ||||||
|       throw new Error(`Failed to fetch masters: ${response.status}`); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return await response.json(); |  | ||||||
|   }; |  | ||||||
| 
 |  | ||||||
|   const updateOrders = async ({ |  | ||||||
|     id, |  | ||||||
|     status, |  | ||||||
|     notes, |  | ||||||
|     masterId, |  | ||||||
|   }: { |  | ||||||
|     id: string; |  | ||||||
|     status?: string; |  | ||||||
|     notes?: string; |  | ||||||
|     masterId?: string; |  | ||||||
|   }) => { |  | ||||||
|     const body = JSON.stringify({ status, notes, masterId }); |  | ||||||
| 
 |  | ||||||
|     const response = await fetch(`${endpoint}/order/${id}`, { |  | ||||||
|       method: 'PATCH', |  | ||||||
|       headers: { |  | ||||||
|         'Content-Type': 'application/json', |  | ||||||
|       }, |  | ||||||
|       body, |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     if (!response.ok) { |  | ||||||
|       throw new Error(`Failed to fetch update masters: ${response.status}`); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return await response.json(); |  | ||||||
|   }; |  | ||||||
| 
 |  | ||||||
|   const updateMaster = async ({ |  | ||||||
|     id, |  | ||||||
|     name, |  | ||||||
|     phone, |  | ||||||
|   }: { |  | ||||||
|     id: string; |  | ||||||
|     name?: string; |  | ||||||
|     phone?: string; |  | ||||||
|   }) => { |  | ||||||
|     const body = JSON.stringify({ name, phone }); |  | ||||||
| 
 |  | ||||||
|     const response = await fetch(`${endpoint}${ArmEndpoints.MASTERS}/${id}`, { |  | ||||||
|       method: 'PATCH', |  | ||||||
|       headers: { |  | ||||||
|         'Content-Type': 'application/json', |  | ||||||
|       }, |  | ||||||
|       body, |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     if (!response.ok) { |  | ||||||
|       throw new Error(`Failed to fetch update masters: ${response.status}`); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return await response.json(); |  | ||||||
|   }; |  | ||||||
| 
 |  | ||||||
|   return { |  | ||||||
|     fetchOrders, |  | ||||||
|     fetchMasters, |  | ||||||
|     addMaster, |  | ||||||
|     deleteMaster, |  | ||||||
|     updateMaster, |  | ||||||
|     updateOrders, |  | ||||||
|   }; |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| export { armService, ArmEndpoints }; |  | ||||||
| @ -1,4 +1,4 @@ | |||||||
| import React, { useState } from 'react'; | import React, { useEffect, useState } from 'react'; | ||||||
| import { | import { | ||||||
|   Editable, |   Editable, | ||||||
|   EditableInput, |   EditableInput, | ||||||
| @ -9,63 +9,51 @@ import { | |||||||
|   useEditableControls, |   useEditableControls, | ||||||
|   ButtonGroup, |   ButtonGroup, | ||||||
|   Stack, |   Stack, | ||||||
|   useToast, |  | ||||||
| } from '@chakra-ui/react'; | } from '@chakra-ui/react'; | ||||||
| import { CheckIcon, CloseIcon, EditIcon } from '@chakra-ui/icons'; | import { CheckIcon, CloseIcon, EditIcon } from '@chakra-ui/icons'; | ||||||
| import { useTranslation } from 'react-i18next'; | import { useTranslation } from 'react-i18next'; | ||||||
| 
 | 
 | ||||||
|  | import { useUpdateMasterMutation } from '../../__data__/service/api'; | ||||||
|  | import useShowToast from '../../hooks/useShowToast'; | ||||||
|  | 
 | ||||||
| interface EditableWrapperProps { | interface EditableWrapperProps { | ||||||
|   value: string; |   value: string; | ||||||
|   onSubmit: ({ |   fieldName: 'phone' | 'name'; | ||||||
|     id, |  | ||||||
|     name, |  | ||||||
|     phone, |  | ||||||
|   }: { |  | ||||||
|     id: string; |  | ||||||
|     name?: string; |  | ||||||
|     phone?: string; |  | ||||||
|   }) => Promise<unknown>; |  | ||||||
|   as: 'phone' | 'name'; |  | ||||||
|   id: string; |   id: string; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const EditableWrapper = ({ value, onSubmit, as, id }: EditableWrapperProps) => { | const EditableWrapper = ({ value, fieldName, id }: EditableWrapperProps) => { | ||||||
|  |   const [updateMaster, { isError, isSuccess, error }] = | ||||||
|  |     useUpdateMasterMutation(); | ||||||
|  | 
 | ||||||
|   const { t } = useTranslation('~', { |   const { t } = useTranslation('~', { | ||||||
|     keyPrefix: 'dry-wash.arm.master.editable', |     keyPrefix: 'dry-wash.arm.master.editable', | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|   const toast = useToast(); |   const showToast = useShowToast(); | ||||||
|   const [currentValue, setCurrentValue] = useState<string>(value); |   const [currentValue, setCurrentValue] = useState<string>(value); | ||||||
| 
 | 
 | ||||||
|   const handleSubmit = async (newValue: string) => { |   const handleSubmit = async (newValue: string) => { | ||||||
|     if (currentValue === newValue) return; |     if (currentValue === newValue) return; | ||||||
| 
 | 
 | ||||||
|     try { |     await updateMaster({ id, [fieldName]: newValue }); | ||||||
|       await onSubmit({ id, [as]: newValue }); |  | ||||||
| 
 | 
 | ||||||
|       setCurrentValue(newValue); |     setCurrentValue(newValue); | ||||||
| 
 |  | ||||||
|       toast({ |  | ||||||
|         title: 'Успешно!', |  | ||||||
|         description: 'Данные обновлены.', |  | ||||||
|         status: 'success', |  | ||||||
|         duration: 2000, |  | ||||||
|         isClosable: true, |  | ||||||
|         position: 'top-right', |  | ||||||
|       }); |  | ||||||
|     } catch (error) { |  | ||||||
|       toast({ |  | ||||||
|         title: 'Ошибка!', |  | ||||||
|         description: 'Не удалось обновить данные.', |  | ||||||
|         status: 'error', |  | ||||||
|         duration: 2000, |  | ||||||
|         isClosable: true, |  | ||||||
|         position: 'top-right', |  | ||||||
|       }); |  | ||||||
|       console.error('Ошибка при обновлении данных:', error); |  | ||||||
|     } |  | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|  |   useEffect(() => { | ||||||
|  |     if (isSuccess) { | ||||||
|  |       showToast(t('toast.success'), 'success'); | ||||||
|  |     } | ||||||
|  |   }, [isSuccess]); | ||||||
|  | 
 | ||||||
|  |   useEffect(() => { | ||||||
|  |     if (isError) { | ||||||
|  |       showToast(t('toast.error.title'), 'error', t('toast.error.description')); | ||||||
|  |       console.error(t('toast.error.description'), error); | ||||||
|  |     } | ||||||
|  |   }, [isError, error]); | ||||||
|  | 
 | ||||||
|   function EditableControls() { |   function EditableControls() { | ||||||
|     const { |     const { | ||||||
|       isEditing, |       isEditing, | ||||||
|  | |||||||
| @ -1,16 +1,16 @@ | |||||||
| import React from 'react'; | import React, { useEffect } from 'react'; | ||||||
| import { | import { | ||||||
|   Menu, |   Menu, | ||||||
|   MenuButton, |   MenuButton, | ||||||
|   MenuList, |   MenuList, | ||||||
|   MenuItem, |   MenuItem, | ||||||
|   IconButton, |   IconButton, | ||||||
|   useToast, |  | ||||||
| } from '@chakra-ui/react'; | } from '@chakra-ui/react'; | ||||||
| import { EditIcon } from '@chakra-ui/icons'; | import { EditIcon } from '@chakra-ui/icons'; | ||||||
| import { useTranslation } from 'react-i18next'; | import { useTranslation } from 'react-i18next'; | ||||||
| 
 | 
 | ||||||
| import { armService } from '../../api/arm'; | import { useDeleteMasterMutation } from '../../__data__/service/api'; | ||||||
|  | import useShowToast from '../../hooks/useShowToast'; | ||||||
| 
 | 
 | ||||||
| interface MasterActionsMenu { | interface MasterActionsMenu { | ||||||
|   id: string; |   id: string; | ||||||
| @ -21,38 +21,35 @@ const MasterActionsMenu = ({ id }: MasterActionsMenu) => { | |||||||
|     keyPrefix: 'dry-wash.arm.master.table.actionsMenu', |     keyPrefix: 'dry-wash.arm.master.table.actionsMenu', | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|   const { deleteMaster } = armService(); |   const [deleteMaster, { isSuccess, isError, error, isLoading }] = | ||||||
|   const toast = useToast(); |     useDeleteMasterMutation(); | ||||||
|  | 
 | ||||||
|  |   const showToast = useShowToast(); | ||||||
| 
 | 
 | ||||||
|   const handleClickDelete = async () => { |   const handleClickDelete = async () => { | ||||||
|     try { |     await deleteMaster({ id }); | ||||||
|       await deleteMaster({ id }); |   }; | ||||||
|       toast({ | 
 | ||||||
|         title: 'Мастер удалён.', |   useEffect(() => { | ||||||
|         description: `Мастер с ID "${id}" успешно удалён.`, |     if (isSuccess) { | ||||||
|         status: 'success', |       showToast(t('toast.success'), 'success'); | ||||||
|         duration: 5000, |     } | ||||||
|         isClosable: true, |   }, [isSuccess]); | ||||||
|         position: 'top-right', | 
 | ||||||
|       }); |   useEffect(() => { | ||||||
|     } catch (error) { |     if (isError) { | ||||||
|       toast({ |       showToast(t('toast.error.title'), 'error', t('toast.error.description')); | ||||||
|         title: 'Ошибка удаления мастера.', |  | ||||||
|         description: 'Не удалось удалить мастера. Попробуйте ещё раз.', |  | ||||||
|         status: 'error', |  | ||||||
|         duration: 5000, |  | ||||||
|         isClosable: true, |  | ||||||
|         position: 'top-right', |  | ||||||
|       }); |  | ||||||
|       console.error(error); |       console.error(error); | ||||||
|     } |     } | ||||||
|   }; |   }, [isError]); | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <Menu> |     <Menu> | ||||||
|       <MenuButton icon={<EditIcon />} as={IconButton} variant='outline' /> |       <MenuButton icon={<EditIcon />} as={IconButton} variant='outline' /> | ||||||
|       <MenuList> |       <MenuList> | ||||||
|         <MenuItem onClick={handleClickDelete}>{t('delete')}</MenuItem> |         <MenuItem onClick={handleClickDelete} isDisabled={isLoading}> | ||||||
|  |           {t('delete')} | ||||||
|  |         </MenuItem> | ||||||
|       </MenuList> |       </MenuList> | ||||||
|     </Menu> |     </Menu> | ||||||
|   ); |   ); | ||||||
|  | |||||||
| @ -12,7 +12,6 @@ import { | |||||||
|   DrawerFooter, |   DrawerFooter, | ||||||
|   DrawerHeader, |   DrawerHeader, | ||||||
|   DrawerOverlay, |   DrawerOverlay, | ||||||
|   useToast, |  | ||||||
|   InputGroup, |   InputGroup, | ||||||
|   InputLeftElement, |   InputLeftElement, | ||||||
|   FormErrorMessage, |   FormErrorMessage, | ||||||
| @ -20,9 +19,9 @@ import { | |||||||
| import { useTranslation } from 'react-i18next'; | import { useTranslation } from 'react-i18next'; | ||||||
| import { PhoneIcon } from '@chakra-ui/icons'; | import { PhoneIcon } from '@chakra-ui/icons'; | ||||||
| 
 | 
 | ||||||
| import { api } from '../../__data__/service/api'; | import { useAddMasterMutation } from '../../__data__/service/api'; | ||||||
| import showToast from '../../helpers/showToast'; |  | ||||||
| import { DrawerInputs } from '../../models/arm/form'; | import { DrawerInputs } from '../../models/arm/form'; | ||||||
|  | import useShowToast from '../../hooks/useShowToast'; | ||||||
| 
 | 
 | ||||||
| interface MasterDrawerProps { | interface MasterDrawerProps { | ||||||
|   isOpen: boolean; |   isOpen: boolean; | ||||||
| @ -50,28 +49,19 @@ const MasterDrawer = ({ isOpen, onClose }: MasterDrawerProps) => { | |||||||
|     const isEmptyFields = trimMaster.name === '' || trimMaster.phone === ''; |     const isEmptyFields = trimMaster.name === '' || trimMaster.phone === ''; | ||||||
| 
 | 
 | ||||||
|     if (isEmptyFields) { |     if (isEmptyFields) { | ||||||
|       showToast({ |       showToast(t('toast.error.base'), 'error', t('toast.error.empty-fields')); | ||||||
|         toast, |  | ||||||
|         title: t('toast.error.base'), |  | ||||||
|         description: t('toast.error.empty-fields'), |  | ||||||
|         status: 'error', |  | ||||||
|       }); |  | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     await addMaster(trimMaster); |     await addMaster(trimMaster); | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|   const [addMaster, { error, isSuccess }] = api.useAddMasterMutation(); |   const [addMaster, { error, isSuccess }] = useAddMasterMutation(); | ||||||
|   const toast = useToast(); |   const showToast = useShowToast(); | ||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     if (isSuccess) { |     if (isSuccess) { | ||||||
|       showToast({ |       showToast(t('toast.create-master'), 'success'); | ||||||
|         toast, |  | ||||||
|         title: t('toast.create-master'), |  | ||||||
|         status: 'success', |  | ||||||
|       }); |  | ||||||
|       reset(); |       reset(); | ||||||
|       onClose(); |       onClose(); | ||||||
|     } |     } | ||||||
| @ -79,12 +69,11 @@ const MasterDrawer = ({ isOpen, onClose }: MasterDrawerProps) => { | |||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     if (error) { |     if (error) { | ||||||
|       showToast({ |       showToast( | ||||||
|         toast, |         t('toast.error.create-master'), | ||||||
|         title: t('toast.error.create-master'), |         'error', | ||||||
|         description: t('toast.error.create-master-details'), |         t('toast.error.create-master-details'), | ||||||
|         status: 'error', |       ); | ||||||
|       }); |  | ||||||
|       console.error(error); |       console.error(error); | ||||||
|     } |     } | ||||||
|   }, [error]); |   }, [error]); | ||||||
|  | |||||||
| @ -5,10 +5,8 @@ import { useTranslation } from 'react-i18next'; | |||||||
| import MasterActionsMenu from '../MasterActionsMenu'; | import MasterActionsMenu from '../MasterActionsMenu'; | ||||||
| import { getTimeSlot } from '../../lib'; | import { getTimeSlot } from '../../lib'; | ||||||
| import EditableWrapper from '../Editable/Editable'; | import EditableWrapper from '../Editable/Editable'; | ||||||
| import { armService } from '../../api/arm'; |  | ||||||
| 
 | 
 | ||||||
| const MasterItem = ({ name, phone, id, schedule }) => { | const MasterItem = ({ name, phone, id, schedule }) => { | ||||||
|   const { updateMaster } = armService(); |  | ||||||
|   const { t } = useTranslation('~', { |   const { t } = useTranslation('~', { | ||||||
|     keyPrefix: 'dry-wash.arm.master', |     keyPrefix: 'dry-wash.arm.master', | ||||||
|   }); |   }); | ||||||
| @ -16,12 +14,7 @@ const MasterItem = ({ name, phone, id, schedule }) => { | |||||||
|   return ( |   return ( | ||||||
|     <Tr> |     <Tr> | ||||||
|       <Td> |       <Td> | ||||||
|         <EditableWrapper |         <EditableWrapper id={id} fieldName={'name'} value={name} /> | ||||||
|           id={id} |  | ||||||
|           as={'name'} |  | ||||||
|           value={name} |  | ||||||
|           onSubmit={updateMaster} |  | ||||||
|         /> |  | ||||||
|       </Td> |       </Td> | ||||||
|       <Td> |       <Td> | ||||||
|         {schedule?.length > 0 ? ( |         {schedule?.length > 0 ? ( | ||||||
| @ -37,12 +30,7 @@ const MasterItem = ({ name, phone, id, schedule }) => { | |||||||
|         )} |         )} | ||||||
|       </Td> |       </Td> | ||||||
|       <Td> |       <Td> | ||||||
|         <EditableWrapper |         <EditableWrapper id={id} fieldName={'phone'} value={phone} /> | ||||||
|           id={id} |  | ||||||
|           as={'phone'} |  | ||||||
|           value={phone} |  | ||||||
|           onSubmit={updateMaster} |  | ||||||
|         /> |  | ||||||
|       </Td> |       </Td> | ||||||
|       <Td> |       <Td> | ||||||
|         <MasterActionsMenu id={id} /> |         <MasterActionsMenu id={id} /> | ||||||
|  | |||||||
| @ -10,7 +10,6 @@ import { | |||||||
|   Button, |   Button, | ||||||
|   useDisclosure, |   useDisclosure, | ||||||
|   Flex, |   Flex, | ||||||
|   useToast, |  | ||||||
|   Td, |   Td, | ||||||
|   Text, |   Text, | ||||||
|   Spinner, |   Spinner, | ||||||
| @ -19,7 +18,8 @@ import { useTranslation } from 'react-i18next'; | |||||||
| 
 | 
 | ||||||
| import MasterItem from '../MasterItem'; | import MasterItem from '../MasterItem'; | ||||||
| import MasterDrawer from '../MasterDrawer'; | import MasterDrawer from '../MasterDrawer'; | ||||||
| import { api } from '../../__data__/service/api'; | import { useGetMastersQuery } from '../../__data__/service/api'; | ||||||
|  | import useShowToast from '../../hooks/useShowToast'; | ||||||
| 
 | 
 | ||||||
| const TABLE_HEADERS = [ | const TABLE_HEADERS = [ | ||||||
|   'name' as const, |   'name' as const, | ||||||
| @ -30,26 +30,17 @@ const TABLE_HEADERS = [ | |||||||
| 
 | 
 | ||||||
| const Masters = () => { | const Masters = () => { | ||||||
|   const { isOpen, onOpen, onClose } = useDisclosure(); |   const { isOpen, onOpen, onClose } = useDisclosure(); | ||||||
|   const toast = useToast(); |   const showToast = useShowToast(); | ||||||
|  | 
 | ||||||
|   const { t } = useTranslation('~', { |   const { t } = useTranslation('~', { | ||||||
|     keyPrefix: 'dry-wash.arm.master', |     keyPrefix: 'dry-wash.arm.master', | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|   const { |   const { data: masters, error, isLoading, isSuccess } = useGetMastersQuery(); | ||||||
|     data: masters, |  | ||||||
|     error, |  | ||||||
|     isLoading, |  | ||||||
|     isSuccess, |  | ||||||
|   } = api.useGetMastersQuery(); |  | ||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     if (error) { |     if (error) { | ||||||
|       toast({ |       showToast(t('error.title'), 'error'); | ||||||
|         title: t('error.title'), |  | ||||||
|         status: 'error', |  | ||||||
|         isClosable: true, |  | ||||||
|         position: 'bottom-right', |  | ||||||
|       }); |  | ||||||
|     } |     } | ||||||
|   }, [error]); |   }, [error]); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -4,35 +4,8 @@ import { useTranslation } from 'react-i18next'; | |||||||
| import dayjs from 'dayjs'; | import dayjs from 'dayjs'; | ||||||
| 
 | 
 | ||||||
| import { getTimeSlot } from '../../lib'; | import { getTimeSlot } from '../../lib'; | ||||||
| import { Master } from '../../models/api/master'; | import { useUpdateOrdersMutation } from '../../__data__/service/api'; | ||||||
| import { armService } from '../../api/arm'; | import { OrderArm, Status, statuses } from '../../models/api'; | ||||||
| 
 |  | ||||||
| const statuses = [ |  | ||||||
|   'pending' as const, |  | ||||||
|   'progress' as const, |  | ||||||
|   'working' as const, |  | ||||||
|   'canceled' as const, |  | ||||||
|   'complete' as const, |  | ||||||
| ]; |  | ||||||
| 
 |  | ||||||
| type GetArrItemType<ArrType> = |  | ||||||
|   ArrType extends Array<infer ItemType> ? ItemType : never; |  | ||||||
| 
 |  | ||||||
| export type OrderProps = { |  | ||||||
|   carNumber?: string; |  | ||||||
|   startWashTime?: string; |  | ||||||
|   endWashTime?: string; |  | ||||||
|   orderDate?: string; |  | ||||||
|   status?: GetArrItemType<typeof statuses>; |  | ||||||
|   phone?: string; |  | ||||||
|   location?: string; |  | ||||||
|   master: Master; |  | ||||||
|   notes: ''; |  | ||||||
|   allMasters: Master[]; |  | ||||||
|   id: string; |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| type Status = (typeof statuses)[number]; |  | ||||||
| 
 | 
 | ||||||
| const statusColors: Record<Status, string> = { | const statusColors: Record<Status, string> = { | ||||||
|   pending: 'yellow.100', |   pending: 'yellow.100', | ||||||
| @ -53,9 +26,8 @@ const OrderItem = ({ | |||||||
|   master, |   master, | ||||||
|   allMasters, |   allMasters, | ||||||
|   id, |   id, | ||||||
| }: OrderProps) => { | }: OrderArm) => { | ||||||
|   const { updateOrders } = armService(); |   const [updateOrders] = useUpdateOrdersMutation(); | ||||||
| 
 |  | ||||||
|   const { t } = useTranslation('~', { |   const { t } = useTranslation('~', { | ||||||
|     keyPrefix: 'dry-wash.arm.order', |     keyPrefix: 'dry-wash.arm.order', | ||||||
|   }); |   }); | ||||||
| @ -72,16 +44,16 @@ const OrderItem = ({ | |||||||
| 
 | 
 | ||||||
|     if (selectedMaster) { |     if (selectedMaster) { | ||||||
|       setMaster(masterName); |       setMaster(masterName); | ||||||
|       updateOrders({ id, masterId: selectedMaster.id }); |       updateOrders({ id, master: selectedMaster.id }); | ||||||
|     } else { |     } else { | ||||||
|       console.error('Master not found'); |       console.error('Master not found'); | ||||||
|     } |     } | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|   const handeChangeStatus = (e: ChangeEvent<HTMLSelectElement>) => { |   const handeChangeStatus = (e: ChangeEvent<HTMLSelectElement>) => { | ||||||
|     const status = e.target.value; |     const status = e.target.value as OrderArm['status']; | ||||||
|     updateOrders({ id, status }); |     updateOrders({ id, status }); | ||||||
|     setStatus(e.target.value as OrderProps['status']); |     setStatus(status); | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|  | |||||||
| @ -10,16 +10,18 @@ import { | |||||||
|   Spinner, |   Spinner, | ||||||
|   Text, |   Text, | ||||||
|   Td, |   Td, | ||||||
|   useToast, |  | ||||||
| } from '@chakra-ui/react'; | } from '@chakra-ui/react'; | ||||||
| import { useTranslation } from 'react-i18next'; | import { useTranslation } from 'react-i18next'; | ||||||
| import dayjs from 'dayjs'; | import dayjs from 'dayjs'; | ||||||
| 
 | 
 | ||||||
| import OrderItem from '../OrderItem'; | import OrderItem from '../OrderItem'; | ||||||
| import { OrderProps } from '../OrderItem/OrderItem'; | import { OrderProps } from '../OrderItem/OrderItem'; | ||||||
| import { armService } from '../../api/arm'; |  | ||||||
| import DateNavigator from '../DateNavigator'; | import DateNavigator from '../DateNavigator'; | ||||||
| import { Master } from '../../models/api/master'; | import { | ||||||
|  |   useGetMastersQuery, | ||||||
|  |   useGetOrdersQuery, | ||||||
|  | } from '../../__data__/service/api'; | ||||||
|  | import useShowToast from '../../hooks/useShowToast'; | ||||||
| 
 | 
 | ||||||
| const TABLE_HEADERS = [ | const TABLE_HEADERS = [ | ||||||
|   'carNumber' as const, |   'carNumber' as const, | ||||||
| @ -34,47 +36,34 @@ const Orders = () => { | |||||||
|   const { t } = useTranslation('~', { |   const { t } = useTranslation('~', { | ||||||
|     keyPrefix: 'dry-wash.arm.order', |     keyPrefix: 'dry-wash.arm.order', | ||||||
|   }); |   }); | ||||||
|  |   const showToast = useShowToast(); | ||||||
| 
 | 
 | ||||||
|   const { fetchOrders } = armService(); |  | ||||||
|   const { fetchMasters } = armService(); |  | ||||||
| 
 |  | ||||||
|   const toast = useToast(); |  | ||||||
| 
 |  | ||||||
|   const [orders, setOrders] = useState<OrderProps[]>([]); |  | ||||||
|   const [allMasters, setAllMasters] = useState<Master[]>([]); |  | ||||||
|   const [loading, setLoading] = useState(false); |  | ||||||
|   const [error, setError] = useState<string | null>(null); |  | ||||||
|   const [currentDate, setCurrentDate] = useState(new Date()); |   const [currentDate, setCurrentDate] = useState(new Date()); | ||||||
|  |   const { | ||||||
|  |     data: orders, | ||||||
|  |     isLoading: isOrdersLoading, | ||||||
|  |     isSuccess: isOrdersSuccess, | ||||||
|  |     isError: isOrdersError, | ||||||
|  |     error: ordersError, | ||||||
|  |   } = useGetOrdersQuery({ date: currentDate }); | ||||||
|  | 
 | ||||||
|  |   const { | ||||||
|  |     data: masters, | ||||||
|  |     isLoading: isMastersLoading, | ||||||
|  |     isSuccess: isMastersSuccess, | ||||||
|  |     isError: isMastersError, | ||||||
|  |     error: mastersError, | ||||||
|  |   } = useGetMastersQuery(); | ||||||
|  | 
 | ||||||
|  |   const isLoading = isOrdersLoading || isMastersLoading; | ||||||
|  |   const isSuccess = isOrdersSuccess && isMastersSuccess; | ||||||
|  |   const isError = isOrdersError || isMastersError; | ||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     const loadData = async () => { |     if (isError) { | ||||||
|       setLoading(true); |       showToast(t('error.title'), 'error'); | ||||||
|       setError(null); |     } | ||||||
| 
 |   }, [isError, ordersError, mastersError, t]); | ||||||
|       try { |  | ||||||
|         const [ordersData, mastersData] = await Promise.all([ |  | ||||||
|           fetchOrders({ date: currentDate }), |  | ||||||
|           fetchMasters(), |  | ||||||
|         ]); |  | ||||||
| 
 |  | ||||||
|         setOrders(ordersData.body); |  | ||||||
|         setAllMasters(mastersData.body); |  | ||||||
|       } catch (err) { |  | ||||||
|         setError(err.message); |  | ||||||
|         toast({ |  | ||||||
|           title: t('error.title'), |  | ||||||
|           status: 'error', |  | ||||||
|           duration: 5000, |  | ||||||
|           isClosable: true, |  | ||||||
|           position: 'bottom-right', |  | ||||||
|         }); |  | ||||||
|       } finally { |  | ||||||
|         setLoading(false); |  | ||||||
|       } |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     loadData(); |  | ||||||
|   }, [currentDate, toast, t]); |  | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <Box p='8'> |     <Box p='8'> | ||||||
| @ -103,25 +92,24 @@ const Orders = () => { | |||||||
|           </Tr> |           </Tr> | ||||||
|         </Thead> |         </Thead> | ||||||
|         <Tbody> |         <Tbody> | ||||||
|           {loading && ( |           {isLoading && ( | ||||||
|             <Tr> |             <Tr> | ||||||
|               <Td colSpan={TABLE_HEADERS.length} textAlign='center' py='8'> |               <Td colSpan={TABLE_HEADERS.length} textAlign='center' py='8'> | ||||||
|                 <Spinner size='lg' /> |                 <Spinner size='lg' /> | ||||||
|               </Td> |               </Td> | ||||||
|             </Tr> |             </Tr> | ||||||
|           )} |           )} | ||||||
|           {!loading && orders.length === 0 && !error && ( |           {isSuccess && orders.length === 0 && ( | ||||||
|             <Tr> |             <Tr> | ||||||
|               <Td colSpan={TABLE_HEADERS.length}> |               <Td colSpan={TABLE_HEADERS.length}> | ||||||
|                 <Text>{t('table.empty')}</Text> |                 <Text>{t('table.empty')}</Text> | ||||||
|               </Td> |               </Td> | ||||||
|             </Tr> |             </Tr> | ||||||
|           )} |           )} | ||||||
|           {!loading && |           {isSuccess && | ||||||
|             !error && |  | ||||||
|             orders.map((order, index) => ( |             orders.map((order, index) => ( | ||||||
|               <OrderItem |               <OrderItem | ||||||
|                 allMasters={allMasters} |                 allMasters={masters} | ||||||
|                 key={index} |                 key={index} | ||||||
|                 {...order} |                 {...order} | ||||||
|                 status={order.status as OrderProps['status']} |                 status={order.status as OrderProps['status']} | ||||||
|  | |||||||
| @ -1,21 +0,0 @@ | |||||||
| import { UseToastOptions } from '@chakra-ui/react'; |  | ||||||
| 
 |  | ||||||
| interface ShowToast { |  | ||||||
|   toast: (options: UseToastOptions) => void; |  | ||||||
|   title: string; |  | ||||||
|   description?: string; |  | ||||||
|   status: 'info' | 'warning' | 'success' | 'error'; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| const showToast = ({ toast, title, description, status }: ShowToast) => { |  | ||||||
|   toast({ |  | ||||||
|     title, |  | ||||||
|     description, |  | ||||||
|     status, |  | ||||||
|     duration: 5000, |  | ||||||
|     isClosable: true, |  | ||||||
|     position: 'top-right', |  | ||||||
|   }); |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| export default showToast; |  | ||||||
							
								
								
									
										28
									
								
								src/hooks/useShowToast.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								src/hooks/useShowToast.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,28 @@ | |||||||
|  | import { useToast } from '@chakra-ui/react'; | ||||||
|  | import { useCallback } from 'react'; | ||||||
|  | 
 | ||||||
|  | const useShowToast = () => { | ||||||
|  |   const toast = useToast(); | ||||||
|  | 
 | ||||||
|  |   const showToast = useCallback( | ||||||
|  |     ( | ||||||
|  |       title: string, | ||||||
|  |       status: 'info' | 'warning' | 'success' | 'error', | ||||||
|  |       description?: string, | ||||||
|  |     ) => { | ||||||
|  |       toast({ | ||||||
|  |         title, | ||||||
|  |         description, | ||||||
|  |         status, | ||||||
|  |         duration: 5000, | ||||||
|  |         isClosable: true, | ||||||
|  |         position: 'top-right', | ||||||
|  |       }); | ||||||
|  |     }, | ||||||
|  |     [toast], | ||||||
|  |   ); | ||||||
|  | 
 | ||||||
|  |   return showToast; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export default useShowToast; | ||||||
| @ -1,2 +1,3 @@ | |||||||
| export * from './common'; | export * from './common'; | ||||||
| export * from './order'; | export * from './order'; | ||||||
|  | export * from './master'; | ||||||
|  | |||||||
| @ -1,21 +1,49 @@ | |||||||
| /* eslint-disable @typescript-eslint/no-namespace */ | /* eslint-disable @typescript-eslint/no-namespace */ | ||||||
| import { Order } from "../landing"; | import { Order } from '../landing'; | ||||||
| 
 | 
 | ||||||
| import { ErrorMessage } from "./common"; | import { ErrorMessage } from './common'; | ||||||
|  | import { Master } from './master'; | ||||||
| 
 | 
 | ||||||
| export namespace GetOrder { | export namespace GetOrder { | ||||||
|   export type Response = Order.View; |   export type Response = Order.View; | ||||||
|   export type Params = { |   export type Params = { | ||||||
|     orderId: Order.Id |     orderId: Order.Id; | ||||||
|   }; |   }; | ||||||
|   export type Error = ErrorMessage; |   export type Error = ErrorMessage; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export namespace CreateOrder { | export namespace CreateOrder { | ||||||
|   export type Response = { |   export type Response = { | ||||||
|     id: Order.Id |     id: Order.Id; | ||||||
|   }; |   }; | ||||||
|   export type Params = { |   export type Params = { | ||||||
|     body: Order.Create |     body: Order.Create; | ||||||
|   }; |   }; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type GetArrItemType<ArrType> = | ||||||
|  |   ArrType extends Array<infer ItemType> ? ItemType : never; | ||||||
|  | 
 | ||||||
|  | export const statuses = [ | ||||||
|  |   'pending' as const, | ||||||
|  |   'progress' as const, | ||||||
|  |   'working' as const, | ||||||
|  |   'canceled' as const, | ||||||
|  |   'complete' as const, | ||||||
|  | ]; | ||||||
|  | 
 | ||||||
|  | export type Status = (typeof statuses)[number]; | ||||||
|  | 
 | ||||||
|  | export type OrderArm = { | ||||||
|  |   carNumber?: string; | ||||||
|  |   startWashTime?: string; | ||||||
|  |   endWashTime?: string; | ||||||
|  |   orderDate?: string; | ||||||
|  |   status?: GetArrItemType<typeof statuses>; | ||||||
|  |   phone?: string; | ||||||
|  |   location?: string; | ||||||
|  |   master: Master; | ||||||
|  |   notes: ''; | ||||||
|  |   allMasters: Master[]; | ||||||
|  |   id: string; | ||||||
| }; | }; | ||||||
| @ -53,7 +53,7 @@ router.delete('/arm/masters/:id', (req, res) => { | |||||||
|     ); |     ); | ||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
| router.patch('/orders/:id', (req, res) => { | router.patch('/order/:id', (req, res) => { | ||||||
|   res |   res | ||||||
|     .status(/error/.test(STUBS.orders) ? 500 : 200) |     .status(/error/.test(STUBS.orders) ? 500 : 200) | ||||||
|     .send( |     .send( | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user