@zap-wunschlachen/wl-shared-components 1.0.38 → 1.0.40
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 +205 -205
- package/.github/workflows/static.yml +61 -61
- package/.github/workflows/update-snapshots.yml +37 -37
- package/.prettierrc +5 -5
- package/.storybook/main.ts +18 -18
- package/.storybook/preview.ts +37 -37
- package/.storybook/storyWrapper.vue +18 -18
- package/.storybook/withVuetifyTheme.decorator.ts +21 -21
- package/App.vue +33 -33
- package/README.md +56 -56
- package/heroicons.ts +75 -75
- package/index.html +19 -19
- package/package.json +67 -67
- package/playwright.config.ts +48 -48
- package/public/background.svg +60 -60
- package/public/style.css +187 -187
- package/public/technologies.svg +22 -22
- package/src/assets/css/base.css +234 -232
- package/src/assets/css/variables.css +112 -109
- package/src/components/Accordion/Accordion.css +59 -59
- package/src/components/Accordion/AccordionGroup.vue +51 -51
- package/src/components/Accordion/AccordionItem.vue +66 -66
- package/src/components/Appointment/Card/Actions.css +54 -54
- package/src/components/Appointment/Card/Actions.vue +99 -99
- package/src/components/Appointment/Card/AnamneseNotification.css +15 -15
- package/src/components/Appointment/Card/AnamneseNotification.vue +23 -23
- package/src/components/Appointment/Card/Card.css +80 -80
- package/src/components/Appointment/Card/Card.vue +93 -93
- package/src/components/Appointment/Card/Details.css +50 -50
- package/src/components/Appointment/Card/Details.vue +43 -43
- package/src/components/Audio/Audio.vue +187 -187
- package/src/components/Audio/Waveform.vue +118 -118
- package/src/components/Button/Button.vue +174 -174
- package/src/components/CheckBox/CheckBox.css +214 -214
- package/src/components/CheckBox/Checkbox.vue +138 -138
- package/src/components/DateInput/DateInput.css +2 -2
- package/src/components/DateInput/DateInput.vue +262 -262
- package/src/components/Dialog/Dialog.css +6 -6
- package/src/components/Dialog/Dialog.vue +38 -38
- package/src/components/EditField/EditField.css +19 -19
- package/src/components/EditField/EditField.vue +202 -202
- package/src/components/ErrorPage/ErrorPage.css +172 -172
- package/src/components/IconBullet/IconBullet.vue +86 -86
- package/src/components/IconBullet/IconBulletList.vue +41 -41
- package/src/components/Icons/AdvanceAppointments.vue +153 -153
- package/src/components/Icons/Audio/CloudFailed.vue +20 -20
- package/src/components/Icons/Audio/CloudSaved.vue +21 -21
- package/src/components/Icons/Audio/Delete.vue +15 -15
- package/src/components/Icons/Audio/Pause.vue +18 -18
- package/src/components/Icons/Audio/Play.vue +15 -15
- package/src/components/Icons/{calendar.vue → Calendar.vue} +17 -17
- package/src/components/Icons/CalendarNotification.vue +126 -126
- package/src/components/Icons/Chair.vue +32 -32
- package/src/components/Icons/ChairNotification.vue +35 -35
- package/src/components/Icons/Circle.vue +66 -66
- package/src/components/Icons/FavIcon.vue +22 -22
- package/src/components/Icons/FilledCircle.vue +11 -11
- package/src/components/Icons/Group3.vue +46 -46
- package/src/components/Icons/RingNotification.vue +54 -54
- package/src/components/Icons/SolidArrowRight.vue +14 -14
- package/src/components/Icons/checkbox.vue +19 -19
- package/src/components/Icons/outlineChecked.vue +27 -27
- package/src/components/Icons/play.vue +5 -5
- package/src/components/Input/Input.css +187 -187
- package/src/components/Input/Input.vue +253 -253
- package/src/components/Laboratory/AppointmentCard/AppointmentCard.css +7 -7
- package/src/components/Laboratory/AppointmentCard/AppointmentCard.vue +116 -116
- package/src/components/Laboratory/ChatBoxImage/ChatBoxImage.vue +81 -81
- package/src/components/Laboratory/ChatMessage/ChatMessage.vue +113 -113
- package/src/components/Laboratory/ChatMessage/ChatMessageBadge.css +4 -4
- package/src/components/Laboratory/ChatMessage/ChatMessageBadge.vue +99 -99
- package/src/components/Laboratory/ChatNotification/ChatNotification.vue +130 -130
- package/src/components/Laboratory/DocumentCard/DocumentCard.css +3 -3
- package/src/components/Laboratory/DocumentCard/DocumentCard.vue +50 -50
- package/src/components/Laboratory/DocumentCard/DocumentCardItem.vue +53 -53
- package/src/components/Laboratory/InfoCard/InfoCard.vue +162 -162
- package/src/components/Laboratory/MainColumnsBar/MainColumnsBar.vue +102 -102
- package/src/components/Laboratory/ProgressCircle/ProgressCircle.vue +152 -152
- package/src/components/Laboratory/ProgressLinear/ProgressLinear.css +33 -33
- package/src/components/Laboratory/ProgressLinear/ProgressLinear.vue +75 -75
- package/src/components/Laboratory/SelectionColumnBar/SelectionColumnBar.vue +92 -92
- package/src/components/Laboratory/StatusNotification/StatusNotification.vue +49 -49
- package/src/components/Laboratory/TagLabel/TagLabel.vue +126 -126
- package/src/components/Laboratory/TagLabelGroup/TagLabelGroup.vue +97 -97
- package/src/components/Laboratory/TicketCard/TicketCard.css +3 -3
- package/src/components/Laboratory/TicketCard/TicketCard.vue +143 -143
- package/src/components/Laboratory/TimeLine/TimeLineEvent.css +18 -18
- package/src/components/Laboratory/TimeLine/TimeLineEvent.vue +119 -119
- package/src/components/Laboratory/TimeLine/Timeline.css +4 -4
- package/src/components/Laboratory/TimeLine/Timeline.vue +30 -30
- package/src/components/Loader/Loader.css +71 -71
- package/src/components/MaintenanceBanner/MaintenanceBanner.css +353 -353
- package/src/components/MaintenanceBanner/MaintenanceBanner.vue +127 -127
- package/src/components/MaintenanceBanner/MaintenanceIllustration.vue +54 -54
- package/src/components/Modal/Modal.css +5 -5
- package/src/components/Modal/Modal.vue +22 -22
- package/src/components/NotificationBubble/NotificationBubble.css +4 -4
- package/src/components/NotificationBubble/NotificationBubble.vue +90 -90
- package/src/components/OtpInput/OtpInput.css +39 -39
- package/src/components/OtpInput/OtpInput.vue +151 -151
- package/src/components/PhoneInput/PhoneInput.css +31 -31
- package/src/components/PhoneInput/PhoneInput.vue +113 -113
- package/src/components/Select/Select.css +150 -150
- package/src/components/Select/Select.vue +316 -316
- package/src/components/TextArea/TextArea.css +3 -3
- package/src/components/TextArea/TextArea.vue +126 -126
- package/src/components/TickBox/TickBox.css +49 -49
- package/src/components/TickBox/TickBox.vue +126 -126
- package/src/components/accessibility.css +218 -218
- package/src/components/index.ts +29 -29
- package/src/constants/iconEnums.ts +3 -3
- package/src/i18n/i18n.ts +15 -15
- package/src/i18n/locales/de.json +30 -30
- package/src/i18n/locales/en.json +30 -30
- package/src/index.ts +34 -34
- package/src/main.ts +11 -11
- package/src/plugins/vuetify.ts +141 -141
- package/src/shims-vue.d.ts +10 -10
- package/src/stories/Accordion.stories.ts +650 -650
- package/src/stories/Audio.stories.ts +28 -28
- package/src/stories/Button.stories.ts +263 -263
- package/src/stories/CheckBox.stories.ts +348 -348
- package/src/stories/DateInput.stories.ts +53 -53
- package/src/stories/Dialog.stories.ts +147 -147
- package/src/stories/EditField.stories.ts +78 -78
- package/src/stories/IconBullet/IconBullet.stories.ts +201 -201
- package/src/stories/IconBullet/IconBulletList.stories.ts +275 -275
- package/src/stories/Input.stories.ts +351 -351
- package/src/stories/Laboratory/Cards/AppointmentCard/AppointmentCard.stories.ts +260 -260
- package/src/stories/Laboratory/Cards/DocumentCard/DocumentCard.stories.ts +176 -176
- package/src/stories/Laboratory/Cards/DocumentCard/DocumentCardItem.stories.ts +119 -119
- package/src/stories/Laboratory/Cards/InfoCard/InfoCard.stories.ts +320 -320
- package/src/stories/Laboratory/Cards/TicketCard/TicketCard.stories.ts +335 -335
- package/src/stories/Laboratory/Chat/ChatBoxImage.stories.ts +82 -82
- package/src/stories/Laboratory/Chat/ChatMessage.stories.ts +198 -198
- package/src/stories/Laboratory/Chat/ChatMessageBadge.stories.ts +204 -204
- package/src/stories/Laboratory/Chat/ChatNotification.stories.ts +144 -144
- package/src/stories/Laboratory/Chat/ProgressLinear.stories.ts +186 -186
- package/src/stories/Laboratory/Chat/StatusNotification.stories.ts +111 -111
- package/src/stories/Laboratory/MainColumnsBar.stories.ts +48 -48
- package/src/stories/Laboratory/ProgressCircle.stories.ts +261 -261
- package/src/stories/Laboratory/SelectionColumnBar.stories.ts +234 -234
- package/src/stories/Laboratory/TagLabel.stories.ts +418 -418
- package/src/stories/Laboratory/TagLabelGroup.stories.ts +234 -234
- package/src/stories/Laboratory/Timeline.stories.ts +403 -403
- package/src/stories/NotificationBubble.stories.ts +194 -194
- package/src/stories/OtpInput.stories.ts +100 -100
- package/src/stories/PhoneInput.stories.ts +52 -52
- package/src/stories/Select.stories.ts +419 -419
- package/src/stories/TextArea.stories.ts +112 -112
- package/src/stories/TickBox.stories.ts +294 -294
- package/src/stories/v-icon.stories.ts +91 -91
- package/src/utils/index.ts +116 -109
- package/src/vite-env.d.ts +1 -1
- package/tests/e2e/README.md +220 -220
- package/tests/e2e/accessibility.spec.ts +638 -638
- package/tests/e2e/accordion.spec.ts +42 -42
- package/tests/e2e/additional-components.spec.ts +437 -437
- package/tests/e2e/all-components.spec.ts +135 -135
- package/tests/e2e/appointment-card.spec.ts +816 -816
- package/tests/e2e/button-fixed.spec.ts +58 -58
- package/tests/e2e/button.spec.ts +76 -76
- package/tests/e2e/checkbox.spec.ts +50 -50
- package/tests/e2e/date-input.spec.ts +46 -46
- package/tests/e2e/debug.spec.ts +51 -51
- package/tests/e2e/dialog.spec.ts +58 -58
- package/tests/e2e/input.spec.ts +55 -55
- package/tests/e2e/laboratory-components.spec.ts +320 -320
- package/tests/e2e/otp-input.spec.ts +50 -50
- package/tests/e2e/select.spec.ts +52 -52
- package/tests/e2e/storybook-utils.ts +59 -59
- package/tests/e2e/test-basic.spec.ts +33 -33
- package/tests/e2e/visual-regression.spec.ts +350 -350
- package/tests/unit/accessibility/component-a11y.spec.ts +469 -469
- package/tests/unit/components/Accordion/AccordionGroup.spec.ts +228 -228
- package/tests/unit/components/Accordion/AccordionGroup.spec.ts.skip +342 -342
- package/tests/unit/components/Accordion/AccordionItem.spec.ts +292 -292
- package/tests/unit/components/Accordion/AccordionItem.spec.ts.skip +383 -383
- package/tests/unit/components/Appointment/AnamneseNotification.spec.ts +176 -176
- package/tests/unit/components/Appointment/Card/Actions.spec.ts +407 -407
- package/tests/unit/components/Appointment/Card/Card.spec.ts +485 -485
- package/tests/unit/components/Appointment/Card/Details.spec.ts +397 -397
- package/tests/unit/components/Audio/Audio.spec.ts +403 -403
- package/tests/unit/components/Audio/Waveform.spec.ts +483 -483
- package/tests/unit/components/Background/Background.spec.ts +177 -177
- package/tests/unit/components/Core/Button.spec.ts +336 -336
- package/tests/unit/components/Core/Checkbox.spec.ts +544 -544
- package/tests/unit/components/Core/DateInput.spec.ts +690 -690
- package/tests/unit/components/Core/Dialog.spec.ts +485 -485
- package/tests/unit/components/Core/EditField.spec.ts +782 -782
- package/tests/unit/components/Core/Input.spec.ts +512 -512
- package/tests/unit/components/Core/Modal.spec.ts +518 -518
- package/tests/unit/components/Core/NotificationBubble.spec.ts +606 -606
- package/tests/unit/components/Core/OtpInput.spec.ts +708 -708
- package/tests/unit/components/Core/PhoneInput.spec.ts +619 -619
- package/tests/unit/components/Core/Select.spec.ts +712 -712
- package/tests/unit/components/Core/TextArea.spec.ts +565 -565
- package/tests/unit/components/Core/TickBox.spec.ts +779 -779
- package/tests/unit/components/ErrorPage/ErrorPage.spec.ts +313 -313
- package/tests/unit/components/ErrorPage/ErrorPageLogo.spec.ts +153 -153
- package/tests/unit/components/IconBullet/IconBullet.spec.ts +356 -356
- package/tests/unit/components/IconBullet/IconBulletList.spec.ts +371 -371
- package/tests/unit/components/Icons/AdvanceAppointments.spec.ts +61 -61
- package/tests/unit/components/Icons/Audio/CloudFailed.spec.ts +108 -108
- package/tests/unit/components/Icons/Audio/CloudSaved.spec.ts +149 -149
- package/tests/unit/components/Icons/Audio/Delete.spec.ts +158 -158
- package/tests/unit/components/Icons/Audio/Pause.spec.ts +208 -208
- package/tests/unit/components/Icons/Audio/Play.spec.ts +217 -217
- package/tests/unit/components/Icons/CalendarNotification.spec.ts +186 -186
- package/tests/unit/components/Icons/Chair.spec.ts +234 -234
- package/tests/unit/components/Icons/ChairNotification.spec.ts +311 -311
- package/tests/unit/components/Icons/Circle.spec.ts +255 -255
- package/tests/unit/components/Icons/FavIcon.spec.ts +251 -251
- package/tests/unit/components/Icons/FilledCircle.spec.ts +274 -274
- package/tests/unit/components/Icons/Group3.spec.ts +355 -355
- package/tests/unit/components/Icons/Logo.spec.ts +228 -228
- package/tests/unit/components/Icons/MiniLogo.spec.ts +38 -38
- package/tests/unit/components/Icons/RingNotification.spec.ts +393 -393
- package/tests/unit/components/Icons/SolidArrowRight.spec.ts +49 -49
- package/tests/unit/components/Icons/calendar.spec.ts +286 -286
- package/tests/unit/components/Icons/checkbox.spec.ts +315 -315
- package/tests/unit/components/Icons/outlineChecked.spec.ts +434 -434
- package/tests/unit/components/Icons/play.spec.ts +308 -308
- package/tests/unit/components/Laboratory/AppointmentCard.spec.ts +167 -167
- package/tests/unit/components/Laboratory/ChatBoxImage.spec.ts +179 -179
- package/tests/unit/components/Laboratory/ChatMessage.spec.ts +263 -263
- package/tests/unit/components/Laboratory/ChatMessageBadge.spec.ts +282 -282
- package/tests/unit/components/Laboratory/ChatNotification.spec.ts +256 -256
- package/tests/unit/components/Laboratory/DocumentCard.spec.ts +228 -228
- package/tests/unit/components/Laboratory/DocumentCardItem.spec.ts +236 -236
- package/tests/unit/components/Laboratory/InfoCard.spec.ts +308 -308
- package/tests/unit/components/Laboratory/MainColumnsBar.spec.ts +251 -251
- package/tests/unit/components/Laboratory/ProgressCircle.spec.ts +290 -290
- package/tests/unit/components/Laboratory/ProgressLinear.spec.ts +275 -275
- package/tests/unit/components/Laboratory/SelectionColumnBar.spec.ts +288 -288
- package/tests/unit/components/Laboratory/StatusNotification.spec.ts +296 -296
- package/tests/unit/components/Laboratory/TagLabel.spec.ts +353 -353
- package/tests/unit/components/Laboratory/TagLabelGroup.spec.ts +377 -377
- package/tests/unit/components/Laboratory/TicketCard.spec.ts +351 -351
- package/tests/unit/components/Laboratory/TimeLineEvent.spec.ts +381 -381
- package/tests/unit/components/Laboratory/Timeline.spec.ts +419 -419
- package/tests/unit/components/Loader/Loader.spec.ts +197 -197
- package/tests/unit/components/MaintenanceBanner/MaintenanceBanner.spec.ts +302 -302
- package/tests/unit/constants/iconEnums.spec.ts +39 -39
- package/tests/unit/i18n/i18n.spec.ts +88 -88
- package/tests/unit/plugins/vuetify.spec.ts +220 -220
- package/tests/unit/setup.ts +189 -189
- package/tests/unit/src/components/index.spec.ts.skip +192 -192
- package/tests/unit/src/index.spec.ts.skip +182 -182
- package/tests/unit/src/main.spec.ts +151 -151
- package/tests/unit/utils/accessibility.spec.ts +318 -318
- package/tsconfig.json +26 -26
- package/vite.config.ts +29 -29
- package/vitest.config.ts +83 -83
|
@@ -1,394 +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
|
-
});
|
|
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
394
|
});
|