diff --git a/backend/src/api/documents/document-contract.controller.ts b/backend/src/api/documents/document-contract.controller.ts index feed2adaa..9be80edea 100644 --- a/backend/src/api/documents/document-contract.controller.ts +++ b/backend/src/api/documents/document-contract.controller.ts @@ -1,6 +1,7 @@ import { Body, Controller, + Delete, Get, Param, Patch, @@ -32,6 +33,7 @@ import { GetOneDocumentContractForNgoUsecase } from 'src/usecases/documents/new_ import { DocumentContractStatisticsPresenter } from './presenters/document-contract-statistics.presenter'; import { GetDocumentContractStatisticsUsecase } from 'src/usecases/documents/new_contracts/get-document-contract-statistics.usecase'; import { SignDocumentContractByNgoDto } from './dto/sign-document-contract-by-ngo.dto'; +import { DeleteDocumentContractUsecase } from 'src/usecases/documents/new_contracts/delete-document-contract.usecase'; @ApiBearerAuth() @UseGuards(WebJwtAuthGuard) @@ -45,6 +47,7 @@ export class DocumentContractController { private readonly signDocumentContractByNGO: SignDocumentContractByNgoUsecase, private readonly getOneDocumentContractForNgoUsecase: GetOneDocumentContractForNgoUsecase, private readonly getDocumentContractStatisticsUsecase: GetDocumentContractStatisticsUsecase, + private readonly deleteDocumentContractUsecase: DeleteDocumentContractUsecase, ) {} @Post() @@ -146,4 +149,12 @@ export class DocumentContractController { admin, }); } + + @Delete(':id') + async deleteDocumentContract( + @Param('id', UuidValidationPipe) id: string, + @ExtractUser() admin: IAdminUserModel, + ): Promise { + await this.deleteDocumentContractUsecase.execute(id, admin); + } } diff --git a/backend/src/modules/documents/exceptions/contract.exceptions.ts b/backend/src/modules/documents/exceptions/contract.exceptions.ts index a341ae617..cdcfdeb35 100644 --- a/backend/src/modules/documents/exceptions/contract.exceptions.ts +++ b/backend/src/modules/documents/exceptions/contract.exceptions.ts @@ -14,6 +14,8 @@ export enum ContractExceptionCodes { CONTRACT_011 = 'CONTRACT_011', CONTRACT_012 = 'CONTRACT_012', CONTRACT_013 = 'CONTRACT_013', + CONTRACT_014 = 'CONTRACT_014', + CONTRACT_015 = 'CONTRACT_015', } type ContractExceptionCodeType = keyof typeof ContractExceptionCodes; @@ -75,4 +77,13 @@ export const ContractExceptionMessages: Record< code_error: ContractExceptionCodes.CONTRACT_013, message: '[Create Contract] Missing volunteer personal data', }, + [ContractExceptionCodes.CONTRACT_014]: { + code_error: ContractExceptionCodes.CONTRACT_014, + message: + 'Cannot delete a contract when it has been signed by any of the parties', + }, + [ContractExceptionCodes.CONTRACT_015]: { + code_error: ContractExceptionCodes.CONTRACT_015, + message: 'Error while deleting the contract', + }, }; diff --git a/backend/src/modules/volunteer/repositories/volunteer.repository.ts b/backend/src/modules/volunteer/repositories/volunteer.repository.ts index 8ce237c08..41f467cc1 100644 --- a/backend/src/modules/volunteer/repositories/volunteer.repository.ts +++ b/backend/src/modules/volunteer/repositories/volunteer.repository.ts @@ -21,11 +21,15 @@ import { UpdateVolunteerOptions, VolunteerModelTransformer, } from '../model/volunteer.model'; -import { ContractStatus } from 'src/modules/documents/enums/contract-status.enum'; +import { + ContractStatus, + DocumentContractStatus, +} from 'src/modules/documents/enums/contract-status.enum'; import { ActivityLogStatus } from 'src/modules/activity-log/enums/activity-log-status.enum'; import { ContractEntity } from 'src/modules/documents/entities/contract.entity'; import { ActivityLogEntity } from 'src/modules/activity-log/entities/activity-log.entity'; import { VolunteerProfileEntity } from '../entities/volunteer-profile.entity'; +import { DocumentContractEntity } from 'src/modules/documents/entities/document-contract.entity'; export class VolunteerRepositoryService extends RepositoryWithPagination @@ -268,12 +272,13 @@ export class VolunteerRepositoryService .addSelect((subQuery) => { return subQuery .select('COUNT(contract.id)', 'contractCount') - .from(ContractEntity, 'contract') + .from(DocumentContractEntity, 'contract') .where('contract.volunteerId = :volunteerId', { volunteerId }) .andWhere('contract.status IN (:...statuses)', { statuses: [ - ContractStatus.PENDING_ADMIN, - ContractStatus.PENDING_VOLUNTEER, + DocumentContractStatus.PENDING_VOLUNTEER_SIGNATURE, + DocumentContractStatus.PENDING_NGO_REPRESENTATIVE_SIGNATURE, + DocumentContractStatus.PENDING_APPROVAL_NGO, ], }); }, 'contractCount') diff --git a/backend/src/usecases/documents/new_contracts/delete-document-contract.usecase.ts b/backend/src/usecases/documents/new_contracts/delete-document-contract.usecase.ts new file mode 100644 index 000000000..87b101980 --- /dev/null +++ b/backend/src/usecases/documents/new_contracts/delete-document-contract.usecase.ts @@ -0,0 +1,80 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { JSONStringifyError } from 'src/common/helpers/utils'; +import { IUseCaseService } from 'src/common/interfaces/use-case-service.interface'; +import { ExceptionsService } from 'src/infrastructure/exceptions/exceptions.service'; +import { ActionsArchiveFacade } from 'src/modules/actions-archive/actions-archive.facade'; +import { TrackedEventName } from 'src/modules/actions-archive/enums/action-resource-types.enum'; +import { DocumentContractStatus } from 'src/modules/documents/enums/contract-status.enum'; +import { ContractExceptionMessages } from 'src/modules/documents/exceptions/contract.exceptions'; +import { DocumentContractFacade } from 'src/modules/documents/services/document-contract.facade'; +import { IAdminUserModel } from 'src/modules/user/models/admin-user.model'; + +@Injectable() +export class DeleteDocumentContractUsecase implements IUseCaseService { + private readonly logger = new Logger(DeleteDocumentContractUsecase.name); + constructor( + private readonly documentContractFacade: DocumentContractFacade, + private readonly exceptionService: ExceptionsService, + private readonly actionsArchiveFacade: ActionsArchiveFacade, + ) {} + + public async execute(id: string, admin: IAdminUserModel): Promise { + try { + // 1. Does the contract exists in the callers' organization? + const contract = await this.documentContractFacade.findOne({ + id, + organizationId: admin.organizationId, + }); + + if (!contract) { + this.exceptionService.notFoundException( + ContractExceptionMessages.CONTRACT_002, + ); + } + + // 2. Status of the contract must not be signed by any of the parties + if ( + [ + DocumentContractStatus.PENDING_APPROVAL_NGO, + DocumentContractStatus.PENDING_NGO_REPRESENTATIVE_SIGNATURE, + DocumentContractStatus.APPROVED, + DocumentContractStatus.REJECTED_NGO, + DocumentContractStatus.ACTION_EXPIRED, + ].includes(contract.status) + ) { + this.exceptionService.badRequestException( + ContractExceptionMessages.CONTRACT_014, + ); + } + + await this.documentContractFacade.delete(id); + + // 10. Track event + this.actionsArchiveFacade.trackEvent( + TrackedEventName.DELETE_DOCUMENT_CONTRACT, + { + organizationId: contract.organizationId, + volunteerId: contract.volunteerId, + volunteerName: contract.volunteer.user.name, + documentContractId: contract.id, + documentContractNumber: contract.documentNumber, + }, + admin, + ); + } catch (error) { + if (error?.status === 400) { + // Rethrow errors that we've thrown above, and catch the others + throw error; + } + + this.logger.error({ + ...ContractExceptionMessages.CONTRACT_015, + error: JSONStringifyError(error), + }); + this.exceptionService.internalServerErrorException({ + ...ContractExceptionMessages.CONTRACT_015, + details: JSONStringifyError(error), + }); + } + } +} diff --git a/backend/src/usecases/use-case.module.ts b/backend/src/usecases/use-case.module.ts index 0f58f027a..8f53515ea 100644 --- a/backend/src/usecases/use-case.module.ts +++ b/backend/src/usecases/use-case.module.ts @@ -153,6 +153,7 @@ import { RejectDocumentContractByNgoUsecase } from './documents/new_contracts/re import { GetOneDocumentContractForNgoUsecase } from './documents/new_contracts/get-one-document-contract-for-ngo.usecase'; import { UpdateDocumentTemplateUsecase } from './documents/new_contracts/update-document-template.usecase'; import { GetDocumentContractStatisticsUsecase } from './documents/new_contracts/get-document-contract-statistics.usecase'; +import { DeleteDocumentContractUsecase } from './documents/new_contracts/delete-document-contract.usecase'; const providers = [ // Organization @@ -313,6 +314,7 @@ const providers = [ RejectDocumentContractByNgoUsecase, GetOneDocumentContractForNgoUsecase, GetDocumentContractStatisticsUsecase, + DeleteDocumentContractUsecase, // Notifications UpdateSettingsUsecase, ]; diff --git a/frontend/src/assets/locales/en/translation.json b/frontend/src/assets/locales/en/translation.json index b07820e10..a6e36a243 100644 --- a/frontend/src/assets/locales/en/translation.json +++ b/frontend/src/assets/locales/en/translation.json @@ -8,6 +8,7 @@ "close": "Close", "name": "Name", "people": "People", + "volunteer": "Volunteer", "contract": "Contract", "age": "Age", "hours": "Ore", diff --git a/frontend/src/assets/locales/ro/translation.json b/frontend/src/assets/locales/ro/translation.json index bde51a3fc..f48c5fac2 100644 --- a/frontend/src/assets/locales/ro/translation.json +++ b/frontend/src/assets/locales/ro/translation.json @@ -8,6 +8,7 @@ "close": "Închide", "name": "Nume", "people": "Persoane", + "volunteer": "Voluntar", "contract": "Contract", "age": "Vârsta", "hours": "Ore", @@ -989,7 +990,9 @@ "CONTRACT_002": "Contractul nu a fost găsit!", "CONTRACT_001": "Eroare la încărcarea contractului!", "CONTRACT_003": "Eroare la upload-ul documentului!", - "CONTRACT_006": "Nu se poate șterge un contract deja aprobat!" + "CONTRACT_006": "Nu se poate șterge un contract deja aprobat!", + "CONTRACT_014": "Nu se poate șterge un contract care a fost semnat de una dintre părți!", + "CONTRACT_015": "Eroare la ștergerea contractului!" } }, "upload": { @@ -1070,8 +1073,8 @@ "DONE": "Încheiat", "NOT_STARTED": "Neînceput" }, - "REJECTED_VOLUNTEER": "Refuzat", - "REJECTED_NGO": "Refuzat", + "REJECTED_VOLUNTEER": "Refuzat Voluntar", + "REJECTED_NGO": "Refuzat NGO", "ACTION_EXPIRED": "Expirat" } }, @@ -1138,7 +1141,7 @@ "download_uncompleted": "Descarcă necompletat" }, "table_header": { - "create": "Creeaază template", + "create": "Creează template", "download_all": "Descarcă toate" } }, diff --git a/frontend/src/common/errors/entities/contract.errors.ts b/frontend/src/common/errors/entities/contract.errors.ts index 9d6e6f4f5..9ee8c9ad7 100644 --- a/frontend/src/common/errors/entities/contract.errors.ts +++ b/frontend/src/common/errors/entities/contract.errors.ts @@ -11,6 +11,9 @@ export enum CONTRACT_ERRORS { CONTRACT_001 = 'CONTRACT_001', CONTRACT_003 = 'CONTRACT_003', CONTRACT_006 = 'CONTRACT_006', + + CONTRACT_014 = 'CONTRACT_014', + CONTRACT_015 = 'CONTRACT_015', } export class ContractError extends ErrorClass { @@ -29,6 +32,9 @@ export class ContractError extends ErrorClass { [CONTRACT_ERRORS.CONTRACT_001]: i18n.t('documents:contract.submit.errors.CONTRACT_001'), [CONTRACT_ERRORS.CONTRACT_003]: i18n.t('documents:contract.submit.errors.CONTRACT_003'), [CONTRACT_ERRORS.CONTRACT_006]: i18n.t('documents:contract.submit.errors.CONTRACT_006'), + + [CONTRACT_ERRORS.CONTRACT_014]: i18n.t('documents:contract.submit.errors.CONTRACT_014'), + [CONTRACT_ERRORS.CONTRACT_015]: i18n.t('documents:contract.submit.errors.CONTRACT_015'), }); } diff --git a/frontend/src/components/DocumentContractsTable.tsx b/frontend/src/components/DocumentContractsTable.tsx index d0fe6216c..cea179ade 100644 --- a/frontend/src/components/DocumentContractsTable.tsx +++ b/frontend/src/components/DocumentContractsTable.tsx @@ -5,7 +5,7 @@ import CardBody from './CardBody'; import Card from '../layouts/CardLayout'; import { IHOCQueryProps } from '../common/interfaces/hoc-query-props.interface'; import i18n from '../common/config/i18n'; -import { ArrowDownTrayIcon, EyeIcon, PlusIcon } from '@heroicons/react/24/outline'; +import { ArrowDownTrayIcon, EyeIcon, PlusIcon, TrashIcon } from '@heroicons/react/24/outline'; import { SortOrder, TableColumn } from 'react-data-table-component'; import { OrderDirection } from '../common/enums/order-direction.enum'; import Popover from './Popover'; @@ -27,51 +27,76 @@ import { useNavigate } from 'react-router-dom'; import { VolunteerTabsOptions } from '../pages/Volunteer'; import { useTranslation } from 'react-i18next'; import { getContractsForDownload } from '../services/contracts/contracts.api'; -import { useGetDocumentsContractsQuery } from '../services/document-contracts/document-contracts.service'; +import { + useDeleteDocumentContractMutation, + useGetDocumentsContractsQuery, +} from '../services/document-contracts/document-contracts.service'; import { ApprovedDocumentContractStatus, DocumentContractStatus, } from '../common/enums/document-contract-status.enum'; import { IPaginationQueryParams } from '../common/constants/pagination'; + import { IDocumentContract } from '../common/interfaces/document-contract.interface'; import DocumentsContractSidePanel from './DocumentsContractSidePanel'; +import VolunteerSelect from '../containers/VolunteerSelect'; +import { ListItem } from '../common/interfaces/list-item.interface'; +import { SelectItem } from './Select'; +import SelectFilter from '../containers/SelectFilter'; +import ConfirmationModal from './ConfirmationModal'; +import { useErrorToast, useSuccessToast } from '../hooks/useToast'; +import { InternalErrors } from '../common/errors/internal-errors.class'; + +interface StatusOption { + key: string; + internalValue: DocumentContractStatus; + value: string; +} -// const StatusOptions: SelectItem[] = [ -// { -// key: ContractStatus.ACTIVE, -// value: `${i18n.t(`documents:contract.status.${ContractStatus.ACTIVE}`)}`, -// }, -// { -// key: ContractStatus.CLOSED, -// value: `${i18n.t(`documents:contract.status.${ContractStatus.CLOSED}`)}`, -// }, -// { -// key: ContractStatus.NOT_STARTED, -// value: `${i18n.t(`documents:contract.status.${ContractStatus.NOT_STARTED}`)}`, -// }, -// { -// key: ContractStatus.REJECTED, -// value: `${i18n.t(`documents:contract.status.${ContractStatus.REJECTED}`)}`, -// }, -// { -// key: ContractStatus.PENDING_ADMIN, -// value: `${i18n.t(`documents:contract.status.${ContractStatus.PENDING_ADMIN}`)}`, -// }, -// { -// key: ContractStatus.PENDING_VOLUNTEER, -// value: `${i18n.t(`documents:contract.status.${ContractStatus.PENDING_VOLUNTEER}`)}`, -// }, -// ]; +const StatusOptions = Object.values(DocumentContractStatus).flatMap((status: string) => { + if (status === DocumentContractStatus.APPROVED) { + return [ + { + key: ApprovedDocumentContractStatus.ACTIVE, + internalValue: DocumentContractStatus.APPROVED, + value: i18n.t( + `document_contract:contract.status.${DocumentContractStatus.APPROVED}.${ApprovedDocumentContractStatus.ACTIVE}`, + ), + }, + { + key: ApprovedDocumentContractStatus.DONE, + internalValue: DocumentContractStatus.APPROVED, + value: i18n.t( + `document_contract:contract.status.${DocumentContractStatus.APPROVED}.${ApprovedDocumentContractStatus.DONE}`, + ), + }, + { + key: ApprovedDocumentContractStatus.NOT_STARTED, + internalValue: DocumentContractStatus.APPROVED, + value: i18n.t( + `document_contract:contract.status.${DocumentContractStatus.APPROVED}.${ApprovedDocumentContractStatus.NOT_STARTED}`, + ), + }, + ]; + } + return [ + { + key: status, + value: i18n.t(`document_contract:contract.status.${status}`), + internalValue: status, + }, + ]; +}); const ContractsTableHeader = [ { - id: 'contractNumber', + id: 'documentNumber', name: i18n.t('documents:contracts.headers.contract_number'), sortable: true, selector: (row: IDocumentContract) => row.documentNumber, }, { - id: 'volunteer', + id: 'volunteerName', name: i18n.t('documents:contracts.headers.volunteer'), grow: 2, sortable: true, @@ -80,13 +105,13 @@ const ContractsTableHeader = [ ), }, { - id: 'startDate', + id: 'documentStartDate', name: i18n.t('documents:contracts.headers.start_date'), sortable: true, selector: (row: IDocumentContract) => formatDate(row.documentStartDate), }, { - id: 'endDate', + id: 'documentEndDate', name: i18n.t('documents:contracts.headers.end_date'), sortable: true, selector: (row: IDocumentContract) => formatDate(row.documentEndDate), @@ -139,6 +164,7 @@ const ContractsTableHeader = [ interface DocumentContractsTableQueryProps extends IPaginationQueryParams { volunteerId?: string; + volunteerName?: string; search?: string; startDate?: Date; endDate?: Date; @@ -147,19 +173,14 @@ interface DocumentContractsTableQueryProps extends IPaginationQueryParams { } type DocumentContractsTableBasicProps = IHOCQueryProps; -interface DocumentContractsTableProps extends DocumentContractsTableBasicProps { - volunteerName?: string; - volunteerId?: string; -} -const ContractsTable = ({ - query, - setQuery, - volunteerName, - volunteerId, -}: DocumentContractsTableProps) => { +const DocumentContractsTable = ({ query, setQuery }: DocumentContractsTableBasicProps) => { // selected contract id const [selectedContract, setSelectedContract] = useState(); + const [selectedVolunteer, setSelectedVolunteer] = useState(); + const [selectedDeleteContract, setSelectedDeleteContract] = useState( + null, + ); // side panel state const [isViewContractSidePanelOpen, setIsViewContractSidePanelOpen] = useState(false); // translation @@ -167,16 +188,24 @@ const ContractsTable = ({ // navigation const navigate = useNavigate(); - const { data: contracts, isLoading: isLoadingContracts } = useGetDocumentsContractsQuery({ + const { + data: contracts, + isLoading: isLoadingContracts, + refetch, + } = useGetDocumentsContractsQuery({ page: query?.page as number, limit: query?.limit as number, search: query?.search, orderBy: query?.orderBy as string, orderDirection: query?.orderDirection as OrderDirection, - volunteerId, + volunteerId: query?.volunteerId as string, status: query?.status as DocumentContractStatus, + startDate: query?.startDate as Date, + endDate: query?.endDate as Date, }); + const { mutate: deleteContract } = useDeleteDocumentContractMutation(); + const onView = (row: IDocumentContract) => { setSelectedContract(row.documentId); setIsViewContractSidePanelOpen(true); @@ -191,12 +220,31 @@ const ContractsTable = ({ startDate: query?.startDate, endDate: query?.endDate, // status: query?.status as ContractStatus, - volunteerId, + volunteerId: query?.volunteerId, }); - downloadExcel(data as BlobPart, t('contracts.download')); }; + const showDeleteContractModal = (row: IDocumentContract) => { + setSelectedDeleteContract(row); + }; + + const confirmDelete = () => { + if (selectedDeleteContract) { + const contractId = selectedDeleteContract.documentId; + deleteContract(contractId, { + onSuccess: () => { + useSuccessToast(t('contract.submit.delete')); + refetch(); + }, + // eslint-disable-next-line @typescript-eslint/no-explicit-any + onError: (error: any) => { + useErrorToast(InternalErrors.CONTRACT_ERRORS.getError(error?.response?.data.code_error)); + }, + }); + } + }; + const buildContractActionColumn = (): TableColumn => { const contractsMenuItems = [ { @@ -218,17 +266,28 @@ const ContractsTable = ({ }, ]; + const deleteContractsMenuItems = [ + ...contractsMenuItems, + { + label: t('general:delete'), + icon: , + onClick: showDeleteContractModal, + alert: true, + }, + ]; + const mapContractStatusToPopoverItems = (status: DocumentContractStatus) => { switch (status) { case DocumentContractStatus.APPROVED: case DocumentContractStatus.SCHEDULED: + case DocumentContractStatus.CREATED: + case DocumentContractStatus.PENDING_VOLUNTEER_SIGNATURE: + return deleteContractsMenuItems; case DocumentContractStatus.ACTION_EXPIRED: case DocumentContractStatus.REJECTED_NGO: case DocumentContractStatus.REJECTED_VOLUNTEER: - case DocumentContractStatus.CREATED: case DocumentContractStatus.PENDING_APPROVAL_NGO: case DocumentContractStatus.PENDING_NGO_REPRESENTATIVE_SIGNATURE: - case DocumentContractStatus.PENDING_VOLUNTEER_SIGNATURE: return contractsMenuItems; default: return []; @@ -289,16 +348,14 @@ const ContractsTable = ({ setQuery({ endDate: endDate as Date }); }; - // const onVolunteerChange = (volunteer: ListItem) => { - // setQuery({ volunteer: volunteer.label }); - // }; + const onVolunteerChange = (volunteer: ListItem) => { + setSelectedVolunteer(volunteer); + setQuery({ volunteerId: volunteer.value }); + }; const onResetFilters = () => { - if (volunteerName) { - setQuery({ activeTab: VolunteerTabsOptions.DOCUMENTS }, 'push'); - } else { - setQuery({}, 'push'); - } + setSelectedVolunteer(undefined); + setQuery({}, 'push'); }; const onSearch = (search: string) => { @@ -307,15 +364,27 @@ const ContractsTable = ({ }); }; - // const onStatusChange = (item: SelectItem | undefined) => { - // setQuery({ status: item?.key }); - // }; - - // const onCloseSidePanel = (shouldRefetch?: boolean) => { - // setIsViewContractSidePanelOpen(false); - // setSelectedContract(undefined); - // if (shouldRefetch) refetch(); - // }; + const onStatusChange = (item: StatusOption) => { + if (item.key === 'ACTIVE') { + // setQuery({ status: item?.internalValue as DocumentContractStatus, endDate: new Date() }); + } else if (item.key === 'DONE') { + const yesterday = new Date(); + yesterday.setDate(yesterday.getDate() - 1); + setQuery({ + status: item?.internalValue as DocumentContractStatus, + endDate: new Date(yesterday), + }); + } else if (item.key === 'NOT_STARTED') { + const tomorrow = new Date(); + tomorrow.setDate(tomorrow.getDate() + 1); + setQuery({ + status: item?.internalValue as DocumentContractStatus, + startDate: new Date(tomorrow), + }); + } else { + setQuery({ status: item?.internalValue as DocumentContractStatus }); + } + }; // todo: do we need shouldRefetch? const onCloseSidePanel = () => { @@ -323,68 +392,6 @@ const ContractsTable = ({ setSelectedContract(undefined); }; - // const confirmReject = (rejectMessage?: string) => { - // if (showRejectContract) - // rejectContract( - // { - // id: showRejectContract.id, - // rejectMessage, - // }, - // { - // onSuccess: () => { - // useSuccessToast(t('contract.submit.reject')); - // // refetch(); - // }, - // onError: (error) => { - // useErrorToast(InternalErrors.CONTRACT_ERRORS.getError(error.response?.data.code_error)); - // }, - // onSettled: () => { - // setShowRejectContract(null); - // }, - // }, - // ); - // }; - - // const confirmDelete = () => { - // if (showDeleteContract) { - // const contractId = showDeleteContract.id; - // setShowDeleteContract(null); - // deleteContract(contractId, { - // onSuccess: () => { - // useSuccessToast(t('contract.submit.delete')); - // // refetch(); - // }, - // onError: (error) => { - // useErrorToast(InternalErrors.CONTRACT_ERRORS.getError(error.response?.data.code_error)); - // }, - // }); - // } - // }; - - // const onConfirmSign = (contract?: File) => { - // if (!contract) return; - - // // store id and close modal - // const contractId = showApproveContract?.id; - // setShowApproveContract(null); - - // // approval process - // approveContract( - // { - // id: contractId as string, - // contract, - // }, - // { - // onSuccess: () => { - // useSuccessToast(t('contract.submit.confirm')); - // }, - // onError: (error) => { - // useErrorToast(InternalErrors.CONTRACT_ERRORS.getError(error.response?.data.code_error)); - // }, - // }, - // ); - // }; - return ( <> - {/* {!volunteerName && ( + { - )} */} + } - {/*