@pattern-stack/frontend-patterns 0.0.3 → 0.0.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (154) hide show
  1. package/dist/index.es.js +1 -1
  2. package/dist/index.js +1 -0
  3. package/package.json +5 -3
  4. package/src/App.css +42 -0
  5. package/src/App.tsx +54 -0
  6. package/src/__tests__/README.md +221 -0
  7. package/src/__tests__/atoms/hooks/simple-hooks.test.ts +44 -0
  8. package/src/__tests__/atoms/ui/button.test.tsx +68 -0
  9. package/src/__tests__/atoms/utils/simple.test.ts +18 -0
  10. package/src/__tests__/atoms/utils/utils.test.ts +77 -0
  11. package/src/__tests__/features/auth/simple-auth.test.tsx +40 -0
  12. package/src/__tests__/molecules/layout/simple-layout.test.tsx +81 -0
  13. package/src/__tests__/organisms/showcase/simple-showcase.test.tsx +167 -0
  14. package/src/__tests__/setup.ts +51 -0
  15. package/src/__tests__/utils.tsx +123 -0
  16. package/src/atoms/composed/Accordion/Accordion.tsx +271 -0
  17. package/src/atoms/composed/Accordion/index.ts +1 -0
  18. package/src/atoms/composed/Alert/Alert.tsx +132 -0
  19. package/src/atoms/composed/Alert/index.ts +1 -0
  20. package/src/atoms/composed/Breadcrumb/Breadcrumb.tsx +83 -0
  21. package/src/atoms/composed/Breadcrumb/index.ts +1 -0
  22. package/src/atoms/composed/Chart/Chart.tsx +425 -0
  23. package/src/atoms/composed/Chart/index.ts +2 -0
  24. package/src/atoms/composed/ColorSwatch/ColorSwatch.tsx +72 -0
  25. package/src/atoms/composed/ColorSwatch/index.ts +1 -0
  26. package/src/atoms/composed/DarkModeToggle.tsx +66 -0
  27. package/src/atoms/composed/DataBadge/DataBadge.tsx +81 -0
  28. package/src/atoms/composed/DataBadge/index.ts +1 -0
  29. package/src/atoms/composed/DataTable/DataTable.tsx +394 -0
  30. package/src/atoms/composed/DataTable/TableCellWithTooltip.tsx +41 -0
  31. package/src/atoms/composed/DataTable/index.ts +2 -0
  32. package/src/atoms/composed/DateTimePicker/DateTimePicker.tsx +611 -0
  33. package/src/atoms/composed/DateTimePicker/index.ts +2 -0
  34. package/src/atoms/composed/DetailedCard/DetailedCard.tsx +181 -0
  35. package/src/atoms/composed/DetailedCard/index.ts +2 -0
  36. package/src/atoms/composed/EmptyState/EmptyState.tsx +90 -0
  37. package/src/atoms/composed/EmptyState/index.ts +1 -0
  38. package/src/atoms/composed/FileUpload/FileUpload.tsx +477 -0
  39. package/src/atoms/composed/FileUpload/index.ts +2 -0
  40. package/src/atoms/composed/FormField/FormField.tsx +92 -0
  41. package/src/atoms/composed/FormField/index.ts +1 -0
  42. package/src/atoms/composed/GlobalSearch/GlobalSearch.tsx +37 -0
  43. package/src/atoms/composed/GlobalSearch/index.ts +1 -0
  44. package/src/atoms/composed/IconBadge/IconBadge.tsx +95 -0
  45. package/src/atoms/composed/IconBadge/index.ts +2 -0
  46. package/src/atoms/composed/Modal/Modal.tsx +223 -0
  47. package/src/atoms/composed/Modal/index.ts +2 -0
  48. package/src/atoms/composed/PaletteSwitcher.tsx +386 -0
  49. package/src/atoms/composed/ProgressBar/ProgressBar.tsx +116 -0
  50. package/src/atoms/composed/ProgressBar/index.ts +1 -0
  51. package/src/atoms/composed/StatCard/StatCard.tsx +219 -0
  52. package/src/atoms/composed/StatCard/index.ts +1 -0
  53. package/src/atoms/composed/StyleGuide.tsx +717 -0
  54. package/src/atoms/composed/Toast/Toast.tsx +219 -0
  55. package/src/atoms/composed/Toast/index.ts +1 -0
  56. package/src/atoms/composed/Tooltip/Tooltip.tsx +213 -0
  57. package/src/atoms/composed/Tooltip/index.ts +1 -0
  58. package/src/atoms/composed/UserAvatar/UserAvatar.tsx +139 -0
  59. package/src/atoms/composed/UserAvatar/index.ts +1 -0
  60. package/src/atoms/composed/UserMenu/UserMenu.tsx +16 -0
  61. package/src/atoms/composed/UserMenu/index.ts +1 -0
  62. package/src/atoms/composed/index.ts +29 -0
  63. package/src/atoms/hooks/useApi.ts +80 -0
  64. package/src/atoms/hooks/useHealth.ts +17 -0
  65. package/src/atoms/index.ts +13 -0
  66. package/src/atoms/services/api/client.ts +134 -0
  67. package/src/atoms/services/auth-service.ts +248 -0
  68. package/src/atoms/services/health.ts +15 -0
  69. package/src/atoms/services/index.ts +3 -0
  70. package/src/atoms/shared/config/constants.ts +17 -0
  71. package/src/atoms/shared/config/dashboard-sizes.ts +111 -0
  72. package/src/atoms/shared/config/environment.ts +10 -0
  73. package/src/atoms/shared/index.ts +4 -0
  74. package/src/atoms/shared/styles/color-palettes.css +566 -0
  75. package/src/atoms/types/auth.ts +62 -0
  76. package/src/atoms/types/generated.ts +1469 -0
  77. package/src/atoms/types/index.ts +4 -0
  78. package/src/atoms/types/loading.ts +28 -0
  79. package/src/atoms/ui/Badge.tsx +30 -0
  80. package/src/atoms/ui/ErrorBoundary.tsx +59 -0
  81. package/src/atoms/ui/Select.tsx +53 -0
  82. package/src/atoms/ui/Switch.tsx +42 -0
  83. package/src/atoms/ui/Tabs.tsx +118 -0
  84. package/src/atoms/ui/avatar.tsx +48 -0
  85. package/src/atoms/ui/button.tsx +70 -0
  86. package/src/atoms/ui/card.tsx +76 -0
  87. package/src/atoms/ui/dropdown-menu.tsx +199 -0
  88. package/src/atoms/ui/index.ts +39 -0
  89. package/src/atoms/ui/input.tsx +23 -0
  90. package/src/atoms/ui/label.tsx +23 -0
  91. package/src/atoms/ui/skeleton.tsx +13 -0
  92. package/src/atoms/ui/spinner.tsx +49 -0
  93. package/src/atoms/ui/table.tsx +116 -0
  94. package/src/atoms/utils/animations.ts +135 -0
  95. package/src/atoms/utils/tooltip-helpers.ts +140 -0
  96. package/src/atoms/utils/utils.ts +9 -0
  97. package/src/features/auth/components/LoginForm.tsx +168 -0
  98. package/src/features/auth/components/LogoutButton.tsx +19 -0
  99. package/src/features/auth/components/ProtectedRoute.tsx +60 -0
  100. package/src/features/auth/components/index.ts +4 -0
  101. package/src/features/auth/hooks/index.ts +2 -0
  102. package/src/features/auth/hooks/useAuth.tsx +205 -0
  103. package/src/features/auth/hooks/usePermissions.ts +35 -0
  104. package/src/features/auth/index.ts +2 -0
  105. package/src/features/index.ts +2 -0
  106. package/src/index.css +704 -0
  107. package/src/index.ts +13 -0
  108. package/src/main.tsx +48 -0
  109. package/src/molecules/.gitkeep +0 -0
  110. package/src/molecules/forms/FormGroup.tsx +75 -0
  111. package/src/molecules/forms/SearchInput.tsx +259 -0
  112. package/src/molecules/forms/index.ts +4 -0
  113. package/src/molecules/index.ts +4 -0
  114. package/src/molecules/layout/AppHeader/AppHeader.tsx +42 -0
  115. package/src/molecules/layout/AppHeader/index.ts +1 -0
  116. package/src/molecules/layout/AppLayout.tsx +29 -0
  117. package/src/molecules/layout/PageTemplate.tsx +87 -0
  118. package/src/molecules/layout/SectionHeader/SectionHeader.tsx +87 -0
  119. package/src/molecules/layout/SectionHeader/index.ts +1 -0
  120. package/src/molecules/layout/ShowcaseSection.tsx +57 -0
  121. package/src/molecules/layout/Sidebar.tsx +144 -0
  122. package/src/molecules/layout/SidebarButton/SidebarButton.tsx +99 -0
  123. package/src/molecules/layout/SidebarButton/index.ts +1 -0
  124. package/src/molecules/layout/SidebarContext.tsx +31 -0
  125. package/src/molecules/layout/index.ts +7 -0
  126. package/src/molecules/navigation/NavMenu.tsx +188 -0
  127. package/src/molecules/navigation/Pagination.tsx +172 -0
  128. package/src/molecules/navigation/index.ts +4 -0
  129. package/src/organisms/index.ts +5 -0
  130. package/src/organisms/showcase/ComponentShowcasePage.tsx +2496 -0
  131. package/src/organisms/showcase/index.ts +1 -0
  132. package/src/pages/AdminShowcase/AdminCRUDShowcase.tsx +242 -0
  133. package/src/pages/AdminShowcase/AdminDashboardShowcase.tsx +171 -0
  134. package/src/pages/AdminShowcase/AdminDetailShowcase.tsx +385 -0
  135. package/src/pages/AdminShowcase/index.tsx +3 -0
  136. package/src/pages/ComponentShowcase/BadgesShowcase.tsx +188 -0
  137. package/src/pages/ComponentShowcase/CardsShowcase.tsx +392 -0
  138. package/src/pages/ComponentShowcase/PalettesShowcase.tsx +207 -0
  139. package/src/pages/ComponentShowcase/StatesShowcase.tsx +485 -0
  140. package/src/pages/ComponentShowcase/TablesShowcase.tsx +134 -0
  141. package/src/pages/ComponentShowcase/TypographyShowcase.tsx +255 -0
  142. package/src/pages/ComponentShowcase/index.tsx +188 -0
  143. package/src/pages/index.ts +2 -0
  144. package/src/templates/AuthTemplate.tsx +216 -0
  145. package/src/templates/ComponentShowcaseTemplate.tsx +173 -0
  146. package/src/templates/DashboardTemplate.tsx +232 -0
  147. package/src/templates/DataTemplate.tsx +319 -0
  148. package/src/templates/admin/AdminCRUDTemplate.tsx +630 -0
  149. package/src/templates/admin/AdminDashboardTemplate.tsx +351 -0
  150. package/src/templates/admin/AdminDetailTemplate.tsx +563 -0
  151. package/src/templates/admin/index.ts +29 -0
  152. package/src/templates/factory.tsx +169 -0
  153. package/src/templates/index.ts +37 -0
  154. package/src/vite-env.d.ts +1 -0
