@zap-wunschlachen/wl-shared-components 1.0.16 → 1.0.18

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.
Files changed (236) hide show
  1. package/.github/workflows/playwright.yml +205 -205
  2. package/.github/workflows/static.yml +61 -61
  3. package/.github/workflows/update-snapshots.yml +37 -37
  4. package/.prettierrc +5 -5
  5. package/.storybook/main.ts +18 -18
  6. package/.storybook/preview.ts +37 -37
  7. package/.storybook/storyWrapper.vue +18 -18
  8. package/.storybook/withVuetifyTheme.decorator.ts +21 -21
  9. package/App.vue +149 -149
  10. package/README.md +56 -56
  11. package/heroicons.ts +75 -75
  12. package/index.html +19 -19
  13. package/package.json +67 -67
  14. package/playwright.config.ts +48 -48
  15. package/public/background.svg +60 -60
  16. package/public/style.css +187 -187
  17. package/public/technologies.svg +22 -22
  18. package/src/assets/css/base.css +235 -235
  19. package/src/assets/css/variables.css +96 -96
  20. package/src/components/Accordion/Accordion.css +59 -59
  21. package/src/components/Accordion/AccordionGroup.vue +51 -51
  22. package/src/components/Accordion/AccordionItem.vue +66 -66
  23. package/src/components/Appointment/Card/Actions.css +34 -28
  24. package/src/components/Appointment/Card/Actions.vue +73 -72
  25. package/src/components/Appointment/Card/AnamneseNotification.css +16 -0
  26. package/src/components/Appointment/Card/AnamneseNotification.vue +24 -0
  27. package/src/components/Appointment/Card/Card.css +80 -80
  28. package/src/components/Appointment/Card/Card.vue +93 -87
  29. package/src/components/Appointment/Card/Details.css +50 -50
  30. package/src/components/Appointment/Card/Details.vue +43 -43
  31. package/src/components/Audio/Audio.vue +187 -187
  32. package/src/components/Audio/Waveform.vue +117 -117
  33. package/src/components/Button/Button.vue +119 -119
  34. package/src/components/CheckBox/CheckBox.css +185 -185
  35. package/src/components/CheckBox/Checkbox.vue +130 -130
  36. package/src/components/DateInput/DateInput.css +2 -2
  37. package/src/components/DateInput/DateInput.vue +262 -262
  38. package/src/components/Dialog/Dialog.css +6 -6
  39. package/src/components/Dialog/Dialog.vue +29 -29
  40. package/src/components/EditField/EditField.css +19 -19
  41. package/src/components/EditField/EditField.vue +202 -202
  42. package/src/components/IconBullet/IconBullet.vue +86 -86
  43. package/src/components/IconBullet/IconBulletList.vue +41 -41
  44. package/src/components/Icons/AdvanceAppointments.vue +153 -153
  45. package/src/components/Icons/Audio/CloudFailed.vue +20 -20
  46. package/src/components/Icons/Audio/CloudSaved.vue +21 -21
  47. package/src/components/Icons/Audio/Delete.vue +15 -15
  48. package/src/components/Icons/Audio/Pause.vue +18 -18
  49. package/src/components/Icons/Audio/Play.vue +15 -15
  50. package/src/components/Icons/CalendarNotification.vue +126 -126
  51. package/src/components/Icons/Chair.vue +32 -32
  52. package/src/components/Icons/ChairNotification.vue +35 -35
  53. package/src/components/Icons/Circle.vue +66 -66
  54. package/src/components/Icons/FavIcon.vue +22 -22
  55. package/src/components/Icons/FilledCircle.vue +11 -11
  56. package/src/components/Icons/Group3.vue +46 -46
  57. package/src/components/Icons/RingNotification.vue +54 -54
  58. package/src/components/Icons/SolidArrowRight.vue +14 -14
  59. package/src/components/Icons/calendar.vue +17 -17
  60. package/src/components/Icons/checkbox.vue +19 -19
  61. package/src/components/Icons/outlineChecked.vue +27 -27
  62. package/src/components/Icons/play.vue +5 -5
  63. package/src/components/Input/Input.css +187 -187
  64. package/src/components/Input/Input.vue +247 -247
  65. package/src/components/Laboratory/AppointmentCard/AppointmentCard.css +7 -7
  66. package/src/components/Laboratory/AppointmentCard/AppointmentCard.vue +116 -116
  67. package/src/components/Laboratory/ChatBoxImage/ChatBoxImage.vue +81 -81
  68. package/src/components/Laboratory/ChatMessage/ChatMessage.vue +113 -113
  69. package/src/components/Laboratory/ChatMessage/ChatMessageBadge.css +4 -4
  70. package/src/components/Laboratory/ChatMessage/ChatMessageBadge.vue +99 -99
  71. package/src/components/Laboratory/ChatNotification/ChatNotification.vue +130 -130
  72. package/src/components/Laboratory/DocumentCard/DocumentCard.css +3 -3
  73. package/src/components/Laboratory/DocumentCard/DocumentCard.vue +50 -50
  74. package/src/components/Laboratory/DocumentCard/DocumentCardItem.vue +53 -53
  75. package/src/components/Laboratory/InfoCard/InfoCard.vue +162 -162
  76. package/src/components/Laboratory/MainColumnsBar/MainColumnsBar.vue +102 -102
  77. package/src/components/Laboratory/ProgressCircle/ProgressCircle.vue +152 -152
  78. package/src/components/Laboratory/ProgressLinear/ProgressLinear.css +33 -33
  79. package/src/components/Laboratory/ProgressLinear/ProgressLinear.vue +75 -75
  80. package/src/components/Laboratory/SelectionColumnBar/SelectionColumnBar.vue +92 -92
  81. package/src/components/Laboratory/StatusNotification/StatusNotification.vue +49 -49
  82. package/src/components/Laboratory/TagLabel/TagLabel.vue +126 -126
  83. package/src/components/Laboratory/TagLabelGroup/TagLabelGroup.vue +97 -97
  84. package/src/components/Laboratory/TicketCard/TicketCard.css +3 -3
  85. package/src/components/Laboratory/TicketCard/TicketCard.vue +143 -143
  86. package/src/components/Laboratory/TimeLine/TimeLineEvent.css +18 -18
  87. package/src/components/Laboratory/TimeLine/TimeLineEvent.vue +119 -119
  88. package/src/components/Laboratory/TimeLine/Timeline.css +4 -4
  89. package/src/components/Laboratory/TimeLine/Timeline.vue +30 -30
  90. package/src/components/Loader/Loader.css +51 -51
  91. package/src/components/Modal/Modal.css +5 -5
  92. package/src/components/Modal/Modal.vue +22 -22
  93. package/src/components/NotificationBubble/NotificationBubble.css +4 -4
  94. package/src/components/NotificationBubble/NotificationBubble.vue +90 -90
  95. package/src/components/OtpInput/OtpInput.css +39 -39
  96. package/src/components/OtpInput/OtpInput.vue +143 -143
  97. package/src/components/PhoneInput/PhoneInput.css +31 -31
  98. package/src/components/PhoneInput/PhoneInput.vue +113 -113
  99. package/src/components/Select/Select.css +150 -150
  100. package/src/components/Select/Select.vue +304 -304
  101. package/src/components/TextArea/TextArea.css +3 -3
  102. package/src/components/TextArea/TextArea.vue +126 -126
  103. package/src/components/TickBox/TickBox.css +49 -49
  104. package/src/components/TickBox/TickBox.vue +126 -126
  105. package/src/components/index.ts +24 -23
  106. package/src/constants/iconEnums.ts +3 -3
  107. package/src/i18n/i18n.ts +15 -15
  108. package/src/i18n/locales/de.json +30 -27
  109. package/src/i18n/locales/en.json +30 -27
  110. package/src/index.ts +34 -34
  111. package/src/main.ts +11 -11
  112. package/src/plugins/vuetify.ts +131 -131
  113. package/src/shims-vue.d.ts +10 -10
  114. package/src/stories/Accordion.stories.ts +650 -650
  115. package/src/stories/Audio.stories.ts +28 -28
  116. package/src/stories/Button.stories.ts +263 -263
  117. package/src/stories/CheckBox.stories.ts +348 -348
  118. package/src/stories/DateInput.stories.ts +53 -53
  119. package/src/stories/Dialog.stories.ts +147 -147
  120. package/src/stories/EditField.stories.ts +78 -78
  121. package/src/stories/IconBullet/IconBullet.stories.ts +201 -201
  122. package/src/stories/IconBullet/IconBulletList.stories.ts +275 -275
  123. package/src/stories/Input.stories.ts +351 -351
  124. package/src/stories/Laboratory/Cards/AppointmentCard/AppointmentCard.stories.ts +260 -260
  125. package/src/stories/Laboratory/Cards/DocumentCard/DocumentCard.stories.ts +176 -176
  126. package/src/stories/Laboratory/Cards/DocumentCard/DocumentCardItem.stories.ts +119 -119
  127. package/src/stories/Laboratory/Cards/InfoCard/InfoCard.stories.ts +320 -320
  128. package/src/stories/Laboratory/Cards/TicketCard/TicketCard.stories.ts +335 -335
  129. package/src/stories/Laboratory/Chat/ChatBoxImage.stories.ts +82 -82
  130. package/src/stories/Laboratory/Chat/ChatMessage.stories.ts +198 -198
  131. package/src/stories/Laboratory/Chat/ChatMessageBadge.stories.ts +204 -204
  132. package/src/stories/Laboratory/Chat/ChatNotification.stories.ts +144 -144
  133. package/src/stories/Laboratory/Chat/ProgressLinear.stories.ts +186 -186
  134. package/src/stories/Laboratory/Chat/StatusNotification.stories.ts +111 -111
  135. package/src/stories/Laboratory/MainColumnsBar.stories.ts +48 -48
  136. package/src/stories/Laboratory/ProgressCircle.stories.ts +261 -261
  137. package/src/stories/Laboratory/SelectionColumnBar.stories.ts +234 -234
  138. package/src/stories/Laboratory/TagLabel.stories.ts +418 -418
  139. package/src/stories/Laboratory/TagLabelGroup.stories.ts +234 -234
  140. package/src/stories/Laboratory/Timeline.stories.ts +403 -403
  141. package/src/stories/NotificationBubble.stories.ts +194 -194
  142. package/src/stories/OtpInput.stories.ts +100 -100
  143. package/src/stories/PhoneInput.stories.ts +52 -52
  144. package/src/stories/Select.stories.ts +419 -419
  145. package/src/stories/TextArea.stories.ts +112 -112
  146. package/src/stories/TickBox.stories.ts +294 -294
  147. package/src/stories/v-icon.stories.ts +91 -91
  148. package/src/types/index.ts +1 -0
  149. package/src/utils/index.ts +100 -100
  150. package/src/vite-env.d.ts +1 -1
  151. package/tests/e2e/README.md +220 -220
  152. package/tests/e2e/accessibility.spec.ts +638 -638
  153. package/tests/e2e/accordion.spec.ts +42 -42
  154. package/tests/e2e/additional-components.spec.ts +437 -437
  155. package/tests/e2e/all-components.spec.ts +135 -135
  156. package/tests/e2e/appointment-card.spec.ts +816 -816
  157. package/tests/e2e/button-fixed.spec.ts +58 -58
  158. package/tests/e2e/button.spec.ts +76 -76
  159. package/tests/e2e/checkbox.spec.ts +50 -50
  160. package/tests/e2e/date-input.spec.ts +46 -46
  161. package/tests/e2e/debug.spec.ts +51 -51
  162. package/tests/e2e/dialog.spec.ts +58 -58
  163. package/tests/e2e/input.spec.ts +55 -55
  164. package/tests/e2e/laboratory-components.spec.ts +320 -320
  165. package/tests/e2e/otp-input.spec.ts +50 -50
  166. package/tests/e2e/select.spec.ts +52 -52
  167. package/tests/e2e/storybook-utils.ts +59 -59
  168. package/tests/e2e/test-basic.spec.ts +33 -33
  169. package/tests/e2e/visual-regression.spec.ts +350 -350
  170. package/tests/unit/components/Accordion/AccordionGroup.spec.ts.skip +342 -342
  171. package/tests/unit/components/Accordion/AccordionItem.spec.ts.skip +383 -383
  172. package/tests/unit/components/Appointment/Card/Actions.spec.ts +407 -407
  173. package/tests/unit/components/Appointment/Card/Card.spec.ts +485 -485
  174. package/tests/unit/components/Appointment/Card/Details.spec.ts +397 -397
  175. package/tests/unit/components/Audio/Audio.spec.ts +403 -403
  176. package/tests/unit/components/Audio/Waveform.spec.ts +483 -483
  177. package/tests/unit/components/Core/Button.spec.ts +336 -336
  178. package/tests/unit/components/Core/Checkbox.spec.ts +544 -544
  179. package/tests/unit/components/Core/DateInput.spec.ts +690 -690
  180. package/tests/unit/components/Core/Dialog.spec.ts +485 -485
  181. package/tests/unit/components/Core/EditField.spec.ts +782 -782
  182. package/tests/unit/components/Core/Input.spec.ts +512 -512
  183. package/tests/unit/components/Core/Modal.spec.ts +518 -518
  184. package/tests/unit/components/Core/NotificationBubble.spec.ts +606 -606
  185. package/tests/unit/components/Core/OtpInput.spec.ts +708 -708
  186. package/tests/unit/components/Core/PhoneInput.spec.ts +619 -619
  187. package/tests/unit/components/Core/Select.spec.ts +712 -712
  188. package/tests/unit/components/Core/TextArea.spec.ts +565 -565
  189. package/tests/unit/components/Core/TickBox.spec.ts +779 -779
  190. package/tests/unit/components/IconBullet/IconBullet.spec.ts +356 -356
  191. package/tests/unit/components/IconBullet/IconBulletList.spec.ts +371 -371
  192. package/tests/unit/components/Icons/Audio/CloudFailed.spec.ts +108 -108
  193. package/tests/unit/components/Icons/Audio/CloudSaved.spec.ts +149 -149
  194. package/tests/unit/components/Icons/Audio/Delete.spec.ts +158 -158
  195. package/tests/unit/components/Icons/Audio/Pause.spec.ts +208 -208
  196. package/tests/unit/components/Icons/Audio/Play.spec.ts +217 -217
  197. package/tests/unit/components/Icons/CalendarNotification.spec.ts +186 -186
  198. package/tests/unit/components/Icons/Chair.spec.ts +234 -234
  199. package/tests/unit/components/Icons/ChairNotification.spec.ts +311 -311
  200. package/tests/unit/components/Icons/Circle.spec.ts +255 -255
  201. package/tests/unit/components/Icons/FavIcon.spec.ts +251 -251
  202. package/tests/unit/components/Icons/FilledCircle.spec.ts +274 -274
  203. package/tests/unit/components/Icons/Group3.spec.ts +355 -355
  204. package/tests/unit/components/Icons/RingNotification.spec.ts +393 -393
  205. package/tests/unit/components/Icons/calendar.spec.ts +286 -286
  206. package/tests/unit/components/Icons/checkbox.spec.ts +315 -315
  207. package/tests/unit/components/Icons/outlineChecked.spec.ts +434 -434
  208. package/tests/unit/components/Icons/play.spec.ts +308 -308
  209. package/tests/unit/components/Laboratory/AppointmentCard.spec.ts +167 -167
  210. package/tests/unit/components/Laboratory/ChatBoxImage.spec.ts +179 -179
  211. package/tests/unit/components/Laboratory/ChatMessage.spec.ts +263 -263
  212. package/tests/unit/components/Laboratory/ChatMessageBadge.spec.ts +282 -282
  213. package/tests/unit/components/Laboratory/ChatNotification.spec.ts +256 -256
  214. package/tests/unit/components/Laboratory/DocumentCard.spec.ts +228 -228
  215. package/tests/unit/components/Laboratory/DocumentCardItem.spec.ts +236 -236
  216. package/tests/unit/components/Laboratory/InfoCard.spec.ts +308 -308
  217. package/tests/unit/components/Laboratory/MainColumnsBar.spec.ts +251 -251
  218. package/tests/unit/components/Laboratory/ProgressCircle.spec.ts +290 -290
  219. package/tests/unit/components/Laboratory/ProgressLinear.spec.ts +275 -275
  220. package/tests/unit/components/Laboratory/SelectionColumnBar.spec.ts +288 -288
  221. package/tests/unit/components/Laboratory/StatusNotification.spec.ts +296 -296
  222. package/tests/unit/components/Laboratory/TagLabel.spec.ts +353 -353
  223. package/tests/unit/components/Laboratory/TagLabelGroup.spec.ts +377 -377
  224. package/tests/unit/components/Laboratory/TicketCard.spec.ts +351 -351
  225. package/tests/unit/components/Laboratory/TimeLineEvent.spec.ts +381 -381
  226. package/tests/unit/components/Laboratory/Timeline.spec.ts +419 -419
  227. package/tests/unit/constants/iconEnums.spec.ts +39 -39
  228. package/tests/unit/i18n/i18n.spec.ts +88 -88
  229. package/tests/unit/plugins/vuetify.spec.ts +220 -220
  230. package/tests/unit/setup.ts +189 -189
  231. package/tests/unit/src/components/index.spec.ts.skip +192 -192
  232. package/tests/unit/src/index.spec.ts.skip +182 -182
  233. package/tests/unit/src/main.spec.ts +151 -151
  234. package/tsconfig.json +26 -26
  235. package/vite.config.ts +29 -29
  236. package/vitest.config.ts +83 -83
