ezfw-core 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (154) hide show
  1. package/components/EzBaseComponent.ts +648 -0
  2. package/components/EzComponent.ts +89 -0
  3. package/components/EzInput.module.scss +183 -0
  4. package/components/EzInput.ts +104 -0
  5. package/components/EzLabel.ts +22 -0
  6. package/components/EzOutlet.ts +181 -0
  7. package/components/HtmlWrapper.ts +305 -0
  8. package/components/avatar/EzAvatar.module.scss +200 -0
  9. package/components/avatar/EzAvatar.ts +130 -0
  10. package/components/badge/EzBadge.module.scss +202 -0
  11. package/components/badge/EzBadge.ts +77 -0
  12. package/components/button/EzButton.module.scss +402 -0
  13. package/components/button/EzButton.ts +175 -0
  14. package/components/button/EzButtonGroup.ts +48 -0
  15. package/components/card/EzCard.module.scss +71 -0
  16. package/components/card/EzCard.ts +120 -0
  17. package/components/chart/EzBarChart.ts +47 -0
  18. package/components/chart/EzChart.module.scss +14 -0
  19. package/components/chart/EzChart.ts +279 -0
  20. package/components/chart/EzDoughnutChart.ts +47 -0
  21. package/components/chart/EzLineChart.ts +53 -0
  22. package/components/checkbox/EzCheckbox.module.scss +145 -0
  23. package/components/checkbox/EzCheckbox.ts +115 -0
  24. package/components/dataview/EzDataView.module.scss +115 -0
  25. package/components/dataview/EzDataView.ts +355 -0
  26. package/components/dataview/modes/EzDataViewCards.ts +322 -0
  27. package/components/dataview/modes/EzDataViewGrid.ts +76 -0
  28. package/components/datepicker/EzDatePicker.module.scss +348 -0
  29. package/components/datepicker/EzDatePicker.ts +519 -0
  30. package/components/dialog/EzDialog.module.scss +180 -0
  31. package/components/dropdown/EzDropdown.module.scss +107 -0
  32. package/components/dropdown/EzDropdown.ts +235 -0
  33. package/components/feed/EzActivityFeed.module.scss +90 -0
  34. package/components/feed/EzActivityFeed.ts +78 -0
  35. package/components/form/EzForm.ts +364 -0
  36. package/components/form/EzValidators.test.js +421 -0
  37. package/components/form/EzValidators.ts +202 -0
  38. package/components/grid/EzGrid.scss +88 -0
  39. package/components/grid/EzGrid.ts +1085 -0
  40. package/components/grid/EzGridContainer.ts +104 -0
  41. package/components/grid/body/EzGridBody.scss +283 -0
  42. package/components/grid/body/EzGridBody.ts +549 -0
  43. package/components/grid/body/EzGridCell.ts +211 -0
  44. package/components/grid/body/EzGridRow.ts +196 -0
  45. package/components/grid/filter/EzGridFilters.scss +78 -0
  46. package/components/grid/filter/EzGridFilters.ts +285 -0
  47. package/components/grid/footer/EzGridFooter.scss +136 -0
  48. package/components/grid/footer/EzGridFooter.ts +448 -0
  49. package/components/grid/header/EzGridHeader.scss +199 -0
  50. package/components/grid/header/EzGridHeader.ts +430 -0
  51. package/components/grid/query/EzGridQuery.ts +81 -0
  52. package/components/grid/state/EzGridColumns.ts +155 -0
  53. package/components/grid/state/EzGridController.ts +470 -0
  54. package/components/grid/state/EzGridLifecycle.ts +136 -0
  55. package/components/grid/state/EzGridNormalizers.test.js +273 -0
  56. package/components/grid/state/EzGridNormalizers.ts +162 -0
  57. package/components/grid/state/EzGridParts.ts +233 -0
  58. package/components/grid/state/EzGridPersistence.ts +140 -0
  59. package/components/grid/state/EzGridRemote.test.js +573 -0
  60. package/components/grid/state/EzGridRemote.ts +335 -0
  61. package/components/grid/state/EzGridSelection.ts +231 -0
  62. package/components/grid/state/EzGridSort.ts +286 -0
  63. package/components/grid/title/EzGridActionBar.ts +98 -0
  64. package/components/grid/title/EzGridTitle.ts +114 -0
  65. package/components/grid/title/EzGridTitleBar.scss +65 -0
  66. package/components/grid/title/EzGridTitleBar.ts +87 -0
  67. package/components/grid/types.ts +607 -0
  68. package/components/panel/EzPanel.module.scss +133 -0
  69. package/components/panel/EzPanel.ts +147 -0
  70. package/components/radio/EzRadio.module.scss +190 -0
  71. package/components/radio/EzRadio.ts +149 -0
  72. package/components/select/EzSelect.module.scss +153 -0
  73. package/components/select/EzSelect.ts +238 -0
  74. package/components/skeleton/EzSkeleton.module.scss +95 -0
  75. package/components/skeleton/EzSkeleton.ts +70 -0
  76. package/components/store/EzStore.ts +344 -0
  77. package/components/switch/EzSwitch.module.scss +164 -0
  78. package/components/switch/EzSwitch.ts +117 -0
  79. package/components/tabs/EzTabPanel.module.scss +181 -0
  80. package/components/tabs/EzTabPanel.ts +402 -0
  81. package/components/textarea/EzTextarea.module.scss +131 -0
  82. package/components/textarea/EzTextarea.ts +161 -0
  83. package/components/timepicker/EzTimePicker.module.scss +282 -0
  84. package/components/timepicker/EzTimePicker.ts +540 -0
  85. package/components/toast/EzToast.module.scss +291 -0
  86. package/components/tooltip/EzTooltip.module.scss +124 -0
  87. package/components/tooltip/EzTooltip.ts +153 -0
  88. package/core/EzComponentTypes.ts +693 -0
  89. package/core/EzError.ts +63 -0
  90. package/core/EzModel.ts +268 -0
  91. package/core/EzTypes.ts +328 -0
  92. package/core/eventBus.ts +284 -0
  93. package/core/ez.ts +617 -0
  94. package/core/loader.ts +725 -0
  95. package/core/renderer.ts +1010 -0
  96. package/core/router.ts +490 -0
  97. package/core/services.ts +124 -0
  98. package/core/state.ts +142 -0
  99. package/core/utils.ts +81 -0
  100. package/package.json +51 -0
  101. package/services/RouteUI.js +17 -0
  102. package/services/crypto.js +64 -0
  103. package/services/dialog.js +222 -0
  104. package/services/fetchApi.js +63 -0
  105. package/services/firebase.js +30 -0
  106. package/services/toast.js +214 -0
  107. package/template/doc/EzDocs.js +15 -0
  108. package/template/doc/EzDocs.module.scss +627 -0
  109. package/template/doc/EzDocsController.js +164 -0
  110. package/template/doc/data/activityfeed/EzActivityFeedDoc.js +42 -0
  111. package/template/doc/data/avatar/EzAvatarDoc.js +71 -0
  112. package/template/doc/data/badge/EzBadgeDoc.js +92 -0
  113. package/template/doc/data/button/EzButtonDoc.js +77 -0
  114. package/template/doc/data/buttongroup/EzButtonGroupDoc.js +102 -0
  115. package/template/doc/data/card/EzCardDoc.js +39 -0
  116. package/template/doc/data/chart/EzChartDoc.js +60 -0
  117. package/template/doc/data/checkbox/EzCheckboxDoc.js +67 -0
  118. package/template/doc/data/component/EzComponentDoc.js +34 -0
  119. package/template/doc/data/cssmodules/CSSModulesDoc.js +70 -0
  120. package/template/doc/data/datepicker/EzDatePickerDoc.js +126 -0
  121. package/template/doc/data/dialog/EzDialogDoc.js +217 -0
  122. package/template/doc/data/dropdown/EzDropdownDoc.js +178 -0
  123. package/template/doc/data/form/EzFormDoc.js +90 -0
  124. package/template/doc/data/grid/EzGridDoc.js +99 -0
  125. package/template/doc/data/input/EzInputDoc.js +92 -0
  126. package/template/doc/data/label/EzLabelDoc.js +40 -0
  127. package/template/doc/data/model/EzModelDoc.js +53 -0
  128. package/template/doc/data/outlet/EzOutletDoc.js +63 -0
  129. package/template/doc/data/panel/EzPanelDoc.js +214 -0
  130. package/template/doc/data/radio/EzRadioDoc.js +174 -0
  131. package/template/doc/data/router/EzRouterDoc.js +75 -0
  132. package/template/doc/data/select/EzSelectDoc.js +37 -0
  133. package/template/doc/data/skeleton/EzSkeletonDoc.js +149 -0
  134. package/template/doc/data/switch/EzSwitchDoc.js +82 -0
  135. package/template/doc/data/tabpanel/EzTabPanelDoc.js +44 -0
  136. package/template/doc/data/textarea/EzTextareaDoc.js +131 -0
  137. package/template/doc/data/timepicker/EzTimePickerDoc.js +107 -0
  138. package/template/doc/data/tooltip/EzTooltipDoc.js +193 -0
  139. package/template/doc/data/validators/EzValidatorsDoc.js +37 -0
  140. package/template/doc/sidebar/EzDocsSidebar.js +32 -0
  141. package/template/doc/sidebar/category/EzDocsCategory.js +33 -0
  142. package/template/doc/sidebar/item/EzDocsComponentItem.js +24 -0
  143. package/template/doc/viewer/EzDocsViewer.js +18 -0
  144. package/template/doc/viewer/codepanel/EzDocsCodePanel.js +51 -0
  145. package/template/doc/viewer/content/EzDocsContent.js +315 -0
  146. package/template/doc/viewer/header/EzDocsViewerHeader.js +46 -0
  147. package/template/doc/viewer/showcase/EzDocsShowcase.js +59 -0
  148. package/template/doc/viewer/showcase/EzDocsShowcaseSection.js +25 -0
  149. package/template/doc/viewer/showcase/EzDocsVariantItem.js +29 -0
  150. package/template/doc/welcome/EzDocsWelcome.js +48 -0
  151. package/themes/ez-theme.scss +179 -0
  152. package/themes/nature-fresh.scss +169 -0
  153. package/types/global.d.ts +21 -0
  154. package/utils/cssModules.js +81 -0
