@zap-wunschlachen/wl-shared-components 1.0.0
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/.github/workflows/playwright.yml +215 -0
- package/.github/workflows/static.yml +62 -0
- package/.prettierrc +5 -0
- package/.storybook/main.ts +18 -0
- package/.storybook/preview.ts +37 -0
- package/.storybook/storyWrapper.vue +18 -0
- package/.storybook/withVuetifyTheme.decorator.ts +21 -0
- package/App.vue +95 -0
- package/README.md +56 -0
- package/heroicons.ts +75 -0
- package/index.html +19 -0
- package/package.json +66 -0
- package/playwright.config.ts +35 -0
- package/public/audio/dummy_pink_noise.wav +0 -0
- package/public/background.svg +60 -0
- package/public/javascript.svg +1 -0
- package/public/style.css +187 -0
- package/public/technologies.svg +22 -0
- package/src/assets/css/base.css +235 -0
- package/src/assets/css/variables.css +96 -0
- package/src/assets/fonts/Outfit-Black.ttf +0 -0
- package/src/assets/fonts/Outfit-Bold.ttf +0 -0
- package/src/assets/fonts/Outfit-ExtraBold.ttf +0 -0
- package/src/assets/fonts/Outfit-ExtraLight.ttf +0 -0
- package/src/assets/fonts/Outfit-Light.ttf +0 -0
- package/src/assets/fonts/Outfit-Medium.ttf +0 -0
- package/src/assets/fonts/Outfit-Regular.ttf +0 -0
- package/src/assets/fonts/Outfit-SemiBold.ttf +0 -0
- package/src/assets/fonts/Outfit-Thin.ttf +0 -0
- package/src/components/Accordion/Accordion.css +59 -0
- package/src/components/Accordion/AccordionGroup.vue +51 -0
- package/src/components/Accordion/AccordionItem.vue +66 -0
- package/src/components/Appointment/Card/Actions.css +30 -0
- package/src/components/Appointment/Card/Actions.vue +66 -0
- package/src/components/Appointment/Card/Card.css +49 -0
- package/src/components/Appointment/Card/Card.vue +55 -0
- package/src/components/Appointment/Card/Details.css +51 -0
- package/src/components/Appointment/Card/Details.vue +44 -0
- package/src/components/Audio/Audio.vue +188 -0
- package/src/components/Audio/Waveform.vue +118 -0
- package/src/components/Button/Button.vue +119 -0
- package/src/components/CheckBox/CheckBox.css +185 -0
- package/src/components/CheckBox/Checkbox.vue +130 -0
- package/src/components/DateInput/DateInput.css +3 -0
- package/src/components/DateInput/DateInput.vue +263 -0
- package/src/components/Dialog/Dialog.css +6 -0
- package/src/components/Dialog/Dialog.vue +29 -0
- package/src/components/EditField/EditField.css +20 -0
- package/src/components/EditField/EditField.vue +202 -0
- package/src/components/IconBullet/IconBullet.vue +86 -0
- package/src/components/IconBullet/IconBulletList.vue +41 -0
- package/src/components/Icons/Audio/CloudFailed.vue +21 -0
- package/src/components/Icons/Audio/CloudSaved.vue +22 -0
- package/src/components/Icons/Audio/Delete.vue +16 -0
- package/src/components/Icons/Audio/Pause.vue +19 -0
- package/src/components/Icons/Audio/Play.vue +16 -0
- package/src/components/Icons/CalendarNotification.vue +126 -0
- package/src/components/Icons/Chair.vue +32 -0
- package/src/components/Icons/ChairNotification.vue +35 -0
- package/src/components/Icons/Circle.vue +66 -0
- package/src/components/Icons/FavIcon.vue +22 -0
- package/src/components/Icons/FilledCircle.vue +11 -0
- package/src/components/Icons/Group3.vue +46 -0
- package/src/components/Icons/RingNotification.vue +54 -0
- package/src/components/Icons/SolidArrowRight.vue +14 -0
- package/src/components/Icons/calendar.vue +17 -0
- package/src/components/Icons/checkbox.vue +19 -0
- package/src/components/Icons/outlineChecked.vue +27 -0
- package/src/components/Icons/play.vue +6 -0
- package/src/components/Input/Input.css +187 -0
- package/src/components/Input/Input.vue +247 -0
- package/src/components/Laboratory/AppointmentCard/AppointmentCard.css +7 -0
- package/src/components/Laboratory/AppointmentCard/AppointmentCard.vue +116 -0
- package/src/components/Laboratory/ChatBoxImage/ChatBoxImage.vue +81 -0
- package/src/components/Laboratory/ChatMessage/ChatMessage.vue +113 -0
- package/src/components/Laboratory/ChatMessage/ChatMessageBadge.css +4 -0
- package/src/components/Laboratory/ChatMessage/ChatMessageBadge.vue +99 -0
- package/src/components/Laboratory/ChatNotification/ChatNotification.vue +130 -0
- package/src/components/Laboratory/DocumentCard/DocumentCard.css +3 -0
- package/src/components/Laboratory/DocumentCard/DocumentCard.vue +50 -0
- package/src/components/Laboratory/DocumentCard/DocumentCardItem.vue +53 -0
- package/src/components/Laboratory/InfoCard/InfoCard.vue +162 -0
- package/src/components/Laboratory/MainColumnsBar/MainColumnsBar.vue +102 -0
- package/src/components/Laboratory/ProgressCircle/ProgressCircle.vue +152 -0
- package/src/components/Laboratory/ProgressLinear/ProgressLinear.css +33 -0
- package/src/components/Laboratory/ProgressLinear/ProgressLinear.vue +75 -0
- package/src/components/Laboratory/SelectionColumnBar/SelectionColumnBar.vue +92 -0
- package/src/components/Laboratory/StatusNotification/StatusNotification.vue +49 -0
- package/src/components/Laboratory/TagLabel/TagLabel.vue +126 -0
- package/src/components/Laboratory/TagLabelGroup/TagLabelGroup.vue +97 -0
- package/src/components/Laboratory/TicketCard/TicketCard.css +3 -0
- package/src/components/Laboratory/TicketCard/TicketCard.vue +143 -0
- package/src/components/Laboratory/TimeLine/TimeLineEvent.css +18 -0
- package/src/components/Laboratory/TimeLine/TimeLineEvent.vue +119 -0
- package/src/components/Laboratory/TimeLine/Timeline.css +4 -0
- package/src/components/Laboratory/TimeLine/Timeline.vue +30 -0
- package/src/components/Modal/Modal.css +6 -0
- package/src/components/Modal/Modal.vue +23 -0
- package/src/components/NotificationBubble/NotificationBubble.css +4 -0
- package/src/components/NotificationBubble/NotificationBubble.vue +90 -0
- package/src/components/OtpInput/OtpInput.css +39 -0
- package/src/components/OtpInput/OtpInput.vue +144 -0
- package/src/components/PhoneInput/PhoneInput.css +32 -0
- package/src/components/PhoneInput/PhoneInput.vue +114 -0
- package/src/components/Select/Select.css +150 -0
- package/src/components/Select/Select.vue +304 -0
- package/src/components/TextArea/TextArea.css +3 -0
- package/src/components/TextArea/TextArea.vue +126 -0
- package/src/components/TickBox/TickBox.css +49 -0
- package/src/components/TickBox/TickBox.vue +126 -0
- package/src/components/index.ts +20 -0
- package/src/constants/buttonEnums.ts +0 -0
- package/src/constants/iconEnums.ts +4 -0
- package/src/i18n/i18n.ts +16 -0
- package/src/i18n/locales/de.json +19 -0
- package/src/i18n/locales/en.json +19 -0
- package/src/index.ts +31 -0
- package/src/main.ts +11 -0
- package/src/plugins/vuetify.ts +131 -0
- package/src/shims-vue.d.ts +10 -0
- package/src/stories/Accordion.stories.ts +650 -0
- package/src/stories/Audio.stories.ts +29 -0
- package/src/stories/Button.stories.ts +263 -0
- package/src/stories/CheckBox.stories.ts +348 -0
- package/src/stories/DateInput.stories.ts +54 -0
- package/src/stories/Dialog.stories.ts +147 -0
- package/src/stories/EditField.stories.ts +79 -0
- package/src/stories/IconBullet/IconBullet.stories.ts +201 -0
- package/src/stories/IconBullet/IconBulletList.stories.ts +275 -0
- package/src/stories/Input.stories.ts +351 -0
- package/src/stories/Laboratory/Cards/AppointmentCard/AppointmentCard.stories.ts +260 -0
- package/src/stories/Laboratory/Cards/DocumentCard/DocumentCard.stories.ts +176 -0
- package/src/stories/Laboratory/Cards/DocumentCard/DocumentCardItem.stories.ts +119 -0
- package/src/stories/Laboratory/Cards/InfoCard/InfoCard.stories.ts +320 -0
- package/src/stories/Laboratory/Cards/TicketCard/TicketCard.stories.ts +335 -0
- package/src/stories/Laboratory/Chat/ChatBoxImage.stories.ts +82 -0
- package/src/stories/Laboratory/Chat/ChatMessage.stories.ts +198 -0
- package/src/stories/Laboratory/Chat/ChatMessageBadge.stories.ts +204 -0
- package/src/stories/Laboratory/Chat/ChatNotification.stories.ts +144 -0
- package/src/stories/Laboratory/Chat/ProgressLinear.stories.ts +186 -0
- package/src/stories/Laboratory/Chat/StatusNotification.stories.ts +111 -0
- package/src/stories/Laboratory/MainColumnsBar.stories.ts +48 -0
- package/src/stories/Laboratory/ProgressCircle.stories.ts +261 -0
- package/src/stories/Laboratory/SelectionColumnBar.stories.ts +234 -0
- package/src/stories/Laboratory/TagLabel.stories.ts +418 -0
- package/src/stories/Laboratory/TagLabelGroup.stories.ts +234 -0
- package/src/stories/Laboratory/Timeline.stories.ts +403 -0
- package/src/stories/NotificationBubble.stories.ts +194 -0
- package/src/stories/OtpInput.stories.ts +101 -0
- package/src/stories/PhoneInput.stories.ts +53 -0
- package/src/stories/Select.stories.ts +419 -0
- package/src/stories/TextArea.stories.ts +112 -0
- package/src/stories/TickBox.stories.ts +294 -0
- package/src/stories/assets/accessibility.png +0 -0
- package/src/stories/assets/accessibility.svg +1 -0
- package/src/stories/assets/addon-library.png +0 -0
- package/src/stories/assets/assets.png +0 -0
- package/src/stories/assets/avif-test-image.avif +0 -0
- package/src/stories/assets/context.png +0 -0
- package/src/stories/assets/discord.svg +1 -0
- package/src/stories/assets/docs.png +0 -0
- package/src/stories/assets/figma-plugin.png +0 -0
- package/src/stories/assets/github.svg +1 -0
- package/src/stories/assets/share.png +0 -0
- package/src/stories/assets/styling.png +0 -0
- package/src/stories/assets/testing.png +0 -0
- package/src/stories/assets/theming.png +0 -0
- package/src/stories/assets/tutorials.svg +1 -0
- package/src/stories/assets/youtube.svg +1 -0
- package/src/stories/v-icon.stories.ts +91 -0
- package/src/types/index.ts +21 -0
- package/src/vite-env.d.ts +1 -0
- package/tests/e2e/README.md +221 -0
- package/tests/e2e/accessibility.spec.ts +639 -0
- package/tests/e2e/accordion.spec.ts +42 -0
- package/tests/e2e/additional-components.spec.ts +438 -0
- package/tests/e2e/all-components.spec.ts +135 -0
- package/tests/e2e/button-fixed.spec.ts +59 -0
- package/tests/e2e/button.spec.ts +76 -0
- package/tests/e2e/checkbox.spec.ts +50 -0
- package/tests/e2e/date-input.spec.ts +46 -0
- package/tests/e2e/debug.spec.ts +52 -0
- package/tests/e2e/dialog.spec.ts +58 -0
- package/tests/e2e/input.spec.ts +55 -0
- package/tests/e2e/laboratory-components.spec.ts +321 -0
- package/tests/e2e/otp-input.spec.ts +50 -0
- package/tests/e2e/select.spec.ts +52 -0
- package/tests/e2e/storybook-utils.ts +59 -0
- package/tests/e2e/test-basic.spec.ts +34 -0
- package/tests/e2e/visual-regression.spec.ts +351 -0
- package/tests/unit/components/Accordion/AccordionGroup.spec.ts +343 -0
- package/tests/unit/components/Accordion/AccordionItem.spec.ts +384 -0
- package/tests/unit/components/Audio/Audio.spec.ts +404 -0
- package/tests/unit/components/Audio/Waveform.spec.ts +484 -0
- package/tests/unit/components/Core/Button.spec.ts +337 -0
- package/tests/unit/components/Core/Checkbox.spec.ts +545 -0
- package/tests/unit/components/Core/DateInput.spec.ts +691 -0
- package/tests/unit/components/Core/Dialog.spec.ts +486 -0
- package/tests/unit/components/Core/EditField.spec.ts +783 -0
- package/tests/unit/components/Core/Input.spec.ts +513 -0
- package/tests/unit/components/Core/Modal.spec.ts +519 -0
- package/tests/unit/components/Core/NotificationBubble.spec.ts +607 -0
- package/tests/unit/components/Core/OtpInput.spec.ts +709 -0
- package/tests/unit/components/Core/PhoneInput.spec.ts +620 -0
- package/tests/unit/components/Core/Select.spec.ts +713 -0
- package/tests/unit/components/Core/TextArea.spec.ts +566 -0
- package/tests/unit/components/Core/TickBox.spec.ts +780 -0
- package/tests/unit/components/IconBullet/IconBullet.spec.ts +357 -0
- package/tests/unit/components/IconBullet/IconBulletList.spec.ts +372 -0
- package/tests/unit/components/Icons/Audio/CloudFailed.spec.ts +109 -0
- package/tests/unit/components/Icons/Audio/CloudSaved.spec.ts +150 -0
- package/tests/unit/components/Icons/Audio/Delete.spec.ts +159 -0
- package/tests/unit/components/Icons/Audio/Pause.spec.ts +209 -0
- package/tests/unit/components/Icons/Audio/Play.spec.ts +218 -0
- package/tests/unit/components/Icons/CalendarNotification.spec.ts +187 -0
- package/tests/unit/components/Icons/Chair.spec.ts +235 -0
- package/tests/unit/components/Icons/ChairNotification.spec.ts +312 -0
- package/tests/unit/components/Icons/Circle.spec.ts +256 -0
- package/tests/unit/components/Icons/FavIcon.spec.ts +252 -0
- package/tests/unit/components/Icons/FilledCircle.spec.ts +275 -0
- package/tests/unit/components/Icons/Group3.spec.ts +356 -0
- package/tests/unit/components/Icons/RingNotification.spec.ts +394 -0
- package/tests/unit/components/Icons/calendar.spec.ts +287 -0
- package/tests/unit/components/Icons/checkbox.spec.ts +316 -0
- package/tests/unit/components/Icons/outlineChecked.spec.ts +435 -0
- package/tests/unit/components/Icons/play.spec.ts +309 -0
- package/tests/unit/components/Laboratory/AppointmentCard.spec.ts +168 -0
- package/tests/unit/components/Laboratory/ChatBoxImage.spec.ts +180 -0
- package/tests/unit/components/Laboratory/ChatMessage.spec.ts +264 -0
- package/tests/unit/components/Laboratory/ChatMessageBadge.spec.ts +283 -0
- package/tests/unit/components/Laboratory/ChatNotification.spec.ts +257 -0
- package/tests/unit/components/Laboratory/DocumentCard.spec.ts +229 -0
- package/tests/unit/components/Laboratory/DocumentCardItem.spec.ts +237 -0
- package/tests/unit/components/Laboratory/InfoCard.spec.ts +309 -0
- package/tests/unit/components/Laboratory/MainColumnsBar.spec.ts +252 -0
- package/tests/unit/components/Laboratory/ProgressCircle.spec.ts +291 -0
- package/tests/unit/components/Laboratory/ProgressLinear.spec.ts +276 -0
- package/tests/unit/components/Laboratory/SelectionColumnBar.spec.ts +289 -0
- package/tests/unit/components/Laboratory/StatusNotification.spec.ts +297 -0
- package/tests/unit/components/Laboratory/TagLabel.spec.ts +354 -0
- package/tests/unit/components/Laboratory/TagLabelGroup.spec.ts +378 -0
- package/tests/unit/components/Laboratory/TicketCard.spec.ts +352 -0
- package/tests/unit/components/Laboratory/TimeLineEvent.spec.ts +382 -0
- package/tests/unit/components/Laboratory/Timeline.spec.ts +420 -0
- package/tests/unit/constants/iconEnums.spec.ts +40 -0
- package/tests/unit/i18n/i18n.spec.ts +89 -0
- package/tests/unit/plugins/vuetify.spec.ts +221 -0
- package/tests/unit/setup.ts +190 -0
- package/tests/unit/src/components/index.spec.ts +193 -0
- package/tests/unit/src/index.spec.ts +183 -0
- package/tests/unit/src/main.spec.ts +152 -0
- package/tsconfig.json +26 -0
- package/vite.config.ts +29 -0
- package/vitest.config.ts +84 -0
|
@@ -0,0 +1,394 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { mount } from '@vue/test-utils';
|
|
3
|
+
import RingNotification from '@components/Icons/RingNotification.vue';
|
|
4
|
+
|
|
5
|
+
describe('RingNotification Icon', () => {
|
|
6
|
+
describe('Rendering', () => {
|
|
7
|
+
it('renders as SVG element', () => {
|
|
8
|
+
const wrapper = mount(RingNotification);
|
|
9
|
+
expect(wrapper.find('svg').exists()).toBe(true);
|
|
10
|
+
});
|
|
11
|
+
|
|
12
|
+
it('has correct SVG attributes', () => {
|
|
13
|
+
const wrapper = mount(RingNotification);
|
|
14
|
+
const svg = wrapper.find('svg');
|
|
15
|
+
|
|
16
|
+
expect(svg.attributes('width')).toBe('23');
|
|
17
|
+
expect(svg.attributes('height')).toBe('23');
|
|
18
|
+
expect(svg.attributes('viewBox')).toBe('0 0 23 23');
|
|
19
|
+
expect(svg.attributes('fill')).toBe('none');
|
|
20
|
+
expect(svg.attributes('xmlns')).toBe('http://www.w3.org/2000/svg');
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
it('contains group with clip path and paths', () => {
|
|
24
|
+
const wrapper = mount(RingNotification);
|
|
25
|
+
const group = wrapper.find('g');
|
|
26
|
+
const paths = wrapper.findAll('path');
|
|
27
|
+
const clipPath = wrapper.find('clipPath');
|
|
28
|
+
|
|
29
|
+
expect(group.exists()).toBe(true);
|
|
30
|
+
expect(paths).toHaveLength(5);
|
|
31
|
+
expect(clipPath.exists()).toBe(true);
|
|
32
|
+
});
|
|
33
|
+
});
|
|
34
|
+
|
|
35
|
+
describe('Path Structure', () => {
|
|
36
|
+
it('first path represents notification circle', () => {
|
|
37
|
+
const wrapper = mount(RingNotification);
|
|
38
|
+
const paths = wrapper.findAll('path');
|
|
39
|
+
|
|
40
|
+
const notificationCircle = paths[0];
|
|
41
|
+
expect(notificationCircle.attributes('d')).toContain('M18.6328 12.6722C20.6723');
|
|
42
|
+
expect(notificationCircle.attributes('fill')).toBe('#F4297E');
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
it('remaining paths represent bell structure', () => {
|
|
46
|
+
const wrapper = mount(RingNotification);
|
|
47
|
+
const paths = wrapper.findAll('path');
|
|
48
|
+
|
|
49
|
+
// Bell outline paths (paths 1-4)
|
|
50
|
+
for (let i = 1; i < paths.length; i++) {
|
|
51
|
+
expect(paths[i].attributes('stroke')).toBe('#172774');
|
|
52
|
+
expect(paths[i].attributes('stroke-width')).toBe('1.5');
|
|
53
|
+
expect(paths[i].attributes('stroke-linecap')).toBe('round');
|
|
54
|
+
expect(paths[i].attributes('stroke-linejoin')).toBe('round');
|
|
55
|
+
}
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
it('contains bell shape and clapper elements', () => {
|
|
59
|
+
const wrapper = mount(RingNotification);
|
|
60
|
+
const paths = wrapper.findAll('path');
|
|
61
|
+
|
|
62
|
+
// Bell body (curved outline)
|
|
63
|
+
expect(paths[1].attributes('d')).toContain('16.9778 3.75416');
|
|
64
|
+
expect(paths[1].attributes('d')).toContain('11.5841 1.14923');
|
|
65
|
+
|
|
66
|
+
// Bell mouth/sound waves
|
|
67
|
+
expect(paths[2].attributes('d')).toContain('14.8633 17.3457');
|
|
68
|
+
expect(paths[2].attributes('d')).toContain('21.1304 15.8441');
|
|
69
|
+
|
|
70
|
+
// Bell clapper/handle
|
|
71
|
+
expect(paths[4].attributes('d')).toContain('14.8633 17.3458');
|
|
72
|
+
expect(paths[4].attributes('d')).toContain('11.5842 21.8508');
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
it('includes notification badge with distinctive color', () => {
|
|
76
|
+
const wrapper = mount(RingNotification);
|
|
77
|
+
const paths = wrapper.findAll('path');
|
|
78
|
+
|
|
79
|
+
const notificationBadge = paths[0];
|
|
80
|
+
expect(notificationBadge.attributes('fill')).toBe('#F4297E');
|
|
81
|
+
expect(notificationBadge.attributes('stroke')).toBeUndefined();
|
|
82
|
+
});
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
describe('Visual Properties', () => {
|
|
86
|
+
it('uses notification color for alert badge', () => {
|
|
87
|
+
const wrapper = mount(RingNotification);
|
|
88
|
+
const paths = wrapper.findAll('path');
|
|
89
|
+
|
|
90
|
+
const notificationCircle = paths[0];
|
|
91
|
+
expect(notificationCircle.attributes('fill')).toBe('#F4297E');
|
|
92
|
+
});
|
|
93
|
+
|
|
94
|
+
it('uses brand color for bell structure', () => {
|
|
95
|
+
const wrapper = mount(RingNotification);
|
|
96
|
+
const paths = wrapper.findAll('path');
|
|
97
|
+
|
|
98
|
+
// Bell structure paths (excluding notification badge)
|
|
99
|
+
for (let i = 1; i < paths.length; i++) {
|
|
100
|
+
expect(paths[i].attributes('stroke')).toBe('#172774');
|
|
101
|
+
}
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
it('svg container has no default fill', () => {
|
|
105
|
+
const wrapper = mount(RingNotification);
|
|
106
|
+
const svg = wrapper.find('svg');
|
|
107
|
+
|
|
108
|
+
expect(svg.attributes('fill')).toBe('none');
|
|
109
|
+
});
|
|
110
|
+
|
|
111
|
+
it('uses proper stroke properties for bell outline', () => {
|
|
112
|
+
const wrapper = mount(RingNotification);
|
|
113
|
+
const paths = wrapper.findAll('path');
|
|
114
|
+
|
|
115
|
+
// Bell paths have consistent stroke styling
|
|
116
|
+
for (let i = 1; i < paths.length; i++) {
|
|
117
|
+
expect(paths[i].attributes('stroke-width')).toBe('1.5');
|
|
118
|
+
expect(paths[i].attributes('stroke-linecap')).toBe('round');
|
|
119
|
+
expect(paths[i].attributes('stroke-linejoin')).toBe('round');
|
|
120
|
+
}
|
|
121
|
+
});
|
|
122
|
+
});
|
|
123
|
+
|
|
124
|
+
describe('Dimensions', () => {
|
|
125
|
+
it('uses square aspect ratio', () => {
|
|
126
|
+
const wrapper = mount(RingNotification);
|
|
127
|
+
const svg = wrapper.find('svg');
|
|
128
|
+
|
|
129
|
+
expect(svg.attributes('width')).toBe('23');
|
|
130
|
+
expect(svg.attributes('height')).toBe('23');
|
|
131
|
+
});
|
|
132
|
+
|
|
133
|
+
it('has correct viewBox dimensions', () => {
|
|
134
|
+
const wrapper = mount(RingNotification);
|
|
135
|
+
const svg = wrapper.find('svg');
|
|
136
|
+
|
|
137
|
+
expect(svg.attributes('viewBox')).toBe('0 0 23 23');
|
|
138
|
+
});
|
|
139
|
+
|
|
140
|
+
it('suitable for notification icon size', () => {
|
|
141
|
+
const wrapper = mount(RingNotification);
|
|
142
|
+
const svg = wrapper.find('svg');
|
|
143
|
+
|
|
144
|
+
// 23x23 is appropriate for notification/alert icons
|
|
145
|
+
expect(svg.attributes('width')).toBe('23');
|
|
146
|
+
expect(svg.attributes('height')).toBe('23');
|
|
147
|
+
});
|
|
148
|
+
});
|
|
149
|
+
|
|
150
|
+
describe('Clipping and Layout', () => {
|
|
151
|
+
it('contains clipping path for proper masking', () => {
|
|
152
|
+
const wrapper = mount(RingNotification);
|
|
153
|
+
const clipPath = wrapper.find('clipPath');
|
|
154
|
+
const rect = clipPath.find('rect');
|
|
155
|
+
|
|
156
|
+
expect(clipPath.attributes('id')).toBe('clip0_1311_589');
|
|
157
|
+
expect(rect.attributes('width')).toBe('21.4524');
|
|
158
|
+
expect(rect.attributes('height')).toBe('23');
|
|
159
|
+
expect(rect.attributes('fill')).toBe('white');
|
|
160
|
+
});
|
|
161
|
+
|
|
162
|
+
it('group references clipping path correctly', () => {
|
|
163
|
+
const wrapper = mount(RingNotification);
|
|
164
|
+
const group = wrapper.find('g');
|
|
165
|
+
|
|
166
|
+
expect(group.attributes('clip-path')).toBe('url(#clip0_1311_589)');
|
|
167
|
+
});
|
|
168
|
+
|
|
169
|
+
it('notification badge positioned at top right', () => {
|
|
170
|
+
const wrapper = mount(RingNotification);
|
|
171
|
+
const paths = wrapper.findAll('path');
|
|
172
|
+
|
|
173
|
+
const notificationPath = paths[0].attributes('d');
|
|
174
|
+
// Badge should be positioned in upper right area
|
|
175
|
+
expect(notificationPath).toMatch(/18\.6328.*12\.6722/);
|
|
176
|
+
expect(notificationPath).toMatch(/22\.3257.*8\.97937/);
|
|
177
|
+
});
|
|
178
|
+
|
|
179
|
+
it('bell structure spans appropriate area', () => {
|
|
180
|
+
const wrapper = mount(RingNotification);
|
|
181
|
+
const paths = wrapper.findAll('path');
|
|
182
|
+
|
|
183
|
+
// Bell should span significant portion of viewBox
|
|
184
|
+
const bellPath = paths[1].attributes('d');
|
|
185
|
+
expect(bellPath).toMatch(/1\.14923/); // Near left edge
|
|
186
|
+
expect(bellPath).toMatch(/17\.3304/); // Extends right
|
|
187
|
+
});
|
|
188
|
+
});
|
|
189
|
+
|
|
190
|
+
describe('Component Structure', () => {
|
|
191
|
+
it('represents notification bell with alert badge', () => {
|
|
192
|
+
const wrapper = mount(RingNotification);
|
|
193
|
+
const paths = wrapper.findAll('path');
|
|
194
|
+
|
|
195
|
+
// 1 notification badge + 4 bell structure paths
|
|
196
|
+
expect(paths).toHaveLength(5);
|
|
197
|
+
|
|
198
|
+
// Badge is filled, bell is outlined
|
|
199
|
+
expect(paths[0].attributes('fill')).toBeTruthy();
|
|
200
|
+
expect(paths[1].attributes('stroke')).toBeTruthy();
|
|
201
|
+
});
|
|
202
|
+
|
|
203
|
+
it('includes defs section for clipping', () => {
|
|
204
|
+
const wrapper = mount(RingNotification);
|
|
205
|
+
const defs = wrapper.find('defs');
|
|
206
|
+
const clipPath = defs.find('clipPath');
|
|
207
|
+
|
|
208
|
+
expect(defs.exists()).toBe(true);
|
|
209
|
+
expect(clipPath.exists()).toBe(true);
|
|
210
|
+
});
|
|
211
|
+
|
|
212
|
+
it('maintains proper visual hierarchy', () => {
|
|
213
|
+
const wrapper = mount(RingNotification);
|
|
214
|
+
const paths = wrapper.findAll('path');
|
|
215
|
+
|
|
216
|
+
// Notification badge uses bright color for attention
|
|
217
|
+
expect(paths[0].attributes('fill')).toBe('#F4297E');
|
|
218
|
+
|
|
219
|
+
// Bell structure uses subtle strokes
|
|
220
|
+
expect(paths[1].attributes('stroke-width')).toBe('1.5');
|
|
221
|
+
});
|
|
222
|
+
});
|
|
223
|
+
|
|
224
|
+
describe('Bell Geometry', () => {
|
|
225
|
+
it('bell shape has characteristic curves', () => {
|
|
226
|
+
const wrapper = mount(RingNotification);
|
|
227
|
+
const paths = wrapper.findAll('path');
|
|
228
|
+
|
|
229
|
+
// Bell outline should have curved elements
|
|
230
|
+
const bellOutline = paths[1].attributes('d');
|
|
231
|
+
expect(bellOutline).toMatch(/16\.9778.*3\.75416/); // Bell top curve
|
|
232
|
+
expect(bellOutline).toMatch(/15\.6141.*2\.4517/); // Bell side curve
|
|
233
|
+
});
|
|
234
|
+
|
|
235
|
+
it('includes bell mouth and sound representation', () => {
|
|
236
|
+
const wrapper = mount(RingNotification);
|
|
237
|
+
const paths = wrapper.findAll('path');
|
|
238
|
+
|
|
239
|
+
// Bell mouth/sound waves
|
|
240
|
+
const soundPath = paths[2].attributes('d');
|
|
241
|
+
expect(soundPath).toContain('21.1304 15.8441'); // Sound wave extent
|
|
242
|
+
expect(soundPath).toMatch(/20\.9312.*15\.6142/); // Wave details
|
|
243
|
+
});
|
|
244
|
+
|
|
245
|
+
it('has bell handle/clapper element', () => {
|
|
246
|
+
const wrapper = mount(RingNotification);
|
|
247
|
+
const paths = wrapper.findAll('path');
|
|
248
|
+
|
|
249
|
+
// Bell handle/clapper at bottom
|
|
250
|
+
const handlePath = paths[4].attributes('d');
|
|
251
|
+
expect(handlePath).toMatch(/11\.5842.*21\.8508/); // Handle center
|
|
252
|
+
expect(handlePath).toMatch(/8\.30503.*17\.3458/); // Handle width
|
|
253
|
+
});
|
|
254
|
+
});
|
|
255
|
+
|
|
256
|
+
describe('Accessibility', () => {
|
|
257
|
+
it('svg is properly structured for screen readers', () => {
|
|
258
|
+
const wrapper = mount(RingNotification);
|
|
259
|
+
const svg = wrapper.find('svg');
|
|
260
|
+
|
|
261
|
+
expect(svg.element.tagName.toLowerCase()).toBe('svg');
|
|
262
|
+
});
|
|
263
|
+
|
|
264
|
+
it('combines bell icon with clear notification indicator', () => {
|
|
265
|
+
const wrapper = mount(RingNotification);
|
|
266
|
+
const paths = wrapper.findAll('path');
|
|
267
|
+
|
|
268
|
+
// Bell structure (outline style for clarity)
|
|
269
|
+
expect(paths[1].attributes('stroke')).toBeTruthy();
|
|
270
|
+
|
|
271
|
+
// Notification badge (filled style for visibility)
|
|
272
|
+
expect(paths[0].attributes('fill')).toBe('#F4297E');
|
|
273
|
+
});
|
|
274
|
+
|
|
275
|
+
it('uses high contrast colors', () => {
|
|
276
|
+
const wrapper = mount(RingNotification);
|
|
277
|
+
const paths = wrapper.findAll('path');
|
|
278
|
+
|
|
279
|
+
// Dark stroke on light background
|
|
280
|
+
expect(paths[1].attributes('stroke')).toBe('#172774');
|
|
281
|
+
|
|
282
|
+
// Bright notification color
|
|
283
|
+
expect(paths[0].attributes('fill')).toBe('#F4297E');
|
|
284
|
+
});
|
|
285
|
+
});
|
|
286
|
+
|
|
287
|
+
describe('Component Integration', () => {
|
|
288
|
+
it('can be mounted without props', () => {
|
|
289
|
+
expect(() => {
|
|
290
|
+
mount(RingNotification);
|
|
291
|
+
}).not.toThrow();
|
|
292
|
+
});
|
|
293
|
+
|
|
294
|
+
it('renders consistently on multiple mounts', () => {
|
|
295
|
+
const wrapper1 = mount(RingNotification);
|
|
296
|
+
const wrapper2 = mount(RingNotification);
|
|
297
|
+
|
|
298
|
+
expect(wrapper1.html()).toBe(wrapper2.html());
|
|
299
|
+
});
|
|
300
|
+
|
|
301
|
+
it('maintains bell notification structure integrity', () => {
|
|
302
|
+
const wrapper = mount(RingNotification);
|
|
303
|
+
const paths = wrapper.findAll('path');
|
|
304
|
+
const group = wrapper.find('g');
|
|
305
|
+
const clipPath = wrapper.find('clipPath');
|
|
306
|
+
|
|
307
|
+
expect(paths).toHaveLength(5);
|
|
308
|
+
expect(group.exists()).toBe(true);
|
|
309
|
+
expect(clipPath.exists()).toBe(true);
|
|
310
|
+
|
|
311
|
+
paths.forEach(path => {
|
|
312
|
+
expect(path.attributes('d')).toBeTruthy();
|
|
313
|
+
});
|
|
314
|
+
});
|
|
315
|
+
|
|
316
|
+
it('suitable for notification/alert contexts', () => {
|
|
317
|
+
const wrapper = mount(RingNotification);
|
|
318
|
+
|
|
319
|
+
// Appropriate size and clear visual hierarchy for notifications
|
|
320
|
+
expect(wrapper.find('svg').attributes('width')).toBe('23');
|
|
321
|
+
expect(wrapper.find('svg').attributes('height')).toBe('23');
|
|
322
|
+
expect(wrapper.findAll('path')[0].attributes('fill')).toBe('#F4297E');
|
|
323
|
+
});
|
|
324
|
+
});
|
|
325
|
+
|
|
326
|
+
describe('Icon Semantics', () => {
|
|
327
|
+
it('represents notification/alert with bell metaphor', () => {
|
|
328
|
+
const wrapper = mount(RingNotification);
|
|
329
|
+
const paths = wrapper.findAll('path');
|
|
330
|
+
|
|
331
|
+
// Bell = notification, badge = alert/count indicator
|
|
332
|
+
expect(paths.length).toBe(5); // Bell structure + notification badge
|
|
333
|
+
expect(paths[0].attributes('fill')).toBe('#F4297E'); // Alert color
|
|
334
|
+
});
|
|
335
|
+
|
|
336
|
+
it('uses universally recognized notification color', () => {
|
|
337
|
+
const wrapper = mount(RingNotification);
|
|
338
|
+
const paths = wrapper.findAll('path');
|
|
339
|
+
|
|
340
|
+
// Pink/magenta is widely used for notifications/alerts
|
|
341
|
+
const notificationColor = paths[0].attributes('fill');
|
|
342
|
+
expect(notificationColor).toBe('#F4297E');
|
|
343
|
+
expect(notificationColor.toLowerCase()).toMatch(/^#[a-f0-9]/); // Valid hex color
|
|
344
|
+
});
|
|
345
|
+
|
|
346
|
+
it('maintains bell recognizability with notification overlay', () => {
|
|
347
|
+
const wrapper = mount(RingNotification);
|
|
348
|
+
const paths = wrapper.findAll('path');
|
|
349
|
+
|
|
350
|
+
// Bell structure should still be clearly defined
|
|
351
|
+
expect(paths[1].attributes('d')).toMatch(/16\.9778.*3\.75416/); // Bell curve
|
|
352
|
+
expect(paths[2].attributes('d')).toMatch(/21\.1304.*15\.8441/); // Bell mouth
|
|
353
|
+
expect(paths[3].attributes('d')).toMatch(/11\.5841.*17\.545/); // Bell handle
|
|
354
|
+
});
|
|
355
|
+
});
|
|
356
|
+
|
|
357
|
+
describe('Visual Design', () => {
|
|
358
|
+
it('provides clear visual hierarchy', () => {
|
|
359
|
+
const wrapper = mount(RingNotification);
|
|
360
|
+
const paths = wrapper.findAll('path');
|
|
361
|
+
|
|
362
|
+
// Notification badge is prominent (bright fill)
|
|
363
|
+
expect(paths[0].attributes('fill')).toBe('#F4297E');
|
|
364
|
+
|
|
365
|
+
// Bell structure is subtle but clear (strokes)
|
|
366
|
+
expect(paths[1].attributes('stroke')).toBe('#172774');
|
|
367
|
+
expect(paths[1].attributes('stroke-width')).toBe('1.5');
|
|
368
|
+
});
|
|
369
|
+
|
|
370
|
+
it('balances bell detail with notification visibility', () => {
|
|
371
|
+
const wrapper = mount(RingNotification);
|
|
372
|
+
const paths = wrapper.findAll('path');
|
|
373
|
+
|
|
374
|
+
// Bell has multiple detail paths but consistent styling
|
|
375
|
+
for (let i = 1; i < paths.length; i++) {
|
|
376
|
+
expect(paths[i].attributes('stroke')).toBe('#172774');
|
|
377
|
+
expect(paths[i].attributes('stroke-width')).toBe('1.5');
|
|
378
|
+
}
|
|
379
|
+
|
|
380
|
+
// Notification is simple but prominent
|
|
381
|
+
expect(paths[0].attributes('d')).toMatch(/C.*C.*C/); // Circle path
|
|
382
|
+
});
|
|
383
|
+
|
|
384
|
+
it('uses clipping for clean composition', () => {
|
|
385
|
+
const wrapper = mount(RingNotification);
|
|
386
|
+
const group = wrapper.find('g');
|
|
387
|
+
const clipPath = wrapper.find('clipPath');
|
|
388
|
+
|
|
389
|
+
// Clipping ensures clean edges and proper overlay
|
|
390
|
+
expect(group.attributes('clip-path')).toBe('url(#clip0_1311_589)');
|
|
391
|
+
expect(clipPath.exists()).toBe(true);
|
|
392
|
+
});
|
|
393
|
+
});
|
|
394
|
+
});
|
|
@@ -0,0 +1,287 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { mount } from '@vue/test-utils';
|
|
3
|
+
import Calendar from '@components/Icons/calendar.vue';
|
|
4
|
+
|
|
5
|
+
describe('Calendar Icon', () => {
|
|
6
|
+
describe('Rendering', () => {
|
|
7
|
+
it('renders as SVG element', () => {
|
|
8
|
+
const wrapper = mount(Calendar);
|
|
9
|
+
expect(wrapper.find('svg').exists()).toBe(true);
|
|
10
|
+
});
|
|
11
|
+
|
|
12
|
+
it('has correct SVG attributes', () => {
|
|
13
|
+
const wrapper = mount(Calendar);
|
|
14
|
+
const svg = wrapper.find('svg');
|
|
15
|
+
|
|
16
|
+
expect(svg.attributes('width')).toBe('21');
|
|
17
|
+
expect(svg.attributes('height')).toBe('20');
|
|
18
|
+
expect(svg.attributes('viewBox')).toBe('0 0 21 20');
|
|
19
|
+
expect(svg.attributes('fill')).toBe('none');
|
|
20
|
+
expect(svg.attributes('xmlns')).toBe('http://www.w3.org/2000/svg');
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
it('contains exactly 1 path element', () => {
|
|
24
|
+
const wrapper = mount(Calendar);
|
|
25
|
+
const paths = wrapper.findAll('path');
|
|
26
|
+
expect(paths).toHaveLength(1);
|
|
27
|
+
});
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
describe('Path Structure', () => {
|
|
31
|
+
it('path represents complete calendar structure', () => {
|
|
32
|
+
const wrapper = mount(Calendar);
|
|
33
|
+
const path = wrapper.find('path');
|
|
34
|
+
|
|
35
|
+
const pathData = path.attributes('d');
|
|
36
|
+
expect(pathData).toContain('M5.07568 1V3.25'); // Top left binding ring
|
|
37
|
+
expect(pathData).toContain('M15.5757 1V3.25'); // Top right binding ring
|
|
38
|
+
expect(pathData).toContain('M1.32568 16.75V5.5'); // Calendar body
|
|
39
|
+
});
|
|
40
|
+
|
|
41
|
+
it('path has correct stroke properties', () => {
|
|
42
|
+
const wrapper = mount(Calendar);
|
|
43
|
+
const path = wrapper.find('path');
|
|
44
|
+
|
|
45
|
+
expect(path.attributes('stroke')).toBe('#172774');
|
|
46
|
+
expect(path.attributes('stroke-width')).toBe('1.5');
|
|
47
|
+
expect(path.attributes('stroke-linecap')).toBe('round');
|
|
48
|
+
expect(path.attributes('stroke-linejoin')).toBe('round');
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
it('includes binding rings at top', () => {
|
|
52
|
+
const wrapper = mount(Calendar);
|
|
53
|
+
const path = wrapper.find('path');
|
|
54
|
+
|
|
55
|
+
const pathData = path.attributes('d');
|
|
56
|
+
// Binding rings that extend above the calendar body
|
|
57
|
+
expect(pathData).toContain('M5.07568 1V3.25'); // Left ring
|
|
58
|
+
expect(pathData).toContain('M15.5757 1V3.25'); // Right ring
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
it('includes calendar body structure', () => {
|
|
62
|
+
const wrapper = mount(Calendar);
|
|
63
|
+
const path = wrapper.find('path');
|
|
64
|
+
|
|
65
|
+
const pathData = path.attributes('d');
|
|
66
|
+
// Calendar body with rounded corners
|
|
67
|
+
expect(pathData).toContain('C1.32568 4.25736 2.33304 3.25'); // Top body
|
|
68
|
+
expect(pathData).toContain('C1.32568 17.9926 2.33304 19'); // Bottom body
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
it('includes date grid dots', () => {
|
|
72
|
+
const wrapper = mount(Calendar);
|
|
73
|
+
const path = wrapper.find('path');
|
|
74
|
+
|
|
75
|
+
const pathData = path.attributes('d');
|
|
76
|
+
// Multiple small rectangles representing dates
|
|
77
|
+
expect(pathData).toContain('H10.3332V10.7575'); // Date dots
|
|
78
|
+
expect(pathData).toContain('H12.5832V13.0075'); // More date dots
|
|
79
|
+
expect(pathData).toContain('H14.8332V10.7575'); // Even more date dots
|
|
80
|
+
});
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
describe('Visual Properties', () => {
|
|
84
|
+
it('uses correct brand color for stroke', () => {
|
|
85
|
+
const wrapper = mount(Calendar);
|
|
86
|
+
const path = wrapper.find('path');
|
|
87
|
+
|
|
88
|
+
expect(path.attributes('stroke')).toBe('#172774');
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
it('has appropriate stroke width for icon size', () => {
|
|
92
|
+
const wrapper = mount(Calendar);
|
|
93
|
+
const path = wrapper.find('path');
|
|
94
|
+
|
|
95
|
+
expect(path.attributes('stroke-width')).toBe('1.5');
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
it('uses rounded line caps and joins', () => {
|
|
99
|
+
const wrapper = mount(Calendar);
|
|
100
|
+
const path = wrapper.find('path');
|
|
101
|
+
|
|
102
|
+
expect(path.attributes('stroke-linecap')).toBe('round');
|
|
103
|
+
expect(path.attributes('stroke-linejoin')).toBe('round');
|
|
104
|
+
});
|
|
105
|
+
|
|
106
|
+
it('path has no fill (outline style)', () => {
|
|
107
|
+
const wrapper = mount(Calendar);
|
|
108
|
+
const svg = wrapper.find('svg');
|
|
109
|
+
|
|
110
|
+
expect(svg.attributes('fill')).toBe('none');
|
|
111
|
+
});
|
|
112
|
+
});
|
|
113
|
+
|
|
114
|
+
describe('Calendar Geometry', () => {
|
|
115
|
+
it('binding rings positioned at top', () => {
|
|
116
|
+
const wrapper = mount(Calendar);
|
|
117
|
+
const path = wrapper.find('path');
|
|
118
|
+
|
|
119
|
+
const pathData = path.attributes('d');
|
|
120
|
+
// Rings extend from y=1 to y=3.25
|
|
121
|
+
expect(pathData).toContain('1V3.25');
|
|
122
|
+
});
|
|
123
|
+
|
|
124
|
+
it('calendar body spans most of viewBox', () => {
|
|
125
|
+
const wrapper = mount(Calendar);
|
|
126
|
+
const path = wrapper.find('path');
|
|
127
|
+
|
|
128
|
+
const pathData = path.attributes('d');
|
|
129
|
+
// Body from y=3.25 to y=19, x=1.32568 to x=19.3257
|
|
130
|
+
expect(pathData).toContain('1.32568');
|
|
131
|
+
expect(pathData).toContain('19.3257');
|
|
132
|
+
expect(pathData).toContain('3.25');
|
|
133
|
+
expect(pathData).toContain('19');
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
it('includes header section separation', () => {
|
|
137
|
+
const wrapper = mount(Calendar);
|
|
138
|
+
const path = wrapper.find('path');
|
|
139
|
+
|
|
140
|
+
const pathData = path.attributes('d');
|
|
141
|
+
// Separator line at y=7 between header and dates
|
|
142
|
+
expect(pathData).toContain('7H17.0757');
|
|
143
|
+
expect(pathData).toContain('9.25V16.75');
|
|
144
|
+
});
|
|
145
|
+
|
|
146
|
+
it('date dots arranged in grid pattern', () => {
|
|
147
|
+
const wrapper = mount(Calendar);
|
|
148
|
+
const path = wrapper.find('path');
|
|
149
|
+
|
|
150
|
+
const pathData = path.attributes('d');
|
|
151
|
+
// Multiple columns and rows
|
|
152
|
+
expect(pathData).toContain('10.3257'); // Center column
|
|
153
|
+
expect(pathData).toContain('8.07568'); // Left column
|
|
154
|
+
expect(pathData).toContain('12.5757'); // Right column
|
|
155
|
+
expect(pathData).toContain('10.75'); // First row
|
|
156
|
+
expect(pathData).toContain('13H'); // Second row
|
|
157
|
+
expect(pathData).toContain('15.25'); // Third row
|
|
158
|
+
});
|
|
159
|
+
});
|
|
160
|
+
|
|
161
|
+
describe('Component Structure', () => {
|
|
162
|
+
it('uses appropriate dimensions for calendar icon', () => {
|
|
163
|
+
const wrapper = mount(Calendar);
|
|
164
|
+
const svg = wrapper.find('svg');
|
|
165
|
+
|
|
166
|
+
// Slightly wider than tall to accommodate calendar proportions
|
|
167
|
+
expect(svg.attributes('width')).toBe('21');
|
|
168
|
+
expect(svg.attributes('height')).toBe('20');
|
|
169
|
+
});
|
|
170
|
+
|
|
171
|
+
it('suitable for UI calendar representations', () => {
|
|
172
|
+
const wrapper = mount(Calendar);
|
|
173
|
+
const path = wrapper.find('path');
|
|
174
|
+
|
|
175
|
+
// Includes key calendar elements
|
|
176
|
+
const pathData = path.attributes('d');
|
|
177
|
+
expect(pathData).toContain('1V3.25'); // Binding rings
|
|
178
|
+
expect(pathData).toMatch(/H\d+\.\d+V\d+\.\d+/); // Grid structure
|
|
179
|
+
});
|
|
180
|
+
});
|
|
181
|
+
|
|
182
|
+
describe('Accessibility', () => {
|
|
183
|
+
it('svg is properly structured for screen readers', () => {
|
|
184
|
+
const wrapper = mount(Calendar);
|
|
185
|
+
const svg = wrapper.find('svg');
|
|
186
|
+
|
|
187
|
+
expect(svg.element.tagName.toLowerCase()).toBe('svg');
|
|
188
|
+
});
|
|
189
|
+
|
|
190
|
+
it('uses outline style for better contrast', () => {
|
|
191
|
+
const wrapper = mount(Calendar);
|
|
192
|
+
const path = wrapper.find('path');
|
|
193
|
+
|
|
194
|
+
// Outline style is more universally accessible
|
|
195
|
+
expect(path.attributes('stroke')).toBeTruthy();
|
|
196
|
+
expect(path.attributes('fill')).toBeFalsy();
|
|
197
|
+
});
|
|
198
|
+
});
|
|
199
|
+
|
|
200
|
+
describe('Component Integration', () => {
|
|
201
|
+
it('can be mounted without props', () => {
|
|
202
|
+
expect(() => {
|
|
203
|
+
mount(Calendar);
|
|
204
|
+
}).not.toThrow();
|
|
205
|
+
});
|
|
206
|
+
|
|
207
|
+
it('renders consistently on multiple mounts', () => {
|
|
208
|
+
const wrapper1 = mount(Calendar);
|
|
209
|
+
const wrapper2 = mount(Calendar);
|
|
210
|
+
|
|
211
|
+
expect(wrapper1.html()).toBe(wrapper2.html());
|
|
212
|
+
});
|
|
213
|
+
|
|
214
|
+
it('maintains calendar structure integrity', () => {
|
|
215
|
+
const wrapper = mount(Calendar);
|
|
216
|
+
const path = wrapper.find('path');
|
|
217
|
+
|
|
218
|
+
expect(path.exists()).toBe(true);
|
|
219
|
+
expect(path.attributes('d')).toBeTruthy();
|
|
220
|
+
expect(path.attributes('stroke')).toBeTruthy();
|
|
221
|
+
expect(path.attributes('stroke-width')).toBeTruthy();
|
|
222
|
+
});
|
|
223
|
+
|
|
224
|
+
it('suitable for various calendar UI contexts', () => {
|
|
225
|
+
const wrapper = mount(Calendar);
|
|
226
|
+
|
|
227
|
+
// Good size for buttons, form fields, navigation
|
|
228
|
+
expect(wrapper.find('svg').attributes('width')).toBe('21');
|
|
229
|
+
expect(wrapper.find('svg').attributes('height')).toBe('20');
|
|
230
|
+
});
|
|
231
|
+
});
|
|
232
|
+
|
|
233
|
+
describe('Icon Semantics', () => {
|
|
234
|
+
it('represents calendar with recognizable elements', () => {
|
|
235
|
+
const wrapper = mount(Calendar);
|
|
236
|
+
const path = wrapper.find('path');
|
|
237
|
+
|
|
238
|
+
const pathData = path.attributes('d');
|
|
239
|
+
// Key calendar features
|
|
240
|
+
expect(pathData).toMatch(/1V3\.25/); // Binding rings
|
|
241
|
+
expect(pathData).toMatch(/H\d+\.\d+V\d+\.\d+/); // Date grid
|
|
242
|
+
expect(pathData).toMatch(/C\d+/); // Rounded corners
|
|
243
|
+
});
|
|
244
|
+
|
|
245
|
+
it('uses standard calendar proportions', () => {
|
|
246
|
+
const wrapper = mount(Calendar);
|
|
247
|
+
const svg = wrapper.find('svg');
|
|
248
|
+
|
|
249
|
+
// Typical calendar aspect ratio (slightly wider than tall)
|
|
250
|
+
const width = parseInt(svg.attributes('width'));
|
|
251
|
+
const height = parseInt(svg.attributes('height'));
|
|
252
|
+
expect(width).toBeGreaterThan(height);
|
|
253
|
+
expect(width / height).toBeCloseTo(1.05, 1);
|
|
254
|
+
});
|
|
255
|
+
|
|
256
|
+
it('includes visual date indicators', () => {
|
|
257
|
+
const wrapper = mount(Calendar);
|
|
258
|
+
const path = wrapper.find('path');
|
|
259
|
+
|
|
260
|
+
const pathData = path.attributes('d');
|
|
261
|
+
// Should have multiple small date markers
|
|
262
|
+
const dateMarkers = pathData.match(/H\d+\.\d+V\d+\.\d+/g);
|
|
263
|
+
expect(dateMarkers).toBeTruthy();
|
|
264
|
+
expect(dateMarkers.length).toBeGreaterThan(5);
|
|
265
|
+
});
|
|
266
|
+
});
|
|
267
|
+
|
|
268
|
+
describe('Visual Design', () => {
|
|
269
|
+
it('provides clear calendar recognition', () => {
|
|
270
|
+
const wrapper = mount(Calendar);
|
|
271
|
+
const path = wrapper.find('path');
|
|
272
|
+
|
|
273
|
+
const pathData = path.attributes('d');
|
|
274
|
+
// Distinctive calendar elements
|
|
275
|
+
expect(pathData).toContain('1V3.25'); // Top rings
|
|
276
|
+
expect(pathData).toMatch(/\d+\.\d+H\d+\.\d+V/); // Grid pattern
|
|
277
|
+
});
|
|
278
|
+
|
|
279
|
+
it('uses appropriate line weight for clarity', () => {
|
|
280
|
+
const wrapper = mount(Calendar);
|
|
281
|
+
const path = wrapper.find('path');
|
|
282
|
+
|
|
283
|
+
// 1.5px stroke provides good visibility without being too heavy
|
|
284
|
+
expect(path.attributes('stroke-width')).toBe('1.5');
|
|
285
|
+
});
|
|
286
|
+
});
|
|
287
|
+
});
|