diff --git a/apps/api/prisma/migrations/20231007073203_citizen_record_payments/migration.sql b/apps/api/prisma/migrations/20231007073203_citizen_record_payments/migration.sql new file mode 100644 index 000000000..e92db9dc8 --- /dev/null +++ b/apps/api/prisma/migrations/20231007073203_citizen_record_payments/migration.sql @@ -0,0 +1,2 @@ +-- AlterEnum +ALTER TYPE "Feature" ADD VALUE 'CITIZEN_RECORD_PAYMENTS'; diff --git a/apps/api/prisma/schema.prisma b/apps/api/prisma/schema.prisma index 1fefcb4f1..5946fca67 100644 --- a/apps/api/prisma/schema.prisma +++ b/apps/api/prisma/schema.prisma @@ -1989,6 +1989,7 @@ enum Feature { LEO_EDITABLE_CITIZEN_PROFILE // see #1698 ALLOW_MULTIPLE_UNITS_DEPARTMENTS_PER_USER // see #1722 OPEN_LAW_BOOK + CITIZEN_RECORD_PAYMENTS } enum DashboardLayoutCardType { diff --git a/apps/api/src/controllers/record/records-controller.ts b/apps/api/src/controllers/record/records-controller.ts index 5884b9ad9..98717510f 100644 --- a/apps/api/src/controllers/record/records-controller.ts +++ b/apps/api/src/controllers/record/records-controller.ts @@ -25,6 +25,7 @@ import { Officer, User, Business, + PaymentStatus, } from "@prisma/client"; import { validateSchema } from "lib/data/validate-schema"; import { combinedUnitProperties, leoProperties } from "utils/leo/includes"; @@ -52,6 +53,7 @@ import { Descendant, slateDataToString } from "@snailycad/utils/editor"; import puppeteer from "puppeteer"; import { AuditLogActionType, createAuditLogEntry } from "@snailycad/audit-logger/server"; import { captureException } from "@sentry/node"; +import { shouldCheckCitizenUserId } from "~/lib/citizen/has-citizen-access"; export const assignedOfficersInclude = { combinedUnit: { include: combinedUnitProperties }, @@ -389,6 +391,54 @@ export class RecordsController { return recordItem; } + @Post("/mark-as-paid/:id") + @Description("Allow a citizen to mark a record as paid") + @IsFeatureEnabled({ + feature: Feature.CITIZEN_RECORD_PAYMENTS, + }) + async markRecordAsPaid( + @Context("cad") cad: { features?: Record }, + @Context("user") user: User, + @PathParams("id") recordId: string, + ): Promise { + const checkCitizenUserId = shouldCheckCitizenUserId({ cad, user }); + + const citizen = await prisma.citizen.findFirst({ + where: { + userId: checkCitizenUserId ? user.id : undefined, + Record: { some: { id: recordId } }, + }, + }); + + if (!citizen) { + throw new NotFound("citizenNotFound"); + } + + const record = await prisma.record.findFirst({ + where: { id: recordId }, + }); + + if (!record) { + throw new NotFound("recordNotFound"); + } + + const isEnabled = isFeatureEnabled({ + feature: Feature.CITIZEN_RECORD_APPROVAL, + features: cad.features, + defaultReturn: false, + }); + + const updatedRecord = await prisma.record.update({ + where: { id: recordId }, + data: { + paymentStatus: PaymentStatus.PAID, + }, + include: recordsInclude(isEnabled).include, + }); + + return updatedRecord; + } + @UseBefore(ActiveOfficer) @Delete("/:id") @Description("Delete a warrant, ticket, written warning or arrest report by its id") diff --git a/apps/api/src/middlewares/is-enabled.ts b/apps/api/src/middlewares/is-enabled.ts index ba1cb4b73..8aaec468e 100644 --- a/apps/api/src/middlewares/is-enabled.ts +++ b/apps/api/src/middlewares/is-enabled.ts @@ -34,6 +34,7 @@ export const DEFAULT_DISABLED_FEATURES: Partial< REQUIRED_CITIZEN_IMAGE: { isEnabled: false }, LEO_EDITABLE_CITIZEN_PROFILE: { isEnabled: false }, ALLOW_MULTIPLE_UNITS_DEPARTMENTS_PER_USER: { isEnabled: false }, + CITIZEN_RECORD_PAYMENTS: { isEnabled: false }, }; export type CadFeatures = Record & { diff --git a/apps/api/src/migrations/set-default-cad-features.ts b/apps/api/src/migrations/set-default-cad-features.ts index acef84c00..54ddf9f1c 100644 --- a/apps/api/src/migrations/set-default-cad-features.ts +++ b/apps/api/src/migrations/set-default-cad-features.ts @@ -25,6 +25,7 @@ const DEFAULT_DISABLED_FEATURES: Partial REQUIRED_CITIZEN_IMAGE: { isEnabled: false }, LEO_EDITABLE_CITIZEN_PROFILE: { isEnabled: false }, ALLOW_MULTIPLE_UNITS_DEPARTMENTS_PER_USER: { isEnabled: false }, + CITIZEN_RECORD_PAYMENTS: { isEnabled: false }, }; /** diff --git a/apps/client/locales/en/cad-settings.json b/apps/client/locales/en/cad-settings.json index 0318ff111..8cafe3126 100644 --- a/apps/client/locales/en/cad-settings.json +++ b/apps/client/locales/en/cad-settings.json @@ -349,7 +349,9 @@ "ALLOW_MULTIPLE_UNITS_DEPARTMENTS_PER_USER": "Allow multiple units with the same callsign and department per user", "ALLOW_MULTIPLE_UNITS_DEPARTMENTS_PER_USER-description": "When enabled, officers and deputies can create multiple units with the same callsign and department.", "OPEN_LAW_BOOK": "Open Law Book", - "OPEN_LAW_BOOK-description": "When enabled, this will allow every user to view the CAD's penal codes (Law Book)" + "OPEN_LAW_BOOK-description": "When enabled, this will allow every user to view the CAD's penal codes (Law Book)", + "CITIZEN_RECORD_PAYMENTS": "Citizen Record Payments", + "CITIZEN_RECORD_PAYMENTS-description": "When enabled, this will allow citizens to mark their own records as paid." }, "Permissions": { "defaultPermissions": "Default Permissions", diff --git a/apps/client/locales/en/citizen.json b/apps/client/locales/en/citizen.json index a079a3c14..5985e87d8 100644 --- a/apps/client/locales/en/citizen.json +++ b/apps/client/locales/en/citizen.json @@ -68,7 +68,8 @@ "huntingLicenseCategory": "Hunting License Categories", "huntingLicensePoints": "Hunting License Points", "fishingLicensePoints": "Fishing License Points", - "otherLicenseCategory": "Other License Categories" + "otherLicenseCategory": "Other License Categories", + "markAsPaid": "Mark As Paid" }, "Vehicles": { "model": "Model", diff --git a/apps/client/src/components/leo/modals/NameSearchModal/tabs/records-tab.tsx b/apps/client/src/components/leo/modals/NameSearchModal/tabs/records-tab.tsx index d62ffff88..51bb3b6f4 100644 --- a/apps/client/src/components/leo/modals/NameSearchModal/tabs/records-tab.tsx +++ b/apps/client/src/components/leo/modals/NameSearchModal/tabs/records-tab.tsx @@ -1,7 +1,7 @@ import * as React from "react"; import compareDesc from "date-fns/compareDesc"; import { useRouter } from "next/router"; -import { Record, RecordType } from "@snailycad/types"; +import { PaymentStatus, Record, RecordType } from "@snailycad/types"; import { useTranslations } from "use-intl"; import { Button, FullDate, Loader, Status, TabsContent } from "@snailycad/ui"; import { ModalIds } from "types/modal-ids"; @@ -14,10 +14,11 @@ import { Table, useTableState } from "components/shared/Table"; import { ManageRecordModal } from "../../manage-record/manage-record-modal"; import { Permissions, usePermission } from "hooks/usePermission"; import { ViolationsColumn } from "components/leo/ViolationsColumn"; -import type { DeleteRecordsByIdData } from "@snailycad/types/api"; +import type { DeleteRecordsByIdData, PutRecordsByIdData } from "@snailycad/types/api"; import { RecordsCaseNumberColumn } from "components/leo/records-case-number-column"; import { CallDescription } from "components/dispatch/active-calls/CallDescription"; import { getAPIUrl } from "@snailycad/utils/api-url"; +import { useFeatureEnabled } from "hooks/useFeatureEnabled"; interface RecordsTabProps { records: Record[]; @@ -95,7 +96,7 @@ export function RecordsTab({ {data.length <= 0 ? (

{noValuesText}

) : ( - + )} ) : ( @@ -151,6 +152,7 @@ export function RecordsTable({ const modalState = useModal(); const t = useTranslations(); const router = useRouter(); + const { execute, state } = useFetch(); const isCitizenCreation = router.pathname === "/citizen/create"; const isCitizen = router.pathname.startsWith("/citizen") && !isCitizenCreation; @@ -158,6 +160,7 @@ export function RecordsTable({ const { generateCallsign } = useGenerateCallsign(); const tableState = useTableState(); const currency = common("currency"); + const { CITIZEN_RECORD_PAYMENTS } = useFeatureEnabled(); const { hasPermissions } = usePermission(); const _hasDeletePermissions = @@ -187,6 +190,17 @@ export function RecordsTable({ document.body.removeChild(anchor); } + async function handleMarkAsPaid(record: Record) { + const { json } = await execute({ + path: `/records/mark-as-paid/${record.id}`, + method: "POST", + }); + + if (json.id) { + onEdit?.(json); + } + } + async function handleExportClick(record: Record) { setExportState("loading"); @@ -265,9 +279,27 @@ export function RecordsTable({ /> ), createdAt: {record.createdAt}, - actions: isCitizen ? null : ( + actions: ( <> - {isCitizenCreation ? null : ( + {isCitizen && CITIZEN_RECORD_PAYMENTS ? ( + record.paymentStatus === PaymentStatus.PAID ? ( + "—" + ) : ( + + ) + ) : null} + + {isCitizen ? null : ( <> + + {_hasDeletePermissions ? ( + + ) : null} )} - - {_hasDeletePermissions ? ( - - ) : null} ), }; @@ -314,8 +346,8 @@ export function RecordsTable({ { header: t("Leo.officer"), accessorKey: "officer" }, { header: t("Leo.paymentStatus"), accessorKey: "paymentStatus" }, isCitizen ? { header: t("Leo.totalCost"), accessorKey: "totalCost" } : null, - isCitizenCreation ? null : { header: common("createdAt"), accessorKey: "createdAt" }, - isCitizen ? null : { header: common("actions"), accessorKey: "actions" }, + { header: common("createdAt"), accessorKey: "createdAt" }, + { header: common("actions"), accessorKey: "actions" }, ]} /> diff --git a/apps/client/src/hooks/useFeatureEnabled.ts b/apps/client/src/hooks/useFeatureEnabled.ts index 8118df475..7150e5c2f 100644 --- a/apps/client/src/hooks/useFeatureEnabled.ts +++ b/apps/client/src/hooks/useFeatureEnabled.ts @@ -25,6 +25,7 @@ export const DEFAULT_DISABLED_FEATURES = { REQUIRED_CITIZEN_IMAGE: { isEnabled: false }, LEO_EDITABLE_CITIZEN_PROFILE: { isEnabled: false }, ALLOW_MULTIPLE_UNITS_DEPARTMENTS_PER_USER: { isEnabled: false }, + CITIZEN_RECORD_PAYMENTS: { isEnabled: false }, } satisfies Partial>; export const DEFAULT_FEATURE_OPTIONS = { diff --git a/packages/types/src/enums.ts b/packages/types/src/enums.ts index 5e668d2bf..2a06cb5a1 100644 --- a/packages/types/src/enums.ts +++ b/packages/types/src/enums.ts @@ -54,6 +54,7 @@ export const Feature = { LEO_EDITABLE_CITIZEN_PROFILE: "LEO_EDITABLE_CITIZEN_PROFILE", ALLOW_MULTIPLE_UNITS_DEPARTMENTS_PER_USER: "ALLOW_MULTIPLE_UNITS_DEPARTMENTS_PER_USER", OPEN_LAW_BOOK: "OPEN_LAW_BOOK", + CITIZEN_RECORD_PAYMENTS: "CITIZEN_RECORD_PAYMENTS", } as const; export type Feature = (typeof Feature)[keyof typeof Feature]; diff --git a/packages/utils/src/api-url.ts b/packages/utils/src/api-url.ts index fdf33abb1..4afce3b3b 100644 --- a/packages/utils/src/api-url.ts +++ b/packages/utils/src/api-url.ts @@ -4,10 +4,6 @@ export function getAPIUrl(): string { const envUrl = process.env.NEXT_PUBLIC_PROD_ORIGIN ?? "http://localhost:8080/v1"; - if (process.env.NODE_ENV === "development") { - return "http://localhost:8080/v1"; - } - if (envUrl.endsWith("/v1")) { return envUrl; }