@getmicdrop/svelte-components 2.0.4 → 2.0.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/components/Alert/Alert.spec.js +170 -170
- package/dist/components/Badges/Badge.spec.js +103 -103
- package/dist/components/BottomSheet/BottomSheet.spec.js +127 -127
- package/dist/components/Breadcrumb/Breadcrumb.spec.js +120 -120
- package/dist/components/Button/Button.spec.js +211 -211
- package/dist/components/Button/ButtonSaveDemo.spec.js +48 -48
- package/dist/components/Calendar/Calendar.spec.js +131 -131
- package/dist/components/Calendar/QuarterView.spec.js +394 -394
- package/dist/components/Card.spec.js +47 -47
- package/dist/components/CropImage/CropImage.spec.js +216 -216
- package/dist/components/DarkModeToggle.spec.js +357 -357
- package/dist/components/ErrorDisplay.spec.js +69 -69
- package/dist/components/FormActions.spec.js +88 -88
- package/dist/components/FormValidationSummary.spec.js +203 -203
- package/dist/components/Icons/Icon.spec.js +175 -175
- package/dist/components/Icons/MoreHori.spec.js +67 -67
- package/dist/components/Icons/WarningIcon.spec.js +30 -30
- package/dist/components/Input/Input.spec.js +573 -573
- package/dist/components/Input/MultiSelect.spec.js +257 -257
- package/dist/components/Input/OTPInput.spec.js +238 -238
- package/dist/components/Input/Select.spec.js +218 -218
- package/dist/components/Layout/BottomNav.spec.js +130 -130
- package/dist/components/Layout/Header.spec.js +203 -203
- package/dist/components/Modal/ConfirmationModal.spec.js +191 -191
- package/dist/components/Modal/Modal.spec.js +95 -95
- package/dist/components/Modal/ModalStateManager.spec.js +100 -100
- package/dist/components/PageLoader.spec.js +54 -54
- package/dist/components/PasswordStrengthIndicator/PasswordStrengthIndicator.spec.js +173 -173
- package/dist/components/PlaceAutocomplete/PlaceAutocomplete.spec.js +300 -300
- package/dist/components/Spinner/Spinner.spec.js +75 -75
- package/dist/components/StatusIndicator/StatusIndicator.spec.js +129 -129
- package/dist/components/Toaster/Toaster.stories.svelte +1 -1
- package/dist/components/Toggle.spec.js +158 -158
- package/dist/components/ValidationError.spec.js +79 -79
- package/dist/components/pages/performers/AvailabilityCalendarModal.spec.js +606 -606
- package/dist/components/pages/performers/AvailabilityCalendarModal.svelte +4 -4
- package/dist/components/pages/performers/ModalShowInfo.spec.js +124 -124
- package/dist/components/pages/performers/ModalShowInfo.svelte +1 -1
- package/dist/components/pages/performers/PageBackButton.spec.js +89 -89
- package/dist/components/pages/performers/SectionHeader.spec.js +75 -75
- package/dist/components/pages/performers/ShowDetails.spec.js +166 -166
- package/dist/components/pages/performers/ShowItemCard.spec.js +793 -793
- package/dist/components/pages/performers/ShowItemCard.svelte +4 -4
- package/dist/components/pages/performers/SwitchOption.spec.js +127 -127
- package/dist/components/pages/performers/VenueInfo.spec.js +167 -167
- package/dist/components/pages/performers/VenueInfo.svelte +1 -1
- package/dist/components/pages/performers/VenueItemCard.spec.js +763 -763
- package/dist/components/pages/performers/VenueItemCard.svelte +4 -4
- package/dist/components/pages/profile/profile-form.spec.js +9 -9
- package/dist/components/pages/settings/tabs/CustomImageDropzone.svelte +3 -3
- package/dist/components/pages/shows/ShowList.spec.js +33 -33
- package/dist/components/pages/shows/TabContent.spec.js +90 -90
- package/dist/components/pages/shows/TabNavigation.spec.js +143 -143
- package/dist/config.js +5 -5
- package/dist/config.spec.js +29 -29
- package/dist/constants/formOptions.js +25 -25
- package/dist/constants/formOptions.spec.js +88 -88
- package/dist/index.js +111 -111
- package/dist/stores/auth.d.ts +9 -0
- package/dist/stores/auth.d.ts.map +1 -0
- package/dist/stores/auth.js +36 -0
- package/dist/stores/auth.spec.d.ts +2 -0
- package/dist/stores/auth.spec.d.ts.map +1 -0
- package/dist/stores/auth.spec.js +139 -0
- package/dist/stores/formDataStore.d.ts +17 -0
- package/dist/stores/formDataStore.d.ts.map +1 -0
- package/dist/stores/formDataStore.js +25 -0
- package/dist/stores/formDataStore.spec.d.ts +2 -0
- package/dist/stores/formDataStore.spec.d.ts.map +1 -0
- package/dist/stores/formDataStore.spec.js +257 -0
- package/dist/stores/formSave.d.ts +24 -0
- package/dist/stores/formSave.d.ts.map +1 -0
- package/dist/stores/formSave.js +132 -0
- package/dist/stores/formSave.spec.d.ts +2 -0
- package/dist/stores/formSave.spec.d.ts.map +1 -0
- package/dist/stores/formSave.spec.js +296 -0
- package/dist/stores/index.d.ts +1 -0
- package/dist/stores/index.d.ts.map +1 -0
- package/dist/stores/index.js +0 -0
- package/dist/stores/navigation.d.ts +5 -0
- package/dist/stores/navigation.d.ts.map +1 -0
- package/dist/stores/navigation.js +12 -0
- package/dist/stores/navigation.spec.d.ts +2 -0
- package/dist/stores/navigation.spec.d.ts.map +1 -0
- package/dist/stores/navigation.spec.js +136 -0
- package/dist/stores/toaster.d.ts +4 -0
- package/dist/stores/toaster.d.ts.map +1 -0
- package/dist/stores/toaster.js +13 -0
- package/dist/stores/toaster.spec.d.ts +2 -0
- package/dist/stores/toaster.spec.d.ts.map +1 -0
- package/dist/stores/toaster.spec.js +59 -0
- package/dist/telemetry.js +357 -357
- package/dist/telemetry.server.js +211 -211
- package/dist/telemetry.server.spec.js +434 -434
- package/dist/telemetry.spec.js +660 -660
- package/dist/utils/apiConfig.js +49 -49
- package/dist/utils/apiConfig.spec.js +118 -118
- package/dist/utils/greetings.js +187 -187
- package/dist/utils/greetings.spec.js +337 -337
- package/dist/utils/imageValidation.js +121 -121
- package/dist/utils/imageValidation.spec.js +220 -220
- package/dist/utils/portal.js +25 -25
- package/dist/utils/portal.spec.js +143 -143
- package/dist/utils/utils/utils.js +323 -323
- package/dist/utils/utils/utils.spec.js +698 -698
- package/dist/utils/utils.spec.js +643 -643
- package/package.json +1 -1
|
@@ -3,21 +3,21 @@
|
|
|
3
3
|
import Icon from "../../Icons/Icon.svelte";
|
|
4
4
|
import Input from "../../Input/Input.svelte";
|
|
5
5
|
import Modal from "../../Modal/Modal.svelte";
|
|
6
|
-
import { showToast } from "
|
|
7
|
-
import { microphonePlaceholder, getUserDetails } from "
|
|
6
|
+
import { showToast } from "../../stores/toaster";
|
|
7
|
+
import { microphonePlaceholder, getUserDetails } from "../../utils/utils";
|
|
8
8
|
import {
|
|
9
9
|
formatHour,
|
|
10
10
|
formattedDate,
|
|
11
11
|
formattedFullDate,
|
|
12
12
|
timeAgo,
|
|
13
|
-
} from "
|
|
13
|
+
} from "../../utils/utils/utils";
|
|
14
14
|
import {
|
|
15
15
|
acceptInvite,
|
|
16
16
|
declineInvite,
|
|
17
17
|
cancelInvite,
|
|
18
18
|
sendVenueMessage,
|
|
19
19
|
getEventUrl,
|
|
20
|
-
} from "
|
|
20
|
+
} from "../../../services/ShowService";
|
|
21
21
|
import { Dropdown, DropdownItem } from "flowbite-svelte";
|
|
22
22
|
import { DotsHorizontalOutline, FileCopyOutline } from "flowbite-svelte-icons";
|
|
23
23
|
import { onMount } from "svelte";
|
|
@@ -1,127 +1,127 @@
|
|
|
1
|
-
import { render, screen, fireEvent } from "@testing-library/svelte";
|
|
2
|
-
import userEvent from "@testing-library/user-event";
|
|
3
|
-
import { expect, describe, test, vi } from "vitest";
|
|
4
|
-
import SwitchOption from "./SwitchOption.svelte";
|
|
5
|
-
|
|
6
|
-
function setupTest(args = {}) {
|
|
7
|
-
const user = userEvent.setup();
|
|
8
|
-
const onToggle = vi.fn();
|
|
9
|
-
const { component, container } = render(SwitchOption, {
|
|
10
|
-
props: {
|
|
11
|
-
label: "Test Switch",
|
|
12
|
-
checked: false,
|
|
13
|
-
onToggle,
|
|
14
|
-
...args,
|
|
15
|
-
},
|
|
16
|
-
});
|
|
17
|
-
return { user, component, container, onToggle };
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
describe("SwitchOption Component Tests", () => {
|
|
21
|
-
test("Renders switch with label", () => {
|
|
22
|
-
setupTest({ label: "Enable Feature" });
|
|
23
|
-
expect(screen.getByText("Enable Feature")).toBeInTheDocument();
|
|
24
|
-
});
|
|
25
|
-
|
|
26
|
-
test("Renders checkbox input", () => {
|
|
27
|
-
setupTest();
|
|
28
|
-
const checkbox = screen.getByRole("checkbox");
|
|
29
|
-
expect(checkbox).toBeInTheDocument();
|
|
30
|
-
});
|
|
31
|
-
|
|
32
|
-
test("Checkbox is hidden for accessibility (sr-only)", () => {
|
|
33
|
-
setupTest();
|
|
34
|
-
const checkbox = screen.getByRole("checkbox");
|
|
35
|
-
expect(checkbox).toHaveClass("sr-only");
|
|
36
|
-
});
|
|
37
|
-
|
|
38
|
-
test("Checkbox is unchecked by default", () => {
|
|
39
|
-
setupTest({ checked: false });
|
|
40
|
-
const checkbox = screen.getByRole("checkbox");
|
|
41
|
-
expect(checkbox).not.toBeChecked();
|
|
42
|
-
});
|
|
43
|
-
|
|
44
|
-
test("Checkbox is checked when checked prop is true", () => {
|
|
45
|
-
setupTest({ checked: true });
|
|
46
|
-
const checkbox = screen.getByRole("checkbox");
|
|
47
|
-
expect(checkbox).toBeChecked();
|
|
48
|
-
});
|
|
49
|
-
|
|
50
|
-
test("Calls onToggle when clicked", async () => {
|
|
51
|
-
const { user, onToggle } = setupTest();
|
|
52
|
-
const checkbox = screen.getByRole("checkbox");
|
|
53
|
-
|
|
54
|
-
await user.click(checkbox);
|
|
55
|
-
|
|
56
|
-
expect(onToggle).toHaveBeenCalledTimes(1);
|
|
57
|
-
});
|
|
58
|
-
|
|
59
|
-
test("Passes checked state to onToggle", async () => {
|
|
60
|
-
const { user, onToggle } = setupTest({ checked: false });
|
|
61
|
-
const checkbox = screen.getByRole("checkbox");
|
|
62
|
-
|
|
63
|
-
await user.click(checkbox);
|
|
64
|
-
|
|
65
|
-
// After clicking, checkbox becomes checked (true)
|
|
66
|
-
expect(onToggle).toHaveBeenCalledWith(true);
|
|
67
|
-
});
|
|
68
|
-
|
|
69
|
-
test("Has full width container", () => {
|
|
70
|
-
const { container } = setupTest();
|
|
71
|
-
const wrapper = container.firstChild;
|
|
72
|
-
expect(wrapper).toHaveClass("w-full");
|
|
73
|
-
});
|
|
74
|
-
|
|
75
|
-
test("Uses flexbox layout", () => {
|
|
76
|
-
const { container } = setupTest();
|
|
77
|
-
const wrapper = container.firstChild;
|
|
78
|
-
expect(wrapper).toHaveClass("flex");
|
|
79
|
-
expect(wrapper).toHaveClass("items-center");
|
|
80
|
-
expect(wrapper).toHaveClass("justify-between");
|
|
81
|
-
});
|
|
82
|
-
|
|
83
|
-
test("Label has correct styling", () => {
|
|
84
|
-
setupTest({ label: "My Label" });
|
|
85
|
-
const label = screen.getByText("My Label");
|
|
86
|
-
expect(label).toHaveClass("text-sm");
|
|
87
|
-
expect(label).toHaveClass("font-medium");
|
|
88
|
-
expect(label).toHaveClass("text-gray-700");
|
|
89
|
-
});
|
|
90
|
-
|
|
91
|
-
test("Has pointer cursor on label", () => {
|
|
92
|
-
const { container } = setupTest();
|
|
93
|
-
const labelElement = container.querySelector("label");
|
|
94
|
-
expect(labelElement).toHaveClass("cursor-pointer");
|
|
95
|
-
});
|
|
96
|
-
|
|
97
|
-
test("Toggle track has correct base classes", () => {
|
|
98
|
-
const { container } = setupTest();
|
|
99
|
-
const track = container.querySelector(".bg-gray-200");
|
|
100
|
-
expect(track).toBeInTheDocument();
|
|
101
|
-
expect(track).toHaveClass("rounded-full");
|
|
102
|
-
expect(track).toHaveClass("w-11");
|
|
103
|
-
expect(track).toHaveClass("h-6");
|
|
104
|
-
});
|
|
105
|
-
|
|
106
|
-
test("Toggle knob has correct base classes", () => {
|
|
107
|
-
const { container } = setupTest();
|
|
108
|
-
const knob = container.querySelector(".bg-white");
|
|
109
|
-
expect(knob).toBeInTheDocument();
|
|
110
|
-
expect(knob).toHaveClass("rounded-full");
|
|
111
|
-
expect(knob).toHaveClass("w-5");
|
|
112
|
-
expect(knob).toHaveClass("h-5");
|
|
113
|
-
});
|
|
114
|
-
|
|
115
|
-
test("Updates visual state on toggle", async () => {
|
|
116
|
-
const { user, container, component } = setupTest({ checked: false });
|
|
117
|
-
const checkbox = screen.getByRole("checkbox");
|
|
118
|
-
|
|
119
|
-
// Initially unchecked
|
|
120
|
-
expect(checkbox).not.toBeChecked();
|
|
121
|
-
|
|
122
|
-
await user.click(checkbox);
|
|
123
|
-
|
|
124
|
-
// The component's checked state should update
|
|
125
|
-
expect(checkbox).toBeChecked();
|
|
126
|
-
});
|
|
127
|
-
});
|
|
1
|
+
import { render, screen, fireEvent } from "@testing-library/svelte";
|
|
2
|
+
import userEvent from "@testing-library/user-event";
|
|
3
|
+
import { expect, describe, test, vi } from "vitest";
|
|
4
|
+
import SwitchOption from "./SwitchOption.svelte";
|
|
5
|
+
|
|
6
|
+
function setupTest(args = {}) {
|
|
7
|
+
const user = userEvent.setup();
|
|
8
|
+
const onToggle = vi.fn();
|
|
9
|
+
const { component, container } = render(SwitchOption, {
|
|
10
|
+
props: {
|
|
11
|
+
label: "Test Switch",
|
|
12
|
+
checked: false,
|
|
13
|
+
onToggle,
|
|
14
|
+
...args,
|
|
15
|
+
},
|
|
16
|
+
});
|
|
17
|
+
return { user, component, container, onToggle };
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
describe("SwitchOption Component Tests", () => {
|
|
21
|
+
test("Renders switch with label", () => {
|
|
22
|
+
setupTest({ label: "Enable Feature" });
|
|
23
|
+
expect(screen.getByText("Enable Feature")).toBeInTheDocument();
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
test("Renders checkbox input", () => {
|
|
27
|
+
setupTest();
|
|
28
|
+
const checkbox = screen.getByRole("checkbox");
|
|
29
|
+
expect(checkbox).toBeInTheDocument();
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
test("Checkbox is hidden for accessibility (sr-only)", () => {
|
|
33
|
+
setupTest();
|
|
34
|
+
const checkbox = screen.getByRole("checkbox");
|
|
35
|
+
expect(checkbox).toHaveClass("sr-only");
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
test("Checkbox is unchecked by default", () => {
|
|
39
|
+
setupTest({ checked: false });
|
|
40
|
+
const checkbox = screen.getByRole("checkbox");
|
|
41
|
+
expect(checkbox).not.toBeChecked();
|
|
42
|
+
});
|
|
43
|
+
|
|
44
|
+
test("Checkbox is checked when checked prop is true", () => {
|
|
45
|
+
setupTest({ checked: true });
|
|
46
|
+
const checkbox = screen.getByRole("checkbox");
|
|
47
|
+
expect(checkbox).toBeChecked();
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
test("Calls onToggle when clicked", async () => {
|
|
51
|
+
const { user, onToggle } = setupTest();
|
|
52
|
+
const checkbox = screen.getByRole("checkbox");
|
|
53
|
+
|
|
54
|
+
await user.click(checkbox);
|
|
55
|
+
|
|
56
|
+
expect(onToggle).toHaveBeenCalledTimes(1);
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
test("Passes checked state to onToggle", async () => {
|
|
60
|
+
const { user, onToggle } = setupTest({ checked: false });
|
|
61
|
+
const checkbox = screen.getByRole("checkbox");
|
|
62
|
+
|
|
63
|
+
await user.click(checkbox);
|
|
64
|
+
|
|
65
|
+
// After clicking, checkbox becomes checked (true)
|
|
66
|
+
expect(onToggle).toHaveBeenCalledWith(true);
|
|
67
|
+
});
|
|
68
|
+
|
|
69
|
+
test("Has full width container", () => {
|
|
70
|
+
const { container } = setupTest();
|
|
71
|
+
const wrapper = container.firstChild;
|
|
72
|
+
expect(wrapper).toHaveClass("w-full");
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
test("Uses flexbox layout", () => {
|
|
76
|
+
const { container } = setupTest();
|
|
77
|
+
const wrapper = container.firstChild;
|
|
78
|
+
expect(wrapper).toHaveClass("flex");
|
|
79
|
+
expect(wrapper).toHaveClass("items-center");
|
|
80
|
+
expect(wrapper).toHaveClass("justify-between");
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
test("Label has correct styling", () => {
|
|
84
|
+
setupTest({ label: "My Label" });
|
|
85
|
+
const label = screen.getByText("My Label");
|
|
86
|
+
expect(label).toHaveClass("text-sm");
|
|
87
|
+
expect(label).toHaveClass("font-medium");
|
|
88
|
+
expect(label).toHaveClass("text-gray-700");
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
test("Has pointer cursor on label", () => {
|
|
92
|
+
const { container } = setupTest();
|
|
93
|
+
const labelElement = container.querySelector("label");
|
|
94
|
+
expect(labelElement).toHaveClass("cursor-pointer");
|
|
95
|
+
});
|
|
96
|
+
|
|
97
|
+
test("Toggle track has correct base classes", () => {
|
|
98
|
+
const { container } = setupTest();
|
|
99
|
+
const track = container.querySelector(".bg-gray-200");
|
|
100
|
+
expect(track).toBeInTheDocument();
|
|
101
|
+
expect(track).toHaveClass("rounded-full");
|
|
102
|
+
expect(track).toHaveClass("w-11");
|
|
103
|
+
expect(track).toHaveClass("h-6");
|
|
104
|
+
});
|
|
105
|
+
|
|
106
|
+
test("Toggle knob has correct base classes", () => {
|
|
107
|
+
const { container } = setupTest();
|
|
108
|
+
const knob = container.querySelector(".bg-white");
|
|
109
|
+
expect(knob).toBeInTheDocument();
|
|
110
|
+
expect(knob).toHaveClass("rounded-full");
|
|
111
|
+
expect(knob).toHaveClass("w-5");
|
|
112
|
+
expect(knob).toHaveClass("h-5");
|
|
113
|
+
});
|
|
114
|
+
|
|
115
|
+
test("Updates visual state on toggle", async () => {
|
|
116
|
+
const { user, container, component } = setupTest({ checked: false });
|
|
117
|
+
const checkbox = screen.getByRole("checkbox");
|
|
118
|
+
|
|
119
|
+
// Initially unchecked
|
|
120
|
+
expect(checkbox).not.toBeChecked();
|
|
121
|
+
|
|
122
|
+
await user.click(checkbox);
|
|
123
|
+
|
|
124
|
+
// The component's checked state should update
|
|
125
|
+
expect(checkbox).toBeChecked();
|
|
126
|
+
});
|
|
127
|
+
});
|
|
@@ -1,167 +1,167 @@
|
|
|
1
|
-
import { describe, it, expect, vi, beforeEach } from 'vitest';
|
|
2
|
-
import { render } from '@testing-library/svelte';
|
|
3
|
-
import VenueInfo from './VenueInfo.svelte';
|
|
4
|
-
|
|
5
|
-
// Mock the utils
|
|
6
|
-
vi.mock('@/utils/utils', () => ({
|
|
7
|
-
microphonePlaceholder: '/placeholder-mic.png',
|
|
8
|
-
parseLocation: vi.fn((location) => {
|
|
9
|
-
if (!location) return { street: '', cityStateZip: '' };
|
|
10
|
-
const parts = location.split(',').map((s) => s.trim());
|
|
11
|
-
if (parts.length >= 2) {
|
|
12
|
-
return {
|
|
13
|
-
street: parts[0],
|
|
14
|
-
cityStateZip: parts.slice(1).join(', '),
|
|
15
|
-
};
|
|
16
|
-
}
|
|
17
|
-
return { street: location, cityStateZip: '' };
|
|
18
|
-
}),
|
|
19
|
-
}));
|
|
20
|
-
|
|
21
|
-
describe('VenueInfo Component', () => {
|
|
22
|
-
const defaultProps = {
|
|
23
|
-
name: 'Comedy Club',
|
|
24
|
-
image: '/venue-image.jpg',
|
|
25
|
-
location: '123 Main St, Los Angeles, CA 90001',
|
|
26
|
-
};
|
|
27
|
-
|
|
28
|
-
beforeEach(() => {
|
|
29
|
-
vi.clearAllMocks();
|
|
30
|
-
});
|
|
31
|
-
|
|
32
|
-
it('renders the component', () => {
|
|
33
|
-
const { container } = render(VenueInfo, { props: defaultProps });
|
|
34
|
-
expect(container.querySelector('.venue-info')).toBeDefined();
|
|
35
|
-
});
|
|
36
|
-
|
|
37
|
-
it('renders the venue name', () => {
|
|
38
|
-
const { container } = render(VenueInfo, { props: defaultProps });
|
|
39
|
-
expect(container.querySelector('.venue-name').textContent).toBe('Comedy Club');
|
|
40
|
-
});
|
|
41
|
-
|
|
42
|
-
it('renders venue image with CDN URL for relative paths', () => {
|
|
43
|
-
const { container } = render(VenueInfo, { props: defaultProps });
|
|
44
|
-
const img = container.querySelector('.venue-image img');
|
|
45
|
-
expect(img.src).toBe('https://moxy.sfo3.digitaloceanspaces.com/venue-image.jpg');
|
|
46
|
-
});
|
|
47
|
-
|
|
48
|
-
it('renders venue image directly for http URLs', () => {
|
|
49
|
-
const props = { ...defaultProps, image: 'https://example.com/image.jpg' };
|
|
50
|
-
const { container } = render(VenueInfo, { props });
|
|
51
|
-
const img = container.querySelector('.venue-image img');
|
|
52
|
-
expect(img.src).toBe('https://example.com/image.jpg');
|
|
53
|
-
});
|
|
54
|
-
|
|
55
|
-
it('uses placeholder when image is empty', () => {
|
|
56
|
-
const props = { ...defaultProps, image: '' };
|
|
57
|
-
const { container } = render(VenueInfo, { props });
|
|
58
|
-
const img = container.querySelector('.venue-image img');
|
|
59
|
-
expect(img.src).toContain('placeholder-mic.png');
|
|
60
|
-
});
|
|
61
|
-
|
|
62
|
-
it('uses placeholder when image is whitespace only', () => {
|
|
63
|
-
const props = { ...defaultProps, image: ' ' };
|
|
64
|
-
const { container } = render(VenueInfo, { props });
|
|
65
|
-
const img = container.querySelector('.venue-image img');
|
|
66
|
-
expect(img.src).toContain('placeholder-mic.png');
|
|
67
|
-
});
|
|
68
|
-
|
|
69
|
-
it('renders street address', () => {
|
|
70
|
-
const { container } = render(VenueInfo, { props: defaultProps });
|
|
71
|
-
const locations = container.querySelectorAll('.venue-location');
|
|
72
|
-
expect(locations.length).toBeGreaterThan(0);
|
|
73
|
-
expect(locations[0].textContent).toBe('123 Main St');
|
|
74
|
-
});
|
|
75
|
-
|
|
76
|
-
it('renders city state zip', () => {
|
|
77
|
-
const { container } = render(VenueInfo, { props: defaultProps });
|
|
78
|
-
const locations = container.querySelectorAll('.venue-location');
|
|
79
|
-
expect(locations.length).toBe(2);
|
|
80
|
-
expect(locations[1].textContent).toBe('Los Angeles, CA 90001');
|
|
81
|
-
});
|
|
82
|
-
|
|
83
|
-
it('handles empty location', () => {
|
|
84
|
-
const props = { ...defaultProps, location: '' };
|
|
85
|
-
const { container } = render(VenueInfo, { props });
|
|
86
|
-
const locations = container.querySelectorAll('.venue-location');
|
|
87
|
-
expect(locations.length).toBe(0);
|
|
88
|
-
});
|
|
89
|
-
|
|
90
|
-
it('handles null location', () => {
|
|
91
|
-
const props = { ...defaultProps, location: null };
|
|
92
|
-
const { container } = render(VenueInfo, { props });
|
|
93
|
-
expect(container.querySelector('.venue-info')).toBeDefined();
|
|
94
|
-
});
|
|
95
|
-
|
|
96
|
-
it('handles undefined location', () => {
|
|
97
|
-
const props = { ...defaultProps, location: undefined };
|
|
98
|
-
const { container } = render(VenueInfo, { props });
|
|
99
|
-
expect(container.querySelector('.venue-info')).toBeDefined();
|
|
100
|
-
});
|
|
101
|
-
|
|
102
|
-
it('sets image alt text to venue name', () => {
|
|
103
|
-
const { container } = render(VenueInfo, { props: defaultProps });
|
|
104
|
-
const img = container.querySelector('.venue-image img');
|
|
105
|
-
expect(img.alt).toBe('Comedy Club');
|
|
106
|
-
});
|
|
107
|
-
|
|
108
|
-
it('renders with default size', () => {
|
|
109
|
-
const { container } = render(VenueInfo, { props: defaultProps });
|
|
110
|
-
expect(container.querySelector('.venue-info--small')).toBeNull();
|
|
111
|
-
});
|
|
112
|
-
|
|
113
|
-
it('renders with small size', () => {
|
|
114
|
-
const props = { ...defaultProps, size: 'small' };
|
|
115
|
-
const { container } = render(VenueInfo, { props });
|
|
116
|
-
expect(container.querySelector('.venue-info--small')).toBeDefined();
|
|
117
|
-
});
|
|
118
|
-
|
|
119
|
-
it('renders venue-image container', () => {
|
|
120
|
-
const { container } = render(VenueInfo, { props: defaultProps });
|
|
121
|
-
expect(container.querySelector('.venue-image')).toBeDefined();
|
|
122
|
-
});
|
|
123
|
-
|
|
124
|
-
it('renders venue-details container', () => {
|
|
125
|
-
const { container } = render(VenueInfo, { props: defaultProps });
|
|
126
|
-
expect(container.querySelector('.venue-details')).toBeDefined();
|
|
127
|
-
});
|
|
128
|
-
|
|
129
|
-
it('handles empty name', () => {
|
|
130
|
-
const props = { ...defaultProps, name: '' };
|
|
131
|
-
const { container } = render(VenueInfo, { props });
|
|
132
|
-
expect(container.querySelector('.venue-name').textContent).toBe('');
|
|
133
|
-
});
|
|
134
|
-
|
|
135
|
-
it('handles location with single part', () => {
|
|
136
|
-
const props = { ...defaultProps, location: 'Los Angeles' };
|
|
137
|
-
const { container } = render(VenueInfo, { props });
|
|
138
|
-
const locations = container.querySelectorAll('.venue-location');
|
|
139
|
-
expect(locations.length).toBeGreaterThan(0);
|
|
140
|
-
});
|
|
141
|
-
|
|
142
|
-
it('handles https image URLs', () => {
|
|
143
|
-
const props = { ...defaultProps, image: 'https://secure.example.com/image.jpg' };
|
|
144
|
-
const { container } = render(VenueInfo, { props });
|
|
145
|
-
const img = container.querySelector('.venue-image img');
|
|
146
|
-
expect(img.src).toBe('https://secure.example.com/image.jpg');
|
|
147
|
-
});
|
|
148
|
-
|
|
149
|
-
it('handles http image URLs', () => {
|
|
150
|
-
const props = { ...defaultProps, image: 'http://example.com/image.jpg' };
|
|
151
|
-
const { container } = render(VenueInfo, { props });
|
|
152
|
-
const img = container.querySelector('.venue-image img');
|
|
153
|
-
expect(img.src).toBe('http://example.com/image.jpg');
|
|
154
|
-
});
|
|
155
|
-
|
|
156
|
-
it('renders with very long name', () => {
|
|
157
|
-
const props = { ...defaultProps, name: 'A Very Long Venue Name That Might Need To Be Truncated In The UI' };
|
|
158
|
-
const { container } = render(VenueInfo, { props });
|
|
159
|
-
expect(container.querySelector('.venue-name').textContent).toBe('A Very Long Venue Name That Might Need To Be Truncated In The UI');
|
|
160
|
-
});
|
|
161
|
-
|
|
162
|
-
it('renders with special characters in name', () => {
|
|
163
|
-
const props = { ...defaultProps, name: "Joe's Comedy & Bar" };
|
|
164
|
-
const { container } = render(VenueInfo, { props });
|
|
165
|
-
expect(container.querySelector('.venue-name').textContent).toBe("Joe's Comedy & Bar");
|
|
166
|
-
});
|
|
167
|
-
});
|
|
1
|
+
import { describe, it, expect, vi, beforeEach } from 'vitest';
|
|
2
|
+
import { render } from '@testing-library/svelte';
|
|
3
|
+
import VenueInfo from './VenueInfo.svelte';
|
|
4
|
+
|
|
5
|
+
// Mock the utils
|
|
6
|
+
vi.mock('@/utils/utils', () => ({
|
|
7
|
+
microphonePlaceholder: '/placeholder-mic.png',
|
|
8
|
+
parseLocation: vi.fn((location) => {
|
|
9
|
+
if (!location) return { street: '', cityStateZip: '' };
|
|
10
|
+
const parts = location.split(',').map((s) => s.trim());
|
|
11
|
+
if (parts.length >= 2) {
|
|
12
|
+
return {
|
|
13
|
+
street: parts[0],
|
|
14
|
+
cityStateZip: parts.slice(1).join(', '),
|
|
15
|
+
};
|
|
16
|
+
}
|
|
17
|
+
return { street: location, cityStateZip: '' };
|
|
18
|
+
}),
|
|
19
|
+
}));
|
|
20
|
+
|
|
21
|
+
describe('VenueInfo Component', () => {
|
|
22
|
+
const defaultProps = {
|
|
23
|
+
name: 'Comedy Club',
|
|
24
|
+
image: '/venue-image.jpg',
|
|
25
|
+
location: '123 Main St, Los Angeles, CA 90001',
|
|
26
|
+
};
|
|
27
|
+
|
|
28
|
+
beforeEach(() => {
|
|
29
|
+
vi.clearAllMocks();
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
it('renders the component', () => {
|
|
33
|
+
const { container } = render(VenueInfo, { props: defaultProps });
|
|
34
|
+
expect(container.querySelector('.venue-info')).toBeDefined();
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
it('renders the venue name', () => {
|
|
38
|
+
const { container } = render(VenueInfo, { props: defaultProps });
|
|
39
|
+
expect(container.querySelector('.venue-name').textContent).toBe('Comedy Club');
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
it('renders venue image with CDN URL for relative paths', () => {
|
|
43
|
+
const { container } = render(VenueInfo, { props: defaultProps });
|
|
44
|
+
const img = container.querySelector('.venue-image img');
|
|
45
|
+
expect(img.src).toBe('https://moxy.sfo3.digitaloceanspaces.com/venue-image.jpg');
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
it('renders venue image directly for http URLs', () => {
|
|
49
|
+
const props = { ...defaultProps, image: 'https://example.com/image.jpg' };
|
|
50
|
+
const { container } = render(VenueInfo, { props });
|
|
51
|
+
const img = container.querySelector('.venue-image img');
|
|
52
|
+
expect(img.src).toBe('https://example.com/image.jpg');
|
|
53
|
+
});
|
|
54
|
+
|
|
55
|
+
it('uses placeholder when image is empty', () => {
|
|
56
|
+
const props = { ...defaultProps, image: '' };
|
|
57
|
+
const { container } = render(VenueInfo, { props });
|
|
58
|
+
const img = container.querySelector('.venue-image img');
|
|
59
|
+
expect(img.src).toContain('placeholder-mic.png');
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
it('uses placeholder when image is whitespace only', () => {
|
|
63
|
+
const props = { ...defaultProps, image: ' ' };
|
|
64
|
+
const { container } = render(VenueInfo, { props });
|
|
65
|
+
const img = container.querySelector('.venue-image img');
|
|
66
|
+
expect(img.src).toContain('placeholder-mic.png');
|
|
67
|
+
});
|
|
68
|
+
|
|
69
|
+
it('renders street address', () => {
|
|
70
|
+
const { container } = render(VenueInfo, { props: defaultProps });
|
|
71
|
+
const locations = container.querySelectorAll('.venue-location');
|
|
72
|
+
expect(locations.length).toBeGreaterThan(0);
|
|
73
|
+
expect(locations[0].textContent).toBe('123 Main St');
|
|
74
|
+
});
|
|
75
|
+
|
|
76
|
+
it('renders city state zip', () => {
|
|
77
|
+
const { container } = render(VenueInfo, { props: defaultProps });
|
|
78
|
+
const locations = container.querySelectorAll('.venue-location');
|
|
79
|
+
expect(locations.length).toBe(2);
|
|
80
|
+
expect(locations[1].textContent).toBe('Los Angeles, CA 90001');
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
it('handles empty location', () => {
|
|
84
|
+
const props = { ...defaultProps, location: '' };
|
|
85
|
+
const { container } = render(VenueInfo, { props });
|
|
86
|
+
const locations = container.querySelectorAll('.venue-location');
|
|
87
|
+
expect(locations.length).toBe(0);
|
|
88
|
+
});
|
|
89
|
+
|
|
90
|
+
it('handles null location', () => {
|
|
91
|
+
const props = { ...defaultProps, location: null };
|
|
92
|
+
const { container } = render(VenueInfo, { props });
|
|
93
|
+
expect(container.querySelector('.venue-info')).toBeDefined();
|
|
94
|
+
});
|
|
95
|
+
|
|
96
|
+
it('handles undefined location', () => {
|
|
97
|
+
const props = { ...defaultProps, location: undefined };
|
|
98
|
+
const { container } = render(VenueInfo, { props });
|
|
99
|
+
expect(container.querySelector('.venue-info')).toBeDefined();
|
|
100
|
+
});
|
|
101
|
+
|
|
102
|
+
it('sets image alt text to venue name', () => {
|
|
103
|
+
const { container } = render(VenueInfo, { props: defaultProps });
|
|
104
|
+
const img = container.querySelector('.venue-image img');
|
|
105
|
+
expect(img.alt).toBe('Comedy Club');
|
|
106
|
+
});
|
|
107
|
+
|
|
108
|
+
it('renders with default size', () => {
|
|
109
|
+
const { container } = render(VenueInfo, { props: defaultProps });
|
|
110
|
+
expect(container.querySelector('.venue-info--small')).toBeNull();
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
it('renders with small size', () => {
|
|
114
|
+
const props = { ...defaultProps, size: 'small' };
|
|
115
|
+
const { container } = render(VenueInfo, { props });
|
|
116
|
+
expect(container.querySelector('.venue-info--small')).toBeDefined();
|
|
117
|
+
});
|
|
118
|
+
|
|
119
|
+
it('renders venue-image container', () => {
|
|
120
|
+
const { container } = render(VenueInfo, { props: defaultProps });
|
|
121
|
+
expect(container.querySelector('.venue-image')).toBeDefined();
|
|
122
|
+
});
|
|
123
|
+
|
|
124
|
+
it('renders venue-details container', () => {
|
|
125
|
+
const { container } = render(VenueInfo, { props: defaultProps });
|
|
126
|
+
expect(container.querySelector('.venue-details')).toBeDefined();
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
it('handles empty name', () => {
|
|
130
|
+
const props = { ...defaultProps, name: '' };
|
|
131
|
+
const { container } = render(VenueInfo, { props });
|
|
132
|
+
expect(container.querySelector('.venue-name').textContent).toBe('');
|
|
133
|
+
});
|
|
134
|
+
|
|
135
|
+
it('handles location with single part', () => {
|
|
136
|
+
const props = { ...defaultProps, location: 'Los Angeles' };
|
|
137
|
+
const { container } = render(VenueInfo, { props });
|
|
138
|
+
const locations = container.querySelectorAll('.venue-location');
|
|
139
|
+
expect(locations.length).toBeGreaterThan(0);
|
|
140
|
+
});
|
|
141
|
+
|
|
142
|
+
it('handles https image URLs', () => {
|
|
143
|
+
const props = { ...defaultProps, image: 'https://secure.example.com/image.jpg' };
|
|
144
|
+
const { container } = render(VenueInfo, { props });
|
|
145
|
+
const img = container.querySelector('.venue-image img');
|
|
146
|
+
expect(img.src).toBe('https://secure.example.com/image.jpg');
|
|
147
|
+
});
|
|
148
|
+
|
|
149
|
+
it('handles http image URLs', () => {
|
|
150
|
+
const props = { ...defaultProps, image: 'http://example.com/image.jpg' };
|
|
151
|
+
const { container } = render(VenueInfo, { props });
|
|
152
|
+
const img = container.querySelector('.venue-image img');
|
|
153
|
+
expect(img.src).toBe('http://example.com/image.jpg');
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
it('renders with very long name', () => {
|
|
157
|
+
const props = { ...defaultProps, name: 'A Very Long Venue Name That Might Need To Be Truncated In The UI' };
|
|
158
|
+
const { container } = render(VenueInfo, { props });
|
|
159
|
+
expect(container.querySelector('.venue-name').textContent).toBe('A Very Long Venue Name That Might Need To Be Truncated In The UI');
|
|
160
|
+
});
|
|
161
|
+
|
|
162
|
+
it('renders with special characters in name', () => {
|
|
163
|
+
const props = { ...defaultProps, name: "Joe's Comedy & Bar" };
|
|
164
|
+
const { container } = render(VenueInfo, { props });
|
|
165
|
+
expect(container.querySelector('.venue-name').textContent).toBe("Joe's Comedy & Bar");
|
|
166
|
+
});
|
|
167
|
+
});
|