@tsiky/components-r19 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 (287) hide show
  1. package/.prettierignore +4 -0
  2. package/.prettierrc +9 -0
  3. package/.storybook/main.ts +17 -0
  4. package/.storybook/preview.ts +21 -0
  5. package/.storybook/vitest.setup.ts +7 -0
  6. package/README.md +69 -0
  7. package/chart.ts +1 -0
  8. package/eslint.config.js +40 -0
  9. package/index.html +13 -0
  10. package/index.ts +33 -0
  11. package/package.json +63 -0
  12. package/public/vite.svg +1 -0
  13. package/src/App.css +42 -0
  14. package/src/App.tsx +12 -0
  15. package/src/assets/accessibility.png +0 -0
  16. package/src/assets/accessibility.svg +1 -0
  17. package/src/assets/addon-library.png +0 -0
  18. package/src/assets/assets.png +0 -0
  19. package/src/assets/avif-test-image.avif +0 -0
  20. package/src/assets/context.png +0 -0
  21. package/src/assets/discord.svg +1 -0
  22. package/src/assets/docs.png +0 -0
  23. package/src/assets/figma-plugin.png +0 -0
  24. package/src/assets/github.svg +1 -0
  25. package/src/assets/react.svg +1 -0
  26. package/src/assets/share.png +0 -0
  27. package/src/assets/styling.png +0 -0
  28. package/src/assets/testing.png +0 -0
  29. package/src/assets/theming.png +0 -0
  30. package/src/assets/tutorials.svg +1 -0
  31. package/src/assets/youtube.svg +1 -0
  32. package/src/components/AddItemModal/AddItemModal.module.css +72 -0
  33. package/src/components/AddItemModal/AddItemModal.tsx +82 -0
  34. package/src/components/AddItemModal/index.ts +1 -0
  35. package/src/components/Alert/Alert.css +54 -0
  36. package/src/components/Alert/Alert.stories.tsx +82 -0
  37. package/src/components/Alert/Alert.tsx +85 -0
  38. package/src/components/Alert/AlertContext.tsx +200 -0
  39. package/src/components/Alert/AlertModels.ts +34 -0
  40. package/src/components/Alert/index.ts +3 -0
  41. package/src/components/AnnouncementPanel/FlexRowContainer.css +17 -0
  42. package/src/components/AnnouncementPanel/FlexRowContainer.stories.tsx +329 -0
  43. package/src/components/AnnouncementPanel/FlexRowContainer.tsx +24 -0
  44. package/src/components/AnnouncementPanel/ListBox/CounterListBox.css +56 -0
  45. package/src/components/AnnouncementPanel/ListBox/CounterListBox.stories.tsx +292 -0
  46. package/src/components/AnnouncementPanel/ListBox/CounterListBox.tsx +106 -0
  47. package/src/components/AnnouncementPanel/ListBox/SimpleListBox.css +57 -0
  48. package/src/components/AnnouncementPanel/ListBox/SimpleListBox.stories.tsx +189 -0
  49. package/src/components/AnnouncementPanel/ListBox/SimpleListBox.tsx +138 -0
  50. package/src/components/AnnouncementPanel/ListBox/TrendListBox.css +61 -0
  51. package/src/components/AnnouncementPanel/ListBox/TrendListBox.stories.tsx +257 -0
  52. package/src/components/AnnouncementPanel/ListBox/TrendListBox.tsx +90 -0
  53. package/src/components/AnnouncementPanel/ListBox/index.ts +3 -0
  54. package/src/components/AnnouncementPanel/ListContentContainer.css +23 -0
  55. package/src/components/AnnouncementPanel/ListContentContainer.stories.tsx +212 -0
  56. package/src/components/AnnouncementPanel/ListContentContainer.tsx +33 -0
  57. package/src/components/AnnouncementPanel/index.ts +3 -0
  58. package/src/components/BandChart/index.tsx +282 -0
  59. package/src/components/Button/Button.module.css +165 -0
  60. package/src/components/Button/Button.stories.ts +132 -0
  61. package/src/components/Button/Button.tsx +55 -0
  62. package/src/components/Button/button.css +29 -0
  63. package/src/components/Button/index.ts +2 -0
  64. package/src/components/ChartContainer/ChartContainer.css +116 -0
  65. package/src/components/ChartContainer/ChartContainer.stories.tsx +159 -0
  66. package/src/components/ChartContainer/ChartContainer.tsx +155 -0
  67. package/src/components/ChartContainer/index.ts +1 -0
  68. package/src/components/Charts/area-chart-admission/AreaChartAdmission.stories.tsx +65 -0
  69. package/src/components/Charts/area-chart-admission/AreaChartAdmission.tsx +89 -0
  70. package/src/components/Charts/area-chart-admission/content.json +48 -0
  71. package/src/components/Charts/area-chart-hospitalisation/AreaChartHospitalisation.stories.tsx +141 -0
  72. package/src/components/Charts/area-chart-hospitalisation/AreaChartHospitalisation.tsx +262 -0
  73. package/src/components/Charts/area-chart-hospitalisation/content.json +55 -0
  74. package/src/components/Charts/bar-chart/BarChart.stories.tsx +50 -0
  75. package/src/components/Charts/bar-chart/BarChart.tsx +132 -0
  76. package/src/components/Charts/bar-chart/content.json +15 -0
  77. package/src/components/Charts/boxplot-chart/BoxPlotChart.stories.tsx +63 -0
  78. package/src/components/Charts/boxplot-chart/BoxPlotChart.tsx +114 -0
  79. package/src/components/Charts/boxplot-chart/boxUtils.ts +22 -0
  80. package/src/components/Charts/boxplot-chart/content.json +11 -0
  81. package/src/components/Charts/mixed-chart/MixedChart.stories.tsx +83 -0
  82. package/src/components/Charts/mixed-chart/MixedChart.tsx +625 -0
  83. package/src/components/Charts/mixed-chart/content.json +34 -0
  84. package/src/components/Charts/sankey-adaptation/sankey.tsx +70 -0
  85. package/src/components/Charts/sankey-chart/SankeyChart.stories.tsx +69 -0
  86. package/src/components/Charts/sankey-chart/SankeyChart.tsx +155 -0
  87. package/src/components/Charts/sankey-chart/content.json +15 -0
  88. package/src/components/Charts/stacked-column/StackedColumn.stories.tsx +72 -0
  89. package/src/components/Charts/stacked-column/StackedColumn.tsx +406 -0
  90. package/src/components/Charts/stacked-column/content.json +37 -0
  91. package/src/components/Charts/stacked-column-one-hundred-percent/StackedColumnOneHundredPercent.stories.tsx +43 -0
  92. package/src/components/Charts/stacked-column-one-hundred-percent/StackedColumnOneHundredPercent.tsx +75 -0
  93. package/src/components/Charts/stacked-column-one-hundred-percent/content.json +6 -0
  94. package/src/components/CircularProgress/CircularProgress.css +79 -0
  95. package/src/components/CircularProgress/CircularProgress.stories.tsx +251 -0
  96. package/src/components/CircularProgress/CircularProgress.tsx +101 -0
  97. package/src/components/CircularProgress/index.ts +2 -0
  98. package/src/components/Configure.mdx +369 -0
  99. package/src/components/DayStatCard/DayStatCard.css +50 -0
  100. package/src/components/DayStatCard/DayStatCard.stories.tsx +273 -0
  101. package/src/components/DayStatCard/DayStatCard.tsx +69 -0
  102. package/src/components/DayStatCard/index.ts +2 -0
  103. package/src/components/DraggableSwitcher/DraggableSwitcherButton.tsx +58 -0
  104. package/src/components/DraggableSwitcher/context/useDraggableSwitcher.tsx +45 -0
  105. package/src/components/DraggableSwitcher/index.ts +2 -0
  106. package/src/components/DropdownMenu/DropdownMenu.css +100 -0
  107. package/src/components/DropdownMenu/DropdownMenu.stories.tsx +174 -0
  108. package/src/components/DropdownMenu/DropdownMenu.tsx +106 -0
  109. package/src/components/DropdownMenu/index.ts +1 -0
  110. package/src/components/DynamicForm/DynamicFom.stories.ts +773 -0
  111. package/src/components/DynamicForm/DynamicForm.module.css +468 -0
  112. package/src/components/DynamicForm/DynamicForm.tsx +224 -0
  113. package/src/components/DynamicForm/index.ts +3 -0
  114. package/src/components/DynamicForm/tools/form-metadata.ts +82 -0
  115. package/src/components/DynamicForm/tools/validation.ts +168 -0
  116. package/src/components/DynamicInput/DynamicInput.module.css +126 -0
  117. package/src/components/DynamicInput/DynamicInput.stories.ts +350 -0
  118. package/src/components/DynamicInput/DynamicInput.tsx +144 -0
  119. package/src/components/DynamicInput/index.ts +2 -0
  120. package/src/components/DynamicInput/input/CheckboxInput.tsx +41 -0
  121. package/src/components/DynamicInput/input/ColorInput.tsx +46 -0
  122. package/src/components/DynamicInput/input/DateInput.tsx +57 -0
  123. package/src/components/DynamicInput/input/FileInput.tsx +174 -0
  124. package/src/components/DynamicInput/input/InputWrapper.tsx +26 -0
  125. package/src/components/DynamicInput/input/RadioInput.tsx +46 -0
  126. package/src/components/DynamicInput/input/RangeInput.tsx +46 -0
  127. package/src/components/DynamicInput/input/SelectInput.tsx +75 -0
  128. package/src/components/DynamicInput/input/TextInput.tsx +101 -0
  129. package/src/components/DynamicInput/input/TextareaInput.tsx +47 -0
  130. package/src/components/DynamicInput/input/assets/ColorInput.module.css +48 -0
  131. package/src/components/DynamicInput/input/assets/DateInput.module.css +83 -0
  132. package/src/components/DynamicInput/input/assets/FileInput.module.css +133 -0
  133. package/src/components/DynamicInput/input/assets/RangeInput.module.css +169 -0
  134. package/src/components/DynamicInput/input/assets/SelectInput.module.css +95 -0
  135. package/src/components/DynamicInput/input/index.ts +10 -0
  136. package/src/components/DynamicTable/AdvancedFilters.tsx +196 -0
  137. package/src/components/DynamicTable/ColumnSorter.tsx +185 -0
  138. package/src/components/DynamicTable/Pagination.tsx +115 -0
  139. package/src/components/DynamicTable/TableBody.tsx +42 -0
  140. package/src/components/DynamicTable/TableCell.tsx +30 -0
  141. package/src/components/DynamicTable/TableHeader.tsx +34 -0
  142. package/src/components/DynamicTable/TableRow.tsx +56 -0
  143. package/src/components/DynamicTable/TableauDynamic.stories.ts +1014 -0
  144. package/src/components/DynamicTable/TableauDynamique.module.css +1287 -0
  145. package/src/components/DynamicTable/TableauDynamique.tsx +154 -0
  146. package/src/components/DynamicTable/filters/BooleanFilter.tsx +30 -0
  147. package/src/components/DynamicTable/filters/DateFilter.tsx +28 -0
  148. package/src/components/DynamicTable/filters/DateRangeFilter.tsx +51 -0
  149. package/src/components/DynamicTable/filters/FilterRenderer.tsx +117 -0
  150. package/src/components/DynamicTable/filters/MultiSelectFilter.tsx +59 -0
  151. package/src/components/DynamicTable/filters/NumberFilter.tsx +37 -0
  152. package/src/components/DynamicTable/filters/OperatorFilter.tsx +64 -0
  153. package/src/components/DynamicTable/filters/SelectFilter.tsx +69 -0
  154. package/src/components/DynamicTable/filters/TextFilter.tsx +39 -0
  155. package/src/components/DynamicTable/filters/index.ts +9 -0
  156. package/src/components/DynamicTable/hooks/index.ts +2 -0
  157. package/src/components/DynamicTable/hooks/useAsyncActions.ts +36 -0
  158. package/src/components/DynamicTable/hooks/useFilters.ts +142 -0
  159. package/src/components/DynamicTable/hooks/useTableData.ts +216 -0
  160. package/src/components/DynamicTable/index.ts +11 -0
  161. package/src/components/DynamicTable/tools/filterTypes.ts +118 -0
  162. package/src/components/DynamicTable/tools/index.ts +3 -0
  163. package/src/components/DynamicTable/tools/tableConfig.ts +96 -0
  164. package/src/components/DynamicTable/tools/tableTypes.ts +63 -0
  165. package/src/components/EntryControl/EntryControl.module.css +218 -0
  166. package/src/components/EntryControl/EntryControl.stories.tsx +71 -0
  167. package/src/components/EntryControl/EntryControl.tsx +117 -0
  168. package/src/components/EntryControl/index.ts +2 -0
  169. package/src/components/Grid/Grid.stories.tsx +94 -0
  170. package/src/components/Grid/Grid.tsx +214 -0
  171. package/src/components/Grid/grid.css +285 -0
  172. package/src/components/Grid/index.ts +2 -0
  173. package/src/components/Header/Header.stories.tsx +164 -0
  174. package/src/components/Header/Header.tsx +59 -0
  175. package/src/components/Header/header.css +31 -0
  176. package/src/components/Header/index.ts +1 -0
  177. package/src/components/IconText/IconText..stories.tsx +135 -0
  178. package/src/components/IconText/IconText.css +43 -0
  179. package/src/components/IconText/IconText.tsx +43 -0
  180. package/src/components/IconText/index.ts +1 -0
  181. package/src/components/LanguageSelector/LanguageSelector.css +31 -0
  182. package/src/components/LanguageSelector/LanguageSelector.stories.tsx +38 -0
  183. package/src/components/LanguageSelector/LanguageSelector.tsx +37 -0
  184. package/src/components/LanguageSelector/index.ts +1 -0
  185. package/src/components/Logo/Logo.css +89 -0
  186. package/src/components/Logo/Logo.stories.tsx +79 -0
  187. package/src/components/Logo/Logo.tsx +22 -0
  188. package/src/components/Logo/index.ts +2 -0
  189. package/src/components/MetricsPanel/MetricsItem.tsx +128 -0
  190. package/src/components/MetricsPanel/MetricsPanel.module.css +636 -0
  191. package/src/components/MetricsPanel/MetricsPanel.stories.tsx +692 -0
  192. package/src/components/MetricsPanel/MetricsPanel.tsx +282 -0
  193. package/src/components/MetricsPanel/PanelHeader.tsx +19 -0
  194. package/src/components/MetricsPanel/SummaryCard.tsx +61 -0
  195. package/src/components/MetricsPanel/index.ts +4 -0
  196. package/src/components/MetricsPanel/renderers/CompactRenderer.tsx +125 -0
  197. package/src/components/MetricsPanel/renderers/ImageCard.tsx +62 -0
  198. package/src/components/MetricsPanel/renderers/PertinenceCard.tsx +55 -0
  199. package/src/components/MetricsPanel/tools/MetricsPanelTypes.ts +62 -0
  200. package/src/components/MetricsPanel/tools/chooseDefaultRender.ts +50 -0
  201. package/src/components/MetricsPanel/tools/colorUtils.ts +39 -0
  202. package/src/components/MetricsPanel/tools/index.ts +2 -0
  203. package/src/components/ModuleHeader/ModuleHeader.css +37 -0
  204. package/src/components/ModuleHeader/ModuleHeader.stories.tsx +37 -0
  205. package/src/components/ModuleHeader/ModuleHeader.tsx +42 -0
  206. package/src/components/ModuleHeader/index.ts +1 -0
  207. package/src/components/ModuleSideBar/ModuleSideBar.css +227 -0
  208. package/src/components/ModuleSideBar/ModuleSideBar.stories.tsx +40 -0
  209. package/src/components/ModuleSideBar/ModuleSideBar.tsx +155 -0
  210. package/src/components/ModuleSideBar/index.ts +1 -0
  211. package/src/components/NavBar/NavBar.css +58 -0
  212. package/src/components/NavBar/NavBar.stories.tsx +169 -0
  213. package/src/components/NavBar/NavBar.tsx +100 -0
  214. package/src/components/NavBar/NavContext.tsx +30 -0
  215. package/src/components/NavBar/index.ts +2 -0
  216. package/src/components/NavItem/NavItem.css +29 -0
  217. package/src/components/NavItem/NavItem.tsx +58 -0
  218. package/src/components/NavItem/index.ts +1 -0
  219. package/src/components/Page/Dashboard.tsx +93 -0
  220. package/src/components/Page/Page.stories.ts +33 -0
  221. package/src/components/Page/Page.tsx +73 -0
  222. package/src/components/Page/page.css +81 -0
  223. package/src/components/PerformanceCard/PerformanceCard.module.css +232 -0
  224. package/src/components/PerformanceCard/PerformanceCard.stories.tsx +441 -0
  225. package/src/components/PerformanceCard/PerformanceCard.tsx +198 -0
  226. package/src/components/PerformanceCard/defaultHistogramRenderer.tsx +32 -0
  227. package/src/components/PerformanceCard/tools/types.ts +50 -0
  228. package/src/components/PerformanceCard/tools/usePerformanceCard.ts +93 -0
  229. package/src/components/PeriodRange/PeriodRange.module.css +158 -0
  230. package/src/components/PeriodRange/PeriodRange.stories.tsx +66 -0
  231. package/src/components/PeriodRange/PeriodRange.tsx +130 -0
  232. package/src/components/PeriodSelect/PeriodSelect.module.css +65 -0
  233. package/src/components/PeriodSelect/PeriodSelect.stories.tsx +40 -0
  234. package/src/components/PeriodSelect/PeriodSelect.tsx +42 -0
  235. package/src/components/PeriodSelect/index.ts +1 -0
  236. package/src/components/ScrollableHorizontale/ScrollableHorizontale.css +40 -0
  237. package/src/components/ScrollableHorizontale/ScrollableHorizontale.stories.tsx +133 -0
  238. package/src/components/ScrollableHorizontale/ScrollableHorizontale.tsx +29 -0
  239. package/src/components/ScrollableHorizontale/index.ts +1 -0
  240. package/src/components/SearchBar/SearchBar.css +40 -0
  241. package/src/components/SearchBar/SearchBar.stories.tsx +36 -0
  242. package/src/components/SearchBar/SearchBar.tsx +30 -0
  243. package/src/components/SearchBar/index.ts +1 -0
  244. package/src/components/SectionTitle/SectionTitle.css +21 -0
  245. package/src/components/SectionTitle/SectionTitle.stories.tsx +39 -0
  246. package/src/components/SectionTitle/SectionTitle.tsx +18 -0
  247. package/src/components/SideComponent/PatientEditor.tsx +64 -0
  248. package/src/components/SideComponent/SideComponent.css +179 -0
  249. package/src/components/SideComponent/SideComponent.stories.tsx +547 -0
  250. package/src/components/SideComponent/SideComponent.tsx +243 -0
  251. package/src/components/SideComponent/hooks/useBodyScrollLock.ts +15 -0
  252. package/src/components/SideComponent/index.ts +2 -0
  253. package/src/components/SideComponent/portal.ts +11 -0
  254. package/src/components/SubNavBar/SubNavBar.tsx +41 -0
  255. package/src/components/SubNavBar/index.ts +1 -0
  256. package/src/components/Switcher/Switcher.css +65 -0
  257. package/src/components/Switcher/Switcher.stories.tsx +153 -0
  258. package/src/components/Switcher/Switcher.tsx +83 -0
  259. package/src/components/Switcher/index.ts +1 -0
  260. package/src/components/Title/Title.stories.tsx +18 -0
  261. package/src/components/Title/Title.tsx +26 -0
  262. package/src/components/Title/index.ts +1 -0
  263. package/src/components/TranslationKey/TranslationKey.css +272 -0
  264. package/src/components/TranslationKey/TranslationKey.stories.tsx +50 -0
  265. package/src/components/TranslationKey/TranslationKey.tsx +245 -0
  266. package/src/components/TranslationKey/index.ts +1 -0
  267. package/src/components/TrendItem/TrendItem.css +36 -0
  268. package/src/components/TrendItem/TrendItem.stories.tsx +276 -0
  269. package/src/components/TrendItem/TrendItem.tsx +46 -0
  270. package/src/components/TrendItem/index.ts +1 -0
  271. package/src/components/TrendList/TrendList.css +16 -0
  272. package/src/components/TrendList/TrendList.stories.tsx +337 -0
  273. package/src/components/TrendList/TrendList.tsx +45 -0
  274. package/src/components/TrendList/index.ts +1 -0
  275. package/src/components/theme/ThemeSwitcherButton.tsx +64 -0
  276. package/src/components/theme/context/ThemeContext.tsx +61 -0
  277. package/src/components/theme/context/index.ts +2 -0
  278. package/src/components/theme/context/useThemeSwitcher.ts +18 -0
  279. package/src/components/theme/index.ts +1 -0
  280. package/src/index.css +68 -0
  281. package/src/main.tsx +10 -0
  282. package/src/vite-env.d.ts +1 -0
  283. package/tsconfig.app.json +27 -0
  284. package/tsconfig.json +4 -0
  285. package/tsconfig.node.json +25 -0
  286. package/vite.config.ts +43 -0
  287. package/vitest.shims.d.ts +1 -0
