@stack-spot/portal-components 2.27.0 → 2.27.2

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 (250) hide show
  1. package/CHANGELOG.md +635 -621
  2. package/dist/components/AnimatedHeight.d.ts +1 -1
  3. package/dist/components/AnimatedHeight.js +26 -26
  4. package/dist/components/AsyncContent.d.ts +1 -1
  5. package/dist/components/AsyncContent.js +1 -1
  6. package/dist/components/BannerWarning.d.ts +1 -1
  7. package/dist/components/BannerWarning.js +1 -1
  8. package/dist/components/Breadcrumb/index.d.ts +2 -2
  9. package/dist/components/Breadcrumb/index.js +1 -1
  10. package/dist/components/Breadcrumb/styled.js +31 -31
  11. package/dist/components/ButtonLoading.d.ts +1 -1
  12. package/dist/components/ButtonLoading.js +1 -1
  13. package/dist/components/ChatBot.d.ts +1 -1
  14. package/dist/components/ChatBot.js +1 -1
  15. package/dist/components/ContentValidateFilter.d.ts +1 -1
  16. package/dist/components/ContentValidateFilter.js +1 -1
  17. package/dist/components/FadingOverflow.d.ts +1 -1
  18. package/dist/components/FadingOverflow.js +69 -69
  19. package/dist/components/FileTreeView/More.d.ts +1 -1
  20. package/dist/components/FileTreeView/More.js +1 -1
  21. package/dist/components/FileTreeView/index.d.ts +1 -1
  22. package/dist/components/FileTreeView/index.js +1 -1
  23. package/dist/components/InfiniteScroll.d.ts +1 -1
  24. package/dist/components/InfiniteScroll.js +1 -1
  25. package/dist/components/InfoMaintenanceBanner.d.ts +1 -1
  26. package/dist/components/InfoMaintenanceBanner.js +2 -2
  27. package/dist/components/LazyMarkdown/BlockquoteMd.d.ts +1 -1
  28. package/dist/components/LazyMarkdown/BlockquoteMd.js +1 -1
  29. package/dist/components/LazyMarkdown/CodeViewer.d.ts +1 -1
  30. package/dist/components/LazyMarkdown/CodeViewer.js +76 -76
  31. package/dist/components/LazyMarkdown/Markdown.d.ts +1 -1
  32. package/dist/components/LazyMarkdown/Markdown.js +1 -1
  33. package/dist/components/LazyMarkdown/MarkdownButton.d.ts +1 -1
  34. package/dist/components/LazyMarkdown/MarkdownButton.js +1 -1
  35. package/dist/components/LazyMarkdown/Video.d.ts +1 -1
  36. package/dist/components/LazyMarkdown/Video.js +1 -1
  37. package/dist/components/LazyMarkdown/index.d.ts +1 -1
  38. package/dist/components/LazyMarkdown/index.js +1 -1
  39. package/dist/components/Placeholder.d.ts +7 -3
  40. package/dist/components/Placeholder.d.ts.map +1 -1
  41. package/dist/components/Placeholder.js +3 -3
  42. package/dist/components/Placeholder.js.map +1 -1
  43. package/dist/components/ScrollView.js +16 -16
  44. package/dist/components/Select/BadgeItem.d.ts +1 -1
  45. package/dist/components/Select/BadgeItem.js +1 -1
  46. package/dist/components/Select/ClearInput.d.ts +1 -1
  47. package/dist/components/Select/ClearInput.js +1 -1
  48. package/dist/components/Select/CloseItem.d.ts +1 -1
  49. package/dist/components/Select/CloseItem.js +1 -1
  50. package/dist/components/Select/CreatableSelect.js +1 -1
  51. package/dist/components/Select/CustomMenu.d.ts +1 -1
  52. package/dist/components/Select/CustomMenu.js +1 -1
  53. package/dist/components/Select/LabelItem.d.ts +1 -1
  54. package/dist/components/Select/LabelItem.js +1 -1
  55. package/dist/components/Select/MultiValue.d.ts +1 -1
  56. package/dist/components/Select/MultiValue.js +1 -1
  57. package/dist/components/Select/SelectInfiniteScroll.d.ts +1 -1
  58. package/dist/components/Select/SelectInfiniteScroll.js +1 -1
  59. package/dist/components/Select/SelectSearch.d.ts +1 -1
  60. package/dist/components/Select/SelectSearch.js +1 -1
  61. package/dist/components/SelectionList.d.ts +1 -1
  62. package/dist/components/SelectionList.js +61 -61
  63. package/dist/components/StatusCircle.d.ts +1 -1
  64. package/dist/components/StatusCircle.js +6 -6
  65. package/dist/components/Stepper/Navigation.js +4 -4
  66. package/dist/components/Stepper/Step.js +3 -3
  67. package/dist/components/Stepper/Stepper.js +6 -6
  68. package/dist/components/Stepper/headers.js +22 -22
  69. package/dist/components/Table/HeaderItem.js +1 -1
  70. package/dist/components/Table/SettingsVerticalMenu.d.ts +1 -1
  71. package/dist/components/Table/SettingsVerticalMenu.js +1 -1
  72. package/dist/components/Table/StyledLinkTable.d.ts +1 -1
  73. package/dist/components/Table/StyledLinkTable.js +5 -5
  74. package/dist/components/Table/TableData.d.ts +1 -1
  75. package/dist/components/Table/TableData.js +25 -25
  76. package/dist/components/TimelineSection.d.ts +1 -1
  77. package/dist/components/TimelineSection.js +14 -14
  78. package/dist/components/error/ErrorFeedback.d.ts +1 -1
  79. package/dist/components/error/ErrorFeedback.js +35 -35
  80. package/dist/components/error/NotFound.d.ts +1 -1
  81. package/dist/components/error/NotFound.js +1 -1
  82. package/dist/components/error/UnderMaintenance.d.ts +1 -1
  83. package/dist/components/error/UnderMaintenance.js +1 -1
  84. package/dist/components/form/Form/Form.d.ts +1 -1
  85. package/dist/components/form/Form/Form.js +1 -1
  86. package/dist/components/form/Form/FormGroup.d.ts +2 -2
  87. package/dist/components/form/Form/FormGroup.js +1 -1
  88. package/dist/components/form/SearchInput.d.ts +1 -1
  89. package/dist/components/form/SearchInput.js +1 -1
  90. package/dist/components/form/Select/CustomSelect.d.ts +1 -1
  91. package/dist/components/form/Select/CustomSelect.js +1 -1
  92. package/dist/components/form/Select/DetailedSelect.d.ts +1 -1
  93. package/dist/components/form/Select/DetailedSelect.js +1 -1
  94. package/dist/components/form/Select/Select.d.ts +1 -1
  95. package/dist/components/form/Select/Select.js +1 -1
  96. package/dist/components/form/Select/styled.js +161 -161
  97. package/dist/components/form/Select/utils.js +1 -1
  98. package/dist/components/notification/NotificationComponent.d.ts +1 -1
  99. package/dist/components/notification/NotificationComponent.js +54 -54
  100. package/dist/components/notification/NotificationItem.d.ts +1 -1
  101. package/dist/components/notification/NotificationItem.d.ts.map +1 -1
  102. package/dist/components/notification/NotificationItem.js +11 -5
  103. package/dist/components/notification/NotificationItem.js.map +1 -1
  104. package/dist/components/notification/NotificationList.d.ts +1 -1
  105. package/dist/components/notification/NotificationList.d.ts.map +1 -1
  106. package/dist/components/notification/NotificationList.js +44 -44
  107. package/dist/components/notification/NotificationList.js.map +1 -1
  108. package/dist/components/notification/NotificationPlaceholder.d.ts +1 -1
  109. package/dist/components/notification/NotificationPlaceholder.d.ts.map +1 -1
  110. package/dist/components/notification/NotificationPlaceholder.js +2 -2
  111. package/dist/components/notification/NotificationPlaceholder.js.map +1 -1
  112. package/dist/containers/NotificationsPage.d.ts +1 -1
  113. package/dist/containers/NotificationsPage.d.ts.map +1 -1
  114. package/dist/containers/NotificationsPage.js +24 -11
  115. package/dist/containers/NotificationsPage.js.map +1 -1
  116. package/dist/context/anchor.d.ts +1 -1
  117. package/dist/context/anchor.js +1 -1
  118. package/dist/context/loading.d.ts +1 -1
  119. package/dist/context/loading.js +1 -1
  120. package/dist/context/notification/context.d.ts +1 -1
  121. package/dist/context/notification/context.js +1 -1
  122. package/dist/context/notification/types.d.ts +1 -0
  123. package/dist/context/notification/types.d.ts.map +1 -1
  124. package/dist/hooks/date.js +1 -1
  125. package/dist/hooks/service-now.js +28 -28
  126. package/dist/svg/AI.d.ts +1 -1
  127. package/dist/svg/AI.js +1 -1
  128. package/dist/svg/CS.d.ts +1 -1
  129. package/dist/svg/CS.js +1 -1
  130. package/dist/svg/EDP.d.ts +1 -1
  131. package/dist/svg/EDP.js +1 -1
  132. package/dist/svg/Forbidden.d.ts +1 -1
  133. package/dist/svg/Forbidden.js +1 -1
  134. package/dist/svg/GenericPlaceholder.d.ts +4 -2
  135. package/dist/svg/GenericPlaceholder.d.ts.map +1 -1
  136. package/dist/svg/GenericPlaceholder.js +2 -2
  137. package/dist/svg/GenericPlaceholder.js.map +1 -1
  138. package/dist/svg/HUB.d.ts +1 -1
  139. package/dist/svg/HUB.js +1 -1
  140. package/dist/svg/Logo.d.ts +1 -1
  141. package/dist/svg/Logo.js +1 -1
  142. package/dist/svg/MiniLogo.d.ts +1 -1
  143. package/dist/svg/MiniLogo.js +1 -1
  144. package/dist/svg/NotFound.d.ts +1 -1
  145. package/dist/svg/NotFound.js +1 -1
  146. package/dist/svg/ServerError.d.ts +1 -1
  147. package/dist/svg/ServerError.js +1 -1
  148. package/dist/svg/Unauthenticated.d.ts +1 -1
  149. package/dist/svg/Unauthenticated.js +1 -1
  150. package/package.json +6 -6
  151. package/readme.md +66 -66
  152. package/src/components/AnimatedHeight.tsx +174 -174
  153. package/src/components/AsyncContent.tsx +78 -78
  154. package/src/components/BannerWarning.tsx +91 -91
  155. package/src/components/Breadcrumb/index.tsx +76 -76
  156. package/src/components/Breadcrumb/styled.ts +37 -37
  157. package/src/components/ButtonLoading.tsx +29 -29
  158. package/src/components/ChatBot.tsx +82 -82
  159. package/src/components/ContentValidateFilter.tsx +15 -15
  160. package/src/components/FadingOverflow.tsx +265 -265
  161. package/src/components/FileTreeView/More.tsx +114 -114
  162. package/src/components/FileTreeView/index.tsx +186 -186
  163. package/src/components/InfiniteScroll.tsx +24 -24
  164. package/src/components/InfoMaintenanceBanner.tsx +29 -29
  165. package/src/components/LazyMarkdown/BlockquoteMd.tsx +107 -107
  166. package/src/components/LazyMarkdown/CodeViewer.tsx +161 -161
  167. package/src/components/LazyMarkdown/Markdown.tsx +122 -122
  168. package/src/components/LazyMarkdown/MarkdownButton.tsx +24 -24
  169. package/src/components/LazyMarkdown/Video.tsx +13 -13
  170. package/src/components/LazyMarkdown/index.tsx +21 -21
  171. package/src/components/Placeholder.tsx +123 -118
  172. package/src/components/ScrollView.tsx +57 -57
  173. package/src/components/Select/BadgeItem.tsx +58 -58
  174. package/src/components/Select/ClearInput.tsx +24 -24
  175. package/src/components/Select/CloseItem.tsx +38 -38
  176. package/src/components/Select/CreatableSelect.tsx +155 -155
  177. package/src/components/Select/CustomMenu.tsx +16 -16
  178. package/src/components/Select/LabelItem.tsx +8 -8
  179. package/src/components/Select/MultiValue.tsx +49 -49
  180. package/src/components/Select/SelectInfiniteScroll.tsx +82 -82
  181. package/src/components/Select/SelectSearch.tsx +195 -195
  182. package/src/components/Select/index.tsx +7 -7
  183. package/src/components/Select/types.ts +8 -8
  184. package/src/components/SelectionList.tsx +427 -427
  185. package/src/components/StatusCircle.tsx +67 -67
  186. package/src/components/Stepper/Navigation.tsx +97 -97
  187. package/src/components/Stepper/Step.tsx +30 -30
  188. package/src/components/Stepper/Stepper.tsx +113 -113
  189. package/src/components/Stepper/headers.tsx +64 -64
  190. package/src/components/Stepper/index.ts +3 -3
  191. package/src/components/Table/HeaderItem.tsx +52 -52
  192. package/src/components/Table/SettingsVerticalMenu.tsx +50 -50
  193. package/src/components/Table/StyledLinkTable.tsx +22 -22
  194. package/src/components/Table/TableData.tsx +251 -251
  195. package/src/components/Table/index.tsx +2 -2
  196. package/src/components/TimelineSection.tsx +66 -66
  197. package/src/components/error/ErrorFeedback.tsx +217 -217
  198. package/src/components/error/NotFound.tsx +24 -24
  199. package/src/components/error/UnderMaintenance.tsx +30 -30
  200. package/src/components/error/index.ts +4 -4
  201. package/src/components/form/Form/Form.tsx +101 -101
  202. package/src/components/form/Form/FormGroup.tsx +221 -221
  203. package/src/components/form/Form/index.ts +2 -2
  204. package/src/components/form/SearchInput.tsx +69 -69
  205. package/src/components/form/Select/CustomSelect.tsx +232 -232
  206. package/src/components/form/Select/DetailedSelect.tsx +85 -85
  207. package/src/components/form/Select/Select.tsx +67 -67
  208. package/src/components/form/Select/index.ts +4 -4
  209. package/src/components/form/Select/styled.ts +165 -165
  210. package/src/components/form/Select/types.ts +112 -112
  211. package/src/components/form/Select/utils.tsx +28 -28
  212. package/src/components/notification/NotificationComponent.tsx +340 -340
  213. package/src/components/notification/NotificationItem.tsx +345 -336
  214. package/src/components/notification/NotificationList.tsx +179 -178
  215. package/src/components/notification/NotificationPlaceholder.tsx +44 -43
  216. package/src/components/notification/types.ts +72 -72
  217. package/src/containers/NotificationsPage.tsx +119 -98
  218. package/src/context/anchor.tsx +37 -37
  219. package/src/context/loading.tsx +36 -36
  220. package/src/context/notification/LazyNotificationList.ts +103 -103
  221. package/src/context/notification/NotificationController.ts +104 -104
  222. package/src/context/notification/context.tsx +23 -23
  223. package/src/context/notification/hooks.ts +98 -98
  224. package/src/context/notification/types.ts +66 -65
  225. package/src/hooks/date.ts +31 -31
  226. package/src/hooks/keyboard.tsx +128 -128
  227. package/src/hooks/manual-render.tsx +10 -10
  228. package/src/hooks/service-now.tsx +233 -233
  229. package/src/hooks/text.tsx +30 -30
  230. package/src/hooks/title.tsx +28 -28
  231. package/src/hooks/use-effect-once.tsx +43 -43
  232. package/src/index.ts +19 -19
  233. package/src/notifications.ts +11 -11
  234. package/src/svg/AI.tsx +41 -41
  235. package/src/svg/CS.tsx +48 -48
  236. package/src/svg/EDP.tsx +31 -31
  237. package/src/svg/Forbidden.tsx +22 -22
  238. package/src/svg/GenericPlaceholder.tsx +20 -20
  239. package/src/svg/HUB.tsx +48 -48
  240. package/src/svg/Logo.tsx +16 -16
  241. package/src/svg/MiniLogo.tsx +12 -12
  242. package/src/svg/NotFound.tsx +16 -16
  243. package/src/svg/ServerError.tsx +33 -33
  244. package/src/svg/Unauthenticated.tsx +16 -16
  245. package/src/svg/index.ts +11 -11
  246. package/src/utils/accessibility.ts +135 -135
  247. package/src/utils/cookie.ts +73 -73
  248. package/src/utils/promise.ts +5 -5
  249. package/src/utils/read-file.ts +16 -16
  250. package/tsconfig.json +10 -10
