@igstack/app-catalog-frontend-core 0.0.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 (652) hide show
  1. package/LICENSE +21 -0
  2. package/dist/esm/App.d.ts +12 -0
  3. package/dist/esm/App.js +12 -0
  4. package/dist/esm/App.js.map +1 -0
  5. package/dist/esm/__tests__/setupTests.d.ts +0 -0
  6. package/dist/esm/api/infra/createQueryClient.d.ts +9 -0
  7. package/dist/esm/api/infra/createQueryClient.js +25 -0
  8. package/dist/esm/api/infra/createQueryClient.js.map +1 -0
  9. package/dist/esm/api/infra/trpc.d.ts +1624 -0
  10. package/dist/esm/api/infra/trpc.js +8 -0
  11. package/dist/esm/api/infra/trpc.js.map +1 -0
  12. package/dist/esm/api/unsorted/appCatalogFetcher.d.ts +10 -0
  13. package/dist/esm/api/unsorted/appCatalogFetcher.js +18 -0
  14. package/dist/esm/api/unsorted/appCatalogFetcher.js.map +1 -0
  15. package/dist/esm/api/unsorted/createCachingFetcher.d.ts +23 -0
  16. package/dist/esm/api/unsorted/createCachingFetcher.js +93 -0
  17. package/dist/esm/api/unsorted/createCachingFetcher.js.map +1 -0
  18. package/dist/esm/appPropsFactory.d.ts +2 -0
  19. package/dist/esm/appPropsFactory.js +34 -0
  20. package/dist/esm/appPropsFactory.js.map +1 -0
  21. package/dist/esm/assets/app-catalog.svg.js +6 -0
  22. package/dist/esm/assets/app-catalog.svg.js.map +1 -0
  23. package/dist/esm/components/IconPickerDialog.d.ts +8 -0
  24. package/dist/esm/components/IconPickerDialog.js +98 -0
  25. package/dist/esm/components/IconPickerDialog.js.map +1 -0
  26. package/dist/esm/components/IconPickerField.d.ts +9 -0
  27. package/dist/esm/components/IconPickerField.js +76 -0
  28. package/dist/esm/components/IconPickerField.js.map +1 -0
  29. package/dist/esm/components/ThemeSwitcher.d.ts +1 -0
  30. package/dist/esm/components/ThemeSwitcher.js +25 -0
  31. package/dist/esm/components/ThemeSwitcher.js.map +1 -0
  32. package/dist/esm/components/theme-provider.d.ts +2 -0
  33. package/dist/esm/components/theme-provider.js +10 -0
  34. package/dist/esm/components/theme-provider.js.map +1 -0
  35. package/dist/esm/errors/AuthorizationError.d.ts +19 -0
  36. package/dist/esm/errors/AuthorizationError.js +20 -0
  37. package/dist/esm/errors/AuthorizationError.js.map +1 -0
  38. package/dist/esm/errors/index.d.ts +1 -0
  39. package/dist/esm/index.d.ts +3 -0
  40. package/dist/esm/index.js +7 -0
  41. package/dist/esm/index.js.map +1 -0
  42. package/dist/esm/lib/utils.d.ts +2 -0
  43. package/dist/esm/lib/utils.js +9 -0
  44. package/dist/esm/lib/utils.js.map +1 -0
  45. package/dist/esm/main.d.ts +0 -0
  46. package/dist/esm/modules/admin-base/components/AdminChat.d.ts +1 -0
  47. package/dist/esm/modules/admin-base/components/AdminChat.js +82 -0
  48. package/dist/esm/modules/admin-base/components/AdminChat.js.map +1 -0
  49. package/dist/esm/modules/admin-base/components/AdminLayout.d.ts +5 -0
  50. package/dist/esm/modules/admin-base/components/AdminLayout.js +83 -0
  51. package/dist/esm/modules/admin-base/components/AdminLayout.js.map +1 -0
  52. package/dist/esm/modules/admin-base/components/AdminWelcome.d.ts +1 -0
  53. package/dist/esm/modules/admin-base/components/AdminWelcome.js +37 -0
  54. package/dist/esm/modules/admin-base/components/AdminWelcome.js.map +1 -0
  55. package/dist/esm/modules/admin-base/context/AdminConfigContext.d.ts +8 -0
  56. package/dist/esm/modules/admin-base/context/AdminConfigContext.js +27 -0
  57. package/dist/esm/modules/admin-base/context/AdminConfigContext.js.map +1 -0
  58. package/dist/esm/modules/admin-base/index.d.ts +5 -0
  59. package/dist/esm/modules/admin-base/types/adminTypes.d.ts +10 -0
  60. package/dist/esm/modules/appCatalog/AppCatalogAdminPage.d.ts +1 -0
  61. package/dist/esm/modules/appCatalog/AppCatalogAdminPage.js +196 -0
  62. package/dist/esm/modules/appCatalog/AppCatalogAdminPage.js.map +1 -0
  63. package/dist/esm/modules/appCatalog/ScreenshotItem.d.ts +7 -0
  64. package/dist/esm/modules/appCatalog/ScreenshotItem.js +57 -0
  65. package/dist/esm/modules/appCatalog/ScreenshotItem.js.map +1 -0
  66. package/dist/esm/modules/appCatalog/ScreenshotManager.d.ts +15 -0
  67. package/dist/esm/modules/appCatalog/ScreenshotManager.js +155 -0
  68. package/dist/esm/modules/appCatalog/ScreenshotManager.js.map +1 -0
  69. package/dist/esm/modules/appCatalog/api/ApiQueryMagazineAppCatalog.d.ts +11 -0
  70. package/dist/esm/modules/appCatalog/api/ApiQueryMagazineAppCatalog.js +14 -0
  71. package/dist/esm/modules/appCatalog/api/ApiQueryMagazineAppCatalog.js.map +1 -0
  72. package/dist/esm/modules/appCatalog/catalogRouteLoader.d.ts +5 -0
  73. package/dist/esm/modules/appCatalog/catalogRouteLoader.js +20 -0
  74. package/dist/esm/modules/appCatalog/catalogRouteLoader.js.map +1 -0
  75. package/dist/esm/modules/appCatalog/context/AppCatalogContext.d.ts +15 -0
  76. package/dist/esm/modules/appCatalog/context/AppCatalogContext.js +47 -0
  77. package/dist/esm/modules/appCatalog/context/AppCatalogContext.js.map +1 -0
  78. package/dist/esm/modules/appCatalog/index.d.ts +10 -0
  79. package/dist/esm/modules/appCatalog/routeLoader.d.ts +3 -0
  80. package/dist/esm/modules/appCatalog/routeLoader.js +7 -0
  81. package/dist/esm/modules/appCatalog/routeLoader.js.map +1 -0
  82. package/dist/esm/modules/appCatalog/ui/components/AccessRequestSection.d.ts +7 -0
  83. package/dist/esm/modules/appCatalog/ui/components/AccessRequestSection.js +257 -0
  84. package/dist/esm/modules/appCatalog/ui/components/AccessRequestSection.js.map +1 -0
  85. package/dist/esm/modules/appCatalog/ui/components/AppDetailModal.d.ts +7 -0
  86. package/dist/esm/modules/appCatalog/ui/components/ApproverDisplay.d.ts +30 -0
  87. package/dist/esm/modules/appCatalog/ui/components/GroupingColumn.d.ts +11 -0
  88. package/dist/esm/modules/appCatalog/ui/components/GroupingTabs.d.ts +7 -0
  89. package/dist/esm/modules/appCatalog/ui/components/ScreenshotGallery.d.ts +10 -0
  90. package/dist/esm/modules/appCatalog/ui/components/ScreenshotGallery.js +35 -0
  91. package/dist/esm/modules/appCatalog/ui/components/ScreenshotGallery.js.map +1 -0
  92. package/dist/esm/modules/appCatalog/ui/grid/AppCatalogFiltersCard.d.ts +9 -0
  93. package/dist/esm/modules/appCatalog/ui/grid/AppCatalogGrid.d.ts +8 -0
  94. package/dist/esm/modules/appCatalog/ui/grid/AppCatalogGrid.js +325 -0
  95. package/dist/esm/modules/appCatalog/ui/grid/AppCatalogGrid.js.map +1 -0
  96. package/dist/esm/modules/appCatalog/ui/grid/AppCatalogTable.d.ts +5 -0
  97. package/dist/esm/modules/appCatalog/ui/grid/appCatalogUtils.d.ts +2 -0
  98. package/dist/esm/modules/appCatalog/ui/hooks/useKeyboardNavigation.d.ts +10 -0
  99. package/dist/esm/modules/appCatalog/ui/hooks/useKeyboardNavigation.js +36 -0
  100. package/dist/esm/modules/appCatalog/ui/hooks/useKeyboardNavigation.js.map +1 -0
  101. package/dist/esm/modules/appCatalog/ui/layout/AppCatalogLayout.d.ts +9 -0
  102. package/dist/esm/modules/appCatalog/ui/layout/AppCatalogLayout.js +28 -0
  103. package/dist/esm/modules/appCatalog/ui/layout/AppCatalogLayout.js.map +1 -0
  104. package/dist/esm/modules/appCatalog/ui/pages/AppCatalogPage.d.ts +1 -0
  105. package/dist/esm/modules/appCatalog/ui/pages/AppCatalogPage.js +112 -0
  106. package/dist/esm/modules/appCatalog/ui/pages/AppCatalogPage.js.map +1 -0
  107. package/dist/esm/modules/approvalMethod/AccessRequestFormFields.d.ts +7 -0
  108. package/dist/esm/modules/approvalMethod/AccessRequestFormFields.js +323 -0
  109. package/dist/esm/modules/approvalMethod/AccessRequestFormFields.js.map +1 -0
  110. package/dist/esm/modules/approvalMethod/ApprovalMethodForm.d.ts +14 -0
  111. package/dist/esm/modules/approvalMethod/ApprovalMethodForm.js +227 -0
  112. package/dist/esm/modules/approvalMethod/ApprovalMethodForm.js.map +1 -0
  113. package/dist/esm/modules/approvalMethod/ApprovalMethodSelector.d.ts +7 -0
  114. package/dist/esm/modules/approvalMethod/ApprovalMethodSelector.js +124 -0
  115. package/dist/esm/modules/approvalMethod/ApprovalMethodSelector.js.map +1 -0
  116. package/dist/esm/modules/approvalMethod/api/ApiQueryMagazineApprovalMethod.d.ts +381 -0
  117. package/dist/esm/modules/approvalMethod/api/ApiQueryMagazineApprovalMethod.js +26 -0
  118. package/dist/esm/modules/approvalMethod/api/ApiQueryMagazineApprovalMethod.js.map +1 -0
  119. package/dist/esm/modules/auth/AuthContext.d.ts +38 -0
  120. package/dist/esm/modules/auth/AuthContext.js +88 -0
  121. package/dist/esm/modules/auth/AuthContext.js.map +1 -0
  122. package/dist/esm/modules/auth/AuthModalContext.d.ts +19 -0
  123. package/dist/esm/modules/auth/AuthModalContext.js +33 -0
  124. package/dist/esm/modules/auth/AuthModalContext.js.map +1 -0
  125. package/dist/esm/modules/auth/ProtectedContent.d.ts +19 -0
  126. package/dist/esm/modules/auth/authClient.d.ts +704 -0
  127. package/dist/esm/modules/auth/authClient.js +8 -0
  128. package/dist/esm/modules/auth/authClient.js.map +1 -0
  129. package/dist/esm/modules/auth/authUtils.d.ts +6 -0
  130. package/dist/esm/modules/auth/authUtils.js +25 -0
  131. package/dist/esm/modules/auth/authUtils.js.map +1 -0
  132. package/dist/esm/modules/auth/index.d.ts +5 -0
  133. package/dist/esm/modules/auth/ui/LoginModal.d.ts +6 -0
  134. package/dist/esm/modules/auth/ui/LoginModal.js +22 -0
  135. package/dist/esm/modules/auth/ui/LoginModal.js.map +1 -0
  136. package/dist/esm/modules/auth/ui/LoginPage.d.ts +5 -0
  137. package/dist/esm/modules/auth/ui/LoginPage.js +50 -0
  138. package/dist/esm/modules/auth/ui/LoginPage.js.map +1 -0
  139. package/dist/esm/modules/auth/useAuthActions.d.ts +17 -0
  140. package/dist/esm/modules/auth/useAuthActions.js +62 -0
  141. package/dist/esm/modules/auth/useAuthActions.js.map +1 -0
  142. package/dist/esm/modules/config/GlobalConfigContext.d.ts +14 -0
  143. package/dist/esm/modules/config/GlobalConfigContext.js +21 -0
  144. package/dist/esm/modules/config/GlobalConfigContext.js.map +1 -0
  145. package/dist/esm/modules/config/HealthStateContext.d.ts +16 -0
  146. package/dist/esm/modules/gallery/Gallery.d.ts +12 -0
  147. package/dist/esm/modules/gallery/Gallery.js +246 -0
  148. package/dist/esm/modules/gallery/Gallery.js.map +1 -0
  149. package/dist/esm/modules/icons/IconManagementPage.d.ts +1 -0
  150. package/dist/esm/modules/icons/IconManagementPage.js +177 -0
  151. package/dist/esm/modules/icons/IconManagementPage.js.map +1 -0
  152. package/dist/esm/modules/pluginCore/PluginManagerContext.d.ts +19 -0
  153. package/dist/esm/modules/pluginCore/PluginManagerContext.js +29 -0
  154. package/dist/esm/modules/pluginCore/PluginManagerContext.js.map +1 -0
  155. package/dist/esm/modules/pluginCore/makePluginManagerContext.d.ts +4 -0
  156. package/dist/esm/modules/pluginCore/makePluginManagerContext.js +9 -0
  157. package/dist/esm/modules/pluginCore/makePluginManagerContext.js.map +1 -0
  158. package/dist/esm/modules/pluginCore/types.d.ts +24 -0
  159. package/dist/esm/node_modules/.pnpm/@dnd-kit_accessibility@3.1.1_react@19.1.2/node_modules/@dnd-kit/accessibility/dist/accessibility.esm.js +60 -0
  160. package/dist/esm/node_modules/.pnpm/@dnd-kit_accessibility@3.1.1_react@19.1.2/node_modules/@dnd-kit/accessibility/dist/accessibility.esm.js.map +1 -0
  161. package/dist/esm/node_modules/.pnpm/@dnd-kit_core@6.3.1_react-dom@19.1.2_react@19.1.2__react@19.1.2/node_modules/@dnd-kit/core/dist/core.esm.js +3055 -0
  162. package/dist/esm/node_modules/.pnpm/@dnd-kit_core@6.3.1_react-dom@19.1.2_react@19.1.2__react@19.1.2/node_modules/@dnd-kit/core/dist/core.esm.js.map +1 -0
  163. package/dist/esm/node_modules/.pnpm/@dnd-kit_sortable@10.0.0_@dnd-kit_core@6.3.1_react-dom@19.1.2_react@19.1.2__react@19.1.2__react@19.1.2/node_modules/@dnd-kit/sortable/dist/sortable.esm.js +593 -0
  164. package/dist/esm/node_modules/.pnpm/@dnd-kit_sortable@10.0.0_@dnd-kit_core@6.3.1_react-dom@19.1.2_react@19.1.2__react@19.1.2__react@19.1.2/node_modules/@dnd-kit/sortable/dist/sortable.esm.js.map +1 -0
  165. package/dist/esm/node_modules/.pnpm/@dnd-kit_utilities@3.2.2_react@19.1.2/node_modules/@dnd-kit/utilities/dist/utilities.esm.js +302 -0
  166. package/dist/esm/node_modules/.pnpm/@dnd-kit_utilities@3.2.2_react@19.1.2/node_modules/@dnd-kit/utilities/dist/utilities.esm.js.map +1 -0
  167. package/dist/esm/node_modules/.pnpm/@hookform_resolvers@5.2.2_react-hook-form@7.71.1_react@19.1.2_/node_modules/@hookform/resolvers/dist/resolvers.js +34 -0
  168. package/dist/esm/node_modules/.pnpm/@hookform_resolvers@5.2.2_react-hook-form@7.71.1_react@19.1.2_/node_modules/@hookform/resolvers/dist/resolvers.js.map +1 -0
  169. package/dist/esm/node_modules/.pnpm/@hookform_resolvers@5.2.2_react-hook-form@7.71.1_react@19.1.2_/node_modules/@hookform/resolvers/zod/dist/zod.js +94 -0
  170. package/dist/esm/node_modules/.pnpm/@hookform_resolvers@5.2.2_react-hook-form@7.71.1_react@19.1.2_/node_modules/@hookform/resolvers/zod/dist/zod.js.map +1 -0
  171. package/dist/esm/node_modules/.pnpm/@tanstack_react-router-devtools@1.151.6_@tanstack_react-router@1.151.6_react-dom@19.1.2_20b78d099e847743bc64b4d051c71d30/node_modules/@tanstack/react-router-devtools/dist/esm/TanStackRouterDevtools.js +72 -0
  172. package/dist/esm/node_modules/.pnpm/@tanstack_react-router-devtools@1.151.6_@tanstack_react-router@1.151.6_react-dom@19.1.2_20b78d099e847743bc64b4d051c71d30/node_modules/@tanstack/react-router-devtools/dist/esm/TanStackRouterDevtools.js.map +1 -0
  173. package/dist/esm/node_modules/.pnpm/@tanstack_react-router-devtools@1.151.6_@tanstack_react-router@1.151.6_react-dom@19.1.2_20b78d099e847743bc64b4d051c71d30/node_modules/@tanstack/react-router-devtools/dist/esm/TanStackRouterDevtoolsPanel.js +44 -0
  174. package/dist/esm/node_modules/.pnpm/@tanstack_react-router-devtools@1.151.6_@tanstack_react-router@1.151.6_react-dom@19.1.2_20b78d099e847743bc64b4d051c71d30/node_modules/@tanstack/react-router-devtools/dist/esm/TanStackRouterDevtoolsPanel.js.map +1 -0
  175. package/dist/esm/node_modules/.pnpm/@tanstack_react-router-devtools@1.151.6_@tanstack_react-router@1.151.6_react-dom@19.1.2_20b78d099e847743bc64b4d051c71d30/node_modules/@tanstack/react-router-devtools/dist/esm/index.js +12 -0
  176. package/dist/esm/node_modules/.pnpm/@tanstack_react-router-devtools@1.151.6_@tanstack_react-router@1.151.6_react-dom@19.1.2_20b78d099e847743bc64b4d051c71d30/node_modules/@tanstack/react-router-devtools/dist/esm/index.js.map +1 -0
  177. package/dist/esm/node_modules/.pnpm/@tanstack_router-core@1.151.6/node_modules/@tanstack/router-core/dist/esm/new-process-route-tree.js +95 -0
  178. package/dist/esm/node_modules/.pnpm/@tanstack_router-core@1.151.6/node_modules/@tanstack/router-core/dist/esm/new-process-route-tree.js.map +1 -0
  179. package/dist/esm/node_modules/.pnpm/@tanstack_router-core@1.151.6/node_modules/@tanstack/router-core/dist/esm/path.js +107 -0
  180. package/dist/esm/node_modules/.pnpm/@tanstack_router-core@1.151.6/node_modules/@tanstack/router-core/dist/esm/path.js.map +1 -0
  181. package/dist/esm/node_modules/.pnpm/@tanstack_router-core@1.151.6/node_modules/@tanstack/router-core/dist/esm/root.js +5 -0
  182. package/dist/esm/node_modules/.pnpm/@tanstack_router-core@1.151.6/node_modules/@tanstack/router-core/dist/esm/root.js.map +1 -0
  183. package/dist/esm/node_modules/.pnpm/@tanstack_router-devtools-core@1.151.6_@tanstack_router-core@1.151.6_csstype@3.2.3/node_modules/@tanstack/router-devtools-core/dist/BaseTanStackRouterDevtoolsPanel-DxaCaP75.js +1963 -0
  184. package/dist/esm/node_modules/.pnpm/@tanstack_router-devtools-core@1.151.6_@tanstack_router-core@1.151.6_csstype@3.2.3/node_modules/@tanstack/router-devtools-core/dist/BaseTanStackRouterDevtoolsPanel-DxaCaP75.js.map +1 -0
  185. package/dist/esm/node_modules/.pnpm/@tanstack_router-devtools-core@1.151.6_@tanstack_router-core@1.151.6_csstype@3.2.3/node_modules/@tanstack/router-devtools-core/dist/FloatingTanStackRouterDevtools-CaypUmOS.js +284 -0
  186. package/dist/esm/node_modules/.pnpm/@tanstack_router-devtools-core@1.151.6_@tanstack_router-core@1.151.6_csstype@3.2.3/node_modules/@tanstack/router-devtools-core/dist/FloatingTanStackRouterDevtools-CaypUmOS.js.map +1 -0
  187. package/dist/esm/node_modules/.pnpm/@tanstack_router-devtools-core@1.151.6_@tanstack_router-core@1.151.6_csstype@3.2.3/node_modules/@tanstack/router-devtools-core/dist/index-5jfj03XJ.js +1855 -0
  188. package/dist/esm/node_modules/.pnpm/@tanstack_router-devtools-core@1.151.6_@tanstack_router-core@1.151.6_csstype@3.2.3/node_modules/@tanstack/router-devtools-core/dist/index-5jfj03XJ.js.map +1 -0
  189. package/dist/esm/node_modules/.pnpm/goober@2.1.18_csstype@3.2.3/node_modules/goober/dist/goober.modern.js +56 -0
  190. package/dist/esm/node_modules/.pnpm/goober@2.1.18_csstype@3.2.3/node_modules/goober/dist/goober.modern.js.map +1 -0
  191. package/dist/esm/node_modules/.pnpm/react-hook-form@7.71.1_react@19.1.2/node_modules/react-hook-form/dist/index.esm.js +1894 -0
  192. package/dist/esm/node_modules/.pnpm/react-hook-form@7.71.1_react@19.1.2/node_modules/react-hook-form/dist/index.esm.js.map +1 -0
  193. package/dist/esm/node_modules/.pnpm/tiny-invariant@1.3.3/node_modules/tiny-invariant/dist/esm/tiny-invariant.js +17 -0
  194. package/dist/esm/node_modules/.pnpm/tiny-invariant@1.3.3/node_modules/tiny-invariant/dist/esm/tiny-invariant.js.map +1 -0
  195. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/classic/errors.js +45 -0
  196. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/classic/errors.js.map +1 -0
  197. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/classic/iso.js +44 -0
  198. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/classic/iso.js.map +1 -0
  199. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/classic/parse.js +30 -0
  200. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/classic/parse.js.map +1 -0
  201. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/classic/schemas.js +632 -0
  202. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/classic/schemas.js.map +1 -0
  203. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/api.js +500 -0
  204. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/api.js.map +1 -0
  205. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/checks.js +244 -0
  206. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/checks.js.map +1 -0
  207. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/core.js +75 -0
  208. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/core.js.map +1 -0
  209. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/doc.js +38 -0
  210. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/doc.js.map +1 -0
  211. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/errors.js +73 -0
  212. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/errors.js.map +1 -0
  213. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/json-schema-processors.js +274 -0
  214. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/json-schema-processors.js.map +1 -0
  215. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/parse.js +100 -0
  216. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/parse.js.map +1 -0
  217. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/regexes.js +80 -0
  218. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/regexes.js.map +1 -0
  219. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/registries.js +52 -0
  220. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/registries.js.map +1 -0
  221. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/schemas.js +1247 -0
  222. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/schemas.js.map +1 -0
  223. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/to-json-schema.js +363 -0
  224. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/to-json-schema.js.map +1 -0
  225. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/util.js +407 -0
  226. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/util.js.map +1 -0
  227. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/versions.js +9 -0
  228. package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/versions.js.map +1 -0
  229. package/dist/esm/plugins/builtin/pageUrl/pageUrlTypes.d.ts +11 -0
  230. package/dist/esm/routeTree.gen.d.ts +263 -0
  231. package/dist/esm/routeTree.gen.js +117 -0
  232. package/dist/esm/routeTree.gen.js.map +1 -0
  233. package/dist/esm/routes/__root.d.ts +2 -0
  234. package/dist/esm/routes/__root.js +22 -0
  235. package/dist/esm/routes/__root.js.map +1 -0
  236. package/dist/esm/routes/_layout/catalog.apps.index.d.ts +5 -0
  237. package/dist/esm/routes/_layout/catalog.apps.index.js +25 -0
  238. package/dist/esm/routes/_layout/catalog.apps.index.js.map +1 -0
  239. package/dist/esm/routes/_layout/index.d.ts +7 -0
  240. package/dist/esm/routes/_layout/index.js +28 -0
  241. package/dist/esm/routes/_layout/index.js.map +1 -0
  242. package/dist/esm/routes/_layout/login.d.ts +1 -0
  243. package/dist/esm/routes/_layout/login.js +16 -0
  244. package/dist/esm/routes/_layout/login.js.map +1 -0
  245. package/dist/esm/routes/_layout.d.ts +1 -0
  246. package/dist/esm/routes/_layout.js +12 -0
  247. package/dist/esm/routes/_layout.js.map +1 -0
  248. package/dist/esm/routes/admin/app-for-catalog/$id.d.ts +5 -0
  249. package/dist/esm/routes/admin/app-for-catalog/_id.js +67 -0
  250. package/dist/esm/routes/admin/app-for-catalog/_id.js.map +1 -0
  251. package/dist/esm/routes/admin/app-for-catalog/_id2.js +321 -0
  252. package/dist/esm/routes/admin/app-for-catalog/_id2.js.map +1 -0
  253. package/dist/esm/routes/admin/app-for-catalog/index.d.ts +1 -0
  254. package/dist/esm/routes/admin/app-for-catalog/index.js +9 -0
  255. package/dist/esm/routes/admin/app-for-catalog/index.js.map +1 -0
  256. package/dist/esm/routes/admin/app-for-catalog/index2.js +12 -0
  257. package/dist/esm/routes/admin/app-for-catalog/index2.js.map +1 -0
  258. package/dist/esm/routes/admin/app-for-catalog.d.ts +1 -0
  259. package/dist/esm/routes/admin/app-for-catalog.js +14 -0
  260. package/dist/esm/routes/admin/app-for-catalog.js.map +1 -0
  261. package/dist/esm/routes/admin/app-for-catalog2.js +9 -0
  262. package/dist/esm/routes/admin/app-for-catalog2.js.map +1 -0
  263. package/dist/esm/routes/admin/approval-methods/index.d.ts +32 -0
  264. package/dist/esm/routes/admin/approval-methods/index.js +24 -0
  265. package/dist/esm/routes/admin/approval-methods/index.js.map +1 -0
  266. package/dist/esm/routes/admin/approval-methods/index2.js +100 -0
  267. package/dist/esm/routes/admin/approval-methods/index2.js.map +1 -0
  268. package/dist/esm/routes/admin/approval-methods.d.ts +1 -0
  269. package/dist/esm/routes/admin/approval-methods.js +14 -0
  270. package/dist/esm/routes/admin/approval-methods.js.map +1 -0
  271. package/dist/esm/routes/admin/approval-methods2.js +7 -0
  272. package/dist/esm/routes/admin/approval-methods2.js.map +1 -0
  273. package/dist/esm/routes/admin/chat.d.ts +1 -0
  274. package/dist/esm/routes/admin/chat.js +14 -0
  275. package/dist/esm/routes/admin/chat.js.map +1 -0
  276. package/dist/esm/routes/admin/chat2.js +9 -0
  277. package/dist/esm/routes/admin/chat2.js.map +1 -0
  278. package/dist/esm/routes/admin/icons.d.ts +1 -0
  279. package/dist/esm/routes/admin/icons.js +14 -0
  280. package/dist/esm/routes/admin/icons.js.map +1 -0
  281. package/dist/esm/routes/admin/icons2.js +12 -0
  282. package/dist/esm/routes/admin/icons2.js.map +1 -0
  283. package/dist/esm/routes/admin/index.d.ts +1 -0
  284. package/dist/esm/routes/admin/index.js +9 -0
  285. package/dist/esm/routes/admin/index.js.map +1 -0
  286. package/dist/esm/routes/admin/index2.js +9 -0
  287. package/dist/esm/routes/admin/index2.js.map +1 -0
  288. package/dist/esm/routes/admin.d.ts +1 -0
  289. package/dist/esm/routes/admin.js +37 -0
  290. package/dist/esm/routes/admin.js.map +1 -0
  291. package/dist/esm/routes/admin2.js +18 -0
  292. package/dist/esm/routes/admin2.js.map +1 -0
  293. package/dist/esm/routes/auth.callback.d.ts +4 -0
  294. package/dist/esm/routes/auth.callback.js +56 -0
  295. package/dist/esm/routes/auth.callback.js.map +1 -0
  296. package/dist/esm/types/slateTypes.d.ts +28 -0
  297. package/dist/esm/types/table.d.ts +7 -0
  298. package/dist/esm/types/tanstackQuery.d.ts +13 -0
  299. package/dist/esm/types/types.d.ts +25 -0
  300. package/dist/esm/types/userBehaviourTypes.d.ts +76 -0
  301. package/dist/esm/types/utilityTypes.d.ts +1 -0
  302. package/dist/esm/ui/accordion.d.ts +7 -0
  303. package/dist/esm/ui/accordion.js +65 -0
  304. package/dist/esm/ui/accordion.js.map +1 -0
  305. package/dist/esm/ui/alert-dialog.d.ts +14 -0
  306. package/dist/esm/ui/alert-dialog.js +141 -0
  307. package/dist/esm/ui/alert-dialog.js.map +1 -0
  308. package/dist/esm/ui/autocomplete.d.ts +29 -0
  309. package/dist/esm/ui/badge.d.ts +9 -0
  310. package/dist/esm/ui/badge.js +27 -0
  311. package/dist/esm/ui/badge.js.map +1 -0
  312. package/dist/esm/ui/breadcrumb.d.ts +11 -0
  313. package/dist/esm/ui/breadcrumb.js +84 -0
  314. package/dist/esm/ui/breadcrumb.js.map +1 -0
  315. package/dist/esm/ui/button-group.d.ts +11 -0
  316. package/dist/esm/ui/button.d.ts +13 -0
  317. package/dist/esm/ui/button.js +53 -0
  318. package/dist/esm/ui/button.js.map +1 -0
  319. package/dist/esm/ui/card.d.ts +9 -0
  320. package/dist/esm/ui/card.js +66 -0
  321. package/dist/esm/ui/card.js.map +1 -0
  322. package/dist/esm/ui/checkbox.d.ts +4 -0
  323. package/dist/esm/ui/collapsible.d.ts +5 -0
  324. package/dist/esm/ui/command.d.ts +20 -0
  325. package/dist/esm/ui/command.js +131 -0
  326. package/dist/esm/ui/command.js.map +1 -0
  327. package/dist/esm/ui/components/ActionCard.d.ts +9 -0
  328. package/dist/esm/ui/components/AppIcon.d.ts +7 -0
  329. package/dist/esm/ui/components/Breadcrumbs.d.ts +19 -0
  330. package/dist/esm/ui/components/Breadcrumbs.js +36 -0
  331. package/dist/esm/ui/components/Breadcrumbs.js.map +1 -0
  332. package/dist/esm/ui/components/commandInput/EhBaseSelector.d.ts +2 -0
  333. package/dist/esm/ui/components/commandInput/types.d.ts +13 -0
  334. package/dist/esm/ui/components/error/BaseErrorPage.d.ts +10 -0
  335. package/dist/esm/ui/components/error/BaseErrorPage.js +16 -0
  336. package/dist/esm/ui/components/error/BaseErrorPage.js.map +1 -0
  337. package/dist/esm/ui/components/error/DefaultErrorComponent.d.ts +3 -0
  338. package/dist/esm/ui/components/error/DefaultErrorComponent.js +73 -0
  339. package/dist/esm/ui/components/error/DefaultErrorComponent.js.map +1 -0
  340. package/dist/esm/ui/components/error/ForbiddenErrorPage.d.ts +5 -0
  341. package/dist/esm/ui/components/error/ForbiddenErrorPage.js +54 -0
  342. package/dist/esm/ui/components/error/ForbiddenErrorPage.js.map +1 -0
  343. package/dist/esm/ui/components/error/RootErrorPage.d.ts +5 -0
  344. package/dist/esm/ui/components/error/RootErrorPage.js +15 -0
  345. package/dist/esm/ui/components/error/RootErrorPage.js.map +1 -0
  346. package/dist/esm/ui/components/footer/Footer.d.ts +1 -0
  347. package/dist/esm/ui/components/header/Header.d.ts +5 -0
  348. package/dist/esm/ui/components/header/Header.js +103 -0
  349. package/dist/esm/ui/components/header/Header.js.map +1 -0
  350. package/dist/esm/ui/components/widgetPanel/AddWidgetCard.d.ts +5 -0
  351. package/dist/esm/ui/components/widgetPanel/WidgetGrid.d.ts +6 -0
  352. package/dist/esm/ui/components/widgets/CredentialsWidget.d.ts +11 -0
  353. package/dist/esm/ui/components/widgets/VersionWidget.d.ts +7 -0
  354. package/dist/esm/ui/crud-list/CrudList.d.ts +2 -0
  355. package/dist/esm/ui/crud-list/CrudList.js +189 -0
  356. package/dist/esm/ui/crud-list/CrudList.js.map +1 -0
  357. package/dist/esm/ui/crud-list/index.d.ts +2 -0
  358. package/dist/esm/ui/crud-list/types.d.ts +35 -0
  359. package/dist/esm/ui/dialog.d.ts +15 -0
  360. package/dist/esm/ui/dialog.js +111 -0
  361. package/dist/esm/ui/dialog.js.map +1 -0
  362. package/dist/esm/ui/dropdown-menu.d.ts +25 -0
  363. package/dist/esm/ui/dropdown-menu.js +79 -0
  364. package/dist/esm/ui/dropdown-menu.js.map +1 -0
  365. package/dist/esm/ui/editable-list/EditableListField.d.ts +2 -0
  366. package/dist/esm/ui/editable-list/EditableListField.js +130 -0
  367. package/dist/esm/ui/editable-list/EditableListField.js.map +1 -0
  368. package/dist/esm/ui/editable-list/index.d.ts +2 -0
  369. package/dist/esm/ui/editable-list/types.d.ts +17 -0
  370. package/dist/esm/ui/empty.d.ts +11 -0
  371. package/dist/esm/ui/empty.js +103 -0
  372. package/dist/esm/ui/empty.js.map +1 -0
  373. package/dist/esm/ui/error/NotFoundError.d.ts +1 -0
  374. package/dist/esm/ui/error/NotFoundError.js +17 -0
  375. package/dist/esm/ui/error/NotFoundError.js.map +1 -0
  376. package/dist/esm/ui/form.d.ts +56 -0
  377. package/dist/esm/ui/form.js +134 -0
  378. package/dist/esm/ui/form.js.map +1 -0
  379. package/dist/esm/ui/input-group.d.ts +16 -0
  380. package/dist/esm/ui/input.d.ts +3 -0
  381. package/dist/esm/ui/input.js +22 -0
  382. package/dist/esm/ui/input.js.map +1 -0
  383. package/dist/esm/ui/item.d.ts +23 -0
  384. package/dist/esm/ui/label.d.ts +4 -0
  385. package/dist/esm/ui/label.js +23 -0
  386. package/dist/esm/ui/label.js.map +1 -0
  387. package/dist/esm/ui/layout/Footer.d.ts +1 -0
  388. package/dist/esm/ui/layout/LoadingScreen.d.ts +6 -0
  389. package/dist/esm/ui/layout/LoadingScreen.js +54 -0
  390. package/dist/esm/ui/layout/LoadingScreen.js.map +1 -0
  391. package/dist/esm/ui/layout/MainLayout.d.ts +7 -0
  392. package/dist/esm/ui/layout/MainLayout.js +17 -0
  393. package/dist/esm/ui/layout/MainLayout.js.map +1 -0
  394. package/dist/esm/ui/layout/SideColumn.d.ts +1 -0
  395. package/dist/esm/ui/layout/TopLevelProviders.d.ts +10 -0
  396. package/dist/esm/ui/layout/TopLevelProviders.js +46 -0
  397. package/dist/esm/ui/layout/TopLevelProviders.js.map +1 -0
  398. package/dist/esm/ui/layout/TopLevelProvidersForErrors.d.ts +10 -0
  399. package/dist/esm/ui/layout/TopLevelProvidersForErrors.js +24 -0
  400. package/dist/esm/ui/layout/TopLevelProvidersForErrors.js.map +1 -0
  401. package/dist/esm/ui/link.d.ts +13 -0
  402. package/dist/esm/ui/link.js +28 -0
  403. package/dist/esm/ui/link.js.map +1 -0
  404. package/dist/esm/ui/linkExternal.d.ts +8 -0
  405. package/dist/esm/ui/linkExternal.js +26 -0
  406. package/dist/esm/ui/linkExternal.js.map +1 -0
  407. package/dist/esm/ui/main/JumpTabContent.d.ts +1 -0
  408. package/dist/esm/ui/main/Tabs.d.ts +4 -0
  409. package/dist/esm/ui/markdown-editor/MarkdownEditor.d.ts +9 -0
  410. package/dist/esm/ui/markdown-editor/MarkdownEditor.js +116 -0
  411. package/dist/esm/ui/markdown-editor/MarkdownEditor.js.map +1 -0
  412. package/dist/esm/ui/markdown-editor/MarkdownToolbar.d.ts +6 -0
  413. package/dist/esm/ui/markdown-editor/MarkdownToolbar.js +99 -0
  414. package/dist/esm/ui/markdown-editor/MarkdownToolbar.js.map +1 -0
  415. package/dist/esm/ui/markdown-editor/index.d.ts +2 -0
  416. package/dist/esm/ui/popover.d.ts +7 -0
  417. package/dist/esm/ui/popover.js +40 -0
  418. package/dist/esm/ui/popover.js.map +1 -0
  419. package/dist/esm/ui/radio-group.d.ts +5 -0
  420. package/dist/esm/ui/resizable.d.ts +7 -0
  421. package/dist/esm/ui/resizable.js +47 -0
  422. package/dist/esm/ui/resizable.js.map +1 -0
  423. package/dist/esm/ui/scroll-area.d.ts +5 -0
  424. package/dist/esm/ui/scroll-area.js +62 -0
  425. package/dist/esm/ui/scroll-area.js.map +1 -0
  426. package/dist/esm/ui/search-input-with-shortcut.d.ts +25 -0
  427. package/dist/esm/ui/select.d.ts +15 -0
  428. package/dist/esm/ui/select.js +138 -0
  429. package/dist/esm/ui/select.js.map +1 -0
  430. package/dist/esm/ui/separator.d.ts +4 -0
  431. package/dist/esm/ui/shortcut-button.d.ts +24 -0
  432. package/dist/esm/ui/skeleton.d.ts +2 -0
  433. package/dist/esm/ui/skeleton.js +16 -0
  434. package/dist/esm/ui/skeleton.js.map +1 -0
  435. package/dist/esm/ui/spinner.d.ts +2 -0
  436. package/dist/esm/ui/spinner.js +18 -0
  437. package/dist/esm/ui/spinner.js.map +1 -0
  438. package/dist/esm/ui/table.d.ts +10 -0
  439. package/dist/esm/ui/table.js +87 -0
  440. package/dist/esm/ui/table.js.map +1 -0
  441. package/dist/esm/ui/tabs.d.ts +7 -0
  442. package/dist/esm/ui/textarea.d.ts +3 -0
  443. package/dist/esm/ui/textarea.js +19 -0
  444. package/dist/esm/ui/textarea.js.map +1 -0
  445. package/dist/esm/ui/tooltip.d.ts +7 -0
  446. package/dist/esm/userDb/AcDb.d.ts +14 -0
  447. package/dist/esm/userDb/AcDb.js +28 -0
  448. package/dist/esm/userDb/AcDb.js.map +1 -0
  449. package/dist/esm/userDb/DbContext.d.ts +9 -0
  450. package/dist/esm/userDb/DbContext.js +18 -0
  451. package/dist/esm/userDb/DbContext.js.map +1 -0
  452. package/dist/esm/util/createEhRouter.d.ts +7 -0
  453. package/dist/esm/util/createEhRouter.js +19 -0
  454. package/dist/esm/util/createEhRouter.js.map +1 -0
  455. package/dist/esm/util/error-utils.d.ts +6 -0
  456. package/dist/esm/util/error-utils.js +28 -0
  457. package/dist/esm/util/error-utils.js.map +1 -0
  458. package/dist/esm/util/highlightMatches.d.ts +2 -0
  459. package/dist/esm/util/reactQueryUtils.d.ts +10 -0
  460. package/dist/esm/util/reactQueryUtils.js +17 -0
  461. package/dist/esm/util/reactQueryUtils.js.map +1 -0
  462. package/dist/esm/util/slug-utils.d.ts +12 -0
  463. package/dist/index.css +225 -0
  464. package/dist/public/app-catalog-16x16.png +0 -0
  465. package/dist/public/app-catalog-192x192.png +0 -0
  466. package/dist/public/app-catalog-32x32.png +0 -0
  467. package/dist/public/app-catalog-48x48.png +0 -0
  468. package/dist/public/app-catalog-512x512.png +0 -0
  469. package/dist/public/app-catalog-square.svg +160 -0
  470. package/dist/public/app-catalog.png +0 -0
  471. package/dist/public/app-catalog.svg +198 -0
  472. package/dist/public/apple-touch-180x180.png +0 -0
  473. package/dist/public/favicon-app-catalog.ico +0 -0
  474. package/dist/public/favicon.ico +0 -0
  475. package/dist/public/robots.txt +2 -0
  476. package/package.json +159 -0
  477. package/public/app-catalog-16x16.png +0 -0
  478. package/public/app-catalog-192x192.png +0 -0
  479. package/public/app-catalog-32x32.png +0 -0
  480. package/public/app-catalog-48x48.png +0 -0
  481. package/public/app-catalog-512x512.png +0 -0
  482. package/public/app-catalog-square.svg +160 -0
  483. package/public/app-catalog.png +0 -0
  484. package/public/app-catalog.svg +198 -0
  485. package/public/apple-touch-180x180.png +0 -0
  486. package/public/favicon-app-catalog.ico +0 -0
  487. package/public/favicon.ico +0 -0
  488. package/public/robots.txt +2 -0
  489. package/src/App.tsx +28 -0
  490. package/src/__tests__/setupTests.tsx +0 -0
  491. package/src/api/infra/createQueryClient.ts +29 -0
  492. package/src/api/infra/trpc.ts +6 -0
  493. package/src/api/unsorted/appCatalogFetcher.ts +32 -0
  494. package/src/api/unsorted/createCachingFetcher.ts +159 -0
  495. package/src/appPropsFactory.ts +37 -0
  496. package/src/assets/app-catalog.svg +198 -0
  497. package/src/components/IconPickerDialog.tsx +136 -0
  498. package/src/components/IconPickerField.tsx +88 -0
  499. package/src/components/ThemeSwitcher.tsx +22 -0
  500. package/src/components/theme-provider.tsx +8 -0
  501. package/src/errors/AuthorizationError.ts +32 -0
  502. package/src/errors/index.ts +1 -0
  503. package/src/index.css +225 -0
  504. package/src/index.tsx +3 -0
  505. package/src/lib/utils.ts +7 -0
  506. package/src/main.tsx +57 -0
  507. package/src/modules/admin-base/components/AdminChat.tsx +122 -0
  508. package/src/modules/admin-base/components/AdminLayout.tsx +111 -0
  509. package/src/modules/admin-base/components/AdminWelcome.tsx +52 -0
  510. package/src/modules/admin-base/context/AdminConfigContext.tsx +36 -0
  511. package/src/modules/admin-base/index.ts +16 -0
  512. package/src/modules/admin-base/types/adminTypes.ts +11 -0
  513. package/src/modules/appCatalog/AppCatalogAdminPage.tsx +274 -0
  514. package/src/modules/appCatalog/ScreenshotItem.tsx +59 -0
  515. package/src/modules/appCatalog/ScreenshotManager.tsx +193 -0
  516. package/src/modules/appCatalog/api/ApiQueryMagazineAppCatalog.ts +12 -0
  517. package/src/modules/appCatalog/catalogRouteLoader.ts +28 -0
  518. package/src/modules/appCatalog/context/AppCatalogContext.tsx +52 -0
  519. package/src/modules/appCatalog/index.ts +16 -0
  520. package/src/modules/appCatalog/routeLoader.ts +9 -0
  521. package/src/modules/appCatalog/ui/components/AccessRequestSection.tsx +370 -0
  522. package/src/modules/appCatalog/ui/components/AppDetailModal.tsx +355 -0
  523. package/src/modules/appCatalog/ui/components/ApproverDisplay.tsx +260 -0
  524. package/src/modules/appCatalog/ui/components/GroupingColumn.tsx +65 -0
  525. package/src/modules/appCatalog/ui/components/GroupingTabs.tsx +41 -0
  526. package/src/modules/appCatalog/ui/components/ScreenshotGallery.tsx +51 -0
  527. package/src/modules/appCatalog/ui/grid/AppCatalogFiltersCard.tsx +79 -0
  528. package/src/modules/appCatalog/ui/grid/AppCatalogGrid.tsx +487 -0
  529. package/src/modules/appCatalog/ui/grid/AppCatalogTable.tsx +89 -0
  530. package/src/modules/appCatalog/ui/grid/appCatalogUtils.ts +5 -0
  531. package/src/modules/appCatalog/ui/hooks/useKeyboardNavigation.ts +54 -0
  532. package/src/modules/appCatalog/ui/layout/AppCatalogLayout.tsx +57 -0
  533. package/src/modules/appCatalog/ui/pages/AppCatalogPage.tsx +142 -0
  534. package/src/modules/approvalMethod/AccessRequestFormFields.tsx +393 -0
  535. package/src/modules/approvalMethod/ApprovalMethodForm.tsx +323 -0
  536. package/src/modules/approvalMethod/ApprovalMethodSelector.tsx +150 -0
  537. package/src/modules/approvalMethod/api/ApiQueryMagazineApprovalMethod.ts +34 -0
  538. package/src/modules/auth/AuthContext.tsx +130 -0
  539. package/src/modules/auth/AuthModalContext.tsx +49 -0
  540. package/src/modules/auth/ProtectedContent.tsx +55 -0
  541. package/src/modules/auth/authClient.ts +9 -0
  542. package/src/modules/auth/authUtils.ts +41 -0
  543. package/src/modules/auth/index.ts +6 -0
  544. package/src/modules/auth/ui/LoginModal.tsx +36 -0
  545. package/src/modules/auth/ui/LoginPage.tsx +62 -0
  546. package/src/modules/auth/useAuthActions.ts +85 -0
  547. package/src/modules/config/GlobalConfigContext.tsx +42 -0
  548. package/src/modules/config/HealthStateContext.tsx +47 -0
  549. package/src/modules/gallery/Gallery.tsx +317 -0
  550. package/src/modules/icons/IconManagementPage.tsx +245 -0
  551. package/src/modules/pluginCore/PluginManagerContext.tsx +78 -0
  552. package/src/modules/pluginCore/makePluginManagerContext.ts +13 -0
  553. package/src/modules/pluginCore/types.ts +27 -0
  554. package/src/plugins/builtin/pageUrl/pageUrlTypes.ts +16 -0
  555. package/src/routeTree.gen.ts +361 -0
  556. package/src/routes/__root.tsx +23 -0
  557. package/src/routes/_layout/catalog.apps.index.tsx +28 -0
  558. package/src/routes/_layout/index.tsx +31 -0
  559. package/src/routes/_layout/login.tsx +20 -0
  560. package/src/routes/_layout.tsx +9 -0
  561. package/src/routes/admin/app-for-catalog/$id.tsx +571 -0
  562. package/src/routes/admin/app-for-catalog/index.tsx +19 -0
  563. package/src/routes/admin/app-for-catalog.tsx +12 -0
  564. package/src/routes/admin/approval-methods/index.tsx +161 -0
  565. package/src/routes/admin/approval-methods.tsx +10 -0
  566. package/src/routes/admin/chat.tsx +13 -0
  567. package/src/routes/admin/icons.tsx +22 -0
  568. package/src/routes/admin/index.tsx +9 -0
  569. package/src/routes/admin.tsx +60 -0
  570. package/src/routes/auth.callback.tsx +74 -0
  571. package/src/types/slateTypes.ts +22 -0
  572. package/src/types/table.ts +9 -0
  573. package/src/types/tanstackQuery.ts +16 -0
  574. package/src/types/types.ts +30 -0
  575. package/src/types/userBehaviourTypes.ts +100 -0
  576. package/src/types/utilityTypes.ts +1 -0
  577. package/src/types/vite-env.d.ts +2 -0
  578. package/src/ui/accordion.tsx +64 -0
  579. package/src/ui/alert-dialog.tsx +155 -0
  580. package/src/ui/autocomplete.tsx +272 -0
  581. package/src/ui/badge.tsx +38 -0
  582. package/src/ui/breadcrumb.tsx +106 -0
  583. package/src/ui/button-group.tsx +85 -0
  584. package/src/ui/button.tsx +66 -0
  585. package/src/ui/card.tsx +92 -0
  586. package/src/ui/checkbox.tsx +30 -0
  587. package/src/ui/collapsible.tsx +45 -0
  588. package/src/ui/command.tsx +196 -0
  589. package/src/ui/components/ActionCard.tsx +30 -0
  590. package/src/ui/components/AppIcon.tsx +48 -0
  591. package/src/ui/components/Breadcrumbs.tsx +97 -0
  592. package/src/ui/components/commandInput/EhBaseSelector.tsx +17 -0
  593. package/src/ui/components/commandInput/types.ts +22 -0
  594. package/src/ui/components/error/BaseErrorPage.tsx +26 -0
  595. package/src/ui/components/error/DefaultErrorComponent.tsx +107 -0
  596. package/src/ui/components/error/ForbiddenErrorPage.tsx +89 -0
  597. package/src/ui/components/error/RootErrorPage.tsx +23 -0
  598. package/src/ui/components/footer/Footer.tsx +51 -0
  599. package/src/ui/components/header/Header.tsx +135 -0
  600. package/src/ui/components/widgetPanel/AddWidgetCard.tsx +17 -0
  601. package/src/ui/components/widgetPanel/WidgetGrid.tsx +18 -0
  602. package/src/ui/components/widgets/CredentialsWidget.tsx +55 -0
  603. package/src/ui/components/widgets/VersionWidget.tsx +29 -0
  604. package/src/ui/crud-list/CrudList.tsx +274 -0
  605. package/src/ui/crud-list/index.ts +2 -0
  606. package/src/ui/crud-list/types.ts +44 -0
  607. package/src/ui/dialog.tsx +141 -0
  608. package/src/ui/dropdown-menu.tsx +255 -0
  609. package/src/ui/editable-list/EditableListField.tsx +188 -0
  610. package/src/ui/editable-list/index.ts +2 -0
  611. package/src/ui/editable-list/types.ts +18 -0
  612. package/src/ui/empty.tsx +105 -0
  613. package/src/ui/error/NotFoundError.tsx +16 -0
  614. package/src/ui/form.tsx +188 -0
  615. package/src/ui/input-group.tsx +167 -0
  616. package/src/ui/input.tsx +21 -0
  617. package/src/ui/item.tsx +194 -0
  618. package/src/ui/label.tsx +22 -0
  619. package/src/ui/layout/Footer.tsx +16 -0
  620. package/src/ui/layout/LoadingScreen.tsx +67 -0
  621. package/src/ui/layout/MainLayout.tsx +28 -0
  622. package/src/ui/layout/SideColumn.tsx +3 -0
  623. package/src/ui/layout/TopLevelProviders.tsx +63 -0
  624. package/src/ui/layout/TopLevelProvidersForErrors.tsx +34 -0
  625. package/src/ui/link.tsx +33 -0
  626. package/src/ui/linkExternal.tsx +26 -0
  627. package/src/ui/main/JumpTabContent.tsx +12 -0
  628. package/src/ui/main/Tabs.tsx +29 -0
  629. package/src/ui/markdown-editor/MarkdownEditor.tsx +132 -0
  630. package/src/ui/markdown-editor/MarkdownToolbar.tsx +90 -0
  631. package/src/ui/markdown-editor/index.ts +2 -0
  632. package/src/ui/popover.tsx +48 -0
  633. package/src/ui/radio-group.tsx +43 -0
  634. package/src/ui/resizable.tsx +51 -0
  635. package/src/ui/scroll-area.tsx +58 -0
  636. package/src/ui/search-input-with-shortcut.tsx +54 -0
  637. package/src/ui/select.tsx +185 -0
  638. package/src/ui/separator.tsx +26 -0
  639. package/src/ui/shortcut-button.tsx +84 -0
  640. package/src/ui/skeleton.tsx +13 -0
  641. package/src/ui/spinner.tsx +16 -0
  642. package/src/ui/table.tsx +114 -0
  643. package/src/ui/tabs.tsx +64 -0
  644. package/src/ui/textarea.tsx +18 -0
  645. package/src/ui/tooltip.tsx +60 -0
  646. package/src/userDb/AcDb.ts +29 -0
  647. package/src/userDb/DbContext.tsx +22 -0
  648. package/src/util/createEhRouter.tsx +22 -0
  649. package/src/util/error-utils.ts +31 -0
  650. package/src/util/highlightMatches.tsx +29 -0
  651. package/src/util/reactQueryUtils.ts +20 -0
  652. package/src/util/slug-utils.ts +17 -0
