openvsx-webui-test 0.17.2 → 0.18.0-dev.1

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 (403) hide show
  1. package/lib/components/button-with-progress.d.ts +1 -0
  2. package/lib/components/button-with-progress.d.ts.map +1 -1
  3. package/lib/components/button-with-progress.js +1 -1
  4. package/lib/components/button-with-progress.js.map +1 -1
  5. package/lib/components/sanitized-markdown.d.ts.map +1 -1
  6. package/lib/components/sanitized-markdown.js +79 -1
  7. package/lib/components/sanitized-markdown.js.map +1 -1
  8. package/lib/components/scan-admin/common/auto-refresh.d.ts +21 -0
  9. package/lib/components/scan-admin/common/auto-refresh.d.ts.map +1 -0
  10. package/lib/components/scan-admin/common/auto-refresh.js +45 -0
  11. package/lib/components/scan-admin/common/auto-refresh.js.map +1 -0
  12. package/lib/components/scan-admin/common/conditional-tooltip.d.ts +24 -0
  13. package/lib/components/scan-admin/common/conditional-tooltip.d.ts.map +1 -0
  14. package/lib/components/scan-admin/common/conditional-tooltip.js +63 -0
  15. package/lib/components/scan-admin/common/conditional-tooltip.js.map +1 -0
  16. package/lib/components/scan-admin/common/file-table.d.ts +23 -0
  17. package/lib/components/scan-admin/common/file-table.d.ts.map +1 -0
  18. package/lib/components/scan-admin/common/file-table.js +350 -0
  19. package/lib/components/scan-admin/common/file-table.js.map +1 -0
  20. package/lib/components/scan-admin/common/index.d.ts +18 -0
  21. package/lib/components/scan-admin/common/index.d.ts.map +1 -0
  22. package/lib/components/scan-admin/common/index.js +18 -0
  23. package/lib/components/scan-admin/common/index.js.map +1 -0
  24. package/lib/components/scan-admin/common/tab-panel.d.ts +25 -0
  25. package/lib/components/scan-admin/common/tab-panel.d.ts.map +1 -0
  26. package/lib/components/scan-admin/common/tab-panel.js +22 -0
  27. package/lib/components/scan-admin/common/tab-panel.js.map +1 -0
  28. package/lib/components/scan-admin/common/utils.d.ts +15 -0
  29. package/lib/components/scan-admin/common/utils.d.ts.map +1 -0
  30. package/lib/components/scan-admin/common/utils.js +46 -0
  31. package/lib/components/scan-admin/common/utils.js.map +1 -0
  32. package/lib/components/scan-admin/dialogs/file-dialog.d.ts +19 -0
  33. package/lib/components/scan-admin/dialogs/file-dialog.d.ts.map +1 -0
  34. package/lib/components/scan-admin/dialogs/file-dialog.js +85 -0
  35. package/lib/components/scan-admin/dialogs/file-dialog.js.map +1 -0
  36. package/lib/components/scan-admin/dialogs/index.d.ts +15 -0
  37. package/lib/components/scan-admin/dialogs/index.d.ts.map +1 -0
  38. package/lib/components/scan-admin/dialogs/index.js +15 -0
  39. package/lib/components/scan-admin/dialogs/index.js.map +1 -0
  40. package/lib/components/scan-admin/dialogs/quarantine-dialog.d.ts +23 -0
  41. package/lib/components/scan-admin/dialogs/quarantine-dialog.d.ts.map +1 -0
  42. package/lib/components/scan-admin/dialogs/quarantine-dialog.js +192 -0
  43. package/lib/components/scan-admin/dialogs/quarantine-dialog.js.map +1 -0
  44. package/lib/components/scan-admin/index.d.ts +20 -0
  45. package/lib/components/scan-admin/index.d.ts.map +1 -0
  46. package/lib/components/scan-admin/index.js +25 -0
  47. package/lib/components/scan-admin/index.js.map +1 -0
  48. package/lib/components/scan-admin/scan-card/index.d.ts +20 -0
  49. package/lib/components/scan-admin/scan-card/index.d.ts.map +1 -0
  50. package/lib/components/scan-admin/scan-card/index.js +20 -0
  51. package/lib/components/scan-admin/scan-card/index.js.map +1 -0
  52. package/lib/components/scan-admin/scan-card/scan-card-content.d.ts +30 -0
  53. package/lib/components/scan-admin/scan-card/scan-card-content.d.ts.map +1 -0
  54. package/lib/components/scan-admin/scan-card/scan-card-content.js +232 -0
  55. package/lib/components/scan-admin/scan-card/scan-card-content.js.map +1 -0
  56. package/lib/components/scan-admin/scan-card/scan-card-expand-strip-badges.d.ts +26 -0
  57. package/lib/components/scan-admin/scan-card/scan-card-expand-strip-badges.d.ts.map +1 -0
  58. package/lib/components/scan-admin/scan-card/scan-card-expand-strip-badges.js +121 -0
  59. package/lib/components/scan-admin/scan-card/scan-card-expand-strip-badges.js.map +1 -0
  60. package/lib/components/scan-admin/scan-card/scan-card-expand-strip.d.ts +27 -0
  61. package/lib/components/scan-admin/scan-card/scan-card-expand-strip.d.ts.map +1 -0
  62. package/lib/components/scan-admin/scan-card/scan-card-expand-strip.js +66 -0
  63. package/lib/components/scan-admin/scan-card/scan-card-expand-strip.js.map +1 -0
  64. package/lib/components/scan-admin/scan-card/scan-card-expanded-content.d.ts +26 -0
  65. package/lib/components/scan-admin/scan-card/scan-card-expanded-content.d.ts.map +1 -0
  66. package/lib/components/scan-admin/scan-card/scan-card-expanded-content.js +151 -0
  67. package/lib/components/scan-admin/scan-card/scan-card-expanded-content.js.map +1 -0
  68. package/lib/components/scan-admin/scan-card/scan-card-header.d.ts +26 -0
  69. package/lib/components/scan-admin/scan-card/scan-card-header.d.ts.map +1 -0
  70. package/lib/components/scan-admin/scan-card/scan-card-header.js +101 -0
  71. package/lib/components/scan-admin/scan-card/scan-card-header.js.map +1 -0
  72. package/lib/components/scan-admin/scan-card/scan-card.d.ts +34 -0
  73. package/lib/components/scan-admin/scan-card/scan-card.d.ts.map +1 -0
  74. package/lib/components/scan-admin/scan-card/scan-card.js +90 -0
  75. package/lib/components/scan-admin/scan-card/scan-card.js.map +1 -0
  76. package/lib/components/scan-admin/scan-card/scan-detail-card.d.ts +38 -0
  77. package/lib/components/scan-admin/scan-card/scan-detail-card.d.ts.map +1 -0
  78. package/lib/components/scan-admin/scan-card/scan-detail-card.js +72 -0
  79. package/lib/components/scan-admin/scan-card/scan-detail-card.js.map +1 -0
  80. package/lib/components/scan-admin/scan-card/utils.d.ts +48 -0
  81. package/lib/components/scan-admin/scan-card/utils.d.ts.map +1 -0
  82. package/lib/components/scan-admin/scan-card/utils.js +166 -0
  83. package/lib/components/scan-admin/scan-card/utils.js.map +1 -0
  84. package/lib/components/scan-admin/tab-contents/allow-list-tab-content.d.ts +19 -0
  85. package/lib/components/scan-admin/tab-contents/allow-list-tab-content.d.ts.map +1 -0
  86. package/lib/components/scan-admin/tab-contents/allow-list-tab-content.js +55 -0
  87. package/lib/components/scan-admin/tab-contents/allow-list-tab-content.js.map +1 -0
  88. package/lib/components/scan-admin/tab-contents/auto-rejected-tab-content.d.ts +19 -0
  89. package/lib/components/scan-admin/tab-contents/auto-rejected-tab-content.d.ts.map +1 -0
  90. package/lib/components/scan-admin/tab-contents/auto-rejected-tab-content.js +43 -0
  91. package/lib/components/scan-admin/tab-contents/auto-rejected-tab-content.js.map +1 -0
  92. package/lib/components/scan-admin/tab-contents/block-list-tab-content.d.ts +19 -0
  93. package/lib/components/scan-admin/tab-contents/block-list-tab-content.d.ts.map +1 -0
  94. package/lib/components/scan-admin/tab-contents/block-list-tab-content.js +55 -0
  95. package/lib/components/scan-admin/tab-contents/block-list-tab-content.js.map +1 -0
  96. package/lib/components/scan-admin/tab-contents/index.d.ts +18 -0
  97. package/lib/components/scan-admin/tab-contents/index.d.ts.map +1 -0
  98. package/lib/components/scan-admin/tab-contents/index.js +18 -0
  99. package/lib/components/scan-admin/tab-contents/index.js.map +1 -0
  100. package/lib/components/scan-admin/tab-contents/quarantined-tab-content.d.ts +19 -0
  101. package/lib/components/scan-admin/tab-contents/quarantined-tab-content.d.ts.map +1 -0
  102. package/lib/components/scan-admin/tab-contents/quarantined-tab-content.js +78 -0
  103. package/lib/components/scan-admin/tab-contents/quarantined-tab-content.js.map +1 -0
  104. package/lib/components/scan-admin/tab-contents/scans-tab-content.d.ts +19 -0
  105. package/lib/components/scan-admin/tab-contents/scans-tab-content.d.ts.map +1 -0
  106. package/lib/components/scan-admin/tab-contents/scans-tab-content.js +52 -0
  107. package/lib/components/scan-admin/tab-contents/scans-tab-content.js.map +1 -0
  108. package/lib/components/scan-admin/toolbars/counts-toolbar.d.ts +35 -0
  109. package/lib/components/scan-admin/toolbars/counts-toolbar.d.ts.map +1 -0
  110. package/lib/components/scan-admin/toolbars/counts-toolbar.js +132 -0
  111. package/lib/components/scan-admin/toolbars/counts-toolbar.js.map +1 -0
  112. package/lib/components/scan-admin/toolbars/index.d.ts +16 -0
  113. package/lib/components/scan-admin/toolbars/index.d.ts.map +1 -0
  114. package/lib/components/scan-admin/toolbars/index.js +16 -0
  115. package/lib/components/scan-admin/toolbars/index.js.map +1 -0
  116. package/lib/components/scan-admin/toolbars/search-toolbar.d.ts +43 -0
  117. package/lib/components/scan-admin/toolbars/search-toolbar.d.ts.map +1 -0
  118. package/lib/components/scan-admin/toolbars/search-toolbar.js +106 -0
  119. package/lib/components/scan-admin/toolbars/search-toolbar.js.map +1 -0
  120. package/lib/components/scan-admin/toolbars/tab-toolbar.d.ts +25 -0
  121. package/lib/components/scan-admin/toolbars/tab-toolbar.d.ts.map +1 -0
  122. package/lib/components/scan-admin/toolbars/tab-toolbar.js +33 -0
  123. package/lib/components/scan-admin/toolbars/tab-toolbar.js.map +1 -0
  124. package/lib/context/scan-admin/index.d.ts +21 -0
  125. package/lib/context/scan-admin/index.d.ts.map +1 -0
  126. package/lib/context/scan-admin/index.js +27 -0
  127. package/lib/context/scan-admin/index.js.map +1 -0
  128. package/lib/context/scan-admin/scan-actions.d.ts +19 -0
  129. package/lib/context/scan-admin/scan-actions.d.ts.map +1 -0
  130. package/lib/context/scan-admin/scan-actions.js +64 -0
  131. package/lib/context/scan-admin/scan-actions.js.map +1 -0
  132. package/lib/context/scan-admin/scan-api-actions.d.ts +26 -0
  133. package/lib/context/scan-admin/scan-api-actions.d.ts.map +1 -0
  134. package/lib/context/scan-admin/scan-api-actions.js +118 -0
  135. package/lib/context/scan-admin/scan-api-actions.js.map +1 -0
  136. package/lib/context/scan-admin/scan-api-effects.d.ts +40 -0
  137. package/lib/context/scan-admin/scan-api-effects.d.ts.map +1 -0
  138. package/lib/context/scan-admin/scan-api-effects.js +446 -0
  139. package/lib/context/scan-admin/scan-api-effects.js.map +1 -0
  140. package/lib/context/scan-admin/scan-context-types.d.ts +69 -0
  141. package/lib/context/scan-admin/scan-context-types.d.ts.map +1 -0
  142. package/lib/context/scan-admin/scan-context-types.js +14 -0
  143. package/lib/context/scan-admin/scan-context-types.js.map +1 -0
  144. package/lib/context/scan-admin/scan-context.d.ts +17 -0
  145. package/lib/context/scan-admin/scan-context.d.ts.map +1 -0
  146. package/lib/context/scan-admin/scan-context.js +90 -0
  147. package/lib/context/scan-admin/scan-context.js.map +1 -0
  148. package/lib/context/scan-admin/scan-helpers.d.ts +28 -0
  149. package/lib/context/scan-admin/scan-helpers.d.ts.map +1 -0
  150. package/lib/context/scan-admin/scan-helpers.js +56 -0
  151. package/lib/context/scan-admin/scan-helpers.js.map +1 -0
  152. package/lib/context/scan-admin/scan-reducer.d.ts +18 -0
  153. package/lib/context/scan-admin/scan-reducer.d.ts.map +1 -0
  154. package/lib/context/scan-admin/scan-reducer.js +194 -0
  155. package/lib/context/scan-admin/scan-reducer.js.map +1 -0
  156. package/lib/context/scan-admin/scan-types.d.ts +292 -0
  157. package/lib/context/scan-admin/scan-types.d.ts.map +1 -0
  158. package/lib/context/scan-admin/scan-types.js +67 -0
  159. package/lib/context/scan-admin/scan-types.js.map +1 -0
  160. package/lib/default/page-settings.d.ts.map +1 -1
  161. package/lib/default/page-settings.js +3 -0
  162. package/lib/default/page-settings.js.map +1 -1
  163. package/lib/default/theme.d.ts +50 -12
  164. package/lib/default/theme.d.ts.map +1 -1
  165. package/lib/default/theme.js +42 -0
  166. package/lib/default/theme.js.map +1 -1
  167. package/lib/extension-registry-service.d.ts +106 -1
  168. package/lib/extension-registry-service.d.ts.map +1 -1
  169. package/lib/extension-registry-service.js +348 -1
  170. package/lib/extension-registry-service.js.map +1 -1
  171. package/lib/extension-registry-types.d.ts +181 -0
  172. package/lib/extension-registry-types.d.ts.map +1 -1
  173. package/lib/extension-registry-types.js +16 -0
  174. package/lib/extension-registry-types.js.map +1 -1
  175. package/lib/hooks/scan-admin/index.d.ts +34 -0
  176. package/lib/hooks/scan-admin/index.d.ts.map +1 -0
  177. package/lib/hooks/scan-admin/index.js +28 -0
  178. package/lib/hooks/scan-admin/index.js.map +1 -0
  179. package/lib/hooks/scan-admin/use-auto-rejected-tab.d.ts +76 -0
  180. package/lib/hooks/scan-admin/use-auto-rejected-tab.d.ts.map +1 -0
  181. package/lib/hooks/scan-admin/use-auto-rejected-tab.js +79 -0
  182. package/lib/hooks/scan-admin/use-auto-rejected-tab.js.map +1 -0
  183. package/lib/hooks/scan-admin/use-dialogs.d.ts +43 -0
  184. package/lib/hooks/scan-admin/use-dialogs.d.ts.map +1 -0
  185. package/lib/hooks/scan-admin/use-dialogs.js +55 -0
  186. package/lib/hooks/scan-admin/use-dialogs.js.map +1 -0
  187. package/lib/hooks/scan-admin/use-file-list-tab.d.ts +208 -0
  188. package/lib/hooks/scan-admin/use-file-list-tab.d.ts.map +1 -0
  189. package/lib/hooks/scan-admin/use-file-list-tab.js +125 -0
  190. package/lib/hooks/scan-admin/use-file-list-tab.js.map +1 -0
  191. package/lib/hooks/scan-admin/use-pagination.d.ts +36 -0
  192. package/lib/hooks/scan-admin/use-pagination.d.ts.map +1 -0
  193. package/lib/hooks/scan-admin/use-pagination.js +79 -0
  194. package/lib/hooks/scan-admin/use-pagination.js.map +1 -0
  195. package/lib/hooks/scan-admin/use-quarantined-tab.d.ts +92 -0
  196. package/lib/hooks/scan-admin/use-quarantined-tab.d.ts.map +1 -0
  197. package/lib/hooks/scan-admin/use-quarantined-tab.js +125 -0
  198. package/lib/hooks/scan-admin/use-quarantined-tab.js.map +1 -0
  199. package/lib/hooks/scan-admin/use-scan-card-state.d.ts +29 -0
  200. package/lib/hooks/scan-admin/use-scan-card-state.d.ts.map +1 -0
  201. package/lib/hooks/scan-admin/use-scan-card-state.js +50 -0
  202. package/lib/hooks/scan-admin/use-scan-card-state.js.map +1 -0
  203. package/lib/hooks/scan-admin/use-scan-filters.d.ts +53 -0
  204. package/lib/hooks/scan-admin/use-scan-filters.d.ts.map +1 -0
  205. package/lib/hooks/scan-admin/use-scan-filters.js +75 -0
  206. package/lib/hooks/scan-admin/use-scan-filters.js.map +1 -0
  207. package/lib/hooks/scan-admin/use-scans-tab.d.ts +85 -0
  208. package/lib/hooks/scan-admin/use-scans-tab.d.ts.map +1 -0
  209. package/lib/hooks/scan-admin/use-scans-tab.js +87 -0
  210. package/lib/hooks/scan-admin/use-scans-tab.js.map +1 -0
  211. package/lib/hooks/scan-admin/use-search.d.ts +32 -0
  212. package/lib/hooks/scan-admin/use-search.d.ts.map +1 -0
  213. package/lib/hooks/scan-admin/use-search.js +48 -0
  214. package/lib/hooks/scan-admin/use-search.js.map +1 -0
  215. package/lib/hooks/scan-admin/use-tab-navigation.d.ts +96 -0
  216. package/lib/hooks/scan-admin/use-tab-navigation.d.ts.map +1 -0
  217. package/lib/hooks/scan-admin/use-tab-navigation.js +70 -0
  218. package/lib/hooks/scan-admin/use-tab-navigation.js.map +1 -0
  219. package/lib/hooks/scan-admin/use-url-sync.d.ts +22 -0
  220. package/lib/hooks/scan-admin/use-url-sync.d.ts.map +1 -0
  221. package/lib/hooks/scan-admin/use-url-sync.js +252 -0
  222. package/lib/hooks/scan-admin/use-url-sync.js.map +1 -0
  223. package/lib/page-settings.d.ts +4 -0
  224. package/lib/page-settings.d.ts.map +1 -1
  225. package/lib/pages/admin-dashboard/admin-dashboard.d.ts +4 -0
  226. package/lib/pages/admin-dashboard/admin-dashboard.d.ts.map +1 -1
  227. package/lib/pages/admin-dashboard/admin-dashboard.js +55 -5
  228. package/lib/pages/admin-dashboard/admin-dashboard.js.map +1 -1
  229. package/lib/pages/admin-dashboard/components/data-grid-filter-operators.d.ts +28 -0
  230. package/lib/pages/admin-dashboard/components/data-grid-filter-operators.d.ts.map +1 -0
  231. package/lib/pages/admin-dashboard/components/data-grid-filter-operators.js +93 -0
  232. package/lib/pages/admin-dashboard/components/data-grid-filter-operators.js.map +1 -0
  233. package/lib/pages/admin-dashboard/components/index.d.ts +2 -0
  234. package/lib/pages/admin-dashboard/components/index.d.ts.map +1 -0
  235. package/lib/pages/admin-dashboard/components/index.js +14 -0
  236. package/lib/pages/admin-dashboard/components/index.js.map +1 -0
  237. package/lib/pages/admin-dashboard/customers/customer-form-dialog.d.ts +23 -0
  238. package/lib/pages/admin-dashboard/customers/customer-form-dialog.d.ts.map +1 -0
  239. package/lib/pages/admin-dashboard/customers/customer-form-dialog.js +225 -0
  240. package/lib/pages/admin-dashboard/customers/customer-form-dialog.js.map +1 -0
  241. package/lib/pages/admin-dashboard/customers/customers.d.ts +15 -0
  242. package/lib/pages/admin-dashboard/customers/customers.d.ts.map +1 -0
  243. package/lib/pages/admin-dashboard/customers/customers.js +175 -0
  244. package/lib/pages/admin-dashboard/customers/customers.js.map +1 -0
  245. package/lib/pages/admin-dashboard/customers/delete-customer-dialog.d.ts +23 -0
  246. package/lib/pages/admin-dashboard/customers/delete-customer-dialog.d.ts.map +1 -0
  247. package/lib/pages/admin-dashboard/customers/delete-customer-dialog.js +64 -0
  248. package/lib/pages/admin-dashboard/customers/delete-customer-dialog.js.map +1 -0
  249. package/lib/pages/admin-dashboard/publisher-admin.js +5 -5
  250. package/lib/pages/admin-dashboard/publisher-admin.js.map +1 -1
  251. package/lib/pages/admin-dashboard/publisher-revoke-dialog.d.ts.map +1 -1
  252. package/lib/pages/admin-dashboard/publisher-revoke-dialog.js +1 -1
  253. package/lib/pages/admin-dashboard/publisher-revoke-dialog.js.map +1 -1
  254. package/lib/pages/admin-dashboard/scan-admin.d.ts +20 -0
  255. package/lib/pages/admin-dashboard/scan-admin.d.ts.map +1 -0
  256. package/lib/pages/admin-dashboard/scan-admin.js +66 -0
  257. package/lib/pages/admin-dashboard/scan-admin.js.map +1 -0
  258. package/lib/pages/admin-dashboard/tiers/delete-tier-dialog.d.ts +23 -0
  259. package/lib/pages/admin-dashboard/tiers/delete-tier-dialog.d.ts.map +1 -0
  260. package/lib/pages/admin-dashboard/tiers/delete-tier-dialog.js +55 -0
  261. package/lib/pages/admin-dashboard/tiers/delete-tier-dialog.js.map +1 -0
  262. package/lib/pages/admin-dashboard/tiers/tier-form-dialog.d.ts +23 -0
  263. package/lib/pages/admin-dashboard/tiers/tier-form-dialog.d.ts.map +1 -0
  264. package/lib/pages/admin-dashboard/tiers/tier-form-dialog.js +215 -0
  265. package/lib/pages/admin-dashboard/tiers/tier-form-dialog.js.map +1 -0
  266. package/lib/pages/admin-dashboard/tiers/tiers.d.ts +15 -0
  267. package/lib/pages/admin-dashboard/tiers/tiers.d.ts.map +1 -0
  268. package/lib/pages/admin-dashboard/tiers/tiers.js +174 -0
  269. package/lib/pages/admin-dashboard/tiers/tiers.js.map +1 -0
  270. package/lib/pages/admin-dashboard/usage-stats/usage-stats-chart.d.ts +23 -0
  271. package/lib/pages/admin-dashboard/usage-stats/usage-stats-chart.d.ts.map +1 -0
  272. package/lib/pages/admin-dashboard/usage-stats/usage-stats-chart.js +106 -0
  273. package/lib/pages/admin-dashboard/usage-stats/usage-stats-chart.js.map +1 -0
  274. package/lib/pages/admin-dashboard/usage-stats/usage-stats-search.d.ts +26 -0
  275. package/lib/pages/admin-dashboard/usage-stats/usage-stats-search.d.ts.map +1 -0
  276. package/lib/pages/admin-dashboard/usage-stats/usage-stats-search.js +50 -0
  277. package/lib/pages/admin-dashboard/usage-stats/usage-stats-search.js.map +1 -0
  278. package/lib/pages/admin-dashboard/usage-stats/usage-stats-utils.d.ts +14 -0
  279. package/lib/pages/admin-dashboard/usage-stats/usage-stats-utils.d.ts.map +1 -0
  280. package/lib/pages/admin-dashboard/usage-stats/usage-stats-utils.js +16 -0
  281. package/lib/pages/admin-dashboard/usage-stats/usage-stats-utils.js.map +1 -0
  282. package/lib/pages/admin-dashboard/usage-stats/usage-stats.d.ts +15 -0
  283. package/lib/pages/admin-dashboard/usage-stats/usage-stats.d.ts.map +1 -0
  284. package/lib/pages/admin-dashboard/usage-stats/usage-stats.js +103 -0
  285. package/lib/pages/admin-dashboard/usage-stats/usage-stats.js.map +1 -0
  286. package/lib/pages/admin-dashboard/welcome.d.ts.map +1 -1
  287. package/lib/pages/admin-dashboard/welcome.js +5 -1
  288. package/lib/pages/admin-dashboard/welcome.js.map +1 -1
  289. package/lib/pages/extension-detail/extension-detail-overview.d.ts.map +1 -1
  290. package/lib/pages/extension-detail/extension-detail-overview.js +9 -1
  291. package/lib/pages/extension-detail/extension-detail-overview.js.map +1 -1
  292. package/lib/pages/extension-detail/extension-detail-reviews.d.ts.map +1 -1
  293. package/lib/pages/extension-detail/extension-detail-reviews.js +67 -18
  294. package/lib/pages/extension-detail/extension-detail-reviews.js.map +1 -1
  295. package/lib/pages/extension-detail/extension-detail.js +2 -2
  296. package/lib/pages/extension-detail/extension-detail.js.map +1 -1
  297. package/lib/pages/extension-detail/extension-rating-stars.d.ts +2 -2
  298. package/lib/pages/extension-detail/extension-rating-stars.d.ts.map +1 -1
  299. package/lib/pages/extension-detail/extension-rating-stars.js +1 -1
  300. package/lib/pages/extension-detail/extension-rating-stars.js.map +1 -1
  301. package/lib/pages/extension-list/extension-list-item.js +2 -2
  302. package/lib/pages/extension-list/extension-list-item.js.map +1 -1
  303. package/lib/pages/user/user-namespace-extension-list-item.d.ts.map +1 -1
  304. package/lib/pages/user/user-namespace-extension-list-item.js +23 -9
  305. package/lib/pages/user/user-namespace-extension-list-item.js.map +1 -1
  306. package/lib/pages/user/user-publisher-agreement.d.ts.map +1 -1
  307. package/lib/pages/user/user-publisher-agreement.js +34 -13
  308. package/lib/pages/user/user-publisher-agreement.js.map +1 -1
  309. package/lib/pages/user/user-settings-profile.d.ts.map +1 -1
  310. package/lib/pages/user/user-settings-profile.js +10 -5
  311. package/lib/pages/user/user-settings-profile.js.map +1 -1
  312. package/lib/pages/user/user-settings-tokens.d.ts.map +1 -1
  313. package/lib/pages/user/user-settings-tokens.js +13 -5
  314. package/lib/pages/user/user-settings-tokens.js.map +1 -1
  315. package/lib/server-request.d.ts +1 -1
  316. package/lib/server-request.d.ts.map +1 -1
  317. package/lib/server-request.js +3 -3
  318. package/lib/server-request.js.map +1 -1
  319. package/package.json +17 -9
  320. package/src/components/button-with-progress.tsx +2 -1
  321. package/src/components/sanitized-markdown.tsx +81 -1
  322. package/src/components/scan-admin/common/auto-refresh.tsx +79 -0
  323. package/src/components/scan-admin/common/conditional-tooltip.tsx +74 -0
  324. package/src/components/scan-admin/common/file-table.tsx +508 -0
  325. package/src/components/scan-admin/common/index.ts +18 -0
  326. package/src/components/scan-admin/common/tab-panel.tsx +42 -0
  327. package/src/components/scan-admin/common/utils.ts +48 -0
  328. package/src/components/scan-admin/dialogs/file-dialog.tsx +137 -0
  329. package/src/components/scan-admin/dialogs/index.ts +15 -0
  330. package/src/components/scan-admin/dialogs/quarantine-dialog.tsx +270 -0
  331. package/src/components/scan-admin/index.ts +52 -0
  332. package/src/components/scan-admin/scan-card/index.ts +20 -0
  333. package/src/components/scan-admin/scan-card/scan-card-content.tsx +457 -0
  334. package/src/components/scan-admin/scan-card/scan-card-expand-strip-badges.tsx +186 -0
  335. package/src/components/scan-admin/scan-card/scan-card-expand-strip.tsx +104 -0
  336. package/src/components/scan-admin/scan-card/scan-card-expanded-content.tsx +262 -0
  337. package/src/components/scan-admin/scan-card/scan-card-header.tsx +176 -0
  338. package/src/components/scan-admin/scan-card/scan-card.tsx +152 -0
  339. package/src/components/scan-admin/scan-card/scan-detail-card.tsx +144 -0
  340. package/src/components/scan-admin/scan-card/utils.ts +199 -0
  341. package/src/components/scan-admin/tab-contents/allow-list-tab-content.tsx +119 -0
  342. package/src/components/scan-admin/tab-contents/auto-rejected-tab-content.tsx +106 -0
  343. package/src/components/scan-admin/tab-contents/block-list-tab-content.tsx +119 -0
  344. package/src/components/scan-admin/tab-contents/index.ts +18 -0
  345. package/src/components/scan-admin/tab-contents/quarantined-tab-content.tsx +158 -0
  346. package/src/components/scan-admin/tab-contents/scans-tab-content.tsx +113 -0
  347. package/src/components/scan-admin/toolbars/counts-toolbar.tsx +262 -0
  348. package/src/components/scan-admin/toolbars/index.ts +16 -0
  349. package/src/components/scan-admin/toolbars/search-toolbar.tsx +255 -0
  350. package/src/components/scan-admin/toolbars/tab-toolbar.tsx +56 -0
  351. package/src/context/scan-admin/index.ts +43 -0
  352. package/src/context/scan-admin/scan-actions.ts +87 -0
  353. package/src/context/scan-admin/scan-api-actions.ts +137 -0
  354. package/src/context/scan-admin/scan-api-effects.ts +491 -0
  355. package/src/context/scan-admin/scan-context-types.ts +102 -0
  356. package/src/context/scan-admin/scan-context.tsx +123 -0
  357. package/src/context/scan-admin/scan-helpers.ts +63 -0
  358. package/src/context/scan-admin/scan-reducer.ts +321 -0
  359. package/src/context/scan-admin/scan-types.ts +337 -0
  360. package/src/default/page-settings.tsx +3 -0
  361. package/src/default/theme.tsx +102 -13
  362. package/src/extension-registry-service.ts +372 -1
  363. package/src/extension-registry-types.ts +207 -0
  364. package/src/hooks/scan-admin/index.ts +48 -0
  365. package/src/hooks/scan-admin/use-auto-rejected-tab.ts +85 -0
  366. package/src/hooks/scan-admin/use-dialogs.ts +83 -0
  367. package/src/hooks/scan-admin/use-file-list-tab.ts +149 -0
  368. package/src/hooks/scan-admin/use-pagination.ts +90 -0
  369. package/src/hooks/scan-admin/use-quarantined-tab.ts +138 -0
  370. package/src/hooks/scan-admin/use-scan-card-state.ts +73 -0
  371. package/src/hooks/scan-admin/use-scan-filters.ts +83 -0
  372. package/src/hooks/scan-admin/use-scans-tab.ts +92 -0
  373. package/src/hooks/scan-admin/use-search.ts +54 -0
  374. package/src/hooks/scan-admin/use-tab-navigation.ts +82 -0
  375. package/src/hooks/scan-admin/use-url-sync.ts +293 -0
  376. package/src/page-settings.ts +4 -0
  377. package/src/pages/admin-dashboard/admin-dashboard.tsx +71 -4
  378. package/src/pages/admin-dashboard/components/data-grid-filter-operators.tsx +126 -0
  379. package/src/pages/admin-dashboard/components/index.ts +18 -0
  380. package/src/pages/admin-dashboard/customers/customer-form-dialog.tsx +348 -0
  381. package/src/pages/admin-dashboard/customers/customers.tsx +279 -0
  382. package/src/pages/admin-dashboard/customers/delete-customer-dialog.tsx +94 -0
  383. package/src/pages/admin-dashboard/publisher-admin.tsx +5 -5
  384. package/src/pages/admin-dashboard/publisher-revoke-dialog.tsx +2 -1
  385. package/src/pages/admin-dashboard/scan-admin.tsx +109 -0
  386. package/src/pages/admin-dashboard/tiers/delete-tier-dialog.tsx +83 -0
  387. package/src/pages/admin-dashboard/tiers/tier-form-dialog.tsx +372 -0
  388. package/src/pages/admin-dashboard/tiers/tiers.tsx +254 -0
  389. package/src/pages/admin-dashboard/usage-stats/usage-stats-chart.tsx +189 -0
  390. package/src/pages/admin-dashboard/usage-stats/usage-stats-search.tsx +83 -0
  391. package/src/pages/admin-dashboard/usage-stats/usage-stats-utils.ts +16 -0
  392. package/src/pages/admin-dashboard/usage-stats/usage-stats.tsx +127 -0
  393. package/src/pages/admin-dashboard/welcome.tsx +4 -0
  394. package/src/pages/extension-detail/extension-detail-overview.tsx +16 -2
  395. package/src/pages/extension-detail/extension-detail-reviews.tsx +116 -30
  396. package/src/pages/extension-detail/extension-detail.tsx +2 -2
  397. package/src/pages/extension-detail/extension-rating-stars.tsx +2 -2
  398. package/src/pages/extension-list/extension-list-item.tsx +2 -2
  399. package/src/pages/user/user-namespace-extension-list-item.tsx +38 -13
  400. package/src/pages/user/user-publisher-agreement.tsx +30 -11
  401. package/src/pages/user/user-settings-profile.tsx +10 -6
  402. package/src/pages/user/user-settings-tokens.tsx +15 -6
  403. package/src/server-request.ts +3 -3