@@ -1,251 +1,251 @@
1
- import { Button, IconBox, Text } from '@citric/core'
2
- import { SxProperties } from '@citric/core/dist/sx'
3
- import { Trash } from '@citric/icons'
4
- import { LoadingCircular, Table, Tbody, Td, Th, Thead, Tr } from '@citric/ui'
5
- import { SelectionList } from '@stack-spot/portal-components/SelectionList'
6
- import { theme } from '@stack-spot/portal-theme'
7
- import { useTranslate } from '@stack-spot/portal-translate'
8
- import { useState } from 'react'
9
- import styled from 'styled-components'
10
- import { ContentValidateFilter } from '../ContentValidateFilter'
11
- import { HeaderSort, SortTypes } from './HeaderItem'
12
- import { SettingsVerticalMenu } from './SettingsVerticalMenu'
13
-
14
- export type TableColumnsDefault = 'name' | 'description'
15
-
16
- export type ColumnDefinition<T, K extends keyof T> = {
17
- key: K,
18
- header: string,
19
- sortable?: boolean,
20
- customRender?: (data: T) => JSX.Element | string | null,
21
- sx?: SxProperties,
22
- width?: string,
23
- }
24
-
25
- type SortDetailsType<T, K extends keyof T> = {
26
- sort: K,
27
- direction: SortTypes,
28
- index?: number,
29
- }
30
-
31
- type TableHeaderProps<T, K extends keyof T> = {
32
- columns: ColumnDefinition<T, K>[],
33
- onSort?: ({ sort, direction, index }: SortDetailsType<T, K>) => void,
34
- onRemove?: (row: T) => void,
35
- sxTh?: SxProperties,
36
- hasBorder?: boolean,
37
- initialSortDetails?: Omit<SortDetailsType<T, K>, 'sort'>,
38
- }
39
-
40
- const StyledTh = styled(Th)<{ $hasBorder?: boolean }>`
41
- border-top: ${({ $hasBorder }) => ($hasBorder ? `1px solid ${theme.color.light[400]}` : '')};
42
- border-bottom: ${({ $hasBorder }) => ($hasBorder ? `1px solid ${theme.color.light[400]}` : '')};
43
- &:first-child {
44
- border-left: ${({ $hasBorder }) => ($hasBorder ? `1px solid ${theme.color.light[400]}` : '')};
45
- border-top-left-radius: 4px;
46
- border-bottom-left-radius: 4px;
47
- }
48
- &:last-child {
49
- border-right: ${({ $hasBorder }) => ($hasBorder ? `1px solid ${theme.color.light[400]}` : '')};
50
- border-top-right-radius: 4px;
51
- border-bottom-right-radius: 4px;
52
- }
53
- `
54
-
55
- const StyledTd = styled(Td)<{ $hasBorder?: boolean }>`
56
- border-radius: 0px !important;
57
-
58
- &:first-child {
59
- border-top-left-radius: 4px !important;
60
- border-bottom-left-radius: 4px !important;
61
- }
62
- &:last-child {
63
- border-top-right-radius: 4px !important;
64
- border-bottom-right-radius: 4px !important;
65
- }
66
- `
67
-
68
- const TableHeader = <T, K extends keyof T>({ columns, onSort, onRemove, sxTh, hasBorder, initialSortDetails }: TableHeaderProps<T, K>) => {
69
- const [sort, setSort] = useState<SortTypes>(initialSortDetails?.direction)
70
- const [activeSort, setActiveSort] = useState<number | undefined>(initialSortDetails?.index)
71
-
72
- const Sortable = ({ sort, direction, index }: { sort: K, direction: SortTypes, index: number }) => {
73
- onSort?.({ sort, direction, index })
74
- setSort(direction)
75
- setActiveSort(index)
76
- }
77
-
78
- return (
79
- <Thead>
80
- <Tr>
81
- {columns.map((column, index) => column?.sortable && onSort ? (
82
- <StyledTh key={column.key.toString()} sx={sxTh} $hasBorder={hasBorder}>
83
- <HeaderSort
84
- sortType={activeSort !== index ? undefined : sort}
85
- onSort={(sortType) => Sortable?.({ sort: column.key, direction: sortType, index })}
86
- >
87
- {column.header}
88
- </HeaderSort>
89
- </StyledTh>
90
- ) : (
91
- <StyledTh key={column.key.toString()} sx={sxTh} $hasBorder={hasBorder}>
92
- <Text appearance="microtext1" mr="2">
93
- {column.header}
94
- </Text>
95
- </StyledTh>
96
- ))}
97
- {onRemove && <Th />}
98
- </Tr>
99
- </Thead>
100
- )
101
- }
102
-
103
- type TableRowsProps<T, K extends keyof T> = {
104
- data: T[],
105
- columns: ColumnDefinition<T, K>[],
106
- onRemove?: (row: T) => void,
107
- canRemove?: (row: T) => boolean,
108
- options?: React.ComponentProps<typeof SelectionList>['items'],
109
- onClickRow?: (data: T) => void,
110
- }
111
-
112
- interface RemoveButtonProps<T> {
113
- row: T,
114
- onRemove?: (row: T) => void,
115
- canRemove?: (row: T) => boolean,
116
- }
117
-
118
- function RemoveButton<T>({ row, onRemove, canRemove }: RemoveButtonProps<T>) {
119
- const t = useTranslate(dictionary)
120
- const [isLoading, setLoading] = useState(false)
121
-
122
- if (!onRemove || (canRemove && !canRemove(row))) {
123
- return null
124
- }
125
-
126
- const handleRemove = async () => {
127
- if (isLoading) return
128
- setLoading(true)
129
- try {
130
- await onRemove(row)
131
- } finally {
132
- setLoading(false)
133
- }
134
- }
135
-
136
- return (
137
- <Button colorScheme="danger" onClick={handleRemove} size="sm">
138
- {isLoading ? <LoadingCircular/> : <>
139
- <IconBox size="xs" mr={2} colorIcon="danger">
140
- <Trash />
141
- </IconBox><Text appearance="microtext1" mr="2">
142
- {t.remove}
143
- </Text>
144
- </>}
145
- </Button>
146
- )
147
- }
148
-
149
- const TableRows = <T, K extends keyof T>({ data, columns, options, onRemove, canRemove }: TableRowsProps<T, K>) => {
150
- const rows = data.map((row, index) => (
151
- <Tr key={`row-${index}`}>
152
- {columns.map((column, columnIndex) => (
153
- <StyledTd
154
- key={columnIndex}
155
- sx={{ ...column.sx }}>
156
- <> {column?.customRender ? column.customRender(row) : row[column.key]}</>
157
- </StyledTd>
158
- ))}
159
-
160
- {canRemove && <StyledTd>
161
- <RemoveButton row={row} onRemove={onRemove} canRemove={canRemove} />
162
- </StyledTd>}
163
- {options && <StyledTd>
164
- <SettingsVerticalMenu
165
- id="workflow-studio-menu"
166
- items={options}
167
- size="xs"
168
- />
169
- </StyledTd>}
170
- </Tr>
171
- ))
172
-
173
- return <Tbody sx={{ borderSpacing: '0 8px' }}>{rows}</Tbody>
174
- }
175
-
176
- type TableProps<T, K extends keyof T> = {
177
- data: T[],
178
- columns: ColumnDefinition<T, K>[],
179
- onSort?: ({ sort, direction, index }: SortDetailsType<T, K>) => void,
180
- onRemove?: (data: T) => void,
181
- canRemove?: (row: T) => boolean,
182
- tableAppearance?: 'striped' | 'filled',
183
- options?: React.ComponentProps<typeof SelectionList>['items'],
184
- sxHeader?: SxProperties,
185
- hasHeaderBorder?: boolean,
186
- searchParamName?: string,
187
- initialSortDetails?: Omit<SortDetailsType<T, K>, 'sort'>,
188
- /**
189
- * Accepts a react element that is rendered below the table, used mostly for pagination.
190
- */
191
- afterContent?: React.ReactElement,
192
- }
193
-
194
- export const TableData = <T, K extends keyof T>({
195
- data,
196
- columns,
197
- onSort,
198
- onRemove,
199
- canRemove,
200
- tableAppearance = 'striped',
201
- options,
202
- sxHeader,
203
- hasHeaderBorder,
204
- searchParamName = 'search',
205
- initialSortDetails,
206
- afterContent,
207
- }: TableProps<T, K>) => {
208
- const searchParams = new URLSearchParams(window.location.search)
209
- const search = searchParams.get(searchParamName) ?? ''
210
-
211
- return (
212
- <ContentValidateFilter listLength={data.length} searchText={search}>
213
- {!!data.length && (
214
- <>
215
- <Table appearance={tableAppearance} sx={{ mb: 5 }}>
216
- <colgroup>
217
- {columns.map((column, index) => (
218
- <col key={index} style={{ width: column.width || 'auto' }} />
219
- ))}
220
- </colgroup>
221
- <TableHeader
222
- sxTh={sxHeader}
223
- columns={columns}
224
- onSort={onSort}
225
- onRemove={onRemove}
226
- hasBorder={tableAppearance === 'striped' ? false : hasHeaderBorder}
227
- initialSortDetails={initialSortDetails}
228
- />
229
- <TableRows
230
- data={data}
231
- columns={columns}
232
- onRemove={onRemove}
233
- canRemove={canRemove}
234
- options={options}
235
- />
236
- </Table>
237
- {!!afterContent && afterContent}
238
- </>
239
- )}
240
- </ContentValidateFilter>
241
- )
242
- }
243
-
244
- const dictionary = {
245
- en: {
246
- remove: 'Remove',
247
- },
248
- pt: {
249
- remove: 'Remover',
250
- },
251
- }
1
+ import { Button, IconBox, Text } from '@citric/core'
2
+ import { SxProperties } from '@citric/core/dist/sx'
3
+ import { Trash } from '@citric/icons'
4
+ import { LoadingCircular, Table, Tbody, Td, Th, Thead, Tr } from '@citric/ui'
5
+ import { SelectionList } from '@stack-spot/portal-components/SelectionList'
6
+ import { theme } from '@stack-spot/portal-theme'
7
+ import { useTranslate } from '@stack-spot/portal-translate'
8
+ import { useState } from 'react'
9
+ import styled from 'styled-components'
10
+ import { ContentValidateFilter } from '../ContentValidateFilter'
11
+ import { HeaderSort, SortTypes } from './HeaderItem'
12
+ import { SettingsVerticalMenu } from './SettingsVerticalMenu'
13
+
14
+ export type TableColumnsDefault = 'name' | 'description'
15
+
16
+ export type ColumnDefinition<T, K extends keyof T> = {
17
+ key: K,
18
+ header: string,
19
+ sortable?: boolean,
20
+ customRender?: (data: T) => JSX.Element | string | null,
21
+ sx?: SxProperties,
22
+ width?: string,
23
+ }
24
+
25
+ type SortDetailsType<T, K extends keyof T> = {
26
+ sort: K,
27
+ direction: SortTypes,
28
+ index?: number,
29
+ }
30
+
31
+ type TableHeaderProps<T, K extends keyof T> = {
32
+ columns: ColumnDefinition<T, K>[],
33
+ onSort?: ({ sort, direction, index }: SortDetailsType<T, K>) => void,
34
+ onRemove?: (row: T) => void,
35
+ sxTh?: SxProperties,
36
+ hasBorder?: boolean,
37
+ initialSortDetails?: Omit<SortDetailsType<T, K>, 'sort'>,
38
+ }
39
+
40
+ const StyledTh = styled(Th)<{ $hasBorder?: boolean }>`
41
+ border-top: ${({ $hasBorder }) => ($hasBorder ? `1px solid ${theme.color.light[400]}` : '')};
42
+ border-bottom: ${({ $hasBorder }) => ($hasBorder ? `1px solid ${theme.color.light[400]}` : '')};
43
+ &:first-child {
44
+ border-left: ${({ $hasBorder }) => ($hasBorder ? `1px solid ${theme.color.light[400]}` : '')};
45
+ border-top-left-radius: 4px;
46
+ border-bottom-left-radius: 4px;
47
+ }
48
+ &:last-child {
49
+ border-right: ${({ $hasBorder }) => ($hasBorder ? `1px solid ${theme.color.light[400]}` : '')};
50
+ border-top-right-radius: 4px;
51
+ border-bottom-right-radius: 4px;
52
+ }
53
+ `
54
+
55
+ const StyledTd = styled(Td)<{ $hasBorder?: boolean }>`
56
+ border-radius: 0px !important;
57
+
58
+ &:first-child {
59
+ border-top-left-radius: 4px !important;
60
+ border-bottom-left-radius: 4px !important;
61
+ }
62
+ &:last-child {
63
+ border-top-right-radius: 4px !important;
64
+ border-bottom-right-radius: 4px !important;
65
+ }
66
+ `
67
+
68
+ const TableHeader = <T, K extends keyof T>({ columns, onSort, onRemove, sxTh, hasBorder, initialSortDetails }: TableHeaderProps<T, K>) => {
69
+ const [sort, setSort] = useState<SortTypes>(initialSortDetails?.direction)
70
+ const [activeSort, setActiveSort] = useState<number | undefined>(initialSortDetails?.index)
71
+
72
+ const Sortable = ({ sort, direction, index }: { sort: K, direction: SortTypes, index: number }) => {
73
+ onSort?.({ sort, direction, index })
74
+ setSort(direction)
75
+ setActiveSort(index)
76
+ }
77
+
78
+ return (
79
+ <Thead>
80
+ <Tr>
81
+ {columns.map((column, index) => column?.sortable && onSort ? (
82
+ <StyledTh key={column.key.toString()} sx={sxTh} $hasBorder={hasBorder}>
83
+ <HeaderSort
84
+ sortType={activeSort !== index ? undefined : sort}
85
+ onSort={(sortType) => Sortable?.({ sort: column.key, direction: sortType, index })}
86
+ >
87
+ {column.header}
88
+ </HeaderSort>
89
+ </StyledTh>
90
+ ) : (
91
+ <StyledTh key={column.key.toString()} sx={sxTh} $hasBorder={hasBorder}>
92
+ <Text appearance="microtext1" mr="2">
93
+ {column.header}
94
+ </Text>
95
+ </StyledTh>
96
+ ))}
97
+ {onRemove && <Th />}
98
+ </Tr>
99
+ </Thead>
100
+ )
101
+ }
102
+
103
+ type TableRowsProps<T, K extends keyof T> = {
104
+ data: T[],
105
+ columns: ColumnDefinition<T, K>[],
106
+ onRemove?: (row: T) => void,
107
+ canRemove?: (row: T) => boolean,
108
+ options?: React.ComponentProps<typeof SelectionList>['items'],
109
+ onClickRow?: (data: T) => void,
110
+ }
111
+
112
+ interface RemoveButtonProps<T> {
113
+ row: T,
114
+ onRemove?: (row: T) => void,
115
+ canRemove?: (row: T) => boolean,
116
+ }
117
+
118
+ function RemoveButton<T>({ row, onRemove, canRemove }: RemoveButtonProps<T>) {
119
+ const t = useTranslate(dictionary)
120
+ const [isLoading, setLoading] = useState(false)
121
+
122
+ if (!onRemove || (canRemove && !canRemove(row))) {
123
+ return null
124
+ }
125
+
126
+ const handleRemove = async () => {
127
+ if (isLoading) return
128
+ setLoading(true)
129
+ try {
130
+ await onRemove(row)
131
+ } finally {
132
+ setLoading(false)
133
+ }
134
+ }
135
+
136
+ return (
137
+ <Button colorScheme="danger" onClick={handleRemove} size="sm">
138
+ {isLoading ? <LoadingCircular/> : <>
139
+ <IconBox size="xs" mr={2} colorIcon="danger">
140
+ <Trash />
141
+ </IconBox><Text appearance="microtext1" mr="2">
142
+ {t.remove}
143
+ </Text>
144
+ </>}
145
+ </Button>
146
+ )
147
+ }
148
+
149
+ const TableRows = <T, K extends keyof T>({ data, columns, options, onRemove, canRemove }: TableRowsProps<T, K>) => {
150
+ const rows = data.map((row, index) => (
151
+ <Tr key={`row-${index}`}>
152
+ {columns.map((column, columnIndex) => (
153
+ <StyledTd
154
+ key={columnIndex}
155
+ sx={{ ...column.sx }}>
156
+ <> {column?.customRender ? column.customRender(row) : row[column.key]}</>
157
+ </StyledTd>
158
+ ))}
159
+
160
+ {canRemove && <StyledTd>
161
+ <RemoveButton row={row} onRemove={onRemove} canRemove={canRemove} />
162
+ </StyledTd>}
163
+ {options && <StyledTd>
164
+ <SettingsVerticalMenu
165
+ id="workflow-studio-menu"
166
+ items={options}
167
+ size="xs"
168
+ />
169
+ </StyledTd>}
170
+ </Tr>
171
+ ))
172
+
173
+ return <Tbody sx={{ borderSpacing: '0 8px' }}>{rows}</Tbody>
174
+ }
175
+
176
+ type TableProps<T, K extends keyof T> = {
177
+ data: T[],
178
+ columns: ColumnDefinition<T, K>[],
179
+ onSort?: ({ sort, direction, index }: SortDetailsType<T, K>) => void,
180
+ onRemove?: (data: T) => void,
181
+ canRemove?: (row: T) => boolean,
182
+ tableAppearance?: 'striped' | 'filled',
183
+ options?: React.ComponentProps<typeof SelectionList>['items'],
184
+ sxHeader?: SxProperties,
185
+ hasHeaderBorder?: boolean,
186
+ searchParamName?: string,
187
+ initialSortDetails?: Omit<SortDetailsType<T, K>, 'sort'>,
188
+ /**
189
+ * Accepts a react element that is rendered below the table, used mostly for pagination.
190
+ */
191
+ afterContent?: React.ReactElement,
192
+ }
193
+
194
+ export const TableData = <T, K extends keyof T>({
195
+ data,
196
+ columns,
197
+ onSort,
198
+ onRemove,
199
+ canRemove,
200
+ tableAppearance = 'striped',
201
+ options,
202
+ sxHeader,
203
+ hasHeaderBorder,
204
+ searchParamName = 'search',
205
+ initialSortDetails,
206
+ afterContent,
207
+ }: TableProps<T, K>) => {
208
+ const searchParams = new URLSearchParams(window.location.search)
209
+ const search = searchParams.get(searchParamName) ?? ''
210
+
211
+ return (
212
+ <ContentValidateFilter listLength={data.length} searchText={search}>
213
+ {!!data.length && (
214
+ <>
215
+ <Table appearance={tableAppearance} sx={{ mb: 5 }}>
216
+ <colgroup>
217
+ {columns.map((column, index) => (
218
+ <col key={index} style={{ width: column.width || 'auto' }} />
219
+ ))}
220
+ </colgroup>
221
+ <TableHeader
222
+ sxTh={sxHeader}
223
+ columns={columns}
224
+ onSort={onSort}
225
+ onRemove={onRemove}
226
+ hasBorder={tableAppearance === 'striped' ? false : hasHeaderBorder}
227
+ initialSortDetails={initialSortDetails}
228
+ />
229
+ <TableRows
230
+ data={data}
231
+ columns={columns}
232
+ onRemove={onRemove}
233
+ canRemove={canRemove}
234
+ options={options}
235
+ />
236
+ </Table>
237
+ {!!afterContent && afterContent}
238
+ </>
239
+ )}
240
+ </ContentValidateFilter>
241
+ )
242
+ }
243
+
244
+ const dictionary = {
245
+ en: {
246
+ remove: 'Remove',
247
+ },
248
+ pt: {
249
+ remove: 'Remover',
250
+ },
251
+ }
@@ -1,2 +1,2 @@
1
- export { StyledLinkTable } from '../Table/StyledLinkTable'
2
- export { TableData, ColumnDefinition } from '../Table/TableData'
1
+ export { StyledLinkTable } from '../Table/StyledLinkTable'
2
+ export { TableData, ColumnDefinition } from '../Table/TableData'
@@ -1,66 +1,66 @@
1
- import { Text } from '@citric/core'
2
- import { Month, useMonthDictionary, useTranslate } from '@stack-spot/portal-translate'
3
- import { styled } from 'styled-components'
4
-
5
- interface Props {
6
- /**
7
- * The month. 0 for January, 11 for December.
8
- */
9
- month: Month,
10
- /**
11
- * The day of the month (1 - 31).
12
- */
13
- day: number,
14
- /**
15
- * The section's content.
16
- */
17
- children: React.ReactNode,
18
- style?: React.CSSProperties,
19
- className?: string,
20
- }
21
-
22
- const Styled = styled.div`
23
- display: flex;
24
- flex-direction: row;
25
- align-items: start;
26
-
27
- .timeline-section-header {
28
- width: 80px;
29
- }
30
-
31
- .timeline-section-content {
32
- min-width: 0;
33
- flex: 1;
34
- }
35
- `
36
-
37
- /**
38
- * Renders a section in a timeline where the left side is a header with the date and the right side is the content (children).
39
- * @param props React props: {@link Props}.
40
- */
41
- export const TimelineSection = ({ children, day, month, className, style }: Props) => {
42
- const months = useMonthDictionary()
43
- const t = useTranslate(locale)
44
-
45
- const ariaLabel = t.dayMonth.replace('%d', `${day}`).replace('%m', months[month])
46
- return (
47
- <Styled style={style} className={className}>
48
- <div className="timeline-section-header">
49
- <Text appearance="h2" aria-label={ariaLabel}>{day >= 10 ? day : `0${day}`}</Text>
50
- <Text appearance="subtitle4" aria-hidden="true">{months[month].toUpperCase().substring(0, 3)}</Text>
51
- </div>
52
- <div className="timeline-section-content">
53
- {children}
54
- </div>
55
- </Styled>
56
- )
57
- }
58
-
59
- const locale = {
60
- pt: {
61
- dayMonth: '%d de %m',
62
- },
63
- en: {
64
- dayMonth: '%m %d',
65
- },
66
- }
1
+ import { Text } from '@citric/core'
2
+ import { Month, useMonthDictionary, useTranslate } from '@stack-spot/portal-translate'
3
+ import { styled } from 'styled-components'
4
+
5
+ interface Props {
6
+ /**
7
+ * The month. 0 for January, 11 for December.
8
+ */
9
+ month: Month,
10
+ /**
11
+ * The day of the month (1 - 31).
12
+ */
13
+ day: number,
14
+ /**
15
+ * The section's content.
16
+ */
17
+ children: React.ReactNode,
18
+ style?: React.CSSProperties,
19
+ className?: string,
20
+ }
21
+
22
+ const Styled = styled.div`
23
+ display: flex;
24
+ flex-direction: row;
25
+ align-items: start;
26
+
27
+ .timeline-section-header {
28
+ width: 80px;
29
+ }
30
+
31
+ .timeline-section-content {
32
+ min-width: 0;
33
+ flex: 1;
34
+ }
35
+ `
36
+
37
+ /**
38
+ * Renders a section in a timeline where the left side is a header with the date and the right side is the content (children).
39
+ * @param props React props: {@link Props}.
40
+ */
41
+ export const TimelineSection = ({ children, day, month, className, style }: Props) => {
42
+ const months = useMonthDictionary()
43
+ const t = useTranslate(locale)
44
+
45
+ const ariaLabel = t.dayMonth.replace('%d', `${day}`).replace('%m', months[month])
46
+ return (
47
+ <Styled style={style} className={className}>
48
+ <div className="timeline-section-header">
49
+ <Text appearance="h2" aria-label={ariaLabel}>{day >= 10 ? day : `0${day}`}</Text>
50
+ <Text appearance="subtitle4" aria-hidden="true">{months[month].toUpperCase().substring(0, 3)}</Text>
51
+ </div>
52
+ <div className="timeline-section-content">
53
+ {children}
54
+ </div>
55
+ </Styled>
56
+ )
57
+ }
58
+
59
+ const locale = {
60
+ pt: {
61
+ dayMonth: '%d de %m',
62
+ },
63
+ en: {
64
+ dayMonth: '%m %d',
65
+ },
66
+ }