@@ -0,0 +1,325 @@
1
+ import { jsxs, jsx, Fragment } from "react/jsx-runtime";
2
+ import { useReactTable, getCoreRowModel, flexRender } from "@tanstack/react-table";
3
+ import { AppWindow, ExternalLink } from "lucide-react";
4
+ import React__default from "react";
5
+ import { cn } from "../../../../lib/utils.js";
6
+ import { Badge } from "../../../../ui/badge.js";
7
+ import { ResizablePanelGroup, ResizablePanel, ResizableHandle } from "../../../../ui/resizable.js";
8
+ import { Table, TableHeader, TableRow, TableHead, TableBody, TableCell } from "../../../../ui/table.js";
9
+ import { AccessRequestSection } from "../components/AccessRequestSection.js";
10
+ import { ScreenshotGallery } from "../components/ScreenshotGallery.js";
11
+ import { useAppCatalogContext } from "../../context/AppCatalogContext.js";
12
+ import { useKeyboardNavigation } from "../hooks/useKeyboardNavigation.js";
13
+ function getIconUrl(iconName) {
14
+ return `/api/icons/${iconName}`;
15
+ }
16
+ function AppIcon({
17
+ app,
18
+ className
19
+ }) {
20
+ const [imageError, setImageError] = React__default.useState(false);
21
+ if (app.iconName && !imageError) {
22
+ return /* @__PURE__ */ jsx("div", { className: cn("size-12 shrink-0", className), children: /* @__PURE__ */ jsx(
23
+ "img",
24
+ {
25
+ src: getIconUrl(app.iconName),
26
+ alt: `${app.displayName} icon`,
27
+ className: "size-12 rounded-lg object-contain",
28
+ onError: () => setImageError(true)
29
+ }
30
+ ) });
31
+ }
32
+ return /* @__PURE__ */ jsx(
33
+ "div",
34
+ {
35
+ className: cn(
36
+ "flex items-center justify-center rounded-lg bg-primary/10 text-primary size-12 shrink-0",
37
+ className
38
+ ),
39
+ children: /* @__PURE__ */ jsx(AppWindow, { className: "size-6" })
40
+ }
41
+ );
42
+ }
43
+ function AppScreenshot({ app }) {
44
+ var _a;
45
+ const [imageError, setImageError] = React__default.useState(false);
46
+ const [isLoadingImage, setIsLoadingImage] = React__default.useState(true);
47
+ const screenshotId = (_a = app.screenshotIds) == null ? void 0 : _a[0];
48
+ if (!screenshotId) {
49
+ return /* @__PURE__ */ jsx("div", { className: "w-full bg-muted/50 rounded-lg overflow-hidden flex items-center justify-center min-h-64", children: /* @__PURE__ */ jsx("div", { className: "w-full h-64 bg-muted/30 flex items-center justify-center text-muted-foreground text-sm", children: "No screenshot available" }) });
50
+ }
51
+ const screenshotImageUrl = `/api/screenshots/${screenshotId}?size=512`;
52
+ return /* @__PURE__ */ jsx("div", { className: "w-full flex justify-center", children: /* @__PURE__ */ jsxs("div", { className: "rounded-lg overflow-hidden inline-flex items-center justify-center min-h-64", children: [
53
+ !imageError ? /* @__PURE__ */ jsx(
54
+ "img",
55
+ {
56
+ src: screenshotImageUrl,
57
+ alt: `${app.displayName} screenshot`,
58
+ className: "h-64 object-contain",
59
+ onError: () => {
60
+ setImageError(true);
61
+ setIsLoadingImage(false);
62
+ },
63
+ onLoad: () => setIsLoadingImage(false)
64
+ }
65
+ ) : null,
66
+ (imageError || isLoadingImage) && /* @__PURE__ */ jsx("div", { className: "w-full h-64 bg-muted/30 flex items-center justify-center text-muted-foreground text-sm", children: isLoadingImage ? "Loading screenshot..." : "No screenshot available" })
67
+ ] }) });
68
+ }
69
+ function AppDetails({ app }) {
70
+ const [isGalleryOpen, setIsGalleryOpen] = React__default.useState(false);
71
+ const [galleryInitialIndex, setGalleryInitialIndex] = React__default.useState(0);
72
+ const { approvalMethods } = useAppCatalogContext();
73
+ const handleScreenshotClick = (index) => {
74
+ setGalleryInitialIndex(index);
75
+ setIsGalleryOpen(true);
76
+ };
77
+ return /* @__PURE__ */ jsxs(Fragment, { children: [
78
+ /* @__PURE__ */ jsxs("div", { className: "flex h-full flex-col p-6", children: [
79
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-4 border-b pb-6", children: [
80
+ /* @__PURE__ */ jsx(AppIcon, { app, className: "size-16" }),
81
+ /* @__PURE__ */ jsxs("div", { children: [
82
+ /* @__PURE__ */ jsx("h2", { className: "text-2xl font-semibold", children: app.displayName }),
83
+ app.appUrl && /* @__PURE__ */ jsxs(
84
+ "a",
85
+ {
86
+ href: app.appUrl,
87
+ target: "_blank",
88
+ rel: "noopener noreferrer",
89
+ className: "mt-1 inline-flex items-center gap-1 text-sm text-muted-foreground hover:text-primary",
90
+ children: [
91
+ app.appUrl.replaceAll(/https?:\/\//g, ""),
92
+ /* @__PURE__ */ jsx(ExternalLink, { className: "size-3" })
93
+ ]
94
+ }
95
+ )
96
+ ] })
97
+ ] }),
98
+ app.description && /* @__PURE__ */ jsxs("div", { className: "mt-6", children: [
99
+ /* @__PURE__ */ jsx("h3", { className: "mb-2 text-sm font-medium", children: "Description" }),
100
+ /* @__PURE__ */ jsx("p", { className: "text-sm text-muted-foreground", children: app.description })
101
+ ] }),
102
+ app.screenshotIds && app.screenshotIds.length > 0 && /* @__PURE__ */ jsxs("div", { className: "mt-6", children: [
103
+ /* @__PURE__ */ jsxs("h3", { className: "mb-2 text-sm font-medium", children: [
104
+ "Screenshots (",
105
+ app.screenshotIds.length,
106
+ ")"
107
+ ] }),
108
+ /* @__PURE__ */ jsxs(
109
+ "div",
110
+ {
111
+ className: "cursor-pointer hover:opacity-80 transition-opacity",
112
+ onClick: () => handleScreenshotClick(0),
113
+ children: [
114
+ /* @__PURE__ */ jsx(AppScreenshot, { app }),
115
+ app.screenshotIds.length > 1 && /* @__PURE__ */ jsxs("p", { className: "text-xs text-muted-foreground mt-2 text-center", children: [
116
+ "Click to view all ",
117
+ app.screenshotIds.length,
118
+ " screenshots"
119
+ ] })
120
+ ]
121
+ }
122
+ )
123
+ ] }),
124
+ /* @__PURE__ */ jsx(AccessRequestSection, { app, approvalMethods }),
125
+ app.tags && app.tags.length > 0 && /* @__PURE__ */ jsxs("div", { className: "mt-6", children: [
126
+ /* @__PURE__ */ jsx("h3", { className: "mb-2 text-sm font-medium", children: "Tags" }),
127
+ /* @__PURE__ */ jsx("div", { className: "flex flex-wrap gap-2", children: app.tags.map((tag) => /* @__PURE__ */ jsx(Badge, { variant: "secondary", className: "text-xs", children: tag }, tag)) })
128
+ ] }),
129
+ app.teams && app.teams.length > 0 && /* @__PURE__ */ jsxs("div", { className: "mt-6", children: [
130
+ /* @__PURE__ */ jsx("h3", { className: "mb-2 text-sm font-medium", children: "Teams" }),
131
+ /* @__PURE__ */ jsx("div", { className: "flex flex-wrap gap-2", children: app.teams.map((team) => /* @__PURE__ */ jsx(Badge, { variant: "outline", className: "text-xs", children: team }, team)) })
132
+ ] })
133
+ ] }),
134
+ /* @__PURE__ */ jsx(
135
+ ScreenshotGallery,
136
+ {
137
+ app,
138
+ screenshotIds: app.screenshotIds || [],
139
+ open: isGalleryOpen,
140
+ onOpenChange: setIsGalleryOpen,
141
+ initialIndex: galleryInitialIndex,
142
+ title: `${app.displayName} - Screenshots`
143
+ }
144
+ )
145
+ ] });
146
+ }
147
+ function groupApps(apps, groupingDef) {
148
+ var _a;
149
+ if (!groupingDef) {
150
+ const sortedApps = [...apps].sort(
151
+ (a, b) => a.displayName.localeCompare(b.displayName)
152
+ );
153
+ return [{ groupName: "All Apps", apps: sortedApps }];
154
+ }
155
+ const grouped = /* @__PURE__ */ new Map();
156
+ const ungrouped = [];
157
+ for (const app of apps) {
158
+ const matchingTag = (_a = app.tags) == null ? void 0 : _a.find(
159
+ (tag) => tag.startsWith(`${groupingDef.prefix}:`)
160
+ );
161
+ if (matchingTag) {
162
+ const value = matchingTag.split(":")[1];
163
+ if (value) {
164
+ const tagValue = groupingDef.values.find((v) => v.value === value);
165
+ const displayName = (tagValue == null ? void 0 : tagValue.displayName) || value;
166
+ if (!grouped.has(displayName)) {
167
+ grouped.set(displayName, []);
168
+ }
169
+ grouped.get(displayName).push(app);
170
+ } else {
171
+ ungrouped.push(app);
172
+ }
173
+ } else {
174
+ ungrouped.push(app);
175
+ }
176
+ }
177
+ const result = [];
178
+ for (const [groupName, appsInGroup] of grouped) {
179
+ const sortedGroupApps = appsInGroup.sort(
180
+ (a, b) => a.displayName.localeCompare(b.displayName)
181
+ );
182
+ result.push({ groupName, apps: sortedGroupApps });
183
+ }
184
+ if (ungrouped.length > 0) {
185
+ const sortedUngrouped = ungrouped.sort(
186
+ (a, b) => a.displayName.localeCompare(b.displayName)
187
+ );
188
+ result.push({ groupName: "Other", apps: sortedUngrouped });
189
+ }
190
+ return result;
191
+ }
192
+ function AppCatalogGrid({
193
+ apps,
194
+ selectedAppSlug,
195
+ groupingDefinition,
196
+ onAppClick
197
+ }) {
198
+ const selectedApp = selectedAppSlug ? apps.find((a) => a.slug === selectedAppSlug) : null;
199
+ const groupedApps = groupApps(apps, groupingDefinition);
200
+ const appsInDisplayOrder = React__default.useMemo(
201
+ () => groupedApps.flatMap((group) => group.apps),
202
+ [groupedApps]
203
+ );
204
+ const { rowRefs } = useKeyboardNavigation({
205
+ apps: appsInDisplayOrder,
206
+ selectedAppSlug,
207
+ onAppClick
208
+ });
209
+ const columns = React__default.useMemo(
210
+ () => [
211
+ {
212
+ id: "application",
213
+ header: "Application",
214
+ cell: ({ row }) => /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-3", children: [
215
+ /* @__PURE__ */ jsx(AppIcon, { app: row.original, className: "size-6" }),
216
+ /* @__PURE__ */ jsx("span", { className: "font-medium", children: row.original.displayName || "Unnamed App" })
217
+ ] }),
218
+ meta: {
219
+ className: "w-[300px]"
220
+ }
221
+ },
222
+ {
223
+ id: "description",
224
+ header: "Description",
225
+ cell: ({ row }) => /* @__PURE__ */ jsx("span", { className: "text-sm text-muted-foreground line-clamp-2", children: row.original.description || "—" })
226
+ }
227
+ ],
228
+ []
229
+ );
230
+ const table = useReactTable({
231
+ data: apps,
232
+ columns,
233
+ getCoreRowModel: getCoreRowModel(),
234
+ getRowId: (row) => row.id
235
+ });
236
+ const hasScrolledRef = React__default.useRef(false);
237
+ React__default.useEffect(() => {
238
+ if (selectedAppSlug && !hasScrolledRef.current) {
239
+ const rowElement = rowRefs.current.get(selectedAppSlug);
240
+ if (rowElement) {
241
+ rowElement.scrollIntoView({ behavior: "smooth", block: "center" });
242
+ }
243
+ hasScrolledRef.current = true;
244
+ }
245
+ }, [selectedAppSlug, rowRefs]);
246
+ const handleAppClick = (app) => {
247
+ onAppClick == null ? void 0 : onAppClick(app);
248
+ };
249
+ return /* @__PURE__ */ jsxs(ResizablePanelGroup, { orientation: "horizontal", className: "h-full w-full", children: [
250
+ /* @__PURE__ */ jsx(ResizablePanel, { defaultSize: 60, minSize: 30, className: "overflow-hidden", children: /* @__PURE__ */ jsx("div", { className: "h-full overflow-y-auto pr-2", children: /* @__PURE__ */ jsxs(Table, { children: [
251
+ /* @__PURE__ */ jsx(TableHeader, { className: "sticky top-0 border-b bg-background z-10", children: table.getHeaderGroups().map((headerGroup) => /* @__PURE__ */ jsx(TableRow, { children: headerGroup.headers.map((header) => {
252
+ var _a;
253
+ return /* @__PURE__ */ jsx(
254
+ TableHead,
255
+ {
256
+ className: cn(
257
+ "px-4 py-3 text-left font-medium text-sm",
258
+ (_a = header.column.columnDef.meta) == null ? void 0 : _a.className
259
+ ),
260
+ children: header.isPlaceholder ? null : flexRender(
261
+ header.column.columnDef.header,
262
+ header.getContext()
263
+ )
264
+ },
265
+ header.id
266
+ );
267
+ }) }, headerGroup.id)) }),
268
+ /* @__PURE__ */ jsx(TableBody, { children: groupedApps.map((group) => /* @__PURE__ */ jsxs(React__default.Fragment, { children: [
269
+ /* @__PURE__ */ jsx(TableRow, { className: "bg-muted/50 hover:bg-muted/50", children: /* @__PURE__ */ jsx(
270
+ TableCell,
271
+ {
272
+ colSpan: columns.length,
273
+ className: "px-4 py-6 sticky top-[49px] bg-muted/90 backdrop-blur z-10",
274
+ children: /* @__PURE__ */ jsx("div", { className: "flex items-center justify-center", children: /* @__PURE__ */ jsx("span", { className: "font-bold text-lg tracking-widest uppercase leading-loose text-muted-foreground", children: group.groupName }) })
275
+ }
276
+ ) }),
277
+ group.apps.map((app) => {
278
+ const row = table.getRowModel().rows.find((r) => r.id === app.id);
279
+ if (!row) return null;
280
+ return /* @__PURE__ */ jsx(
281
+ TableRow,
282
+ {
283
+ ref: (el) => {
284
+ if (el && row.original.slug) {
285
+ rowRefs.current.set(row.original.slug, el);
286
+ } else if (row.original.slug) {
287
+ rowRefs.current.delete(row.original.slug);
288
+ }
289
+ },
290
+ onClick: () => handleAppClick(row.original),
291
+ className: cn(
292
+ "border-b cursor-pointer transition-colors",
293
+ (selectedApp == null ? void 0 : selectedApp.id) === row.original.id ? "bg-blue-100 dark:bg-blue-950 hover:bg-blue-200 dark:hover:bg-blue-900" : "hover:bg-muted/30"
294
+ ),
295
+ children: row.getVisibleCells().map((cell) => {
296
+ var _a;
297
+ return /* @__PURE__ */ jsx(
298
+ TableCell,
299
+ {
300
+ className: cn(
301
+ "px-4 py-4",
302
+ (_a = cell.column.columnDef.meta) == null ? void 0 : _a.className
303
+ ),
304
+ children: flexRender(
305
+ cell.column.columnDef.cell,
306
+ cell.getContext()
307
+ )
308
+ },
309
+ cell.id
310
+ );
311
+ })
312
+ },
313
+ row.id
314
+ );
315
+ })
316
+ ] }, group.groupName)) })
317
+ ] }) }) }),
318
+ /* @__PURE__ */ jsx(ResizableHandle, { withHandle: true }),
319
+ /* @__PURE__ */ jsx(ResizablePanel, { defaultSize: 40, minSize: 25, className: "overflow-hidden", children: /* @__PURE__ */ jsx("div", { className: "h-full overflow-y-auto border-l bg-background pl-4", children: selectedApp ? /* @__PURE__ */ jsx(AppDetails, { app: selectedApp }) : /* @__PURE__ */ jsx("div", { className: "flex items-center justify-center h-full text-muted-foreground", children: "Select an app to view details" }) }) })
320
+ ] });
321
+ }
322
+ export {
323
+ AppCatalogGrid
324
+ };
325
+ //# sourceMappingURL=AppCatalogGrid.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AppCatalogGrid.js","sources":["../../../../../../src/modules/appCatalog/ui/grid/AppCatalogGrid.tsx"],"sourcesContent":["import type {\n AppForCatalog,\n GroupingTagDefinition,\n} from '@igstack/app-catalog-backend-core'\nimport type { ColumnDef } from '@tanstack/react-table'\nimport {\n flexRender,\n getCoreRowModel,\n useReactTable,\n} from '@tanstack/react-table'\nimport { AppWindow, ExternalLink } from 'lucide-react'\nimport React from 'react'\n\nimport { cn } from '~/lib/utils'\nimport type {} from '~/types/table'\nimport { Badge } from '~/ui/badge'\nimport {\n ResizableHandle,\n ResizablePanel,\n ResizablePanelGroup,\n} from '~/ui/resizable'\nimport {\n Table,\n TableBody,\n TableCell,\n TableHead,\n TableHeader,\n TableRow,\n} from '~/ui/table'\nimport { AccessRequestSection } from '../components/AccessRequestSection'\nimport { ScreenshotGallery } from '../components/ScreenshotGallery'\nimport { useAppCatalogContext } from '../../context/AppCatalogContext'\nimport { useKeyboardNavigation } from '../hooks/useKeyboardNavigation'\n\nexport interface AppCatalogGridProps {\n apps: Array<AppForCatalog>\n selectedAppSlug?: string\n groupingDefinition?: GroupingTagDefinition\n onAppClick?: (app: AppForCatalog) => void\n}\n\nfunction getIconUrl(iconName: string): string {\n return `/api/icons/${iconName}`\n}\n\nfunction AppIcon({\n app,\n className,\n}: {\n app: AppForCatalog\n className?: string\n}) {\n const [imageError, setImageError] = React.useState(false)\n\n // Use iconName from backend if available\n if (app.iconName && !imageError) {\n return (\n <div className={cn('size-12 shrink-0', className)}>\n <img\n src={getIconUrl(app.iconName)}\n alt={`${app.displayName} icon`}\n className=\"size-12 rounded-lg object-contain\"\n onError={() => setImageError(true)}\n />\n </div>\n )\n }\n\n // Fallback icon\n return (\n <div\n className={cn(\n 'flex items-center justify-center rounded-lg bg-primary/10 text-primary size-12 shrink-0',\n className,\n )}\n >\n <AppWindow className=\"size-6\" />\n </div>\n )\n}\n\nfunction AppScreenshot({ app }: { app: AppForCatalog }) {\n const [imageError, setImageError] = React.useState(false)\n const [isLoadingImage, setIsLoadingImage] = React.useState(true)\n\n // Check if app has screenshots\n const screenshotId = app.screenshotIds?.[0]\n if (!screenshotId) {\n return (\n <div className=\"w-full bg-muted/50 rounded-lg overflow-hidden flex items-center justify-center min-h-64\">\n <div className=\"w-full h-64 bg-muted/30 flex items-center justify-center text-muted-foreground text-sm\">\n No screenshot available\n </div>\n </div>\n )\n }\n\n const screenshotImageUrl = `/api/screenshots/${screenshotId}?size=512`\n\n return (\n <div className=\"w-full flex justify-center\">\n <div className=\"rounded-lg overflow-hidden inline-flex items-center justify-center min-h-64\">\n {!imageError ? (\n <img\n src={screenshotImageUrl}\n alt={`${app.displayName} screenshot`}\n className=\"h-64 object-contain\"\n onError={() => {\n setImageError(true)\n setIsLoadingImage(false)\n }}\n onLoad={() => setIsLoadingImage(false)}\n />\n ) : null}\n {(imageError || isLoadingImage) && (\n <div className=\"w-full h-64 bg-muted/30 flex items-center justify-center text-muted-foreground text-sm\">\n {isLoadingImage\n ? 'Loading screenshot...'\n : 'No screenshot available'}\n </div>\n )}\n </div>\n </div>\n )\n}\n\nfunction AppDetails({ app }: { app: AppForCatalog }) {\n const [isGalleryOpen, setIsGalleryOpen] = React.useState(false)\n const [galleryInitialIndex, setGalleryInitialIndex] = React.useState(0)\n const { approvalMethods } = useAppCatalogContext()\n\n const handleScreenshotClick = (index: number) => {\n setGalleryInitialIndex(index)\n setIsGalleryOpen(true)\n }\n\n return (\n <>\n <div className=\"flex h-full flex-col p-6\">\n {/* Icon and Title */}\n <div className=\"flex items-center gap-4 border-b pb-6\">\n <AppIcon app={app} className=\"size-16\" />\n <div>\n <h2 className=\"text-2xl font-semibold\">{app.displayName}</h2>\n {app.appUrl && (\n <a\n href={app.appUrl}\n target=\"_blank\"\n rel=\"noopener noreferrer\"\n className=\"mt-1 inline-flex items-center gap-1 text-sm text-muted-foreground hover:text-primary\"\n >\n {app.appUrl.replaceAll(/https?:\\/\\//g, '')}\n <ExternalLink className=\"size-3\" />\n </a>\n )}\n </div>\n </div>\n\n {/* Description */}\n {app.description && (\n <div className=\"mt-6\">\n <h3 className=\"mb-2 text-sm font-medium\">Description</h3>\n <p className=\"text-sm text-muted-foreground\">{app.description}</p>\n </div>\n )}\n\n {/* Screenshots - Clickable preview */}\n {app.screenshotIds && app.screenshotIds.length > 0 && (\n <div className=\"mt-6\">\n <h3 className=\"mb-2 text-sm font-medium\">\n Screenshots ({app.screenshotIds.length})\n </h3>\n <div\n className=\"cursor-pointer hover:opacity-80 transition-opacity\"\n onClick={() => handleScreenshotClick(0)}\n >\n <AppScreenshot app={app} />\n {app.screenshotIds.length > 1 && (\n <p className=\"text-xs text-muted-foreground mt-2 text-center\">\n Click to view all {app.screenshotIds.length} screenshots\n </p>\n )}\n </div>\n </div>\n )}\n\n {/* Access Request Section */}\n <AccessRequestSection app={app} approvalMethods={approvalMethods} />\n\n {/* Tags */}\n {app.tags && app.tags.length > 0 && (\n <div className=\"mt-6\">\n <h3 className=\"mb-2 text-sm font-medium\">Tags</h3>\n <div className=\"flex flex-wrap gap-2\">\n {app.tags.map((tag) => (\n <Badge key={tag} variant=\"secondary\" className=\"text-xs\">\n {tag}\n </Badge>\n ))}\n </div>\n </div>\n )}\n\n {/* Teams */}\n {app.teams && app.teams.length > 0 && (\n <div className=\"mt-6\">\n <h3 className=\"mb-2 text-sm font-medium\">Teams</h3>\n <div className=\"flex flex-wrap gap-2\">\n {app.teams.map((team) => (\n <Badge key={team} variant=\"outline\" className=\"text-xs\">\n {team}\n </Badge>\n ))}\n </div>\n </div>\n )}\n </div>\n\n {/* Screenshot Gallery Dialog */}\n <ScreenshotGallery\n app={app}\n screenshotIds={app.screenshotIds || []}\n open={isGalleryOpen}\n onOpenChange={setIsGalleryOpen}\n initialIndex={galleryInitialIndex}\n title={`${app.displayName} - Screenshots`}\n />\n </>\n )\n}\n\ninterface GroupedApps {\n groupName: string\n apps: Array<AppForCatalog>\n}\n\nfunction groupApps(\n apps: Array<AppForCatalog>,\n groupingDef?: GroupingTagDefinition,\n): Array<GroupedApps> {\n if (!groupingDef) {\n const sortedApps = [...apps].sort((a, b) =>\n a.displayName.localeCompare(b.displayName),\n )\n return [{ groupName: 'All Apps', apps: sortedApps }]\n }\n\n const grouped = new Map<string, Array<AppForCatalog>>()\n const ungrouped: Array<AppForCatalog> = []\n\n for (const app of apps) {\n const matchingTag = app.tags?.find((tag) =>\n tag.startsWith(`${groupingDef.prefix}:`),\n )\n\n if (matchingTag) {\n const value = matchingTag.split(':')[1]\n if (value) {\n const tagValue = groupingDef.values.find((v) => v.value === value)\n const displayName = tagValue?.displayName || value\n\n if (!grouped.has(displayName)) {\n grouped.set(displayName, [])\n }\n grouped.get(displayName)!.push(app)\n } else {\n ungrouped.push(app)\n }\n } else {\n ungrouped.push(app)\n }\n }\n\n const result: Array<GroupedApps> = []\n for (const [groupName, appsInGroup] of grouped) {\n // Sort apps alphabetically within each group\n const sortedGroupApps = appsInGroup.sort((a, b) =>\n a.displayName.localeCompare(b.displayName),\n )\n result.push({ groupName, apps: sortedGroupApps })\n }\n\n if (ungrouped.length > 0) {\n // Sort ungrouped apps alphabetically\n const sortedUngrouped = ungrouped.sort((a, b) =>\n a.displayName.localeCompare(b.displayName),\n )\n result.push({ groupName: 'Other', apps: sortedUngrouped })\n }\n\n return result\n}\n\nexport function AppCatalogGrid({\n apps,\n selectedAppSlug,\n groupingDefinition,\n onAppClick,\n}: AppCatalogGridProps) {\n const selectedApp = selectedAppSlug\n ? apps.find((a) => a.slug === selectedAppSlug)\n : null\n\n const groupedApps = groupApps(apps, groupingDefinition)\n\n // Flatten grouped apps to get display order for keyboard navigation\n const appsInDisplayOrder = React.useMemo(\n () => groupedApps.flatMap((group) => group.apps),\n [groupedApps],\n )\n\n // Use keyboard navigation hook with apps in display order\n const { rowRefs } = useKeyboardNavigation({\n apps: appsInDisplayOrder,\n selectedAppSlug,\n onAppClick,\n })\n\n // Define columns\n const columns = React.useMemo<Array<ColumnDef<AppForCatalog>>>(\n () => [\n {\n id: 'application',\n header: 'Application',\n cell: ({ row }) => (\n <div className=\"flex items-center gap-3\">\n <AppIcon app={row.original} className=\"size-6\" />\n <span className=\"font-medium\">\n {row.original.displayName || 'Unnamed App'}\n </span>\n </div>\n ),\n meta: {\n className: 'w-[300px]',\n },\n },\n {\n id: 'description',\n header: 'Description',\n cell: ({ row }) => (\n <span className=\"text-sm text-muted-foreground line-clamp-2\">\n {row.original.description || '—'}\n </span>\n ),\n },\n ],\n [],\n )\n\n // Create a single table instance with all apps\n const table = useReactTable({\n data: apps,\n columns,\n getCoreRowModel: getCoreRowModel(),\n getRowId: (row) => row.id,\n })\n\n // Auto-scroll to selected app (only on initial load)\n const hasScrolledRef = React.useRef(false)\n React.useEffect(() => {\n // Only scroll once on initial load if there's a selection\n if (selectedAppSlug && !hasScrolledRef.current) {\n const rowElement = rowRefs.current.get(selectedAppSlug)\n if (rowElement) {\n rowElement.scrollIntoView({ behavior: 'smooth', block: 'center' })\n }\n hasScrolledRef.current = true\n }\n }, [selectedAppSlug, rowRefs])\n\n const handleAppClick = (app: AppForCatalog) => {\n onAppClick?.(app)\n }\n\n return (\n <ResizablePanelGroup orientation=\"horizontal\" className=\"h-full w-full\">\n {/* Left Panel - Table */}\n <ResizablePanel defaultSize={60} minSize={30} className=\"overflow-hidden\">\n <div className=\"h-full overflow-y-auto pr-2\">\n <Table>\n <TableHeader className=\"sticky top-0 border-b bg-background z-10\">\n {table.getHeaderGroups().map((headerGroup) => (\n <TableRow key={headerGroup.id}>\n {headerGroup.headers.map((header) => (\n <TableHead\n key={header.id}\n className={cn(\n 'px-4 py-3 text-left font-medium text-sm',\n header.column.columnDef.meta?.className,\n )}\n >\n {header.isPlaceholder\n ? null\n : flexRender(\n header.column.columnDef.header,\n header.getContext(),\n )}\n </TableHead>\n ))}\n </TableRow>\n ))}\n </TableHeader>\n\n <TableBody>\n {groupedApps.map((group) => (\n <React.Fragment key={group.groupName}>\n {/* Group Header Row */}\n <TableRow className=\"bg-muted/50 hover:bg-muted/50\">\n <TableCell\n colSpan={columns.length}\n className=\"px-4 py-6 sticky top-[49px] bg-muted/90 backdrop-blur z-10\"\n >\n <div className=\"flex items-center justify-center\">\n <span className=\"font-bold text-lg tracking-widest uppercase leading-loose text-muted-foreground\">\n {group.groupName}\n </span>\n </div>\n </TableCell>\n </TableRow>\n\n {/* Group Apps */}\n {group.apps.map((app) => {\n const row = table\n .getRowModel()\n .rows.find((r) => r.id === app.id)\n if (!row) return null\n\n return (\n <TableRow\n key={row.id}\n ref={(el) => {\n if (el && row.original.slug) {\n rowRefs.current.set(row.original.slug, el)\n } else if (row.original.slug) {\n rowRefs.current.delete(row.original.slug)\n }\n }}\n onClick={() => handleAppClick(row.original)}\n className={cn(\n 'border-b cursor-pointer transition-colors',\n selectedApp?.id === row.original.id\n ? 'bg-blue-100 dark:bg-blue-950 hover:bg-blue-200 dark:hover:bg-blue-900'\n : 'hover:bg-muted/30',\n )}\n >\n {row.getVisibleCells().map((cell) => (\n <TableCell\n key={cell.id}\n className={cn(\n 'px-4 py-4',\n cell.column.columnDef.meta?.className,\n )}\n >\n {flexRender(\n cell.column.columnDef.cell,\n cell.getContext(),\n )}\n </TableCell>\n ))}\n </TableRow>\n )\n })}\n </React.Fragment>\n ))}\n </TableBody>\n </Table>\n </div>\n </ResizablePanel>\n\n {/* Resizable Handle */}\n <ResizableHandle withHandle />\n\n {/* Right Panel - Details */}\n <ResizablePanel defaultSize={40} minSize={25} className=\"overflow-hidden\">\n <div className=\"h-full overflow-y-auto border-l bg-background pl-4\">\n {selectedApp ? (\n <AppDetails app={selectedApp} />\n ) : (\n <div className=\"flex items-center justify-center h-full text-muted-foreground\">\n Select an app to view details\n </div>\n )}\n </div>\n </ResizablePanel>\n </ResizablePanelGroup>\n )\n}\n"],"names":["React"],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ import { AppForCatalog } from '@igstack/app-catalog-backend-core';
2
+ export interface AppCatalogTableProps {
3
+ apps: Array<AppForCatalog>;
4
+ }
5
+ export declare function AppCatalogTable({ apps }: AppCatalogTableProps): import("react/jsx-runtime").JSX.Element;
@@ -0,0 +1,2 @@
1
+ import { AppForCatalog } from '@igstack/app-catalog-backend-core';
2
+ export declare function getAppUrl(app: AppForCatalog): string;
@@ -0,0 +1,10 @@
1
+ import { AppForCatalog } from '@igstack/app-catalog-backend-core';
2
+ import { default as React } from 'react';
3
+ export interface UseKeyboardNavigationProps {
4
+ apps: Array<AppForCatalog>;
5
+ selectedAppSlug?: string;
6
+ onAppClick?: (app: AppForCatalog) => void;
7
+ }
8
+ export declare function useKeyboardNavigation({ apps, selectedAppSlug, onAppClick, }: UseKeyboardNavigationProps): {
9
+ rowRefs: React.RefObject<Map<string, HTMLTableRowElement>>;
10
+ };
@@ -0,0 +1,36 @@
1
+ import React__default from "react";
2
+ function useKeyboardNavigation({
3
+ apps,
4
+ selectedAppSlug,
5
+ onAppClick
6
+ }) {
7
+ const rowRefs = React__default.useRef(/* @__PURE__ */ new Map());
8
+ React__default.useEffect(() => {
9
+ const handleKeyDown = (event) => {
10
+ if (event.key !== "ArrowUp" && event.key !== "ArrowDown") return;
11
+ event.preventDefault();
12
+ const currentIndex = selectedAppSlug ? apps.findIndex((app) => app.slug === selectedAppSlug) : -1;
13
+ let nextIndex;
14
+ if (event.key === "ArrowDown") {
15
+ nextIndex = currentIndex === -1 ? 0 : Math.min(currentIndex + 1, apps.length - 1);
16
+ } else {
17
+ nextIndex = currentIndex === -1 ? 0 : Math.max(currentIndex - 1, 0);
18
+ }
19
+ const nextApp = apps[nextIndex];
20
+ if (nextApp && nextApp.slug !== selectedAppSlug) {
21
+ onAppClick == null ? void 0 : onAppClick(nextApp);
22
+ const rowElement = rowRefs.current.get(nextApp.slug);
23
+ if (rowElement) {
24
+ rowElement.scrollIntoView({ behavior: "smooth", block: "center" });
25
+ }
26
+ }
27
+ };
28
+ window.addEventListener("keydown", handleKeyDown);
29
+ return () => window.removeEventListener("keydown", handleKeyDown);
30
+ }, [apps, selectedAppSlug, onAppClick]);
31
+ return { rowRefs };
32
+ }
33
+ export {
34
+ useKeyboardNavigation
35
+ };
36
+ //# sourceMappingURL=useKeyboardNavigation.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useKeyboardNavigation.js","sources":["../../../../../../src/modules/appCatalog/ui/hooks/useKeyboardNavigation.ts"],"sourcesContent":["import type { AppForCatalog } from '@igstack/app-catalog-backend-core'\nimport React from 'react'\n\nexport interface UseKeyboardNavigationProps {\n apps: Array<AppForCatalog>\n selectedAppSlug?: string\n onAppClick?: (app: AppForCatalog) => void\n}\n\nexport function useKeyboardNavigation({\n apps,\n selectedAppSlug,\n onAppClick,\n}: UseKeyboardNavigationProps) {\n const rowRefs = React.useRef<Map<string, HTMLTableRowElement>>(new Map())\n\n // Keyboard navigation (ArrowUp/ArrowDown)\n React.useEffect(() => {\n const handleKeyDown = (event: KeyboardEvent) => {\n if (event.key !== 'ArrowUp' && event.key !== 'ArrowDown') return\n\n // Prevent default scrolling behavior\n event.preventDefault()\n\n const currentIndex = selectedAppSlug\n ? apps.findIndex((app) => app.slug === selectedAppSlug)\n : -1\n\n let nextIndex: number\n if (event.key === 'ArrowDown') {\n nextIndex =\n currentIndex === -1 ? 0 : Math.min(currentIndex + 1, apps.length - 1)\n } else {\n nextIndex = currentIndex === -1 ? 0 : Math.max(currentIndex - 1, 0)\n }\n\n const nextApp = apps[nextIndex]\n if (nextApp && nextApp.slug !== selectedAppSlug) {\n onAppClick?.(nextApp)\n\n // Scroll the newly selected row into view\n const rowElement = rowRefs.current.get(nextApp.slug)\n if (rowElement) {\n rowElement.scrollIntoView({ behavior: 'smooth', block: 'center' })\n }\n }\n }\n\n window.addEventListener('keydown', handleKeyDown)\n return () => window.removeEventListener('keydown', handleKeyDown)\n }, [apps, selectedAppSlug, onAppClick])\n\n return { rowRefs }\n}\n"],"names":["React"],"mappings":";AASO,SAAS,sBAAsB;AAAA,EACpC;AAAA,EACA;AAAA,EACA;AACF,GAA+B;AAC7B,QAAM,UAAUA,eAAM,OAAyC,oBAAI,KAAK;AAGxEA,iBAAM,UAAU,MAAM;AACpB,UAAM,gBAAgB,CAAC,UAAyB;AAC9C,UAAI,MAAM,QAAQ,aAAa,MAAM,QAAQ,YAAa;AAG1D,YAAM,eAAA;AAEN,YAAM,eAAe,kBACjB,KAAK,UAAU,CAAC,QAAQ,IAAI,SAAS,eAAe,IACpD;AAEJ,UAAI;AACJ,UAAI,MAAM,QAAQ,aAAa;AAC7B,oBACE,iBAAiB,KAAK,IAAI,KAAK,IAAI,eAAe,GAAG,KAAK,SAAS,CAAC;AAAA,MACxE,OAAO;AACL,oBAAY,iBAAiB,KAAK,IAAI,KAAK,IAAI,eAAe,GAAG,CAAC;AAAA,MACpE;AAEA,YAAM,UAAU,KAAK,SAAS;AAC9B,UAAI,WAAW,QAAQ,SAAS,iBAAiB;AAC/C,iDAAa;AAGb,cAAM,aAAa,QAAQ,QAAQ,IAAI,QAAQ,IAAI;AACnD,YAAI,YAAY;AACd,qBAAW,eAAe,EAAE,UAAU,UAAU,OAAO,UAAU;AAAA,QACnE;AAAA,MACF;AAAA,IACF;AAEA,WAAO,iBAAiB,WAAW,aAAa;AAChD,WAAO,MAAM,OAAO,oBAAoB,WAAW,aAAa;AAAA,EAClE,GAAG,CAAC,MAAM,iBAAiB,UAAU,CAAC;AAEtC,SAAO,EAAE,QAAA;AACX;"}
@@ -0,0 +1,9 @@
1
+ import { TRPCRouter } from '@igstack/app-catalog-backend-core';
2
+ import { QueryClient } from '@tanstack/react-query';
3
+ import { TRPCClient } from '@trpc/client';
4
+ export interface AppCatalogLayoutProps {
5
+ children: React.ReactNode;
6
+ queryClient: QueryClient;
7
+ trpcClient: TRPCClient<TRPCRouter>;
8
+ }
9
+ export declare function AppCatalogLayout({ children, queryClient, trpcClient, }: AppCatalogLayoutProps): import("react/jsx-runtime").JSX.Element;
@@ -0,0 +1,28 @@
1
+ import { jsx, jsxs } from "react/jsx-runtime";
2
+ import { HomeIcon } from "lucide-react";
3
+ import { AppCatalogProvider } from "../../context/AppCatalogContext.js";
4
+ import { Breadcrumb, BreadcrumbList, BreadcrumbItem, BreadcrumbLink, BreadcrumbSeparator } from "../../../../ui/breadcrumb.js";
5
+ import { MainLayout } from "../../../../ui/layout/MainLayout.js";
6
+ import { TopLevelProviders } from "../../../../ui/layout/TopLevelProviders.js";
7
+ import { Link } from "../../../../ui/link.js";
8
+ function AppCatalogLayout({
9
+ children,
10
+ queryClient,
11
+ trpcClient
12
+ }) {
13
+ return /* @__PURE__ */ jsx(TopLevelProviders, { queryClient, trpcClient, children: /* @__PURE__ */ jsx(AppCatalogProvider, { children: /* @__PURE__ */ jsxs(MainLayout, { children: [
14
+ /* @__PURE__ */ jsx(Breadcrumb, { className: "pb-4", children: /* @__PURE__ */ jsxs(BreadcrumbList, { children: [
15
+ /* @__PURE__ */ jsx(BreadcrumbItem, { children: /* @__PURE__ */ jsx(BreadcrumbLink, { asChild: true, children: /* @__PURE__ */ jsxs(Link, { to: "/", className: "flex items-center gap-1", children: [
16
+ /* @__PURE__ */ jsx(HomeIcon, { className: "size-4" }),
17
+ /* @__PURE__ */ jsx("span", { className: "sr-only", children: "Home" })
18
+ ] }) }) }),
19
+ /* @__PURE__ */ jsx(BreadcrumbSeparator, {}),
20
+ /* @__PURE__ */ jsx(BreadcrumbItem, { children: /* @__PURE__ */ jsx(BreadcrumbLink, { asChild: true, children: /* @__PURE__ */ jsx(Link, { to: "/catalog/apps", children: "App Catalog" }) }) })
21
+ ] }) }),
22
+ /* @__PURE__ */ jsx("div", { className: "flex flex-1 w-full justify-center min-h-0", children })
23
+ ] }) }) });
24
+ }
25
+ export {
26
+ AppCatalogLayout
27
+ };
28
+ //# sourceMappingURL=AppCatalogLayout.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AppCatalogLayout.js","sources":["../../../../../../src/modules/appCatalog/ui/layout/AppCatalogLayout.tsx"],"sourcesContent":["import type { TRPCRouter } from '@igstack/app-catalog-backend-core'\nimport type { QueryClient } from '@tanstack/react-query'\nimport type { TRPCClient } from '@trpc/client'\nimport { HomeIcon } from 'lucide-react'\nimport { AppCatalogProvider } from '~/modules/appCatalog/context/AppCatalogContext'\nimport {\n Breadcrumb,\n BreadcrumbItem,\n BreadcrumbLink,\n BreadcrumbList,\n BreadcrumbSeparator,\n} from '~/ui/breadcrumb'\nimport { MainLayout } from '~/ui/layout/MainLayout'\nimport { TopLevelProviders } from '~/ui/layout/TopLevelProviders'\nimport { Link } from '~/ui/link'\n\nexport interface AppCatalogLayoutProps {\n children: React.ReactNode\n queryClient: QueryClient\n trpcClient: TRPCClient<TRPCRouter>\n}\n\nexport function AppCatalogLayout({\n children,\n queryClient,\n trpcClient,\n}: AppCatalogLayoutProps) {\n return (\n <TopLevelProviders queryClient={queryClient} trpcClient={trpcClient}>\n <AppCatalogProvider>\n <MainLayout>\n <Breadcrumb className=\"pb-4\">\n <BreadcrumbList>\n <BreadcrumbItem>\n <BreadcrumbLink asChild>\n <Link to={'/'} className=\"flex items-center gap-1\">\n <HomeIcon className=\"size-4\" />\n <span className=\"sr-only\">Home</span>\n </Link>\n </BreadcrumbLink>\n </BreadcrumbItem>\n <BreadcrumbSeparator />\n <BreadcrumbItem>\n <BreadcrumbLink asChild>\n <Link to=\"/catalog/apps\">App Catalog</Link>\n </BreadcrumbLink>\n </BreadcrumbItem>\n </BreadcrumbList>\n </Breadcrumb>\n <div className=\"flex flex-1 w-full justify-center min-h-0\">\n {children}\n </div>\n </MainLayout>\n </AppCatalogProvider>\n </TopLevelProviders>\n )\n}\n"],"names":[],"mappings":";;;;;;;AAsBO,SAAS,iBAAiB;AAAA,EAC/B;AAAA,EACA;AAAA,EACA;AACF,GAA0B;AACxB,6BACG,mBAAA,EAAkB,aAA0B,YAC3C,UAAA,oBAAC,oBAAA,EACC,+BAAC,YAAA,EACC,UAAA;AAAA,IAAA,oBAAC,YAAA,EAAW,WAAU,QACpB,UAAA,qBAAC,gBAAA,EACC,UAAA;AAAA,MAAA,oBAAC,gBAAA,EACC,UAAA,oBAAC,gBAAA,EAAe,SAAO,MACrB,+BAAC,MAAA,EAAK,IAAI,KAAK,WAAU,2BACvB,UAAA;AAAA,QAAA,oBAAC,UAAA,EAAS,WAAU,SAAA,CAAS;AAAA,QAC7B,oBAAC,QAAA,EAAK,WAAU,WAAU,UAAA,OAAA,CAAI;AAAA,MAAA,EAAA,CAChC,GACF,GACF;AAAA,0BACC,qBAAA,EAAoB;AAAA,MACrB,oBAAC,gBAAA,EACC,UAAA,oBAAC,gBAAA,EAAe,SAAO,MACrB,UAAA,oBAAC,MAAA,EAAK,IAAG,iBAAgB,UAAA,cAAA,CAAW,EAAA,CACtC,EAAA,CACF;AAAA,IAAA,EAAA,CACF,EAAA,CACF;AAAA,IACA,oBAAC,OAAA,EAAI,WAAU,6CACZ,SAAA,CACH;AAAA,EAAA,EAAA,CACF,GACF,GACF;AAEJ;"}
@@ -0,0 +1 @@
1
+ export declare function AppCatalogPage(): import("react/jsx-runtime").JSX.Element;
@@ -0,0 +1,112 @@
1
+ import { jsx, jsxs } from "react/jsx-runtime";
2
+ import { useNavigate, useRouter, useSearch } from "@tanstack/react-router";
3
+ import { useState, useRef, useEffect, useMemo } from "react";
4
+ import { useAppCatalogContext } from "../../context/AppCatalogContext.js";
5
+ import { AppCatalogGrid } from "../grid/AppCatalogGrid.js";
6
+ import { Input } from "../../../../ui/input.js";
7
+ function AppCatalogPage() {
8
+ const navigate = useNavigate();
9
+ const router = useRouter();
10
+ const search = useSearch({ strict: false });
11
+ const { apps, isLoadingApps, tagsDefinitions } = useAppCatalogContext();
12
+ const [searchValue, setSearchValue] = useState("");
13
+ const [selectedAppSlug, setSelectedAppSlug] = useState();
14
+ const filterTag = search.filterTag;
15
+ const isInitializedRef = useRef(false);
16
+ useEffect(() => {
17
+ if (!isInitializedRef.current) {
18
+ if (search.app) {
19
+ setSelectedAppSlug(search.app);
20
+ }
21
+ if (search.q) {
22
+ setSearchValue(search.q);
23
+ }
24
+ isInitializedRef.current = true;
25
+ }
26
+ }, [search.app, search.q]);
27
+ useEffect(() => {
28
+ if (!isInitializedRef.current) return;
29
+ if (selectedAppSlug === search.app) return;
30
+ const currentPath = router.state.location.pathname;
31
+ navigate({
32
+ to: currentPath,
33
+ search: { ...search, app: selectedAppSlug },
34
+ replace: true
35
+ // Use replace to avoid polluting history
36
+ });
37
+ }, [selectedAppSlug, navigate, router.state.location.pathname, search]);
38
+ useEffect(() => {
39
+ if (!isInitializedRef.current) return;
40
+ const normalizedSearchValue = searchValue.trim();
41
+ const urlSearchValue = search.q || "";
42
+ if (normalizedSearchValue === urlSearchValue) return;
43
+ const currentPath = router.state.location.pathname;
44
+ navigate({
45
+ to: currentPath,
46
+ search: {
47
+ ...search,
48
+ q: normalizedSearchValue || void 0
49
+ // Remove param if empty
50
+ },
51
+ replace: true
52
+ // Use replace to avoid polluting history
53
+ });
54
+ }, [searchValue, navigate, router.state.location.pathname, search]);
55
+ const filteredApps = useMemo(() => {
56
+ const normalizedSearchValue = searchValue.trim().toLowerCase();
57
+ return apps.filter((app) => {
58
+ var _a, _b;
59
+ if (normalizedSearchValue === "") {
60
+ return true;
61
+ }
62
+ const name = app.displayName.toLowerCase() || "";
63
+ const slug = app.slug.toLowerCase() || "";
64
+ const description = ((_a = app.description) == null ? void 0 : _a.toLowerCase()) || "";
65
+ const tags = ((_b = app.tags) == null ? void 0 : _b.join(" ").toLowerCase()) || "";
66
+ return name.includes(normalizedSearchValue) || slug.includes(normalizedSearchValue) || description.includes(normalizedSearchValue) || tags.includes(normalizedSearchValue);
67
+ }).filter((app) => {
68
+ var _a;
69
+ return filterTag === void 0 || ((_a = app.tags) == null ? void 0 : _a.some((tag) => tag.toLowerCase() === filterTag.toLowerCase()));
70
+ });
71
+ }, [apps, searchValue, filterTag]);
72
+ const handleAppClick = (app) => {
73
+ setSelectedAppSlug(app.slug);
74
+ };
75
+ if (isLoadingApps) {
76
+ return /* @__PURE__ */ jsx("div", { className: "py-6 text-muted-foreground", children: "Loading…" });
77
+ }
78
+ const groupingDefinition = tagsDefinitions[0];
79
+ return /* @__PURE__ */ jsxs("div", { className: "flex flex-col flex-1 min-h-0", children: [
80
+ /* @__PURE__ */ jsxs("div", { className: "pb-4 shrink-0", children: [
81
+ /* @__PURE__ */ jsx("div", { className: "flex items-start justify-between gap-4 pb-4", children: /* @__PURE__ */ jsxs("div", { children: [
82
+ /* @__PURE__ */ jsx("div", { className: "font-medium text-2xl", children: "App Catalog" }),
83
+ /* @__PURE__ */ jsxs("div", { className: "text-sm text-muted-foreground", children: [
84
+ filteredApps.length,
85
+ " apps available"
86
+ ] })
87
+ ] }) }),
88
+ /* @__PURE__ */ jsx("div", { className: "w-full", children: /* @__PURE__ */ jsx(
89
+ Input,
90
+ {
91
+ value: searchValue,
92
+ onChange: (e) => setSearchValue(e.target.value),
93
+ placeholder: "Search apps by name, description, or tags…",
94
+ "aria-label": "Search apps"
95
+ }
96
+ ) })
97
+ ] }),
98
+ /* @__PURE__ */ jsx("div", { className: "flex-1 min-h-0", children: /* @__PURE__ */ jsx(
99
+ AppCatalogGrid,
100
+ {
101
+ apps: filteredApps,
102
+ selectedAppSlug,
103
+ groupingDefinition,
104
+ onAppClick: handleAppClick
105
+ }
106
+ ) })
107
+ ] });
108
+ }
109
+ export {
110
+ AppCatalogPage
111
+ };
112
+ //# sourceMappingURL=AppCatalogPage.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AppCatalogPage.js","sources":["../../../../../../src/modules/appCatalog/ui/pages/AppCatalogPage.tsx"],"sourcesContent":["import { useNavigate, useRouter, useSearch } from '@tanstack/react-router'\nimport { useEffect, useMemo, useRef, useState } from 'react'\n\nimport type { AppForCatalog } from '@igstack/app-catalog-backend-core'\nimport { useAppCatalogContext } from '../../context/AppCatalogContext'\nimport { AppCatalogGrid } from '../grid/AppCatalogGrid'\n\nimport { Input } from '~/ui/input'\n\nexport function AppCatalogPage() {\n const navigate = useNavigate()\n const router = useRouter()\n const search = useSearch({ strict: false })\n const { apps, isLoadingApps, tagsDefinitions } = useAppCatalogContext()\n const [searchValue, setSearchValue] = useState('')\n\n // Local state for app selection (source of truth)\n const [selectedAppSlug, setSelectedAppSlug] = useState<string | undefined>()\n\n const filterTag = search.filterTag\n\n // Initialize from URL on mount (once only)\n const isInitializedRef = useRef(false)\n useEffect(() => {\n if (!isInitializedRef.current) {\n if (search.app) {\n setSelectedAppSlug(search.app)\n }\n if (search.q) {\n setSearchValue(search.q)\n }\n isInitializedRef.current = true\n }\n }, [search.app, search.q])\n\n // Sync app selection state to URL (async side effect)\n useEffect(() => {\n // Don't sync until after initialization\n if (!isInitializedRef.current) return\n if (selectedAppSlug === search.app) return // Already in sync\n\n const currentPath = router.state.location.pathname\n navigate({\n to: currentPath,\n search: { ...search, app: selectedAppSlug },\n replace: true, // Use replace to avoid polluting history\n })\n }, [selectedAppSlug, navigate, router.state.location.pathname, search])\n\n // Sync search value state to URL (async side effect)\n useEffect(() => {\n // Don't sync until after initialization\n if (!isInitializedRef.current) return\n\n const normalizedSearchValue = searchValue.trim()\n const urlSearchValue = search.q || ''\n\n if (normalizedSearchValue === urlSearchValue) return // Already in sync\n\n const currentPath = router.state.location.pathname\n navigate({\n to: currentPath,\n search: {\n ...search,\n q: normalizedSearchValue || undefined, // Remove param if empty\n },\n replace: true, // Use replace to avoid polluting history\n })\n }, [searchValue, navigate, router.state.location.pathname, search])\n\n const filteredApps = useMemo(() => {\n const normalizedSearchValue = searchValue.trim().toLowerCase()\n\n return apps\n .filter((app) => {\n if (normalizedSearchValue === '') {\n return true\n }\n const name = app.displayName.toLowerCase() || ''\n const slug = app.slug.toLowerCase() || ''\n const description = app.description?.toLowerCase() || ''\n const tags = app.tags?.join(' ').toLowerCase() || ''\n\n return (\n name.includes(normalizedSearchValue) ||\n slug.includes(normalizedSearchValue) ||\n description.includes(normalizedSearchValue) ||\n tags.includes(normalizedSearchValue)\n )\n })\n .filter((app) => {\n return (\n filterTag === undefined ||\n app.tags?.some((tag) => tag.toLowerCase() === filterTag.toLowerCase())\n )\n })\n }, [apps, searchValue, filterTag])\n\n const handleAppClick = (app: AppForCatalog) => {\n setSelectedAppSlug(app.slug)\n }\n\n if (isLoadingApps) {\n return <div className=\"py-6 text-muted-foreground\">Loading…</div>\n }\n\n // Use first tag definition for grouping\n const groupingDefinition = tagsDefinitions[0]\n\n return (\n <div className=\"flex flex-col flex-1 min-h-0\">\n <div className=\"pb-4 shrink-0\">\n <div className=\"flex items-start justify-between gap-4 pb-4\">\n <div>\n <div className=\"font-medium text-2xl\">App Catalog</div>\n <div className=\"text-sm text-muted-foreground\">\n {filteredApps.length} apps available\n </div>\n </div>\n </div>\n\n <div className=\"w-full\">\n <Input\n value={searchValue}\n onChange={(e) => setSearchValue(e.target.value)}\n placeholder=\"Search apps by name, description, or tags…\"\n aria-label=\"Search apps\"\n />\n </div>\n </div>\n\n <div className=\"flex-1 min-h-0\">\n <AppCatalogGrid\n apps={filteredApps}\n selectedAppSlug={selectedAppSlug}\n groupingDefinition={groupingDefinition}\n onAppClick={handleAppClick}\n />\n </div>\n </div>\n )\n}\n"],"names":[],"mappings":";;;;;;AASO,SAAS,iBAAiB;AAC/B,QAAM,WAAW,YAAA;AACjB,QAAM,SAAS,UAAA;AACf,QAAM,SAAS,UAAU,EAAE,QAAQ,OAAO;AAC1C,QAAM,EAAE,MAAM,eAAe,gBAAA,IAAoB,qBAAA;AACjD,QAAM,CAAC,aAAa,cAAc,IAAI,SAAS,EAAE;AAGjD,QAAM,CAAC,iBAAiB,kBAAkB,IAAI,SAAA;AAE9C,QAAM,YAAY,OAAO;AAGzB,QAAM,mBAAmB,OAAO,KAAK;AACrC,YAAU,MAAM;AACd,QAAI,CAAC,iBAAiB,SAAS;AAC7B,UAAI,OAAO,KAAK;AACd,2BAAmB,OAAO,GAAG;AAAA,MAC/B;AACA,UAAI,OAAO,GAAG;AACZ,uBAAe,OAAO,CAAC;AAAA,MACzB;AACA,uBAAiB,UAAU;AAAA,IAC7B;AAAA,EACF,GAAG,CAAC,OAAO,KAAK,OAAO,CAAC,CAAC;AAGzB,YAAU,MAAM;AAEd,QAAI,CAAC,iBAAiB,QAAS;AAC/B,QAAI,oBAAoB,OAAO,IAAK;AAEpC,UAAM,cAAc,OAAO,MAAM,SAAS;AAC1C,aAAS;AAAA,MACP,IAAI;AAAA,MACJ,QAAQ,EAAE,GAAG,QAAQ,KAAK,gBAAA;AAAA,MAC1B,SAAS;AAAA;AAAA,IAAA,CACV;AAAA,EACH,GAAG,CAAC,iBAAiB,UAAU,OAAO,MAAM,SAAS,UAAU,MAAM,CAAC;AAGtE,YAAU,MAAM;AAEd,QAAI,CAAC,iBAAiB,QAAS;AAE/B,UAAM,wBAAwB,YAAY,KAAA;AAC1C,UAAM,iBAAiB,OAAO,KAAK;AAEnC,QAAI,0BAA0B,eAAgB;AAE9C,UAAM,cAAc,OAAO,MAAM,SAAS;AAC1C,aAAS;AAAA,MACP,IAAI;AAAA,MACJ,QAAQ;AAAA,QACN,GAAG;AAAA,QACH,GAAG,yBAAyB;AAAA;AAAA,MAAA;AAAA,MAE9B,SAAS;AAAA;AAAA,IAAA,CACV;AAAA,EACH,GAAG,CAAC,aAAa,UAAU,OAAO,MAAM,SAAS,UAAU,MAAM,CAAC;AAElE,QAAM,eAAe,QAAQ,MAAM;AACjC,UAAM,wBAAwB,YAAY,KAAA,EAAO,YAAA;AAEjD,WAAO,KACJ,OAAO,CAAC,QAAQ;;AACf,UAAI,0BAA0B,IAAI;AAChC,eAAO;AAAA,MACT;AACA,YAAM,OAAO,IAAI,YAAY,YAAA,KAAiB;AAC9C,YAAM,OAAO,IAAI,KAAK,YAAA,KAAiB;AACvC,YAAM,gBAAc,SAAI,gBAAJ,mBAAiB,kBAAiB;AACtD,YAAM,SAAO,SAAI,SAAJ,mBAAU,KAAK,KAAK,kBAAiB;AAElD,aACE,KAAK,SAAS,qBAAqB,KACnC,KAAK,SAAS,qBAAqB,KACnC,YAAY,SAAS,qBAAqB,KAC1C,KAAK,SAAS,qBAAqB;AAAA,IAEvC,CAAC,EACA,OAAO,CAAC,QAAQ;;AACf,aACE,cAAc,YACd,SAAI,SAAJ,mBAAU,KAAK,CAAC,QAAQ,IAAI,YAAA,MAAkB,UAAU,YAAA;AAAA,IAE5D,CAAC;AAAA,EACL,GAAG,CAAC,MAAM,aAAa,SAAS,CAAC;AAEjC,QAAM,iBAAiB,CAAC,QAAuB;AAC7C,uBAAmB,IAAI,IAAI;AAAA,EAC7B;AAEA,MAAI,eAAe;AACjB,WAAO,oBAAC,OAAA,EAAI,WAAU,8BAA6B,UAAA,YAAQ;AAAA,EAC7D;AAGA,QAAM,qBAAqB,gBAAgB,CAAC;AAE5C,SACE,qBAAC,OAAA,EAAI,WAAU,gCACb,UAAA;AAAA,IAAA,qBAAC,OAAA,EAAI,WAAU,iBACb,UAAA;AAAA,MAAA,oBAAC,OAAA,EAAI,WAAU,+CACb,UAAA,qBAAC,OAAA,EACC,UAAA;AAAA,QAAA,oBAAC,OAAA,EAAI,WAAU,wBAAuB,UAAA,eAAW;AAAA,QACjD,qBAAC,OAAA,EAAI,WAAU,iCACZ,UAAA;AAAA,UAAA,aAAa;AAAA,UAAO;AAAA,QAAA,EAAA,CACvB;AAAA,MAAA,EAAA,CACF,EAAA,CACF;AAAA,MAEA,oBAAC,OAAA,EAAI,WAAU,UACb,UAAA;AAAA,QAAC;AAAA,QAAA;AAAA,UACC,OAAO;AAAA,UACP,UAAU,CAAC,MAAM,eAAe,EAAE,OAAO,KAAK;AAAA,UAC9C,aAAY;AAAA,UACZ,cAAW;AAAA,QAAA;AAAA,MAAA,EACb,CACF;AAAA,IAAA,GACF;AAAA,IAEA,oBAAC,OAAA,EAAI,WAAU,kBACb,UAAA;AAAA,MAAC;AAAA,MAAA;AAAA,QACC,MAAM;AAAA,QACN;AAAA,QACA;AAAA,QACA,YAAY;AAAA,MAAA;AAAA,IAAA,EACd,CACF;AAAA,EAAA,GACF;AAEJ;"}
@@ -0,0 +1,7 @@
1
+ import { Control, FieldValues } from 'react-hook-form';
2
+ interface AccessRequestFormFieldsProps {
3
+ control: Control<FieldValues>;
4
+ approvalMethodId: string | undefined;
5
+ }
6
+ export declare function AccessRequestFormFields({ control, approvalMethodId, }: AccessRequestFormFieldsProps): import("react/jsx-runtime").JSX.Element;
7
+ export {};