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,37 @@
import * as z from 'zod';
export const OfficeContactAggregateResultSchema = z.object({ _count: z.object({
id: z.number(),
userId: z.number(),
receptionistName: z.number(),
dentistName: z.number(),
phoneNumber: z.number(),
email: z.number(),
fax: z.number(),
user: z.number()
}).optional(),
_sum: z.object({
id: z.number().nullable(),
userId: z.number().nullable()
}).nullable().optional(),
_avg: z.object({
id: z.number().nullable(),
userId: z.number().nullable()
}).nullable().optional(),
_min: z.object({
id: z.number().int().nullable(),
userId: z.number().int().nullable(),
receptionistName: z.string().nullable(),
dentistName: z.string().nullable(),
phoneNumber: z.string().nullable(),
email: z.string().nullable(),
fax: z.string().nullable()
}).nullable().optional(),
_max: z.object({
id: z.number().int().nullable(),
userId: z.number().int().nullable(),
receptionistName: z.string().nullable(),
dentistName: z.string().nullable(),
phoneNumber: z.string().nullable(),
email: z.string().nullable(),
fax: z.string().nullable()
}).nullable().optional()});

View File

@@ -0,0 +1,2 @@
import * as z from 'zod';
export const OfficeContactCountResultSchema = z.number();

View File

@@ -0,0 +1,4 @@
import * as z from 'zod';
export const OfficeContactCreateManyResultSchema = z.object({
count: z.number()
});

View File

@@ -0,0 +1,11 @@
import * as z from 'zod';
export const OfficeContactCreateResultSchema = z.object({
id: z.number().int(),
userId: z.number().int(),
receptionistName: z.string().optional(),
dentistName: z.string().optional(),
phoneNumber: z.string().optional(),
email: z.string().optional(),
fax: z.string().optional(),
user: z.unknown()
});

View File

@@ -0,0 +1,4 @@
import * as z from 'zod';
export const OfficeContactDeleteManyResultSchema = z.object({
count: z.number()
});

View File

@@ -0,0 +1,11 @@
import * as z from 'zod';
export const OfficeContactDeleteResultSchema = z.nullable(z.object({
id: z.number().int(),
userId: z.number().int(),
receptionistName: z.string().optional(),
dentistName: z.string().optional(),
phoneNumber: z.string().optional(),
email: z.string().optional(),
fax: z.string().optional(),
user: z.unknown()
}));

View File

@@ -0,0 +1,11 @@
import * as z from 'zod';
export const OfficeContactFindFirstResultSchema = z.nullable(z.object({
id: z.number().int(),
userId: z.number().int(),
receptionistName: z.string().optional(),
dentistName: z.string().optional(),
phoneNumber: z.string().optional(),
email: z.string().optional(),
fax: z.string().optional(),
user: z.unknown()
}));

View File

@@ -0,0 +1,21 @@
import * as z from 'zod';
export const OfficeContactFindManyResultSchema = z.object({
data: z.array(z.object({
id: z.number().int(),
userId: z.number().int(),
receptionistName: z.string().optional(),
dentistName: z.string().optional(),
phoneNumber: z.string().optional(),
email: z.string().optional(),
fax: z.string().optional(),
user: z.unknown()
})),
pagination: z.object({
page: z.number().int().min(1),
pageSize: z.number().int().min(1),
total: z.number().int().min(0),
totalPages: z.number().int().min(0),
hasNext: z.boolean(),
hasPrev: z.boolean()
})
});

View File

@@ -0,0 +1,11 @@
import * as z from 'zod';
export const OfficeContactFindUniqueResultSchema = z.nullable(z.object({
id: z.number().int(),
userId: z.number().int(),
receptionistName: z.string().optional(),
dentistName: z.string().optional(),
phoneNumber: z.string().optional(),
email: z.string().optional(),
fax: z.string().optional(),
user: z.unknown()
}));

View File

@@ -0,0 +1,46 @@
import * as z from 'zod';
export const OfficeContactGroupByResultSchema = z.array(z.object({
id: z.number().int(),
userId: z.number().int(),
receptionistName: z.string(),
dentistName: z.string(),
phoneNumber: z.string(),
email: z.string(),
fax: z.string(),
_count: z.object({
id: z.number(),
userId: z.number(),
receptionistName: z.number(),
dentistName: z.number(),
phoneNumber: z.number(),
email: z.number(),
fax: z.number(),
user: z.number()
}).optional(),
_sum: z.object({
id: z.number().nullable(),
userId: z.number().nullable()
}).nullable().optional(),
_avg: z.object({
id: z.number().nullable(),
userId: z.number().nullable()
}).nullable().optional(),
_min: z.object({
id: z.number().int().nullable(),
userId: z.number().int().nullable(),
receptionistName: z.string().nullable(),
dentistName: z.string().nullable(),
phoneNumber: z.string().nullable(),
email: z.string().nullable(),
fax: z.string().nullable()
}).nullable().optional(),
_max: z.object({
id: z.number().int().nullable(),
userId: z.number().int().nullable(),
receptionistName: z.string().nullable(),
dentistName: z.string().nullable(),
phoneNumber: z.string().nullable(),
email: z.string().nullable(),
fax: z.string().nullable()
}).nullable().optional()
}));

View File

@@ -0,0 +1,4 @@
import * as z from 'zod';
export const OfficeContactUpdateManyResultSchema = z.object({
count: z.number()
});

View File

