@stack-spot/portal-components 2.27.1 → 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 (246) hide show
  1. package/CHANGELOG.md +635 -628
  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 +8 -2
  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.js +10 -10
  114. package/dist/context/anchor.d.ts +1 -1
  115. package/dist/context/anchor.js +1 -1
  116. package/dist/context/loading.d.ts +1 -1
  117. package/dist/context/loading.js +1 -1
  118. package/dist/context/notification/context.d.ts +1 -1
  119. package/dist/context/notification/context.js +1 -1
  120. package/dist/hooks/date.js +1 -1
  121. package/dist/hooks/service-now.js +28 -28
  122. package/dist/svg/AI.d.ts +1 -1
  123. package/dist/svg/AI.js +1 -1
  124. package/dist/svg/CS.d.ts +1 -1
  125. package/dist/svg/CS.js +1 -1
  126. package/dist/svg/EDP.d.ts +1 -1
  127. package/dist/svg/EDP.js +1 -1
  128. package/dist/svg/Forbidden.d.ts +1 -1
  129. package/dist/svg/Forbidden.js +1 -1
  130. package/dist/svg/GenericPlaceholder.d.ts +4 -2
  131. package/dist/svg/GenericPlaceholder.d.ts.map +1 -1
  132. package/dist/svg/GenericPlaceholder.js +2 -2
  133. package/dist/svg/GenericPlaceholder.js.map +1 -1
  134. package/dist/svg/HUB.d.ts +1 -1
  135. package/dist/svg/HUB.js +1 -1
  136. package/dist/svg/Logo.d.ts +1 -1
  137. package/dist/svg/Logo.js +1 -1
  138. package/dist/svg/MiniLogo.d.ts +1 -1
  139. package/dist/svg/MiniLogo.js +1 -1
  140. package/dist/svg/NotFound.d.ts +1 -1
  141. package/dist/svg/NotFound.js +1 -1
  142. package/dist/svg/ServerError.d.ts +1 -1
  143. package/dist/svg/ServerError.js +1 -1
  144. package/dist/svg/Unauthenticated.d.ts +1 -1
  145. package/dist/svg/Unauthenticated.js +1 -1
  146. package/package.json +80 -80
  147. package/readme.md +66 -66
  148. package/src/components/AnimatedHeight.tsx +174 -174
  149. package/src/components/AsyncContent.tsx +78 -78
  150. package/src/components/BannerWarning.tsx +91 -91
  151. package/src/components/Breadcrumb/index.tsx +76 -76
  152. package/src/components/Breadcrumb/styled.ts +37 -37
  153. package/src/components/ButtonLoading.tsx +29 -29
  154. package/src/components/ChatBot.tsx +82 -82
  155. package/src/components/ContentValidateFilter.tsx +15 -15
  156. package/src/components/FadingOverflow.tsx +265 -265
  157. package/src/components/FileTreeView/More.tsx +114 -114
  158. package/src/components/FileTreeView/index.tsx +186 -186
  159. package/src/components/InfiniteScroll.tsx +24 -24
  160. package/src/components/InfoMaintenanceBanner.tsx +29 -29
  161. package/src/components/LazyMarkdown/BlockquoteMd.tsx +107 -107
  162. package/src/components/LazyMarkdown/CodeViewer.tsx +161 -161
  163. package/src/components/LazyMarkdown/Markdown.tsx +122 -122
  164. package/src/components/LazyMarkdown/MarkdownButton.tsx +24 -24
  165. package/src/components/LazyMarkdown/Video.tsx +13 -13
  166. package/src/components/LazyMarkdown/index.tsx +21 -21
  167. package/src/components/Placeholder.tsx +123 -118
  168. package/src/components/ScrollView.tsx +57 -57
  169. package/src/components/Select/BadgeItem.tsx +58 -58
  170. package/src/components/Select/ClearInput.tsx +24 -24
  171. package/src/components/Select/CloseItem.tsx +38 -38
  172. package/src/components/Select/CreatableSelect.tsx +155 -155
  173. package/src/components/Select/CustomMenu.tsx +16 -16
  174. package/src/components/Select/LabelItem.tsx +8 -8
  175. package/src/components/Select/MultiValue.tsx +49 -49
  176. package/src/components/Select/SelectInfiniteScroll.tsx +82 -82
  177. package/src/components/Select/SelectSearch.tsx +195 -195
  178. package/src/components/Select/index.tsx +7 -7
  179. package/src/components/Select/types.ts +8 -8
  180. package/src/components/SelectionList.tsx +427 -427
  181. package/src/components/StatusCircle.tsx +67 -67
  182. package/src/components/Stepper/Navigation.tsx +97 -97
  183. package/src/components/Stepper/Step.tsx +30 -30
  184. package/src/components/Stepper/Stepper.tsx +113 -113
  185. package/src/components/Stepper/headers.tsx +64 -64
  186. package/src/components/Stepper/index.ts +3 -3
  187. package/src/components/Table/HeaderItem.tsx +52 -52
  188. package/src/components/Table/SettingsVerticalMenu.tsx +50 -50
  189. package/src/components/Table/StyledLinkTable.tsx +22 -22
  190. package/src/components/Table/TableData.tsx +251 -251
  191. package/src/components/Table/index.tsx +2 -2
  192. package/src/components/TimelineSection.tsx +66 -66
  193. package/src/components/error/ErrorFeedback.tsx +217 -217
  194. package/src/components/error/NotFound.tsx +24 -24
  195. package/src/components/error/UnderMaintenance.tsx +30 -30
  196. package/src/components/error/index.ts +4 -4
  197. package/src/components/form/Form/Form.tsx +101 -101
  198. package/src/components/form/Form/FormGroup.tsx +221 -221
  199. package/src/components/form/Form/index.ts +2 -2
  200. package/src/components/form/SearchInput.tsx +69 -69
  201. package/src/components/form/Select/CustomSelect.tsx +232 -232
  202. package/src/components/form/Select/DetailedSelect.tsx +85 -85
  203. package/src/components/form/Select/Select.tsx +67 -67
  204. package/src/components/form/Select/index.ts +4 -4
  205. package/src/components/form/Select/styled.ts +165 -165
  206. package/src/components/form/Select/types.ts +112 -112
  207. package/src/components/form/Select/utils.tsx +28 -28
  208. package/src/components/notification/NotificationComponent.tsx +340 -340
  209. package/src/components/notification/NotificationItem.tsx +345 -337
  210. package/src/components/notification/NotificationList.tsx +179 -178
  211. package/src/components/notification/NotificationPlaceholder.tsx +44 -43
  212. package/src/components/notification/types.ts +72 -72
  213. package/src/containers/NotificationsPage.tsx +119 -119
  214. package/src/context/anchor.tsx +37 -37
  215. package/src/context/loading.tsx +36 -36
  216. package/src/context/notification/LazyNotificationList.ts +103 -103
  217. package/src/context/notification/NotificationController.ts +104 -104
  218. package/src/context/notification/context.tsx +23 -23
  219. package/src/context/notification/hooks.ts +98 -98
  220. package/src/context/notification/types.ts +66 -66
  221. package/src/hooks/date.ts +31 -31
  222. package/src/hooks/keyboard.tsx +128 -128
  223. package/src/hooks/manual-render.tsx +10 -10
  224. package/src/hooks/service-now.tsx +233 -233
  225. package/src/hooks/text.tsx +30 -30
  226. package/src/hooks/title.tsx +28 -28
  227. package/src/hooks/use-effect-once.tsx +43 -43
  228. package/src/index.ts +19 -19
  229. package/src/notifications.ts +11 -11
  230. package/src/svg/AI.tsx +41 -41
  231. package/src/svg/CS.tsx +48 -48
  232. package/src/svg/EDP.tsx +31 -31
  233. package/src/svg/Forbidden.tsx +22 -22
  234. package/src/svg/GenericPlaceholder.tsx +20 -20
  235. package/src/svg/HUB.tsx +48 -48
  236. package/src/svg/Logo.tsx +16 -16
  237. package/src/svg/MiniLogo.tsx +12 -12
  238. package/src/svg/NotFound.tsx +16 -16
  239. package/src/svg/ServerError.tsx +33 -33
  240. package/src/svg/Unauthenticated.tsx +16 -16
  241. package/src/svg/index.ts +11 -11
  242. package/src/utils/accessibility.ts +135 -135
  243. package/src/utils/cookie.ts +73 -73
  244. package/src/utils/promise.ts +5 -5
  245. package/src/utils/read-file.ts +16 -16
  246. package/tsconfig.json +10 -10
