feat: office address, multi-template SMS manager, hardcoded defaults with auto-seed

- Add streetAddress/city/state/zipCode fields to OfficeContact (schema + storage + UI)
- Support {officeAddress} variable in batch reminder SMS
- Replace single SMS template field with full CRUD template list (add/rename/edit/delete)
- Store SMS template list under _sms_template_list; first template synced to batch reminder
- Hardcode all AI chat template defaults into codebase (reminder SMS, greetings, fallback)
- Add seed-templates.ts that auto-seeds default templates for all users on server boot
- Update README: note that templates are auto-configured on first boot

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
Gitead
2026-05-11 23:18:04 -04:00
parent 11244ace7f
commit 7929dc6e19
56 changed files with 763 additions and 46 deletions

View File

@@ -12,6 +12,10 @@ type OfficeContact = {
phoneNumber?: string | null;
email?: string | null;
fax?: string | null;
streetAddress?: string | null;
city?: string | null;
state?: string | null;
zipCode?: string | null;
};
export function OfficeContactCard() {
@@ -23,6 +27,10 @@ export function OfficeContactCard() {
const [phoneNumber, setPhoneNumber] = useState("");
const [email, setEmail] = useState("");
const [fax, setFax] = useState("");
const [streetAddress, setStreetAddress] = useState("");
const [city, setCity] = useState("");
const [state, setState] = useState("");
const [zipCode, setZipCode] = useState("");
const { data: contact, isLoading } = useQuery<OfficeContact | null>({
queryKey: ["/api/office-contact"],
@@ -41,6 +49,10 @@ export function OfficeContactCard() {
setPhoneNumber(contact.phoneNumber ?? "");
setEmail(contact.email ?? "");
setFax(contact.fax ?? "");
setStreetAddress(contact.streetAddress ?? "");
setCity(contact.city ?? "");
setState(contact.state ?? "");
setZipCode(contact.zipCode ?? "");
}
}, [contact]);
@@ -64,7 +76,7 @@ export function OfficeContactCard() {
const handleSubmit = (e: React.FormEvent) => {
e.preventDefault();
saveMutation.mutate({ officeName, receptionistName, dentistName, phoneNumber, email, fax });
saveMutation.mutate({ officeName, receptionistName, dentistName, phoneNumber, email, fax, streetAddress, city, state, zipCode });
};
return (
@@ -149,6 +161,54 @@ export function OfficeContactCard() {
</div>
</div>
<div className="pt-2">
<h4 className="text-sm font-semibold text-gray-700 mb-3">Office Address</h4>
<div className="space-y-3">
<div>
<label className="block text-sm font-medium">Street Address</label>
<input
type="text"
value={streetAddress}
onChange={(e) => setStreetAddress(e.target.value)}
className="mt-1 p-2 border rounded w-full text-sm"
placeholder="e.g. 123 Main Street"
/>
</div>
<div className="grid grid-cols-1 gap-3 sm:grid-cols-3">
<div>
<label className="block text-sm font-medium">City</label>
<input
type="text"
value={city}
onChange={(e) => setCity(e.target.value)}
className="mt-1 p-2 border rounded w-full text-sm"
placeholder="e.g. Framingham"
/>
</div>
<div>
<label className="block text-sm font-medium">State</label>
<input
type="text"
value={state}
onChange={(e) => setState(e.target.value)}
className="mt-1 p-2 border rounded w-full text-sm"
placeholder="e.g. MA"
/>
</div>
<div>
<label className="block text-sm font-medium">ZIP Code</label>
<input
type="text"
value={zipCode}
onChange={(e) => setZipCode(e.target.value)}
className="mt-1 p-2 border rounded w-full text-sm"
placeholder="e.g. 01701"
/>
</div>
</div>
</div>
</div>
<div className="pt-1">
<button
type="submit"