@@ -0,0 +1,11 @@
import * as z from 'zod';
export const OfficeContactUpdateResultSchema = z.nullable(z.object({
id: z.number().int(),
userId: z.number().int(),
receptionistName: z.string().optional(),
dentistName: z.string().optional(),
phoneNumber: z.string().optional(),
email: z.string().optional(),
fax: z.string().optional(),
user: z.unknown()
}));

View File

@@ -0,0 +1,11 @@
import * as z from 'zod';
export const OfficeContactUpsertResultSchema = z.object({
id: z.number().int(),
userId: z.number().int(),
receptionistName: z.string().optional(),
dentistName: z.string().optional(),
phoneNumber: z.string().optional(),
email: z.string().optional(),
fax: z.string().optional(),
user: z.unknown()
});

View File

@@ -20,7 +20,8 @@ export const UserAggregateResultSchema = z.object({ _count: z.object({
communications: z.number(),
twilioSettings: z.number(),
aiSettings: z.number(),
officeHours: z.number()
officeHours: z.number(),
officeContact: z.number()
}).optional(),
_sum: z.object({
id: z.number().nullable()

View File

@@ -20,5 +20,6 @@ export const UserCreateResultSchema = z.object({
communications: z.array(z.unknown()),
twilioSettings: z.unknown().optional(),
aiSettings: z.unknown().optional(),
officeHours: z.unknown().optional()
officeHours: z.unknown().optional(),
officeContact: z.unknown().optional()
});

View File

@@ -20,5 +20,6 @@ export const UserDeleteResultSchema = z.nullable(z.object({
communications: z.array(z.unknown()),
twilioSettings: z.unknown().optional(),
aiSettings: z.unknown().optional(),
officeHours: z.unknown().optional()
officeHours: z.unknown().optional(),
officeContact: z.unknown().optional()
}));

View File

@@ -20,5 +20,6 @@ export const UserFindFirstResultSchema = z.nullable(z.object({
communications: z.array(z.unknown()),
twilioSettings: z.unknown().optional(),
aiSettings: z.unknown().optional(),
officeHours: z.unknown().optional()
officeHours: z.unknown().optional(),
officeContact: z.unknown().optional()
}));

View File

@@ -21,7 +21,8 @@ export const UserFindManyResultSchema = z.object({
communications: z.array(z.unknown()),
twilioSettings: z.unknown().optional(),
aiSettings: z.unknown().optional(),
officeHours: z.unknown().optional()
officeHours: z.unknown().optional(),
officeContact: z.unknown().optional()
})),
pagination: z.object({
page: z.number().int().min(1),

View File

@@ -20,5 +20,6 @@ export const UserFindUniqueResultSchema = z.nullable(z.object({
communications: z.array(z.unknown()),
twilioSettings: z.unknown().optional(),
aiSettings: z.unknown().optional(),
officeHours: z.unknown().optional()
officeHours: z.unknown().optional(),
officeContact: z.unknown().optional()
}));

View File

@@ -26,7 +26,8 @@ export const UserGroupByResultSchema = z.array(z.object({
communications: z.number(),
twilioSettings: z.number(),
aiSettings: z.number(),
officeHours: z.number()
officeHours: z.number(),
officeContact: z.number()
}).optional(),
_sum: z.object({
id: z.number().nullable()

View File

@@ -20,5 +20,6 @@ export const UserUpdateResultSchema = z.nullable(z.object({
communications: z.array(z.unknown()),
twilioSettings: z.unknown().optional(),
aiSettings: z.unknown().optional(),
officeHours: z.unknown().optional()
officeHours: z.unknown().optional(),
officeContact: z.unknown().optional()
}));

View File

@@ -20,5 +20,6 @@ export const UserUpsertResultSchema = z.object({
communications: z.array(z.unknown()),
twilioSettings: z.unknown().optional(),
aiSettings: z.unknown().optional(),
officeHours: z.unknown().optional()
officeHours: z.unknown().optional(),
officeContact: z.unknown().optional()
});

View File

@@ -349,3 +349,16 @@ export { OfficeHoursDeleteManyResultSchema } from './OfficeHoursDeleteManyResult
export { OfficeHoursAggregateResultSchema } from './OfficeHoursAggregateResult.schema';
export { OfficeHoursGroupByResultSchema } from './OfficeHoursGroupByResult.schema';
export { OfficeHoursCountResultSchema } from './OfficeHoursCountResult.schema';
export { OfficeContactFindUniqueResultSchema } from './OfficeContactFindUniqueResult.schema';
export { OfficeContactFindFirstResultSchema } from './OfficeContactFindFirstResult.schema';
export { OfficeContactFindManyResultSchema } from './OfficeContactFindManyResult.schema';
export { OfficeContactCreateResultSchema } from './OfficeContactCreateResult.schema';
export { OfficeContactCreateManyResultSchema } from './OfficeContactCreateManyResult.schema';
export { OfficeContactUpdateResultSchema } from './OfficeContactUpdateResult.schema';
export { OfficeContactUpdateManyResultSchema } from './OfficeContactUpdateManyResult.schema';
export { OfficeContactUpsertResultSchema } from './OfficeContactUpsertResult.schema';
export { OfficeContactDeleteResultSchema } from './OfficeContactDeleteResult.schema';
export { OfficeContactDeleteManyResultSchema } from './OfficeContactDeleteManyResult.schema';
export { OfficeContactAggregateResultSchema } from './OfficeContactAggregateResult.schema';
export { OfficeContactGroupByResultSchema } from './OfficeContactGroupByResult.schema';
export { OfficeContactCountResultSchema } from './OfficeContactCountResult.schema';