@cobaltcore-dev/aurora 0.3.0 → 0.3.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 (111) hide show
  1. package/dist/client/{ContentHeader-B_PWrxbw.mjs → ContentHeader-BXZoN3B9.mjs} +16 -16
  2. package/dist/client/{ContentHeader-B_PWrxbw.mjs.map → ContentHeader-BXZoN3B9.mjs.map} +1 -1
  3. package/dist/client/{DeleteFlavorModal-BkPrQKyE.mjs → DeleteFlavorModal-BusYn32r.mjs} +152 -152
  4. package/dist/client/{DeleteFlavorModal-BkPrQKyE.mjs.map → DeleteFlavorModal-BusYn32r.mjs.map} +1 -1
  5. package/dist/client/{EditSecurityGroupModal-ad4uzlt0.mjs → EditSecurityGroupModal-Dl6m7wUe.mjs} +15 -15
  6. package/dist/client/{EditSecurityGroupModal-ad4uzlt0.mjs.map → EditSecurityGroupModal-Dl6m7wUe.mjs.map} +1 -1
  7. package/dist/client/FiltersInput-BgNaHFBt.mjs +81 -0
  8. package/dist/client/FiltersInput-BgNaHFBt.mjs.map +1 -0
  9. package/dist/client/{FloatingIpActionModals-lKMwLuL8.mjs → FloatingIpActionModals-By2hXR9m.mjs} +50 -50
  10. package/dist/client/{FloatingIpActionModals-lKMwLuL8.mjs.map → FloatingIpActionModals-By2hXR9m.mjs.map} +1 -1
  11. package/dist/client/{ImageToastNotifications--U13YiQ_.mjs → ImageToastNotifications-fHI8jB2j.mjs} +321 -321
  12. package/dist/client/{ImageToastNotifications--U13YiQ_.mjs.map → ImageToastNotifications-fHI8jB2j.mjs.map} +1 -1
  13. package/dist/client/ListToolbar-BvtCo8dk.mjs +129 -0
  14. package/dist/client/ListToolbar-BvtCo8dk.mjs.map +1 -0
  15. package/dist/client/{RouteError-iP1Vd6bY.mjs → RouteError-pDEWC_k7.mjs} +2 -2
  16. package/dist/client/{RouteError-iP1Vd6bY.mjs.map → RouteError-pDEWC_k7.mjs.map} +1 -1
  17. package/dist/client/SortInput-DXWSqSny.mjs +34 -0
  18. package/dist/client/SortInput-DXWSqSny.mjs.map +1 -0
  19. package/dist/client/{_flavorId-BaNXUJhA.mjs → _flavorId-BRonXvCo.mjs} +40 -40
  20. package/dist/client/{_flavorId-BaNXUJhA.mjs.map → _flavorId-BRonXvCo.mjs.map} +1 -1
  21. package/dist/client/{_flavorId-CnO76tuy.mjs → _flavorId-BoNcxYmF.mjs} +9 -9
  22. package/dist/client/{_flavorId-CnO76tuy.mjs.map → _flavorId-BoNcxYmF.mjs.map} +1 -1
  23. package/dist/client/_floatingIpId-BpUfL8Im2.mjs +228 -0
  24. package/dist/client/_floatingIpId-BpUfL8Im2.mjs.map +1 -0
  25. package/dist/client/{_floatingIpId-C8G20nNt.mjs → _floatingIpId-CwHiXazi.mjs} +2 -2
  26. package/dist/client/{_floatingIpId-C8G20nNt.mjs.map → _floatingIpId-CwHiXazi.mjs.map} +1 -1
  27. package/dist/client/{_imageId-DdSbxFqG.mjs → _imageId-CdOOJjw0.mjs} +127 -127
  28. package/dist/client/{_imageId-DdSbxFqG.mjs.map → _imageId-CdOOJjw0.mjs.map} +1 -1
  29. package/dist/client/{_pcaId-DFkYJEb5.mjs → _pcaId-CwlH1Kvl.mjs} +82 -82
  30. package/dist/client/{_pcaId-DFkYJEb5.mjs.map → _pcaId-CwlH1Kvl.mjs.map} +1 -1
  31. package/dist/client/{_pcaId-DX_S-3hE.mjs → _pcaId-D1ZEaCdp.mjs} +2 -2
  32. package/dist/client/{_pcaId-DX_S-3hE.mjs.map → _pcaId-D1ZEaCdp.mjs.map} +1 -1
  33. package/dist/client/{_projectId-DsSVV2EA.mjs → _projectId-D1gGribM.mjs} +69 -69
  34. package/dist/client/{_projectId-DsSVV2EA.mjs.map → _projectId-D1gGribM.mjs.map} +1 -1
  35. package/dist/client/{_projectId-DYrcZ3E3.mjs → _projectId-DhLpIalx.mjs} +3 -3
  36. package/dist/client/{_projectId-DYrcZ3E3.mjs.map → _projectId-DhLpIalx.mjs.map} +1 -1
  37. package/dist/client/{_projectId-Dha4XqX4.mjs → _projectId-Dj_InfSc.mjs} +3 -3
  38. package/dist/client/{_projectId-Dha4XqX4.mjs.map → _projectId-Dj_InfSc.mjs.map} +1 -1
  39. package/dist/client/{_projectId-KH5si25Q.mjs → _projectId-OW2xkK43.mjs} +2 -2
  40. package/dist/client/{_projectId-KH5si25Q.mjs.map → _projectId-OW2xkK43.mjs.map} +1 -1
  41. package/dist/client/{_securityGroupId-DgaSqYex.mjs → _securityGroupId-B0llWH9A.mjs} +2 -2
  42. package/dist/client/{_securityGroupId-DgaSqYex.mjs.map → _securityGroupId-B0llWH9A.mjs.map} +1 -1
  43. package/dist/client/{_securityGroupId-O7FXfx0M.mjs → _securityGroupId-gbUnd5Wv.mjs} +306 -306
  44. package/dist/client/{_securityGroupId-O7FXfx0M.mjs.map → _securityGroupId-gbUnd5Wv.mjs.map} +1 -1
  45. package/dist/client/{about-DN8n8sN8.mjs → about-DCe6LsKz.mjs} +2 -2
  46. package/dist/client/{about-DN8n8sN8.mjs.map → about-DCe6LsKz.mjs.map} +1 -1
  47. package/dist/client/{build-CZRvXrAL.mjs → build-BJDfnAyi.mjs} +541 -537
  48. package/dist/client/{build-CZRvXrAL.mjs.map → build-BJDfnAyi.mjs.map} +1 -1
  49. package/dist/client/{buildFilterParams-ngVK3ybs.mjs → buildFilterParams-By33pG59.mjs} +1 -1
  50. package/dist/client/{buildFilterParams-ngVK3ybs.mjs.map → buildFilterParams-By33pG59.mjs.map} +1 -1
  51. package/dist/client/{constants-akdIBeTX.mjs → constants-CAjjRTo_.mjs} +9 -9
  52. package/dist/client/{constants-akdIBeTX.mjs.map → constants-CAjjRTo_.mjs.map} +1 -1
  53. package/dist/client/{containers-BE2QiLBs.mjs → containers-BuXUVb1N.mjs} +702 -702
  54. package/dist/client/{containers-BE2QiLBs.mjs.map → containers-BuXUVb1N.mjs.map} +1 -1
  55. package/dist/client/{containers-Cs5vOeR2.mjs → containers-Ca5V1EBS.mjs} +1 -1
  56. package/dist/client/{containers-Cs5vOeR2.mjs.map → containers-Ca5V1EBS.mjs.map} +1 -1
  57. package/dist/client/{containers-DmwhE9Uz.mjs → containers-NW7RnHTI.mjs} +2 -2
  58. package/dist/client/{containers-DmwhE9Uz.mjs.map → containers-NW7RnHTI.mjs.map} +1 -1
  59. package/dist/client/{flavors-CY7A6--v.mjs → flavors-D8oElC2K.mjs} +2 -2
  60. package/dist/client/{flavors-CY7A6--v.mjs.map → flavors-D8oElC2K.mjs.map} +1 -1
  61. package/dist/client/flavors-qvgPSI7J.mjs +613 -0
  62. package/dist/client/flavors-qvgPSI7J.mjs.map +1 -0
  63. package/dist/client/{floatingips-BtL4d4m-.mjs → floatingips-Be3zLoaD.mjs} +85 -85
  64. package/dist/client/{floatingips-BtL4d4m-.mjs.map → floatingips-Be3zLoaD.mjs.map} +1 -1
  65. package/dist/client/{images-I9gQfRa7.mjs → images-BiEBENaj.mjs} +2 -2
  66. package/dist/client/{images-I9gQfRa7.mjs.map → images-BiEBENaj.mjs.map} +1 -1
  67. package/dist/client/images-CCYBAphP2.mjs +1900 -0
  68. package/dist/client/images-CCYBAphP2.mjs.map +1 -0
  69. package/dist/client/index.js +109 -109
  70. package/dist/client/index.js.map +1 -1
  71. package/dist/client/{network-SCVadZsv.mjs → network-nbSbl0X0.mjs} +1 -1
  72. package/dist/client/{network-SCVadZsv.mjs.map → network-nbSbl0X0.mjs.map} +1 -1
  73. package/dist/client/{objects-BrYe_RaJ.mjs → objects-CU5ws07o.mjs} +2 -2
  74. package/dist/client/{objects-BrYe_RaJ.mjs.map → objects-CU5ws07o.mjs.map} +1 -1
  75. package/dist/client/{objects-DOYFFn3Y.mjs → objects-FXN0VWLI.mjs} +1791 -1791
  76. package/dist/client/{objects-DOYFFn3Y.mjs.map → objects-FXN0VWLI.mjs.map} +1 -1
  77. package/dist/client/{objects-B4yrYf_a.mjs → objects-GmuIOaHd.mjs} +1 -1
  78. package/dist/client/{objects-B4yrYf_a.mjs.map → objects-GmuIOaHd.mjs.map} +1 -1
  79. package/dist/client/{overview-DTLIAKkJ.mjs → overview-B3gdnWTG.mjs} +2 -2
  80. package/dist/client/{overview-DTLIAKkJ.mjs.map → overview-B3gdnWTG.mjs.map} +1 -1
  81. package/dist/client/{overview-BjRSFSBh.mjs → overview-DzYBiNfD.mjs} +2 -2
  82. package/dist/client/{overview-BjRSFSBh.mjs.map → overview-DzYBiNfD.mjs.map} +1 -1
  83. package/dist/client/{overview-CmQkJ4Hh.mjs → overview-EhfPY8Je.mjs} +2 -2
  84. package/dist/client/{overview-CmQkJ4Hh.mjs.map → overview-EhfPY8Je.mjs.map} +1 -1
  85. package/dist/client/{overview-C4gjtc2q.mjs → overview-XueZI4LQ.mjs} +3 -3
  86. package/dist/client/{overview-C4gjtc2q.mjs.map → overview-XueZI4LQ.mjs.map} +1 -1
  87. package/dist/client/{pca-DzixU9Dl.mjs → pca-DSM71LhW.mjs} +2 -2
  88. package/dist/client/{pca-DzixU9Dl.mjs.map → pca-DSM71LhW.mjs.map} +1 -1
  89. package/dist/client/{pca-CYFJxSZ2.mjs → pca-x9if8xU-.mjs} +51 -51
  90. package/dist/client/{pca-CYFJxSZ2.mjs.map → pca-x9if8xU-.mjs.map} +1 -1
  91. package/dist/client/{projects-DwVawmll.mjs → projects-B6BPo2Ar.mjs} +2 -2
  92. package/dist/client/{projects-DwVawmll.mjs.map → projects-B6BPo2Ar.mjs.map} +1 -1
  93. package/dist/client/{projects-BsN4bvU2.mjs → projects-BilrmHLu.mjs} +1 -1
  94. package/dist/client/{projects-BsN4bvU2.mjs.map → projects-BilrmHLu.mjs.map} +1 -1
  95. package/dist/client/{projects-BtyjXGq2.mjs → projects-Bt0XptpG.mjs} +2 -2
  96. package/dist/client/{projects-BtyjXGq2.mjs.map → projects-Bt0XptpG.mjs.map} +1 -1
  97. package/dist/client/{projects-D3hOC1cy.mjs → projects-CnmZIB2Q.mjs} +25 -25
  98. package/dist/client/{projects-D3hOC1cy.mjs.map → projects-CnmZIB2Q.mjs.map} +1 -1
  99. package/dist/client/{securitygroups-DMCIDHQS.mjs → securitygroups-BdzieS7Z.mjs} +101 -101
  100. package/dist/client/{securitygroups-DMCIDHQS.mjs.map → securitygroups-BdzieS7Z.mjs.map} +1 -1
  101. package/dist/client/{useListWithFiltering-CEDh1LO-.mjs → useListWithFiltering-CqQbAjEe.mjs} +1 -1
  102. package/dist/client/{useListWithFiltering-CEDh1LO-.mjs.map → useListWithFiltering-CqQbAjEe.mjs.map} +1 -1
  103. package/package.json +2 -2
  104. package/dist/client/ListToolbar-C5lzTrit.mjs +0 -223
  105. package/dist/client/ListToolbar-C5lzTrit.mjs.map +0 -1
  106. package/dist/client/_floatingIpId-DdKnjdIV.mjs +0 -228
  107. package/dist/client/_floatingIpId-DdKnjdIV.mjs.map +0 -1
  108. package/dist/client/flavors-pEcGkCut.mjs +0 -565
  109. package/dist/client/flavors-pEcGkCut.mjs.map +0 -1
  110. package/dist/client/images-C19gCFSy.mjs +0 -1797
  111. package/dist/client/images-C19gCFSy.mjs.map +0 -1