@@ -0,0 +1,106 @@
1
+ import React from 'react';
2
+ import './CounterListBox.css';
3
+
4
+ export type CounterListBoxProps = {
5
+ className?: string;
6
+ style?: React.CSSProperties;
7
+ title?: string;
8
+ titleColor?: string;
9
+ data?: (string | [string, string])[];
10
+ badgeColor?: string;
11
+ badgeColors?: { [index: number]: string };
12
+ badgePadding?: string;
13
+ badgeWidth?: string;
14
+ itemBg?: string;
15
+ lineClasses?: { [index: number]: string };
16
+ colGap?: string;
17
+ };
18
+
19
+ export const CounterListBox: React.FC<CounterListBoxProps> = ({
20
+ className = '',
21
+ style = {},
22
+ title,
23
+ titleColor,
24
+ data = [],
25
+ badgeColor = '#0ea5e9',
26
+ badgeColors = {},
27
+ badgePadding = '8px 16px',
28
+ badgeWidth,
29
+ itemBg,
30
+ lineClasses = {},
31
+ colGap = '1rem',
32
+ }) => {
33
+ const has2D = data.some((r) => Array.isArray(r));
34
+ const gridStyle: React.CSSProperties | undefined = colGap
35
+ ? ({ ['--col-gap' as any]: colGap } as React.CSSProperties)
36
+ : undefined;
37
+
38
+ return (
39
+ <div className={`clb-root ${className}`} style={{ ...style }}>
40
+ {title && (
41
+ <div className='clb-title' style={titleColor ? { color: titleColor } : undefined}>
42
+ {title}
43
+ </div>
44
+ )}
45
+
46
+ <div className='clb-grid' style={gridStyle}>
47
+ {(data as (string | [string, string])[]).map((row, idx) => {
48
+ const lineClass = lineClasses[idx] || '';
49
+
50
+ if (Array.isArray(row)) {
51
+ const [left, right] = row;
52
+ const currentBadgeColor = badgeColors[idx] || badgeColor;
53
+
54
+ return (
55
+ <div
56
+ key={idx}
57
+ className={`clb-row ${lineClass}`}
58
+ style={{ background: itemBg ?? undefined }}
59
+ >
60
+ <div className='clb-badge-col'>
61
+ <div
62
+ className='counter-badge'
63
+ style={{
64
+ background: currentBadgeColor,
65
+ padding: badgePadding,
66
+ width: badgeWidth,
67
+ minWidth: badgeWidth,
68
+ }}
69
+ >
70
+ {left}
71
+ </div>
72
+ </div>
73
+
74
+ <div className='clb-text'>{right}</div>
75
+ </div>
76
+ );
77
+ }
78
+
79
+ if (!has2D) {
80
+ return (
81
+ <div
82
+ key={idx}
83
+ className={`clb-fullrow ${lineClass}`}
84
+ style={{ background: itemBg ?? undefined }}
85
+ >
86
+ <span>{row as string}</span>
87
+ </div>
88
+ );
89
+ }
90
+
91
+ // 1D in mixed mode: empty badge + text
92
+ return (
93
+ <div
94
+ key={idx}
95
+ className={`clb-row ${lineClass}`}
96
+ style={{ background: itemBg ?? undefined }}
97
+ >
98
+ <div className='clb-badge-col clb-empty-badge' />
99
+ <div className='clb-text'>{row as string}</div>
100
+ </div>
101
+ );
102
+ })}
103
+ </div>
104
+ </div>
105
+ );
106
+ };
@@ -0,0 +1,57 @@
1
+ .slb-root {
2
+ padding: 0.125rem;
3
+ }
4
+ .slb-title {
5
+ font-weight: var(--font-bold);
6
+ padding: 0.0625rem 0.125rem;
7
+ border-radius: 0.03125rem;
8
+ margin-bottom: 0.0625rem;
9
+ }
10
+ .slb-list {
11
+ list-style: none;
12
+ margin: 0;
13
+ padding: 0;
14
+ display: flex;
15
+ flex-direction: column;
16
+ gap: 0.03125rem;
17
+ }
18
+ .slb-item {
19
+ padding: 0.0625rem 0.125rem;
20
+ }
21
+
22
+ .slb-grid {
23
+ display: flex;
24
+ flex-direction: column;
25
+ gap: 0.03125rem;
26
+ }
27
+
28
+ .slb-row {
29
+ display: grid;
30
+ grid-template-columns: minmax(max-content, 1fr) minmax(max-content, 1fr);
31
+ gap: var(--col-gap, 0.125rem);
32
+ align-items: center;
33
+ padding: 0;
34
+ }
35
+
36
+ .slb-grid-left {
37
+ padding: 0.0625rem 0.125rem;
38
+ min-width: 0;
39
+ justify-self: start;
40
+ overflow-wrap: break-word;
41
+ word-wrap: break-word;
42
+ }
43
+
44
+ .slb-grid-right {
45
+ padding: 0.0625rem 0.125rem;
46
+ min-width: 0;
47
+ justify-self: start;
48
+ }
49
+
50
+ .slb-empty {
51
+ color: #666;
52
+ padding: 0.125rem 0.125rem;
53
+ text-align: center;
54
+ }
55
+ .slb-empty-cell {
56
+ opacity: 0.5;
57
+ }
@@ -0,0 +1,189 @@
1
+ import type { Meta, StoryObj } from '@storybook/react';
2
+ import { SimpleListBox } from './SimpleListBox';
3
+
4
+ const meta: Meta<typeof SimpleListBox> = {
5
+ title: 'Components/ListBox/SimpleListBox',
6
+ component: SimpleListBox,
7
+ parameters: {
8
+ layout: 'centered',
9
+ docs: {
10
+ description: {
11
+ component:
12
+ 'A simple list component for displaying data in one or two columns with a clean design.',
13
+ },
14
+ },
15
+ },
16
+ tags: ['autodocs'],
17
+ argTypes: {
18
+ data: {
19
+ control: 'object',
20
+ description: 'Array of strings or [left, right] tuples for two-column display',
21
+ },
22
+ titleBg: {
23
+ control: 'color',
24
+ description: 'Background color for the title',
25
+ },
26
+ titleColor: {
27
+ control: 'color',
28
+ description: 'Text color for the title',
29
+ },
30
+ itemBg: {
31
+ control: 'color',
32
+ description: 'Background color for items',
33
+ },
34
+ },
35
+ };
36
+
37
+ export default meta;
38
+ type Story = StoryObj<typeof meta>;
39
+
40
+ export const Default: Story = {
41
+ name: 'Default',
42
+ args: {
43
+ title: 'Simple List',
44
+ data: ['Item 1', 'Item 2', 'Item 3', 'Item 4'],
45
+ },
46
+ };
47
+
48
+ export const TwoColumns: Story = {
49
+ name: 'Two Columns',
50
+ args: {
51
+ title: 'Product Inventory',
52
+ data: [
53
+ ['Apples', '42 units'],
54
+ ['Bananas', '28 units'],
55
+ ['Oranges', '15 units'],
56
+ ['Pears', '33 units'],
57
+ ['Pineapples', '8 units'],
58
+ ],
59
+ },
60
+ };
61
+
62
+ export const MixedData: Story = {
63
+ name: 'Mixed Data',
64
+ args: {
65
+ title: 'System Report',
66
+ data: [
67
+ ['CPU', '45% usage'],
68
+ 'No critical alerts detected',
69
+ ['Memory', '68% used'],
70
+ 'All services running normally',
71
+ ['Storage', '82% occupied'],
72
+ 'Backup scheduled for tonight',
73
+ ],
74
+ },
75
+ };
76
+
77
+ export const PriorityLevels: Story = {
78
+ name: 'Priority Levels',
79
+ args: {
80
+ title: 'Alerts by Priority',
81
+ data: [
82
+ ['Critical', '3 alerts'],
83
+ ['High', '7 alerts'],
84
+ ['Medium', '12 alerts'],
85
+ ['Low', '5 alerts'],
86
+ ],
87
+ lineClasses: {
88
+ 0: 'priority-critical',
89
+ 1: 'priority-high',
90
+ 2: 'priority-medium',
91
+ 3: 'priority-low',
92
+ },
93
+ },
94
+ decorators: [
95
+ (Story) => (
96
+ <>
97
+ <style>
98
+ {`
99
+ .priority-critical { background-color: #fef2f2; color: #dc2626; }
100
+ .priority-high { background-color: #fffbeb; color: #d97706; }
101
+ .priority-medium { background-color: #f0f9ff; color: #0369a1; }
102
+ .priority-low { background-color: #f0fdf4; color: #16a34a; }
103
+ `}
104
+ </style>
105
+ <Story />
106
+ </>
107
+ ),
108
+ ],
109
+ };
110
+
111
+ export const ServiceStatus: Story = {
112
+ name: 'Service Status',
113
+ args: {
114
+ title: 'Service Status',
115
+ data: [
116
+ ['API Gateway', '✅ Online'],
117
+ ['Database', '✅ Online'],
118
+ ['Redis Cache', '⚠️ Degraded'],
119
+ ['Email Service', '❌ Offline'],
120
+ ['Payment Service', '✅ Online'],
121
+ ],
122
+ lineClasses: {
123
+ 2: 'status-degraded',
124
+ 3: 'status-offline',
125
+ },
126
+ },
127
+ decorators: [
128
+ (Story) => (
129
+ <>
130
+ <style>
131
+ {`
132
+ .status-degraded { background-color: #fffbeb; color: #d97706; }
133
+ .status-offline { background-color: #fef2f2; color: #dc2626; }
134
+ `}
135
+ </style>
136
+ <Story />
137
+ </>
138
+ ),
139
+ ],
140
+ };
141
+
142
+ export const CustomBackgrounds: Story = {
143
+ name: 'Custom Backgrounds',
144
+ args: {
145
+ title: 'System Metrics',
146
+ titleBg: '#1e40af',
147
+ titleColor: 'white',
148
+ itemBg: '#f0f9ff',
149
+ data: [
150
+ ['CPU Temperature', '42°C'],
151
+ ['Uptime', '99.9%'],
152
+ ['Network Latency', '24ms'],
153
+ ['Requests/min', '1,245'],
154
+ ],
155
+ },
156
+ };
157
+
158
+ export const EmptyState: Story = {
159
+ name: 'Empty State',
160
+ args: {
161
+ title: 'Pending Tasks',
162
+ data: [],
163
+ emptyContent: '🎉 No pending tasks - everything is up to date!',
164
+ },
165
+ };
166
+
167
+ export const CompactView: Story = {
168
+ name: 'Compact View',
169
+ args: {
170
+ title: 'Quick Metrics',
171
+ data: [
172
+ ['Visits', '1.2K'],
173
+ ['Conversions', '42'],
174
+ ['Rate', '3.5%'],
175
+ ],
176
+ },
177
+ };
178
+
179
+ export const LongContent: Story = {
180
+ name: 'Long Content',
181
+ args: {
182
+ title: 'Detailed Descriptions',
183
+ data: [
184
+ ['Customer Service', 'Available 24/7 with multilingual support and quick problem resolution'],
185
+ ['Delivery', 'Free for orders over €50 with real-time tracking'],
186
+ ['Returns', '30-day return policy with no fees or questions'],
187
+ ],
188
+ },
189
+ };
@@ -0,0 +1,138 @@
1
+ import React from 'react';
2
+ import './SimpleListBox.css';
3
+
4
+ export type SimpleListBoxProps = {
5
+ className?: string;
6
+ style?: React.CSSProperties;
7
+ title?: string;
8
+ titleClassName?: string;
9
+ titleStyle?: React.CSSProperties;
10
+ data?: (string | [string, string])[];
11
+ lineClasses?: { [index: number]: string };
12
+ renderCell?: (value: string, row?: number, col?: number) => React.ReactNode;
13
+ bg?: string;
14
+ titleBg?: string;
15
+ titleColor?: string;
16
+ itemBg?: string;
17
+ emptyContent?: React.ReactNode;
18
+ colGap?: string;
19
+ };
20
+
21
+ export const SimpleListBox: React.FC<SimpleListBoxProps> = ({
22
+ className = '',
23
+ style = {},
24
+ title,
25
+ titleClassName = '',
26
+ titleStyle,
27
+ data,
28
+ lineClasses = {},
29
+ renderCell,
30
+ bg,
31
+ titleBg,
32
+ titleColor,
33
+ itemBg,
34
+ emptyContent,
35
+ colGap = '1rem',
36
+ }) => {
37
+ const rows = Array.isArray(data) ? data : [];
38
+ const isEmpty = rows.length === 0;
39
+ const all1D = rows.every((r) => typeof r === 'string');
40
+ const has2D = rows.some((r) => Array.isArray(r));
41
+
42
+ const cellRenderer = (value: string, r?: number, c?: number) =>
43
+ renderCell ? renderCell(value, r, c) : <span className='slb-cell-text'>{value}</span>;
44
+
45
+ const gridStyle: React.CSSProperties | undefined = colGap
46
+ ? ({ ['--col-gap' as any]: colGap } as React.CSSProperties)
47
+ : undefined;
48
+
49
+ if (isEmpty) {
50
+ return (
51
+ <div className={`slb-root ${className}`} style={{ ...style, background: bg ?? undefined }}>
52
+ {title && (
53
+ <div
54
+ className={`slb-title ${titleClassName}`}
55
+ style={{
56
+ ...titleStyle,
57
+ background: titleBg ?? undefined,
58
+ color: titleColor ?? undefined,
59
+ }}
60
+ >
61
+ {title}
62
+ </div>
63
+ )}
64
+ <div className='slb-empty'>{emptyContent ?? null}</div>
65
+ </div>
66
+ );
67
+ }
68
+
69
+ if (all1D && !has2D) {
70
+ return (
71
+ <div className={`slb-root ${className}`} style={{ ...style, background: bg ?? undefined }}>
72
+ {title && (
73
+ <div
74
+ className={`slb-title ${titleClassName}`}
75
+ style={{ background: titleBg ?? undefined, color: titleColor ?? undefined }}
76
+ >
77
+ {title}
78
+ </div>
79
+ )}
80
+ <ul className='slb-list'>
81
+ {(rows as string[]).map((it, i) => (
82
+ <li
83
+ key={i}
84
+ className={`slb-item ${lineClasses[i] || ''}`}
85
+ style={{ background: itemBg ?? undefined }}
86
+ >
87
+ {cellRenderer(it, i, 0)}
88
+ </li>
89
+ ))}
90
+ </ul>
91
+ </div>
92
+ );
93
+ }
94
+
95
+ // Mixed mode: each row is a wrapper (ligne) that contains left/right cells.
96
+ return (
97
+ <div className={`slb-root ${className}`} style={{ ...style, background: bg ?? undefined }}>
98
+ {title && (
99
+ <div
100
+ className={`slb-title ${titleClassName}`}
101
+ style={{ background: titleBg ?? undefined, color: titleColor ?? undefined }}
102
+ >
103
+ {title}
104
+ </div>
105
+ )}
106
+
107
+ <div className='slb-grid' style={gridStyle}>
108
+ {rows.map((r, ridx) => {
109
+ const lineClass = lineClasses[ridx] || '';
110
+ if (Array.isArray(r)) {
111
+ const [left, right] = r as [string, string];
112
+ return (
113
+ <div
114
+ key={ridx}
115
+ className={`slb-row ${lineClass}`}
116
+ style={{ background: itemBg ?? undefined }}
117
+ >
118
+ <div className='slb-grid-left'>{cellRenderer(left, ridx, 0)}</div>
119
+ <div className='slb-grid-right'>{cellRenderer(right, ridx, 1)}</div>
120
+ </div>
121
+ );
122
+ }
123
+
124
+ return (
125
+ <div
126
+ key={ridx}
127
+ className={`slb-row ${lineClass}`}
128
+ style={{ background: itemBg ?? undefined }}
129
+ >
130
+ <div className='slb-grid-left'>{cellRenderer(r as string, ridx, 0)}</div>
131
+ <div className='slb-grid-right slb-empty-cell' aria-hidden='true' />
132
+ </div>
133
+ );
134
+ })}
135
+ </div>
136
+ </div>
137
+ );
138
+ };
@@ -0,0 +1,61 @@
1
+ .tlb-root {
2
+ padding: 0.125rem;
3
+ }
4
+ .tlb-title {
5
+ font-weight: var(--font-bold);
6
+ padding: 0.0625rem 0.125rem;
7
+ margin-bottom: 0.0625rem;
8
+ }
9
+
10
+ .tlb-grid {
11
+ display: flex;
12
+ flex-direction: column;
13
+ gap: 0.03125rem;
14
+ }
15
+
16
+ /* row wrapper */
17
+ .tlb-row {
18
+ display: grid;
19
+ grid-template-columns: minmax(max-content, 1fr) minmax(max-content, 1fr);
20
+ gap: var(--col-gap, 0.125rem);
21
+ align-items: center;
22
+ padding: 0;
23
+ }
24
+
25
+ .tlb-label {
26
+ padding: 0.0625rem 0 0.0625rem 0.0625rem;
27
+ min-width: 0;
28
+ justify-self: start;
29
+ }
30
+ .tlb-value-group {
31
+ display: flex;
32
+ align-items: center;
33
+ gap: 0.015625rem;
34
+ min-width: 0;
35
+ justify-self: center;
36
+ }
37
+ .tlb-value-wrap {
38
+ display: flex;
39
+ gap: 0.03125rem;
40
+ align-items: center;
41
+ min-width: 0;
42
+ }
43
+ .tlb-value,
44
+ .tlb-delta {
45
+ min-width: 0;
46
+ white-space: nowrap;
47
+ }
48
+ .tlb-icon-col {
49
+ display: flex;
50
+ align-items: center;
51
+ justify-content: center;
52
+ }
53
+ .tlb-delta.positive {
54
+ color: #15803d;
55
+ }
56
+ .tlb-delta.negative {
57
+ color: #dc2626;
58
+ }
59
+ .tlb-delta.neutral {
60
+ color: #6b7280;
61
+ }