@@ -0,0 +1,107 @@
1
+ .dropdown {
2
+ position: relative;
3
+ display: inline-flex;
4
+ }
5
+
6
+ .trigger {
7
+ // Styles come from the trigger component
8
+ }
9
+
10
+ .isOpen {
11
+ .trigger {
12
+ // Can add active state styles
13
+ }
14
+ }
15
+
16
+ .menu {
17
+ position: fixed;
18
+ z-index: 10000;
19
+
20
+ min-width: 160px;
21
+ padding: 4px;
22
+
23
+ background: var(--ez-surface-secondary, #ffffff);
24
+ border: 1px solid var(--ez-border, #e2e8f0);
25
+ border-radius: var(--ez-radius-md, 6px);
26
+ box-shadow: 0 10px 25px rgba(0, 0, 0, 0.12);
27
+
28
+ transition: opacity 0.15s ease, visibility 0.15s ease;
29
+ }
30
+
31
+ .item {
32
+ display: flex;
33
+ align-items: center;
34
+ gap: 8px;
35
+ width: 100%;
36
+
37
+ padding: 8px 12px;
38
+
39
+ font-size: 13px;
40
+ font-weight: 500;
41
+ text-align: left;
42
+ color: var(--ez-text-primary, #1e293b);
43
+
44
+ background: transparent;
45
+ border: none;
46
+ border-radius: var(--ez-radius-sm, 4px);
47
+
48
+ cursor: pointer;
49
+ transition: background 0.1s ease;
50
+
51
+ &:hover:not(.disabled) {
52
+ background: var(--ez-surface-primary, #f1f5f9);
53
+ }
54
+
55
+ &:active:not(.disabled) {
56
+ background: var(--ez-surface-secondary, #e2e8f0);
57
+ }
58
+ }
59
+
60
+ .itemIcon {
61
+ width: 16px;
62
+ font-size: 14px;
63
+ color: var(--ez-text-tertiary, #64748b);
64
+ text-align: center;
65
+ }
66
+
67
+ .itemText {
68
+ flex: 1;
69
+ }
70
+
71
+ .shortcut {
72
+ font-size: 11px;
73
+ color: var(--ez-text-tertiary, #94a3b8);
74
+ margin-left: auto;
75
+ }
76
+
77
+ .danger {
78
+ color: var(--ez-danger, #ef4444);
79
+
80
+ .itemIcon {
81
+ color: var(--ez-danger, #ef4444);
82
+ }
83
+
84
+ &:hover:not(.disabled) {
85
+ background: var(--ez-danger-light, rgba(239, 68, 68, 0.1));
86
+ }
87
+ }
88
+
89
+ .disabled {
90
+ opacity: 0.5;
91
+ cursor: not-allowed;
92
+ }
93
+
94
+ .divider {
95
+ height: 1px;
96
+ margin: 4px 0;
97
+ background: var(--ez-border, #e2e8f0);
98
+ }
99
+
100
+ .header {
101
+ padding: 8px 12px 4px;
102
+ font-size: 11px;
103
+ font-weight: 600;
104
+ text-transform: uppercase;
105
+ letter-spacing: 0.5px;
106
+ color: var(--ez-text-tertiary, #94a3b8);
107
+ }
@@ -0,0 +1,235 @@
1
+ import styles from './EzDropdown.module.scss';
2
+ import { cx } from '../../utils/cssModules.js';
3
+ import { EzBaseComponent, EzBaseComponentConfig } from '../EzBaseComponent.js';
4
+
5
+ declare const ez: {
6
+ _createElement(config: unknown): Promise<HTMLElement>;
7
+ };
8
+
9
+ const cls = cx(styles);
10
+
11
+ export interface DropdownItem {
12
+ text?: string;
13
+ icon?: string;
14
+ shortcut?: string;
15
+ divider?: boolean;
16
+ header?: string;
17
+ danger?: boolean;
18
+ disabled?: boolean;
19
+ closeOnClick?: boolean;
20
+ onClick?: (e: MouseEvent) => void;
21
+ }
22
+
23
+ export interface EzDropdownConfig extends EzBaseComponentConfig {
24
+ trigger?: unknown;
25
+ items?: DropdownItem[];
26
+ position?: 'bottom-start' | 'bottom-end' | 'bottom' | 'top-start' | 'top-end' | 'top';
27
+ }
28
+
29
+ export class EzDropdown extends EzBaseComponent {
30
+ declare config: EzDropdownConfig;
31
+
32
+ private _wrapper: HTMLDivElement | null = null;
33
+ private _menu: HTMLDivElement | null = null;
34
+ private _triggerEl: HTMLElement | null = null;
35
+ private _isOpen: boolean = false;
36
+ private _position: string = 'bottom-start';
37
+ private _onOutsideClick: ((e: MouseEvent) => void) | null = null;
38
+ private _onEscape: ((e: KeyboardEvent) => void) | null = null;
39
+
40
+ async render(): Promise<HTMLDivElement> {
41
+ const cfg = this.config;
42
+
43
+ const wrapper = document.createElement('div');
44
+ wrapper.className = cls('dropdown');
45
+
46
+ if (cfg.trigger) {
47
+ const triggerEl = await ez._createElement(cfg.trigger);
48
+ triggerEl.classList.add(cls('trigger'));
49
+ triggerEl.addEventListener('click', (e) => {
50
+ e.stopPropagation();
51
+ this._toggle();
52
+ });
53
+ wrapper.appendChild(triggerEl);
54
+ this._triggerEl = triggerEl;
55
+ }
56
+
57
+ const menu = document.createElement('div');
58
+ menu.className = cls('menu');
59
+ menu.style.opacity = '0';
60
+ menu.style.visibility = 'hidden';
61
+
62
+ this._buildItems(menu, cfg.items || []);
63
+
64
+ document.body.appendChild(menu);
65
+
66
+ this._wrapper = wrapper;
67
+ this._menu = menu;
68
+ this._isOpen = false;
69
+ this._position = cfg.position || 'bottom-start';
70
+
71
+ this._onOutsideClick = (e: MouseEvent) => {
72
+ if (!wrapper.contains(e.target as Node) && !menu.contains(e.target as Node)) {
73
+ this._close();
74
+ }
75
+ };
76
+ document.addEventListener('click', this._onOutsideClick);
77
+
78
+ this._onEscape = (e: KeyboardEvent) => {
79
+ if (e.key === 'Escape' && this._isOpen) {
80
+ this._close();
81
+ }
82
+ };
83
+ document.addEventListener('keydown', this._onEscape);
84
+
85
+ return wrapper;
86
+ }
87
+
88
+ private _buildItems(menu: HTMLDivElement, items: DropdownItem[]): void {
89
+ items.forEach(item => {
90
+ if (item.divider) {
91
+ const divider = document.createElement('div');
92
+ divider.className = cls('divider');
93
+ menu.appendChild(divider);
94
+ return;
95
+ }
96
+
97
+ if (item.header) {
98
+ const header = document.createElement('div');
99
+ header.className = cls('header');
100
+ header.textContent = item.header;
101
+ menu.appendChild(header);
102
+ return;
103
+ }
104
+
105
+ const menuItem = document.createElement('button');
106
+ menuItem.type = 'button';
107
+ menuItem.className = cls(
108
+ 'item',
109
+ item.danger && 'danger',
110
+ item.disabled && 'disabled'
111
+ );
112
+
113
+ if (item.disabled) {
114
+ menuItem.disabled = true;
115
+ }
116
+
117
+ if (item.icon) {
118
+ const icon = document.createElement('i');
119
+ icon.className = cls('itemIcon', item.icon);
120
+ menuItem.appendChild(icon);
121
+ }
122
+
123
+ const text = document.createElement('span');
124
+ text.className = cls('itemText');
125
+ text.textContent = item.text || '';
126
+ menuItem.appendChild(text);
127
+
128
+ if (item.shortcut) {
129
+ const shortcut = document.createElement('span');
130
+ shortcut.className = cls('shortcut');
131
+ shortcut.textContent = item.shortcut;
132
+ menuItem.appendChild(shortcut);
133
+ }
134
+
135
+ menuItem.addEventListener('click', (e) => {
136
+ if (item.disabled) return;
137
+ if (item.onClick) {
138
+ item.onClick(e);
139
+ }
140
+ if (item.closeOnClick !== false) {
141
+ this._close();
142
+ }
143
+ });
144
+
145
+ menu.appendChild(menuItem);
146
+ });
147
+ }
148
+
149
+ private _toggle(): void {
150
+ if (this._isOpen) {
151
+ this._close();
152
+ } else {
153
+ this._open();
154
+ }
155
+ }
156
+
157
+ private _open(): void {
158
+ if (!this._triggerEl || !this._menu) return;
159
+
160
+ const rect = this._triggerEl.getBoundingClientRect();
161
+ const menuRect = this._menu.getBoundingClientRect();
162
+
163
+ let top: number, left: number;
164
+
165
+ switch (this._position) {
166
+ case 'bottom-start':
167
+ top = rect.bottom + 4;
168
+ left = rect.left;
169
+ break;
170
+ case 'bottom-end':
171
+ top = rect.bottom + 4;
172
+ left = rect.right - menuRect.width;
173
+ break;
174
+ case 'bottom':
175
+ top = rect.bottom + 4;
176
+ left = rect.left + (rect.width - menuRect.width) / 2;
177
+ break;
178
+ case 'top-start':
179
+ top = rect.top - menuRect.height - 4;
180
+ left = rect.left;
181
+ break;
182
+ case 'top-end':
183
+ top = rect.top - menuRect.height - 4;
184
+ left = rect.right - menuRect.width;
185
+ break;
186
+ case 'top':
187
+ top = rect.top - menuRect.height - 4;
188
+ left = rect.left + (rect.width - menuRect.width) / 2;
189
+ break;
190
+ default:
191
+ top = rect.bottom + 4;
192
+ left = rect.left;
193
+ }
194
+
195
+ const padding = 8;
196
+ if (left < padding) left = padding;
197
+ if (left + menuRect.width > window.innerWidth - padding) {
198
+ left = window.innerWidth - menuRect.width - padding;
199
+ }
200
+ if (top + menuRect.height > window.innerHeight - padding) {
201
+ top = rect.top - menuRect.height - 4;
202
+ }
203
+ if (top < padding) top = padding;
204
+
205
+ this._menu.style.top = `${top + window.scrollY}px`;
206
+ this._menu.style.left = `${left + window.scrollX}px`;
207
+ this._menu.style.opacity = '1';
208
+ this._menu.style.visibility = 'visible';
209
+
210
+ this._wrapper?.classList.add(cls('isOpen'));
211
+ this._isOpen = true;
212
+ }
213
+
214
+ private _close(): void {
215
+ if (!this._menu) return;
216
+
217
+ this._menu.style.opacity = '0';
218
+ this._menu.style.visibility = 'hidden';
219
+ this._wrapper?.classList.remove(cls('isOpen'));
220
+ this._isOpen = false;
221
+ }
222
+
223
+ destroy(): void {
224
+ if (this._onOutsideClick) {
225
+ document.removeEventListener('click', this._onOutsideClick);
226
+ }
227
+ if (this._onEscape) {
228
+ document.removeEventListener('keydown', this._onEscape);
229
+ }
230
+ if (this._menu?.parentNode) {
231
+ this._menu.parentNode.removeChild(this._menu);
232
+ }
233
+ super.destroy();
234
+ }
235
+ }
@@ -0,0 +1,90 @@
1
+ // ==========================================================
2
+ // EzActivityFeed - Activity Feed Styles
3
+ // ==========================================================
4
+
5
+ .list {
6
+ list-style: none;
7
+ padding: 0;
8
+ margin: 0;
9
+ overflow-y: auto;
10
+ }
11
+
12
+ .item {
13
+ display: flex;
14
+ align-items: flex-start;
15
+ gap: 12px;
16
+ padding: 14px 20px;
17
+ border-bottom: 1px solid var(--ez-border, #e2e8f0);
18
+ transition: background 0.15s ease;
19
+
20
+ &:last-child {
21
+ border-bottom: none;
22
+ }
23
+
24
+ &:hover {
25
+ background: var(--ez-surface-primary, #f8fafc);
26
+ }
27
+ }
28
+
29
+ .icon {
30
+ width: 36px;
31
+ height: 36px;
32
+ border-radius: 50%;
33
+ display: flex;
34
+ align-items: center;
35
+ justify-content: center;
36
+ flex-shrink: 0;
37
+ font-size: 14px;
38
+ }
39
+
40
+ .content {
41
+ flex: 1;
42
+ min-width: 0;
43
+ }
44
+
45
+ .description {
46
+ font-size: 13px;
47
+ color: var(--ez-text-primary, #1e293b);
48
+ line-height: 1.4;
49
+ margin: 0;
50
+ }
51
+
52
+ .timestamp {
53
+ font-size: 12px;
54
+ color: var(--ez-text-tertiary, #94a3b8);
55
+ margin: 2px 0 0 0;
56
+ }
57
+
58
+ // ==========================================================
59
+ // Icon Variants
60
+ // ==========================================================
61
+
62
+ .primary {
63
+ background: rgba(0, 88, 113, 0.1);
64
+ color: var(--ez-primary, #005871);
65
+ }
66
+
67
+ .success {
68
+ background: rgba(16, 185, 129, 0.1);
69
+ color: #059669;
70
+ }
71
+
72
+ .warning {
73
+ background: rgba(245, 158, 11, 0.1);
74
+ color: #d97706;
75
+ }
76
+
77
+ .danger {
78
+ background: rgba(239, 68, 68, 0.1);
79
+ color: #dc2626;
80
+ }
81
+
82
+ .info {
83
+ background: rgba(59, 130, 246, 0.1);
84
+ color: #2563eb;
85
+ }
86
+
87
+ .accent {
88
+ background: rgba(46, 82, 131, 0.1);
89
+ color: #2E5283;
90
+ }
@@ -0,0 +1,78 @@
1
+ import { EzBaseComponent, EzBaseComponentConfig } from '../EzBaseComponent.js';
2
+ import styles from './EzActivityFeed.module.scss';
3
+
4
+ declare const ez: {
5
+ registerComponent(name: string, component: unknown): void;
6
+ };
7
+
8
+ interface ActivityItem {
9
+ icon: string;
10
+ variant?: string;
11
+ iconColor?: string;
12
+ description?: string;
13
+ text?: string;
14
+ timestamp?: string;
15
+ time?: string;
16
+ }
17
+
18
+ export interface EzActivityFeedConfig extends EzBaseComponentConfig {
19
+ items?: ActivityItem[];
20
+ maxHeight?: number | string;
21
+ }
22
+
23
+ export class EzActivityFeed extends EzBaseComponent {
24
+ declare config: EzActivityFeedConfig;
25
+
26
+ constructor(config: EzActivityFeedConfig = {}) {
27
+ super(config);
28
+ this.config = config;
29
+ }
30
+
31
+ render(): HTMLElement {
32
+ const {
33
+ items = [],
34
+ maxHeight = 320,
35
+ cls = ''
36
+ } = this.config;
37
+
38
+ const el = document.createElement('ul');
39
+ el.className = `${styles.list} ${cls}`.trim();
40
+ if (maxHeight) {
41
+ el.style.maxHeight = typeof maxHeight === 'number' ? `${maxHeight}px` : String(maxHeight);
42
+ }
43
+
44
+ items.forEach(item => {
45
+ const li = document.createElement('li');
46
+ li.className = styles.item;
47
+
48
+ const iconWrapper = document.createElement('div');
49
+ const variantClass = styles[item.variant || item.iconColor || 'primary'] || '';
50
+ iconWrapper.className = `${styles.icon} ${variantClass}`.trim();
51
+ iconWrapper.innerHTML = `<i class="${item.icon}"></i>`;
52
+ li.appendChild(iconWrapper);
53
+
54
+ const content = document.createElement('div');
55
+ content.className = styles.content;
56
+
57
+ const description = document.createElement('p');
58
+ description.className = styles.description;
59
+ description.textContent = item.description || item.text || '';
60
+ content.appendChild(description);
61
+
62
+ if (item.timestamp || item.time) {
63
+ const timestamp = document.createElement('p');
64
+ timestamp.className = styles.timestamp;
65
+ timestamp.textContent = item.timestamp || item.time || '';
66
+ content.appendChild(timestamp);
67
+ }
68
+
69
+ li.appendChild(content);
70
+ el.appendChild(li);
71
+ });
72
+
73
+ this.applyStyles(el);
74
+ return el;
75
+ }
76
+ }
77
+
78
+ ez.registerComponent('EzActivityFeed', EzActivityFeed);