feat: add Procedure Duration/Time Slot settings and custom appointment type

- Add Settings > Advanced > Procedure Duration/Time Slot page with three sections:
  1. Procedure Duration: CDT codes with durations (editable table, save per section)
  2. Doctor Time Slot: drag-to-block visual grid (A/B/C columns, 8 AM–9 PM, edit/delete slots)
  3. Hygienist Time Slot: procedure descriptions with durations
- Backend: ProcedureTimeslot Prisma model, storage, and GET/PUT /api/procedure-timeslot route
- DB migration: procedure_timeslot table
- Appointment form: when type is "Other", show a free-text input for custom description; saved as other:<description> and decoded for display on the schedule card

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
Gitead
2026-05-05 23:08:34 -04:00
parent fea0dd4d59
commit ceb95f1915
11 changed files with 900 additions and 4 deletions

View File

@@ -817,7 +817,11 @@ export default function AppointmentsPage() {
<Move className="h-3 w-3" />
{appointment.patientName}
</div>
<div className="truncate">{appointment.type}</div>
<div className="truncate">
{appointment.type?.startsWith("other:")
? appointment.type.slice(6)
: appointment.type}
</div>
</div>
);
}

View File

@@ -17,6 +17,7 @@ import { TwilioSettingsCard } from "@/components/settings/twilio-settings-card";
import { AiSettingsCard } from "@/components/settings/ai-settings-card";
import { OfficeHoursCard } from "@/components/settings/office-hours-card";
import { OfficeContactCard } from "@/components/settings/office-contact-card";
import { ProcedureTimeslotCard } from "@/components/settings/procedure-timeslot-card";
type SectionId =
| "staff"
@@ -28,7 +29,8 @@ type SectionId =
| "twilio"
| "ai"
| "officehours"
| "officecontact";
| "officecontact"
| "proceduretimeslot";
export default function SettingsPage() {
const { toast } = useToast();
@@ -261,6 +263,9 @@ export default function SettingsPage() {
case "officecontact":
return <OfficeContactCard />;
case "proceduretimeslot":
return <ProcedureTimeslotCard />;
default:
return null;
}