@dhasdk/simple-ui 1.0.7 → 1.0.8
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/.babelrc +12 -0
- package/.storybook/main.ts +35 -0
- package/.storybook/preview.ts +4 -0
- package/BAKpostcss.config.jsBAK +15 -0
- package/BAKtailwind.config.mjsBAK +99 -0
- package/README.md +464 -16
- package/coverage/storybook/coverage-storybook.json +32411 -0
- package/coverage/storybook/lcov-report/Accordion.tsx.html +805 -0
- package/coverage/storybook/lcov-report/Badge.tsx.html +346 -0
- package/coverage/storybook/lcov-report/Breadcrumbs.tsx.html +742 -0
- package/coverage/storybook/lcov-report/Button.tsx.html +448 -0
- package/coverage/storybook/lcov-report/ButtonGroup.tsx.html +403 -0
- package/coverage/storybook/lcov-report/Card.tsx.html +292 -0
- package/coverage/storybook/lcov-report/CharacterCounter.tsx.html +253 -0
- package/coverage/storybook/lcov-report/CheckBox.tsx.html +1555 -0
- package/coverage/storybook/lcov-report/DatePicker.tsx.html +826 -0
- package/coverage/storybook/lcov-report/Input.tsx.html +1012 -0
- package/coverage/storybook/lcov-report/List.tsx.html +364 -0
- package/coverage/storybook/lcov-report/Modal.tsx.html +745 -0
- package/coverage/storybook/lcov-report/Pill.tsx.html +358 -0
- package/coverage/storybook/lcov-report/Search.tsx.html +997 -0
- package/coverage/storybook/lcov-report/SearchContent.tsx.html +235 -0
- package/coverage/storybook/lcov-report/SectionHeader.tsx.html +358 -0
- package/coverage/storybook/lcov-report/Select.tsx.html +1012 -0
- package/coverage/storybook/lcov-report/Shield.tsx.html +802 -0
- package/coverage/storybook/lcov-report/SideBarNav.tsx.html +490 -0
- package/coverage/storybook/lcov-report/Skeleton.tsx.html +394 -0
- package/coverage/storybook/lcov-report/Slider.tsx.html +385 -0
- package/coverage/storybook/lcov-report/Status.tsx.html +322 -0
- package/coverage/storybook/lcov-report/Tabs.tsx.html +610 -0
- package/coverage/storybook/lcov-report/Toggle.tsx.html +373 -0
- package/coverage/storybook/lcov-report/Tooltip.tsx.html +496 -0
- package/coverage/storybook/lcov-report/base.css +224 -0
- package/coverage/storybook/lcov-report/block-navigation.js +87 -0
- package/coverage/storybook/lcov-report/favicon.png +0 -0
- package/coverage/storybook/lcov-report/index.html +476 -0
- package/coverage/storybook/lcov-report/prettify.css +1 -0
- package/coverage/storybook/lcov-report/prettify.js +2 -0
- package/coverage/storybook/lcov-report/sort-arrow-sprite.png +0 -0
- package/coverage/storybook/lcov-report/sorter.js +196 -0
- package/coverage/storybook/lcov.info +2312 -0
- package/dist/README.md +1815 -0
- package/eslint.config.mjs +13 -0
- package/package.json +6 -7
- package/project.json +11 -0
- package/src/assets/img/Frame.svg +5 -0
- package/src/assets/img/backArrowRight.svg +10 -0
- package/src/assets/img/bc-separator.png +0 -0
- package/src/assets/img/calendar.png +0 -0
- package/src/assets/img/calendar.svg +4 -0
- package/src/assets/img/check.svg +5 -0
- package/src/assets/img/check_box.svg +10 -0
- package/src/assets/img/check_box_empty.svg +10 -0
- package/src/assets/img/check_box_fill.svg +10 -0
- package/src/assets/img/check_box_fill_empty.svg +10 -0
- package/src/assets/img/chevron-down-white.svg +2 -0
- package/src/assets/img/chevron-down.svg +2 -0
- package/src/assets/img/chevron-left.svg +1 -0
- package/src/assets/img/chevron-right-light.svg +4 -0
- package/src/assets/img/chevron-right.svg +3 -0
- package/src/assets/img/chevron-up-white.svg +1 -0
- package/src/assets/img/chevron-up.svg +1 -0
- package/src/assets/img/clock.svg +6 -0
- package/src/assets/img/close.svg +1 -0
- package/src/assets/img/close2.svg +6 -0
- package/src/assets/img/closeModal.svg +10 -0
- package/src/assets/img/close_icon_dark.svg +10 -0
- package/src/assets/img/close_small.svg +3 -0
- package/src/assets/img/emergency_home.svg +10 -0
- package/src/assets/img/first-aid-kit.svg +7 -0
- package/src/assets/img/heartbeat.svg +4 -0
- package/src/assets/img/home-gray.svg +3 -0
- package/src/assets/img/home.svg +3 -0
- package/src/assets/img/hospital.jpg +0 -0
- package/src/assets/img/indeterminate_check_box.svg +10 -0
- package/src/assets/img/indeterminate_check_box_fill.svg +10 -0
- package/src/assets/img/info_24_ 1d4ed8.svg +3 -0
- package/src/assets/img/info_24_ 2c6441.svg +3 -0
- package/src/assets/img/marker_check_by_default.svg +10 -0
- package/src/assets/img/marker_check_by_default_fill.svg +10 -0
- package/src/assets/img/minus-accordion.svg +5 -0
- package/src/assets/img/minus.svg +3 -0
- package/src/assets/img/open.svg +1 -0
- package/src/assets/img/pill-white.svg +7 -0
- package/src/assets/img/pill.svg +5 -0
- package/src/assets/img/plus-accordion.svg +5 -0
- package/src/assets/img/plus.svg +4 -0
- package/src/assets/img/prescription.svg +6 -0
- package/src/assets/img/search.svg +10 -0
- package/src/assets/img/search_icon_light.svg +10 -0
- package/src/assets/img/separator.svg +3 -0
- package/src/assets/img/stethoscope-white.svg +8 -0
- package/src/assets/img/stethoscope.svg +8 -0
- package/src/assets/img/thumb_up.svg +10 -0
- package/src/assets/img/vector.svg +3 -0
- package/src/assets/img/warning-badge-disabled.svg +11 -0
- package/src/assets/img/warning-badge-green.svg +11 -0
- package/src/assets/img/warning-badge-red.svg +11 -0
- package/src/assets/img/warning-badge-yellow.svg +11 -0
- package/src/assets/img/warning.svg +10 -0
- package/src/global.d.ts +13 -0
- package/{index.d.ts → src/index.ts} +13 -5
- package/src/lib/Accordian--Accordian.stories.tsx +312 -0
- package/src/lib/Accordion.spec.tsx +384 -0
- package/src/lib/Accordion.tsx +240 -0
- package/src/lib/AppointmentPicker.spec.tsx +138 -0
- package/src/lib/AppointmentPicker.tsx +97 -0
- package/src/lib/Badge--Badge.stories.tsx +60 -0
- package/src/lib/Badge.spec.tsx +70 -0
- package/src/lib/Badge.tsx +87 -0
- package/src/lib/Breadcrumbs-Breadcrumbs.stories.tsx +114 -0
- package/src/lib/Breadcrumbs.spec.tsx +218 -0
- package/src/lib/Breadcrumbs.tsx +219 -0
- package/src/lib/Button--Button.stories.tsx +220 -0
- package/src/lib/Button.spec.tsx +241 -0
- package/src/lib/Button.tsx +121 -0
- package/src/lib/ButtonGroup--ButtonGroup.stories.tsx +129 -0
- package/src/lib/ButtonGroup.spec.tsx +89 -0
- package/src/lib/ButtonGroup.tsx +107 -0
- package/src/lib/Card--Card.stories.tsx +113 -0
- package/src/lib/Card.spec.tsx +112 -0
- package/src/lib/Card.tsx +69 -0
- package/src/lib/CharacterCounter--CharacterCounter.stories.tsx +169 -0
- package/src/lib/CharacterCounter.spec.tsx +123 -0
- package/src/lib/CharacterCounter.tsx +56 -0
- package/src/lib/CheckBox--CheckBox.stories.tsx +107 -0
- package/src/lib/CheckBox.spec.tsx +412 -0
- package/src/lib/CheckBox.tsx +491 -0
- package/src/lib/DatePicker--DatePicker.stories.tsx +228 -0
- package/src/lib/DatePicker.spec.tsx +424 -0
- package/src/lib/DatePicker.tsx +247 -0
- package/src/lib/Input--Input.stories.tsx +449 -0
- package/src/lib/Input.spec.tsx +281 -0
- package/src/lib/Input.tsx +309 -0
- package/src/lib/List--List.stories.tsx +157 -0
- package/src/lib/List.spec.tsx +211 -0
- package/src/lib/List.tsx +93 -0
- package/src/lib/Modal--Modal.stories.tsx +454 -0
- package/src/lib/Modal.spec.tsx +202 -0
- package/src/lib/Modal.tsx +220 -0
- package/src/lib/Pill--Pill.stories.tsx +98 -0
- package/src/lib/Pill.spec.tsx +103 -0
- package/src/lib/Pill.tsx +91 -0
- package/src/lib/ProgressBar.spec.tsx +106 -0
- package/src/lib/ProgressBar.tsx +112 -0
- package/src/lib/RadioGroup.spec.tsx +84 -0
- package/src/lib/RadioGroup.tsx +74 -0
- package/src/lib/RadioIcon.tsx +13 -0
- package/src/lib/Search--Search.stories.tsx +67 -0
- package/src/lib/Search.spec.tsx +182 -0
- package/src/lib/Search.tsx +304 -0
- package/src/lib/SearchContent.tsx +51 -0
- package/src/lib/SectionHeader--SectionHeader.stories.tsx +98 -0
- package/src/lib/SectionHeader.spec.tsx +60 -0
- package/src/lib/SectionHeader.tsx +91 -0
- package/src/lib/Select--Select.stories.tsx +387 -0
- package/src/lib/Select.spec.tsx +493 -0
- package/src/lib/Select.tsx +311 -0
- package/src/lib/Shield--Shield.stories.tsx +196 -0
- package/src/lib/Shield.spec.tsx +275 -0
- package/src/lib/Shield.tsx +239 -0
- package/src/lib/SideBarNav--SideBarNav.stories.tsx +136 -0
- package/src/lib/SideBarNav.spec.tsx +178 -0
- package/src/lib/SideBarNav.tsx +135 -0
- package/src/lib/Skeleton--Skeleton.stories.tsx +77 -0
- package/src/lib/Skeleton.module.css +16 -0
- package/src/lib/Skeleton.spec.tsx +83 -0
- package/src/lib/Skeleton.tsx +103 -0
- package/src/lib/SkipLink.spec.tsx +76 -0
- package/src/lib/SkipLink.tsx +48 -0
- package/src/lib/Slider--Slider.stories.tsx +108 -0
- package/src/lib/Slider.module.css +109 -0
- package/src/lib/Slider.spec.tsx +67 -0
- package/src/lib/Slider.tsx +101 -0
- package/src/lib/Status--Status.stories.tsx +93 -0
- package/src/lib/Status.spec.tsx +118 -0
- package/src/lib/Status.tsx +79 -0
- package/src/lib/Tabs--Tabs.stories.tsx +294 -0
- package/src/lib/Tabs.spec.tsx +249 -0
- package/src/lib/Tabs.tsx +188 -0
- package/src/lib/Tester.spec.tsx +17 -0
- package/src/lib/Toggle--Toggle.stories.tsx +162 -0
- package/src/lib/Toggle.spec.tsx +122 -0
- package/src/lib/Toggle.tsx +96 -0
- package/src/lib/Tooltip--Tooltip.stories.tsx +315 -0
- package/src/lib/Tooltip.spec.tsx +307 -0
- package/src/lib/Tooltip.tsx +137 -0
- package/src/lib/bak-simple-ui.stories.tsx-bak +24 -0
- package/src/styles.css +190 -0
- package/tsconfig.json +25 -0
- package/tsconfig.lib.json +42 -0
- package/tsconfig.spec.json +29 -0
- package/tsconfig.storybook.json +36 -0
- package/vite.config.mts +87 -0
- package/vitest.setup.ts +12 -0
- package/index.css +0 -1
- package/index.js +0 -35
- package/index.mjs +0 -4981
- package/lib/Accordion.d.ts +0 -36
- package/lib/AppointmentPicker.d.ts +0 -21
- package/lib/Badge.d.ts +0 -11
- package/lib/Breadcrumbs.d.ts +0 -13
- package/lib/Button.d.ts +0 -15
- package/lib/ButtonGroup.d.ts +0 -8
- package/lib/Card.d.ts +0 -11
- package/lib/CharacterCounter.d.ts +0 -11
- package/lib/CheckBox.d.ts +0 -30
- package/lib/DatePicker.d.ts +0 -7
- package/lib/Input.d.ts +0 -16
- package/lib/List.d.ts +0 -22
- package/lib/Modal.d.ts +0 -18
- package/lib/Pill.d.ts +0 -13
- package/lib/ProgressBar.d.ts +0 -19
- package/lib/RadioGroup.d.ts +0 -15
- package/lib/Search.d.ts +0 -26
- package/lib/SearchContent.d.ts +0 -6
- package/lib/SectionHeader.d.ts +0 -18
- package/lib/Select.d.ts +0 -19
- package/lib/Shield.d.ts +0 -12
- package/lib/SideBarNav.d.ts +0 -21
- package/lib/Skeleton.d.ts +0 -15
- package/lib/SkipLink.d.ts +0 -22
- package/lib/Slider.d.ts +0 -14
- package/lib/Status.d.ts +0 -10
- package/lib/Tabs.d.ts +0 -23
- package/lib/Toggle.d.ts +0 -11
- package/lib/Tooltip.d.ts +0 -14
|
@@ -0,0 +1,220 @@
|
|
|
1
|
+
import React, { useEffect, useRef, useState } from "react";
|
|
2
|
+
import { twMerge } from "tailwind-merge";
|
|
3
|
+
import close from '../assets/img/closeModal.svg';
|
|
4
|
+
|
|
5
|
+
interface VariantType {
|
|
6
|
+
[key: string]: {
|
|
7
|
+
container: string;
|
|
8
|
+
title: string;
|
|
9
|
+
titleHeading: string;
|
|
10
|
+
body: string;
|
|
11
|
+
button: string;
|
|
12
|
+
};
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
// do not remove this, it is required to preload tailwind classes
|
|
16
|
+
const blurLevels = 'backdrop-blur-xs backdrop-blur-md backdrop-blur-lg backdrop-blur-xl backdrop-blur-2xl backdrop-blur-3xl';
|
|
17
|
+
|
|
18
|
+
const variants: VariantType = {
|
|
19
|
+
default: {
|
|
20
|
+
container: 'mx-8 md:mx-32 lg:mx-64 bg-white rounded-xs shadow-lg w-full max-w-lg px-2 pt-2 pb-4 md:pb-6 lg:pb-7 focus:outline-hidden ',
|
|
21
|
+
title: "flex justify-between items-center mb-4",
|
|
22
|
+
titleHeading: 'text-black',
|
|
23
|
+
body: '',
|
|
24
|
+
button: 'px-4 py-2 bg-gray-500 text-white rounded-xs hover:bg-gray-600 focus:outline-hidden focus:ring ring-gray-700',
|
|
25
|
+
},
|
|
26
|
+
darker: {
|
|
27
|
+
container: 'mx-8 md:mx-32 lg:mx-64 bg-slate-600 rounded-xs shadow-lg w-full max-w-lg px-2 pt-2 pb-4 md:pb-6 lg:pb-7 focus:outline-hidden text-slate-200 ',
|
|
28
|
+
title: "flex justify-between items-center mb-4",
|
|
29
|
+
titleHeading: 'text-slate-200',
|
|
30
|
+
body: '',
|
|
31
|
+
button: 'px-4 py-2 bg-slate-100 text-gray-700 rounded-xs hover:bg-slate-300 focus:outline-hidden focus:ring ring-gray-400',
|
|
32
|
+
},
|
|
33
|
+
dark: {
|
|
34
|
+
container: 'mx-8 md:mx-32 lg:mx-64 bg-zinc-800 rounded-xs shadow-lg w-full max-w-lg px-2 pt-2 pb-4 md:pb-6 lg:pb-7 focus:outline-hidden text-slate-200 ',
|
|
35
|
+
title: "flex justify-between items-center mb-4",
|
|
36
|
+
titleHeading: 'text-slate-200',
|
|
37
|
+
body: '',
|
|
38
|
+
button: 'px-4 py-2 bg-slate-200 text-gray-700 rounded-xs hover:bg-slate-400 focus:outline-hidden focus:ring ring-slate-500',
|
|
39
|
+
},
|
|
40
|
+
};
|
|
41
|
+
|
|
42
|
+
|
|
43
|
+
export interface AccessibleModalProps {
|
|
44
|
+
isOpen: boolean; // Controls modal visibility
|
|
45
|
+
variant?: string;
|
|
46
|
+
className?: string;
|
|
47
|
+
closeButton?: boolean;
|
|
48
|
+
blurLevel?: 'sm' | 'md' | 'lg' | 'xl' | '2xl' | '3xl';
|
|
49
|
+
onClose: () => void; // Close handler
|
|
50
|
+
title: string; // Modal title for screen readers
|
|
51
|
+
children: React.ReactNode; // Modal content
|
|
52
|
+
clickOutsideCloses?: boolean; // Close modal w/ click outside box? Default true
|
|
53
|
+
displayClosingX?: boolean; // Display closing 'x' in top right or not
|
|
54
|
+
closeButtonText?: string; // Text label for close button, default is "Close"
|
|
55
|
+
continueButton?: boolean; // Display a Continue button?
|
|
56
|
+
continueButtonText?: string; // Text label for continue button, default is "Continue"
|
|
57
|
+
continueButtonHandler?: () => void; // handler for Continue button
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
export const Modal = ({isOpen, variant = 'default', onClose, title, className,
|
|
61
|
+
closeButton = true, clickOutsideCloses = false, displayClosingX = true, closeButtonText='Close',
|
|
62
|
+
continueButton = false, continueButtonHandler, continueButtonText = 'Continue', blurLevel,
|
|
63
|
+
children}: AccessibleModalProps) => {
|
|
64
|
+
const modalRef = useRef<HTMLDivElement>(null);
|
|
65
|
+
const closeButtonRef = useRef<HTMLButtonElement>(null);
|
|
66
|
+
|
|
67
|
+
const defaultContainerClasses = 'fixed inset-0 z-50 flex items-center justify-center bg-black/50';
|
|
68
|
+
|
|
69
|
+
const [containerClass, setContainerClass] = useState<string>('');
|
|
70
|
+
|
|
71
|
+
useEffect( () => {
|
|
72
|
+
if (blurLevel) {
|
|
73
|
+
// console.log('blurLevel: ' + blurLevel);
|
|
74
|
+
const blur = 'backdrop-blur-' + blurLevel;
|
|
75
|
+
// console.log('blur: ' + blur);
|
|
76
|
+
setContainerClass(twMerge(defaultContainerClasses, blur));
|
|
77
|
+
// setContainerClass(twMerge(defaultContainerClasses, 'backdrop-blur-xs'));
|
|
78
|
+
// console.log('blur set');
|
|
79
|
+
}
|
|
80
|
+
else {
|
|
81
|
+
setContainerClass(defaultContainerClasses);
|
|
82
|
+
// console.log('no blur');
|
|
83
|
+
}
|
|
84
|
+
}, [blurLevel]);
|
|
85
|
+
|
|
86
|
+
// console.log('blurLevel: ' + blurLevel);
|
|
87
|
+
|
|
88
|
+
useEffect(() => {
|
|
89
|
+
if (clickOutsideCloses) {
|
|
90
|
+
const handleClickOutside = (event: MouseEvent) => {
|
|
91
|
+
if (modalRef.current && !modalRef.current.contains(event.target as Node)) {
|
|
92
|
+
onClose();
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
|
|
97
|
+
// attach event listener
|
|
98
|
+
document.addEventListener('mousedown', handleClickOutside);
|
|
99
|
+
|
|
100
|
+
return () => {
|
|
101
|
+
// cleanup listener on unmount
|
|
102
|
+
document.removeEventListener('mousedown', handleClickOutside);
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
}, [onClose]);
|
|
107
|
+
|
|
108
|
+
useEffect(() => {
|
|
109
|
+
const handleKeyDown = (event: KeyboardEvent) => {
|
|
110
|
+
if (event.key === "Escape" && isOpen) {
|
|
111
|
+
onClose();
|
|
112
|
+
}
|
|
113
|
+
};
|
|
114
|
+
|
|
115
|
+
const trapFocus = (event: KeyboardEvent) => {
|
|
116
|
+
const focusableElements = modalRef.current?.querySelectorAll<HTMLElement>(
|
|
117
|
+
'button, [href], input, select, textarea, [tabindex]:not([tabindex="-1"])'
|
|
118
|
+
);
|
|
119
|
+
const firstElement = focusableElements?.[0];
|
|
120
|
+
const lastElement = focusableElements?.[focusableElements.length - 1];
|
|
121
|
+
|
|
122
|
+
if (event.key === "Tab" && focusableElements) {
|
|
123
|
+
if (event.shiftKey && document.activeElement === firstElement) {
|
|
124
|
+
event.preventDefault();
|
|
125
|
+
lastElement?.focus();
|
|
126
|
+
} else if (!event.shiftKey && document.activeElement === lastElement) {
|
|
127
|
+
event.preventDefault();
|
|
128
|
+
firstElement?.focus();
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
};
|
|
132
|
+
|
|
133
|
+
if (isOpen) {
|
|
134
|
+
document.body.style.overflow = "hidden";
|
|
135
|
+
window.addEventListener("keydown", handleKeyDown);
|
|
136
|
+
window.addEventListener("keydown", trapFocus);
|
|
137
|
+
closeButtonRef.current?.focus();
|
|
138
|
+
} else {
|
|
139
|
+
document.body.style.overflow = "auto";
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
return () => {
|
|
143
|
+
window.removeEventListener("keydown", handleKeyDown);
|
|
144
|
+
window.removeEventListener("keydown", trapFocus);
|
|
145
|
+
document.body.style.overflow = "auto";
|
|
146
|
+
};
|
|
147
|
+
|
|
148
|
+
}, [isOpen, onClose]);
|
|
149
|
+
|
|
150
|
+
if (!isOpen) return null;
|
|
151
|
+
|
|
152
|
+
const continueHandler = () => {
|
|
153
|
+
if (continueButtonHandler) {
|
|
154
|
+
continueButtonHandler() // user supplied continue handler
|
|
155
|
+
}
|
|
156
|
+
onClose();
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
return (
|
|
160
|
+
<div
|
|
161
|
+
className={containerClass}
|
|
162
|
+
id="backdrop"
|
|
163
|
+
>
|
|
164
|
+
{/* Modal Content */}
|
|
165
|
+
<div
|
|
166
|
+
className={twMerge(variants[variant].container, className)}
|
|
167
|
+
ref={modalRef}
|
|
168
|
+
aria-label="modal-title"
|
|
169
|
+
role="dialog"
|
|
170
|
+
aria-modal="true"
|
|
171
|
+
|
|
172
|
+
>
|
|
173
|
+
{/* Modal Header */}
|
|
174
|
+
<div className="flex flex-col items-end">
|
|
175
|
+
{ displayClosingX &&
|
|
176
|
+
<button
|
|
177
|
+
// ref={closeButtonRef}
|
|
178
|
+
onClick={onClose}
|
|
179
|
+
className={twMerge("text-xl text-gray-500 hover:text-gray-700 focus:outline-hidden focus:ring",
|
|
180
|
+
variants[variant].titleHeading
|
|
181
|
+
)}
|
|
182
|
+
aria-label="Close modal"
|
|
183
|
+
>
|
|
184
|
+
<img src={close} alt="close icon"></img>
|
|
185
|
+
</button>
|
|
186
|
+
}
|
|
187
|
+
</div>
|
|
188
|
+
<div className={twMerge(variants[variant].title)}>
|
|
189
|
+
<h2 id="modal-title" className={twMerge("text-[20px] md:text-[24px] lg:text-[32px] font-medium text-gray-900 text-center w-full", variants[variant].titleHeading)}>
|
|
190
|
+
{title}
|
|
191
|
+
</h2>
|
|
192
|
+
</div>
|
|
193
|
+
|
|
194
|
+
{/* Modal Body */}
|
|
195
|
+
<div className={twMerge('text-center lg:text-[18px] px-2', variants[variant].body)}>{children}</div>
|
|
196
|
+
|
|
197
|
+
{/* Modal Footer */}
|
|
198
|
+
{continueButton || closeButton ?
|
|
199
|
+
<div className="mt-4 flex justify-end px-4">
|
|
200
|
+
{ closeButton &&
|
|
201
|
+
<button
|
|
202
|
+
ref={closeButtonRef}
|
|
203
|
+
onClick={onClose}
|
|
204
|
+
className={twMerge(variants[variant].button)}
|
|
205
|
+
>
|
|
206
|
+
{closeButtonText}
|
|
207
|
+
</button> }
|
|
208
|
+
{/* continueButton = false, continueButtonHandler, */}
|
|
209
|
+
{ continueButton &&
|
|
210
|
+
<button
|
|
211
|
+
onClick={continueHandler}
|
|
212
|
+
className={twMerge(variants[variant].button, 'ms-4')}
|
|
213
|
+
>{continueButtonText}</button>
|
|
214
|
+
}
|
|
215
|
+
</div>
|
|
216
|
+
: ''}
|
|
217
|
+
</div>
|
|
218
|
+
</div>
|
|
219
|
+
);
|
|
220
|
+
};
|
|
@@ -0,0 +1,98 @@
|
|
|
1
|
+
import { Meta} from '@storybook/react';
|
|
2
|
+
import { Pill } from './Pill';
|
|
3
|
+
|
|
4
|
+
// Import your images
|
|
5
|
+
import pill from '../assets/img/pill.svg';
|
|
6
|
+
import pillWhite from '../assets/img/pill-white.svg';
|
|
7
|
+
import stethoscope from '../assets/img/stethoscope.svg';
|
|
8
|
+
import stethoscopeWhite from '../assets/img/stethoscope-white.svg';
|
|
9
|
+
import { useState } from 'react';
|
|
10
|
+
// https://fonts.google.com/icons?icon.set=Material+Icons&icon.style=Filled
|
|
11
|
+
|
|
12
|
+
export default {
|
|
13
|
+
title: 'Components/Pill',
|
|
14
|
+
component: Pill,
|
|
15
|
+
args: {
|
|
16
|
+
iconLeft: <img src={pill} alt='pill icon' />,
|
|
17
|
+
iconRight: <img src={stethoscope} alt='stethoscope icon' />,
|
|
18
|
+
children: 'Pill',
|
|
19
|
+
},
|
|
20
|
+
parameters: {
|
|
21
|
+
layout: 'centered',
|
|
22
|
+
backgrounds: {
|
|
23
|
+
default: 'light',
|
|
24
|
+
values: [
|
|
25
|
+
{ name: 'white', value: '#ffffff' },
|
|
26
|
+
{ name: 'light', value: '#f0f0f0' },
|
|
27
|
+
],
|
|
28
|
+
},
|
|
29
|
+
},
|
|
30
|
+
} as Meta<typeof Pill>;
|
|
31
|
+
|
|
32
|
+
|
|
33
|
+
export const Default = {
|
|
34
|
+
args: { }
|
|
35
|
+
};
|
|
36
|
+
|
|
37
|
+
export const PillSelect = () => {
|
|
38
|
+
const [selected, setSelected] = useState<boolean>(false);
|
|
39
|
+
|
|
40
|
+
const onClick = () => {
|
|
41
|
+
// do something & set selected or !selected
|
|
42
|
+
setSelected(!selected);
|
|
43
|
+
};
|
|
44
|
+
|
|
45
|
+
return (
|
|
46
|
+
<>
|
|
47
|
+
<Pill
|
|
48
|
+
iconLeft={<img src={pill} alt='pill icon' />}
|
|
49
|
+
iconLeftSelected={<img src={pillWhite} alt='pill icon' />}
|
|
50
|
+
iconRight={<img src={stethoscope} alt='stethoscope icon' />}
|
|
51
|
+
iconRightSelected={<img src={stethoscopeWhite} alt='stethoscope icon' />}
|
|
52
|
+
selected={selected}
|
|
53
|
+
onClick={onClick}
|
|
54
|
+
>Pill</Pill>
|
|
55
|
+
<div>{selected ? 'selected' : 'not selected'}</div>
|
|
56
|
+
</>
|
|
57
|
+
);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
|
|
61
|
+
export const Disabled =() => {
|
|
62
|
+
return (
|
|
63
|
+
<Pill
|
|
64
|
+
disabled
|
|
65
|
+
>Pill</Pill>
|
|
66
|
+
);
|
|
67
|
+
};
|
|
68
|
+
|
|
69
|
+
export const IconLeft = () => {
|
|
70
|
+
return (
|
|
71
|
+
<Pill
|
|
72
|
+
iconLeft={<img src={pill} alt='pill icon' />}
|
|
73
|
+
selected={true}
|
|
74
|
+
>Pill</Pill>
|
|
75
|
+
);
|
|
76
|
+
};
|
|
77
|
+
|
|
78
|
+
export const IconRight = () => {
|
|
79
|
+
return (
|
|
80
|
+
<Pill
|
|
81
|
+
iconRight={<img src={pill} alt='pill icon' />}
|
|
82
|
+
>Pill</Pill>
|
|
83
|
+
);
|
|
84
|
+
};
|
|
85
|
+
|
|
86
|
+
|
|
87
|
+
export const NoIcons = () => {
|
|
88
|
+
return (
|
|
89
|
+
<Pill
|
|
90
|
+
iconLeft={null}
|
|
91
|
+
// iconLeftSelected={null}
|
|
92
|
+
iconRight={null}
|
|
93
|
+
// iconRightSelected={null}
|
|
94
|
+
>Pill</Pill>
|
|
95
|
+
);
|
|
96
|
+
};
|
|
97
|
+
|
|
98
|
+
|
|
@@ -0,0 +1,103 @@
|
|
|
1
|
+
import React, { createRef } from 'react';
|
|
2
|
+
import { render, screen, fireEvent } from '@testing-library/react';
|
|
3
|
+
import { Pill } from './Pill';
|
|
4
|
+
import { vi } from 'vitest';
|
|
5
|
+
import pill from '../assets/img/pill.svg';
|
|
6
|
+
describe('Pill Component', () => {
|
|
7
|
+
it('renders the Pill with children and default styles when no icons are provided', () => {
|
|
8
|
+
render(<Pill>Test Pill</Pill>);
|
|
9
|
+
const button = screen.getByRole('button');
|
|
10
|
+
expect(button).toBeInTheDocument();
|
|
11
|
+
expect(button).toHaveTextContent('Test Pill');
|
|
12
|
+
|
|
13
|
+
// When no icons are provided, the children container should get "ml-6 mr-6"
|
|
14
|
+
const childrenDiv = button.querySelector('div');
|
|
15
|
+
expect(childrenDiv?.className).toMatch(/ml-6 mr-6/);
|
|
16
|
+
});
|
|
17
|
+
|
|
18
|
+
it('calls onClick handler when clicked', () => {
|
|
19
|
+
const onClickMock = vi.fn();
|
|
20
|
+
render(<Pill onClick={onClickMock}>Clickable Pill</Pill>);
|
|
21
|
+
const button = screen.getByRole('button');
|
|
22
|
+
fireEvent.click(button);
|
|
23
|
+
expect(onClickMock).toHaveBeenCalled();
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
it('renders left icon and applies proper margin when only iconLeft is provided', () => {
|
|
27
|
+
render(
|
|
28
|
+
<Pill iconLeft={pill} data-testid='icon-left'>
|
|
29
|
+
Pill with Left Icon
|
|
30
|
+
</Pill>
|
|
31
|
+
);
|
|
32
|
+
expect(screen.getByTestId('icon-left')).toBeInTheDocument();
|
|
33
|
+
// When only iconLeft is provided, children container should have "mr-6"
|
|
34
|
+
const childrenDiv = screen.getByText('Pill with Left Icon');
|
|
35
|
+
expect(childrenDiv?.className).toMatch(/mr-6/);
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
it('renders right icon and applies proper margin when only iconRight is provided', () => {
|
|
39
|
+
render(
|
|
40
|
+
<Pill iconRight={pill} data-testid='icon-right'>
|
|
41
|
+
Pill with Right Icon
|
|
42
|
+
</Pill>
|
|
43
|
+
);
|
|
44
|
+
expect(screen.getByTestId('icon-right')).toBeInTheDocument();
|
|
45
|
+
// When only iconRight is provided, children container should have "ml-6"
|
|
46
|
+
const childrenDiv = screen.getByText('Pill with Right Icon');
|
|
47
|
+
expect(childrenDiv?.className).toMatch(/ml-6/);
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
it('renders both icons and does not add extra margin to children', () => {
|
|
51
|
+
render(
|
|
52
|
+
<Pill
|
|
53
|
+
iconLeft={<span data-testid="icon-left">{pill}</span>}
|
|
54
|
+
iconRight={<span data-testid="icon-right">{pill}</span>}
|
|
55
|
+
>
|
|
56
|
+
Pill with Both Icons
|
|
57
|
+
</Pill>
|
|
58
|
+
);
|
|
59
|
+
expect(screen.getByTestId('icon-left')).toBeInTheDocument();
|
|
60
|
+
expect(screen.getByTestId('icon-right')).toBeInTheDocument();
|
|
61
|
+
|
|
62
|
+
// When both icons are provided, the base children classes should be used without extra margin.
|
|
63
|
+
const childrenDiv = screen.getByText('Pill with Both Icons');
|
|
64
|
+
// The base childrenClasses is: "text-lg font-normal font-['Arial'] leading-normal "
|
|
65
|
+
// (without extra margin)
|
|
66
|
+
expect(childrenDiv?.className.trim()).toBe("text-lg font-normal font-['Arial'] leading-normal");
|
|
67
|
+
});
|
|
68
|
+
|
|
69
|
+
it('renders selected state with selected icons', () => {
|
|
70
|
+
render(
|
|
71
|
+
<Pill
|
|
72
|
+
selected
|
|
73
|
+
iconLeft={<span data-testid="icon-left">L</span>}
|
|
74
|
+
iconLeftSelected={<span data-testid="icon-left-selected">L-S</span>}
|
|
75
|
+
iconRight={<span data-testid="icon-right">R</span>}
|
|
76
|
+
iconRightSelected={<span data-testid="icon-right-selected">R-S</span>}
|
|
77
|
+
>
|
|
78
|
+
Selected Pill
|
|
79
|
+
</Pill>
|
|
80
|
+
);
|
|
81
|
+
|
|
82
|
+
// For selected state, the left and right icons should render the selected versions
|
|
83
|
+
expect(screen.getByTestId('icon-left-selected')).toBeInTheDocument();
|
|
84
|
+
expect(screen.getByTestId('icon-right-selected')).toBeInTheDocument();
|
|
85
|
+
|
|
86
|
+
// Additionally, check that the button's classes contain selected styles
|
|
87
|
+
const button = screen.getByRole('button');
|
|
88
|
+
expect(button.className).toMatch(/bg-\[#092068\]/);
|
|
89
|
+
expect(button.className).toMatch(/text-white/);
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
it('merges custom className with computed classes', () => {
|
|
93
|
+
render(<Pill className="custom-class">Custom Class Pill</Pill>);
|
|
94
|
+
const button = screen.getByRole('button');
|
|
95
|
+
expect(button.className).toMatch(/custom-class/);
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
it('forwards ref to the button element', () => {
|
|
99
|
+
const ref = createRef<HTMLButtonElement>();
|
|
100
|
+
render(<Pill ref={ref}>Ref Forwarding Pill</Pill>);
|
|
101
|
+
expect(ref.current).toBeInstanceOf(HTMLButtonElement);
|
|
102
|
+
});
|
|
103
|
+
});
|
package/src/lib/Pill.tsx
ADDED
|
@@ -0,0 +1,91 @@
|
|
|
1
|
+
import { forwardRef, ButtonHTMLAttributes, ReactNode, useEffect, useState } from 'react';
|
|
2
|
+
import { twMerge } from 'tailwind-merge';
|
|
3
|
+
|
|
4
|
+
|
|
5
|
+
export interface PillProps extends ButtonHTMLAttributes<HTMLButtonElement> {
|
|
6
|
+
className?: string;
|
|
7
|
+
iconLeft?: ReactNode;
|
|
8
|
+
iconLeftSelected?: ReactNode;
|
|
9
|
+
iconRight?: ReactNode
|
|
10
|
+
iconRightSelected?: ReactNode;
|
|
11
|
+
children?: ReactNode;
|
|
12
|
+
onClick?: () => void;
|
|
13
|
+
selected?: boolean;
|
|
14
|
+
disabled?: boolean;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
export const Pill = forwardRef<HTMLButtonElement, PillProps>(
|
|
18
|
+
({ className = '', iconLeft, iconLeftSelected,
|
|
19
|
+
iconRight, iconRightSelected, children,
|
|
20
|
+
onClick, selected = false, disabled= false, ...props }, ref) => {
|
|
21
|
+
|
|
22
|
+
const [classValue, setClassValue] = useState<string>();
|
|
23
|
+
const [childrenClasses, setChildrenClasses] = useState<string>();
|
|
24
|
+
const [pillSelectedClasses, setPillSelectedClasses] = useState<string>('bg-white');
|
|
25
|
+
|
|
26
|
+
useEffect(() => {
|
|
27
|
+
const baseClasses = "text-lg font-normal font-['Arial'] leading-normal "
|
|
28
|
+
if (iconLeft && iconRight) { // we have both iconLeft & iconRight
|
|
29
|
+
setChildrenClasses(baseClasses)
|
|
30
|
+
} else if (iconLeft) { // we have only iconLeft
|
|
31
|
+
setChildrenClasses(baseClasses + "mr-6");
|
|
32
|
+
} else if (iconRight) { // we have only iconRight
|
|
33
|
+
setChildrenClasses(baseClasses + "ml-6");
|
|
34
|
+
} else { // we have no icons
|
|
35
|
+
setChildrenClasses(baseClasses + "ml-6 mr-6");
|
|
36
|
+
}
|
|
37
|
+
}, [iconLeft, iconRight, children])
|
|
38
|
+
|
|
39
|
+
useEffect(() => {
|
|
40
|
+
|
|
41
|
+
if (selected) {
|
|
42
|
+
setPillSelectedClasses('bg-[#092068] focus:bg-[#092068] text-white');
|
|
43
|
+
} else {
|
|
44
|
+
setPillSelectedClasses('bg-white');
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
// 'pressed' bg-[#9FC5F0], place inside onClick
|
|
48
|
+
setClassValue(twMerge('relative inline-flex items-center justify-center ' +
|
|
49
|
+
'whitespace-nowrap rounded-3xl ' +
|
|
50
|
+
'transition-colors focus-visible:outline-hidden font-[`Arial`] ' +
|
|
51
|
+
'disabled:pointer-events-none ' +
|
|
52
|
+
'disabled:opacity-50 ' +
|
|
53
|
+
'border-2 border-[#092068] text-[#092068] text-lg ' +
|
|
54
|
+
'focus:shadow-[0px_0px_0px_3px_rgba(251,137,241,1.00)] ' +
|
|
55
|
+
|
|
56
|
+
// active psuedo class applies to click, i.e. 'pressed' in figma
|
|
57
|
+
'bg-white hover:bg-[#D1DBFB] active:bg-[#9fc5f0] focus:bg-white disabled:bg-[#939194] ' +
|
|
58
|
+
|
|
59
|
+
// 'hover:text-white ' +
|
|
60
|
+
'disabled:bg-dha-mc-bottom-nav-background ' +
|
|
61
|
+
'disabled:text-dha-mc-checkbox-inactive ' +
|
|
62
|
+
'focus:border-black ' +
|
|
63
|
+
'disabled:border-dha-mc-secondary-border disabled:border-2 h-[48px] mt-1'
|
|
64
|
+
, pillSelectedClasses, className));
|
|
65
|
+
// console.log('pillSelectedClasses :', pillSelectedClasses);
|
|
66
|
+
}, [className, pillSelectedClasses, selected])
|
|
67
|
+
|
|
68
|
+
return (
|
|
69
|
+
<button
|
|
70
|
+
// type='button' // default to 'button'
|
|
71
|
+
onClick={onClick}
|
|
72
|
+
className={classValue}
|
|
73
|
+
ref={ref}
|
|
74
|
+
{...props}
|
|
75
|
+
disabled={disabled}
|
|
76
|
+
>
|
|
77
|
+
{/* Conditionally render icon on the left or right based on iconPosition */}
|
|
78
|
+
{/* ms/e-3 === 12px */}
|
|
79
|
+
{iconLeft && (
|
|
80
|
+
<span className="ml-6 mr-2.5 size-6">{selected && iconLeftSelected ? iconLeftSelected : iconLeft}</span>
|
|
81
|
+
)}
|
|
82
|
+
|
|
83
|
+
<div className={childrenClasses}>{children}</div>
|
|
84
|
+
|
|
85
|
+
{iconRight && (
|
|
86
|
+
<span className="ml-2.5 mr-6 size-6">{selected && iconRightSelected ? iconRightSelected :
|
|
87
|
+
iconRight}</span>
|
|
88
|
+
)}
|
|
89
|
+
</button>
|
|
90
|
+
);
|
|
91
|
+
});
|
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
import React from "react";
|
|
2
|
+
import { render, screen } from "@testing-library/react";
|
|
3
|
+
import { axe } from "vitest-axe";
|
|
4
|
+
import ProgressBar from "./ProgressBar";
|
|
5
|
+
|
|
6
|
+
describe("ProgressBar Component", () => {
|
|
7
|
+
it("renders the correct number of steps", () => {
|
|
8
|
+
const totalSteps = 5;
|
|
9
|
+
const currentStep = 3;
|
|
10
|
+
const { container } = render(
|
|
11
|
+
<ProgressBar totalSteps={totalSteps} currentStep={currentStep} />
|
|
12
|
+
);
|
|
13
|
+
const circles = container.getElementsByClassName("rounded-full");
|
|
14
|
+
expect(circles).toHaveLength(totalSteps);
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
it("applies default styling to completed and upcoming steps", () => {
|
|
18
|
+
const totalSteps = 3;
|
|
19
|
+
const currentStep = 2;
|
|
20
|
+
const { container } = render(
|
|
21
|
+
<ProgressBar totalSteps={totalSteps} currentStep={currentStep} />
|
|
22
|
+
);
|
|
23
|
+
const circles = container.getElementsByClassName("rounded-full");
|
|
24
|
+
// Only the first step (step 1) should be marked as completed
|
|
25
|
+
expect(circles[0]).toHaveClass("bg-green-500", "border-[#305B25]");
|
|
26
|
+
expect(circles[1]).not.toHaveClass("bg-green-500");
|
|
27
|
+
expect(circles[2]).not.toHaveClass("bg-green-500");
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
it("respects custom className props", () => {
|
|
31
|
+
const totalSteps = 2;
|
|
32
|
+
const currentStep = 2;
|
|
33
|
+
const customFill = "bg-red-500";
|
|
34
|
+
const customBorder = "border-blue-500";
|
|
35
|
+
const { container } = render(
|
|
36
|
+
<ProgressBar
|
|
37
|
+
totalSteps={totalSteps}
|
|
38
|
+
currentStep={currentStep}
|
|
39
|
+
classNameFillColor={customFill}
|
|
40
|
+
classNameBorderColor={customBorder}
|
|
41
|
+
classNameGradient="from-red to-blue"
|
|
42
|
+
classNameArrowColor="fill-yellow-500"
|
|
43
|
+
/>
|
|
44
|
+
);
|
|
45
|
+
const circles = container.getElementsByClassName("rounded-full");
|
|
46
|
+
expect(circles[0]).toHaveClass(customFill, customBorder);
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
it("renders step numbers when isStep is true", () => {
|
|
50
|
+
const totalSteps = 3;
|
|
51
|
+
const currentStep = 1;
|
|
52
|
+
render(
|
|
53
|
+
<ProgressBar totalSteps={totalSteps} currentStep={currentStep} isStep />
|
|
54
|
+
);
|
|
55
|
+
for (let i = 1; i <= totalSteps; i++) {
|
|
56
|
+
expect(screen.getByText(`${i}`)).toBeInTheDocument();
|
|
57
|
+
}
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
it("renders check icons when isStep is false", () => {
|
|
61
|
+
const totalSteps = 1;
|
|
62
|
+
const currentStep = 1;
|
|
63
|
+
const { container } = render(
|
|
64
|
+
<ProgressBar
|
|
65
|
+
totalSteps={totalSteps}
|
|
66
|
+
currentStep={currentStep}
|
|
67
|
+
isStep={false}
|
|
68
|
+
/>
|
|
69
|
+
);
|
|
70
|
+
const svgs = container.getElementsByTagName("svg");
|
|
71
|
+
expect(svgs).toHaveLength(1);
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
it("renders check icons when isStep is false and uses correct styles when at least one step is completed", () => {
|
|
75
|
+
const totalSteps = 5;
|
|
76
|
+
const currentStep = 3;
|
|
77
|
+
const { container } = render(
|
|
78
|
+
<ProgressBar
|
|
79
|
+
totalSteps={totalSteps}
|
|
80
|
+
currentStep={currentStep}
|
|
81
|
+
isStep={false}
|
|
82
|
+
/>
|
|
83
|
+
);
|
|
84
|
+
const svgs = container.getElementsByTagName("svg");
|
|
85
|
+
expect(svgs[5]).toHaveAttribute('fill', 'none')
|
|
86
|
+
expect(svgs[1]).toHaveClass('fill-white')
|
|
87
|
+
});
|
|
88
|
+
});
|
|
89
|
+
|
|
90
|
+
describe("ProgressBar Accessibility Tests", () => {
|
|
91
|
+
it("should have no accessibility violations with default props", async () => {
|
|
92
|
+
const { container } = render(
|
|
93
|
+
<ProgressBar totalSteps={3} currentStep={2} />
|
|
94
|
+
);
|
|
95
|
+
const results = await axe(container);
|
|
96
|
+
expect(results).toHaveNoViolations();
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
it("should have no accessibility violations when there are no steps", async () => {
|
|
100
|
+
const { container } = render(
|
|
101
|
+
<ProgressBar totalSteps={0} currentStep={0} />
|
|
102
|
+
);
|
|
103
|
+
const results = await axe(container);
|
|
104
|
+
expect(results).toHaveNoViolations();
|
|
105
|
+
});
|
|
106
|
+
});
|