Merge pull request 'chore: promote dev → uat (GRO-1795 StatusBadge)' (#28) from dev into uat
Merge PR #28: promote dev → uat (GRO-1795 StatusBadge)
This commit was merged in pull request #28.
This commit is contained in:
@@ -195,6 +195,17 @@ export const { signIn, signOut, useSession, changePassword } = authClient;
|
|||||||
| TC-WEB-5.12.10 | RescheduleFlow error state | Mock API failure on availability fetch | "Failed to load time slots" error shown |
|
| TC-WEB-5.12.10 | RescheduleFlow error state | Mock API failure on availability fetch | "Failed to load time slots" error shown |
|
||||||
| TC-WEB-5.12.11 | RescheduleFlow no slots | Select date with no availability | "No available slots on this date" shown |
|
| TC-WEB-5.12.11 | RescheduleFlow no slots | Select date with no availability | "No available slots on this date" shown |
|
||||||
|
|
||||||
|
#### 5.12c Waitlist/Booking Status Badges (GRO-1795)
|
||||||
|
|
||||||
|
| # | Scenario | Steps | Expected |
|
||||||
|
|---|----------|-------|----------|
|
||||||
|
| TC-WEB-5.12.12 | Confirmed badge | View appointment card with confirmed status | Green "Confirmed" badge displayed |
|
||||||
|
| TC-WEB-5.12.13 | Pending badge | View appointment card with pending status | Amber "Pending" badge displayed |
|
||||||
|
| TC-WEB-5.12.14 | Waitlisted badge | View appointment card with waitlisted status | Blue "Waitlisted" badge displayed |
|
||||||
|
| TC-WEB-5.12.15 | Badge uses CSS classes | Inspect badge element | Badge uses CSS variable-based classes (e.g., bg-green-100, text-amber-600), not hardcoded colors |
|
||||||
|
| TC-WEB-5.12.16 | Badge status from data | Compare badge label to appointment.status field | Badge label matches the API appointment status exactly |
|
||||||
|
| TC-WEB-5.12.17 | Unknown status fallback | Render badge with unknown status value | Badge renders with the raw status string as label and fallback CSS class |
|
||||||
|
|
||||||
### 5.13 Reports UI
|
### 5.13 Reports UI
|
||||||
|
|
||||||
| # | Scenario | Steps | Expected |
|
| # | Scenario | Steps | Expected |
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
|
import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
|
||||||
import { render, screen, fireEvent, waitFor } from "@testing-library/react";
|
import { render, screen, fireEvent, waitFor } from "@testing-library/react";
|
||||||
import { parseTimeTo24Hour, isUpcoming, CustomerNotesSection, ConfirmationSection } from "../portal/sections/Appointments.tsx";
|
import { parseTimeTo24Hour, isUpcoming, CustomerNotesSection, ConfirmationSection, StatusBadge } from "../portal/sections/Appointments.tsx";
|
||||||
|
|
||||||
const UPCOMING_APPT = {
|
const UPCOMING_APPT = {
|
||||||
id: "appt-1",
|
id: "appt-1",
|
||||||
@@ -381,6 +381,66 @@ describe("ConfirmationSection", () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("StatusBadge", () => {
|
||||||
|
it("renders Confirmed for confirmed status", () => {
|
||||||
|
render(<StatusBadge status="confirmed" />);
|
||||||
|
expect(screen.getByText("Confirmed")).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("renders Pending for pending status", () => {
|
||||||
|
render(<StatusBadge status="pending" />);
|
||||||
|
expect(screen.getByText("Pending")).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("renders Waitlisted for waitlisted status", () => {
|
||||||
|
render(<StatusBadge status="waitlisted" />);
|
||||||
|
expect(screen.getByText("Waitlisted")).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("renders Completed for completed status", () => {
|
||||||
|
render(<StatusBadge status="completed" />);
|
||||||
|
expect(screen.getByText("Completed")).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("renders Cancelled for cancelled status", () => {
|
||||||
|
render(<StatusBadge status="cancelled" />);
|
||||||
|
expect(screen.getByText("Cancelled")).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("falls back to status string for unknown status", () => {
|
||||||
|
render(<StatusBadge status="custom-status" />);
|
||||||
|
expect(screen.getByText("custom-status")).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("uses correct CSS class for confirmed status", () => {
|
||||||
|
render(<StatusBadge status="confirmed" />);
|
||||||
|
const badge = screen.getByText("Confirmed").closest('span');
|
||||||
|
expect(badge?.className).toContain("bg-green-100");
|
||||||
|
expect(badge?.className).toContain("text-green-700");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("uses correct CSS class for waitlisted status", () => {
|
||||||
|
render(<StatusBadge status="waitlisted" />);
|
||||||
|
const badge = screen.getByText("Waitlisted").closest('span');
|
||||||
|
expect(badge?.className).toContain("bg-blue-100");
|
||||||
|
expect(badge?.className).toContain("text-blue-600");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("uses correct CSS class for pending status", () => {
|
||||||
|
render(<StatusBadge status="pending" />);
|
||||||
|
const badge = screen.getByText("Pending").closest('span');
|
||||||
|
expect(badge?.className).toContain("bg-amber-100");
|
||||||
|
expect(badge?.className).toContain("text-amber-600");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("uses fallback styling for unknown status", () => {
|
||||||
|
render(<StatusBadge status="unknown" />);
|
||||||
|
const badge = screen.getByText("unknown").closest('span');
|
||||||
|
expect(badge?.className).toContain("bg-stone-100");
|
||||||
|
expect(badge?.className).toContain("text-stone-600");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
describe("RescheduleFlow dynamic time slots", () => {
|
describe("RescheduleFlow dynamic time slots", () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
vi.clearAllMocks();
|
vi.clearAllMocks();
|
||||||
|
|||||||
@@ -83,14 +83,34 @@ export function isUpcoming(appt: Appointment): boolean {
|
|||||||
|
|
||||||
const STATUS_COLORS: Record<string, string> = {
|
const STATUS_COLORS: Record<string, string> = {
|
||||||
confirmed: 'bg-green-100 text-green-700',
|
confirmed: 'bg-green-100 text-green-700',
|
||||||
pending: 'bg-amber-100 text-amber-700',
|
pending: 'bg-amber-100 text-amber-600',
|
||||||
waitlisted: 'bg-blue-100 text-blue-700',
|
waitlisted: 'bg-blue-100 text-blue-600',
|
||||||
completed: 'bg-stone-100 text-stone-600',
|
completed: 'bg-stone-100 text-stone-600',
|
||||||
cancelled: 'bg-red-100 text-red-600',
|
cancelled: 'bg-red-100 text-red-600',
|
||||||
'no-show': 'bg-yellow-100 text-yellow-700',
|
'no-show': 'bg-yellow-100 text-yellow-700',
|
||||||
scheduled: 'bg-blue-100 text-blue-700',
|
scheduled: 'bg-blue-100 text-blue-600',
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const STATUS_LABELS: Record<string, string> = {
|
||||||
|
confirmed: 'Confirmed',
|
||||||
|
pending: 'Pending',
|
||||||
|
waitlisted: 'Waitlisted',
|
||||||
|
completed: 'Completed',
|
||||||
|
cancelled: 'Cancelled',
|
||||||
|
'no-show': 'No-show',
|
||||||
|
scheduled: 'Scheduled',
|
||||||
|
};
|
||||||
|
|
||||||
|
export function StatusBadge({ status }: { status: string }) {
|
||||||
|
const label = STATUS_LABELS[status] ?? status;
|
||||||
|
const colorClass = STATUS_COLORS[status] ?? 'bg-stone-100 text-stone-600';
|
||||||
|
return (
|
||||||
|
<span className={`px-2 py-0.5 rounded-full text-xs font-medium ${colorClass}`}>
|
||||||
|
{label}
|
||||||
|
</span>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
const CONFIRMATION_STATUS_COLORS: Record<string, string> = {
|
const CONFIRMATION_STATUS_COLORS: Record<string, string> = {
|
||||||
confirmed: 'bg-green-100 text-green-700',
|
confirmed: 'bg-green-100 text-green-700',
|
||||||
pending: 'bg-amber-100 text-amber-700',
|
pending: 'bg-amber-100 text-amber-700',
|
||||||
@@ -298,13 +318,7 @@ function AppointmentCard({
|
|||||||
<span>with {appt.groomerName || 'First Available'}</span>
|
<span>with {appt.groomerName || 'First Available'}</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<span
|
<StatusBadge status={appt.status} />
|
||||||
className={`px-2 py-0.5 rounded-full text-xs font-medium ${
|
|
||||||
STATUS_COLORS[appt.status] || ''
|
|
||||||
}`}
|
|
||||||
>
|
|
||||||
{appt.status}
|
|
||||||
</span>
|
|
||||||
{expanded ? (
|
{expanded ? (
|
||||||
<ChevronDown size={16} className="text-stone-400" />
|
<ChevronDown size={16} className="text-stone-400" />
|
||||||
) : (
|
) : (
|
||||||
|
|||||||
Reference in New Issue
Block a user