@@ -1,486 +1,486 @@
1
- import { describe, it, expect, beforeEach, vi } from 'vitest';
2
- import { mount, VueWrapper } from '@vue/test-utils';
3
- import { format } from 'date-fns';
4
- import { de } from 'date-fns/locale';
5
- import Card from '../../../../../src/components/Appointment/Card/Card.vue';
6
- import type { AppointmentData, Dentist } from '../../../../../src/types';
7
-
8
- describe('Appointment Card', () => {
9
- let wrapper: VueWrapper;
10
-
11
- const createMockDentist = (overrides: Partial<Dentist> = {}): Dentist => ({
12
- name: 'Dr. Test Dentist',
13
- gender: 'Male',
14
- imageSrc: 'https://example.com/dentist.jpg',
15
- ...overrides
16
- });
17
-
18
- const createMockAppointment = (overrides: Partial<AppointmentData> = {}): AppointmentData => ({
19
- id: '123',
20
- template_name: 'Test Treatment',
21
- description: 'Test Description',
22
- dentist: createMockDentist(),
23
- start: '2024-01-15T10:30:00.000Z',
24
- type: 1,
25
- status: 'upcoming',
26
- patientName: 'John Doe',
27
- address: 'Test Address 123',
28
- district: 'Test District',
29
- is_confirmed: false,
30
- ...overrides
31
- });
32
-
33
- beforeEach(() => {
34
- const appointment = createMockAppointment();
35
- wrapper = mount(Card, {
36
- props: { appointment },
37
- global: {
38
- stubs: {
39
- Icon: {
40
- template: '<div class="icon-stub" :name="name" :size="size"></div>',
41
- props: ['name', 'size']
42
- },
43
- Details: {
44
- template: '<div class="details-stub">Details Component</div>',
45
- props: ['appointment', 'dentistImageSrc']
46
- },
47
- Actions: {
48
- template: '<div class="actions-stub" @confirmed="$emit(\'confirmed\', $event)" @cancelled="$emit(\'cancelled\', $event)" @rescheduled="$emit(\'rescheduled\', $event)">Actions Component</div>',
49
- props: ['appointment'],
50
- emits: ['confirmed', 'cancelled', 'rescheduled']
51
- }
52
- }
53
- }
54
- });
55
- });
56
-
57
- describe('Component Rendering', () => {
58
- it('should render the component with correct test id', () => {
59
- expect(wrapper.find('[data-testid="root"]').exists()).toBe(true);
60
- });
61
-
62
- it('should render the appointment card structure', () => {
63
- expect(wrapper.find('.appointment-card').exists()).toBe(true);
64
- expect(wrapper.find('.card-header').exists()).toBe(true);
65
- expect(wrapper.find('.card-body').exists()).toBe(true);
66
- expect(wrapper.find('.card-footer').exists()).toBe(true);
67
- });
68
-
69
- it('should render Details and Actions components', () => {
70
- expect(wrapper.find('.details-stub').exists()).toBe(true);
71
- expect(wrapper.find('.actions-stub').exists()).toBe(true);
72
- });
73
- });
74
-
75
- describe('Header Information', () => {
76
- it('should display patient name', () => {
77
- expect(wrapper.text()).toContain('John Doe');
78
- });
79
-
80
- it('should display default patient name when not provided', async () => {
81
- const appointmentWithoutPatient = createMockAppointment({ patientName: undefined });
82
- await wrapper.setProps({ appointment: appointmentWithoutPatient });
83
-
84
- expect(wrapper.text()).toContain('Unbekannter Patient');
85
- });
86
-
87
- it('should display formatted date', () => {
88
- const expectedDate = format(new Date('2024-01-15T10:30:00Z'), 'EEE, d MMM yyyy', { locale: de });
89
- expect(wrapper.text()).toContain(expectedDate);
90
- });
91
-
92
- it('should display formatted time', () => {
93
- const expectedTime = format(new Date('2024-01-15T10:30:00Z'), 'HH:mm');
94
- expect(wrapper.text()).toContain(expectedTime);
95
- });
96
-
97
- it('should render header icons', () => {
98
- const icons = wrapper.findAll('.icon-stub');
99
- expect(icons.length).toBeGreaterThanOrEqual(3);
100
-
101
- const iconNames = icons.map(icon => icon.attributes('name'));
102
- expect(iconNames).toContain('heroicons:user-circle');
103
- expect(iconNames).toContain('heroicons:map-pin');
104
- expect(iconNames).toContain('heroicons:clock');
105
- });
106
-
107
- it('should have correct header items structure', () => {
108
- const headerItems = wrapper.findAll('.header-item');
109
- expect(headerItems).toHaveLength(3);
110
- });
111
- });
112
-
113
- describe('Appointment Status Styling', () => {
114
- it('should apply normal header background for upcoming appointments', () => {
115
- const header = wrapper.find('.card-header');
116
- expect(header.classes()).toContain('header-bg-normal');
117
- expect(header.classes()).not.toContain('header-bg-cancelled');
118
- });
119
-
120
- it('should apply cancelled header background for cancelled appointments', async () => {
121
- const cancelledAppointment = createMockAppointment({ status: 'cancelled' });
122
- await wrapper.setProps({ appointment: cancelledAppointment });
123
-
124
- const header = wrapper.find('.card-header');
125
- expect(header.classes()).toContain('header-bg-cancelled');
126
- expect(header.classes()).not.toContain('header-bg-normal');
127
- });
128
-
129
- it('should apply card opacity for past appointments', async () => {
130
- const pastAppointment = createMockAppointment({ status: 'past' });
131
- await wrapper.setProps({ appointment: pastAppointment });
132
-
133
- expect(wrapper.find('.card-opacity').exists()).toBe(true);
134
- });
135
-
136
- it('should not apply card opacity for non-past appointments', () => {
137
- expect(wrapper.find('.card-opacity').exists()).toBe(false);
138
- });
139
- });
140
-
141
- describe('Template Name Display', () => {
142
- it('should display single template name', () => {
143
- expect(wrapper.text()).toContain('Test Treatment');
144
- });
145
-
146
- it('should display multiple template names as comma-separated string', async () => {
147
- const appointmentWithMultipleTemplates = createMockAppointment({
148
- template_name: ['Treatment A', 'Treatment B', 'Treatment C']
149
- });
150
- await wrapper.setProps({ appointment: appointmentWithMultipleTemplates });
151
-
152
- expect(wrapper.text()).toContain('Treatment A, Treatment B, Treatment C');
153
- });
154
-
155
- it('should handle empty template name array', async () => {
156
- const appointmentWithEmptyTemplates = createMockAppointment({
157
- template_name: []
158
- });
159
- await wrapper.setProps({ appointment: appointmentWithEmptyTemplates });
160
-
161
- expect(wrapper.find('h2').text()).toBe('');
162
- });
163
- });
164
-
165
- describe('Link Functionality', () => {
166
- it('should render appointment link when provided', async () => {
167
- await wrapper.setProps({ appointmentLink: 'https://example.com/appointment/123' });
168
-
169
- const link = wrapper.find('.appointment-card');
170
- expect(link.attributes('href')).toBe('https://example.com/appointment/123');
171
- expect(link.attributes('rel')).toBe('noopener noreferrer');
172
- });
173
-
174
- it('should render default hash link when no appointment link provided', () => {
175
- const link = wrapper.find('.appointment-card');
176
- expect(link.attributes('href')).toBe('#');
177
- });
178
- });
179
-
180
- describe('Event Handling', () => {
181
- it('should emit confirm event when Actions emits confirmed', async () => {
182
- const actionsComponent = wrapper.find('.actions-stub');
183
- await actionsComponent.trigger('confirmed', '123');
184
-
185
- expect(wrapper.emitted('confirm')).toBeTruthy();
186
- expect(wrapper.emitted('confirm')?.[0]).toEqual(['123']);
187
- });
188
-
189
- it('should emit cancel event when Actions emits cancelled', async () => {
190
- const actionsComponent = wrapper.find('.actions-stub');
191
- await actionsComponent.trigger('cancelled', '123');
192
-
193
- expect(wrapper.emitted('cancel')).toBeTruthy();
194
- expect(wrapper.emitted('cancel')?.[0]).toEqual(['123']);
195
- });
196
-
197
- it('should emit reschedule event when Actions emits rescheduled', async () => {
198
- const actionsComponent = wrapper.find('.actions-stub');
199
- await actionsComponent.trigger('rescheduled', '123');
200
-
201
- expect(wrapper.emitted('reschedule')).toBeTruthy();
202
- expect(wrapper.emitted('reschedule')?.[0]).toEqual(['123']);
203
- });
204
- });
205
-
206
- describe('Props Passing', () => {
207
- it('should pass appointment prop to Details component', () => {
208
- const detailsStub = wrapper.find('.details-stub');
209
- expect(detailsStub.exists()).toBe(true);
210
- });
211
-
212
- it('should pass appointment prop to Actions component', () => {
213
- const actionsStub = wrapper.find('.actions-stub');
214
- expect(actionsStub.exists()).toBe(true);
215
- });
216
-
217
- it('should pass dentistImageSrc prop to Details component', async () => {
218
- await wrapper.setProps({ dentistImageSrc: 'https://example.com/custom-dentist.jpg' });
219
- expect(wrapper.props('dentistImageSrc')).toBe('https://example.com/custom-dentist.jpg');
220
- });
221
- });
222
-
223
- describe('Date Formatting Edge Cases', () => {
224
- it('should handle current date when appointment date is null', async () => {
225
- const appointmentWithNullDate = createMockAppointment({ start: null as any });
226
- await wrapper.setProps({ appointment: appointmentWithNullDate });
227
-
228
- // Should not throw an error and should display current date
229
- expect(wrapper.find('.card-header').exists()).toBe(true);
230
- });
231
-
232
- it('should format different date correctly', async () => {
233
- const appointmentWithDifferentDate = createMockAppointment({
234
- start: '2024-12-25T15:45:00Z'
235
- });
236
- await wrapper.setProps({ appointment: appointmentWithDifferentDate });
237
-
238
- const expectedDate = format(new Date('2024-12-25T15:45:00Z'), 'EEE, d MMM yyyy', { locale: de });
239
- const expectedTime = format(new Date('2024-12-25T15:45:00Z'), 'HH:mm');
240
-
241
- expect(wrapper.text()).toContain(expectedDate);
242
- expect(wrapper.text()).toContain(expectedTime);
243
- });
244
- });
245
-
246
- describe('Layout Structure', () => {
247
- it('should have proper layout structure', () => {
248
- const root = wrapper.find('[data-testid="root"]');
249
- expect(root.exists()).toBe(true);
250
-
251
- const appointmentCard = root.find('.appointment-card');
252
- expect(appointmentCard.exists()).toBe(true);
253
-
254
- const cardFooter = root.find('.card-footer');
255
- expect(cardFooter.exists()).toBe(true);
256
- });
257
-
258
- it('should contain horizontal rule separator', () => {
259
- expect(wrapper.find('hr').exists()).toBe(true);
260
- });
261
- });
262
-
263
- describe('Accessibility', () => {
264
- it('should have proper link attributes for accessibility', async () => {
265
- await wrapper.setProps({ appointmentLink: 'https://example.com/appointment/123' });
266
-
267
- const link = wrapper.find('.appointment-card');
268
- expect(link.attributes('rel')).toBe('noopener noreferrer');
269
- });
270
-
271
- it('should have descriptive data-testid', () => {
272
- expect(wrapper.find('[data-testid="root"]').exists()).toBe(true);
273
- });
274
- });
275
-
276
- describe('Component Props Validation', () => {
277
- it('should accept all required props', () => {
278
- expect(wrapper.props('appointment')).toBeDefined();
279
- });
280
-
281
- it('should accept optional props', async () => {
282
- await wrapper.setProps({
283
- appointmentLink: 'https://example.com/test',
284
- dentistImageSrc: 'https://example.com/dentist.jpg'
285
- });
286
-
287
- expect(wrapper.props('appointmentLink')).toBe('https://example.com/test');
288
- expect(wrapper.props('dentistImageSrc')).toBe('https://example.com/dentist.jpg');
289
- });
290
- });
291
-
292
- describe('Accessibility', () => {
293
- beforeEach(() => {
294
- const appointment = createMockAppointment();
295
- wrapper = mount(Card, {
296
- props: { appointment },
297
- global: {
298
- stubs: {
299
- Icon: {
300
- template: '<div class="icon-stub" :aria-hidden="true" :name="name" :size="size" role="img" :aria-label="name"></div>',
301
- props: ['name', 'size']
302
- },
303
- Details: {
304
- template: '<div class="details-stub" role="region" aria-label="Appointment details">Details Component</div>',
305
- props: ['appointment', 'dentistImageSrc']
306
- },
307
- Actions: {
308
- template: '<div class="actions-stub" role="group" aria-label="Appointment actions">Actions Component</div>',
309
- props: ['appointment']
310
- }
311
- }
312
- }
313
- });
314
- });
315
-
316
- it('should have proper semantic HTML structure', () => {
317
- const root = wrapper.find('[data-testid="root"]');
318
- expect(root.exists()).toBe(true);
319
-
320
- const appointmentCard = wrapper.find('.appointment-card');
321
- expect(appointmentCard.exists()).toBe(true);
322
-
323
- // Should have proper sectioning
324
- expect(wrapper.find('.card-header').exists()).toBe(true);
325
- expect(wrapper.find('.card-body').exists()).toBe(true);
326
- expect(wrapper.find('.card-footer').exists()).toBe(true);
327
- });
328
-
329
- it('should use proper heading hierarchy', () => {
330
- const heading = wrapper.find('h2');
331
- expect(heading.exists()).toBe(true);
332
- expect(heading.text()).toBe('Test Treatment');
333
- });
334
-
335
- it('should have accessible link attributes', async () => {
336
- await wrapper.setProps({ appointmentLink: 'https://example.com/appointment/123' });
337
-
338
- const link = wrapper.find('.appointment-card');
339
- expect(link.attributes('href')).toBe('https://example.com/appointment/123');
340
- expect(link.attributes('rel')).toBe('noopener noreferrer');
341
- });
342
-
343
- it('should provide meaningful content for screen readers', () => {
344
- // Patient name should be visible
345
- expect(wrapper.text()).toContain('John Doe');
346
-
347
- // Date and time should be accessible
348
- const appointmentDate = new Date('2024-01-15T10:30:00.000Z');
349
- const expectedDate = format(appointmentDate, 'EEE, d MMM yyyy', { locale: de });
350
-
351
- expect(wrapper.text()).toContain(expectedDate);
352
- // Time will be displayed based on local timezone, check for any valid time format
353
- expect(wrapper.text()).toMatch(/\d{1,2}:\d{2}/);
354
-
355
- // Treatment name should be in heading
356
- const h2 = wrapper.find('h2');
357
- expect(h2.text()).toBe('Test Treatment');
358
- });
359
-
360
- it('should have proper icon accessibility attributes', () => {
361
- const icons = wrapper.findAll('.icon-stub');
362
- expect(icons.length).toBeGreaterThanOrEqual(3);
363
-
364
- icons.forEach(icon => {
365
- expect(icon.attributes('aria-hidden')).toBe('true');
366
- expect(icon.attributes('role')).toBe('img');
367
- expect(icon.attributes('aria-label')).toBeTruthy();
368
- });
369
- });
370
-
371
- it('should provide semantic roles for sub-components', () => {
372
- const detailsStub = wrapper.find('.details-stub');
373
- expect(detailsStub.attributes('role')).toBe('region');
374
- expect(detailsStub.attributes('aria-label')).toBe('Appointment details');
375
-
376
- const actionsStub = wrapper.find('.actions-stub');
377
- expect(actionsStub.attributes('role')).toBe('group');
378
- expect(actionsStub.attributes('aria-label')).toBe('Appointment actions');
379
- });
380
-
381
- it('should handle different appointment statuses accessibly', async () => {
382
- // Test normal status
383
- let header = wrapper.find('.card-header');
384
- expect(header.classes()).toContain('header-bg-normal');
385
-
386
- // Test cancelled status
387
- const cancelledAppointment = createMockAppointment({ status: 'cancelled' });
388
- await wrapper.setProps({ appointment: cancelledAppointment });
389
-
390
- header = wrapper.find('.card-header');
391
- expect(header.classes()).toContain('header-bg-cancelled');
392
-
393
- // Test past status with opacity
394
- const pastAppointment = createMockAppointment({ status: 'past' });
395
- await wrapper.setProps({ appointment: pastAppointment });
396
-
397
- expect(wrapper.find('.card-opacity').exists()).toBe(true);
398
- });
399
-
400
- it('should provide informative default values', async () => {
401
- const appointmentWithoutPatient = createMockAppointment({ patientName: undefined });
402
- await wrapper.setProps({ appointment: appointmentWithoutPatient });
403
-
404
- expect(wrapper.text()).toContain('Unbekannter Patient');
405
- });
406
-
407
- it('should have logical reading order', () => {
408
- // Header items should be in logical order
409
- const headerItems = wrapper.findAll('.header-item');
410
- expect(headerItems).toHaveLength(3);
411
-
412
- // Patient, Date, Time order
413
- expect(headerItems[0].text()).toContain('John Doe');
414
- // Time will be displayed based on local timezone, check for any valid time format
415
- expect(headerItems[2].text()).toMatch(/\d{1,2}:\d{2}/);
416
- });
417
-
418
- it('should handle multiple template names accessibly', async () => {
419
- const appointmentWithMultipleTemplates = createMockAppointment({
420
- template_name: ['Treatment A', 'Treatment B', 'Treatment C']
421
- });
422
- await wrapper.setProps({ appointment: appointmentWithMultipleTemplates });
423
-
424
- const heading = wrapper.find('h2');
425
- expect(heading.text()).toBe('Treatment A, Treatment B, Treatment C');
426
- });
427
-
428
- it('should use proper HTML5 semantic elements', () => {
429
- // Should have proper sectioning
430
- const header = wrapper.find('.card-header');
431
- const body = wrapper.find('.card-body');
432
- const footer = wrapper.find('.card-footer');
433
-
434
- expect(header.exists()).toBe(true);
435
- expect(body.exists()).toBe(true);
436
- expect(footer.exists()).toBe(true);
437
-
438
- // Should have horizontal rule for separation
439
- expect(wrapper.find('hr').exists()).toBe(true);
440
- });
441
-
442
- it('should maintain focus management for interactive elements', async () => {
443
- await wrapper.setProps({ appointmentLink: 'https://example.com/appointment/123' });
444
-
445
- const link = wrapper.find('.appointment-card');
446
- expect(link.attributes('href')).toBeTruthy();
447
-
448
- // Link should be focusable
449
- expect(link.element.tagName.toLowerCase()).toBe('a');
450
- });
451
-
452
- it('should provide context for visual styling changes', () => {
453
- // Visual changes should have semantic meaning
454
- const header = wrapper.find('.card-header');
455
- expect(header.classes()).toContain('header-bg-normal');
456
-
457
- // Check that there are no accessibility violations in basic structure
458
- expect(wrapper.find('[data-testid="root"]').exists()).toBe(true);
459
- });
460
-
461
- it('should handle date formatting for screen readers', async () => {
462
- const appointmentWithDifferentDate = createMockAppointment({
463
- start: '2024-12-25T15:45:00Z'
464
- });
465
- await wrapper.setProps({ appointment: appointmentWithDifferentDate });
466
-
467
- const expectedDate = format(new Date('2024-12-25T15:45:00Z'), 'EEE, d MMM yyyy', { locale: de });
468
- const expectedTime = format(new Date('2024-12-25T15:45:00Z'), 'HH:mm');
469
-
470
- expect(wrapper.text()).toContain(expectedDate);
471
- expect(wrapper.text()).toContain(expectedTime);
472
- });
473
-
474
- it('should provide comprehensive information structure', () => {
475
- // Card should contain all necessary information
476
- expect(wrapper.text()).toContain('John Doe'); // Patient
477
- expect(wrapper.text()).toContain('Test Treatment'); // Treatment
478
- // Time will be displayed based on local timezone, check for any valid time format
479
- expect(wrapper.text()).toMatch(/\d{1,2}:\d{2}/); // Time
480
-
481
- // Should have proper component structure
482
- expect(wrapper.find('.details-stub').exists()).toBe(true);
483
- expect(wrapper.find('.actions-stub').exists()).toBe(true);
484
- });
485
- });
1
+ import { describe, it, expect, beforeEach, vi } from 'vitest';
2
+ import { mount, VueWrapper } from '@vue/test-utils';
3
+ import { format } from 'date-fns';
4
+ import { de } from 'date-fns/locale';
5
+ import Card from '../../../../../src/components/Appointment/Card/Card.vue';
6
+ import type { AppointmentData, Dentist } from '../../../../../src/types';
7
+
8
+ describe('Appointment Card', () => {
9
+ let wrapper: VueWrapper;
10
+
11
+ const createMockDentist = (overrides: Partial<Dentist> = {}): Dentist => ({
12
+ name: 'Dr. Test Dentist',
13
+ gender: 'Male',
14
+ imageSrc: 'https://example.com/dentist.jpg',
15
+ ...overrides
16
+ });
17
+
18
+ const createMockAppointment = (overrides: Partial<AppointmentData> = {}): AppointmentData => ({
19
+ id: '123',
20
+ template_name: 'Test Treatment',
21
+ description: 'Test Description',
22
+ dentist: createMockDentist(),
23
+ start: '2024-01-15T10:30:00.000Z',
24
+ type: 1,
25
+ status: 'upcoming',
26
+ patientName: 'John Doe',
27
+ address: 'Test Address 123',
28
+ district: 'Test District',
29
+ is_confirmed: false,
30
+ ...overrides
31
+ });
32
+
33
+ beforeEach(() => {
34
+ const appointment = createMockAppointment();
35
+ wrapper = mount(Card, {
36
+ props: { appointment },
37
+ global: {
38
+ stubs: {
39
+ Icon: {
40
+ template: '<div class="icon-stub" :name="name" :size="size"></div>',
41
+ props: ['name', 'size']
42
+ },
43
+ Details: {
44
+ template: '<div class="details-stub">Details Component</div>',
45
+ props: ['appointment', 'dentistImageSrc']
46
+ },
47
+ Actions: {
48
+ template: '<div class="actions-stub" @confirmed="$emit(\'confirmed\', $event)" @cancelled="$emit(\'cancelled\', $event)" @rescheduled="$emit(\'rescheduled\', $event)">Actions Component</div>',
49
+ props: ['appointment'],
50
+ emits: ['confirmed', 'cancelled', 'rescheduled']
51
+ }
52
+ }
53
+ }
54
+ });
55
+ });
56
+
57
+ describe('Component Rendering', () => {
58
+ it('should render the component with correct test id', () => {
59
+ expect(wrapper.find('[data-testid="root"]').exists()).toBe(true);
60
+ });
61
+
62
+ it('should render the appointment card structure', () => {
63
+ expect(wrapper.find('.appointment-card').exists()).toBe(true);
64
+ expect(wrapper.find('.card-header').exists()).toBe(true);
65
+ expect(wrapper.find('.card-body').exists()).toBe(true);
66
+ expect(wrapper.find('.card-footer').exists()).toBe(true);
67
+ });
68
+
69
+ it('should render Details and Actions components', () => {
70
+ expect(wrapper.find('.details-stub').exists()).toBe(true);
71
+ expect(wrapper.find('.actions-stub').exists()).toBe(true);
72
+ });
73
+ });
74
+
75
+ describe('Header Information', () => {
76
+ it('should display patient name', () => {
77
+ expect(wrapper.text()).toContain('John Doe');
78
+ });
79
+
80
+ it('should display default patient name when not provided', async () => {
81
+ const appointmentWithoutPatient = createMockAppointment({ patientName: undefined });
82
+ await wrapper.setProps({ appointment: appointmentWithoutPatient });
83
+
84
+ expect(wrapper.text()).toContain('Unbekannter Patient');
85
+ });
86
+
87
+ it('should display formatted date', () => {
88
+ const expectedDate = format(new Date('2024-01-15T10:30:00Z'), 'EEE, d MMM yyyy', { locale: de });
89
+ expect(wrapper.text()).toContain(expectedDate);
90
+ });
91
+
92
+ it('should display formatted time', () => {
93
+ const expectedTime = format(new Date('2024-01-15T10:30:00Z'), 'HH:mm');
94
+ expect(wrapper.text()).toContain(expectedTime);
95
+ });
96
+
97
+ it('should render header icons', () => {
98
+ const icons = wrapper.findAll('.icon-stub');
99
+ expect(icons.length).toBeGreaterThanOrEqual(3);
100
+
101
+ const iconNames = icons.map(icon => icon.attributes('name'));
102
+ expect(iconNames).toContain('heroicons:user-circle');
103
+ expect(iconNames).toContain('heroicons:map-pin');
104
+ expect(iconNames).toContain('heroicons:clock');
105
+ });
106
+
107
+ it('should have correct header items structure', () => {
108
+ const headerItems = wrapper.findAll('.header-item');
109
+ expect(headerItems).toHaveLength(3);
110
+ });
111
+ });
112
+
113
+ describe('Appointment Status Styling', () => {
114
+ it('should apply normal header background for upcoming appointments', () => {
115
+ const header = wrapper.find('.card-header');
116
+ expect(header.classes()).toContain('header-bg-normal');
117
+ expect(header.classes()).not.toContain('header-bg-cancelled');
118
+ });
119
+
120
+ it('should apply cancelled header background for cancelled appointments', async () => {
121
+ const cancelledAppointment = createMockAppointment({ status: 'cancelled' });
122
+ await wrapper.setProps({ appointment: cancelledAppointment });
123
+
124
+ const header = wrapper.find('.card-header');
125
+ expect(header.classes()).toContain('header-bg-cancelled');
126
+ expect(header.classes()).not.toContain('header-bg-normal');
127
+ });
128
+
129
+ it('should apply card opacity for past appointments', async () => {
130
+ const pastAppointment = createMockAppointment({ status: 'past' });
131
+ await wrapper.setProps({ appointment: pastAppointment });
132
+
133
+ expect(wrapper.find('.card-opacity').exists()).toBe(true);
134
+ });
135
+
136
+ it('should not apply card opacity for non-past appointments', () => {
137
+ expect(wrapper.find('.card-opacity').exists()).toBe(false);
138
+ });
139
+ });
140
+
141
+ describe('Template Name Display', () => {
142
+ it('should display single template name', () => {
143
+ expect(wrapper.text()).toContain('Test Treatment');
144
+ });
145
+
146
+ it('should display multiple template names as comma-separated string', async () => {
147
+ const appointmentWithMultipleTemplates = createMockAppointment({
148
+ template_name: ['Treatment A', 'Treatment B', 'Treatment C']
149
+ });
150
+ await wrapper.setProps({ appointment: appointmentWithMultipleTemplates });
151
+
152
+ expect(wrapper.text()).toContain('Treatment A, Treatment B, Treatment C');
153
+ });
154
+
155
+ it('should handle empty template name array', async () => {
156
+ const appointmentWithEmptyTemplates = createMockAppointment({
157
+ template_name: []
158
+ });
159
+ await wrapper.setProps({ appointment: appointmentWithEmptyTemplates });
160
+
161
+ expect(wrapper.find('h2').text()).toBe('');
162
+ });
163
+ });
164
+
165
+ describe('Link Functionality', () => {
166
+ it('should render appointment link when provided', async () => {
167
+ await wrapper.setProps({ appointmentLink: 'https://example.com/appointment/123' });
168
+
169
+ const link = wrapper.find('.appointment-card');
170
+ expect(link.attributes('href')).toBe('https://example.com/appointment/123');
171
+ expect(link.attributes('rel')).toBe('noopener noreferrer');
172
+ });
173
+
174
+ it('should render default hash link when no appointment link provided', () => {
175
+ const link = wrapper.find('.appointment-card');
176
+ expect(link.attributes('href')).toBe('#');
177
+ });
178
+ });
179
+
180
+ describe('Event Handling', () => {
181
+ it('should emit confirm event when Actions emits confirmed', async () => {
182
+ const actionsComponent = wrapper.find('.actions-stub');
183
+ await actionsComponent.trigger('confirmed', '123');
184
+
185
+ expect(wrapper.emitted('confirm')).toBeTruthy();
186
+ expect(wrapper.emitted('confirm')?.[0]).toEqual(['123']);
187
+ });
188
+
189
+ it('should emit cancel event when Actions emits cancelled', async () => {
190
+ const actionsComponent = wrapper.find('.actions-stub');
191
+ await actionsComponent.trigger('cancelled', '123');
192
+
193
+ expect(wrapper.emitted('cancel')).toBeTruthy();
194
+ expect(wrapper.emitted('cancel')?.[0]).toEqual(['123']);
195
+ });
196
+
197
+ it('should emit reschedule event when Actions emits rescheduled', async () => {
198
+ const actionsComponent = wrapper.find('.actions-stub');
199
+ await actionsComponent.trigger('rescheduled', '123');
200
+
201
+ expect(wrapper.emitted('reschedule')).toBeTruthy();
202
+ expect(wrapper.emitted('reschedule')?.[0]).toEqual(['123']);
203
+ });
204
+ });
205
+
206
+ describe('Props Passing', () => {
207
+ it('should pass appointment prop to Details component', () => {
208
+ const detailsStub = wrapper.find('.details-stub');
209
+ expect(detailsStub.exists()).toBe(true);
210
+ });
211
+
212
+ it('should pass appointment prop to Actions component', () => {
213
+ const actionsStub = wrapper.find('.actions-stub');
214
+ expect(actionsStub.exists()).toBe(true);
215
+ });
216
+
217
+ it('should pass dentistImageSrc prop to Details component', async () => {
218
+ await wrapper.setProps({ dentistImageSrc: 'https://example.com/custom-dentist.jpg' });
219
+ expect(wrapper.props('dentistImageSrc')).toBe('https://example.com/custom-dentist.jpg');
220
+ });
221
+ });
222
+
223
+ describe('Date Formatting Edge Cases', () => {
224
+ it('should handle current date when appointment date is null', async () => {
225
+ const appointmentWithNullDate = createMockAppointment({ start: null as any });
226
+ await wrapper.setProps({ appointment: appointmentWithNullDate });
227
+
228
+ // Should not throw an error and should display current date
229
+ expect(wrapper.find('.card-header').exists()).toBe(true);
230
+ });
231
+
232
+ it('should format different date correctly', async () => {
233
+ const appointmentWithDifferentDate = createMockAppointment({
234
+ start: '2024-12-25T15:45:00Z'
235
+ });
236
+ await wrapper.setProps({ appointment: appointmentWithDifferentDate });
237
+
238
+ const expectedDate = format(new Date('2024-12-25T15:45:00Z'), 'EEE, d MMM yyyy', { locale: de });
239
+ const expectedTime = format(new Date('2024-12-25T15:45:00Z'), 'HH:mm');
240
+
241
+ expect(wrapper.text()).toContain(expectedDate);
242
+ expect(wrapper.text()).toContain(expectedTime);
243
+ });
244
+ });
245
+
246
+ describe('Layout Structure', () => {
247
+ it('should have proper layout structure', () => {
248
+ const root = wrapper.find('[data-testid="root"]');
249
+ expect(root.exists()).toBe(true);
250
+
251
+ const appointmentCard = root.find('.appointment-card');
252
+ expect(appointmentCard.exists()).toBe(true);
253
+
254
+ const cardFooter = root.find('.card-footer');
255
+ expect(cardFooter.exists()).toBe(true);
256
+ });
257
+
258
+ it('should contain horizontal rule separator', () => {
259
+ expect(wrapper.find('hr').exists()).toBe(true);
260
+ });
261
+ });
262
+
263
+ describe('Accessibility', () => {
264
+ it('should have proper link attributes for accessibility', async () => {
265
+ await wrapper.setProps({ appointmentLink: 'https://example.com/appointment/123' });
266
+
267
+ const link = wrapper.find('.appointment-card');
268
+ expect(link.attributes('rel')).toBe('noopener noreferrer');
269
+ });
270
+
271
+ it('should have descriptive data-testid', () => {
272
+ expect(wrapper.find('[data-testid="root"]').exists()).toBe(true);
273
+ });
274
+ });
275
+
276
+ describe('Component Props Validation', () => {
277
+ it('should accept all required props', () => {
278
+ expect(wrapper.props('appointment')).toBeDefined();
279
+ });
280
+
281
+ it('should accept optional props', async () => {
282
+ await wrapper.setProps({
283
+ appointmentLink: 'https://example.com/test',
284
+ dentistImageSrc: 'https://example.com/dentist.jpg'
285
+ });
286
+
287
+ expect(wrapper.props('appointmentLink')).toBe('https://example.com/test');
288
+ expect(wrapper.props('dentistImageSrc')).toBe('https://example.com/dentist.jpg');
289
+ });
290
+ });
291
+
292
+ describe('Accessibility', () => {
293
+ beforeEach(() => {
294
+ const appointment = createMockAppointment();
295
+ wrapper = mount(Card, {
296
+ props: { appointment },
297
+ global: {
298
+ stubs: {
299
+ Icon: {
300
+ template: '<div class="icon-stub" :aria-hidden="true" :name="name" :size="size" role="img" :aria-label="name"></div>',
301
+ props: ['name', 'size']
302
+ },
303
+ Details: {
304
+ template: '<div class="details-stub" role="region" aria-label="Appointment details">Details Component</div>',
305
+ props: ['appointment', 'dentistImageSrc']
306
+ },
307
+ Actions: {
308
+ template: '<div class="actions-stub" role="group" aria-label="Appointment actions">Actions Component</div>',
309
+ props: ['appointment']
310
+ }
311
+ }
312
+ }
313
+ });
314
+ });
315
+
316
+ it('should have proper semantic HTML structure', () => {
317
+ const root = wrapper.find('[data-testid="root"]');
318
+ expect(root.exists()).toBe(true);
319
+
320
+ const appointmentCard = wrapper.find('.appointment-card');
321
+ expect(appointmentCard.exists()).toBe(true);
322
+
323
+ // Should have proper sectioning
324
+ expect(wrapper.find('.card-header').exists()).toBe(true);
325
+ expect(wrapper.find('.card-body').exists()).toBe(true);
326
+ expect(wrapper.find('.card-footer').exists()).toBe(true);
327
+ });
328
+
329
+ it('should use proper heading hierarchy', () => {
330
+ const heading = wrapper.find('h2');
331
+ expect(heading.exists()).toBe(true);
332
+ expect(heading.text()).toBe('Test Treatment');
333
+ });
334
+
335
+ it('should have accessible link attributes', async () => {
336
+ await wrapper.setProps({ appointmentLink: 'https://example.com/appointment/123' });
337
+
338
+ const link = wrapper.find('.appointment-card');
339
+ expect(link.attributes('href')).toBe('https://example.com/appointment/123');
340
+ expect(link.attributes('rel')).toBe('noopener noreferrer');
341
+ });
342
+
343
+ it('should provide meaningful content for screen readers', () => {
344
+ // Patient name should be visible
345
+ expect(wrapper.text()).toContain('John Doe');
346
+
347
+ // Date and time should be accessible
348
+ const appointmentDate = new Date('2024-01-15T10:30:00.000Z');
349
+ const expectedDate = format(appointmentDate, 'EEE, d MMM yyyy', { locale: de });
350
+
351
+ expect(wrapper.text()).toContain(expectedDate);
352
+ // Time will be displayed based on local timezone, check for any valid time format
353
+ expect(wrapper.text()).toMatch(/\d{1,2}:\d{2}/);
354
+
355
+ // Treatment name should be in heading
356
+ const h2 = wrapper.find('h2');
357
+ expect(h2.text()).toBe('Test Treatment');
358
+ });
359
+
360
+ it('should have proper icon accessibility attributes', () => {
361
+ const icons = wrapper.findAll('.icon-stub');
362
+ expect(icons.length).toBeGreaterThanOrEqual(3);
363
+
364
+ icons.forEach(icon => {
365
+ expect(icon.attributes('aria-hidden')).toBe('true');
366
+ expect(icon.attributes('role')).toBe('img');
367
+ expect(icon.attributes('aria-label')).toBeTruthy();
368
+ });
369
+ });
370
+
371
+ it('should provide semantic roles for sub-components', () => {
372
+ const detailsStub = wrapper.find('.details-stub');
373
+ expect(detailsStub.attributes('role')).toBe('region');
374
+ expect(detailsStub.attributes('aria-label')).toBe('Appointment details');
375
+
376
+ const actionsStub = wrapper.find('.actions-stub');
377
+ expect(actionsStub.attributes('role')).toBe('group');
378
+ expect(actionsStub.attributes('aria-label')).toBe('Appointment actions');
379
+ });
380
+
381
+ it('should handle different appointment statuses accessibly', async () => {
382
+ // Test normal status
383
+ let header = wrapper.find('.card-header');
384
+ expect(header.classes()).toContain('header-bg-normal');
385
+
386
+ // Test cancelled status
387
+ const cancelledAppointment = createMockAppointment({ status: 'cancelled' });
388
+ await wrapper.setProps({ appointment: cancelledAppointment });
389
+
390
+ header = wrapper.find('.card-header');
391
+ expect(header.classes()).toContain('header-bg-cancelled');
392
+
393
+ // Test past status with opacity
394
+ const pastAppointment = createMockAppointment({ status: 'past' });
395
+ await wrapper.setProps({ appointment: pastAppointment });
396
+
397
+ expect(wrapper.find('.card-opacity').exists()).toBe(true);
398
+ });
399
+
400
+ it('should provide informative default values', async () => {
401
+ const appointmentWithoutPatient = createMockAppointment({ patientName: undefined });
402
+ await wrapper.setProps({ appointment: appointmentWithoutPatient });
403
+
404
+ expect(wrapper.text()).toContain('Unbekannter Patient');
405
+ });
406
+
407
+ it('should have logical reading order', () => {
408
+ // Header items should be in logical order
409
+ const headerItems = wrapper.findAll('.header-item');
410
+ expect(headerItems).toHaveLength(3);
411
+
412
+ // Patient, Date, Time order
413
+ expect(headerItems[0].text()).toContain('John Doe');
414
+ // Time will be displayed based on local timezone, check for any valid time format
415
+ expect(headerItems[2].text()).toMatch(/\d{1,2}:\d{2}/);
416
+ });
417
+
418
+ it('should handle multiple template names accessibly', async () => {
419
+ const appointmentWithMultipleTemplates = createMockAppointment({
420
+ template_name: ['Treatment A', 'Treatment B', 'Treatment C']
421
+ });
422
+ await wrapper.setProps({ appointment: appointmentWithMultipleTemplates });
423
+
424
+ const heading = wrapper.find('h2');
425
+ expect(heading.text()).toBe('Treatment A, Treatment B, Treatment C');
426
+ });
427
+
428
+ it('should use proper HTML5 semantic elements', () => {
429
+ // Should have proper sectioning
430
+ const header = wrapper.find('.card-header');
431
+ const body = wrapper.find('.card-body');
432
+ const footer = wrapper.find('.card-footer');
433
+
434
+ expect(header.exists()).toBe(true);
435
+ expect(body.exists()).toBe(true);
436
+ expect(footer.exists()).toBe(true);
437
+
438
+ // Should have horizontal rule for separation
439
+ expect(wrapper.find('hr').exists()).toBe(true);
440
+ });
441
+
442
+ it('should maintain focus management for interactive elements', async () => {
443
+ await wrapper.setProps({ appointmentLink: 'https://example.com/appointment/123' });
444
+
445
+ const link = wrapper.find('.appointment-card');
446
+ expect(link.attributes('href')).toBeTruthy();
447
+
448
+ // Link should be focusable
449
+ expect(link.element.tagName.toLowerCase()).toBe('a');
450
+ });
451
+
452
+ it('should provide context for visual styling changes', () => {
453
+ // Visual changes should have semantic meaning
454
+ const header = wrapper.find('.card-header');
455
+ expect(header.classes()).toContain('header-bg-normal');
456
+
457
+ // Check that there are no accessibility violations in basic structure
458
+ expect(wrapper.find('[data-testid="root"]').exists()).toBe(true);
459
+ });
460
+
461
+ it('should handle date formatting for screen readers', async () => {
462
+ const appointmentWithDifferentDate = createMockAppointment({
463
+ start: '2024-12-25T15:45:00Z'
464
+ });
465
+ await wrapper.setProps({ appointment: appointmentWithDifferentDate });
466
+
467
+ const expectedDate = format(new Date('2024-12-25T15:45:00Z'), 'EEE, d MMM yyyy', { locale: de });
468
+ const expectedTime = format(new Date('2024-12-25T15:45:00Z'), 'HH:mm');
469
+
470
+ expect(wrapper.text()).toContain(expectedDate);
471
+ expect(wrapper.text()).toContain(expectedTime);
472
+ });
473
+
474
+ it('should provide comprehensive information structure', () => {
475
+ // Card should contain all necessary information
476
+ expect(wrapper.text()).toContain('John Doe'); // Patient
477
+ expect(wrapper.text()).toContain('Test Treatment'); // Treatment
478
+ // Time will be displayed based on local timezone, check for any valid time format
479
+ expect(wrapper.text()).toMatch(/\d{1,2}:\d{2}/); // Time
480
+
481
+ // Should have proper component structure
482
+ expect(wrapper.find('.details-stub').exists()).toBe(true);
483
+ expect(wrapper.find('.actions-stub').exists()).toBe(true);
484
+ });
485
+ });
486
486
  });