feat: add Office Contact settings page and reorder Advanced sidebar

- Add OfficeContact Prisma model with receptionist name, dentist name, phone, email, fax fields
- Create GET/PUT /api/office-contact backend route and storage
- Add OfficeContactCard frontend component under Settings > Advanced
- Reorder Advanced sidebar: Office Hours → Office Contact → Twilio Settings → Google AI Settings

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
Gitead
2026-05-05 21:19:30 -04:00
parent 2312ad66ca
commit 800008792a
188 changed files with 3780 additions and 173 deletions

View File

@@ -0,0 +1,14 @@
import * as z from 'zod';
// prettier-ignore
export const OfficeContactInputSchema = z.object({
id: z.number().int(),
userId: z.number().int(),
receptionistName: z.string().optional().nullable(),
dentistName: z.string().optional().nullable(),
phoneNumber: z.string().optional().nullable(),
email: z.string().optional().nullable(),
fax: z.string().optional().nullable(),
user: z.unknown()
}).strict();
export type OfficeContactInputType = z.infer<typeof OfficeContactInputSchema>;

View File

@@ -21,7 +21,8 @@ export const UserInputSchema = z.object({
communications: z.array(z.unknown()),
twilioSettings: z.unknown().optional().nullable(),
aiSettings: z.unknown().optional().nullable(),
officeHours: z.unknown().optional().nullable()
officeHours: z.unknown().optional().nullable(),
officeContact: z.unknown().optional().nullable()
}).strict();
export type UserInputType = z.infer<typeof UserInputSchema>;

View File

@@ -30,3 +30,4 @@ export { PatientDocumentInputSchema } from './PatientDocument.input';
export { TwilioSettingsInputSchema } from './TwilioSettings.input';
export { AiSettingsInputSchema } from './AiSettings.input';
export { OfficeHoursInputSchema } from './OfficeHours.input';
export { OfficeContactInputSchema } from './OfficeContact.input';

View File

@@ -0,0 +1,14 @@
import * as z from 'zod';
// prettier-ignore
export const OfficeContactModelSchema = z.object({
id: z.number().int(),
userId: z.number().int(),
receptionistName: z.string().nullable(),
dentistName: z.string().nullable(),
phoneNumber: z.string().nullable(),
email: z.string().nullable(),
fax: z.string().nullable(),
user: z.unknown()
}).strict();
export type OfficeContactPureType = z.infer<typeof OfficeContactModelSchema>;

View File

@@ -21,7 +21,8 @@ export const UserModelSchema = z.object({
communications: z.array(z.unknown()),
twilioSettings: z.unknown().nullable(),
aiSettings: z.unknown().nullable(),
officeHours: z.unknown().nullable()
officeHours: z.unknown().nullable(),
officeContact: z.unknown().nullable()
}).strict();
export type UserPureType = z.infer<typeof UserModelSchema>;

View File

@@ -30,3 +30,4 @@ export { PatientDocumentModelSchema } from './PatientDocument.pure';
export { TwilioSettingsModelSchema } from './TwilioSettings.pure';
export { AiSettingsModelSchema } from './AiSettings.pure';
export { OfficeHoursModelSchema } from './OfficeHours.pure';
export { OfficeContactModelSchema } from './OfficeContact.pure';

View File

@@ -0,0 +1,14 @@
import * as z from 'zod';
// prettier-ignore
export const OfficeContactResultSchema = z.object({
id: z.number().int(),
userId: z.number().int(),
receptionistName: z.string().nullable(),
dentistName: z.string().nullable(),
phoneNumber: z.string().nullable(),
email: z.string().nullable(),
fax: z.string().nullable(),
user: z.unknown()
}).strict();
export type OfficeContactResultType = z.infer<typeof OfficeContactResultSchema>;

View File

@@ -21,7 +21,8 @@ export const UserResultSchema = z.object({
communications: z.array(z.unknown()),
twilioSettings: z.unknown().nullable(),
aiSettings: z.unknown().nullable(),
officeHours: z.unknown().nullable()
officeHours: z.unknown().nullable(),
officeContact: z.unknown().nullable()
}).strict();
export type UserResultType = z.infer<typeof UserResultSchema>;

View File

@@ -30,3 +30,4 @@ export { PatientDocumentResultSchema } from './PatientDocument.result';
export { TwilioSettingsResultSchema } from './TwilioSettings.result';
export { AiSettingsResultSchema } from './AiSettings.result';
export { OfficeHoursResultSchema } from './OfficeHours.result';
export { OfficeContactResultSchema } from './OfficeContact.result';