Implement confirm/cancel in customer portal (GRO-50)
Backend: - Add POST /api/portal/appointments/:id/confirm endpoint - Validates impersonation session auth and ownership - Rejects past/in-progress, non-pending, or already-cancelled/completed - Sets confirmationStatus="confirmed", confirmedAt, updatedAt - Add POST /api/portal/appointments/:id/cancel endpoint - Same auth/ownership pattern - Rejects past/in-progress or already-cancelled/completed - Sets status="cancelled", confirmationStatus="cancelled", cancelledAt, updatedAt Frontend (Appointments.tsx): - Add confirmationStatus field to Appointment type and mock data - Add ConfirmationSection component: shows status badge + confirm button - Add CancelAppointmentButton: wires to cancel API with loading/error state - Wire existing Cancel button to CancelAppointmentButton - Show confirmation status badge in expanded view for upcoming appointments Co-Authored-By: Paperclip <noreply@paperclip.ing>
This commit is contained in:
@@ -76,6 +76,133 @@ portalRouter.patch(
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// ─── Appointment confirm/cancel ──────────────────────────────────────────────
|
||||||
|
|
||||||
|
portalRouter.post("/appointments/:id/confirm", async (c) => {
|
||||||
|
const db = getDb();
|
||||||
|
const id = c.req.param("id");
|
||||||
|
|
||||||
|
const sessionId = c.req.header("X-Impersonation-Session-Id");
|
||||||
|
if (!sessionId) {
|
||||||
|
return c.json({ error: "Unauthorized" }, 401);
|
||||||
|
}
|
||||||
|
|
||||||
|
const [session] = await db
|
||||||
|
.select()
|
||||||
|
.from(impersonationSessions)
|
||||||
|
.where(
|
||||||
|
and(
|
||||||
|
eq(impersonationSessions.id, sessionId),
|
||||||
|
eq(impersonationSessions.status, "active")
|
||||||
|
)
|
||||||
|
)
|
||||||
|
.limit(1);
|
||||||
|
|
||||||
|
if (!session || session.expiresAt <= new Date()) {
|
||||||
|
return c.json({ error: "Unauthorized" }, 401);
|
||||||
|
}
|
||||||
|
|
||||||
|
const [appt] = await db
|
||||||
|
.select()
|
||||||
|
.from(appointments)
|
||||||
|
.where(eq(appointments.id, id))
|
||||||
|
.limit(1);
|
||||||
|
|
||||||
|
if (!appt) {
|
||||||
|
return c.json({ error: "Not found" }, 404);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (appt.clientId !== session.clientId) {
|
||||||
|
return c.json({ error: "Forbidden" }, 403);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (appt.startTime <= new Date()) {
|
||||||
|
return c.json({ error: "Cannot confirm a past or in-progress appointment" }, 422);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (appt.confirmationStatus !== "pending") {
|
||||||
|
return c.json({ error: "Appointment is not pending confirmation" }, 422);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (appt.status === "cancelled" || appt.status === "completed") {
|
||||||
|
return c.json({ error: "Cannot confirm a cancelled or completed appointment" }, 422);
|
||||||
|
}
|
||||||
|
|
||||||
|
const [updated] = await db
|
||||||
|
.update(appointments)
|
||||||
|
.set({ confirmationStatus: "confirmed", confirmedAt: new Date(), updatedAt: new Date() })
|
||||||
|
.where(eq(appointments.id, id))
|
||||||
|
.returning();
|
||||||
|
|
||||||
|
return c.json({
|
||||||
|
id: updated.id,
|
||||||
|
confirmationStatus: updated.confirmationStatus,
|
||||||
|
confirmedAt: updated.confirmedAt,
|
||||||
|
updatedAt: updated.updatedAt,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
portalRouter.post("/appointments/:id/cancel", async (c) => {
|
||||||
|
const db = getDb();
|
||||||
|
const id = c.req.param("id");
|
||||||
|
|
||||||
|
const sessionId = c.req.header("X-Impersonation-Session-Id");
|
||||||
|
if (!sessionId) {
|
||||||
|
return c.json({ error: "Unauthorized" }, 401);
|
||||||
|
}
|
||||||
|
|
||||||
|
const [session] = await db
|
||||||
|
.select()
|
||||||
|
.from(impersonationSessions)
|
||||||
|
.where(
|
||||||
|
and(
|
||||||
|
eq(impersonationSessions.id, sessionId),
|
||||||
|
eq(impersonationSessions.status, "active")
|
||||||
|
)
|
||||||
|
)
|
||||||
|
.limit(1);
|
||||||
|
|
||||||
|
if (!session || session.expiresAt <= new Date()) {
|
||||||
|
return c.json({ error: "Unauthorized" }, 401);
|
||||||
|
}
|
||||||
|
|
||||||
|
const [appt] = await db
|
||||||
|
.select()
|
||||||
|
.from(appointments)
|
||||||
|
.where(eq(appointments.id, id))
|
||||||
|
.limit(1);
|
||||||
|
|
||||||
|
if (!appt) {
|
||||||
|
return c.json({ error: "Not found" }, 404);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (appt.clientId !== session.clientId) {
|
||||||
|
return c.json({ error: "Forbidden" }, 403);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (appt.startTime <= new Date()) {
|
||||||
|
return c.json({ error: "Cannot cancel a past or in-progress appointment" }, 422);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (appt.status === "cancelled" || appt.status === "completed") {
|
||||||
|
return c.json({ error: "Appointment is already cancelled or completed" }, 422);
|
||||||
|
}
|
||||||
|
|
||||||
|
const [updated] = await db
|
||||||
|
.update(appointments)
|
||||||
|
.set({ status: "cancelled", confirmationStatus: "cancelled", cancelledAt: new Date(), updatedAt: new Date() })
|
||||||
|
.where(eq(appointments.id, id))
|
||||||
|
.returning();
|
||||||
|
|
||||||
|
return c.json({
|
||||||
|
id: updated.id,
|
||||||
|
status: updated.status,
|
||||||
|
confirmationStatus: updated.confirmationStatus,
|
||||||
|
cancelledAt: updated.cancelledAt,
|
||||||
|
updatedAt: updated.updatedAt,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
// ─── Client-facing waitlist routes ───────────────────────────────────────────
|
// ─── Client-facing waitlist routes ───────────────────────────────────────────
|
||||||
|
|
||||||
const createWaitlistEntrySchema = z.object({
|
const createWaitlistEntrySchema = z.object({
|
||||||
|
|||||||
@@ -41,6 +41,7 @@ export interface Appointment {
|
|||||||
duration: number;
|
duration: number;
|
||||||
price: number;
|
price: number;
|
||||||
status: "confirmed" | "pending" | "waitlisted" | "completed" | "cancelled";
|
status: "confirmed" | "pending" | "waitlisted" | "completed" | "cancelled";
|
||||||
|
confirmationStatus: "pending" | "confirmed" | "cancelled";
|
||||||
notes: string;
|
notes: string;
|
||||||
customerNotes: string;
|
customerNotes: string;
|
||||||
reportCardId?: string;
|
reportCardId?: string;
|
||||||
@@ -177,21 +178,21 @@ export const UPCOMING_APPOINTMENTS: Appointment[] = [
|
|||||||
id: "a1", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
id: "a1", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
||||||
services: ["Full Groom"], addOns: ["De-shedding Treatment"],
|
services: ["Full Groom"], addOns: ["De-shedding Treatment"],
|
||||||
date: "2026-03-21", time: "10:00 AM", duration: 120, price: 145,
|
date: "2026-03-21", time: "10:00 AM", duration: 120, price: 145,
|
||||||
status: "confirmed", notes: "Spring shed is heavy — extra undercoat work needed",
|
status: "confirmed", confirmationStatus: "confirmed", notes: "Spring shed is heavy — extra undercoat work needed",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "a2", petId: "p2", petName: "Mochi", groomerId: "g3", groomerName: "Morgan",
|
id: "a2", petId: "p2", petName: "Mochi", groomerId: "g3", groomerName: "Morgan",
|
||||||
services: ["Full Groom"], addOns: ["Teeth Brushing"],
|
services: ["Full Groom"], addOns: ["Teeth Brushing"],
|
||||||
date: "2026-03-25", time: "2:00 PM", duration: 100, price: 90,
|
date: "2026-03-25", time: "2:00 PM", duration: 100, price: 90,
|
||||||
status: "confirmed", notes: "First visit with Morgan — patient with anxious pets",
|
status: "confirmed", confirmationStatus: "confirmed", notes: "First visit with Morgan — patient with anxious pets",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "a3", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
id: "a3", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
||||||
services: ["Bath & Brush"], addOns: [],
|
services: ["Bath & Brush"], addOns: [],
|
||||||
date: "2026-04-18", time: "11:00 AM", duration: 45, price: 55,
|
date: "2026-04-18", time: "11:00 AM", duration: 45, price: 55,
|
||||||
status: "pending", notes: "",
|
status: "pending", confirmationStatus: "pending", notes: "",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
@@ -201,56 +202,56 @@ export const PAST_APPOINTMENTS: Appointment[] = [
|
|||||||
id: "pa1", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
id: "pa1", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
||||||
services: ["Full Groom"], addOns: ["De-shedding Treatment", "Blueberry Facial"],
|
services: ["Full Groom"], addOns: ["De-shedding Treatment", "Blueberry Facial"],
|
||||||
date: "2026-02-15", time: "10:00 AM", duration: 130, price: 160,
|
date: "2026-02-15", time: "10:00 AM", duration: 130, price: 160,
|
||||||
status: "completed", notes: "", reportCardId: "rc1",
|
status: "completed", confirmationStatus: "confirmed", notes: "", reportCardId: "rc1",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "pa2", petId: "p2", petName: "Mochi", groomerId: "g2", groomerName: "Alex",
|
id: "pa2", petId: "p2", petName: "Mochi", groomerId: "g2", groomerName: "Alex",
|
||||||
services: ["Full Groom"], addOns: ["Teeth Brushing"],
|
services: ["Full Groom"], addOns: ["Teeth Brushing"],
|
||||||
date: "2026-02-20", time: "1:00 PM", duration: 100, price: 88,
|
date: "2026-02-20", time: "1:00 PM", duration: 100, price: 88,
|
||||||
status: "completed", notes: "", reportCardId: "rc2",
|
status: "completed", confirmationStatus: "confirmed", notes: "", reportCardId: "rc2",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "pa3", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
id: "pa3", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
||||||
services: ["Bath & Brush"], addOns: [],
|
services: ["Bath & Brush"], addOns: [],
|
||||||
date: "2026-01-18", time: "9:00 AM", duration: 45, price: 55,
|
date: "2026-01-18", time: "9:00 AM", duration: 45, price: 55,
|
||||||
status: "completed", notes: "",
|
status: "completed", confirmationStatus: "confirmed", notes: "",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "pa4", petId: "p2", petName: "Mochi", groomerId: "g2", groomerName: "Alex",
|
id: "pa4", petId: "p2", petName: "Mochi", groomerId: "g2", groomerName: "Alex",
|
||||||
services: ["Puppy's First Groom"], addOns: [],
|
services: ["Puppy's First Groom"], addOns: [],
|
||||||
date: "2026-01-10", time: "3:00 PM", duration: 60, price: 62,
|
date: "2026-01-10", time: "3:00 PM", duration: 60, price: 62,
|
||||||
status: "completed", notes: "",
|
status: "completed", confirmationStatus: "confirmed", notes: "",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "pa5", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
id: "pa5", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
||||||
services: ["Full Groom"], addOns: ["Nail Grinding"],
|
services: ["Full Groom"], addOns: ["Nail Grinding"],
|
||||||
date: "2025-12-20", time: "10:00 AM", duration: 105, price: 132,
|
date: "2025-12-20", time: "10:00 AM", duration: 105, price: 132,
|
||||||
status: "completed", notes: "Holiday groom",
|
status: "completed", confirmationStatus: "confirmed", notes: "Holiday groom",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "pa6", petId: "p1", petName: "Biscuit", groomerId: "g2", groomerName: "Alex",
|
id: "pa6", petId: "p1", petName: "Biscuit", groomerId: "g2", groomerName: "Alex",
|
||||||
services: ["Full Groom"], addOns: [],
|
services: ["Full Groom"], addOns: [],
|
||||||
date: "2025-11-15", time: "11:00 AM", duration: 90, price: 110,
|
date: "2025-11-15", time: "11:00 AM", duration: 90, price: 110,
|
||||||
status: "completed", notes: "",
|
status: "completed", confirmationStatus: "confirmed", notes: "",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "pa7", petId: "p2", petName: "Mochi", groomerId: "g3", groomerName: "Morgan",
|
id: "pa7", petId: "p2", petName: "Mochi", groomerId: "g3", groomerName: "Morgan",
|
||||||
services: ["Bath & Brush"], addOns: [],
|
services: ["Bath & Brush"], addOns: [],
|
||||||
date: "2025-11-08", time: "2:00 PM", duration: 45, price: 48,
|
date: "2025-11-08", time: "2:00 PM", duration: 45, price: 48,
|
||||||
status: "completed", notes: "",
|
status: "completed", confirmationStatus: "confirmed", notes: "",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "pa8", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
id: "pa8", petId: "p1", petName: "Biscuit", groomerId: "g1", groomerName: "Jamie",
|
||||||
services: ["Bath & Brush"], addOns: ["De-shedding Treatment"],
|
services: ["Bath & Brush"], addOns: ["De-shedding Treatment"],
|
||||||
date: "2025-10-12", time: "10:00 AM", duration: 75, price: 85,
|
date: "2025-10-12", time: "10:00 AM", duration: 75, price: 85,
|
||||||
status: "completed", notes: "",
|
status: "completed", confirmationStatus: "confirmed", notes: "",
|
||||||
customerNotes: "",
|
customerNotes: "",
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -41,6 +41,12 @@ const STATUS_COLORS: Record<string, string> = {
|
|||||||
cancelled: "bg-red-100 text-red-600",
|
cancelled: "bg-red-100 text-red-600",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const CONFIRMATION_STATUS_COLORS: Record<string, string> = {
|
||||||
|
confirmed: "bg-green-100 text-green-700",
|
||||||
|
pending: "bg-amber-100 text-amber-700",
|
||||||
|
cancelled: "bg-red-100 text-red-600",
|
||||||
|
};
|
||||||
|
|
||||||
export function AppointmentsSection({ readOnly, sessionId }: Props) {
|
export function AppointmentsSection({ readOnly, sessionId }: Props) {
|
||||||
const [showBooking, setShowBooking] = useState(false);
|
const [showBooking, setShowBooking] = useState(false);
|
||||||
const [expandedId, setExpandedId] = useState<string | null>(null);
|
const [expandedId, setExpandedId] = useState<string | null>(null);
|
||||||
@@ -165,14 +171,15 @@ function AppointmentCard({
|
|||||||
{isUpcoming(appt) && !readOnly && (
|
{isUpcoming(appt) && !readOnly && (
|
||||||
<CustomerNotesSection appointment={appt} sessionId={sessionId} />
|
<CustomerNotesSection appointment={appt} sessionId={sessionId} />
|
||||||
)}
|
)}
|
||||||
|
{isUpcoming(appt) && (
|
||||||
|
<ConfirmationSection appointment={appt} sessionId={sessionId} onCancel={() => {}} />
|
||||||
|
)}
|
||||||
{appt.status !== "completed" && appt.status !== "cancelled" && !readOnly && (
|
{appt.status !== "completed" && appt.status !== "cancelled" && !readOnly && (
|
||||||
<div className="flex gap-2 mt-3">
|
<div className="flex gap-2 mt-3">
|
||||||
<button className="text-xs px-3 py-1.5 border border-stone-200 rounded-lg text-stone-600 hover:bg-stone-50">
|
<button className="text-xs px-3 py-1.5 border border-stone-200 rounded-lg text-stone-600 hover:bg-stone-50">
|
||||||
Reschedule
|
Reschedule
|
||||||
</button>
|
</button>
|
||||||
<button className="text-xs px-3 py-1.5 border border-red-200 rounded-lg text-red-600 hover:bg-red-50">
|
<CancelAppointmentButton appointment={appt} sessionId={sessionId} />
|
||||||
Cancel
|
|
||||||
</button>
|
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
{appt.reportCardId && (
|
{appt.reportCardId && (
|
||||||
@@ -188,6 +195,112 @@ function AppointmentCard({
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function ConfirmationSection({ appointment: appt, sessionId }: { appointment: Appointment; sessionId?: string | null }) {
|
||||||
|
const [confirming, setConfirming] = useState(false);
|
||||||
|
const [confirmError, setConfirmError] = useState<string | null>(null);
|
||||||
|
const [confirmSuccess, setConfirmSuccess] = useState(false);
|
||||||
|
|
||||||
|
async function handleConfirm() {
|
||||||
|
if (!window.confirm("Confirm this appointment?")) return;
|
||||||
|
setConfirming(true);
|
||||||
|
setConfirmError(null);
|
||||||
|
try {
|
||||||
|
const headers: Record<string, string> = {};
|
||||||
|
if (sessionId) {
|
||||||
|
headers["X-Impersonation-Session-Id"] = sessionId;
|
||||||
|
}
|
||||||
|
const res = await fetch(`/api/portal/appointments/${appt.id}/confirm`, {
|
||||||
|
method: "POST",
|
||||||
|
headers,
|
||||||
|
});
|
||||||
|
if (!res.ok) {
|
||||||
|
const err = await res.json().catch(() => ({ error: "Failed to confirm" }));
|
||||||
|
throw new Error(err.error || `HTTP ${res.status}`);
|
||||||
|
}
|
||||||
|
setConfirmSuccess(true);
|
||||||
|
setTimeout(() => setConfirmSuccess(false), 2000);
|
||||||
|
} catch (e) {
|
||||||
|
setConfirmError(e instanceof Error ? e.message : "Failed to confirm");
|
||||||
|
} finally {
|
||||||
|
setConfirming(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const statusLabel = appt.confirmationStatus === "confirmed"
|
||||||
|
? "✓ Confirmed"
|
||||||
|
: appt.confirmationStatus === "pending"
|
||||||
|
? "Pending confirmation"
|
||||||
|
: "Cancelled";
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="mt-3 p-3 bg-stone-50 rounded-lg">
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<span className={`px-2 py-0.5 rounded-full text-xs font-medium ${CONFIRMATION_STATUS_COLORS[appt.confirmationStatus] || ""}`}>
|
||||||
|
{statusLabel}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
{!confirmSuccess && appt.confirmationStatus === "pending" && (
|
||||||
|
<button
|
||||||
|
onClick={handleConfirm}
|
||||||
|
disabled={confirming}
|
||||||
|
className="flex items-center gap-1.5 text-xs px-3 py-1.5 bg-green-600 text-white rounded-lg font-medium hover:bg-green-700 disabled:opacity-50 disabled:cursor-not-allowed"
|
||||||
|
>
|
||||||
|
{confirming && <Loader2 size={12} className="animate-spin" />}
|
||||||
|
{confirming ? "Confirming..." : "Confirm Appointment"}
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
{confirmSuccess && (
|
||||||
|
<span className="text-xs text-green-600 font-medium">Confirmed!</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
{confirmError && <p className="text-xs text-red-500 mt-1">{confirmError}</p>}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function CancelAppointmentButton({ appointment: appt, sessionId }: { appointment: Appointment; sessionId?: string | null }) {
|
||||||
|
const [cancelling, setCancelling] = useState(false);
|
||||||
|
const [cancelError, setCancelError] = useState<string | null>(null);
|
||||||
|
|
||||||
|
async function handleCancel() {
|
||||||
|
if (!window.confirm("Cancel this appointment? This cannot be undone.")) return;
|
||||||
|
setCancelling(true);
|
||||||
|
setCancelError(null);
|
||||||
|
try {
|
||||||
|
const headers: Record<string, string> = {};
|
||||||
|
if (sessionId) {
|
||||||
|
headers["X-Impersonation-Session-Id"] = sessionId;
|
||||||
|
}
|
||||||
|
const res = await fetch(`/api/portal/appointments/${appt.id}/cancel`, {
|
||||||
|
method: "POST",
|
||||||
|
headers,
|
||||||
|
});
|
||||||
|
if (!res.ok) {
|
||||||
|
const err = await res.json().catch(() => ({ error: "Failed to cancel" }));
|
||||||
|
throw new Error(err.error || `HTTP ${res.status}`);
|
||||||
|
}
|
||||||
|
window.location.reload();
|
||||||
|
} catch (e) {
|
||||||
|
setCancelError(e instanceof Error ? e.message : "Failed to cancel");
|
||||||
|
setCancelling(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<button
|
||||||
|
onClick={handleCancel}
|
||||||
|
disabled={cancelling}
|
||||||
|
className="text-xs px-3 py-1.5 border border-red-200 rounded-lg text-red-600 hover:bg-red-50 disabled:opacity-50 disabled:cursor-not-allowed"
|
||||||
|
>
|
||||||
|
{cancelling ? "Cancelling..." : "Cancel"}
|
||||||
|
</button>
|
||||||
|
{cancelError && <p className="text-xs text-red-500 mt-1">{cancelError}</p>}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
export function CustomerNotesSection({ appointment: appt, sessionId }: { appointment: Appointment; sessionId?: string | null }) {
|
export function CustomerNotesSection({ appointment: appt, sessionId }: { appointment: Appointment; sessionId?: string | null }) {
|
||||||
const [notes, setNotes] = useState(appt.customerNotes || "");
|
const [notes, setNotes] = useState(appt.customerNotes || "");
|
||||||
const [saving, setSaving] = useState(false);
|
const [saving, setSaving] = useState(false);
|
||||||
|
|||||||
Reference in New Issue
Block a user