@@ -1 +1 @@
1
- {"version":3,"file":"_pcaId-DX_S-3hE.mjs","names":["createFileRoute","redirect","getServiceIndex","Route","staticData","section","service","isDetail","sectionCrumb","labelKey","crumb","to","RouteInfo","loader","context","params","pca","trpcClient","services","getById","query","project_id","projectId","certificate_authority_id","pcaId","pcaTitle","configuration","subject","common_name","id","head","loaderData","meta","title","component","lazyRouteComponent","$$splitComponentImporter","beforeLoad","availableServices","auth","getAvailableServices","serviceIndex","pcaServices"],"sources":["../../src/client/routes/_auth/projects/$projectId/services/pca/$pcaId/index.tsx"],"sourcesContent":["import { createFileRoute, redirect, useNavigate } from \"@tanstack/react-router\"\nimport { Trans } from \"@lingui/react/macro\"\nimport { Button, Spinner, Stack } from \"@cloudoperators/juno-ui-components/index\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport type { RouteInfo } from \"@/client/routes/routeInfo\"\nimport { trpcReact } from \"@/client/trpcClient\"\nimport { useProjectId } from \"@/client/hooks/useProjectId\"\nimport { PcaDetailsView } from \"./-components/PcaDetailsView\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId/services/pca/$pcaId/\")({\n staticData: {\n section: \"services\",\n service: \"pca\",\n isDetail: true,\n sectionCrumb: { labelKey: \"Services\" },\n crumb: { labelKey: \"PCA (Clavis)\", to: \"/projects/$projectId/services/pca\" },\n } satisfies RouteInfo,\n loader: async ({ context, params }) => {\n const pca = await context.trpcClient?.services.pca.getById.query({\n project_id: params.projectId,\n certificate_authority_id: params.pcaId,\n })\n return { pcaTitle: pca?.configuration?.subject?.common_name || pca?.id || null }\n },\n head: ({ loaderData }) => ({\n meta: [{ title: loaderData?.pcaTitle ?? \"Certificate Authority\" }],\n }),\n component: RouteComponent,\n beforeLoad: async ({ context, params }) => {\n const { trpcClient } = context\n const availableServices = (await trpcClient?.auth.getAvailableServices.query()) || []\n const serviceIndex = getServiceIndex(availableServices)\n // temporary as clavis is not fully GA, after GA replace with [\"pca\"]?.[\"clavis\"]\n const pcaServices = serviceIndex[\"pca\"]?.[\"clavis-beta\"] || serviceIndex[\"pca\"]?.[\"clavis-dev\"]\n\n // Redirect if clavis service not available\n if (!pcaServices) {\n throw redirect({\n to: \"/projects/$projectId/services/pca\",\n params: { projectId: params.projectId },\n })\n }\n },\n})\n\nfunction RouteComponent() {\n const navigate = useNavigate()\n const projectId = useProjectId()\n const { pcaId } = Route.useParams()\n\n const {\n isLoading,\n isError,\n error,\n data: pca,\n } = trpcReact.services.pca.getById.useQuery({\n project_id: projectId,\n certificate_authority_id: pcaId,\n })\n\n // Loading state\n if (isLoading) {\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\">\n <Spinner variant=\"primary\" size=\"large\" className=\"mb-2\" />\n <Trans>Loading Certificate Authority Details...</Trans>\n </Stack>\n )\n }\n\n const handleBack = () =>\n navigate({\n to: \"/projects/$projectId/services/pca\",\n params: { projectId },\n })\n\n // Error state\n if (isError) {\n const errorMessage = error?.message || \"Unknown error\"\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\" gap=\"5\">\n <p className=\"text-theme-error font-semibold\">\n <Trans>Error loading Certificate Authority</Trans>\n </p>\n <p className=\"text-theme-highest\">{errorMessage}</p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Certificate Authorities</Trans>\n </Button>\n </Stack>\n )\n }\n\n // No data state\n if (!pca) {\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\" gap=\"5\">\n <p className=\"text-theme-secondary\">\n <Trans>Certificate Authority not found</Trans>\n </p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Certificate Authorities</Trans>\n </Button>\n </Stack>\n )\n }\n\n return <PcaDetailsView pca={pca} />\n}\n"],"mappings":";;AASA,IAAaG,IAAQH,EAAgB,kDAAA,CAAmD;CACtFI,YAAY;EACVC,SAAS;EACTC,SAAS;EACTC,UAAU;EACVC,cAAc,EAAEC,UAAU,YAAW;EACrCC,OAAO;GAAED,UAAU;GAAgBE,IAAI;GAAoC;EAC7E;CACAE,QAAQ,OAAO,EAAEC,YAASC,gBAAQ;EAChC,IAAMC,IAAM,MAAMF,EAAQG,YAAYC,SAASF,IAAIG,QAAQC,MAAM;GAC/DC,YAAYN,EAAOO;GACnBC,0BAA0BR,EAAOS;GACnC,CAAA;AACA,SAAO,EAAEC,UAAUT,GAAKU,eAAeC,SAASC,eAAeZ,GAAKa,MAAM,MAAK;;CAEjFC,OAAO,EAAEC,qBAAkB,EACzBC,MAAM,CAAC,EAAEC,OAAOF,GAAYN,YAAY,yBAAwB,CAAC,EACnE;CACAS,WAASC,yCAAA,YAAA;CACTE,YAAY,OAAO,EAAEvB,YAASC,gBAAQ;EACpC,IAAM,EAAEE,kBAAeH,GAEjB2B,IAAevC,EADK,MAAOe,GAAYsB,KAAKC,qBAAqBpB,OAAAA,IAAY,EAAE,CAChDkB;AAKrC,MAAI,EAHgBG,EAAa,MAAS,kBAAkBA,EAAa,MAAS,eAIhF,OAAMxC,EAAS;GACbU,IAAI;GACJI,QAAQ,EAAEO,WAAWP,EAAOO,WAAU;GACxC,CAAA;;CAGN,CAAA"}
1
+ {"version":3,"file":"_pcaId-D1ZEaCdp.mjs","names":["createFileRoute","redirect","getServiceIndex","Route","staticData","section","service","isDetail","sectionCrumb","labelKey","crumb","to","RouteInfo","loader","context","params","pca","trpcClient","services","getById","query","project_id","projectId","certificate_authority_id","pcaId","pcaTitle","configuration","subject","common_name","id","head","loaderData","meta","title","component","lazyRouteComponent","$$splitComponentImporter","beforeLoad","availableServices","auth","getAvailableServices","serviceIndex","pcaServices"],"sources":["../../src/client/routes/_auth/projects/$projectId/services/pca/$pcaId/index.tsx"],"sourcesContent":["import { createFileRoute, redirect, useNavigate } from \"@tanstack/react-router\"\nimport { Trans } from \"@lingui/react/macro\"\nimport { Button, Spinner, Stack } from \"@cloudoperators/juno-ui-components/index\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport type { RouteInfo } from \"@/client/routes/routeInfo\"\nimport { trpcReact } from \"@/client/trpcClient\"\nimport { useProjectId } from \"@/client/hooks/useProjectId\"\nimport { PcaDetailsView } from \"./-components/PcaDetailsView\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId/services/pca/$pcaId/\")({\n staticData: {\n section: \"services\",\n service: \"pca\",\n isDetail: true,\n sectionCrumb: { labelKey: \"Services\" },\n crumb: { labelKey: \"PCA (Clavis)\", to: \"/projects/$projectId/services/pca\" },\n } satisfies RouteInfo,\n loader: async ({ context, params }) => {\n const pca = await context.trpcClient?.services.pca.getById.query({\n project_id: params.projectId,\n certificate_authority_id: params.pcaId,\n })\n return { pcaTitle: pca?.configuration?.subject?.common_name || pca?.id || null }\n },\n head: ({ loaderData }) => ({\n meta: [{ title: loaderData?.pcaTitle ?? \"Certificate Authority\" }],\n }),\n component: RouteComponent,\n beforeLoad: async ({ context, params }) => {\n const { trpcClient } = context\n const availableServices = (await trpcClient?.auth.getAvailableServices.query()) || []\n const serviceIndex = getServiceIndex(availableServices)\n // temporary as clavis is not fully GA, after GA replace with [\"pca\"]?.[\"clavis\"]\n const pcaServices = serviceIndex[\"pca\"]?.[\"clavis-beta\"] || serviceIndex[\"pca\"]?.[\"clavis-dev\"]\n\n // Redirect if clavis service not available\n if (!pcaServices) {\n throw redirect({\n to: \"/projects/$projectId/services/pca\",\n params: { projectId: params.projectId },\n })\n }\n },\n})\n\nfunction RouteComponent() {\n const navigate = useNavigate()\n const projectId = useProjectId()\n const { pcaId } = Route.useParams()\n\n const {\n isLoading,\n isError,\n error,\n data: pca,\n } = trpcReact.services.pca.getById.useQuery({\n project_id: projectId,\n certificate_authority_id: pcaId,\n })\n\n // Loading state\n if (isLoading) {\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\">\n <Spinner variant=\"primary\" size=\"large\" className=\"mb-2\" />\n <Trans>Loading Certificate Authority Details...</Trans>\n </Stack>\n )\n }\n\n const handleBack = () =>\n navigate({\n to: \"/projects/$projectId/services/pca\",\n params: { projectId },\n })\n\n // Error state\n if (isError) {\n const errorMessage = error?.message || \"Unknown error\"\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\" gap=\"5\">\n <p className=\"text-theme-error font-semibold\">\n <Trans>Error loading Certificate Authority</Trans>\n </p>\n <p className=\"text-theme-highest\">{errorMessage}</p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Certificate Authorities</Trans>\n </Button>\n </Stack>\n )\n }\n\n // No data state\n if (!pca) {\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\" gap=\"5\">\n <p className=\"text-theme-secondary\">\n <Trans>Certificate Authority not found</Trans>\n </p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Certificate Authorities</Trans>\n </Button>\n </Stack>\n )\n }\n\n return <PcaDetailsView pca={pca} />\n}\n"],"mappings":";;AASA,IAAaG,IAAQH,EAAgB,kDAAA,CAAmD;CACtFI,YAAY;EACVC,SAAS;EACTC,SAAS;EACTC,UAAU;EACVC,cAAc,EAAEC,UAAU,YAAW;EACrCC,OAAO;GAAED,UAAU;GAAgBE,IAAI;GAAoC;EAC7E;CACAE,QAAQ,OAAO,EAAEC,YAASC,gBAAQ;EAChC,IAAMC,IAAM,MAAMF,EAAQG,YAAYC,SAASF,IAAIG,QAAQC,MAAM;GAC/DC,YAAYN,EAAOO;GACnBC,0BAA0BR,EAAOS;GACnC,CAAA;AACA,SAAO,EAAEC,UAAUT,GAAKU,eAAeC,SAASC,eAAeZ,GAAKa,MAAM,MAAK;;CAEjFC,OAAO,EAAEC,qBAAkB,EACzBC,MAAM,CAAC,EAAEC,OAAOF,GAAYN,YAAY,yBAAwB,CAAC,EACnE;CACAS,WAASC,yCAAA,YAAA;CACTE,YAAY,OAAO,EAAEvB,YAASC,gBAAQ;EACpC,IAAM,EAAEE,kBAAeH,GAEjB2B,IAAevC,EADK,MAAOe,GAAYsB,KAAKC,qBAAqBpB,OAAAA,IAAY,EAAE,CAChDkB;AAKrC,MAAI,EAHgBG,EAAa,MAAS,kBAAkBA,EAAa,MAAS,eAIhF,OAAMxC,EAAS;GACbU,IAAI;GACJI,QAAQ,EAAEO,WAAWP,EAAOO,WAAU;GACxC,CAAA;;CAGN,CAAA"}
@@ -1,5 +1,5 @@
1
- import { B as e, E as t, G as n, O as r, R as i, Y as a, g as o, tt as s } from "./build-CZRvXrAL.mjs";
2
- import { t as c } from "./_projectId-Dha4XqX4.mjs";
1
+ import { D as e, K as t, V as n, X as r, g as i, k as a, nt as o, z as s } from "./build-BJDfnAyi.mjs";
2
+ import { t as c } from "./_projectId-Dj_InfSc.mjs";
3
3
  import { t as l } from "./helpers--JWXi40U.mjs";
4
4
  import { Fragment as u, jsx as d, jsxs as f } from "react/jsx-runtime";
5
5
  import { useEffect as p, useMemo as m, useRef as h, useState as g } from "react";
@@ -49,8 +49,8 @@ function A({ component: e }) {
49
49
  }
50
50
  //#endregion
51
51
  //#region src/client/routes/_auth/projects/-components/SideNavBar.tsx