@@ -0,0 +1,95 @@
1
+ import React from 'react';
2
+ import { cn } from '../../utils/utils';
3
+ import { Tooltip } from '../Tooltip';
4
+ import { getAnimationClasses, animationPresets } from '../../utils/animations';
5
+
6
+ export interface IconBadgeProps {
7
+ children?: React.ReactNode;
8
+ icon?: React.ReactNode;
9
+ variant?: 'category' | 'status';
10
+ category?: 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8;
11
+ status?: 'success' | 'warning' | 'error' | 'info' | 'neutral';
12
+ size?: 'sm' | 'md' | 'lg';
13
+ interactive?: boolean;
14
+ onClick?: () => void;
15
+ className?: string;
16
+ /** Tooltip content to show on hover */
17
+ tooltip?: string;
18
+ }
19
+
20
+ export const IconBadge = ({
21
+ children,
22
+ icon,
23
+ variant = 'category',
24
+ category = 1,
25
+ status = 'neutral',
26
+ size = 'md',
27
+ interactive = false,
28
+ onClick,
29
+ className,
30
+ tooltip
31
+ }: IconBadgeProps) => {
32
+
33
+ const sizeClasses: Record<'sm' | 'md' | 'lg', string> = {
34
+ sm: 'w-8 h-8 text-xs rounded-lg',
35
+ md: 'w-10 h-10 text-sm rounded-xl',
36
+ lg: 'w-12 h-12 text-base rounded-xl',
37
+ };
38
+
39
+ const baseClasses = cn(
40
+ 'inline-flex items-center justify-center font-bold shadow-md',
41
+ sizeClasses[size],
42
+ interactive && [
43
+ 'cursor-pointer',
44
+ getAnimationClasses({
45
+ ...animationPresets.dataBadge,
46
+ size: size === 'lg' ? 'lg' : size === 'sm' ? 'sm' : 'md'
47
+ })
48
+ ],
49
+ className
50
+ );
51
+
52
+ // Generate gradient classes based on variant
53
+ const gradientClasses = variant === 'category'
54
+ ? `bg-gradient-to-br from-category-${category} to-category-${Math.min(category + 1, 8)}`
55
+ : variant === 'status'
56
+ ? `bg-gradient-to-br from-status-${status} to-status-${status}`
57
+ : 'bg-gradient-to-br from-category-1 to-category-2';
58
+
59
+ const badge = (
60
+ <div
61
+ className={cn(baseClasses, gradientClasses, 'animate-fade-in')}
62
+ onClick={onClick}
63
+ role={interactive ? 'button' : undefined}
64
+ tabIndex={interactive ? 0 : undefined}
65
+ onKeyDown={interactive ? (e) => {
66
+ if (e.key === 'Enter' || e.key === ' ') {
67
+ e.preventDefault();
68
+ onClick?.();
69
+ }
70
+ } : undefined}
71
+ data-component-name="IconBadge"
72
+ >
73
+ {icon && (
74
+ <span className="text-primary-foreground drop-shadow">
75
+ {icon}
76
+ </span>
77
+ )}
78
+ {children && !icon && (
79
+ <span className="text-primary-foreground font-bold">
80
+ {children}
81
+ </span>
82
+ )}
83
+ </div>
84
+ );
85
+
86
+ if (tooltip) {
87
+ return (
88
+ <Tooltip content={tooltip} position="top" size="sm">
89
+ {badge}
90
+ </Tooltip>
91
+ );
92
+ }
93
+
94
+ return badge;
95
+ };
@@ -0,0 +1,2 @@
1
+ export { IconBadge } from './IconBadge';
2
+ export type { IconBadgeProps } from './IconBadge';
@@ -0,0 +1,223 @@
1
+ import React, { useEffect, useRef } from 'react';
2
+ import { createPortal } from 'react-dom';
3
+ import { Button } from '../../ui/button';
4
+ import { X } from 'lucide-react';
5
+ import { cn } from '../../utils/utils';
6
+
7
+ export interface ModalProps {
8
+ isOpen: boolean;
9
+ onClose: () => void;
10
+ title?: string;
11
+ children: React.ReactNode;
12
+ size?: 'sm' | 'md' | 'lg' | 'xl';
13
+ variant?: 'default' | 'destructive' | 'success';
14
+ closeOnBackdrop?: boolean;
15
+ closeOnEscape?: boolean;
16
+ showCloseButton?: boolean;
17
+ footer?: React.ReactNode;
18
+ className?: string;
19
+ overlayClassName?: string;
20
+ contentClassName?: string;
21
+ }
22
+
23
+ const sizeClasses = {
24
+ sm: 'max-w-sm',
25
+ md: 'max-w-md',
26
+ lg: 'max-w-lg',
27
+ xl: 'max-w-xl'
28
+ };
29
+
30
+ const variantClasses = {
31
+ default: 'border-border',
32
+ destructive: 'border-l-4 border-l-status-error',
33
+ success: 'border-l-4 border-l-status-success'
34
+ };
35
+
36
+ export const Modal = ({
37
+ isOpen,
38
+ onClose,
39
+ title,
40
+ children,
41
+ size = 'md',
42
+ variant = 'default',
43
+ closeOnBackdrop = true,
44
+ closeOnEscape = true,
45
+ showCloseButton = true,
46
+ footer,
47
+ className,
48
+ overlayClassName,
49
+ contentClassName
50
+ }: ModalProps) => {
51
+ const modalRef = useRef<HTMLDivElement>(null);
52
+ const previousActiveElement = useRef<HTMLElement | null>(null);
53
+
54
+ // Handle escape key
55
+ useEffect(() => {
56
+ if (!isOpen || !closeOnEscape) return;
57
+
58
+ const handleEscape = (e: KeyboardEvent) => {
59
+ if (e.key === 'Escape') {
60
+ onClose();
61
+ }
62
+ };
63
+
64
+ document.addEventListener('keydown', handleEscape);
65
+ return () => document.removeEventListener('keydown', handleEscape);
66
+ }, [isOpen, closeOnEscape, onClose]);
67
+
68
+ // Handle focus management
69
+ useEffect(() => {
70
+ if (!isOpen) return;
71
+
72
+ // Store the currently focused element
73
+ previousActiveElement.current = document.activeElement as HTMLElement;
74
+
75
+ // Focus the modal
76
+ if (modalRef.current) {
77
+ modalRef.current.focus();
78
+ }
79
+
80
+ // Restore focus when modal closes
81
+ return () => {
82
+ if (previousActiveElement.current) {
83
+ previousActiveElement.current.focus();
84
+ }
85
+ };
86
+ }, [isOpen]);
87
+
88
+ // Prevent body scroll when modal is open
89
+ useEffect(() => {
90
+ if (isOpen) {
91
+ document.body.style.overflow = 'hidden';
92
+ } else {
93
+ document.body.style.overflow = '';
94
+ }
95
+
96
+ return () => {
97
+ document.body.style.overflow = '';
98
+ };
99
+ }, [isOpen]);
100
+
101
+ // Handle backdrop click
102
+ const handleBackdropClick = (e: React.MouseEvent) => {
103
+ if (closeOnBackdrop && e.target === e.currentTarget) {
104
+ onClose();
105
+ }
106
+ };
107
+
108
+ // Trap focus within modal
109
+ const handleKeyDown = (e: React.KeyboardEvent) => {
110
+ if (e.key === 'Tab') {
111
+ const focusableElements = modalRef.current?.querySelectorAll(
112
+ 'button, [href], input, select, textarea, [tabindex]:not([tabindex="-1"])'
113
+ );
114
+
115
+ if (!focusableElements?.length) return;
116
+
117
+ const firstElement = focusableElements[0] as HTMLElement;
118
+ const lastElement = focusableElements[focusableElements.length - 1] as HTMLElement;
119
+
120
+ if (e.shiftKey) {
121
+ if (document.activeElement === firstElement) {
122
+ e.preventDefault();
123
+ lastElement.focus();
124
+ }
125
+ } else {
126
+ if (document.activeElement === lastElement) {
127
+ e.preventDefault();
128
+ firstElement.focus();
129
+ }
130
+ }
131
+ }
132
+ };
133
+
134
+ if (!isOpen) return null;
135
+
136
+ return createPortal(
137
+ <div
138
+ className={cn(
139
+ "fixed inset-0 z-50 flex items-center justify-center p-4",
140
+ "bg-black/50 backdrop-blur-sm",
141
+ "animate-in fade-in-0 duration-200",
142
+ overlayClassName
143
+ )}
144
+ onClick={handleBackdropClick}
145
+ data-component-name="ModalOverlay"
146
+ >
147
+ <div
148
+ ref={modalRef}
149
+ role="dialog"
150
+ aria-modal="true"
151
+ aria-labelledby={title ? "modal-title" : undefined}
152
+ tabIndex={-1}
153
+ onKeyDown={handleKeyDown}
154
+ className={cn(
155
+ "relative w-full max-h-[90vh] overflow-hidden",
156
+ "bg-card border rounded-lg shadow-lg",
157
+ "animate-in zoom-in-95 duration-200",
158
+ sizeClasses[size],
159
+ variantClasses[variant],
160
+ className
161
+ )}
162
+ data-component-name="Modal"
163
+ data-variant={variant}
164
+ data-size={size}
165
+ >
166
+ {/* Header */}
167
+ {(title || showCloseButton) && (
168
+ <div className="flex items-center justify-between p-6 border-b border-border">
169
+ {title && (
170
+ <h2
171
+ id="modal-title"
172
+ className={cn(
173
+ "text-lg font-semibold text-foreground",
174
+ variant === 'destructive' && "text-status-error",
175
+ variant === 'success' && "text-status-success"
176
+ )}
177
+ data-component-name="ModalTitle"
178
+ >
179
+ {title}
180
+ </h2>
181
+ )}
182
+ {showCloseButton && (
183
+ <Button
184
+ variant="ghost"
185
+ size="icon"
186
+ onClick={onClose}
187
+ className="h-8 w-8 rounded hover:bg-muted/80"
188
+ aria-label="Close modal"
189
+ data-component-name="ModalCloseButton"
190
+ >
191
+ <X className="h-4 w-4" />
192
+ </Button>
193
+ )}
194
+ </div>
195
+ )}
196
+
197
+ {/* Content */}
198
+ <div
199
+ className={cn(
200
+ "p-6 overflow-y-auto",
201
+ (!title && !showCloseButton) && "pt-6",
202
+ !footer && "pb-6",
203
+ contentClassName
204
+ )}
205
+ data-component-name="ModalContent"
206
+ >
207
+ {children}
208
+ </div>
209
+
210
+ {/* Footer */}
211
+ {footer && (
212
+ <div
213
+ className="flex items-center justify-end gap-3 p-6 pt-0 border-t border-border"
214
+ data-component-name="ModalFooter"
215
+ >
216
+ {footer}
217
+ </div>
218
+ )}
219
+ </div>
220
+ </div>,
221
+ document.body
222
+ );
223
+ };
@@ -0,0 +1,2 @@
1
+ export { Modal } from './Modal';
2
+ export type { ModalProps } from './Modal';