@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.
- package/LICENSE +21 -0
- package/dist/esm/App.d.ts +12 -0
- package/dist/esm/App.js +12 -0
- package/dist/esm/App.js.map +1 -0
- package/dist/esm/__tests__/setupTests.d.ts +0 -0
- package/dist/esm/api/infra/createQueryClient.d.ts +9 -0
- package/dist/esm/api/infra/createQueryClient.js +25 -0
- package/dist/esm/api/infra/createQueryClient.js.map +1 -0
- package/dist/esm/api/infra/trpc.d.ts +1624 -0
- package/dist/esm/api/infra/trpc.js +8 -0
- package/dist/esm/api/infra/trpc.js.map +1 -0
- package/dist/esm/api/unsorted/appCatalogFetcher.d.ts +10 -0
- package/dist/esm/api/unsorted/appCatalogFetcher.js +18 -0
- package/dist/esm/api/unsorted/appCatalogFetcher.js.map +1 -0
- package/dist/esm/api/unsorted/createCachingFetcher.d.ts +23 -0
- package/dist/esm/api/unsorted/createCachingFetcher.js +93 -0
- package/dist/esm/api/unsorted/createCachingFetcher.js.map +1 -0
- package/dist/esm/appPropsFactory.d.ts +2 -0
- package/dist/esm/appPropsFactory.js +34 -0
- package/dist/esm/appPropsFactory.js.map +1 -0
- package/dist/esm/assets/app-catalog.svg.js +6 -0
- package/dist/esm/assets/app-catalog.svg.js.map +1 -0
- package/dist/esm/components/IconPickerDialog.d.ts +8 -0
- package/dist/esm/components/IconPickerDialog.js +98 -0
- package/dist/esm/components/IconPickerDialog.js.map +1 -0
- package/dist/esm/components/IconPickerField.d.ts +9 -0
- package/dist/esm/components/IconPickerField.js +76 -0
- package/dist/esm/components/IconPickerField.js.map +1 -0
- package/dist/esm/components/ThemeSwitcher.d.ts +1 -0
- package/dist/esm/components/ThemeSwitcher.js +25 -0
- package/dist/esm/components/ThemeSwitcher.js.map +1 -0
- package/dist/esm/components/theme-provider.d.ts +2 -0
- package/dist/esm/components/theme-provider.js +10 -0
- package/dist/esm/components/theme-provider.js.map +1 -0
- package/dist/esm/errors/AuthorizationError.d.ts +19 -0
- package/dist/esm/errors/AuthorizationError.js +20 -0
- package/dist/esm/errors/AuthorizationError.js.map +1 -0
- package/dist/esm/errors/index.d.ts +1 -0
- package/dist/esm/index.d.ts +3 -0
- package/dist/esm/index.js +7 -0
- package/dist/esm/index.js.map +1 -0
- package/dist/esm/lib/utils.d.ts +2 -0
- package/dist/esm/lib/utils.js +9 -0
- package/dist/esm/lib/utils.js.map +1 -0
- package/dist/esm/main.d.ts +0 -0
- package/dist/esm/modules/admin-base/components/AdminChat.d.ts +1 -0
- package/dist/esm/modules/admin-base/components/AdminChat.js +82 -0
- package/dist/esm/modules/admin-base/components/AdminChat.js.map +1 -0
- package/dist/esm/modules/admin-base/components/AdminLayout.d.ts +5 -0
- package/dist/esm/modules/admin-base/components/AdminLayout.js +83 -0
- package/dist/esm/modules/admin-base/components/AdminLayout.js.map +1 -0
- package/dist/esm/modules/admin-base/components/AdminWelcome.d.ts +1 -0
- package/dist/esm/modules/admin-base/components/AdminWelcome.js +37 -0
- package/dist/esm/modules/admin-base/components/AdminWelcome.js.map +1 -0
- package/dist/esm/modules/admin-base/context/AdminConfigContext.d.ts +8 -0
- package/dist/esm/modules/admin-base/context/AdminConfigContext.js +27 -0
- package/dist/esm/modules/admin-base/context/AdminConfigContext.js.map +1 -0
- package/dist/esm/modules/admin-base/index.d.ts +5 -0
- package/dist/esm/modules/admin-base/types/adminTypes.d.ts +10 -0
- package/dist/esm/modules/appCatalog/AppCatalogAdminPage.d.ts +1 -0
- package/dist/esm/modules/appCatalog/AppCatalogAdminPage.js +196 -0
- package/dist/esm/modules/appCatalog/AppCatalogAdminPage.js.map +1 -0
- package/dist/esm/modules/appCatalog/ScreenshotItem.d.ts +7 -0
- package/dist/esm/modules/appCatalog/ScreenshotItem.js +57 -0
- package/dist/esm/modules/appCatalog/ScreenshotItem.js.map +1 -0
- package/dist/esm/modules/appCatalog/ScreenshotManager.d.ts +15 -0
- package/dist/esm/modules/appCatalog/ScreenshotManager.js +155 -0
- package/dist/esm/modules/appCatalog/ScreenshotManager.js.map +1 -0
- package/dist/esm/modules/appCatalog/api/ApiQueryMagazineAppCatalog.d.ts +11 -0
- package/dist/esm/modules/appCatalog/api/ApiQueryMagazineAppCatalog.js +14 -0
- package/dist/esm/modules/appCatalog/api/ApiQueryMagazineAppCatalog.js.map +1 -0
- package/dist/esm/modules/appCatalog/catalogRouteLoader.d.ts +5 -0
- package/dist/esm/modules/appCatalog/catalogRouteLoader.js +20 -0
- package/dist/esm/modules/appCatalog/catalogRouteLoader.js.map +1 -0
- package/dist/esm/modules/appCatalog/context/AppCatalogContext.d.ts +15 -0
- package/dist/esm/modules/appCatalog/context/AppCatalogContext.js +47 -0
- package/dist/esm/modules/appCatalog/context/AppCatalogContext.js.map +1 -0
- package/dist/esm/modules/appCatalog/index.d.ts +10 -0
- package/dist/esm/modules/appCatalog/routeLoader.d.ts +3 -0
- package/dist/esm/modules/appCatalog/routeLoader.js +7 -0
- package/dist/esm/modules/appCatalog/routeLoader.js.map +1 -0
- package/dist/esm/modules/appCatalog/ui/components/AccessRequestSection.d.ts +7 -0
- package/dist/esm/modules/appCatalog/ui/components/AccessRequestSection.js +257 -0
- package/dist/esm/modules/appCatalog/ui/components/AccessRequestSection.js.map +1 -0
- package/dist/esm/modules/appCatalog/ui/components/AppDetailModal.d.ts +7 -0
- package/dist/esm/modules/appCatalog/ui/components/ApproverDisplay.d.ts +30 -0
- package/dist/esm/modules/appCatalog/ui/components/GroupingColumn.d.ts +11 -0
- package/dist/esm/modules/appCatalog/ui/components/GroupingTabs.d.ts +7 -0
- package/dist/esm/modules/appCatalog/ui/components/ScreenshotGallery.d.ts +10 -0
- package/dist/esm/modules/appCatalog/ui/components/ScreenshotGallery.js +35 -0
- package/dist/esm/modules/appCatalog/ui/components/ScreenshotGallery.js.map +1 -0
- package/dist/esm/modules/appCatalog/ui/grid/AppCatalogFiltersCard.d.ts +9 -0
- package/dist/esm/modules/appCatalog/ui/grid/AppCatalogGrid.d.ts +8 -0
- package/dist/esm/modules/appCatalog/ui/grid/AppCatalogGrid.js +325 -0
- package/dist/esm/modules/appCatalog/ui/grid/AppCatalogGrid.js.map +1 -0
- package/dist/esm/modules/appCatalog/ui/grid/AppCatalogTable.d.ts +5 -0
- package/dist/esm/modules/appCatalog/ui/grid/appCatalogUtils.d.ts +2 -0
- package/dist/esm/modules/appCatalog/ui/hooks/useKeyboardNavigation.d.ts +10 -0
- package/dist/esm/modules/appCatalog/ui/hooks/useKeyboardNavigation.js +36 -0
- package/dist/esm/modules/appCatalog/ui/hooks/useKeyboardNavigation.js.map +1 -0
- package/dist/esm/modules/appCatalog/ui/layout/AppCatalogLayout.d.ts +9 -0
- package/dist/esm/modules/appCatalog/ui/layout/AppCatalogLayout.js +28 -0
- package/dist/esm/modules/appCatalog/ui/layout/AppCatalogLayout.js.map +1 -0
- package/dist/esm/modules/appCatalog/ui/pages/AppCatalogPage.d.ts +1 -0
- package/dist/esm/modules/appCatalog/ui/pages/AppCatalogPage.js +112 -0
- package/dist/esm/modules/appCatalog/ui/pages/AppCatalogPage.js.map +1 -0
- package/dist/esm/modules/approvalMethod/AccessRequestFormFields.d.ts +7 -0
- package/dist/esm/modules/approvalMethod/AccessRequestFormFields.js +323 -0
- package/dist/esm/modules/approvalMethod/AccessRequestFormFields.js.map +1 -0
- package/dist/esm/modules/approvalMethod/ApprovalMethodForm.d.ts +14 -0
- package/dist/esm/modules/approvalMethod/ApprovalMethodForm.js +227 -0
- package/dist/esm/modules/approvalMethod/ApprovalMethodForm.js.map +1 -0
- package/dist/esm/modules/approvalMethod/ApprovalMethodSelector.d.ts +7 -0
- package/dist/esm/modules/approvalMethod/ApprovalMethodSelector.js +124 -0
- package/dist/esm/modules/approvalMethod/ApprovalMethodSelector.js.map +1 -0
- package/dist/esm/modules/approvalMethod/api/ApiQueryMagazineApprovalMethod.d.ts +381 -0
- package/dist/esm/modules/approvalMethod/api/ApiQueryMagazineApprovalMethod.js +26 -0
- package/dist/esm/modules/approvalMethod/api/ApiQueryMagazineApprovalMethod.js.map +1 -0
- package/dist/esm/modules/auth/AuthContext.d.ts +38 -0
- package/dist/esm/modules/auth/AuthContext.js +88 -0
- package/dist/esm/modules/auth/AuthContext.js.map +1 -0
- package/dist/esm/modules/auth/AuthModalContext.d.ts +19 -0
- package/dist/esm/modules/auth/AuthModalContext.js +33 -0
- package/dist/esm/modules/auth/AuthModalContext.js.map +1 -0
- package/dist/esm/modules/auth/ProtectedContent.d.ts +19 -0
- package/dist/esm/modules/auth/authClient.d.ts +704 -0
- package/dist/esm/modules/auth/authClient.js +8 -0
- package/dist/esm/modules/auth/authClient.js.map +1 -0
- package/dist/esm/modules/auth/authUtils.d.ts +6 -0
- package/dist/esm/modules/auth/authUtils.js +25 -0
- package/dist/esm/modules/auth/authUtils.js.map +1 -0
- package/dist/esm/modules/auth/index.d.ts +5 -0
- package/dist/esm/modules/auth/ui/LoginModal.d.ts +6 -0
- package/dist/esm/modules/auth/ui/LoginModal.js +22 -0
- package/dist/esm/modules/auth/ui/LoginModal.js.map +1 -0
- package/dist/esm/modules/auth/ui/LoginPage.d.ts +5 -0
- package/dist/esm/modules/auth/ui/LoginPage.js +50 -0
- package/dist/esm/modules/auth/ui/LoginPage.js.map +1 -0
- package/dist/esm/modules/auth/useAuthActions.d.ts +17 -0
- package/dist/esm/modules/auth/useAuthActions.js +62 -0
- package/dist/esm/modules/auth/useAuthActions.js.map +1 -0
- package/dist/esm/modules/config/GlobalConfigContext.d.ts +14 -0
- package/dist/esm/modules/config/GlobalConfigContext.js +21 -0
- package/dist/esm/modules/config/GlobalConfigContext.js.map +1 -0
- package/dist/esm/modules/config/HealthStateContext.d.ts +16 -0
- package/dist/esm/modules/gallery/Gallery.d.ts +12 -0
- package/dist/esm/modules/gallery/Gallery.js +246 -0
- package/dist/esm/modules/gallery/Gallery.js.map +1 -0
- package/dist/esm/modules/icons/IconManagementPage.d.ts +1 -0
- package/dist/esm/modules/icons/IconManagementPage.js +177 -0
- package/dist/esm/modules/icons/IconManagementPage.js.map +1 -0
- package/dist/esm/modules/pluginCore/PluginManagerContext.d.ts +19 -0
- package/dist/esm/modules/pluginCore/PluginManagerContext.js +29 -0
- package/dist/esm/modules/pluginCore/PluginManagerContext.js.map +1 -0
- package/dist/esm/modules/pluginCore/makePluginManagerContext.d.ts +4 -0
- package/dist/esm/modules/pluginCore/makePluginManagerContext.js +9 -0
- package/dist/esm/modules/pluginCore/makePluginManagerContext.js.map +1 -0
- package/dist/esm/modules/pluginCore/types.d.ts +24 -0
- 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
- 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
- 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
- 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
- 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
- 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
- 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
- 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
- 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
- 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
- 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
- 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
- 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
- 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
- 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
- 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
- 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
- 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
- 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
- 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
- package/dist/esm/node_modules/.pnpm/@tanstack_router-core@1.151.6/node_modules/@tanstack/router-core/dist/esm/path.js +107 -0
- package/dist/esm/node_modules/.pnpm/@tanstack_router-core@1.151.6/node_modules/@tanstack/router-core/dist/esm/path.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/@tanstack_router-core@1.151.6/node_modules/@tanstack/router-core/dist/esm/root.js +5 -0
- package/dist/esm/node_modules/.pnpm/@tanstack_router-core@1.151.6/node_modules/@tanstack/router-core/dist/esm/root.js.map +1 -0
- 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
- 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
- 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
- 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
- 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
- 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
- package/dist/esm/node_modules/.pnpm/goober@2.1.18_csstype@3.2.3/node_modules/goober/dist/goober.modern.js +56 -0
- package/dist/esm/node_modules/.pnpm/goober@2.1.18_csstype@3.2.3/node_modules/goober/dist/goober.modern.js.map +1 -0
- 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
- 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
- package/dist/esm/node_modules/.pnpm/tiny-invariant@1.3.3/node_modules/tiny-invariant/dist/esm/tiny-invariant.js +17 -0
- package/dist/esm/node_modules/.pnpm/tiny-invariant@1.3.3/node_modules/tiny-invariant/dist/esm/tiny-invariant.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/classic/errors.js +45 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/classic/errors.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/classic/iso.js +44 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/classic/iso.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/classic/parse.js +30 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/classic/parse.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/classic/schemas.js +632 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/classic/schemas.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/api.js +500 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/api.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/checks.js +244 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/checks.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/core.js +75 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/core.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/doc.js +38 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/doc.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/errors.js +73 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/errors.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/json-schema-processors.js +274 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/json-schema-processors.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/parse.js +100 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/parse.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/regexes.js +80 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/regexes.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/registries.js +52 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/registries.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/schemas.js +1247 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/schemas.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/to-json-schema.js +363 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/to-json-schema.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/util.js +407 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/util.js.map +1 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/versions.js +9 -0
- package/dist/esm/node_modules/.pnpm/zod@4.3.5/node_modules/zod/v4/core/versions.js.map +1 -0
- package/dist/esm/plugins/builtin/pageUrl/pageUrlTypes.d.ts +11 -0
- package/dist/esm/routeTree.gen.d.ts +263 -0
- package/dist/esm/routeTree.gen.js +117 -0
- package/dist/esm/routeTree.gen.js.map +1 -0
- package/dist/esm/routes/__root.d.ts +2 -0
- package/dist/esm/routes/__root.js +22 -0
- package/dist/esm/routes/__root.js.map +1 -0
- package/dist/esm/routes/_layout/catalog.apps.index.d.ts +5 -0
- package/dist/esm/routes/_layout/catalog.apps.index.js +25 -0
- package/dist/esm/routes/_layout/catalog.apps.index.js.map +1 -0
- package/dist/esm/routes/_layout/index.d.ts +7 -0
- package/dist/esm/routes/_layout/index.js +28 -0
- package/dist/esm/routes/_layout/index.js.map +1 -0
- package/dist/esm/routes/_layout/login.d.ts +1 -0
- package/dist/esm/routes/_layout/login.js +16 -0
- package/dist/esm/routes/_layout/login.js.map +1 -0
- package/dist/esm/routes/_layout.d.ts +1 -0
- package/dist/esm/routes/_layout.js +12 -0
- package/dist/esm/routes/_layout.js.map +1 -0
- package/dist/esm/routes/admin/app-for-catalog/$id.d.ts +5 -0
- package/dist/esm/routes/admin/app-for-catalog/_id.js +67 -0
- package/dist/esm/routes/admin/app-for-catalog/_id.js.map +1 -0
- package/dist/esm/routes/admin/app-for-catalog/_id2.js +321 -0
- package/dist/esm/routes/admin/app-for-catalog/_id2.js.map +1 -0
- package/dist/esm/routes/admin/app-for-catalog/index.d.ts +1 -0
- package/dist/esm/routes/admin/app-for-catalog/index.js +9 -0
- package/dist/esm/routes/admin/app-for-catalog/index.js.map +1 -0
- package/dist/esm/routes/admin/app-for-catalog/index2.js +12 -0
- package/dist/esm/routes/admin/app-for-catalog/index2.js.map +1 -0
- package/dist/esm/routes/admin/app-for-catalog.d.ts +1 -0
- package/dist/esm/routes/admin/app-for-catalog.js +14 -0
- package/dist/esm/routes/admin/app-for-catalog.js.map +1 -0
- package/dist/esm/routes/admin/app-for-catalog2.js +9 -0
- package/dist/esm/routes/admin/app-for-catalog2.js.map +1 -0
- package/dist/esm/routes/admin/approval-methods/index.d.ts +32 -0
- package/dist/esm/routes/admin/approval-methods/index.js +24 -0
- package/dist/esm/routes/admin/approval-methods/index.js.map +1 -0
- package/dist/esm/routes/admin/approval-methods/index2.js +100 -0
- package/dist/esm/routes/admin/approval-methods/index2.js.map +1 -0
- package/dist/esm/routes/admin/approval-methods.d.ts +1 -0
- package/dist/esm/routes/admin/approval-methods.js +14 -0
- package/dist/esm/routes/admin/approval-methods.js.map +1 -0
- package/dist/esm/routes/admin/approval-methods2.js +7 -0
- package/dist/esm/routes/admin/approval-methods2.js.map +1 -0
- package/dist/esm/routes/admin/chat.d.ts +1 -0
- package/dist/esm/routes/admin/chat.js +14 -0
- package/dist/esm/routes/admin/chat.js.map +1 -0
- package/dist/esm/routes/admin/chat2.js +9 -0
- package/dist/esm/routes/admin/chat2.js.map +1 -0
- package/dist/esm/routes/admin/icons.d.ts +1 -0
- package/dist/esm/routes/admin/icons.js +14 -0
- package/dist/esm/routes/admin/icons.js.map +1 -0
- package/dist/esm/routes/admin/icons2.js +12 -0
- package/dist/esm/routes/admin/icons2.js.map +1 -0
- package/dist/esm/routes/admin/index.d.ts +1 -0
- package/dist/esm/routes/admin/index.js +9 -0
- package/dist/esm/routes/admin/index.js.map +1 -0
- package/dist/esm/routes/admin/index2.js +9 -0
- package/dist/esm/routes/admin/index2.js.map +1 -0
- package/dist/esm/routes/admin.d.ts +1 -0
- package/dist/esm/routes/admin.js +37 -0
- package/dist/esm/routes/admin.js.map +1 -0
- package/dist/esm/routes/admin2.js +18 -0
- package/dist/esm/routes/admin2.js.map +1 -0
- package/dist/esm/routes/auth.callback.d.ts +4 -0
- package/dist/esm/routes/auth.callback.js +56 -0
- package/dist/esm/routes/auth.callback.js.map +1 -0
- package/dist/esm/types/slateTypes.d.ts +28 -0
- package/dist/esm/types/table.d.ts +7 -0
- package/dist/esm/types/tanstackQuery.d.ts +13 -0
- package/dist/esm/types/types.d.ts +25 -0
- package/dist/esm/types/userBehaviourTypes.d.ts +76 -0
- package/dist/esm/types/utilityTypes.d.ts +1 -0
- package/dist/esm/ui/accordion.d.ts +7 -0
- package/dist/esm/ui/accordion.js +65 -0
- package/dist/esm/ui/accordion.js.map +1 -0
- package/dist/esm/ui/alert-dialog.d.ts +14 -0
- package/dist/esm/ui/alert-dialog.js +141 -0
- package/dist/esm/ui/alert-dialog.js.map +1 -0
- package/dist/esm/ui/autocomplete.d.ts +29 -0
- package/dist/esm/ui/badge.d.ts +9 -0
- package/dist/esm/ui/badge.js +27 -0
- package/dist/esm/ui/badge.js.map +1 -0
- package/dist/esm/ui/breadcrumb.d.ts +11 -0
- package/dist/esm/ui/breadcrumb.js +84 -0
- package/dist/esm/ui/breadcrumb.js.map +1 -0
- package/dist/esm/ui/button-group.d.ts +11 -0
- package/dist/esm/ui/button.d.ts +13 -0
- package/dist/esm/ui/button.js +53 -0
- package/dist/esm/ui/button.js.map +1 -0
- package/dist/esm/ui/card.d.ts +9 -0
- package/dist/esm/ui/card.js +66 -0
- package/dist/esm/ui/card.js.map +1 -0
- package/dist/esm/ui/checkbox.d.ts +4 -0
- package/dist/esm/ui/collapsible.d.ts +5 -0
- package/dist/esm/ui/command.d.ts +20 -0
- package/dist/esm/ui/command.js +131 -0
- package/dist/esm/ui/command.js.map +1 -0
- package/dist/esm/ui/components/ActionCard.d.ts +9 -0
- package/dist/esm/ui/components/AppIcon.d.ts +7 -0
- package/dist/esm/ui/components/Breadcrumbs.d.ts +19 -0
- package/dist/esm/ui/components/Breadcrumbs.js +36 -0
- package/dist/esm/ui/components/Breadcrumbs.js.map +1 -0
- package/dist/esm/ui/components/commandInput/EhBaseSelector.d.ts +2 -0
- package/dist/esm/ui/components/commandInput/types.d.ts +13 -0
- package/dist/esm/ui/components/error/BaseErrorPage.d.ts +10 -0
- package/dist/esm/ui/components/error/BaseErrorPage.js +16 -0
- package/dist/esm/ui/components/error/BaseErrorPage.js.map +1 -0
- package/dist/esm/ui/components/error/DefaultErrorComponent.d.ts +3 -0
- package/dist/esm/ui/components/error/DefaultErrorComponent.js +73 -0
- package/dist/esm/ui/components/error/DefaultErrorComponent.js.map +1 -0
- package/dist/esm/ui/components/error/ForbiddenErrorPage.d.ts +5 -0
- package/dist/esm/ui/components/error/ForbiddenErrorPage.js +54 -0
- package/dist/esm/ui/components/error/ForbiddenErrorPage.js.map +1 -0
- package/dist/esm/ui/components/error/RootErrorPage.d.ts +5 -0
- package/dist/esm/ui/components/error/RootErrorPage.js +15 -0
- package/dist/esm/ui/components/error/RootErrorPage.js.map +1 -0
- package/dist/esm/ui/components/footer/Footer.d.ts +1 -0
- package/dist/esm/ui/components/header/Header.d.ts +5 -0
- package/dist/esm/ui/components/header/Header.js +103 -0
- package/dist/esm/ui/components/header/Header.js.map +1 -0
- package/dist/esm/ui/components/widgetPanel/AddWidgetCard.d.ts +5 -0
- package/dist/esm/ui/components/widgetPanel/WidgetGrid.d.ts +6 -0
- package/dist/esm/ui/components/widgets/CredentialsWidget.d.ts +11 -0
- package/dist/esm/ui/components/widgets/VersionWidget.d.ts +7 -0
- package/dist/esm/ui/crud-list/CrudList.d.ts +2 -0
- package/dist/esm/ui/crud-list/CrudList.js +189 -0
- package/dist/esm/ui/crud-list/CrudList.js.map +1 -0
- package/dist/esm/ui/crud-list/index.d.ts +2 -0
- package/dist/esm/ui/crud-list/types.d.ts +35 -0
- package/dist/esm/ui/dialog.d.ts +15 -0
- package/dist/esm/ui/dialog.js +111 -0
- package/dist/esm/ui/dialog.js.map +1 -0
- package/dist/esm/ui/dropdown-menu.d.ts +25 -0
- package/dist/esm/ui/dropdown-menu.js +79 -0
- package/dist/esm/ui/dropdown-menu.js.map +1 -0
- package/dist/esm/ui/editable-list/EditableListField.d.ts +2 -0
- package/dist/esm/ui/editable-list/EditableListField.js +130 -0
- package/dist/esm/ui/editable-list/EditableListField.js.map +1 -0
- package/dist/esm/ui/editable-list/index.d.ts +2 -0
- package/dist/esm/ui/editable-list/types.d.ts +17 -0
- package/dist/esm/ui/empty.d.ts +11 -0
- package/dist/esm/ui/empty.js +103 -0
- package/dist/esm/ui/empty.js.map +1 -0
- package/dist/esm/ui/error/NotFoundError.d.ts +1 -0
- package/dist/esm/ui/error/NotFoundError.js +17 -0
- package/dist/esm/ui/error/NotFoundError.js.map +1 -0
- package/dist/esm/ui/form.d.ts +56 -0
- package/dist/esm/ui/form.js +134 -0
- package/dist/esm/ui/form.js.map +1 -0
- package/dist/esm/ui/input-group.d.ts +16 -0
- package/dist/esm/ui/input.d.ts +3 -0
- package/dist/esm/ui/input.js +22 -0
- package/dist/esm/ui/input.js.map +1 -0
- package/dist/esm/ui/item.d.ts +23 -0
- package/dist/esm/ui/label.d.ts +4 -0
- package/dist/esm/ui/label.js +23 -0
- package/dist/esm/ui/label.js.map +1 -0
- package/dist/esm/ui/layout/Footer.d.ts +1 -0
- package/dist/esm/ui/layout/LoadingScreen.d.ts +6 -0
- package/dist/esm/ui/layout/LoadingScreen.js +54 -0
- package/dist/esm/ui/layout/LoadingScreen.js.map +1 -0
- package/dist/esm/ui/layout/MainLayout.d.ts +7 -0
- package/dist/esm/ui/layout/MainLayout.js +17 -0
- package/dist/esm/ui/layout/MainLayout.js.map +1 -0
- package/dist/esm/ui/layout/SideColumn.d.ts +1 -0
- package/dist/esm/ui/layout/TopLevelProviders.d.ts +10 -0
- package/dist/esm/ui/layout/TopLevelProviders.js +46 -0
- package/dist/esm/ui/layout/TopLevelProviders.js.map +1 -0
- package/dist/esm/ui/layout/TopLevelProvidersForErrors.d.ts +10 -0
- package/dist/esm/ui/layout/TopLevelProvidersForErrors.js +24 -0
- package/dist/esm/ui/layout/TopLevelProvidersForErrors.js.map +1 -0
- package/dist/esm/ui/link.d.ts +13 -0
- package/dist/esm/ui/link.js +28 -0
- package/dist/esm/ui/link.js.map +1 -0
- package/dist/esm/ui/linkExternal.d.ts +8 -0
- package/dist/esm/ui/linkExternal.js +26 -0
- package/dist/esm/ui/linkExternal.js.map +1 -0
- package/dist/esm/ui/main/JumpTabContent.d.ts +1 -0
- package/dist/esm/ui/main/Tabs.d.ts +4 -0
- package/dist/esm/ui/markdown-editor/MarkdownEditor.d.ts +9 -0
- package/dist/esm/ui/markdown-editor/MarkdownEditor.js +116 -0
- package/dist/esm/ui/markdown-editor/MarkdownEditor.js.map +1 -0
- package/dist/esm/ui/markdown-editor/MarkdownToolbar.d.ts +6 -0
- package/dist/esm/ui/markdown-editor/MarkdownToolbar.js +99 -0
- package/dist/esm/ui/markdown-editor/MarkdownToolbar.js.map +1 -0
- package/dist/esm/ui/markdown-editor/index.d.ts +2 -0
- package/dist/esm/ui/popover.d.ts +7 -0
- package/dist/esm/ui/popover.js +40 -0
- package/dist/esm/ui/popover.js.map +1 -0
- package/dist/esm/ui/radio-group.d.ts +5 -0
- package/dist/esm/ui/resizable.d.ts +7 -0
- package/dist/esm/ui/resizable.js +47 -0
- package/dist/esm/ui/resizable.js.map +1 -0
- package/dist/esm/ui/scroll-area.d.ts +5 -0
- package/dist/esm/ui/scroll-area.js +62 -0
- package/dist/esm/ui/scroll-area.js.map +1 -0
- package/dist/esm/ui/search-input-with-shortcut.d.ts +25 -0
- package/dist/esm/ui/select.d.ts +15 -0
- package/dist/esm/ui/select.js +138 -0
- package/dist/esm/ui/select.js.map +1 -0
- package/dist/esm/ui/separator.d.ts +4 -0
- package/dist/esm/ui/shortcut-button.d.ts +24 -0
- package/dist/esm/ui/skeleton.d.ts +2 -0
- package/dist/esm/ui/skeleton.js +16 -0
- package/dist/esm/ui/skeleton.js.map +1 -0
- package/dist/esm/ui/spinner.d.ts +2 -0
- package/dist/esm/ui/spinner.js +18 -0
- package/dist/esm/ui/spinner.js.map +1 -0
- package/dist/esm/ui/table.d.ts +10 -0
- package/dist/esm/ui/table.js +87 -0
- package/dist/esm/ui/table.js.map +1 -0
- package/dist/esm/ui/tabs.d.ts +7 -0
- package/dist/esm/ui/textarea.d.ts +3 -0
- package/dist/esm/ui/textarea.js +19 -0
- package/dist/esm/ui/textarea.js.map +1 -0
- package/dist/esm/ui/tooltip.d.ts +7 -0
- package/dist/esm/userDb/AcDb.d.ts +14 -0
- package/dist/esm/userDb/AcDb.js +28 -0
- package/dist/esm/userDb/AcDb.js.map +1 -0
- package/dist/esm/userDb/DbContext.d.ts +9 -0
- package/dist/esm/userDb/DbContext.js +18 -0
- package/dist/esm/userDb/DbContext.js.map +1 -0
- package/dist/esm/util/createEhRouter.d.ts +7 -0
- package/dist/esm/util/createEhRouter.js +19 -0
- package/dist/esm/util/createEhRouter.js.map +1 -0
- package/dist/esm/util/error-utils.d.ts +6 -0
- package/dist/esm/util/error-utils.js +28 -0
- package/dist/esm/util/error-utils.js.map +1 -0
- package/dist/esm/util/highlightMatches.d.ts +2 -0
- package/dist/esm/util/reactQueryUtils.d.ts +10 -0
- package/dist/esm/util/reactQueryUtils.js +17 -0
- package/dist/esm/util/reactQueryUtils.js.map +1 -0
- package/dist/esm/util/slug-utils.d.ts +12 -0
- package/dist/index.css +225 -0
- package/dist/public/app-catalog-16x16.png +0 -0
- package/dist/public/app-catalog-192x192.png +0 -0
- package/dist/public/app-catalog-32x32.png +0 -0
- package/dist/public/app-catalog-48x48.png +0 -0
- package/dist/public/app-catalog-512x512.png +0 -0
- package/dist/public/app-catalog-square.svg +160 -0
- package/dist/public/app-catalog.png +0 -0
- package/dist/public/app-catalog.svg +198 -0
- package/dist/public/apple-touch-180x180.png +0 -0
- package/dist/public/favicon-app-catalog.ico +0 -0
- package/dist/public/favicon.ico +0 -0
- package/dist/public/robots.txt +2 -0
- package/package.json +159 -0
- package/public/app-catalog-16x16.png +0 -0
- package/public/app-catalog-192x192.png +0 -0
- package/public/app-catalog-32x32.png +0 -0
- package/public/app-catalog-48x48.png +0 -0
- package/public/app-catalog-512x512.png +0 -0
- package/public/app-catalog-square.svg +160 -0
- package/public/app-catalog.png +0 -0
- package/public/app-catalog.svg +198 -0
- package/public/apple-touch-180x180.png +0 -0
- package/public/favicon-app-catalog.ico +0 -0
- package/public/favicon.ico +0 -0
- package/public/robots.txt +2 -0
- package/src/App.tsx +28 -0
- package/src/__tests__/setupTests.tsx +0 -0
- package/src/api/infra/createQueryClient.ts +29 -0
- package/src/api/infra/trpc.ts +6 -0
- package/src/api/unsorted/appCatalogFetcher.ts +32 -0
- package/src/api/unsorted/createCachingFetcher.ts +159 -0
- package/src/appPropsFactory.ts +37 -0
- package/src/assets/app-catalog.svg +198 -0
- package/src/components/IconPickerDialog.tsx +136 -0
- package/src/components/IconPickerField.tsx +88 -0
- package/src/components/ThemeSwitcher.tsx +22 -0
- package/src/components/theme-provider.tsx +8 -0
- package/src/errors/AuthorizationError.ts +32 -0
- package/src/errors/index.ts +1 -0
- package/src/index.css +225 -0
- package/src/index.tsx +3 -0
- package/src/lib/utils.ts +7 -0
- package/src/main.tsx +57 -0
- package/src/modules/admin-base/components/AdminChat.tsx +122 -0
- package/src/modules/admin-base/components/AdminLayout.tsx +111 -0
- package/src/modules/admin-base/components/AdminWelcome.tsx +52 -0
- package/src/modules/admin-base/context/AdminConfigContext.tsx +36 -0
- package/src/modules/admin-base/index.ts +16 -0
- package/src/modules/admin-base/types/adminTypes.ts +11 -0
- package/src/modules/appCatalog/AppCatalogAdminPage.tsx +274 -0
- package/src/modules/appCatalog/ScreenshotItem.tsx +59 -0
- package/src/modules/appCatalog/ScreenshotManager.tsx +193 -0
- package/src/modules/appCatalog/api/ApiQueryMagazineAppCatalog.ts +12 -0
- package/src/modules/appCatalog/catalogRouteLoader.ts +28 -0
- package/src/modules/appCatalog/context/AppCatalogContext.tsx +52 -0
- package/src/modules/appCatalog/index.ts +16 -0
- package/src/modules/appCatalog/routeLoader.ts +9 -0
- package/src/modules/appCatalog/ui/components/AccessRequestSection.tsx +370 -0
- package/src/modules/appCatalog/ui/components/AppDetailModal.tsx +355 -0
- package/src/modules/appCatalog/ui/components/ApproverDisplay.tsx +260 -0
- package/src/modules/appCatalog/ui/components/GroupingColumn.tsx +65 -0
- package/src/modules/appCatalog/ui/components/GroupingTabs.tsx +41 -0
- package/src/modules/appCatalog/ui/components/ScreenshotGallery.tsx +51 -0
- package/src/modules/appCatalog/ui/grid/AppCatalogFiltersCard.tsx +79 -0
- package/src/modules/appCatalog/ui/grid/AppCatalogGrid.tsx +487 -0
- package/src/modules/appCatalog/ui/grid/AppCatalogTable.tsx +89 -0
- package/src/modules/appCatalog/ui/grid/appCatalogUtils.ts +5 -0
- package/src/modules/appCatalog/ui/hooks/useKeyboardNavigation.ts +54 -0
- package/src/modules/appCatalog/ui/layout/AppCatalogLayout.tsx +57 -0
- package/src/modules/appCatalog/ui/pages/AppCatalogPage.tsx +142 -0
- package/src/modules/approvalMethod/AccessRequestFormFields.tsx +393 -0
- package/src/modules/approvalMethod/ApprovalMethodForm.tsx +323 -0
- package/src/modules/approvalMethod/ApprovalMethodSelector.tsx +150 -0
- package/src/modules/approvalMethod/api/ApiQueryMagazineApprovalMethod.ts +34 -0
- package/src/modules/auth/AuthContext.tsx +130 -0
- package/src/modules/auth/AuthModalContext.tsx +49 -0
- package/src/modules/auth/ProtectedContent.tsx +55 -0
- package/src/modules/auth/authClient.ts +9 -0
- package/src/modules/auth/authUtils.ts +41 -0
- package/src/modules/auth/index.ts +6 -0
- package/src/modules/auth/ui/LoginModal.tsx +36 -0
- package/src/modules/auth/ui/LoginPage.tsx +62 -0
- package/src/modules/auth/useAuthActions.ts +85 -0
- package/src/modules/config/GlobalConfigContext.tsx +42 -0
- package/src/modules/config/HealthStateContext.tsx +47 -0
- package/src/modules/gallery/Gallery.tsx +317 -0
- package/src/modules/icons/IconManagementPage.tsx +245 -0
- package/src/modules/pluginCore/PluginManagerContext.tsx +78 -0
- package/src/modules/pluginCore/makePluginManagerContext.ts +13 -0
- package/src/modules/pluginCore/types.ts +27 -0
- package/src/plugins/builtin/pageUrl/pageUrlTypes.ts +16 -0
- package/src/routeTree.gen.ts +361 -0
- package/src/routes/__root.tsx +23 -0
- package/src/routes/_layout/catalog.apps.index.tsx +28 -0
- package/src/routes/_layout/index.tsx +31 -0
- package/src/routes/_layout/login.tsx +20 -0
- package/src/routes/_layout.tsx +9 -0
- package/src/routes/admin/app-for-catalog/$id.tsx +571 -0
- package/src/routes/admin/app-for-catalog/index.tsx +19 -0
- package/src/routes/admin/app-for-catalog.tsx +12 -0
- package/src/routes/admin/approval-methods/index.tsx +161 -0
- package/src/routes/admin/approval-methods.tsx +10 -0
- package/src/routes/admin/chat.tsx +13 -0
- package/src/routes/admin/icons.tsx +22 -0
- package/src/routes/admin/index.tsx +9 -0
- package/src/routes/admin.tsx +60 -0
- package/src/routes/auth.callback.tsx +74 -0
- package/src/types/slateTypes.ts +22 -0
- package/src/types/table.ts +9 -0
- package/src/types/tanstackQuery.ts +16 -0
- package/src/types/types.ts +30 -0
- package/src/types/userBehaviourTypes.ts +100 -0
- package/src/types/utilityTypes.ts +1 -0
- package/src/types/vite-env.d.ts +2 -0
- package/src/ui/accordion.tsx +64 -0
- package/src/ui/alert-dialog.tsx +155 -0
- package/src/ui/autocomplete.tsx +272 -0
- package/src/ui/badge.tsx +38 -0
- package/src/ui/breadcrumb.tsx +106 -0
- package/src/ui/button-group.tsx +85 -0
- package/src/ui/button.tsx +66 -0
- package/src/ui/card.tsx +92 -0
- package/src/ui/checkbox.tsx +30 -0
- package/src/ui/collapsible.tsx +45 -0
- package/src/ui/command.tsx +196 -0
- package/src/ui/components/ActionCard.tsx +30 -0
- package/src/ui/components/AppIcon.tsx +48 -0
- package/src/ui/components/Breadcrumbs.tsx +97 -0
- package/src/ui/components/commandInput/EhBaseSelector.tsx +17 -0
- package/src/ui/components/commandInput/types.ts +22 -0
- package/src/ui/components/error/BaseErrorPage.tsx +26 -0
- package/src/ui/components/error/DefaultErrorComponent.tsx +107 -0
- package/src/ui/components/error/ForbiddenErrorPage.tsx +89 -0
- package/src/ui/components/error/RootErrorPage.tsx +23 -0
- package/src/ui/components/footer/Footer.tsx +51 -0
- package/src/ui/components/header/Header.tsx +135 -0
- package/src/ui/components/widgetPanel/AddWidgetCard.tsx +17 -0
- package/src/ui/components/widgetPanel/WidgetGrid.tsx +18 -0
- package/src/ui/components/widgets/CredentialsWidget.tsx +55 -0
- package/src/ui/components/widgets/VersionWidget.tsx +29 -0
- package/src/ui/crud-list/CrudList.tsx +274 -0
- package/src/ui/crud-list/index.ts +2 -0
- package/src/ui/crud-list/types.ts +44 -0
- package/src/ui/dialog.tsx +141 -0
- package/src/ui/dropdown-menu.tsx +255 -0
- package/src/ui/editable-list/EditableListField.tsx +188 -0
- package/src/ui/editable-list/index.ts +2 -0
- package/src/ui/editable-list/types.ts +18 -0
- package/src/ui/empty.tsx +105 -0
- package/src/ui/error/NotFoundError.tsx +16 -0
- package/src/ui/form.tsx +188 -0
- package/src/ui/input-group.tsx +167 -0
- package/src/ui/input.tsx +21 -0
- package/src/ui/item.tsx +194 -0
- package/src/ui/label.tsx +22 -0
- package/src/ui/layout/Footer.tsx +16 -0
- package/src/ui/layout/LoadingScreen.tsx +67 -0
- package/src/ui/layout/MainLayout.tsx +28 -0
- package/src/ui/layout/SideColumn.tsx +3 -0
- package/src/ui/layout/TopLevelProviders.tsx +63 -0
- package/src/ui/layout/TopLevelProvidersForErrors.tsx +34 -0
- package/src/ui/link.tsx +33 -0
- package/src/ui/linkExternal.tsx +26 -0
- package/src/ui/main/JumpTabContent.tsx +12 -0
- package/src/ui/main/Tabs.tsx +29 -0
- package/src/ui/markdown-editor/MarkdownEditor.tsx +132 -0
- package/src/ui/markdown-editor/MarkdownToolbar.tsx +90 -0
- package/src/ui/markdown-editor/index.ts +2 -0
- package/src/ui/popover.tsx +48 -0
- package/src/ui/radio-group.tsx +43 -0
- package/src/ui/resizable.tsx +51 -0
- package/src/ui/scroll-area.tsx +58 -0
- package/src/ui/search-input-with-shortcut.tsx +54 -0
- package/src/ui/select.tsx +185 -0
- package/src/ui/separator.tsx +26 -0
- package/src/ui/shortcut-button.tsx +84 -0
- package/src/ui/skeleton.tsx +13 -0
- package/src/ui/spinner.tsx +16 -0
- package/src/ui/table.tsx +114 -0
- package/src/ui/tabs.tsx +64 -0
- package/src/ui/textarea.tsx +18 -0
- package/src/ui/tooltip.tsx +60 -0
- package/src/userDb/AcDb.ts +29 -0
- package/src/userDb/DbContext.tsx +22 -0
- package/src/util/createEhRouter.tsx +22 -0
- package/src/util/error-utils.ts +31 -0
- package/src/util/highlightMatches.tsx +29 -0
- package/src/util/reactQueryUtils.ts +20 -0
- 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,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":"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"}
|
|
@@ -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 {};
|