52
- var j = ({ projectId: t, projectName: i, availableServices: a }) => {
53
- let { i18n: o, _: s } = w(), c = x(), m = b(), { provider: h } = S({ strict: !1 }), { slots: _ } = C({ strict: !1 }), v = [...m].reverse().find((e) => O(e.staticData)), y = v && O(v.staticData) ? v.staticData : void 0, T = y?.section ?? null, E = y?.service ?? null, D = l(a), [k, j] = g({
52
+ var j = ({ projectId: e, projectName: r, availableServices: i }) => {
53
+ let { i18n: o, _: s } = w(), c = x(), m = b(), { provider: h } = S({ strict: !1 }), { slots: _ } = C({ strict: !1 }), v = [...m].reverse().find((e) => O(e.staticData)), y = v && O(v.staticData) ? v.staticData : void 0, T = y?.section ?? null, E = y?.service ?? null, D = l(i), [k, j] = g({
54
54
  compute: !0,
55
55
  network: !0,
56
56
  storage: !0,
@@ -80,28 +80,28 @@ var j = ({ projectId: t, projectName: i, availableServices: a }) => {
80
80
  service: "images",
81
81
  label: o._({ id: "an5hVd" }),
82
82
  to: "/projects/$projectId/compute/images",
83
- params: { projectId: t }
83
+ params: { projectId: e }
84
84
  }] : [], ...D?.compute?.nova ? [{
85
85
  service: "flavors",
86
86
  label: o._({ id: "neiJm0" }),
87
87
  to: "/projects/$projectId/compute/flavors",
88
- params: { projectId: t }
88
+ params: { projectId: e }
89
89
  }] : []], R = [...D.network ? [{
90
90
  service: "securitygroups",
91
91
  label: o._({ id: "4opp4r" }),
92
92
  to: "/projects/$projectId/network/securitygroups",
93
- params: { projectId: t }
93
+ params: { projectId: e }
94
94
  }, {
95
95
  service: "floatingips",
96
96
  label: o._({ id: "u77/s4" }),
97
97
  to: "/projects/$projectId/network/floatingips",
98
- params: { projectId: t }
98
+ params: { projectId: e }
99
99
  }] : []], z = [...D?.["object-store"]?.swift ? [{
100
100
  service: "containers",
101
101
  label: o._({ id: "+OEi73" }),
102
102
  to: "/projects/$projectId/storage/$provider/containers",
103
103
  params: {
104
- projectId: t,
104
+ projectId: e,
105
105
  provider: "swift"
106
106
  }
107
107
  }] : [], {
@@ -109,85 +109,85 @@ var j = ({ projectId: t, projectName: i, availableServices: a }) => {
109
109
  label: o._({ id: "KhNDX4" }),
110
110
  to: "/projects/$projectId/storage/$provider/containers",
111
111
  params: {
112
- projectId: t,
112
+ projectId: e,
113
113
  provider: "ceph"
114
114
  }
115
115
  }], B = [...D.pca?.["clavis-beta"] || D.pca?.["clavis-dev"] ? [{
116
116
  service: "pca",
117
117
  label: o._({ id: "miy5mb" }),
118
118
  to: "/projects/$projectId/services/pca",
119
- params: { projectId: t }
119
+ params: { projectId: e }
120
120
  }] : []];
121
- return /* @__PURE__ */ d(n, {
121
+ return /* @__PURE__ */ d(t, {
122
122
  ariaLabel: "Project Side Navigation",
123
- children: /* @__PURE__ */ f(u, { children: [/* @__PURE__ */ d(r, { children: /* @__PURE__ */ f(u, { children: [
124
- /* @__PURE__ */ d(e, {
123
+ children: /* @__PURE__ */ f(u, { children: [/* @__PURE__ */ d(a, { children: /* @__PURE__ */ f(u, { children: [
124
+ /* @__PURE__ */ d(n, {
125
125
  icon: "home",
126
- label: i,
126
+ label: r,
127
127
  onClick: () => c({
128
128
  to: "/projects/$projectId",
129
- params: { projectId: t }
129
+ params: { projectId: e }
130
130
  }),
131
131
  selected: F
132
132
  }),
133
133
  /* @__PURE__ */ d("div", {
134
134
  onClickCapture: I("compute"),
135
- children: /* @__PURE__ */ d(e, {
135
+ children: /* @__PURE__ */ d(n, {
136
136
  label: o._({ id: "rp0Bd0" }),
137
137
  open: k.compute,
138
- children: L.map(({ service: t, label: n, to: r, params: i }) => /* @__PURE__ */ d(e, {
138
+ children: L.map(({ service: e, label: t, to: r, params: i }) => /* @__PURE__ */ d(n, {
139
139
  onClick: () => c({
140
140
  to: r,
141
141
  params: i
142
142
  }),
143
- label: n,
144
- selected: T === "compute" && E === t
145
- }, n))
143
+ label: t,
144
+ selected: T === "compute" && E === e
145
+ }, t))
146
146
  }, N.compute)
147
147
  }),
148
148
  R.length > 0 && /* @__PURE__ */ d("div", {
149
149
  onClickCapture: I("network"),
150
- children: /* @__PURE__ */ d(e, {
150
+ children: /* @__PURE__ */ d(n, {
151
151
  label: o._({ id: "OR475H" }),
152
152
  open: k.network,
153
- children: R.map(({ service: t, label: n, to: r, params: i }) => /* @__PURE__ */ d(e, {
153
+ children: R.map(({ service: e, label: t, to: r, params: i }) => /* @__PURE__ */ d(n, {
154
154
  onClick: () => c({
155
155
  to: r,
156
156
  params: i
157
157
  }),
158
- label: n,
159
- selected: T === "network" && E === t
160
- }, n))
158
+ label: t,
159
+ selected: T === "network" && E === e
160
+ }, t))
161
161
  }, N.network)
162
162
  }),
163
163
  z.length > 0 && /* @__PURE__ */ d("div", {
164
164
  onClickCapture: I("storage"),
165
- children: /* @__PURE__ */ d(e, {
165
+ children: /* @__PURE__ */ d(n, {
166
166
  label: o._({ id: "BrrIs8" }),
167
167
  open: k.storage,
168
- children: z.map(({ service: t, label: n, to: r, params: i }) => /* @__PURE__ */ d(e, {
168
+ children: z.map(({ service: e, label: t, to: r, params: i }) => /* @__PURE__ */ d(n, {
169
169
  onClick: () => c({
170
170
  to: r,
171
171
  params: i
172
172
  }),
173
- label: n,
174
- selected: T === "storage" && E === "containers" ? i.provider === h : E === t
175
- }, n))
173
+ label: t,
174
+ selected: T === "storage" && E === "containers" ? i.provider === h : E === e
175
+ }, t))
176
176
  }, N.storage)
177
177
  }),
178
178
  B.length > 0 && /* @__PURE__ */ d("div", {
179
179
  onClickCapture: I("services"),
180
- children: /* @__PURE__ */ d(e, {
180
+ children: /* @__PURE__ */ d(n, {
181
181
  label: o._({ id: "MILoeL" }),
182
182
  open: k.services,
183
- children: B.map(({ service: t, label: n, to: r, params: i }) => /* @__PURE__ */ d(e, {
183
+ children: B.map(({ service: e, label: t, to: r, params: i }) => /* @__PURE__ */ d(n, {
184
184
  onClick: () => c({
185
185
  to: r,
186
186
  params: i
187
187
  }),
188
- label: n,
189
- selected: T === "services" && E === t
190
- }, n))
188
+ label: t,
189
+ selected: T === "services" && E === e
190
+ }, t))
191
191
  }, N.services)
192
192
  })
193
193
  ] }) }), _?.sideNavBanner && /* @__PURE__ */ d(A, { component: _.sideNavBanner })] })
@@ -195,12 +195,12 @@ var j = ({ projectId: t, projectName: i, availableServices: a }) => {
195
195
  };
196
196
  //#endregion
197
197
  //#region src/client/components/ProjectView/ProjectInfoBox.tsx
198
- function M({ projectInfo: e }) {
199
- let { i18n: n, _: r } = w(), i = x(), o = b(), { projectId: s } = S({ strict: !1 });
200
- return /* @__PURE__ */ d(a, {
198
+ function M({ projectInfo: t }) {
199
+ let { i18n: n, _: i } = w(), a = x(), o = b(), { projectId: s } = S({ strict: !1 });
200
+ return /* @__PURE__ */ d(r, {
201
201
  className: "mt-8 mb-4",
202
202
  children: m(() => {
203
- let t = {
203
+ let e = {
204
204
  Compute: n._({ id: "rp0Bd0" }),
205
205
  Network: n._({ id: "OR475H" }),
206
206
  Storage: n._({ id: "BrrIs8" }),
@@ -210,28 +210,28 @@ function M({ projectInfo: e }) {
210
210
  "Security Groups": n._({ id: "4opp4r" }),
211
211
  "Floating IPs": n._({ id: "u77/s4" }),
212
212
  "PCA (Clavis)": n._({ id: "miy5mb" })
213
- }, r = (e) => e === "swift" ? n._({ id: "+OEi73" }) : e === "ceph" ? n._({ id: "KhNDX4" }) : n._({ id: "BrrIs8" }), a = [];
214
- a.push({
213
+ }, r = (e) => e === "swift" ? n._({ id: "+OEi73" }) : e === "ceph" ? n._({ id: "KhNDX4" }) : n._({ id: "BrrIs8" }), i = [];
214
+ i.push({
215
215
  icon: "home",
216
216
  label: n._({ id: "i0qMbr" }),
217
- onClick: () => i({ to: "/projects" })
218
- }), e.domain?.name && a.push({ label: e.domain.name }), a.push({
219
- label: e.name,
220
- onClick: () => i({
217
+ onClick: () => a({ to: "/projects" })
218
+ }), t.domain?.name && i.push({ label: t.domain.name }), i.push({
219
+ label: t.name,
220
+ onClick: () => a({
221
221
  to: "/projects/$projectId",
222
222
  params: { projectId: s }
223
223
  })
224
224
  });
225
225
  let c = o.filter((e) => e.routeId !== "/_auth/projects/$projectId" && e.routeId.startsWith("/_auth/projects/$projectId")), l = c[c.length - 1];
226
- if (!l) return a;
226
+ if (!l) return i;
227
227
  let u = O(l.staticData) ? l.staticData : void 0;
228
- if (!u) return a;
228
+ if (!u) return i;
229
229
  let d = l.params;
230
230
  if (u.sectionCrumb) {
231
- let { labelKey: e, to: n } = u.sectionCrumb, r = e ? t[e] : void 0, o = !u.crumb;
232
- a.push(n ? {
231
+ let { labelKey: t, to: n } = u.sectionCrumb, r = t ? e[t] : void 0, o = !u.crumb;
232
+ i.push(n ? {
233
233
  label: r,
234
- onClick: () => i({
234
+ onClick: () => a({
235
235
  to: n,
236
236
  params: d
237
237
  })
@@ -241,23 +241,23 @@ function M({ projectInfo: e }) {
241
241
  });
242
242
  }
243
243
  if (u.crumb) {
244
- let { labelKey: e, to: n, useParamAsLabel: o } = u.crumb, s = o ? r(d[o]) : e ? t[e] : void 0;
244
+ let { labelKey: t, to: n, useParamAsLabel: o } = u.crumb, s = o ? r(d[o]) : t ? e[t] : void 0;
245
245
  if (u.isDetail) {
246
- a.push({
246
+ i.push({
247
247
  label: s,
248
- onClick: () => i({
248
+ onClick: () => a({
249
249
  to: n,
250
250
  params: d
251
251
  })
252
252
  });
253
253
  let e = l.meta?.find((e) => e != null && "title" in e)?.title;
254
- e && a.push({
254
+ e && i.push({
255
255
  label: e,
256
256
  active: !0
257
257
  });
258
- } else a.push(n ? {
258
+ } else i.push(n ? {
259
259
  label: s,
260
- onClick: () => i({
260
+ onClick: () => a({
261
261
  to: n,
262
262
  params: d
263
263
  })
@@ -266,18 +266,18 @@ function M({ projectInfo: e }) {
266
266
  active: !0
267
267
  });
268
268
  }
269
- return a;
269
+ return i;
270
270
  }, [
271
271
  o,
272
- e,
272
+ t,
273
273
  s,
274
- i,
275
- r
276
- ]).map((e, n) => /* @__PURE__ */ d(t, {
277
- label: e.label,
278
- icon: e.icon,
279
- onClick: e.onClick,
280
- active: e.active
274
+ a,
275
+ i
276
+ ]).map((t, n) => /* @__PURE__ */ d(e, {
277
+ label: t.label,
278
+ icon: t.icon,
279
+ onClick: t.onClick,
280
+ active: t.active
281
281
  }, n))
282
282
  });
283
283
  }
@@ -285,7 +285,7 @@ function M({ projectInfo: e }) {
285
285
  //#region src/client/routes/_auth/projects/$projectId.tsx?tsr-split=component
286
286
  function N() {
287
287
  let { availableServices: e, projectId: t, crumbProject: n } = y({ from: c.id });
288
- return /* @__PURE__ */ d(o, {
288
+ return /* @__PURE__ */ d(i, {
289
289
  embedded: !0,
290
290
  sideNavigation: /* @__PURE__ */ d(j, {
291
291
  availableServices: e,
@@ -293,7 +293,7 @@ function N() {
293
293
  projectName: n?.name || t
294
294
  }),
295
295
  className: "h-min-screen",
296
- children: /* @__PURE__ */ d(i, { children: /* @__PURE__ */ d(s, {
296
+ children: /* @__PURE__ */ d(s, { children: /* @__PURE__ */ d(o, {
297
297
  direction: "vertical",
298
298
  distribution: "start",
299
299
  alignment: "stretch",
@@ -313,4 +313,4 @@ function N() {
313
313
  //#endregion
314
314
  export { N as component };
315
315
 
316
- //# sourceMappingURL=_projectId-DsSVV2EA.mjs.map
316
+ //# sourceMappingURL=_projectId-D1gGribM.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"_projectId-DsSVV2EA.mjs","names":["z","CRUMB_LABEL_KEYS","CrumbSchema","object","labelKey","enum","optional","to","string","useParamAsLabel","RouteInfoSchema","section","service","isDetail","boolean","crumb","sectionCrumb","isRouteInfo","data","safeParse","success","useRef","useState","useEffect","createPortal","useRouteContext","SlotShadowRoot","children","ref","root","setRoot","current","shadowRoot","attachShadow","mode","div","style","display","Slot","component","Component","trpcClient","strict","auroraContext","client","useNavigate","useMatches","useParams","useRouteContext","useState","useEffect","getServiceIndex","SideNavigation","SideNavigationList","SideNavigationItem","isRouteInfo","Slot","SideNavBar","projectId","projectName","availableServices","useLingui","navigate","matches","provider","strict","slots","activeMatch","reverse","find","m","staticData","activeRouteInfo","undefined","activeSection","section","activeService","service","serviceIndex","openSections","setOpenSections","compute","network","storage","services","toggle","prev","sectionKeys","setSectionKeys","isOverviewActive","handleSectionClick","e","clickedItem","target","closest","parentElement","querySelector","computeServices","label","t","to","params","networkServices","storageServices","pcaServices","clavisServices","ariaLabel","icon","onClick","selected","div","onClickCapture","open","map","length","isStorageContainers","isSelected","sideNavBanner","component","Breadcrumb","BreadcrumbItem","useMatches","useNavigate","useParams","useMemo","isRouteInfo","ProjectInfoBox","projectInfo","useLingui","navigate","matches","projectId","strict","breadcrumbs","crumbLabels","Compute","t","Network","Storage","Services","Images","Flavors","resolveProviderLabel","provider","items","push","icon","label","onClick","to","domain","name","params","projectMatches","filter","m","routeId","startsWith","deepest","length","info","staticData","undefined","sectionCrumb","labelKey","isLeaf","crumb","active","useParamAsLabel","resolvedLabel","isDetail","title","meta","find","className","map","item","index","Outlet","useLoaderData","AppShell","Container","Stack","SideNavBar","ProjectInfoBox","Route","RouteComponent","availableServices","projectId","crumbProject","from","id","name","domain","component"],"sources":["../../src/client/routes/routeInfo.ts","../../src/client/components/Slot.tsx","../../src/client/routes/_auth/projects/-components/SideNavBar.tsx","../../src/client/components/ProjectView/ProjectInfoBox.tsx","../../src/client/routes/_auth/projects/$projectId.tsx?tsr-split=component"],"sourcesContent":["import { z } from \"zod\"\n\nexport const CRUMB_LABEL_KEYS = [\n \"Compute\",\n \"Network\",\n \"Storage\",\n \"Services\",\n \"Images\",\n \"Flavors\",\n \"Security Groups\",\n \"Floating IPs\",\n \"PCA (Clavis)\",\n] as const\n\nexport type CrumbLabelKey = (typeof CRUMB_LABEL_KEYS)[number]\n\nconst CrumbSchema = z.object({\n labelKey: z.enum(CRUMB_LABEL_KEYS).optional(),\n to: z.string().optional(),\n useParamAsLabel: z.string().optional(),\n})\n\nconst RouteInfoSchema = z.object({\n section: z.string(),\n service: z.string().optional(),\n isDetail: z.boolean().optional(),\n crumb: CrumbSchema.optional(),\n sectionCrumb: CrumbSchema.optional(),\n})\n\nexport type Crumb = z.infer<typeof CrumbSchema>\nexport type RouteInfo = z.infer<typeof RouteInfoSchema>\n\nexport function isRouteInfo(data: unknown): data is RouteInfo {\n return RouteInfoSchema.safeParse(data).success\n}\n","import { useRef, useState, useEffect, type ReactNode, type FC } from \"react\"\nimport { createPortal } from \"react-dom\"\nimport { useRouteContext } from \"@tanstack/react-router\"\nimport type { SlotProps } from \"../AuroraApp\"\n\nfunction SlotShadowRoot({ children }: { children: ReactNode }) {\n const ref = useRef<HTMLDivElement>(null)\n const [root, setRoot] = useState<ShadowRoot | null>(null)\n\n useEffect(() => {\n if (!ref.current) return\n setRoot(ref.current.shadowRoot ?? ref.current.attachShadow({ mode: \"open\" }))\n }, [])\n\n return (\n <div ref={ref} style={{ display: \"contents\" }}>\n {root && createPortal(children, root)}\n </div>\n )\n}\n\nexport function Slot({ component: Component }: { component: FC<SlotProps> }) {\n const { trpcClient } = useRouteContext({ strict: false })\n\n if (!trpcClient) return null\n\n return (\n <SlotShadowRoot>\n <Component auroraContext={{ client: trpcClient }} />\n </SlotShadowRoot>\n )\n}\n","import { useNavigate, useMatches, useParams, useRouteContext } from \"@tanstack/react-router\"\nimport { type MouseEvent, useState, useEffect } from \"react\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport { SideNavigation, SideNavigationList, SideNavigationItem } from \"@cloudoperators/juno-ui-components/index\"\nimport { useLingui } from \"@lingui/react/macro\"\nimport { isRouteInfo } from \"@/client/routes/routeInfo\"\nimport { Slot } from \"@/client/components/Slot\"\n\ninterface SideNavBarProps {\n projectId: string\n projectName: string\n availableServices: {\n type: string\n name: string\n }[]\n}\n\nexport const SideNavBar = ({ projectId, projectName, availableServices }: SideNavBarProps) => {\n const { t } = useLingui()\n const navigate = useNavigate()\n const matches = useMatches()\n const { provider } = useParams({ strict: false }) as { provider?: string }\n const { slots } = useRouteContext({ strict: false })\n\n // Read active section/service from the deepest match that has valid RouteInfo staticData\n const activeMatch = [...matches].reverse().find((m) => isRouteInfo(m.staticData))\n const activeRouteInfo = activeMatch && isRouteInfo(activeMatch.staticData) ? activeMatch.staticData : undefined\n const activeSection = activeRouteInfo?.section ?? null\n const activeService = activeRouteInfo?.service ?? null\n\n const serviceIndex = getServiceIndex(availableServices)\n\n const [openSections, setOpenSections] = useState({ compute: true, network: true, storage: true, services: true })\n const toggle = (section: keyof typeof openSections) =>\n setOpenSections((prev) => ({ ...prev, [section]: !prev[section] }))\n\n // When navigating into a section, force Juno to re-open it by resetting the key\n const [sectionKeys, setSectionKeys] = useState({ compute: 0, network: 0, storage: 0, services: 0 })\n\n useEffect(() => {\n if (activeSection && activeSection in sectionKeys) {\n setSectionKeys((prev) => ({ ...prev, [activeSection]: prev[activeSection as keyof typeof sectionKeys] + 1 }))\n setOpenSections((prev) => ({ ...prev, [activeSection]: true }))\n }\n }, [activeSection])\n\n const isOverviewActive = activeSection === null\n\n const handleSectionClick = (section: keyof typeof openSections) => (e: MouseEvent<HTMLDivElement>) => {\n // Only toggle if the click is on the header row (has expand-icon sibling), not on child items\n const clickedItem = (e.target as HTMLElement).closest(\".juno-sidenavigation-item\")\n if (clickedItem && clickedItem.parentElement?.querySelector(\".expand-icon\")) {\n toggle(section)\n }\n }\n\n const computeServices = [\n ...(serviceIndex[\"image\"]?.[\"glance\"]\n ? [\n {\n service: \"images\",\n label: t`Images`,\n to: \"/projects/$projectId/compute/images\" as const,\n params: { projectId },\n },\n ]\n : []),\n ...(serviceIndex?.[\"compute\"]?.[\"nova\"]\n ? [\n {\n service: \"flavors\",\n label: t`Flavors`,\n to: \"/projects/$projectId/compute/flavors\" as const,\n params: { projectId },\n },\n ]\n : []),\n ]\n\n const networkServices = [\n ...(serviceIndex[\"network\"]\n ? [\n {\n service: \"securitygroups\",\n label: t`Security Groups`,\n to: \"/projects/$projectId/network/securitygroups\" as const,\n params: { projectId },\n },\n {\n service: \"floatingips\",\n label: t`Floating IPs`,\n to: \"/projects/$projectId/network/floatingips\" as const,\n params: { projectId },\n },\n ]\n : []),\n ]\n\n const storageServices = [\n ...(serviceIndex?.[\"object-store\"]?.[\"swift\"]\n ? [\n {\n service: \"containers\",\n label: t`Object Storage (Swift)`,\n to: \"/projects/$projectId/storage/$provider/containers\" as const,\n params: { projectId, provider: \"swift\" },\n },\n ]\n : []),\n {\n service: \"ceph-containers\",\n label: t`Object Storage (Ceph)`,\n to: \"/projects/$projectId/storage/$provider/containers\" as const,\n params: { projectId, provider: \"ceph\" },\n },\n ]\n\n // temporary as clavis is not fully GA, after GA replace with [\"pca\"]?.[\"clavis\"]\n const pcaServices = serviceIndex[\"pca\"]?.[\"clavis-beta\"] || serviceIndex[\"pca\"]?.[\"clavis-dev\"]\n const clavisServices = [\n ...(pcaServices\n ? [\n {\n service: \"pca\",\n label: t`PCA (Clavis)`,\n to: \"/projects/$projectId/services/pca\" as const,\n params: { projectId },\n },\n ]\n : []),\n ]\n\n return (\n <SideNavigation ariaLabel=\"Project Side Navigation\">\n <>\n <SideNavigationList>\n <>\n <SideNavigationItem\n icon=\"home\"\n label={projectName}\n onClick={() => navigate({ to: \"/projects/$projectId\", params: { projectId } })}\n selected={isOverviewActive}\n />\n {/* onClickCapture fires before Juno's chevron stopPropagation, keeping our state in sync */}\n <div onClickCapture={handleSectionClick(\"compute\")}>\n <SideNavigationItem key={sectionKeys.compute} label={t`Compute`} open={openSections.compute}>\n {computeServices.map(({ service, label, to, params }) => (\n <SideNavigationItem\n key={label}\n onClick={() => navigate({ to, params })}\n label={label}\n selected={activeSection === \"compute\" && activeService === service}\n />\n ))}\n </SideNavigationItem>\n </div>\n\n {networkServices.length > 0 && (\n <div onClickCapture={handleSectionClick(\"network\")}>\n <SideNavigationItem key={sectionKeys.network} label={t`Network`} open={openSections.network}>\n {networkServices.map(({ service, label, to, params }) => (\n <SideNavigationItem\n key={label}\n onClick={() => navigate({ to, params })}\n label={label}\n selected={activeSection === \"network\" && activeService === service}\n />\n ))}\n </SideNavigationItem>\n </div>\n )}\n\n {storageServices.length > 0 && (\n <div onClickCapture={handleSectionClick(\"storage\")}>\n <SideNavigationItem key={sectionKeys.storage} label={t`Storage`} open={openSections.storage}>\n {storageServices.map(({ service, label, to, params }) => {\n // For storage services with provider param, match against current provider\n const isStorageContainers = activeSection === \"storage\" && activeService === \"containers\"\n const isSelected = isStorageContainers ? params.provider === provider : activeService === service\n\n return (\n <SideNavigationItem\n key={label}\n onClick={() => navigate({ to, params })}\n label={label}\n selected={isSelected}\n />\n )\n })}\n </SideNavigationItem>\n </div>\n )}\n\n {clavisServices.length > 0 && (\n <div onClickCapture={handleSectionClick(\"services\")}>\n <SideNavigationItem key={sectionKeys.services} label={t`Services`} open={openSections.services}>\n {clavisServices.map(({ service, label, to, params }) => (\n <SideNavigationItem\n key={label}\n onClick={() => navigate({ to, params })}\n label={label}\n selected={activeSection === \"services\" && activeService === service}\n />\n ))}\n </SideNavigationItem>\n </div>\n )}\n </>\n </SideNavigationList>\n {slots?.sideNavBanner && <Slot component={slots.sideNavBanner} />}\n </>\n </SideNavigation>\n )\n}\n","import { Breadcrumb, BreadcrumbItem, KnownIcons } from \"@cloudoperators/juno-ui-components\"\nimport { useMatches, useNavigate, useParams } from \"@tanstack/react-router\"\nimport { useMemo } from \"react\"\nimport { useLingui } from \"@lingui/react/macro\"\nimport { isRouteInfo, CrumbLabelKey } from \"@/client/routes/routeInfo\"\n\ninterface ProjectInfoBoxProps {\n projectInfo: {\n id: string\n name: string\n description?: string\n domain?: {\n name?: string\n }\n }\n}\n\nexport function ProjectInfoBox({ projectInfo }: ProjectInfoBoxProps) {\n const { t } = useLingui()\n const navigate = useNavigate()\n const matches = useMatches()\n const { projectId } = useParams({ strict: false }) as { projectId: string }\n\n const breadcrumbs = useMemo(() => {\n const crumbLabels: Record<CrumbLabelKey, string> = {\n Compute: t`Compute`,\n Network: t`Network`,\n Storage: t`Storage`,\n Services: t`Services`,\n Images: t`Images`,\n Flavors: t`Flavors`,\n \"Security Groups\": t`Security Groups`,\n \"Floating IPs\": t`Floating IPs`,\n \"PCA (Clavis)\": t`PCA (Clavis)`,\n }\n\n const resolveProviderLabel = (provider: string | undefined) => {\n if (provider === \"swift\") return t`Object Storage (Swift)`\n if (provider === \"ceph\") return t`Object Storage (Ceph)`\n return t`Storage`\n }\n\n const items: Array<{ label?: string; icon?: KnownIcons; onClick?: () => void; active?: boolean }> = []\n\n items.push({ icon: \"home\", label: t`Home`, onClick: () => navigate({ to: \"/projects\" }) })\n\n if (projectInfo.domain?.name) {\n items.push({ label: projectInfo.domain.name })\n }\n\n items.push({\n label: projectInfo.name,\n onClick: () => navigate({ to: \"/projects/$projectId\", params: { projectId } }),\n })\n\n const projectMatches = matches.filter(\n (m) => m.routeId !== \"/_auth/projects/$projectId\" && m.routeId.startsWith(\"/_auth/projects/$projectId\")\n )\n const deepest = projectMatches[projectMatches.length - 1]\n if (!deepest) return items\n\n const info = isRouteInfo(deepest.staticData) ? deepest.staticData : undefined\n if (!info) return items\n\n const params = deepest.params as Record<string, string>\n\n if (info.sectionCrumb) {\n const { labelKey, to } = info.sectionCrumb\n const label = labelKey ? crumbLabels[labelKey] : undefined\n const isLeaf = !info.crumb\n items.push(\n to\n ? { label, onClick: () => navigate({ to: to as never, params: params as never }) }\n : { label, active: isLeaf }\n )\n }\n\n if (info.crumb) {\n const { labelKey, to, useParamAsLabel } = info.crumb\n const resolvedLabel = useParamAsLabel\n ? resolveProviderLabel(params[useParamAsLabel])\n : labelKey\n ? crumbLabels[labelKey]\n : undefined\n\n if (info.isDetail) {\n items.push({ label: resolvedLabel, onClick: () => navigate({ to: to as never, params: params as never }) })\n const title = deepest.meta?.find((m) => m != null && \"title\" in m)?.title as string | undefined\n if (title) items.push({ label: title, active: true })\n } else {\n items.push(\n to\n ? { label: resolvedLabel, onClick: () => navigate({ to: to as never, params: params as never }) }\n : { label: resolvedLabel, active: true }\n )\n }\n }\n\n return items\n }, [matches, projectInfo, projectId, navigate, t])\n\n return (\n <Breadcrumb className=\"mt-8 mb-4\">\n {breadcrumbs.map((item, index) => (\n <BreadcrumbItem key={index} label={item.label} icon={item.icon} onClick={item.onClick} active={item.active} />\n ))}\n </Breadcrumb>\n )\n}\n","import { createFileRoute, Outlet, useLoaderData } from \"@tanstack/react-router\"\nimport { AppShell, Container, Stack } from \"@cloudoperators/juno-ui-components\"\nimport { SideNavBar } from \"@/client/routes/_auth/projects/-components/SideNavBar\"\nimport { ProjectInfoBox } from \"@/client/components/ProjectView/ProjectInfoBox\"\nimport { RouteError } from \"@/client/components/Error/RouteError\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId\")({\n component: RouteComponent,\n errorComponent: ({ error }) => {\n return <RouteError error={error} />\n },\n loader: async (options) => {\n const { context, params } = options\n const data = await context.trpcClient?.auth.setCurrentScope.mutate({\n type: \"project\",\n projectId: params.projectId || \"\",\n })\n const availableServices = await context.trpcClient?.auth.getAvailableServices.query()\n\n // Extract accountId (domain id) from the rescoped token\n // This is needed for SideNavBar navigation until we refactor it\n const accountId = data?.domain?.id || \"\"\n\n return {\n trpcClient: context.trpcClient,\n crumbDomain: { path: `/projects`, name: data?.domain?.name },\n crumbProject: data?.project,\n availableServices,\n accountId, // Keep for SideNavBar compatibility\n projectId: params.projectId,\n }\n },\n})\n\nfunction RouteComponent() {\n const { availableServices, projectId, crumbProject } = useLoaderData({ from: Route.id })\n\n return (\n <AppShell\n embedded\n sideNavigation={\n <SideNavBar\n availableServices={availableServices!}\n projectId={projectId}\n projectName={crumbProject?.name || projectId}\n />\n }\n className=\"h-min-screen\"\n >\n <Container>\n <Stack direction=\"vertical\" distribution=\"start\" alignment=\"stretch\" className=\"xl:flex-row\" gap=\"6\">\n {/* Main content area */}\n <div className=\"min-w-0 flex-1\">\n <ProjectInfoBox\n projectInfo={{\n id: projectId,\n name: crumbProject?.name || projectId,\n domain: crumbProject?.domain,\n }}\n />\n <Outlet />\n </div>\n </Stack>\n </Container>\n </AppShell>\n )\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"_projectId-D1gGribM.mjs","names":["z","CRUMB_LABEL_KEYS","CrumbSchema","object","labelKey","enum","optional","to","string","useParamAsLabel","RouteInfoSchema","section","service","isDetail","boolean","crumb","sectionCrumb","isRouteInfo","data","safeParse","success","useRef","useState","useEffect","createPortal","useRouteContext","SlotShadowRoot","children","ref","root","setRoot","current","shadowRoot","attachShadow","mode","div","style","display","Slot","component","Component","trpcClient","strict","auroraContext","client","useNavigate","useMatches","useParams","useRouteContext","useState","useEffect","getServiceIndex","SideNavigation","SideNavigationList","SideNavigationItem","isRouteInfo","Slot","SideNavBar","projectId","projectName","availableServices","useLingui","navigate","matches","provider","strict","slots","activeMatch","reverse","find","m","staticData","activeRouteInfo","undefined","activeSection","section","activeService","service","serviceIndex","openSections","setOpenSections","compute","network","storage","services","toggle","prev","sectionKeys","setSectionKeys","isOverviewActive","handleSectionClick","e","clickedItem","target","closest","parentElement","querySelector","computeServices","label","t","to","params","networkServices","storageServices","pcaServices","clavisServices","ariaLabel","icon","onClick","selected","div","onClickCapture","open","map","length","isStorageContainers","isSelected","sideNavBanner","component","Breadcrumb","BreadcrumbItem","useMatches","useNavigate","useParams","useMemo","isRouteInfo","ProjectInfoBox","projectInfo","useLingui","navigate","matches","projectId","strict","breadcrumbs","crumbLabels","Compute","t","Network","Storage","Services","Images","Flavors","resolveProviderLabel","provider","items","push","icon","label","onClick","to","domain","name","params","projectMatches","filter","m","routeId","startsWith","deepest","length","info","staticData","undefined","sectionCrumb","labelKey","isLeaf","crumb","active","useParamAsLabel","resolvedLabel","isDetail","title","meta","find","className","map","item","index","Outlet","useLoaderData","AppShell","Container","Stack","SideNavBar","ProjectInfoBox","Route","RouteComponent","availableServices","projectId","crumbProject","from","id","name","domain","component"],"sources":["../../src/client/routes/routeInfo.ts","../../src/client/components/Slot.tsx","../../src/client/routes/_auth/projects/-components/SideNavBar.tsx","../../src/client/components/ProjectView/ProjectInfoBox.tsx","../../src/client/routes/_auth/projects/$projectId.tsx?tsr-split=component"],"sourcesContent":["import { z } from \"zod\"\n\nexport const CRUMB_LABEL_KEYS = [\n \"Compute\",\n \"Network\",\n \"Storage\",\n \"Services\",\n \"Images\",\n \"Flavors\",\n \"Security Groups\",\n \"Floating IPs\",\n \"PCA (Clavis)\",\n] as const\n\nexport type CrumbLabelKey = (typeof CRUMB_LABEL_KEYS)[number]\n\nconst CrumbSchema = z.object({\n labelKey: z.enum(CRUMB_LABEL_KEYS).optional(),\n to: z.string().optional(),\n useParamAsLabel: z.string().optional(),\n})\n\nconst RouteInfoSchema = z.object({\n section: z.string(),\n service: z.string().optional(),\n isDetail: z.boolean().optional(),\n crumb: CrumbSchema.optional(),\n sectionCrumb: CrumbSchema.optional(),\n})\n\nexport type Crumb = z.infer<typeof CrumbSchema>\nexport type RouteInfo = z.infer<typeof RouteInfoSchema>\n\nexport function isRouteInfo(data: unknown): data is RouteInfo {\n return RouteInfoSchema.safeParse(data).success\n}\n","import { useRef, useState, useEffect, type ReactNode, type FC } from \"react\"\nimport { createPortal } from \"react-dom\"\nimport { useRouteContext } from \"@tanstack/react-router\"\nimport type { SlotProps } from \"../AuroraApp\"\n\nfunction SlotShadowRoot({ children }: { children: ReactNode }) {\n const ref = useRef<HTMLDivElement>(null)\n const [root, setRoot] = useState<ShadowRoot | null>(null)\n\n useEffect(() => {\n if (!ref.current) return\n setRoot(ref.current.shadowRoot ?? ref.current.attachShadow({ mode: \"open\" }))\n }, [])\n\n return (\n <div ref={ref} style={{ display: \"contents\" }}>\n {root && createPortal(children, root)}\n </div>\n )\n}\n\nexport function Slot({ component: Component }: { component: FC<SlotProps> }) {\n const { trpcClient } = useRouteContext({ strict: false })\n\n if (!trpcClient) return null\n\n return (\n <SlotShadowRoot>\n <Component auroraContext={{ client: trpcClient }} />\n </SlotShadowRoot>\n )\n}\n","import { useNavigate, useMatches, useParams, useRouteContext } from \"@tanstack/react-router\"\nimport { type MouseEvent, useState, useEffect } from \"react\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport { SideNavigation, SideNavigationList, SideNavigationItem } from \"@cloudoperators/juno-ui-components/index\"\nimport { useLingui } from \"@lingui/react/macro\"\nimport { isRouteInfo } from \"@/client/routes/routeInfo\"\nimport { Slot } from \"@/client/components/Slot\"\n\ninterface SideNavBarProps {\n projectId: string\n projectName: string\n availableServices: {\n type: string\n name: string\n }[]\n}\n\nexport const SideNavBar = ({ projectId, projectName, availableServices }: SideNavBarProps) => {\n const { t } = useLingui()\n const navigate = useNavigate()\n const matches = useMatches()\n const { provider } = useParams({ strict: false }) as { provider?: string }\n const { slots } = useRouteContext({ strict: false })\n\n // Read active section/service from the deepest match that has valid RouteInfo staticData\n const activeMatch = [...matches].reverse().find((m) => isRouteInfo(m.staticData))\n const activeRouteInfo = activeMatch && isRouteInfo(activeMatch.staticData) ? activeMatch.staticData : undefined\n const activeSection = activeRouteInfo?.section ?? null\n const activeService = activeRouteInfo?.service ?? null\n\n const serviceIndex = getServiceIndex(availableServices)\n\n const [openSections, setOpenSections] = useState({ compute: true, network: true, storage: true, services: true })\n const toggle = (section: keyof typeof openSections) =>\n setOpenSections((prev) => ({ ...prev, [section]: !prev[section] }))\n\n // When navigating into a section, force Juno to re-open it by resetting the key\n const [sectionKeys, setSectionKeys] = useState({ compute: 0, network: 0, storage: 0, services: 0 })\n\n useEffect(() => {\n if (activeSection && activeSection in sectionKeys) {\n setSectionKeys((prev) => ({ ...prev, [activeSection]: prev[activeSection as keyof typeof sectionKeys] + 1 }))\n setOpenSections((prev) => ({ ...prev, [activeSection]: true }))\n }\n }, [activeSection])\n\n const isOverviewActive = activeSection === null\n\n const handleSectionClick = (section: keyof typeof openSections) => (e: MouseEvent<HTMLDivElement>) => {\n // Only toggle if the click is on the header row (has expand-icon sibling), not on child items\n const clickedItem = (e.target as HTMLElement).closest(\".juno-sidenavigation-item\")\n if (clickedItem && clickedItem.parentElement?.querySelector(\".expand-icon\")) {\n toggle(section)\n }\n }\n\n const computeServices = [\n ...(serviceIndex[\"image\"]?.[\"glance\"]\n ? [\n {\n service: \"images\",\n label: t`Images`,\n to: \"/projects/$projectId/compute/images\" as const,\n params: { projectId },\n },\n ]\n : []),\n ...(serviceIndex?.[\"compute\"]?.[\"nova\"]\n ? [\n {\n service: \"flavors\",\n label: t`Flavors`,\n to: \"/projects/$projectId/compute/flavors\" as const,\n params: { projectId },\n },\n ]\n : []),\n ]\n\n const networkServices = [\n ...(serviceIndex[\"network\"]\n ? [\n {\n service: \"securitygroups\",\n label: t`Security Groups`,\n to: \"/projects/$projectId/network/securitygroups\" as const,\n params: { projectId },\n },\n {\n service: \"floatingips\",\n label: t`Floating IPs`,\n to: \"/projects/$projectId/network/floatingips\" as const,\n params: { projectId },\n },\n ]\n : []),\n ]\n\n const storageServices = [\n ...(serviceIndex?.[\"object-store\"]?.[\"swift\"]\n ? [\n {\n service: \"containers\",\n label: t`Object Storage (Swift)`,\n to: \"/projects/$projectId/storage/$provider/containers\" as const,\n params: { projectId, provider: \"swift\" },\n },\n ]\n : []),\n {\n service: \"ceph-containers\",\n label: t`Object Storage (Ceph)`,\n to: \"/projects/$projectId/storage/$provider/containers\" as const,\n params: { projectId, provider: \"ceph\" },\n },\n ]\n\n // temporary as clavis is not fully GA, after GA replace with [\"pca\"]?.[\"clavis\"]\n const pcaServices = serviceIndex[\"pca\"]?.[\"clavis-beta\"] || serviceIndex[\"pca\"]?.[\"clavis-dev\"]\n const clavisServices = [\n ...(pcaServices\n ? [\n {\n service: \"pca\",\n label: t`PCA (Clavis)`,\n to: \"/projects/$projectId/services/pca\" as const,\n params: { projectId },\n },\n ]\n : []),\n ]\n\n return (\n <SideNavigation ariaLabel=\"Project Side Navigation\">\n <>\n <SideNavigationList>\n <>\n <SideNavigationItem\n icon=\"home\"\n label={projectName}\n onClick={() => navigate({ to: \"/projects/$projectId\", params: { projectId } })}\n selected={isOverviewActive}\n />\n {/* onClickCapture fires before Juno's chevron stopPropagation, keeping our state in sync */}\n <div onClickCapture={handleSectionClick(\"compute\")}>\n <SideNavigationItem key={sectionKeys.compute} label={t`Compute`} open={openSections.compute}>\n {computeServices.map(({ service, label, to, params }) => (\n <SideNavigationItem\n key={label}\n onClick={() => navigate({ to, params })}\n label={label}\n selected={activeSection === \"compute\" && activeService === service}\n />\n ))}\n </SideNavigationItem>\n </div>\n\n {networkServices.length > 0 && (\n <div onClickCapture={handleSectionClick(\"network\")}>\n <SideNavigationItem key={sectionKeys.network} label={t`Network`} open={openSections.network}>\n {networkServices.map(({ service, label, to, params }) => (\n <SideNavigationItem\n key={label}\n onClick={() => navigate({ to, params })}\n label={label}\n selected={activeSection === \"network\" && activeService === service}\n />\n ))}\n </SideNavigationItem>\n </div>\n )}\n\n {storageServices.length > 0 && (\n <div onClickCapture={handleSectionClick(\"storage\")}>\n <SideNavigationItem key={sectionKeys.storage} label={t`Storage`} open={openSections.storage}>\n {storageServices.map(({ service, label, to, params }) => {\n // For storage services with provider param, match against current provider\n const isStorageContainers = activeSection === \"storage\" && activeService === \"containers\"\n const isSelected = isStorageContainers ? params.provider === provider : activeService === service\n\n return (\n <SideNavigationItem\n key={label}\n onClick={() => navigate({ to, params })}\n label={label}\n selected={isSelected}\n />\n )\n })}\n </SideNavigationItem>\n </div>\n )}\n\n {clavisServices.length > 0 && (\n <div onClickCapture={handleSectionClick(\"services\")}>\n <SideNavigationItem key={sectionKeys.services} label={t`Services`} open={openSections.services}>\n {clavisServices.map(({ service, label, to, params }) => (\n <SideNavigationItem\n key={label}\n onClick={() => navigate({ to, params })}\n label={label}\n selected={activeSection === \"services\" && activeService === service}\n />\n ))}\n </SideNavigationItem>\n </div>\n )}\n </>\n </SideNavigationList>\n {slots?.sideNavBanner && <Slot component={slots.sideNavBanner} />}\n </>\n </SideNavigation>\n )\n}\n","import { Breadcrumb, BreadcrumbItem, KnownIcons } from \"@cloudoperators/juno-ui-components\"\nimport { useMatches, useNavigate, useParams } from \"@tanstack/react-router\"\nimport { useMemo } from \"react\"\nimport { useLingui } from \"@lingui/react/macro\"\nimport { isRouteInfo, CrumbLabelKey } from \"@/client/routes/routeInfo\"\n\ninterface ProjectInfoBoxProps {\n projectInfo: {\n id: string\n name: string\n description?: string\n domain?: {\n name?: string\n }\n }\n}\n\nexport function ProjectInfoBox({ projectInfo }: ProjectInfoBoxProps) {\n const { t } = useLingui()\n const navigate = useNavigate()\n const matches = useMatches()\n const { projectId } = useParams({ strict: false }) as { projectId: string }\n\n const breadcrumbs = useMemo(() => {\n const crumbLabels: Record<CrumbLabelKey, string> = {\n Compute: t`Compute`,\n Network: t`Network`,\n Storage: t`Storage`,\n Services: t`Services`,\n Images: t`Images`,\n Flavors: t`Flavors`,\n \"Security Groups\": t`Security Groups`,\n \"Floating IPs\": t`Floating IPs`,\n \"PCA (Clavis)\": t`PCA (Clavis)`,\n }\n\n const resolveProviderLabel = (provider: string | undefined) => {\n if (provider === \"swift\") return t`Object Storage (Swift)`\n if (provider === \"ceph\") return t`Object Storage (Ceph)`\n return t`Storage`\n }\n\n const items: Array<{ label?: string; icon?: KnownIcons; onClick?: () => void; active?: boolean }> = []\n\n items.push({ icon: \"home\", label: t`Home`, onClick: () => navigate({ to: \"/projects\" }) })\n\n if (projectInfo.domain?.name) {\n items.push({ label: projectInfo.domain.name })\n }\n\n items.push({\n label: projectInfo.name,\n onClick: () => navigate({ to: \"/projects/$projectId\", params: { projectId } }),\n })\n\n const projectMatches = matches.filter(\n (m) => m.routeId !== \"/_auth/projects/$projectId\" && m.routeId.startsWith(\"/_auth/projects/$projectId\")\n )\n const deepest = projectMatches[projectMatches.length - 1]\n if (!deepest) return items\n\n const info = isRouteInfo(deepest.staticData) ? deepest.staticData : undefined\n if (!info) return items\n\n const params = deepest.params as Record<string, string>\n\n if (info.sectionCrumb) {\n const { labelKey, to } = info.sectionCrumb\n const label = labelKey ? crumbLabels[labelKey] : undefined\n const isLeaf = !info.crumb\n items.push(\n to\n ? { label, onClick: () => navigate({ to: to as never, params: params as never }) }\n : { label, active: isLeaf }\n )\n }\n\n if (info.crumb) {\n const { labelKey, to, useParamAsLabel } = info.crumb\n const resolvedLabel = useParamAsLabel\n ? resolveProviderLabel(params[useParamAsLabel])\n : labelKey\n ? crumbLabels[labelKey]\n : undefined\n\n if (info.isDetail) {\n items.push({ label: resolvedLabel, onClick: () => navigate({ to: to as never, params: params as never }) })\n const title = deepest.meta?.find((m) => m != null && \"title\" in m)?.title as string | undefined\n if (title) items.push({ label: title, active: true })\n } else {\n items.push(\n to\n ? { label: resolvedLabel, onClick: () => navigate({ to: to as never, params: params as never }) }\n : { label: resolvedLabel, active: true }\n )\n }\n }\n\n return items\n }, [matches, projectInfo, projectId, navigate, t])\n\n return (\n <Breadcrumb className=\"mt-8 mb-4\">\n {breadcrumbs.map((item, index) => (\n <BreadcrumbItem key={index} label={item.label} icon={item.icon} onClick={item.onClick} active={item.active} />\n ))}\n </Breadcrumb>\n )\n}\n","import { createFileRoute, Outlet, useLoaderData } from \"@tanstack/react-router\"\nimport { AppShell, Container, Stack } from \"@cloudoperators/juno-ui-components\"\nimport { SideNavBar } from \"@/client/routes/_auth/projects/-components/SideNavBar\"\nimport { ProjectInfoBox } from \"@/client/components/ProjectView/ProjectInfoBox\"\nimport { RouteError } from \"@/client/components/Error/RouteError\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId\")({\n component: RouteComponent,\n errorComponent: ({ error }) => {\n return <RouteError error={error} />\n },\n loader: async (options) => {\n const { context, params } = options\n const data = await context.trpcClient?.auth.setCurrentScope.mutate({\n type: \"project\",\n projectId: params.projectId || \"\",\n })\n const availableServices = await context.trpcClient?.auth.getAvailableServices.query()\n\n // Extract accountId (domain id) from the rescoped token\n // This is needed for SideNavBar navigation until we refactor it\n const accountId = data?.domain?.id || \"\"\n\n return {\n trpcClient: context.trpcClient,\n crumbDomain: { path: `/projects`, name: data?.domain?.name },\n crumbProject: data?.project,\n availableServices,\n accountId, // Keep for SideNavBar compatibility\n projectId: params.projectId,\n }\n },\n})\n\nfunction RouteComponent() {\n const { availableServices, projectId, crumbProject } = useLoaderData({ from: Route.id })\n\n return (\n <AppShell\n embedded\n sideNavigation={\n <SideNavBar\n availableServices={availableServices!}\n projectId={projectId}\n projectName={crumbProject?.name || projectId}\n />\n }\n className=\"h-min-screen\"\n >\n <Container>\n <Stack direction=\"vertical\" distribution=\"start\" alignment=\"stretch\" className=\"xl:flex-row\" gap=\"6\">\n {/* Main content area */}\n <div className=\"min-w-0 flex-1\">\n <ProjectInfoBox\n projectInfo={{\n id: projectId,\n name: crumbProject?.name || projectId,\n domain: crumbProject?.domain,\n }}\n />\n <Outlet />\n </div>\n </Stack>\n </Container>\n </AppShell>\n )\n}\n"],"mappings":"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"}
@@ -1,6 +1,6 @@
1
- import { b as e, tt as t } from "./build-CZRvXrAL.mjs";
1
+ import { b as e, nt as t } from "./build-BJDfnAyi.mjs";
2
2
  import { t as n } from "./helpers--JWXi40U.mjs";
3
- import { t as r } from "./ContentHeader-B_PWrxbw.mjs";
3
+ import { t as r } from "./ContentHeader-BXZoN3B9.mjs";
4
4
  import { jsx as i, jsxs as a } from "react/jsx-runtime";
5
5
  import { Link as o, useLoaderData as s } from "@tanstack/react-router";
6
6
  import { Trans as c, useLingui as l } from "@lingui/react";
@@ -81,4 +81,4 @@ function d() {
81
81
  //#endregion
82
82
  export { d as component };
83
83
 
84
- //# sourceMappingURL=_projectId-DYrcZ3E3.mjs.map
84
+ //# sourceMappingURL=_projectId-DhLpIalx.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"_projectId-DYrcZ3E3.mjs","names":["Link","useLoaderData","Box","Stack","getServiceIndex","Trans","useLingui","ContentHeader","ServiceCardProps","title","links","label","to","ServiceCard","map","RouteComponent","crumbProject","availableServices","projectId","from","t","serviceIndex","base","cards","push","storageLinks","name","length","card","component"],"sources":["../../src/client/routes/_auth/projects/$projectId/index.tsx?tsr-split=component"],"sourcesContent":["import { createFileRoute, Link, useLoaderData } from \"@tanstack/react-router\"\nimport { Box, Stack } from \"@cloudoperators/juno-ui-components\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport { Trans } from \"@lingui/react/macro\"\nimport { useLingui } from \"@lingui/react/macro\"\nimport { ContentHeader } from \"@/client/components/ContentHeader/ContentHeader\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId/\")({\n component: RouteComponent,\n})\n\ninterface ServiceCardProps {\n title: string\n links: { label: string; to: string }[]\n}\n\nfunction ServiceCard({ title, links }: ServiceCardProps) {\n return (\n <Box className=\"p-5\">\n <h3 className=\"text-theme-high mb-3 text-base font-semibold\">{title}</h3>\n <ul className=\"space-y-1.5\">\n {links.map(({ label, to }) => (\n <li key={label}>\n <Link to={to} className=\"text-theme-accent hover:text-theme-accent/80 text-sm\">\n {label}\n </Link>\n </li>\n ))}\n </ul>\n </Box>\n )\n}\n\nfunction RouteComponent() {\n const { crumbProject, availableServices, projectId } = useLoaderData({\n from: \"/_auth/projects/$projectId\",\n })\n const { t } = useLingui()\n\n const serviceIndex = getServiceIndex(availableServices ?? [])\n const base = `/projects/${projectId}`\n\n const cards: ServiceCardProps[] = []\n\n if (serviceIndex[\"image\"]?.[\"glance\"] || serviceIndex[\"compute\"]?.[\"nova\"]) {\n const links: { label: string; to: string }[] = []\n if (serviceIndex[\"image\"]?.[\"glance\"]) links.push({ label: \"Images\", to: `${base}/compute/images` })\n if (serviceIndex[\"compute\"]?.[\"nova\"]) links.push({ label: \"Flavors\", to: `${base}/compute/flavors` })\n cards.push({ title: \"Compute\", links })\n }\n\n if (serviceIndex[\"network\"]) {\n cards.push({\n title: \"Network\",\n links: [\n { label: \"Security Groups\", to: `${base}/network/securitygroups` },\n { label: \"Floating IPs\", to: `${base}/network/floatingips` },\n ],\n })\n }\n\n // Storage section - always show Ceph, optionally add Swift\n const storageLinks: { label: string; to: string }[] = []\n if (serviceIndex[\"object-store\"]?.[\"swift\"]) {\n storageLinks.push({ label: \"Swift\", to: `${base}/storage/swift/containers` })\n }\n storageLinks.push({ label: \"Ceph\", to: `${base}/storage/ceph/containers` })\n cards.push({ title: \"Storage\", links: storageLinks })\n\n // temporary as clavis is not fully GA, after GA replace with [\"pca\"]?.[\"clavis\"]\n if (serviceIndex[\"pca\"]?.[\"clavis-dev\"] || serviceIndex[\"pca\"]?.[\"clavis-beta\"]) {\n cards.push({\n title: \"Services\",\n links: [{ label: \"PCA (Clavis)\", to: `${base}/services/pca` }],\n })\n }\n\n return (\n <Stack direction=\"vertical\" gap=\"6\" className=\"pb-4\">\n <ContentHeader title={crumbProject?.name ?? t`Project`} projectId={projectId} />\n {cards.length > 0 ? (\n <div className=\"grid grid-cols-1 gap-4 sm:grid-cols-2 lg:grid-cols-3\">\n {cards.map((card) => (\n <ServiceCard key={card.title} {...card} />\n ))}\n </div>\n ) : (\n <p className=\"text-theme-light text-sm\">\n <Trans>No services available for this project.</Trans>\n </p>\n )}\n </Stack>\n )\n}\n"],"mappings":";;;;;;;AAgBA,SAASa,EAAY,EAAEJ,UAAOC,YAAyB;AACrD,QACE,kBAAC,GAAA;EAAI,WAAU;aACb,kBAAC,MAAA;GAAG,WAAU;aAAgDD;MAC9D,kBAAC,MAAA;GAAG,WAAU;aACXC,EAAMI,KAAK,EAAEH,UAAOC,YACnB,kBAAC,MAAA,EAAA,UACC,kBAAC,GAAA;IAASA;IAAI,WAAU;cACrBD;SAFIA,EAAAA,CAAAA;;;;AAWnB,SAASI,IAAAA;CACP,IAAM,EAAEC,iBAAcC,sBAAmBC,iBAAcjB,EAAc,EACnEkB,MAAM,8BACR,CAAA,EACM,EAAA,MAAA,GAAA,GAAA,MAAQb,GAAAA,EAERe,IAAejB,EAAgBa,KAAqB,EAAE,CAAA,EACtDK,IAAO,aAAaJ,KAEpBK,IAA4B,EAAE;AAEpC,KAAIF,EAAa,OAAW,UAAaA,EAAa,SAAa,MAAS;EAC1E,IAAMX,IAAyC,EAAE;AAGjDa,EAFIF,EAAa,OAAW,UAAWX,EAAMc,KAAK;GAAEb,OAAO;GAAUC,IAAI,GAAGU,EAAI;GAAkB,CAAA,EAC9FD,EAAa,SAAa,QAASX,EAAMc,KAAK;GAAEb,OAAO;GAAWC,IAAI,GAAGU,EAAI;GAAmB,CAAA,EACpGC,EAAMC,KAAK;GAAEf,OAAO;GAAWC;GAAM,CAAA;;AAGvC,CAAIW,EAAa,WACfE,EAAMC,KAAK;EACTf,OAAO;EACPC,OAAO,CACL;GAAEC,OAAO;GAAmBC,IAAI,GAAGU,EAAI;GAA0B,EACjE;GAAEX,OAAO;GAAgBC,IAAI,GAAGU,EAAI;GAAuB,CAAC;EAEhE,CAAA;CAIF,IAAMG,IAAgD,EAAE;AAexD,QAdIJ,EAAa,iBAAkB,SACjCI,EAAaD,KAAK;EAAEb,OAAO;EAASC,IAAI,GAAGU,EAAI;EAA4B,CAAA,EAE7EG,EAAaD,KAAK;EAAEb,OAAO;EAAQC,IAAI,GAAGU,EAAI;EAA2B,CAAA,EACzEC,EAAMC,KAAK;EAAEf,OAAO;EAAWC,OAAOe;EAAa,CAAA,GAG/CJ,EAAa,MAAS,iBAAiBA,EAAa,MAAS,mBAC/DE,EAAMC,KAAK;EACTf,OAAO;EACPC,OAAO,CAAC;GAAEC,OAAO;GAAgBC,IAAI,GAAGU,EAAI;GAAgB,CAAC;EAC/D,CAAA,EAIA,kBAAC,GAAA;EAAM,WAAU;EAAW,KAAI;EAAI,WAAU;aAC5C,kBAAC,GAAA;GAAc,OAAON,GAAcU,QAAQN,EAAAA,EAAC,EAAA,IAAA,UAAA,CAAA;GAAsBF;MAClEK,EAAMI,SAAS,IACd,kBAAC,OAAA;GAAI,WAAU;aACZJ,EAAMT,KAAKc,MACV,kBAAC,GAAA,EAA6B,GAAIA,GAAAA,EAAhBA,EAAKnB,MAAK,CAAA;OAIhC,kBAAC,KAAA;GAAE,WAAU;aACX,kBAAA,GAAA,EAAA,IAAA,UAAA,CAAA"}
1
+ {"version":3,"file":"_projectId-DhLpIalx.mjs","names":["Link","useLoaderData","Box","Stack","getServiceIndex","Trans","useLingui","ContentHeader","ServiceCardProps","title","links","label","to","ServiceCard","map","RouteComponent","crumbProject","availableServices","projectId","from","t","serviceIndex","base","cards","push","storageLinks","name","length","card","component"],"sources":["../../src/client/routes/_auth/projects/$projectId/index.tsx?tsr-split=component"],"sourcesContent":["import { createFileRoute, Link, useLoaderData } from \"@tanstack/react-router\"\nimport { Box, Stack } from \"@cloudoperators/juno-ui-components\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport { Trans } from \"@lingui/react/macro\"\nimport { useLingui } from \"@lingui/react/macro\"\nimport { ContentHeader } from \"@/client/components/ContentHeader/ContentHeader\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId/\")({\n component: RouteComponent,\n})\n\ninterface ServiceCardProps {\n title: string\n links: { label: string; to: string }[]\n}\n\nfunction ServiceCard({ title, links }: ServiceCardProps) {\n return (\n <Box className=\"p-5\">\n <h3 className=\"text-theme-high mb-3 text-base font-semibold\">{title}</h3>\n <ul className=\"space-y-1.5\">\n {links.map(({ label, to }) => (\n <li key={label}>\n <Link to={to} className=\"text-theme-accent hover:text-theme-accent/80 text-sm\">\n {label}\n </Link>\n </li>\n ))}\n </ul>\n </Box>\n )\n}\n\nfunction RouteComponent() {\n const { crumbProject, availableServices, projectId } = useLoaderData({\n from: \"/_auth/projects/$projectId\",\n })\n const { t } = useLingui()\n\n const serviceIndex = getServiceIndex(availableServices ?? [])\n const base = `/projects/${projectId}`\n\n const cards: ServiceCardProps[] = []\n\n if (serviceIndex[\"image\"]?.[\"glance\"] || serviceIndex[\"compute\"]?.[\"nova\"]) {\n const links: { label: string; to: string }[] = []\n if (serviceIndex[\"image\"]?.[\"glance\"]) links.push({ label: \"Images\", to: `${base}/compute/images` })\n if (serviceIndex[\"compute\"]?.[\"nova\"]) links.push({ label: \"Flavors\", to: `${base}/compute/flavors` })\n cards.push({ title: \"Compute\", links })\n }\n\n if (serviceIndex[\"network\"]) {\n cards.push({\n title: \"Network\",\n links: [\n { label: \"Security Groups\", to: `${base}/network/securitygroups` },\n { label: \"Floating IPs\", to: `${base}/network/floatingips` },\n ],\n })\n }\n\n // Storage section - always show Ceph, optionally add Swift\n const storageLinks: { label: string; to: string }[] = []\n if (serviceIndex[\"object-store\"]?.[\"swift\"]) {\n storageLinks.push({ label: \"Swift\", to: `${base}/storage/swift/containers` })\n }\n storageLinks.push({ label: \"Ceph\", to: `${base}/storage/ceph/containers` })\n cards.push({ title: \"Storage\", links: storageLinks })\n\n // temporary as clavis is not fully GA, after GA replace with [\"pca\"]?.[\"clavis\"]\n if (serviceIndex[\"pca\"]?.[\"clavis-dev\"] || serviceIndex[\"pca\"]?.[\"clavis-beta\"]) {\n cards.push({\n title: \"Services\",\n links: [{ label: \"PCA (Clavis)\", to: `${base}/services/pca` }],\n })\n }\n\n return (\n <Stack direction=\"vertical\" gap=\"6\" className=\"pb-4\">\n <ContentHeader title={crumbProject?.name ?? t`Project`} projectId={projectId} />\n {cards.length > 0 ? (\n <div className=\"grid grid-cols-1 gap-4 sm:grid-cols-2 lg:grid-cols-3\">\n {cards.map((card) => (\n <ServiceCard key={card.title} {...card} />\n ))}\n </div>\n ) : (\n <p className=\"text-theme-light text-sm\">\n <Trans>No services available for this project.</Trans>\n </p>\n )}\n </Stack>\n )\n}\n"],"mappings":";;;;;;;AAgBA,SAASa,EAAY,EAAEJ,UAAOC,YAAyB;AACrD,QACE,kBAAC,GAAA;EAAI,WAAU;aACb,kBAAC,MAAA;GAAG,WAAU;aAAgDD;MAC9D,kBAAC,MAAA;GAAG,WAAU;aACXC,EAAMI,KAAK,EAAEH,UAAOC,YACnB,kBAAC,MAAA,EAAA,UACC,kBAAC,GAAA;IAASA;IAAI,WAAU;cACrBD;SAFIA,EAAAA,CAAAA;;;;AAWnB,SAASI,IAAAA;CACP,IAAM,EAAEC,iBAAcC,sBAAmBC,iBAAcjB,EAAc,EACnEkB,MAAM,8BACR,CAAA,EACM,EAAA,MAAA,GAAA,GAAA,MAAQb,GAAAA,EAERe,IAAejB,EAAgBa,KAAqB,EAAE,CAAA,EACtDK,IAAO,aAAaJ,KAEpBK,IAA4B,EAAE;AAEpC,KAAIF,EAAa,OAAW,UAAaA,EAAa,SAAa,MAAS;EAC1E,IAAMX,IAAyC,EAAE;AAGjDa,EAFIF,EAAa,OAAW,UAAWX,EAAMc,KAAK;GAAEb,OAAO;GAAUC,IAAI,GAAGU,EAAI;GAAkB,CAAA,EAC9FD,EAAa,SAAa,QAASX,EAAMc,KAAK;GAAEb,OAAO;GAAWC,IAAI,GAAGU,EAAI;GAAmB,CAAA,EACpGC,EAAMC,KAAK;GAAEf,OAAO;GAAWC;GAAM,CAAA;;AAGvC,CAAIW,EAAa,WACfE,EAAMC,KAAK;EACTf,OAAO;EACPC,OAAO,CACL;GAAEC,OAAO;GAAmBC,IAAI,GAAGU,EAAI;GAA0B,EACjE;GAAEX,OAAO;GAAgBC,IAAI,GAAGU,EAAI;GAAuB,CAAC;EAEhE,CAAA;CAIF,IAAMG,IAAgD,EAAE;AAexD,QAdIJ,EAAa,iBAAkB,SACjCI,EAAaD,KAAK;EAAEb,OAAO;EAASC,IAAI,GAAGU,EAAI;EAA4B,CAAA,EAE7EG,EAAaD,KAAK;EAAEb,OAAO;EAAQC,IAAI,GAAGU,EAAI;EAA2B,CAAA,EACzEC,EAAMC,KAAK;EAAEf,OAAO;EAAWC,OAAOe;EAAa,CAAA,GAG/CJ,EAAa,MAAS,iBAAiBA,EAAa,MAAS,mBAC/DE,EAAMC,KAAK;EACTf,OAAO;EACPC,OAAO,CAAC;GAAEC,OAAO;GAAgBC,IAAI,GAAGU,EAAI;GAAgB,CAAC;EAC/D,CAAA,EAIA,kBAAC,GAAA;EAAM,WAAU;EAAW,KAAI;EAAI,WAAU;aAC5C,kBAAC,GAAA;GAAc,OAAON,GAAcU,QAAQN,EAAAA,EAAC,EAAA,IAAA,UAAA,CAAA;GAAsBF;MAClEK,EAAMI,SAAS,IACd,kBAAC,OAAA;GAAI,WAAU;aACZJ,EAAMT,KAAKc,MACV,kBAAC,GAAA,EAA6B,GAAIA,GAAAA,EAAhBA,EAAKnB,MAAK,CAAA;OAIhC,kBAAC,KAAA;GAAE,WAAU;aACX,kBAAA,GAAA,EAAA,IAAA,UAAA,CAAA"}
@@ -1,7 +1,7 @@
1
1
  import { createFileRoute as e, lazyRouteComponent as t } from "@tanstack/react-router";
2
2
  var n = e("/_auth/projects/$projectId")({
3
- component: t(() => import("./_projectId-DsSVV2EA.mjs"), "component"),
4
- errorComponent: t(() => import("./_projectId-KH5si25Q.mjs"), "errorComponent"),
3
+ component: t(() => import("./_projectId-D1gGribM.mjs"), "component"),
4
+ errorComponent: t(() => import("./_projectId-OW2xkK43.mjs"), "errorComponent"),
5
5
  loader: async (e) => {
6
6
  let { context: t, params: n } = e, r = await t.trpcClient?.auth.setCurrentScope.mutate({
7
7
  type: "project",
@@ -23,4 +23,4 @@ var n = e("/_auth/projects/$projectId")({
23
23
  //#endregion
24
24
  export { n as t };
25
25
 
26
- //# sourceMappingURL=_projectId-Dha4XqX4.mjs.map
26
+ //# sourceMappingURL=_projectId-Dj_InfSc.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"_projectId-Dha4XqX4.mjs","names":["createFileRoute","Route","component","lazyRouteComponent","$$splitComponentImporter","errorComponent","$$splitErrorComponentImporter","loader","options","context","params","data","trpcClient","auth","setCurrentScope","mutate","type","projectId","availableServices","getAvailableServices","query","accountId","domain","id","crumbDomain","path","name","crumbProject","project"],"sources":["../../src/client/routes/_auth/projects/$projectId.tsx"],"sourcesContent":["import { createFileRoute, Outlet, useLoaderData } from \"@tanstack/react-router\"\nimport { AppShell, Container, Stack } from \"@cloudoperators/juno-ui-components\"\nimport { SideNavBar } from \"@/client/routes/_auth/projects/-components/SideNavBar\"\nimport { ProjectInfoBox } from \"@/client/components/ProjectView/ProjectInfoBox\"\nimport { RouteError } from \"@/client/components/Error/RouteError\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId\")({\n component: RouteComponent,\n errorComponent: ({ error }) => {\n return <RouteError error={error} />\n },\n loader: async (options) => {\n const { context, params } = options\n const data = await context.trpcClient?.auth.setCurrentScope.mutate({\n type: \"project\",\n projectId: params.projectId || \"\",\n })\n const availableServices = await context.trpcClient?.auth.getAvailableServices.query()\n\n // Extract accountId (domain id) from the rescoped token\n // This is needed for SideNavBar navigation until we refactor it\n const accountId = data?.domain?.id || \"\"\n\n return {\n trpcClient: context.trpcClient,\n crumbDomain: { path: `/projects`, name: data?.domain?.name },\n crumbProject: data?.project,\n availableServices,\n accountId, // Keep for SideNavBar compatibility\n projectId: params.projectId,\n }\n },\n})\n\nfunction RouteComponent() {\n const { availableServices, projectId, crumbProject } = useLoaderData({ from: Route.id })\n\n return (\n <AppShell\n embedded\n sideNavigation={\n <SideNavBar\n availableServices={availableServices!}\n projectId={projectId}\n projectName={crumbProject?.name || projectId}\n />\n }\n className=\"h-min-screen\"\n >\n <Container>\n <Stack direction=\"vertical\" distribution=\"start\" alignment=\"stretch\" className=\"xl:flex-row\" gap=\"6\">\n {/* Main content area */}\n <div className=\"min-w-0 flex-1\">\n <ProjectInfoBox\n projectInfo={{\n id: projectId,\n name: crumbProject?.name || projectId,\n domain: crumbProject?.domain,\n }}\n />\n <Outlet />\n </div>\n </Stack>\n </Container>\n </AppShell>\n )\n}\n"],"mappings":";AAMA,IAAaC,IAAQD,EAAgB,6BAAA,CAA8B;CACjEE,WAASC,6CAAA,YAAA;CACTE,gBAAcF,6CAAA,iBAAA;CAGdI,QAAQ,OAAOC,MAAAA;EACb,IAAM,EAAEC,YAASC,cAAWF,GACtBG,IAAO,MAAMF,EAAQG,YAAYC,KAAKC,gBAAgBC,OAAO;GACjEC,MAAM;GACNC,WAAWP,EAAOO,aAAa;GACjC,CAAA,EACMC,IAAoB,MAAMT,EAAQG,YAAYC,KAAKM,qBAAqBC,OAAAA,EAIxEC,IAAYV,GAAMW,QAAQC,MAAM;AAEtC,SAAO;GACLX,YAAYH,EAAQG;GACpBY,aAAa;IAAEC,MAAM;IAAaC,MAAMf,GAAMW,QAAQI;IAAK;GAC3DC,cAAchB,GAAMiB;GACpBV;GACAG;GACAJ,WAAWP,EAAOO;GACpB;;CAEJ,CAAA"}
1
+ {"version":3,"file":"_projectId-Dj_InfSc.mjs","names":["createFileRoute","Route","component","lazyRouteComponent","$$splitComponentImporter","errorComponent","$$splitErrorComponentImporter","loader","options","context","params","data","trpcClient","auth","setCurrentScope","mutate","type","projectId","availableServices","getAvailableServices","query","accountId","domain","id","crumbDomain","path","name","crumbProject","project"],"sources":["../../src/client/routes/_auth/projects/$projectId.tsx"],"sourcesContent":["import { createFileRoute, Outlet, useLoaderData } from \"@tanstack/react-router\"\nimport { AppShell, Container, Stack } from \"@cloudoperators/juno-ui-components\"\nimport { SideNavBar } from \"@/client/routes/_auth/projects/-components/SideNavBar\"\nimport { ProjectInfoBox } from \"@/client/components/ProjectView/ProjectInfoBox\"\nimport { RouteError } from \"@/client/components/Error/RouteError\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId\")({\n component: RouteComponent,\n errorComponent: ({ error }) => {\n return <RouteError error={error} />\n },\n loader: async (options) => {\n const { context, params } = options\n const data = await context.trpcClient?.auth.setCurrentScope.mutate({\n type: \"project\",\n projectId: params.projectId || \"\",\n })\n const availableServices = await context.trpcClient?.auth.getAvailableServices.query()\n\n // Extract accountId (domain id) from the rescoped token\n // This is needed for SideNavBar navigation until we refactor it\n const accountId = data?.domain?.id || \"\"\n\n return {\n trpcClient: context.trpcClient,\n crumbDomain: { path: `/projects`, name: data?.domain?.name },\n crumbProject: data?.project,\n availableServices,\n accountId, // Keep for SideNavBar compatibility\n projectId: params.projectId,\n }\n },\n})\n\nfunction RouteComponent() {\n const { availableServices, projectId, crumbProject } = useLoaderData({ from: Route.id })\n\n return (\n <AppShell\n embedded\n sideNavigation={\n <SideNavBar\n availableServices={availableServices!}\n projectId={projectId}\n projectName={crumbProject?.name || projectId}\n />\n }\n className=\"h-min-screen\"\n >\n <Container>\n <Stack direction=\"vertical\" distribution=\"start\" alignment=\"stretch\" className=\"xl:flex-row\" gap=\"6\">\n {/* Main content area */}\n <div className=\"min-w-0 flex-1\">\n <ProjectInfoBox\n projectInfo={{\n id: projectId,\n name: crumbProject?.name || projectId,\n domain: crumbProject?.domain,\n }}\n />\n <Outlet />\n </div>\n </Stack>\n </Container>\n </AppShell>\n )\n}\n"],"mappings":";AAMA,IAAaC,IAAQD,EAAgB,6BAAA,CAA8B;CACjEE,WAASC,6CAAA,YAAA;CACTE,gBAAcF,6CAAA,iBAAA;CAGdI,QAAQ,OAAOC,MAAAA;EACb,IAAM,EAAEC,YAASC,cAAWF,GACtBG,IAAO,MAAMF,EAAQG,YAAYC,KAAKC,gBAAgBC,OAAO;GACjEC,MAAM;GACNC,WAAWP,EAAOO,aAAa;GACjC,CAAA,EACMC,IAAoB,MAAMT,EAAQG,YAAYC,KAAKM,qBAAqBC,OAAAA,EAIxEC,IAAYV,GAAMW,QAAQC,MAAM;AAEtC,SAAO;GACLX,YAAYH,EAAQG;GACpBY,aAAa;IAAEC,MAAM;IAAaC,MAAMf,GAAMW,QAAQI;IAAK;GAC3DC,cAAchB,GAAMiB;GACpBV;GACAG;GACAJ,WAAWP,EAAOO;GACpB;;CAEJ,CAAA"}
@@ -1,8 +1,8 @@
1
- import { t as e } from "./RouteError-iP1Vd6bY.mjs";
1
+ import { t as e } from "./RouteError-pDEWC_k7.mjs";
2
2
  import { jsx as t } from "react/jsx-runtime";
3
3
  //#region src/client/routes/_auth/projects/$projectId.tsx?tsr-split=errorComponent
4
4
  var n = ({ error: n }) => /* @__PURE__ */ t(e, { error: n });
5
5
  //#endregion
6
6
  export { n as errorComponent };
7
7
 
8
- //# sourceMappingURL=_projectId-KH5si25Q.mjs.map
8
+ //# sourceMappingURL=_projectId-OW2xkK43.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"_projectId-KH5si25Q.mjs","names":["RouteError","SplitErrorComponent","error","errorComponent"],"sources":["../../src/client/routes/_auth/projects/$projectId.tsx?tsr-split=errorComponent"],"sourcesContent":["import { createFileRoute, Outlet, useLoaderData } from \"@tanstack/react-router\"\nimport { AppShell, Container, Stack } from \"@cloudoperators/juno-ui-components\"\nimport { SideNavBar } from \"@/client/routes/_auth/projects/-components/SideNavBar\"\nimport { ProjectInfoBox } from \"@/client/components/ProjectView/ProjectInfoBox\"\nimport { RouteError } from \"@/client/components/Error/RouteError\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId\")({\n component: RouteComponent,\n errorComponent: ({ error }) => {\n return <RouteError error={error} />\n },\n loader: async (options) => {\n const { context, params } = options\n const data = await context.trpcClient?.auth.setCurrentScope.mutate({\n type: \"project\",\n projectId: params.projectId || \"\",\n })\n const availableServices = await context.trpcClient?.auth.getAvailableServices.query()\n\n // Extract accountId (domain id) from the rescoped token\n // This is needed for SideNavBar navigation until we refactor it\n const accountId = data?.domain?.id || \"\"\n\n return {\n trpcClient: context.trpcClient,\n crumbDomain: { path: `/projects`, name: data?.domain?.name },\n crumbProject: data?.project,\n availableServices,\n accountId, // Keep for SideNavBar compatibility\n projectId: params.projectId,\n }\n },\n})\n\nfunction RouteComponent() {\n const { availableServices, projectId, crumbProject } = useLoaderData({ from: Route.id })\n\n return (\n <AppShell\n embedded\n sideNavigation={\n <SideNavBar\n availableServices={availableServices!}\n projectId={projectId}\n projectName={crumbProject?.name || projectId}\n />\n }\n className=\"h-min-screen\"\n >\n <Container>\n <Stack direction=\"vertical\" distribution=\"start\" alignment=\"stretch\" className=\"xl:flex-row\" gap=\"6\">\n {/* Main content area */}\n <div className=\"min-w-0 flex-1\">\n <ProjectInfoBox\n projectInfo={{\n id: projectId,\n name: crumbProject?.name || projectId,\n domain: crumbProject?.domain,\n }}\n />\n <Outlet />\n </div>\n </Stack>\n </Container>\n </AppShell>\n )\n}\n"],"mappings":";;;AAIiE,IAAAC,KAI9C,EAAEC,eACV,kBAAC,GAAA,EAAkBA,UAAAA,CAAAA"}
1
+ {"version":3,"file":"_projectId-OW2xkK43.mjs","names":["RouteError","SplitErrorComponent","error","errorComponent"],"sources":["../../src/client/routes/_auth/projects/$projectId.tsx?tsr-split=errorComponent"],"sourcesContent":["import { createFileRoute, Outlet, useLoaderData } from \"@tanstack/react-router\"\nimport { AppShell, Container, Stack } from \"@cloudoperators/juno-ui-components\"\nimport { SideNavBar } from \"@/client/routes/_auth/projects/-components/SideNavBar\"\nimport { ProjectInfoBox } from \"@/client/components/ProjectView/ProjectInfoBox\"\nimport { RouteError } from \"@/client/components/Error/RouteError\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId\")({\n component: RouteComponent,\n errorComponent: ({ error }) => {\n return <RouteError error={error} />\n },\n loader: async (options) => {\n const { context, params } = options\n const data = await context.trpcClient?.auth.setCurrentScope.mutate({\n type: \"project\",\n projectId: params.projectId || \"\",\n })\n const availableServices = await context.trpcClient?.auth.getAvailableServices.query()\n\n // Extract accountId (domain id) from the rescoped token\n // This is needed for SideNavBar navigation until we refactor it\n const accountId = data?.domain?.id || \"\"\n\n return {\n trpcClient: context.trpcClient,\n crumbDomain: { path: `/projects`, name: data?.domain?.name },\n crumbProject: data?.project,\n availableServices,\n accountId, // Keep for SideNavBar compatibility\n projectId: params.projectId,\n }\n },\n})\n\nfunction RouteComponent() {\n const { availableServices, projectId, crumbProject } = useLoaderData({ from: Route.id })\n\n return (\n <AppShell\n embedded\n sideNavigation={\n <SideNavBar\n availableServices={availableServices!}\n projectId={projectId}\n projectName={crumbProject?.name || projectId}\n />\n }\n className=\"h-min-screen\"\n >\n <Container>\n <Stack direction=\"vertical\" distribution=\"start\" alignment=\"stretch\" className=\"xl:flex-row\" gap=\"6\">\n {/* Main content area */}\n <div className=\"min-w-0 flex-1\">\n <ProjectInfoBox\n projectInfo={{\n id: projectId,\n name: crumbProject?.name || projectId,\n domain: crumbProject?.domain,\n }}\n />\n <Outlet />\n </div>\n </Stack>\n </Container>\n </AppShell>\n )\n}\n"],"mappings":";;;AAIiE,IAAAC,KAI9C,EAAEC,eACV,kBAAC,GAAA,EAAkBA,UAAAA,CAAAA"}
@@ -19,7 +19,7 @@ var i = t("/_auth/projects/$projectId/network/securitygroups/$securityGroupId/")
19
19
  return { sgTitle: n?.name || n?.id || null };
20
20
  },
21
21
  head: ({ loaderData: e }) => ({ meta: [{ title: e?.sgTitle ?? "Security Group" }] }),
22
- component: n(() => import("./_securityGroupId-O7FXfx0M.mjs"), "component"),
22
+ component: n(() => import("./_securityGroupId-gbUnd5Wv.mjs"), "component"),
23
23
  beforeLoad: async ({ context: t, params: n }) => {
24
24
  let { trpcClient: i } = t, a = e(await i?.auth.getAvailableServices.query() || []);
25
25
  if (!a.network || !a.network.neutron) throw r({
@@ -31,4 +31,4 @@ var i = t("/_auth/projects/$projectId/network/securitygroups/$securityGroupId/")
31
31
  //#endregion
32
32
  export { i as t };
33
33
 
34
- //# sourceMappingURL=_securityGroupId-DgaSqYex.mjs.map
34
+ //# sourceMappingURL=_securityGroupId-B0llWH9A.mjs.map