@@ -1,178 +1,179 @@
1
- import { Box, Flex } from '@citric/core'
2
- import { listToClass } from '@stack-spot/portal-theme'
3
- import { Month } from '@stack-spot/portal-translate'
4
- import { last } from 'lodash'
5
- import { useMemo } from 'react'
6
- import { styled } from 'styled-components'
7
- import { InfiniteScroll } from '../InfiniteScroll'
8
- import { TimelineSection } from '../TimelineSection'
9
- import { NotificationItem } from './NotificationItem'
10
- import { NotificationPlaceholder } from './NotificationPlaceholder'
11
- import { GetTenantNotificationsResponse, InfiniteScrollConfig } from './types'
12
-
13
- export interface NotificationListProps {
14
- /**
15
- * Function to call when the message is marked as read (committed).
16
- * @param id the id of the notification where the read status changed.
17
- */
18
- onCommit: (id: string) => void,
19
- /**
20
- * Optional. Function called when the button to perform the notification action is clicked. This function will be run in addition to
21
- * the notification action.
22
- * @param id the id of the notification where the button was clicked
23
- */
24
- onClickAction?: (id: string) => void,
25
- /**
26
- * Optional. Function called when the button read more in notification is clicked.
27
- */
28
- onClickViewNotification?: () => void,
29
- /**
30
- * If you need this notification list to be have an infinite scroll behavior, set this option.
31
- */
32
- infiniteScroll?: InfiniteScrollConfig,
33
- /**
34
- * The notifications themselves.
35
- */
36
- items: GetTenantNotificationsResponse[],
37
- /**
38
- * A compact notification list don't show date headers (as a timeline) or descriptions of notifications.
39
- */
40
- compact?: boolean,
41
- /**
42
- * Whether or not the content is loading. If this is true, the content becomes transparent and the cursor turns into the progress cursor.
43
- */
44
- loading?: boolean,
45
- /**
46
- * If true, when the list is empty, the placeholder will say "nothing found" instead of "no notifications".
47
- */
48
- showEmptySearch?: boolean,
49
- style?: React.CSSProperties,
50
- className?: string,
51
- }
52
-
53
- interface NotificationGroup {
54
- month: Month,
55
- day: number,
56
- year: number,
57
- items: GetTenantNotificationsResponse[],
58
- }
59
-
60
- const StyledBox = styled(Box)`
61
- width: 100%;
62
- position: relative;
63
- transition: opacity 0.3s;
64
-
65
- > div:first-child{
66
- width: 100%;
67
- }
68
-
69
- &.loading {
70
- opacity: 0.6;
71
- .loading-mask {
72
- pointer-events: auto;
73
- }
74
- }
75
-
76
- .loading-mask {
77
- opacity: 0;
78
- position: absolute;
79
- pointer-events: none;
80
- top: 0;
81
- right: 0;
82
- left: 0;
83
- bottom: 0;
84
- cursor: progress;
85
- }
86
-
87
- .placeholder.compact {
88
- & > div {
89
- padding-block: 0;
90
- gap: 15px;
91
- & > div {
92
- text-align: center;
93
- h4 {
94
- margin-bottom: 10px;
95
- }
96
- p {
97
- margin: 0;
98
- }
99
- }
100
- }
101
- }
102
- `
103
-
104
- // this assumes the data from the backend is ordered by date (trigger_at)
105
- function groupNotificationsByDate(notifications: GetTenantNotificationsResponse[]): NotificationGroup[] {
106
- const groups: NotificationGroup[] = []
107
- for (const n of notifications) {
108
- let currentGroup = last(groups)
109
- const date = new Date(n.trigger_at)
110
- const year = date.getFullYear()
111
- const month = date.getMonth() as Month
112
- const day = date.getDate()
113
- if (!currentGroup || currentGroup.day !== day || currentGroup.month !== month || currentGroup.year !== year) {
114
- currentGroup = { year, month, day, items: [] }
115
- groups.push(currentGroup)
116
- }
117
- currentGroup.items.push(n)
118
- }
119
- return groups
120
- }
121
-
122
- export const NotificationList = (
123
- { items, compact = false, onCommit, onClickAction, onClickViewNotification,
124
- infiniteScroll, loading, showEmptySearch, style, className }: NotificationListProps,
125
- ) => {
126
- const groups: NotificationGroup[] = useMemo(
127
- () => compact ? [{ day: 0, month: 0, year: 0, items }] : groupNotificationsByDate(items),
128
- [compact, items],
129
- )
130
-
131
- const renderNotifications = (notifications: GetTenantNotificationsResponse[], key?: string) => (
132
- <Flex sx={{ gap: '4px' }} flexDirection="column" key={key}>
133
- {notifications?.map((item) => (
134
- <NotificationItem
135
- key={item.id}
136
- id={key ? '' : `notification-item-${item.id}`}
137
- notification={item}
138
- isSummary={compact}
139
- onClickViewNotification={onClickViewNotification}
140
- onCommit={() => onCommit(item.id)}
141
- onClickAction={() => onClickAction?.(item.id)}
142
- />
143
- ))}
144
- </Flex>
145
- )
146
-
147
- return infiniteScroll?.scrollableTarget !== null && (
148
- <StyledBox style={style} className={listToClass([className, loading && 'loading'])}>
149
- {items.length ? (
150
- <InfiniteScroll
151
- dataLength={items.length || 0}
152
- next={infiniteScroll?.loadMore ?? (() => { })}
153
- hasMore={infiniteScroll?.hasMore ?? false}
154
- // @ts-ignore: the library is wrongly typed and abandoned, meaning, it will never be fixed. The source code clearly accepts
155
- // HTMLElements as scrollable targets:
156
- // https://github.com/ankeetmaini/react-infinite-scroll-component/blob/master/src/index.tsx#L168
157
- scrollableTarget={infiniteScroll?.scrollableTarget}
158
- >
159
- <Flex sx={{ gap: '24px' }} flexDirection="column">
160
- {groups.map((group) => compact
161
- ? renderNotifications(group.items, 'compact')
162
- : (
163
- <TimelineSection key={`${group.day}-${group.month}-${group.year}`} month={group.month} day={group.day}>
164
- {renderNotifications(group.items)}
165
- </TimelineSection>
166
- ),
167
- )}
168
- </Flex>
169
- </InfiniteScroll>
170
- ) : <NotificationPlaceholder
171
- isSearch={showEmptySearch}
172
- className={listToClass(['placeholder', compact && 'compact'])}
173
- titleAs="p"
174
- />}
175
- <div className="loading-mask"></div>
176
- </StyledBox>
177
- )
178
- }
1
+ import { Box, Flex } from '@citric/core'
2
+ import { listToClass } from '@stack-spot/portal-theme'
3
+ import { Month } from '@stack-spot/portal-translate'
4
+ import { last } from 'lodash'
5
+ import { useMemo } from 'react'
6
+ import { styled } from 'styled-components'
7
+ import { InfiniteScroll } from '../InfiniteScroll'
8
+ import { TimelineSection } from '../TimelineSection'
9
+ import { NotificationItem } from './NotificationItem'
10
+ import { NotificationPlaceholder } from './NotificationPlaceholder'
11
+ import { GetTenantNotificationsResponse, InfiniteScrollConfig } from './types'
12
+
13
+ export interface NotificationListProps {
14
+ /**
15
+ * Function to call when the message is marked as read (committed).
16
+ * @param id the id of the notification where the read status changed.
17
+ */
18
+ onCommit: (id: string) => void,
19
+ /**
20
+ * Optional. Function called when the button to perform the notification action is clicked. This function will be run in addition to
21
+ * the notification action.
22
+ * @param id the id of the notification where the button was clicked
23
+ */
24
+ onClickAction?: (id: string) => void,
25
+ /**
26
+ * Optional. Function called when the button read more in notification is clicked.
27
+ */
28
+ onClickViewNotification?: () => void,
29
+ /**
30
+ * If you need this notification list to be have an infinite scroll behavior, set this option.
31
+ */
32
+ infiniteScroll?: InfiniteScrollConfig,
33
+ /**
34
+ * The notifications themselves.
35
+ */
36
+ items: GetTenantNotificationsResponse[],
37
+ /**
38
+ * A compact notification list don't show date headers (as a timeline) or descriptions of notifications.
39
+ */
40
+ compact?: boolean,
41
+ /**
42
+ * Whether or not the content is loading. If this is true, the content becomes transparent and the cursor turns into the progress cursor.
43
+ */
44
+ loading?: boolean,
45
+ /**
46
+ * If true, when the list is empty, the placeholder will say "nothing found" instead of "no notifications".
47
+ */
48
+ showEmptySearch?: boolean,
49
+ style?: React.CSSProperties,
50
+ className?: string,
51
+ }
52
+
53
+ interface NotificationGroup {
54
+ month: Month,
55
+ day: number,
56
+ year: number,
57
+ items: GetTenantNotificationsResponse[],
58
+ }
59
+
60
+ const StyledBox = styled(Box)`
61
+ width: 100%;
62
+ position: relative;
63
+ transition: opacity 0.3s;
64
+
65
+ > div:first-child{
66
+ width: 100%;
67
+ }
68
+
69
+ &.loading {
70
+ opacity: 0.6;
71
+ .loading-mask {
72
+ pointer-events: auto;
73
+ }
74
+ }
75
+
76
+ .loading-mask {
77
+ opacity: 0;
78
+ position: absolute;
79
+ pointer-events: none;
80
+ top: 0;
81
+ right: 0;
82
+ left: 0;
83
+ bottom: 0;
84
+ cursor: progress;
85
+ }
86
+
87
+ .placeholder.compact {
88
+ & > div {
89
+ padding-block: 0;
90
+ gap: 16px;
91
+ & > div {
92
+ text-align: center;
93
+ h4 {
94
+ margin-bottom: 10px;
95
+ }
96
+ p {
97
+ margin: 0;
98
+ }
99
+ }
100
+ }
101
+ }
102
+ `
103
+
104
+ // this assumes the data from the backend is ordered by date (trigger_at)
105
+ function groupNotificationsByDate(notifications: GetTenantNotificationsResponse[]): NotificationGroup[] {
106
+ const groups: NotificationGroup[] = []
107
+ for (const n of notifications) {
108
+ let currentGroup = last(groups)
109
+ const date = new Date(n.trigger_at)
110
+ const year = date.getFullYear()
111
+ const month = date.getMonth() as Month
112
+ const day = date.getDate()
113
+ if (!currentGroup || currentGroup.day !== day || currentGroup.month !== month || currentGroup.year !== year) {
114
+ currentGroup = { year, month, day, items: [] }
115
+ groups.push(currentGroup)
116
+ }
117
+ currentGroup.items.push(n)
118
+ }
119
+ return groups
120
+ }
121
+
122
+ export const NotificationList = (
123
+ { items, compact = false, onCommit, onClickAction, onClickViewNotification,
124
+ infiniteScroll, loading, showEmptySearch, style, className }: NotificationListProps,
125
+ ) => {
126
+ const groups: NotificationGroup[] = useMemo(
127
+ () => compact ? [{ day: 0, month: 0, year: 0, items }] : groupNotificationsByDate(items),
128
+ [compact, items],
129
+ )
130
+
131
+ const renderNotifications = (notifications: GetTenantNotificationsResponse[], key?: string) => (
132
+ <Flex sx={{ gap: '4px' }} flexDirection="column" key={key}>
133
+ {notifications?.map((item) => (
134
+ <NotificationItem
135
+ key={item.id}
136
+ id={key ? '' : `notification-item-${item.id}`}
137
+ notification={item}
138
+ isSummary={compact}
139
+ onClickViewNotification={onClickViewNotification}
140
+ onCommit={() => onCommit(item.id)}
141
+ onClickAction={() => onClickAction?.(item.id)}
142
+ />
143
+ ))}
144
+ </Flex>
145
+ )
146
+
147
+ return infiniteScroll?.scrollableTarget !== null && (
148
+ <StyledBox style={style} className={listToClass([className, loading && 'loading'])}>
149
+ {items.length ? (
150
+ <InfiniteScroll
151
+ dataLength={items.length || 0}
152
+ next={infiniteScroll?.loadMore ?? (() => { })}
153
+ hasMore={infiniteScroll?.hasMore ?? false}
154
+ // @ts-ignore: the library is wrongly typed and abandoned, meaning, it will never be fixed. The source code clearly accepts
155
+ // HTMLElements as scrollable targets:
156
+ // https://github.com/ankeetmaini/react-infinite-scroll-component/blob/master/src/index.tsx#L168
157
+ scrollableTarget={infiniteScroll?.scrollableTarget}
158
+ >
159
+ <Flex sx={{ gap: '24px' }} flexDirection="column">
160
+ {groups.map((group) => compact
161
+ ? renderNotifications(group.items, 'compact')
162
+ : (
163
+ <TimelineSection key={`${group.day}-${group.month}-${group.year}`} month={group.month} day={group.day}>
164
+ {renderNotifications(group.items)}
165
+ </TimelineSection>
166
+ ),
167
+ )}
168
+ </Flex>
169
+ </InfiniteScroll>
170
+ ) : <NotificationPlaceholder
171
+ isSearch={showEmptySearch}
172
+ className={listToClass(['placeholder', compact && 'compact'])}
173
+ titleAs="p"
174
+ sx={{ minHeight: '350px' }}
175
+ />}
176
+ <div className="loading-mask"></div>
177
+ </StyledBox>
178
+ )
179
+ }
@@ -1,43 +1,44 @@
1
- import { AsProp } from '@citric/core/dist/forward-ref-as'
2
- import { SxProperties } from '@citric/core/dist/sx'
3
- import { Dictionary, useTranslate } from '@stack-spot/portal-translate'
4
- import { PlaceholderCallToAction } from '../Placeholder'
5
-
6
- interface Props {
7
- isSearch?: boolean,
8
- style?: React.CSSProperties,
9
- className?: string,
10
- sx?: SxProperties,
11
- sxCard?: SxProperties,
12
- titleAs?: AsProp,
13
- }
14
-
15
- export const NotificationPlaceholder = ({ isSearch, sx, sxCard, className, style, titleAs }: Props) => {
16
- const t = useTranslate(dictionary)
17
-
18
- return <PlaceholderCallToAction
19
- title={isSearch ? t.placeholderSearchTitle : t.placeholderTitle}
20
- description={isSearch ? t.placeholderSearchDescription : t.placeholderDescription}
21
- fullWidth
22
- sx={sx}
23
- sxCard={sxCard}
24
- className={className}
25
- style={style}
26
- titleAs={titleAs}
27
- />
28
- }
29
-
30
- const dictionary = {
31
- en: {
32
- placeholderTitle: 'You don’t have any notifications yet.',
33
- placeholderDescription: 'Relevant updates will show up here as soon as they’re available.',
34
- placeholderSearchTitle: 'No notifications match your filters',
35
- placeholderSearchDescription: 'Try changing the filters to see more results.',
36
- },
37
- pt: {
38
- placeholderTitle: 'Você ainda não tem notificações.',
39
- placeholderDescription: 'Quando houver alguma atualização relevante, ela será exibida aqui.',
40
- placeholderSearchTitle: 'Nenhuma notificação corresponde aos filtros',
41
- placeholderSearchDescription: 'Experimente mudar os filtros para encontrar mais notificações.',
42
- },
43
- } satisfies Dictionary
1
+ import { AsProp } from '@citric/core/dist/forward-ref-as'
2
+ import { SxProperties } from '@citric/core/dist/sx'
3
+ import { Dictionary, useTranslate } from '@stack-spot/portal-translate'
4
+ import { PlaceholderCallToAction } from '../Placeholder'
5
+
6
+ interface Props {
7
+ isSearch?: boolean,
8
+ style?: React.CSSProperties,
9
+ className?: string,
10
+ sx?: SxProperties,
11
+ sxCard?: SxProperties,
12
+ titleAs?: AsProp,
13
+ }
14
+
15
+ export const NotificationPlaceholder = ({ isSearch, sx, sxCard, className, style, titleAs }: Props) => {
16
+ const t = useTranslate(dictionary)
17
+
18
+ return <PlaceholderCallToAction
19
+ title={isSearch ? t.placeholderSearchTitle : t.placeholderTitle}
20
+ description={isSearch ? t.placeholderSearchDescription : t.placeholderDescription}
21
+ fullWidth
22
+ sx={sx}
23
+ sxCard={sxCard}
24
+ className={className}
25
+ style={style}
26
+ titleAs={titleAs}
27
+ placeholderImageConfig={{ width: '200', height: '150' }}
28
+ />
29
+ }
30
+
31
+ const dictionary = {
32
+ en: {
33
+ placeholderTitle: 'You don’t have any notifications yet.',
34
+ placeholderDescription: 'Relevant updates will show up here as soon as they’re available.',
35
+ placeholderSearchTitle: 'No notifications match your filters',
36
+ placeholderSearchDescription: 'Try changing the filters to see more results.',
37
+ },
38
+ pt: {
39
+ placeholderTitle: 'Você ainda não tem notificações.',
40
+ placeholderDescription: 'Quando houver alguma atualização relevante, ela será exibida aqui.',
41
+ placeholderSearchTitle: 'Nenhuma notificação corresponde aos filtros',
42
+ placeholderSearchDescription: 'Experimente mudar os filtros para encontrar mais notificações.',
43
+ },
44
+ } satisfies Dictionary
@@ -1,72 +1,72 @@
1
- export type NotificationType = 'LOW' | 'MEDIUM' | 'HIGH'
2
-
3
- export type NotificationContext = 'ACCOUNT' | 'STUDIO' | 'WORKSPACE' | 'AI'
4
-
5
- //The following three notification types comes from the api swagger
6
-
7
- export type NotificationContentResponse = {
8
- title: string,
9
- description: string,
10
- };
11
-
12
- export type GetTenantNotificationsResponse = {
13
- id: string,
14
- content: {
15
- [key: string]: NotificationContentResponse,
16
- },
17
- broadcast_level: 'ACCOUNT' | 'INDIVIDUAL' | 'PLATFORM' | 'RESOURCE',
18
- context: 'ACCOUNT' | 'STUDIO' | 'WORKSPACE' | 'AI',
19
- target?: string,
20
- criticality: 'LOW' | 'MEDIUM' | 'HIGH',
21
- call_to_action: string,
22
- persistent: boolean,
23
- committed: boolean,
24
- last_until?: string,
25
- trigger_at: string,
26
- created_at?: string,
27
- isBanner?: boolean,
28
- }
29
-
30
- export type ResponseModelGetTenantNotificationsResponse = {
31
- items: GetTenantNotificationsResponse[],
32
- total: number,
33
- }
34
-
35
- export type NotificationTypeFilters = NotificationType | UnreadType
36
-
37
- export interface NotificationFilters {
38
- filterBy?: 'criticality' | 'context',
39
- filterValue?: string,
40
- sort?: string,
41
- direction?: string,
42
- page?: any,
43
- size?: string,
44
- }
45
-
46
- export interface NotificationCommitted {
47
- id: string,
48
- committed: boolean,
49
- }
50
-
51
- // fixme: please, do not ignore lint rules. They exist for a reason. Next major: transform this enum into a disjoint string type.
52
- // eslint-disable-next-line no-restricted-syntax
53
- export enum UnreadType {
54
- Unread = 'unread',
55
- }
56
-
57
- export interface InfiniteScrollConfig {
58
- /**
59
- * Function to load more items into the list. Called when the scroll is almost reaching its end.
60
- */
61
- loadMore: () => void,
62
- /**
63
- * Set this to false to prevent the scroll from loading more items when it reaches the end.
64
- */
65
- hasMore: boolean,
66
- /**
67
- * Defines which scroll will be used as the target of the infinite scroll.
68
- *
69
- * If null, nothing renders, it waits until it has a value.
70
- */
71
- scrollableTarget?: string | HTMLElement | null,
72
- }
1
+ export type NotificationType = 'LOW' | 'MEDIUM' | 'HIGH'
2
+
3
+ export type NotificationContext = 'ACCOUNT' | 'STUDIO' | 'WORKSPACE' | 'AI'
4
+
5
+ //The following three notification types comes from the api swagger
6
+
7
+ export type NotificationContentResponse = {
8
+ title: string,
9
+ description: string,
10
+ };
11
+
12
+ export type GetTenantNotificationsResponse = {
13
+ id: string,
14
+ content: {
15
+ [key: string]: NotificationContentResponse,
16
+ },
17
+ broadcast_level: 'ACCOUNT' | 'INDIVIDUAL' | 'PLATFORM' | 'RESOURCE',
18
+ context: 'ACCOUNT' | 'STUDIO' | 'WORKSPACE' | 'AI',
19
+ target?: string,
20
+ criticality: 'LOW' | 'MEDIUM' | 'HIGH',
21
+ call_to_action: string,
22
+ persistent: boolean,
23
+ committed: boolean,
24
+ last_until?: string,
25
+ trigger_at: string,
26
+ created_at?: string,
27
+ isBanner?: boolean,
28
+ }
29
+
30
+ export type ResponseModelGetTenantNotificationsResponse = {
31
+ items: GetTenantNotificationsResponse[],
32
+ total: number,
33
+ }
34
+
35
+ export type NotificationTypeFilters = NotificationType | UnreadType
36
+
37
+ export interface NotificationFilters {
38
+ filterBy?: 'criticality' | 'context',
39
+ filterValue?: string,
40
+ sort?: string,
41
+ direction?: string,
42
+ page?: any,
43
+ size?: string,
44
+ }
45
+
46
+ export interface NotificationCommitted {
47
+ id: string,
48
+ committed: boolean,
49
+ }
50
+
51
+ // fixme: please, do not ignore lint rules. They exist for a reason. Next major: transform this enum into a disjoint string type.
52
+ // eslint-disable-next-line no-restricted-syntax
53
+ export enum UnreadType {
54
+ Unread = 'unread',
55
+ }
56
+
57
+ export interface InfiniteScrollConfig {
58
+ /**
59
+ * Function to load more items into the list. Called when the scroll is almost reaching its end.
60
+ */
61
+ loadMore: () => void,
62
+ /**
63
+ * Set this to false to prevent the scroll from loading more items when it reaches the end.
64
+ */
65
+ hasMore: boolean,
66
+ /**
67
+ * Defines which scroll will be used as the target of the infinite scroll.
68
+ *
69
+ * If null, nothing renders, it waits until it has a value.
70
+ */
71
+ scrollableTarget?: string | HTMLElement | null,
72
+ }