From ea8a1c7ebf7fe8f9a50902361f4f2d75881b31c2 Mon Sep 17 00:00:00 2001
From: Casper Iversen <53900565+casperiv0@users.noreply.github.com>
Date: Mon, 23 Oct 2023 19:19:35 +0200
Subject: [PATCH] feat: call 911 disposition code (#1858)
---
.../migration.sql | 8 +++++
apps/api/prisma/schema.prisma | 17 +++++++----
.../admin/values/import-values-controller.ts | 1 +
apps/client/locales/en/calls.json | 3 +-
apps/client/locales/en/values.json | 4 ++-
.../admin/values/ManageValueModal.tsx | 12 ++++++++
.../modals/manage-911-call-form.tsx | 29 ++++++++++++++++++-
.../form/inputs/value-select-field.tsx | 2 ++
apps/client/src/lib/admin/values/values.tsx | 6 +++-
packages/schemas/src/admin/values/import.ts | 1 +
10 files changed, 73 insertions(+), 10 deletions(-)
create mode 100644 apps/api/prisma/migrations/20231023170026_disposition_code_call/migration.sql
diff --git a/apps/api/prisma/migrations/20231023170026_disposition_code_call/migration.sql b/apps/api/prisma/migrations/20231023170026_disposition_code_call/migration.sql
new file mode 100644
index 000000000..6a2de6ae7
--- /dev/null
+++ b/apps/api/prisma/migrations/20231023170026_disposition_code_call/migration.sql
@@ -0,0 +1,8 @@
+-- AlterTable
+ALTER TABLE "Call911" ADD COLUMN "dispositionCodeId" TEXT;
+
+-- AlterTable
+ALTER TABLE "CallTypeValue" ADD COLUMN "isDisposition" BOOLEAN NOT NULL DEFAULT false;
+
+-- AddForeignKey
+ALTER TABLE "Call911" ADD CONSTRAINT "Call911_dispositionCodeId_fkey" FOREIGN KEY ("dispositionCodeId") REFERENCES "CallTypeValue"("id") ON DELETE SET NULL ON UPDATE CASCADE;
diff --git a/apps/api/prisma/schema.prisma b/apps/api/prisma/schema.prisma
index 1bf6cee7c..881bc3936 100644
--- a/apps/api/prisma/schema.prisma
+++ b/apps/api/prisma/schema.prisma
@@ -824,11 +824,14 @@ model WeaponValue {
}
model CallTypeValue {
- id String @id @default(cuid())
- priority String?
- valueId String
- value Value @relation("callTypeValueToValue", fields: [valueId], references: [id], onDelete: Cascade)
- Call911 Call911[]
+ id String @id @default(cuid())
+ priority String?
+ valueId String
+ isDisposition Boolean @default(false)
+ value Value @relation("callTypeValueToValue", fields: [valueId], references: [id], onDelete: Cascade)
+
+ dispositionCodeToCall Call911[] @relation("dispositionCodeToCall")
+ callTypeToCall Call911[] @relation("callTypeToCall")
}
// notifications
@@ -1310,12 +1313,14 @@ model Call911 {
ended Boolean? @default(false)
situationCode StatusValue? @relation(fields: [situationCodeId], references: [id])
situationCodeId String?
+ dispositionCode CallTypeValue? @relation("dispositionCodeToCall", fields: [dispositionCodeId], references: [id])
+ dispositionCodeId String?
viaDispatch Boolean? @default(false)
divisions DivisionValue[]
departments DepartmentValue[]
events Call911Event[]
incidents LeoIncident[]
- type CallTypeValue? @relation(fields: [typeId], references: [id])
+ type CallTypeValue? @relation("callTypeToCall", fields: [typeId], references: [id])
typeId String?
Officer Officer[] @relation("activeCall")
EmsFdDeputy EmsFdDeputy[] @relation("emsFdActiveCall")
diff --git a/apps/api/src/controllers/admin/values/import-values-controller.ts b/apps/api/src/controllers/admin/values/import-values-controller.ts
index 35606388a..566ab5a37 100644
--- a/apps/api/src/controllers/admin/values/import-values-controller.ts
+++ b/apps/api/src/controllers/admin/values/import-values-controller.ts
@@ -483,6 +483,7 @@ export const typeHandlers = {
where: { id: String(id) },
...makePrismaData(ValueType.CALL_TYPE, {
priority: item.priority,
+ isDisposition: item.isDisposition ?? false,
value: item.value,
isDisabled: item.isDisabled,
}),
diff --git a/apps/client/locales/en/calls.json b/apps/client/locales/en/calls.json
index ec8524d4b..087505c3d 100644
--- a/apps/client/locales/en/calls.json
+++ b/apps/client/locales/en/calls.json
@@ -69,7 +69,8 @@
"id": "id",
"towLogsDescription": "Here you can view all archived tow logs. This allows you to view all tow logs that have been ended/archived.",
"addAllUnits": "Add all active units",
- "openCallModalAfterCreation": "Open Manage 911 call modal after call creation?"
+ "openCallModalAfterCreation": "Open Manage 911 call modal after call creation?",
+ "dispositionCode": "Disposition Code"
},
"Events": {
"unitAssignedToCall": "{unit} has been assigned to this call.",
diff --git a/apps/client/locales/en/values.json b/apps/client/locales/en/values.json
index 3fb97e347..eddca38d0 100644
--- a/apps/client/locales/en/values.json
+++ b/apps/client/locales/en/values.json
@@ -74,7 +74,9 @@
"FELONY": "Felony",
"warningApplicable": "Warning Applicable",
"warningNotApplicable": "Warning Not Applicable",
- "isPrimary": "Is Primary"
+ "isPrimary": "Is Primary",
+ "isDispositionDescription": "A disposition code can be used when a call is closed. This will be used to determine the outcome of the call.",
+ "isDisposition": "Is Disposition"
},
"PENAL_CODE_GROUP": {
"MANAGE": "Manage Penal Code Groups",
diff --git a/apps/client/src/components/admin/values/ManageValueModal.tsx b/apps/client/src/components/admin/values/ManageValueModal.tsx
index 7d1964924..706cdc7cc 100644
--- a/apps/client/src/components/admin/values/ManageValueModal.tsx
+++ b/apps/client/src/components/admin/values/ManageValueModal.tsx
@@ -133,6 +133,7 @@ function createInitialValues(options: CreateInitialValuesOptions) {
licenseType: value && isBaseValue(value) ? value.licenseType : null,
isDefault: value && isBaseValue(value) ? value.isDefault : undefined,
priority: value && isCallTypeValue(value) ? value.priority ?? undefined : undefined,
+ isDisposition: value && isCallTypeValue(value) ? value.isDisposition ?? undefined : undefined,
officerRankImageId: "",
officerRankDepartments:
@@ -376,6 +377,17 @@ export function ManageValueModal({ onCreate, onUpdate, type, value }: Props) {
/>
) : null}
+ {type === ValueType.CALL_TYPE ? (
+ setFieldValue("isDisposition", value)}
+ isSelected={values.isDisposition}
+ description={tValues("isDispositionDescription")}
+ >
+ {tValues("isDisposition")}
+
+ ) : null}
+
{type === ValueType.OFFICER_RANK ? (
<>
diff --git a/apps/client/src/components/dispatch/active-calls/modals/manage-911-call-form.tsx b/apps/client/src/components/dispatch/active-calls/modals/manage-911-call-form.tsx
index c572188fe..4e3e8c259 100644
--- a/apps/client/src/components/dispatch/active-calls/modals/manage-911-call-form.tsx
+++ b/apps/client/src/components/dispatch/active-calls/modals/manage-911-call-form.tsx
@@ -111,6 +111,7 @@ export function Manage911CallForm({ call, isDisabled, setShowAlert, handleClose
departments: call?.departments?.map((dep) => ({ value: dep.id, label: dep.value.value })) ?? [],
divisions: call?.divisions?.map((dep) => ({ value: dep.id, label: dep.value.value })) ?? [],
situationCode: call?.situationCodeId ?? null,
+ dispositionCodeId: call?.dispositionCodeId ?? null,
type: call?.typeId ?? null,
notifyAssignedUnits: true,
openCallModalAfterCreation: true,
@@ -208,7 +209,7 @@ export function Manage911CallForm({ call, isDisabled, setShowAlert, handleClose
) : null}
-
+
value.type === StatusValueType.SITUATION_CODE}
+ className="w-full"
+ />
+
+ value.isDisposition}
+ className="w-full"
/>
!value.isDisposition}
+ className="w-full"
/>
+ value.type === StatusValueType.SITUATION_CODE}
+ />
+
(props: Props) {
}, [getDefaultSearchValue]);
function handleSuggestionPress(node?: Node | null) {
+ if (node) setSearch(node?.textValue);
+
const fieldData = { [props.fieldName]: node?.key ?? null };
setValues({ ...values, ...fieldData });
props.onSelectionChange?.((node?.value as T | null) ?? null);
diff --git a/apps/client/src/lib/admin/values/values.tsx b/apps/client/src/lib/admin/values/values.tsx
index 16a74ddf5..da8637f9f 100644
--- a/apps/client/src/lib/admin/values/values.tsx
+++ b/apps/client/src/lib/admin/values/values.tsx
@@ -189,6 +189,7 @@ export function useTableDataOfType(type: ValueType) {
return {
priority: v.priority ?? common("none"),
+ isDisposition: common(yesOrNoText(v.isDisposition)),
};
}
case ValueType.DRIVERSLICENSE_CATEGORY: {
@@ -275,7 +276,10 @@ export function useTableHeadersOfType(type: ValueType): ColumnDef<{ id: string }
];
}
case ValueType.CALL_TYPE: {
- return [{ header: t("priority"), accessorKey: "priority" }];
+ return [
+ { header: t("priority"), accessorKey: "priority" },
+ { header: t("isDisposition"), accessorKey: "isDisposition" },
+ ];
}
case ValueType.DRIVERSLICENSE_CATEGORY: {
return [
diff --git a/packages/schemas/src/admin/values/import.ts b/packages/schemas/src/admin/values/import.ts
index 7eed18f13..b45c13de6 100644
--- a/packages/schemas/src/admin/values/import.ts
+++ b/packages/schemas/src/admin/values/import.ts
@@ -144,6 +144,7 @@ export const QUALIFICATION_ARR = z.array(QUALIFICATION_SCHEMA).min(1);
*/
export const CALL_TYPE_SCHEMA = BASE_VALUE_SCHEMA.extend({
priority: z.string().nullish(),
+ isDisposition: z.boolean().nullish(),
});
export const CALL_TYPE_ARR = z.array(CALL_TYPE_SCHEMA).min(1);