@@ -0,0 +1,372 @@
1
+ /******************************************************************************
2
+ * Copyright (c) 2026 Contributors to the Eclipse Foundation.
3
+ *
4
+ * See the NOTICE file(s) distributed with this work for additional
5
+ * information regarding copyright ownership.
6
+ *
7
+ * This program and the accompanying materials are made available under the
8
+ * terms of the Eclipse Public License 2.0 which is available at
9
+ * https://www.eclipse.org/legal/epl-2.0.
10
+ *
11
+ * SPDX-License-Identifier: EPL-2.0
12
+ *****************************************************************************/
13
+
14
+ import React, { FC, useEffect, useState } from 'react';
15
+ import type { SelectChangeEvent } from '@mui/material';
16
+ import {
17
+ Alert,
18
+ Box,
19
+ Button,
20
+ CircularProgress,
21
+ Dialog,
22
+ DialogActions,
23
+ DialogContent,
24
+ DialogTitle,
25
+ FormControl,
26
+ FormHelperText,
27
+ InputLabel,
28
+ MenuItem,
29
+ Select,
30
+ TextField
31
+ } from '@mui/material';
32
+ import { RefillStrategy, type Tier, TierType } from "../../../extension-registry-types";
33
+ import { handleError } from "../../../utils";
34
+
35
+ type DurationUnit = 'seconds' | 'minutes' | 'hours';
36
+
37
+ const DURATION_MULTIPLIERS: Record<DurationUnit, number> = {
38
+ seconds: 1,
39
+ minutes: 60,
40
+ hours: 3600
41
+ };
42
+
43
+ function formatDuration(duration: number): [number, DurationUnit] {
44
+ const hours = Math.floor(duration / 3600);
45
+ if (hours > 0) {
46
+ return [hours, "hours"];
47
+ }
48
+
49
+ const minutes = Math.floor(duration / 60);
50
+ if (minutes > 0) {
51
+ return [minutes, "minutes"];
52
+ }
53
+
54
+ return [duration, "seconds"];
55
+ }
56
+
57
+ interface TierFormDialogProps {
58
+ open: boolean;
59
+ tier?: Tier;
60
+ onClose: () => void;
61
+ onSubmit: (formData: Tier) => Promise<void>;
62
+ }
63
+
64
+ export const TierFormDialog: FC<TierFormDialogProps> = ({ open, tier, onClose, onSubmit }) => {
65
+ const [formData, setFormData] = useState<Tier>({
66
+ name: '',
67
+ description: '',
68
+ capacity: 100,
69
+ duration: 3600,
70
+ refillStrategy: RefillStrategy.INTERVAL
71
+ } as Tier);
72
+ const [durationValue, setDurationValue] = useState(1);
73
+ const [durationUnit, setDurationUnit] = useState<DurationUnit>('hours');
74
+ const [loading, setLoading] = useState(false);
75
+ const [errors, setErrors] = useState<Record<string, string>>({});
76
+ const [touched, setTouched] = useState<Record<string, boolean>>({});
77
+
78
+ const getDurationInSeconds = (): number => {
79
+ return durationValue * DURATION_MULTIPLIERS[durationUnit];
80
+ };
81
+
82
+ useEffect(() => {
83
+ if (tier) {
84
+ setFormData(_ => ({
85
+ name: tier.name,
86
+ description: tier.description || '',
87
+ tierType: tier.tierType,
88
+ capacity: tier.capacity,
89
+ duration: tier.duration,
90
+ refillStrategy: tier.refillStrategy
91
+ } as Tier));
92
+ // Convert duration seconds to value/unit for display
93
+ const [value, unit] = formatDuration(tier.duration);
94
+ setDurationValue(value);
95
+ setDurationUnit(unit);
96
+ } else {
97
+ setFormData(prev => ({
98
+ ...prev,
99
+ name: '',
100
+ description: '',
101
+ tierType: TierType.NON_FREE,
102
+ capacity: 100,
103
+ duration: 3600,
104
+ refillStrategy: RefillStrategy.INTERVAL
105
+ }));
106
+ setDurationValue(1);
107
+ setDurationUnit('hours');
108
+ }
109
+ setErrors({});
110
+ setTouched({});
111
+ }, [open, tier]);
112
+
113
+ const clearFieldError = (fieldName: string) => {
114
+ if (errors[fieldName]) {
115
+ setErrors(prev => {
116
+ const newErrors = { ...prev };
117
+ delete newErrors[fieldName];
118
+ return newErrors;
119
+ });
120
+ }
121
+ };
122
+
123
+ const handleChange = (e: React.ChangeEvent<HTMLInputElement | { name?: string; value: unknown }> | SelectChangeEvent) => {
124
+ const { name, value } = e.target as any;
125
+ clearFieldError(name);
126
+
127
+ setFormData((prev: Tier) => ({
128
+ ...prev,
129
+ [name]: name === 'capacity' || name === 'duration' ? Number.parseInt(value as string, 10) : value
130
+ } as Tier));
131
+ };
132
+
133
+ const handleBlur = (e: React.FocusEvent<HTMLInputElement | HTMLTextAreaElement>) => {
134
+ const { name } = e.target;
135
+ setTouched(prev => ({ ...prev, [name]: true }));
136
+ validateField(name);
137
+ };
138
+
139
+ const fieldValidators: Record<string, () => string | undefined> = {
140
+ name: () => {
141
+ if (formData.name === undefined) {
142
+ return "Tier name is required";
143
+ } else if (formData.name.trim() !== formData.name) {
144
+ return "Tier name must not contain trailing whitespace";
145
+ } else {
146
+ return undefined;
147
+ }
148
+ },
149
+ tierType: () => formData.tierType ? undefined : 'Tier type is required',
150
+ capacity: () => formData.capacity <= 0 ? 'Capacity must be greater than 0' : undefined,
151
+ duration: () => durationValue <= 0 ? 'Duration must be greater than 0' : undefined,
152
+ refillStrategy: () => formData.refillStrategy ? undefined : 'Refill strategy is required',
153
+ };
154
+
155
+ const validateField = (fieldName: string): string | undefined => {
156
+ const validator = fieldValidators[fieldName];
157
+ const error = validator?.();
158
+
159
+ if (error) {
160
+ setErrors(prev => ({ ...prev, [fieldName]: error }));
161
+ }
162
+ return error;
163
+ };
164
+
165
+ const validateForm = (): boolean => {
166
+ // Mark all fields as touched on submit
167
+ setTouched({
168
+ name: true,
169
+ tierType: true,
170
+ capacity: true,
171
+ duration: true,
172
+ refillStrategy: true,
173
+ });
174
+
175
+ const newErrors: Record<string, string> = {};
176
+ for (const key of Object.keys(formData)) {
177
+ const error = validateField(key);
178
+ if (error !== undefined) {
179
+ newErrors[key] = error;
180
+ }
181
+ }
182
+
183
+ setErrors(newErrors);
184
+ return Object.keys(newErrors).length === 0;
185
+ };
186
+
187
+ const handleSubmit = async () => {
188
+ if (!validateForm()) {
189
+ return;
190
+ }
191
+
192
+ setLoading(true);
193
+
194
+ try {
195
+ const durationInSeconds = getDurationInSeconds();
196
+ await onSubmit({
197
+ ...formData,
198
+ duration: durationInSeconds
199
+ });
200
+ onClose();
201
+ } catch (err: any) {
202
+ setErrors({ submit: handleError(err) });
203
+ } finally {
204
+ setLoading(false);
205
+ }
206
+ };
207
+
208
+ const isEditMode = !!tier;
209
+ const title = isEditMode ? 'Edit Tier' : 'Create New Tier';
210
+
211
+ return (
212
+ <Dialog open={open} onClose={onClose} maxWidth='sm' fullWidth>
213
+ <DialogTitle>{title}</DialogTitle>
214
+ <DialogContent>
215
+ <Box sx={{ display: 'flex', flexDirection: 'column', gap: 2, pt: 2 }}>
216
+ {errors.submit && <Alert severity='error'>{errors.submit}</Alert>}
217
+
218
+ <TextField
219
+ label='Tier Name'
220
+ name='name'
221
+ value={formData.name}
222
+ onChange={handleChange}
223
+ onBlur={handleBlur}
224
+ fullWidth
225
+ placeholder='e.g., Professional, Enterprise'
226
+ required={true}
227
+ disabled={loading}
228
+ error={touched.name && !!errors.name}
229
+ helperText={touched.name && errors.name}
230
+ />
231
+
232
+ <TextField
233
+ label='Description'
234
+ name='description'
235
+ value={formData.description}
236
+ onChange={handleChange}
237
+ fullWidth
238
+ multiline
239
+ rows={3}
240
+ placeholder='Optional description for this tier'
241
+ disabled={loading}
242
+ />
243
+
244
+ <FormControl fullWidth disabled={loading} required={true} error={touched.tierType && !!errors.tierType}>
245
+ <InputLabel>Tier Type</InputLabel>
246
+ <Select
247
+ name='tierType'
248
+ value={formData.tierType || ''}
249
+ onChange={(e: SelectChangeEvent) => {
250
+ const { name, value } = e.target;
251
+ clearFieldError(name);
252
+ setFormData((prev: Tier) => ({
253
+ ...prev,
254
+ [name]: value
255
+ } as Tier));
256
+ }}
257
+ onBlur={() => {
258
+ setTouched(prev => ({ ...prev, tierType: true }));
259
+ validateField('tierType');
260
+ }}
261
+ label='Tier Type'
262
+ >
263
+ {Object.keys(TierType).map(tierType => (
264
+ <MenuItem key={tierType} value={TierType[tierType as keyof typeof TierType]}>
265
+ {tierType}
266
+ </MenuItem>
267
+ ))}
268
+ </Select>
269
+ {touched.tierType && errors.tierType && <FormHelperText>{errors.tierType}</FormHelperText>}
270
+ </FormControl>
271
+
272
+ <TextField
273
+ label='Capacity'
274
+ name='capacity'
275
+ type='number'
276
+ value={formData.capacity}
277
+ onChange={handleChange}
278
+ onBlur={handleBlur}
279
+ fullWidth
280
+ inputProps={{ min: '1' }}
281
+ disabled={loading}
282
+ required={true}
283
+ error={touched.capacity && !!errors.capacity}
284
+ helperText={(touched.capacity && errors.capacity) || 'Maximum number of requests allowed per interval'}
285
+ />
286
+
287
+ <Box sx={{ display: 'flex', gap: 2 }}>
288
+ <TextField
289
+ label='Duration'
290
+ name='duration'
291
+ type='number'
292
+ value={durationValue}
293
+ onChange={(e) => {
294
+ clearFieldError('duration');
295
+ setDurationValue(Math.max(1, Number.parseInt(e.target.value, 10) || 0));
296
+ }}
297
+ onBlur={(e) => {
298
+ setTouched(prev => ({ ...prev, duration: true }));
299
+ validateField('duration');
300
+ }}
301
+ inputProps={{ min: '1' }}
302
+ disabled={loading}
303
+ required={true}
304
+ error={touched.duration && !!errors.duration}
305
+ helperText={touched.duration && errors.duration}
306
+ sx={{ flex: 1 }}
307
+ />
308
+ <FormControl disabled={loading} required={true} sx={{ minWidth: 150 }}>
309
+ <InputLabel>Unit</InputLabel>
310
+ <Select
311
+ value={durationUnit}
312
+ onChange={(e: SelectChangeEvent) => setDurationUnit(e.target.value as DurationUnit)}
313
+ label='Unit'
314
+ >
315
+ <MenuItem value='seconds'>Seconds</MenuItem>
316
+ <MenuItem value='minutes'>Minutes</MenuItem>
317
+ <MenuItem value='hours'>Hours</MenuItem>
318
+ </Select>
319
+ </FormControl>
320
+ </Box>
321
+ <Box sx={{ fontSize: '0.875rem', color: 'text.secondary' }}>
322
+ = {getDurationInSeconds().toLocaleString()} seconds
323
+ </Box>
324
+
325
+ <FormControl fullWidth disabled={loading} required={true} error={touched.refillStrategy && !!errors.refillStrategy}>
326
+ <InputLabel>Refill Strategy</InputLabel>
327
+ <Select
328
+ name='refillStrategy'
329
+ value={formData.refillStrategy || ''}
330
+ onChange={(e: SelectChangeEvent) => {
331
+ const { name, value } = e.target;
332
+ clearFieldError(name);
333
+ setFormData((prev: Tier) => ({
334
+ ...prev,
335
+ [name]: value
336
+ } as Tier));
337
+ }}
338
+ onBlur={() => {
339
+ setTouched(prev => ({ ...prev, refillStrategy: true }));
340
+ validateField('refillStrategy');
341
+ }}
342
+ label='Refill Strategy'
343
+ >
344
+ {Object.keys(RefillStrategy).map(strategy => (
345
+ <MenuItem key={strategy} value={RefillStrategy[strategy as keyof typeof RefillStrategy]}>
346
+ {strategy}
347
+ </MenuItem>
348
+ ))}
349
+ </Select>
350
+ {touched.refillStrategy && errors.refillStrategy && <FormHelperText>{errors.refillStrategy}</FormHelperText>}
351
+ </FormControl>
352
+
353
+
354
+ </Box>
355
+ </DialogContent>
356
+
357
+ <DialogActions sx={{ p: 2 }}>
358
+ <Button onClick={onClose} disabled={loading}>
359
+ Cancel
360
+ </Button>
361
+ <Button
362
+ onClick={handleSubmit}
363
+ variant='contained'
364
+ disabled={loading || Object.keys(errors).length > 0}
365
+ startIcon={loading ? <CircularProgress size={20} /> : undefined}
366
+ >
367
+ {isEditMode ? 'Update' : 'Create'}
368
+ </Button>
369
+ </DialogActions>
370
+ </Dialog>
371
+ );
372
+ };
@@ -0,0 +1,254 @@
1
+ /******************************************************************************
2
+ * Copyright (c) 2026 Contributors to the Eclipse Foundation.
3
+ *
4
+ * See the NOTICE file(s) distributed with this work for additional
5
+ * information regarding copyright ownership.
6
+ *
7
+ * This program and the accompanying materials are made available under the
8
+ * terms of the Eclipse Public License 2.0 which is available at
9
+ * https://www.eclipse.org/legal/epl-2.0.
10
+ *
11
+ * SPDX-License-Identifier: EPL-2.0
12
+ *****************************************************************************/
13
+
14
+ import React, { FC, useState, useEffect, useRef, useMemo } from "react";
15
+ import {
16
+ Box,
17
+ Button,
18
+ Paper,
19
+ Typography,
20
+ CircularProgress,
21
+ Alert,
22
+ IconButton
23
+ } from "@mui/material";
24
+ import { DataGrid, GridColDef, GridRenderCellParams } from "@mui/x-data-grid";
25
+ import EditIcon from "@mui/icons-material/Edit";
26
+ import DeleteIcon from "@mui/icons-material/Delete";
27
+ import AddIcon from "@mui/icons-material/Add";
28
+ import { MainContext } from "../../../context";
29
+ import type { Tier } from "../../../extension-registry-types";
30
+ import { TierFormDialog } from "./tier-form-dialog";
31
+ import { DeleteTierDialog } from "./delete-tier-dialog";
32
+ import { handleError } from "../../../utils";
33
+ import { createMultiSelectFilterOperators } from "../components";
34
+
35
+ export const Tiers: FC = () => {
36
+ const abortController = useRef<AbortController>(new AbortController());
37
+ const { service } = React.useContext(MainContext);
38
+ const [tiers, setTiers] = useState<readonly Tier[]>([]);
39
+ const [loading, setLoading] = useState(true);
40
+ const [error, setError] = useState<string | null>(null);
41
+ const [formDialogOpen, setFormDialogOpen] = useState(false);
42
+ const [deleteDialogOpen, setDeleteDialogOpen] = useState(false);
43
+ const [selectedTier, setSelectedTier] = useState<Tier | undefined>();
44
+
45
+ // load all tiers
46
+ const loadTiers = async () => {
47
+ try {
48
+ setLoading(true);
49
+ setError(null);
50
+ const data = await service.admin.getTiers(abortController.current);
51
+ setTiers(data.tiers);
52
+ } catch (err: any) {
53
+ setError(handleError(err));
54
+ } finally {
55
+ setLoading(false);
56
+ }
57
+ };
58
+
59
+ useEffect(() => {
60
+ loadTiers();
61
+ return () => abortController.current.abort();
62
+ }, []);
63
+
64
+ const handleCreateClick = () => {
65
+ setSelectedTier(undefined);
66
+ setFormDialogOpen(true);
67
+ };
68
+
69
+ const handleEditClick = (tier: Tier) => {
70
+ setSelectedTier(tier);
71
+ setFormDialogOpen(true);
72
+ };
73
+
74
+ const handleDeleteClick = (tier: Tier) => {
75
+ setSelectedTier(tier);
76
+ setDeleteDialogOpen(true);
77
+ };
78
+
79
+ const handleFormSubmit = async (formData: Tier) => {
80
+ if (selectedTier) {
81
+ // update existing tier
82
+ await service.admin.updateTier(abortController.current, selectedTier.name, formData);
83
+ } else {
84
+ // create new tier
85
+ await service.admin.createTier(abortController.current, formData);
86
+ }
87
+ await loadTiers();
88
+ };
89
+
90
+ const handleDeleteConfirm = async () => {
91
+ if (selectedTier) {
92
+ await service.admin.deleteTier(abortController.current, selectedTier.name);
93
+ await loadTiers();
94
+ }
95
+ };
96
+
97
+ const handleFormDialogClose = () => {
98
+ setFormDialogOpen(false);
99
+ setSelectedTier(undefined);
100
+ };
101
+
102
+ const handleDeleteDialogClose = () => {
103
+ setDeleteDialogOpen(false);
104
+ setSelectedTier(undefined);
105
+ };
106
+
107
+ // Extract unique values for filter dropdowns
108
+ const tierTypeOptions = useMemo(() =>
109
+ [...new Set(tiers.map(t => t.tierType).filter(Boolean))],
110
+ [tiers]
111
+ );
112
+
113
+ const refillStrategyOptions = useMemo(() =>
114
+ [...new Set(tiers.map(t => t.refillStrategy).filter(Boolean))],
115
+ [tiers]
116
+ );
117
+
118
+ const columns: GridColDef[] = [
119
+ {
120
+ field: 'name',
121
+ headerName: 'Name',
122
+ flex: 1,
123
+ minWidth: 150
124
+ },
125
+ {
126
+ field: 'description',
127
+ headerName: 'Description',
128
+ flex: 2,
129
+ minWidth: 200,
130
+ valueGetter: (value: string) => value || '-'
131
+ },
132
+ {
133
+ field: 'tierType',
134
+ headerName: 'Tier Type',
135
+ width: 150,
136
+ filterOperators: createMultiSelectFilterOperators(tierTypeOptions)
137
+ },
138
+ {
139
+ field: 'capacity',
140
+ headerName: 'Capacity',
141
+ type: 'number',
142
+ width: 120,
143
+ valueFormatter: (value: number) => value.toLocaleString()
144
+ },
145
+ {
146
+ field: 'duration',
147
+ headerName: 'Duration (s)',
148
+ type: 'number',
149
+ width: 130,
150
+ valueFormatter: (value: number) => value.toLocaleString()
151
+ },
152
+ {
153
+ field: 'refillStrategy',
154
+ headerName: 'Refill Strategy',
155
+ width: 150,
156
+ filterOperators: createMultiSelectFilterOperators(refillStrategyOptions)
157
+ },
158
+ {
159
+ field: 'actions',
160
+ headerName: 'Actions',
161
+ width: 120,
162
+ sortable: false,
163
+ filterable: false,
164
+ renderCell: (params: GridRenderCellParams<Tier>) => (
165
+ <>
166
+ <IconButton
167
+ size='small'
168
+ onClick={() => handleEditClick(params.row)}
169
+ title='Edit tier'
170
+ color='primary'
171
+ >
172
+ <EditIcon />
173
+ </IconButton>
174
+ <IconButton
175
+ size='small'
176
+ onClick={() => handleDeleteClick(params.row)}
177
+ title='Delete tier'
178
+ color='error'
179
+ >
180
+ <DeleteIcon />
181
+ </IconButton>
182
+ </>
183
+ ),
184
+ },
185
+ ];
186
+
187
+ return (
188
+ <Box sx={{ p: 3, height: '100%', display: 'flex', flexDirection: 'column' }}>
189
+ <Box sx={{ display: "flex", justifyContent: "space-between", alignItems: "center", mb: 3 }}>
190
+ <Typography variant='h4' component='h1'>
191
+ Tier Management
192
+ </Typography>
193
+ <Button
194
+ variant='contained'
195
+ startIcon={<AddIcon />}
196
+ onClick={handleCreateClick}
197
+ disabled={loading}
198
+ >
199
+ Create Tier
200
+ </Button>
201
+ </Box>
202
+
203
+ { error &&
204
+ <Alert severity='error' sx={{ mb: 2 }} onClose={() => setError(null)}>
205
+ {error}
206
+ </Alert>
207
+ }
208
+
209
+ { loading &&
210
+ <Box sx={{ display: "flex", justifyContent: "center", p: 4 }}>
211
+ <CircularProgress />
212
+ </Box>
213
+ }
214
+
215
+ { !loading && tiers.length === 0 &&
216
+ <Paper elevation={0} sx={{ p: 3, textAlign: "center" }}>
217
+ <Typography color='textSecondary' gutterBottom>
218
+ No tiers found. Create one to get started.
219
+ </Typography>
220
+ </Paper>
221
+ }
222
+
223
+ { !loading && tiers.length > 0 &&
224
+ <Paper elevation={0} sx={{ flex: 1, minHeight: 400, width: '100%', display: 'flex', flexDirection: 'column' }}>
225
+ <DataGrid
226
+ rows={tiers as Tier[]}
227
+ columns={columns}
228
+ getRowId={(row) => row.name}
229
+ pageSizeOptions={[20, 35, 50]}
230
+ initialState={{
231
+ pagination: { paginationModel: { pageSize: 20 } },
232
+ }}
233
+ disableRowSelectionOnClick
234
+ sx={{ flex: 1 }}
235
+ />
236
+ </Paper>
237
+ }
238
+
239
+ <TierFormDialog
240
+ open={formDialogOpen}
241
+ tier={selectedTier}
242
+ onClose={handleFormDialogClose}
243
+ onSubmit={handleFormSubmit}
244
+ />
245
+
246
+ <DeleteTierDialog
247
+ open={deleteDialogOpen}
248
+ tier={selectedTier}
249
+ onClose={handleDeleteDialogClose}
250
+ onConfirm={handleDeleteConfirm}
251
+ />
252
+ </Box>
253
+ );
254
+ };