@cobaltcore-dev/aurora 0.2.2 → 0.3.0

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 (105) hide show
  1. package/dist/client/AuroraApp.d.ts +10 -0
  2. package/dist/client/{ContentHeader-DtBiIwRY.mjs → ContentHeader-B_PWrxbw.mjs} +17 -17
  3. package/dist/client/{ContentHeader-DtBiIwRY.mjs.map → ContentHeader-B_PWrxbw.mjs.map} +1 -1
  4. package/dist/client/{DeleteFlavorModal-rmuYIafD.mjs → DeleteFlavorModal-BkPrQKyE.mjs} +146 -146
  5. package/dist/client/{DeleteFlavorModal-rmuYIafD.mjs.map → DeleteFlavorModal-BkPrQKyE.mjs.map} +1 -1
  6. package/dist/client/{EditSecurityGroupModal-B7Sz9puM.mjs → EditSecurityGroupModal-ad4uzlt0.mjs} +14 -14
  7. package/dist/client/{EditSecurityGroupModal-B7Sz9puM.mjs.map → EditSecurityGroupModal-ad4uzlt0.mjs.map} +1 -1
  8. package/dist/client/{FloatingIpActionModals-CfRJiZqD.mjs → FloatingIpActionModals-lKMwLuL8.mjs} +51 -51
  9. package/dist/client/{FloatingIpActionModals-CfRJiZqD.mjs.map → FloatingIpActionModals-lKMwLuL8.mjs.map} +1 -1
  10. package/dist/client/{ImageToastNotifications-Cw30RXsw.mjs → ImageToastNotifications--U13YiQ_.mjs} +328 -328
  11. package/dist/client/{ImageToastNotifications-Cw30RXsw.mjs.map → ImageToastNotifications--U13YiQ_.mjs.map} +1 -1
  12. package/dist/client/{ListToolbar-DuazvsAu.mjs → ListToolbar-C5lzTrit.mjs} +56 -56
  13. package/dist/client/{ListToolbar-DuazvsAu.mjs.map → ListToolbar-C5lzTrit.mjs.map} +1 -1
  14. package/dist/client/{RouteError-Cyto623-.mjs → RouteError-iP1Vd6bY.mjs} +2 -2
  15. package/dist/client/{RouteError-Cyto623-.mjs.map → RouteError-iP1Vd6bY.mjs.map} +1 -1
  16. package/dist/client/{_flavorId-B-1fYadl.mjs → _flavorId-BaNXUJhA.mjs} +49 -49
  17. package/dist/client/_flavorId-BaNXUJhA.mjs.map +1 -0
  18. package/dist/client/{_flavorId-BYfIHIV_.mjs → _flavorId-CnO76tuy.mjs} +10 -10
  19. package/dist/client/_flavorId-CnO76tuy.mjs.map +1 -0
  20. package/dist/client/{_floatingIpId-IrnN-ozB.mjs → _floatingIpId-C8G20nNt.mjs} +2 -2
  21. package/dist/client/{_floatingIpId-IrnN-ozB.mjs.map → _floatingIpId-C8G20nNt.mjs.map} +1 -1
  22. package/dist/client/{_floatingIpId-FQ5P2qMV.mjs → _floatingIpId-DdKnjdIV.mjs} +36 -36
  23. package/dist/client/{_floatingIpId-FQ5P2qMV.mjs.map → _floatingIpId-DdKnjdIV.mjs.map} +1 -1
  24. package/dist/client/{_imageId-Tx_9bqEc.mjs → _imageId-DdSbxFqG.mjs} +135 -135
  25. package/dist/client/_imageId-DdSbxFqG.mjs.map +1 -0
  26. package/dist/client/{_pcaId-CFuKY82d.mjs → _pcaId-DFkYJEb5.mjs} +136 -136
  27. package/dist/client/{_pcaId-CFuKY82d.mjs.map → _pcaId-DFkYJEb5.mjs.map} +1 -1
  28. package/dist/client/{_pcaId-Bck7S7gJ.mjs → _pcaId-DX_S-3hE.mjs} +2 -2
  29. package/dist/client/{_pcaId-Bck7S7gJ.mjs.map → _pcaId-DX_S-3hE.mjs.map} +1 -1
  30. package/dist/client/{_projectId-PSpuCKO7.mjs → _projectId-DYrcZ3E3.mjs} +9 -9
  31. package/dist/client/{_projectId-PSpuCKO7.mjs.map → _projectId-DYrcZ3E3.mjs.map} +1 -1
  32. package/dist/client/{_projectId-B1VjDd0Z.mjs → _projectId-Dha4XqX4.mjs} +3 -3
  33. package/dist/client/{_projectId-B1VjDd0Z.mjs.map → _projectId-Dha4XqX4.mjs.map} +1 -1
  34. package/dist/client/_projectId-DsSVV2EA.mjs +316 -0
  35. package/dist/client/_projectId-DsSVV2EA.mjs.map +1 -0
  36. package/dist/client/{_projectId-Pxp-RXK4.mjs → _projectId-KH5si25Q.mjs} +2 -2
  37. package/dist/client/{_projectId-Pxp-RXK4.mjs.map → _projectId-KH5si25Q.mjs.map} +1 -1
  38. package/dist/client/{_securityGroupId-VV2lUcGQ.mjs → _securityGroupId-DgaSqYex.mjs} +2 -2
  39. package/dist/client/{_securityGroupId-VV2lUcGQ.mjs.map → _securityGroupId-DgaSqYex.mjs.map} +1 -1
  40. package/dist/client/{_securityGroupId-Dqi6ddw4.mjs → _securityGroupId-O7FXfx0M.mjs} +306 -306
  41. package/dist/client/{_securityGroupId-Dqi6ddw4.mjs.map → _securityGroupId-O7FXfx0M.mjs.map} +1 -1
  42. package/dist/client/{about-B2AzqxFI.mjs → about-DN8n8sN8.mjs} +8 -8
  43. package/dist/client/{about-B2AzqxFI.mjs.map → about-DN8n8sN8.mjs.map} +1 -1
  44. package/dist/client/{build-Cf7iWbpH.mjs → build-CZRvXrAL.mjs} +1047 -1042
  45. package/dist/client/{build-Cf7iWbpH.mjs.map → build-CZRvXrAL.mjs.map} +1 -1
  46. package/dist/client/{constants-CCgR6fKI.mjs → constants-akdIBeTX.mjs} +6 -6
  47. package/dist/client/{constants-CCgR6fKI.mjs.map → constants-akdIBeTX.mjs.map} +1 -1
  48. package/dist/client/{containers-BWERuY0O.mjs → containers-BE2QiLBs.mjs} +748 -748
  49. package/dist/client/{containers-BWERuY0O.mjs.map → containers-BE2QiLBs.mjs.map} +1 -1
  50. package/dist/client/containers-Cs5vOeR2.mjs.map +1 -1
  51. package/dist/client/{containers-DovytjVP.mjs → containers-DmwhE9Uz.mjs} +6 -6
  52. package/dist/client/containers-DmwhE9Uz.mjs.map +1 -0
  53. package/dist/client/flavors-BXPYAFyQ.mjs.map +1 -1
  54. package/dist/client/{flavors-CUiALHcB.mjs → flavors-CY7A6--v.mjs} +2 -2
  55. package/dist/client/{flavors-CUiALHcB.mjs.map → flavors-CY7A6--v.mjs.map} +1 -1
  56. package/dist/client/{flavors-Bovz-I2U.mjs → flavors-pEcGkCut.mjs} +150 -150
  57. package/dist/client/{flavors-Bovz-I2U.mjs.map → flavors-pEcGkCut.mjs.map} +1 -1
  58. package/dist/client/{floatingips-BrjDiY2t.mjs → floatingips-BtL4d4m-.mjs} +118 -118
  59. package/dist/client/{floatingips-BrjDiY2t.mjs.map → floatingips-BtL4d4m-.mjs.map} +1 -1
  60. package/dist/client/{images-DaaCUXMI.mjs → images-C19gCFSy.mjs} +389 -389
  61. package/dist/client/{images-DaaCUXMI.mjs.map → images-C19gCFSy.mjs.map} +1 -1
  62. package/dist/client/images-DM9I8G0p.mjs.map +1 -1
  63. package/dist/client/{images-BZP3pBqj.mjs → images-I9gQfRa7.mjs} +2 -2
  64. package/dist/client/{images-BZP3pBqj.mjs.map → images-I9gQfRa7.mjs.map} +1 -1
  65. package/dist/client/index.d.ts +2 -1
  66. package/dist/client/index.js +416 -418
  67. package/dist/client/index.js.map +1 -1
  68. package/dist/client/objects-B4yrYf_a.mjs.map +1 -1
  69. package/dist/client/{objects-D4zBka5e.mjs → objects-BrYe_RaJ.mjs} +6 -6
  70. package/dist/client/objects-BrYe_RaJ.mjs.map +1 -0
  71. package/dist/client/{objects-Cw4Vu01M.mjs → objects-DOYFFn3Y.mjs} +1013 -1013
  72. package/dist/client/{objects-Cw4Vu01M.mjs.map → objects-DOYFFn3Y.mjs.map} +1 -1
  73. package/dist/client/{overview-2J54-loz.mjs → overview-BjRSFSBh.mjs} +2 -2
  74. package/dist/client/{overview-2J54-loz.mjs.map → overview-BjRSFSBh.mjs.map} +1 -1
  75. package/dist/client/{overview-BnmukbFh.mjs → overview-C4gjtc2q.mjs} +7 -7
  76. package/dist/client/{overview-BnmukbFh.mjs.map → overview-C4gjtc2q.mjs.map} +1 -1
  77. package/dist/client/{overview-D0AAvsmL.mjs → overview-CmQkJ4Hh.mjs} +2 -2
  78. package/dist/client/{overview-D0AAvsmL.mjs.map → overview-CmQkJ4Hh.mjs.map} +1 -1
  79. package/dist/client/{overview-BtIXpYBo.mjs → overview-DTLIAKkJ.mjs} +2 -2
  80. package/dist/client/{overview-BtIXpYBo.mjs.map → overview-DTLIAKkJ.mjs.map} +1 -1
  81. package/dist/client/{pca-V2aaOxZA.mjs → pca-CYFJxSZ2.mjs} +70 -70
  82. package/dist/client/{pca-V2aaOxZA.mjs.map → pca-CYFJxSZ2.mjs.map} +1 -1
  83. package/dist/client/{pca-BqZycwCu.mjs → pca-DzixU9Dl.mjs} +2 -2
  84. package/dist/client/{pca-BqZycwCu.mjs.map → pca-DzixU9Dl.mjs.map} +1 -1
  85. package/dist/client/projects-BsN4bvU2.mjs.map +1 -1
  86. package/dist/client/{projects-0feOw_b6.mjs → projects-BtyjXGq2.mjs} +2 -2
  87. package/dist/client/projects-BtyjXGq2.mjs.map +1 -0
  88. package/dist/client/projects-D3hOC1cy.mjs +95 -0
  89. package/dist/client/projects-D3hOC1cy.mjs.map +1 -0
  90. package/dist/client/{projects-jyIHL6DE.mjs → projects-DwVawmll.mjs} +2 -2
  91. package/dist/client/projects-DwVawmll.mjs.map +1 -0
  92. package/dist/client/{securitygroups-B4MkSBtI.mjs → securitygroups-DMCIDHQS.mjs} +114 -114
  93. package/dist/client/{securitygroups-B4MkSBtI.mjs.map → securitygroups-DMCIDHQS.mjs.map} +1 -1
  94. package/package.json +3 -3
  95. package/dist/client/_flavorId-B-1fYadl.mjs.map +0 -1
  96. package/dist/client/_flavorId-BYfIHIV_.mjs.map +0 -1
  97. package/dist/client/_imageId-Tx_9bqEc.mjs.map +0 -1
  98. package/dist/client/_projectId-Bs4W9hos.mjs +0 -283
  99. package/dist/client/_projectId-Bs4W9hos.mjs.map +0 -1
  100. package/dist/client/containers-DovytjVP.mjs.map +0 -1
  101. package/dist/client/objects-D4zBka5e.mjs.map +0 -1
  102. package/dist/client/projects-0feOw_b6.mjs.map +0 -1
  103. package/dist/client/projects-C1IYOvFQ.mjs +0 -144
  104. package/dist/client/projects-C1IYOvFQ.mjs.map +0 -1
  105. package/dist/client/projects-jyIHL6DE.mjs.map +0 -1
@@ -1,283 +0,0 @@
1
- import { $ as e, E as t, G as n, H as r, I as i, et as a, g as o, k as s } from "./build-Cf7iWbpH.mjs";
2
- import { t as c } from "./_projectId-B1VjDd0Z.mjs";
3
- import { t as l } from "./helpers--JWXi40U.mjs";
4
- import { Fragment as u, jsx as d, jsxs as f } from "react/jsx-runtime";
5
- import { useMemo as p, useState as m } from "react";
6
- import { Outlet as h, useLoaderData as g, useMatches as _, useNavigate as v, useParams as y } from "@tanstack/react-router";
7
- import { useLingui as b } from "@lingui/react";
8
- import { z as x } from "zod";
9
- var S = x.object({
10
- labelKey: x.enum([
11
- "Compute",
12
- "Network",
13
- "Storage",
14
- "Services",
15
- "Images",
16
- "Flavors",
17
- "Security Groups",
18
- "Floating IPs",
19
- "PCA (Clavis)"
20
- ]).optional(),
21
- to: x.string().optional(),
22
- useParamAsLabel: x.string().optional()
23
- }), C = x.object({
24
- section: x.string(),
25
- service: x.string().optional(),
26
- isDetail: x.boolean().optional(),
27
- crumb: S.optional(),
28
- sectionCrumb: S.optional()
29
- });
30
- function w(e) {
31
- return C.safeParse(e).success;
32
- }
33
- //#endregion
34
- //#region src/client/routes/_auth/projects/-components/SideNavBar.tsx
35
- var T = ({ projectId: e, projectName: t, availableServices: n }) => {
36
- let { i18n: i, _: o } = b(), c = v(), p = _(), { provider: h } = y({ strict: !1 }), [g, x] = m({
37
- compute: !0,
38
- network: !0,
39
- storage: !0,
40
- services: !0
41
- }), S = l(n), C = [...p].reverse().find((e) => w(e.staticData)), T = C && w(C.staticData) ? C.staticData : void 0, E = T?.section ?? null, D = T?.service ?? null, O = [...S.image?.glance ? [{
42
- service: "images",
43
- label: i._({ id: "an5hVd" }),
44
- to: "/projects/$projectId/compute/images",
45
- params: { projectId: e }
46
- }] : [], ...S?.compute?.nova ? [{
47
- service: "flavors",
48
- label: i._({ id: "neiJm0" }),
49
- to: "/projects/$projectId/compute/flavors",
50
- params: { projectId: e }
51
- }] : []], k = [...S.network ? [{
52
- service: "securitygroups",
53
- label: i._({ id: "4opp4r" }),
54
- to: "/projects/$projectId/network/securitygroups",
55
- params: { projectId: e }
56
- }, {
57
- service: "floatingips",
58
- label: i._({ id: "u77/s4" }),
59
- to: "/projects/$projectId/network/floatingips",
60
- params: { projectId: e }
61
- }] : []], A = [...S?.["object-store"]?.swift ? [{
62
- service: "containers",
63
- label: i._({ id: "+OEi73" }),
64
- to: "/projects/$projectId/storage/$provider/containers",
65
- params: {
66
- projectId: e,
67
- provider: "swift"
68
- }
69
- }] : [], {
70
- service: "ceph-containers",
71
- label: i._({ id: "KhNDX4" }),
72
- to: "/projects/$projectId/storage/$provider/containers",
73
- params: {
74
- projectId: e,
75
- provider: "ceph"
76
- }
77
- }], j = [...S.pca?.["clavis-beta"] || S.pca?.["clavis-dev"] ? [{
78
- service: "pca",
79
- label: i._({ id: "miy5mb" }),
80
- to: "/projects/$projectId/services/pca",
81
- params: { projectId: e }
82
- }] : []], M = E === null, N = (e) => g[e] || E === e;
83
- return /* @__PURE__ */ d(r, {
84
- ariaLabel: "Project Side Navigation",
85
- onActiveItemChange: () => {},
86
- children: /* @__PURE__ */ d(s, { children: /* @__PURE__ */ f(u, { children: [
87
- /* @__PURE__ */ d(a, {
88
- icon: "home",
89
- label: t,
90
- onClick: () => c({
91
- to: "/projects/$projectId",
92
- params: { projectId: e }
93
- }),
94
- selected: M
95
- }),
96
- /* @__PURE__ */ d(a, {
97
- label: i._({ id: "rp0Bd0" }),
98
- open: N("compute"),
99
- onClick: () => x((e) => ({
100
- ...e,
101
- compute: !e.compute
102
- })),
103
- children: O.map(({ service: e, label: t, to: n, params: r }) => /* @__PURE__ */ d(a, {
104
- onClick: () => c({
105
- to: n,
106
- params: r
107
- }),
108
- label: t,
109
- selected: E === "compute" && D === e
110
- }, t))
111
- }),
112
- k.length > 0 && /* @__PURE__ */ d(a, {
113
- label: i._({ id: "OR475H" }),
114
- open: N("network"),
115
- onClick: () => x((e) => ({
116
- ...e,
117
- network: !e.network
118
- })),
119
- children: k.map(({ service: e, label: t, to: n, params: r }) => /* @__PURE__ */ d(a, {
120
- onClick: () => c({
121
- to: n,
122
- params: r
123
- }),
124
- label: t,
125
- selected: E === "network" && D === e
126
- }, t))
127
- }),
128
- A.length > 0 && /* @__PURE__ */ d(a, {
129
- label: i._({ id: "BrrIs8" }),
130
- open: N("storage"),
131
- onClick: () => x((e) => ({
132
- ...e,
133
- storage: !e.storage
134
- })),
135
- children: A.map(({ service: e, label: t, to: n, params: r }) => /* @__PURE__ */ d(a, {
136
- onClick: () => c({
137
- to: n,
138
- params: r
139
- }),
140
- label: t,
141
- selected: E === "storage" && D === "containers" ? r.provider === h : D === e
142
- }, t))
143
- }),
144
- j.length > 0 && /* @__PURE__ */ d(a, {
145
- label: i._({ id: "MILoeL" }),
146
- open: N("services"),
147
- onClick: () => x((e) => ({
148
- ...e,
149
- services: !e.services
150
- })),
151
- children: j.map(({ service: e, label: t, to: n, params: r }) => /* @__PURE__ */ d(a, {
152
- onClick: () => c({
153
- to: n,
154
- params: r
155
- }),
156
- label: t,
157
- selected: E === "services" && D === e
158
- }, t))
159
- })
160
- ] }) })
161
- });
162
- };
163
- //#endregion
164
- //#region src/client/components/ProjectView/ProjectInfoBox.tsx
165
- function E({ projectInfo: e }) {
166
- let { i18n: r, _: i } = b(), a = v(), o = _(), { projectId: s } = y({ strict: !1 });
167
- return /* @__PURE__ */ d(n, {
168
- className: "mt-8 mb-4",
169
- children: p(() => {
170
- let t = {
171
- Compute: r._({ id: "rp0Bd0" }),
172
- Network: r._({ id: "OR475H" }),
173
- Storage: r._({ id: "BrrIs8" }),
174
- Services: r._({ id: "MILoeL" }),
175
- Images: r._({ id: "an5hVd" }),
176
- Flavors: r._({ id: "neiJm0" }),
177
- "Security Groups": r._({ id: "4opp4r" }),
178
- "Floating IPs": r._({ id: "u77/s4" }),
179
- "PCA (Clavis)": r._({ id: "miy5mb" })
180
- }, n = (e) => e === "swift" ? r._({ id: "+OEi73" }) : e === "ceph" ? r._({ id: "KhNDX4" }) : r._({ id: "BrrIs8" }), i = [];
181
- i.push({
182
- icon: "home",
183
- label: r._({ id: "i0qMbr" }),
184
- onClick: () => a({ to: "/projects" })
185
- }), e.domain?.name && i.push({ label: e.domain.name }), i.push({
186
- label: e.name,
187
- onClick: () => a({
188
- to: "/projects/$projectId",
189
- params: { projectId: s }
190
- })
191
- });
192
- let c = o.filter((e) => e.routeId !== "/_auth/projects/$projectId" && e.routeId.startsWith("/_auth/projects/$projectId")), l = c[c.length - 1];
193
- if (!l) return i;
194
- let u = w(l.staticData) ? l.staticData : void 0;
195
- if (!u) return i;
196
- let d = l.params;
197
- if (u.sectionCrumb) {
198
- let { labelKey: e, to: n } = u.sectionCrumb, r = e ? t[e] : void 0, o = !u.crumb;
199
- i.push(n ? {
200
- label: r,
201
- onClick: () => a({
202
- to: n,
203
- params: d
204
- })
205
- } : {
206
- label: r,
207
- active: o
208
- });
209
- }
210
- if (u.crumb) {
211
- let { labelKey: e, to: r, useParamAsLabel: o } = u.crumb, s = o ? n(d[o]) : e ? t[e] : void 0;
212
- if (u.isDetail) {
213
- i.push({
214
- label: s,
215
- onClick: () => a({
216
- to: r,
217
- params: d
218
- })
219
- });
220
- let e = l.meta?.find((e) => e != null && "title" in e)?.title;
221
- e && i.push({
222
- label: e,
223
- active: !0
224
- });
225
- } else i.push(r ? {
226
- label: s,
227
- onClick: () => a({
228
- to: r,
229
- params: d
230
- })
231
- } : {
232
- label: s,
233
- active: !0
234
- });
235
- }
236
- return i;
237
- }, [
238
- o,
239
- e,
240
- s,
241
- a,
242
- i
243
- ]).map((e, n) => /* @__PURE__ */ d(t, {
244
- label: e.label,
245
- icon: e.icon,
246
- onClick: e.onClick,
247
- active: e.active
248
- }, n))
249
- });
250
- }
251
- //#endregion
252
- //#region src/client/routes/_auth/projects/$projectId.tsx?tsr-split=component
253
- function D() {
254
- let { availableServices: t, projectId: n, crumbProject: r } = g({ from: c.id });
255
- return /* @__PURE__ */ d(o, {
256
- embedded: !0,
257
- sideNavigation: /* @__PURE__ */ d(T, {
258
- availableServices: t,
259
- projectId: n,
260
- projectName: r?.name || n
261
- }),
262
- className: "h-min-screen",
263
- children: /* @__PURE__ */ d(i, { children: /* @__PURE__ */ d(e, {
264
- direction: "vertical",
265
- distribution: "start",
266
- alignment: "stretch",
267
- className: "xl:flex-row",
268
- gap: "6",
269
- children: /* @__PURE__ */ f("div", {
270
- className: "min-w-0 flex-1",
271
- children: [/* @__PURE__ */ d(E, { projectInfo: {
272
- id: n,
273
- name: r?.name || n,
274
- domain: r?.domain
275
- } }), /* @__PURE__ */ d(h, {})]
276
- })
277
- }) })
278
- });
279
- }
280
- //#endregion
281
- export { D as component };
282
-
283
- //# sourceMappingURL=_projectId-Bs4W9hos.mjs.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"_projectId-Bs4W9hos.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","useNavigate","useMatches","useParams","useState","getServiceIndex","SideNavigation","SideNavigationList","SideNavigationItem","isRouteInfo","SideNavBar","projectId","projectName","availableServices","useLingui","navigate","matches","provider","strict","openSections","setOpenSections","compute","network","storage","services","serviceIndex","activeMatch","reverse","find","m","staticData","activeRouteInfo","undefined","activeSection","section","activeService","service","computeServices","label","t","to","params","networkServices","storageServices","pcaServices","clavisServices","isOverviewActive","isOpen","ariaLabel","onActiveItemChange","icon","onClick","selected","open","prev","map","length","isStorageContainers","isSelected","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/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 { useNavigate, useMatches, useParams } from \"@tanstack/react-router\"\nimport { useState } 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\"\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\n const [openSections, setOpenSections] = useState({ compute: true, network: true, storage: true, services: true })\n\n const serviceIndex = getServiceIndex(availableServices)\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 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 const isOverviewActive = activeSection === null\n const isOpen = (section: string) => openSections[section as keyof typeof openSections] || activeSection === section\n\n return (\n <SideNavigation ariaLabel=\"Project Side Navigation\" onActiveItemChange={() => {}}>\n <SideNavigationList>\n <>\n <SideNavigationItem\n icon=\"home\"\n label={projectName}\n onClick={() => navigate({ to: \"/projects/$projectId\", params: { projectId } })}\n selected={isOverviewActive}\n />\n <SideNavigationItem\n label={t`Compute`}\n open={isOpen(\"compute\")}\n onClick={() => setOpenSections((prev) => ({ ...prev, compute: !prev.compute }))}\n >\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\n {networkServices.length > 0 && (\n <SideNavigationItem\n label={t`Network`}\n open={isOpen(\"network\")}\n onClick={() => setOpenSections((prev) => ({ ...prev, network: !prev.network }))}\n >\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 )}\n\n {storageServices.length > 0 && (\n <SideNavigationItem\n label={t`Storage`}\n open={isOpen(\"storage\")}\n onClick={() => setOpenSections((prev) => ({ ...prev, storage: !prev.storage }))}\n >\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 )}\n\n {clavisServices.length > 0 && (\n <SideNavigationItem\n label={t`Services`}\n open={isOpen(\"services\")}\n onClick={() => setOpenSections((prev) => ({ ...prev, services: !prev.services }))}\n >\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 )}\n </>\n </SideNavigationList>\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 +0,0 @@
1
- {"version":3,"file":"containers-DovytjVP.mjs","names":["createFileRoute","redirect","z","getServiceIndex","checkServiceAvailability","availableServices","type","name","params","projectId","provider","serviceIndex","to","hasSwift","Boolean","hasCeph","cephFallbackEnabled","hasEffectiveCeph","fallbackProvider","containersSearchSchema","object","sortBy","enum","optional","sortDirection","search","string","Route","staticData","section","service","sectionCrumb","labelKey","crumb","useParamAsLabel","RouteInfo","validateSearch","head","match","meta","title","component","lazyRouteComponent","$$splitComponentImporter","notFoundComponent","$$splitNotFoundComponentImporter","loader","context","trpcClient","auth","getAvailableServices","query","client","beforeLoad"],"sources":["../../src/client/routes/_auth/projects/$projectId/storage/$provider/containers/index.tsx"],"sourcesContent":["import { createFileRoute, redirect, useParams } from \"@tanstack/react-router\"\nimport { z } from \"zod\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport { ErrorBoundary } from \"react-error-boundary\"\nimport { SwiftContainers } from \"../../-components/Swift/Containers\"\nimport { CephContainers } from \"../../-components/Ceph/Containers\"\nimport { Trans, useLingui } from \"@lingui/react/macro\"\nimport type { RouteInfo } from \"@/client/routes/routeInfo\"\nimport { ContentHeader } from \"@/client/components/ContentHeader/ContentHeader\"\n\nexport const checkServiceAvailability = (\n availableServices: {\n type: string\n name: string\n }[],\n params: {\n projectId: string\n provider: string\n }\n) => {\n const { provider, projectId } = params\n\n const serviceIndex = getServiceIndex(availableServices)\n\n // Redirect to the \"Projects Overview\" page if no storage services available\n if (!serviceIndex[\"object-store\"]) {\n throw redirect({\n to: \"/projects/$projectId/compute/overview\",\n params: { projectId },\n })\n }\n\n // Check provider availability\n const hasSwift = Boolean(serviceIndex[\"object-store\"][\"swift\"])\n const hasCeph = Boolean(serviceIndex[\"object-store\"][\"ceph\"])\n\n // TEMPORARY: Allow Ceph access even if not in catalog (relies on env config)\n // TODO: Properly register Ceph in OpenStack service catalog\n const cephFallbackEnabled = true // Set to false once Ceph is in catalog\n\n // Effective availability includes fallback flag for Ceph\n const hasEffectiveCeph = hasCeph || cephFallbackEnabled\n const fallbackProvider = hasSwift ? \"swift\" : hasEffectiveCeph ? \"ceph\" : null\n\n if (provider !== \"swift\" && provider !== \"ceph\") {\n if (!fallbackProvider) {\n throw redirect({\n to: \"/projects/$projectId/compute/overview\",\n params: { projectId },\n })\n }\n throw redirect({\n to: \"/projects/$projectId/storage/$provider/containers\",\n params: { ...params, provider: fallbackProvider },\n })\n }\n\n if (provider === \"swift\" && !hasSwift) {\n if (!hasEffectiveCeph) {\n throw redirect({\n to: \"/projects/$projectId/compute/overview\",\n params: { projectId },\n })\n }\n\n throw redirect({\n to: \"/projects/$projectId/storage/$provider/containers\",\n params: { ...params, provider: \"ceph\" },\n })\n }\n\n if (provider === \"ceph\" && !hasEffectiveCeph) {\n if (!hasSwift) {\n throw redirect({\n to: \"/projects/$projectId/compute/overview\",\n params: { projectId },\n })\n }\n\n throw redirect({\n to: \"/projects/$projectId/storage/$provider/containers\",\n params: { ...params, provider: \"swift\" },\n })\n }\n}\n\n// Search params schema\n// - sortBy: active sort column — persisted for deep links and back navigation\n// - sortDirection: \"asc\" | \"desc\" — persisted alongside sortBy\n// - search: active filter string — persisted so deep links preserve the current search\nconst containersSearchSchema = z.object({\n sortBy: z.enum([\"name\", \"count\", \"bytes\", \"last_modified\"]).optional(),\n sortDirection: z.enum([\"asc\", \"desc\"]).optional(),\n search: z.string().optional(),\n})\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId/storage/$provider/containers/\")({\n staticData: {\n section: \"storage\",\n service: \"containers\",\n sectionCrumb: { labelKey: \"Storage\" },\n crumb: { useParamAsLabel: \"provider\" },\n } satisfies RouteInfo,\n validateSearch: containersSearchSchema,\n head: ({ match }) => ({\n meta: [\n {\n title:\n match.params.provider === \"swift\"\n ? \"Object Storage (Swift)\"\n : match.params.provider === \"ceph\"\n ? \"Object Storage (Ceph)\"\n : \"Storage Overview\",\n },\n ],\n }),\n component: () => {\n return <StorageDashboard />\n },\n notFoundComponent: () => {\n return <p>Storage service not found</p>\n },\n loader: async ({ context }) => {\n const { trpcClient } = context\n const availableServices = await trpcClient?.auth.getAvailableServices.query()\n\n return {\n client: trpcClient,\n availableServices,\n }\n },\n beforeLoad: async ({ context, params }) => {\n const { trpcClient } = context\n const availableServices = await trpcClient?.auth.getAvailableServices.query()\n checkServiceAvailability(availableServices!, params)\n },\n})\n\nfunction StorageDashboard() {\n const { project, provider } = useParams({\n from: \"/_auth/projects/$projectId/storage/$provider/containers/\",\n select: (params) => {\n return { project: params.projectId, provider: params.provider }\n },\n })\n\n const { t } = useLingui()\n\n let pageTitle: string\n switch (provider) {\n case \"swift\":\n pageTitle = t`Object Storage (Swift)`\n break\n case \"ceph\":\n pageTitle = t`Object Storage (Ceph)`\n break\n default:\n pageTitle = t`Storage Overview`\n }\n\n return (\n <div>\n <ContentHeader title={pageTitle} projectId={project} />\n {project ? (\n <ErrorBoundary\n fallback={\n <div className=\"p-4 text-center\">\n <Trans>Error loading component</Trans>\n </div>\n }\n >\n {(() => {\n switch (provider) {\n case \"swift\":\n return <SwiftContainers />\n case \"ceph\":\n return <CephContainers />\n default:\n return <div>Storage Overview Page</div> // replace when available\n }\n })()}\n </ErrorBoundary>\n ) : (\n <div className=\"p-4 text-center\">\n <Trans>No project selected</Trans>\n </div>\n )}\n </div>\n )\n}\n"],"mappings":";;;;kGAUaI,KACXC,GAIAG,MAAAA;CAKA,IAAM,EAAEE,aAAUD,iBAAcD,GAE1BG,IAAeR,EAAgBE,EAAAA;AAGrC,KAAI,CAACM,EAAa,gBAChB,OAAMV,EAAS;EACbW,IAAI;EACJJ,QAAQ,EAAEC,cAAU;EACtB,CAAA;CAIF,IAAMI,IAAWC,EAAQH,EAAa,gBAAgB,OAQhDM,IAPUH,EAAQH,EAAa,gBAAgB,QAIzB,IAItBO,IAAmBL,IAAW,UAAUI,IAAmB,SAAS;AAE1E,KAAIP,MAAa,WAAWA,MAAa,OAOvC,OAAMT,EANDiB,IAMU;EACbN,IAAI;EACJJ,QAAQ;GAAE,GAAGA;GAAQE,UAAUQ;GAAiB;EAClD,GARiB;EACbN,IAAI;EACJJ,QAAQ,EAAEC,cAAU;EACtB,CAAA;AAQJ,KAAIC,MAAa,WAAW,CAACG,EAQ3B,OAAMZ,EAPDgB,IAOU;EACbL,IAAI;EACJJ,QAAQ;GAAE,GAAGA;GAAQE,UAAU;GAAO;EACxC,GATiB;EACbE,IAAI;EACJJ,QAAQ,EAAEC,cAAU;EACtB,CAAA;AASJ,KAAIC,MAAa,UAAU,CAACO,EAQ1B,OAAMhB,EAPDY,IAOU;EACbD,IAAI;EACJJ,QAAQ;GAAE,GAAGA;GAAQE,UAAU;GAAQ;EACzC,GATiB;EACbE,IAAI;EACJJ,QAAQ,EAAEC,cAAU;EACtB,CAAA;GAcAU,IAAyBjB,EAAEkB,OAAO;CACtCC,QAAQnB,EAAEoB,KAAK;EAAC;EAAQ;EAAS;EAAS;EAAgB,CAAA,CAAEC,UAAQ;CACpEC,eAAetB,EAAEoB,KAAK,CAAC,OAAO,OAAO,CAAA,CAAEC,UAAQ;CAC/CE,QAAQvB,EAAEwB,QAAM,CAAGH,UAAQ;CAC7B,CAAA,EAEaI,IAAQ3B,EAAgB,2DAAA,CAA4D;CAC/F4B,YAAY;EACVC,SAAS;EACTC,SAAS;EACTC,cAAc,EAAEC,UAAU,WAAU;EACpCC,OAAO,EAAEC,iBAAiB,YAAW;EACvC;CACAE,gBAAgBjB;CAChBkB,OAAO,EAAEC,gBAAa,EACpBC,MAAM,CACJ,EACEC,OACEF,EAAM9B,OAAOE,aAAa,UACtB,2BACA4B,EAAM9B,OAAOE,aAAa,SACxB,0BACA,oBACV,CAAC,EAEL;CACA+B,WAASC,EAAAC,GAAA,YAAA;CAGTC,mBAAiBF,EAAAG,GAAA,oBAAA;CAGjBC,QAAQ,OAAO,EAAEC,iBAAS;EACxB,IAAM,EAAEC,kBAAeD;AAGvB,SAAO;GACLK,QAAQJ;GACR3C,mBAJwB,MAAM2C,GAAYC,KAAKC,qBAAqBC,OAAAA;GAKtE;;CAEFE,YAAY,OAAO,EAAEN,YAASvC,gBAAQ;EACpC,IAAM,EAAEwC,kBAAeD;AAEvB3C,IAD0B,MAAM4C,GAAYC,KAAKC,qBAAqBC,OAAAA,EACzB3C,EAAAA;;CAEjD,CAAA"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"objects-D4zBka5e.mjs","names":["createFileRoute","redirect","getServiceIndex","z","checkServiceAvailability","availableServices","type","name","params","projectId","provider","containerName","serviceIndex","to","hasSwift","Boolean","hasCeph","cephFallbackEnabled","hasEffectiveCeph","fallbackProvider","objectsSearchSchema","object","prefix","string","optional","sortBy","enum","sortDirection","search","Route","staticData","section","service","isDetail","sectionCrumb","labelKey","crumb","useParamAsLabel","RouteInfo","validateSearch","head","match","meta","title","component","lazyRouteComponent","$$splitComponentImporter","notFoundComponent","$$splitNotFoundComponentImporter","loader","context","trpcClient","auth","getAvailableServices","query","client","beforeLoad"],"sources":["../../src/client/routes/_auth/projects/$projectId/storage/$provider/containers/$containerName/objects/index.tsx"],"sourcesContent":["import { createFileRoute, redirect, useParams } from \"@tanstack/react-router\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport { ErrorBoundary } from \"react-error-boundary\"\nimport { Trans } from \"@lingui/react/macro\"\nimport { SwiftObjects } from \"../../../../-components/Swift/Objects\"\nimport { ObjectBrowserView } from \"../../../../-components/Ceph/Objects\"\nimport { z } from \"zod\"\nimport type { RouteInfo } from \"@/client/routes/routeInfo\"\n\nexport const checkServiceAvailability = (\n availableServices: {\n type: string\n name: string\n }[],\n params: {\n projectId: string\n provider: string\n containerName: string\n }\n) => {\n const { provider, projectId, containerName } = params\n\n const serviceIndex = getServiceIndex(availableServices)\n\n // Redirect to the \"Projects Overview\" page if no storage services available\n if (!serviceIndex[\"object-store\"]) {\n throw redirect({\n to: \"/projects/$projectId/compute/overview\",\n params: { projectId },\n })\n }\n\n // Check provider availability\n const hasSwift = Boolean(serviceIndex[\"object-store\"][\"swift\"])\n const hasCeph = Boolean(serviceIndex[\"object-store\"][\"ceph\"])\n\n // TEMPORARY: Allow Ceph access even if not in catalog (relies on env config)\n // TODO: Properly register Ceph in OpenStack service catalog\n const cephFallbackEnabled = true // Set to false once Ceph is in catalog\n\n // Effective availability includes fallback flag for Ceph\n const hasEffectiveCeph = hasCeph || cephFallbackEnabled\n const fallbackProvider = hasSwift ? \"swift\" : hasEffectiveCeph ? \"ceph\" : null\n\n if (provider !== \"swift\" && provider !== \"ceph\") {\n if (!fallbackProvider) {\n throw redirect({\n to: \"/projects/$projectId/compute/overview\",\n params: { projectId },\n })\n }\n throw redirect({\n to: \"/projects/$projectId/storage/$provider/containers/$containerName/objects\",\n params: { projectId, provider: fallbackProvider, containerName },\n })\n }\n\n if (provider === \"swift\" && !hasSwift) {\n if (!hasEffectiveCeph) {\n throw redirect({\n to: \"/projects/$projectId/compute/overview\",\n params: { projectId },\n })\n }\n\n throw redirect({\n to: \"/projects/$projectId/storage/$provider/containers/$containerName/objects\",\n params: { projectId, provider: \"ceph\", containerName },\n })\n }\n\n if (provider === \"ceph\" && !hasEffectiveCeph) {\n if (!hasSwift) {\n throw redirect({\n to: \"/projects/$projectId/compute/overview\",\n params: { projectId },\n })\n }\n\n throw redirect({\n to: \"/projects/$projectId/storage/$provider/containers/$containerName/objects\",\n params: { projectId, provider: \"swift\", containerName },\n })\n }\n}\n\n// Search params schema\n// - prefix: base64-encoded current folder path, safe to carry \"/\" chars in the URL\n// - sortBy: active sort column key — persisted so deep links and back navigation restore sort state\n// Accepts both Swift keys (last_modified, bytes) and Ceph keys (lastModified, size) for compatibility\n// - sortDirection: \"asc\" | \"desc\" — persisted alongside sortBy\nconst objectsSearchSchema = z.object({\n prefix: z.string().optional(),\n sortBy: z.enum([\"name\", \"last_modified\", \"bytes\", \"lastModified\", \"size\"]).optional(),\n sortDirection: z.enum([\"asc\", \"desc\"]).optional(),\n search: z.string().optional(),\n})\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId/storage/$provider/containers/$containerName/objects/\")(\n {\n staticData: {\n section: \"storage\",\n service: \"containers\",\n isDetail: true,\n sectionCrumb: { labelKey: \"Storage\" },\n crumb: { useParamAsLabel: \"provider\", to: \"/projects/$projectId/storage/$provider/containers\" },\n } satisfies RouteInfo,\n validateSearch: objectsSearchSchema,\n head: ({ match }) => ({\n meta: [{ title: match.params.containerName }],\n }),\n component: () => {\n return <ObjectsDashboard />\n },\n notFoundComponent: () => {\n return (\n <p>\n <Trans>Storage container not found</Trans>\n </p>\n )\n },\n loader: async ({ context }) => {\n const { trpcClient } = context\n const availableServices = await trpcClient?.auth.getAvailableServices.query()\n\n return {\n client: trpcClient,\n availableServices,\n }\n },\n beforeLoad: async ({ context, params }) => {\n const { trpcClient } = context\n const availableServices = await trpcClient?.auth.getAvailableServices.query()\n checkServiceAvailability(availableServices!, params)\n },\n }\n)\n\nfunction ObjectsDashboard() {\n const { project, provider, containerName } = useParams({\n from: \"/_auth/projects/$projectId/storage/$provider/containers/$containerName/objects/\",\n select: (params) => ({\n project: params.projectId,\n provider: params.provider,\n containerName: params.containerName,\n }),\n })\n\n const { prefix, sortBy, sortDirection, search } = Route.useSearch()\n\n return (\n <div>\n {project ? (\n <ErrorBoundary\n resetKeys={[project, provider, containerName, prefix, sortBy, sortDirection, search]}\n fallback={\n <div className=\"p-4 text-center\">\n <Trans>Error loading component</Trans>\n </div>\n }\n >\n {(() => {\n switch (provider) {\n case \"swift\":\n return <SwiftObjects />\n case \"ceph\":\n return <ObjectBrowserView bucketName={containerName} />\n default:\n return (\n <div className=\"p-4\">\n <Trans>Objects — {containerName}</Trans>\n </div>\n )\n }\n })()}\n </ErrorBoundary>\n ) : (\n <div className=\"p-4 text-center\">\n <Trans>No project selected</Trans>\n </div>\n )}\n </div>\n )\n}\n"],"mappings":";;;;4FASaI,KACXC,GAIAG,MAAAA;CAMA,IAAM,EAAEE,aAAUD,cAAWE,qBAAkBH,GAEzCI,IAAeV,EAAgBG,EAAAA;AAGrC,KAAI,CAACO,EAAa,gBAChB,OAAMX,EAAS;EACbY,IAAI;EACJL,QAAQ,EAAEC,cAAU;EACtB,CAAA;CAIF,IAAMK,IAAWC,EAAQH,EAAa,gBAAgB,OAQhDM,IAPUH,EAAQH,EAAa,gBAAgB,QAIzB,IAItBO,IAAmBL,IAAW,UAAUI,IAAmB,SAAS;AAE1E,KAAIR,MAAa,WAAWA,MAAa,OAOvC,OAAMT,EANDkB,IAMU;EACbN,IAAI;EACJL,QAAQ;GAAEC;GAAWC,UAAUS;GAAkBR;GAAc;EACjE,GARiB;EACbE,IAAI;EACJL,QAAQ,EAAEC,cAAU;EACtB,CAAA;AAQJ,KAAIC,MAAa,WAAW,CAACI,EAQ3B,OAAMb,EAPDiB,IAOU;EACbL,IAAI;EACJL,QAAQ;GAAEC;GAAWC,UAAU;GAAQC;GAAc;EACvD,GATiB;EACbE,IAAI;EACJL,QAAQ,EAAEC,cAAU;EACtB,CAAA;AASJ,KAAIC,MAAa,UAAU,CAACQ,EAQ1B,OAAMjB,EAPDa,IAOU;EACbD,IAAI;EACJL,QAAQ;GAAEC;GAAWC,UAAU;GAASC;GAAc;EACxD,GATiB;EACbE,IAAI;EACJL,QAAQ,EAAEC,cAAU;EACtB,CAAA;GAeAW,IAAsBjB,EAAEkB,OAAO;CACnCC,QAAQnB,EAAEoB,QAAM,CAAGC,UAAQ;CAC3BC,QAAQtB,EAAEuB,KAAK;EAAC;EAAQ;EAAiB;EAAS;EAAgB;EAAO,CAAA,CAAEF,UAAQ;CACnFG,eAAexB,EAAEuB,KAAK,CAAC,OAAO,OAAO,CAAA,CAAEF,UAAQ;CAC/CI,QAAQzB,EAAEoB,QAAM,CAAGC,UAAQ;CAC7B,CAAA,EAEaK,IAAQ7B,EAAgB,kFAAA,CACnC;CACE8B,YAAY;EACVC,SAAS;EACTC,SAAS;EACTC,UAAU;EACVC,cAAc,EAAEC,UAAU,WAAU;EACpCC,OAAO;GAAEC,iBAAiB;GAAYxB,IAAI;GAAoD;EAChG;CACA0B,gBAAgBnB;CAChBoB,OAAO,EAAEC,gBAAa,EACpBC,MAAM,CAAC,EAAEC,OAAOF,EAAMjC,OAAOG,eAAc,CAAC,EAC9C;CACAiC,WAASC,EAAAC,GAAA,YAAA;CAGTC,mBAAiBF,EAAAG,GAAA,oBAAA;CAOjBC,QAAQ,OAAO,EAAEC,iBAAS;EACxB,IAAM,EAAEC,kBAAeD;AAGvB,SAAO;GACLK,QAAQJ;GACR9C,mBAJwB,MAAM8C,GAAYC,KAAKC,qBAAqBC,OAAAA;GAKtE;;CAEFE,YAAY,OAAO,EAAEN,YAAS1C,gBAAQ;EACpC,IAAM,EAAE2C,kBAAeD;AAEvB9C,IAD0B,MAAM+C,GAAYC,KAAKC,qBAAqBC,OAAAA,EACzB9C,EAAAA;;CAEjD,CAAA"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"projects-0feOw_b6.mjs","names":["createFileRoute","z","searchSchema","object","search","string","optional","Route","component","lazyRouteComponent","$$splitComponentImporter","errorComponent","$$splitErrorComponentImporter","notFoundComponent","$$splitNotFoundComponentImporter","validateSearch","loaderDeps","loader","context","deps","allProjects","trpcClient","project","getAuthProjects","query","projects","trim","searchTermLower","toLowerCase","filter","name","includes","description"],"sources":["../../src/client/routes/_auth/projects/index.tsx"],"sourcesContent":["import { createFileRoute } from \"@tanstack/react-router\"\nimport { useState } from \"react\"\nimport { ProjectsOverviewNavBar, ViewMode } from \"@/client/routes/_auth/projects/-components/ProjectOverviewNavBar\"\nimport { ProjectCardView } from \"@/client/routes/_auth/projects/-components/ProjectCardView\"\nimport { ProjectListView } from \"@/client/routes/_auth/projects/-components/ProjectListView\"\nimport { RouteError } from \"@/client/components/Error/RouteError\"\nimport { TRPCClientError } from \"@trpc/client\"\nimport { ContentHeading } from \"@cloudoperators/juno-ui-components\"\nimport { Trans } from \"@lingui/react/macro\"\n\nimport { z } from \"zod\"\n\nconst searchSchema = z.object({\n search: z.string().optional(),\n})\n\nexport const Route = createFileRoute(\"/_auth/projects/\")({\n component: ProjectsOverview,\n errorComponent: ({ error }) => (\n <RouteError error={error} safeErrorMessage={error instanceof TRPCClientError ? error.message : undefined} />\n ),\n notFoundComponent: () => {\n return <p>Projects not found</p>\n },\n\n validateSearch: searchSchema,\n\n loaderDeps: ({ search }) => ({\n search: search.search || \"\",\n }),\n\n loader: async ({ context, deps }) => {\n const allProjects = await context.trpcClient?.project.getAuthProjects.query()\n\n // Filter projects based on search term\n let projects = allProjects\n if (deps.search && deps.search.trim() !== \"\") {\n const searchTermLower = deps.search.toLowerCase()\n projects = allProjects?.filter(\n (project) =>\n project.name?.toLowerCase().includes(searchTermLower) ||\n project.description?.toLowerCase().includes(searchTermLower)\n )\n }\n\n return {\n projects,\n }\n },\n})\n\nfunction ProjectsOverview() {\n const [viewMode, setViewMode] = useState<ViewMode>(\"card\")\n const { projects } = Route.useLoaderData()\n const { search = \"\" } = Route.useSearch()\n const navigate = Route.useNavigate()\n\n const handleSearch = (value: string) => {\n navigate({\n search: { search: value },\n replace: true,\n })\n }\n\n return (\n <div>\n <div className=\"mx-auto h-full w-full max-w-full p-4 lg:w-3/4 xl:w-2/3 2xl:w-5/8\">\n <div className=\"mx-auto w-full\">\n <ContentHeading className=\"px-4 pt-4\">\n <Trans>Projects</Trans>\n </ContentHeading>\n <ProjectsOverviewNavBar\n viewMode={viewMode}\n setViewMode={setViewMode}\n searchTerm={search}\n onSearch={handleSearch}\n />\n <div className=\"w-full pt-5\">\n {viewMode === \"list\" ? <ProjectListView projects={projects} /> : <ProjectCardView projects={projects} />}\n </div>\n </div>\n </div>\n </div>\n )\n}\n"],"mappings":";;;2IAYME,IAAeD,EAAEE,OAAO,EAC5BC,QAAQH,EAAEI,QAAM,CAAGC,UAAQ,EAC7B,CAAA,EAEaC,IAAQP,EAAgB,mBAAA,CAAoB;CACvDQ,WAASC,EAAAC,GAAA,YAAA;CACTC,gBAAcF,EAAAG,GAAA,iBAAA;CAGdC,mBAAiBJ,EAAAK,GAAA,oBAAA;CAIjBC,gBAAgBb;CAEhBc,aAAa,EAAEZ,iBAAc,EAC3BA,QAAQA,EAAOA,UAAU,IAC3B;CAEAa,QAAQ,OAAO,EAAEC,YAASC,cAAM;EAC9B,IAAMC,IAAc,MAAMF,EAAQG,YAAYC,QAAQC,gBAAgBC,OAAAA,EAGlEC,IAAWL;AACf,MAAID,EAAKf,UAAUe,EAAKf,OAAOsB,MAAI,KAAO,IAAI;GAC5C,IAAMC,IAAkBR,EAAKf,OAAOwB,aAAW;AAC/CH,OAAWL,GAAaS,QACrBP,MACCA,EAAQQ,MAAMF,aAAAA,CAAcG,SAASJ,EAAAA,IACrCL,EAAQU,aAAaJ,aAAAA,CAAcG,SAASJ,EAAAA,CAAAA;;AAIlD,SAAO,EACLF,aACF;;CAEJ,CAAA"}
@@ -1,144 +0,0 @@
1
- import { $ as e, A as t, N as n, O as r, at as i, b as a, l as o, p as s, z as c } from "./build-Cf7iWbpH.mjs";
2
- import { t as l } from "./projects-0feOw_b6.mjs";
3
- import { Fragment as u, jsx as d, jsxs as f } from "react/jsx-runtime";
4
- import { useEffect as p, useRef as m, useState as h } from "react";
5
- import { Link as g } from "@tanstack/react-router";
6
- import { Trans as _, useLingui as v } from "@lingui/react";
7
- //#region src/client/routes/_auth/projects/-components/ProjectOverviewNavBar.tsx
8
- function y({ viewMode: t, setViewMode: n, onSearch: r, searchTerm: a = "" }) {
9
- let { i18n: c, _: l } = v(), g = m(null), _ = m(!1), [y, b] = h(a);
10
- return p(() => {
11
- !_.current && !g.current && a !== y && b(a);
12
- }, [a]), p(() => () => {
13
- g.current && clearTimeout(g.current);
14
- }, []), /* @__PURE__ */ d(u, { children: /* @__PURE__ */ d(e, {
15
- alignment: "center",
16
- gap: "8",
17
- className: "my-px mt-4 px-4",
18
- children: /* @__PURE__ */ d(e, {
19
- direction: "vertical",
20
- gap: "3",
21
- className: "w-full",
22
- children: /* @__PURE__ */ f(e, {
23
- gap: "6",
24
- className: "flex w-full flex-wrap",
25
- children: [/* @__PURE__ */ d(s, {
26
- className: "shrink-0 grow basis-0",
27
- type: "text",
28
- placeholder: c._({ id: "YIix5Y" }),
29
- onChange: (e) => {
30
- let t = e.target.value;
31
- b(t), g.current && clearTimeout(g.current), g.current = setTimeout(() => {
32
- g.current = null, r(t);
33
- }, 300);
34
- },
35
- onFocus: () => {
36
- _.current = !0;
37
- },
38
- onBlur: () => {
39
- _.current = !1;
40
- },
41
- onClear: () => {
42
- g.current &&= (clearTimeout(g.current), null), b(""), r("");
43
- },
44
- value: y
45
- }), /* @__PURE__ */ f(i, { children: [/* @__PURE__ */ d(o, {
46
- variant: t === "list" ? "default" : "subdued",
47
- onClick: () => n("list"),
48
- icon: "dns"
49
- }), /* @__PURE__ */ d(o, {
50
- variant: t === "card" ? "default" : "subdued",
51
- onClick: () => n("card"),
52
- icon: "autoAwesomeMotion"
53
- })] })]
54
- })
55
- })
56
- }) });
57
- }
58
- //#endregion
59
- //#region src/client/routes/_auth/projects/-components/ProjectCardView.tsx
60
- function b({ project: e }) {
61
- return /* @__PURE__ */ d(g, {
62
- to: "/projects/$projectId",
63
- params: { projectId: e.id },
64
- className: "block text-inherit no-underline",
65
- children: /* @__PURE__ */ d(a, {
66
- className: "hover:bg-theme-background-lvl-2 min-h-50 rounded-lg p-6 shadow-md",
67
- children: /* @__PURE__ */ f("div", {
68
- className: "w-full",
69
- children: [/* @__PURE__ */ d(r, {
70
- className: "text-theme-accent",
71
- children: e.name
72
- }), /* @__PURE__ */ d("p", {
73
- className: "mt-4 line-clamp-3 pr-4 leading-relaxed",
74
- children: e.description
75
- })]
76
- })
77
- })
78
- });
79
- }
80
- function x({ projects: e }) {
81
- return /* @__PURE__ */ d("div", {
82
- className: "mx-auto h-full w-full max-w-[95vw] px-4",
83
- children: /* @__PURE__ */ d("div", {
84
- className: "grid grid-cols-1 gap-6 sm:grid-cols-2 lg:grid-cols-3",
85
- children: e?.length ? e.map((e) => /* @__PURE__ */ d(b, { project: e }, e.id)) : /* @__PURE__ */ d("p", {
86
- className: "text-center",
87
- children: /* @__PURE__ */ d(_, { id: "Zgp2Sm" })
88
- })
89
- })
90
- });
91
- }
92
- //#endregion
93
- //#region src/client/routes/_auth/projects/-components/ProjectListView.tsx
94
- function S({ projects: e }) {
95
- return /* @__PURE__ */ d(c, {
96
- className: "overflow-hidden",
97
- columns: 2,
98
- children: e?.length ? e.map((e) => /* @__PURE__ */ f(n, { children: [/* @__PURE__ */ d(t, { children: /* @__PURE__ */ d(g, {
99
- to: "/projects/$projectId",
100
- params: { projectId: e.id },
101
- className: "text-inherit no-underline",
102
- children: e.name
103
- }) }), /* @__PURE__ */ d(t, { children: e.description })] }, e.id)) : /* @__PURE__ */ d("div", {
104
- className: "py-6 text-center",
105
- children: /* @__PURE__ */ d(_, { id: "i30J2U" })
106
- })
107
- });
108
- }
109
- //#endregion
110
- //#region src/client/routes/_auth/projects/index.tsx?tsr-split=component
111
- function C() {
112
- let [e, t] = h("card"), { projects: n } = l.useLoaderData(), { search: i = "" } = l.useSearch(), a = l.useNavigate();
113
- return /* @__PURE__ */ d("div", { children: /* @__PURE__ */ d("div", {
114
- className: "mx-auto h-full w-full max-w-full p-4 lg:w-3/4 xl:w-2/3 2xl:w-5/8",
115
- children: /* @__PURE__ */ f("div", {
116
- className: "mx-auto w-full",
117
- children: [
118
- /* @__PURE__ */ d(r, {
119
- className: "px-4 pt-4",
120
- children: /* @__PURE__ */ d(_, { id: "+0B+ue" })
121
- }),
122
- /* @__PURE__ */ d(y, {
123
- viewMode: e,
124
- setViewMode: t,
125
- searchTerm: i,
126
- onSearch: (e) => {
127
- a({
128
- search: { search: e },
129
- replace: !0
130
- });
131
- }
132
- }),
133
- /* @__PURE__ */ d("div", {
134
- className: "w-full pt-5",
135
- children: d(e === "list" ? S : x, { projects: n })
136
- })
137
- ]
138
- })
139
- }) });
140
- }
141
- //#endregion
142
- export { C as component };
143
-
144
- //# sourceMappingURL=projects-C1IYOvFQ.mjs.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"projects-C1IYOvFQ.mjs","names":["useEffect","useRef","useState","Button","InputGroup","SearchInput","Stack","ProjectsOverviewNavBar","viewMode","setViewMode","onSearch","searchTerm","useLingui","timerRef","inputFocusedRef","inputValue","setInputValue","current","clearTimeout","handleSearchChange","e","value","target","setTimeout","handleClear","alignment","gap","className","direction","type","placeholder","t","onChange","onFocus","onBlur","onClear","variant","onClick","icon","Link","Box","ContentHeading","ProjectCard","project","to","params","projectId","id","className","div","name","p","description","ProjectCardView","projects","length","map","DataGrid","DataGridCell","DataGridRow","Link","ProjectListView","projects","className","columns","length","map","project","to","params","projectId","id","name","description","div","useState","ProjectsOverviewNavBar","ViewMode","ProjectCardView","ProjectListView","ContentHeading","Trans","Route","ProjectsOverview","viewMode","setViewMode","projects","useLoaderData","search","useSearch","navigate","useNavigate","handleSearch","value","replace","component"],"sources":["../../src/client/routes/_auth/projects/-components/ProjectOverviewNavBar.tsx","../../src/client/routes/_auth/projects/-components/ProjectCardView.tsx","../../src/client/routes/_auth/projects/-components/ProjectListView.tsx","../../src/client/routes/_auth/projects/index.tsx?tsr-split=component"],"sourcesContent":["import { ChangeEvent, useEffect, useRef, useState } from \"react\"\nimport { Button, InputGroup, SearchInput, Stack } from \"@cloudoperators/juno-ui-components\"\nimport { useLingui } from \"@lingui/react/macro\"\nexport type ViewMode = \"list\" | \"card\"\n\ntype ProjectsOverviewNavBarProps = {\n viewMode: ViewMode\n setViewMode: (mode: ViewMode) => void\n searchPlaceholder?: string\n filters?: { label: string; value: string }[]\n searchTerm?: string\n onSearch: (value: string) => void\n}\n\nexport function ProjectsOverviewNavBar({\n viewMode,\n setViewMode,\n onSearch,\n searchTerm = \"\",\n}: ProjectsOverviewNavBarProps) {\n const { t } = useLingui()\n const timerRef = useRef<NodeJS.Timeout | null>(null)\n const inputFocusedRef = useRef(false)\n const [inputValue, setInputValue] = useState(searchTerm)\n\n // Sync from URL (e.g. back/forward navigation) only when user isn't actively editing\n useEffect(() => {\n if (!inputFocusedRef.current && !timerRef.current && searchTerm !== inputValue) {\n setInputValue(searchTerm)\n }\n }, [searchTerm])\n\n useEffect(() => {\n return () => {\n if (timerRef.current) {\n clearTimeout(timerRef.current)\n }\n }\n }, [])\n\n const handleSearchChange = (e: ChangeEvent<HTMLInputElement>) => {\n const value = e.target.value\n setInputValue(value)\n\n if (timerRef.current) {\n clearTimeout(timerRef.current)\n }\n\n timerRef.current = setTimeout(() => {\n timerRef.current = null\n onSearch(value)\n }, 300)\n }\n\n const handleClear = () => {\n if (timerRef.current) {\n clearTimeout(timerRef.current)\n timerRef.current = null\n }\n setInputValue(\"\")\n onSearch(\"\")\n }\n\n return (\n <>\n <Stack alignment=\"center\" gap=\"8\" className=\"my-px mt-4 px-4\">\n <Stack direction=\"vertical\" gap=\"3\" className=\"w-full\">\n <Stack gap=\"6\" className=\"flex w-full flex-wrap\">\n <SearchInput\n className=\"shrink-0 grow basis-0\"\n type=\"text\"\n placeholder={t`Search...`}\n onChange={handleSearchChange}\n onFocus={() => {\n inputFocusedRef.current = true\n }}\n onBlur={() => {\n inputFocusedRef.current = false\n }}\n onClear={handleClear}\n value={inputValue}\n />\n\n <InputGroup>\n <Button\n variant={viewMode === \"list\" ? \"default\" : \"subdued\"}\n onClick={() => setViewMode(\"list\")}\n icon=\"dns\"\n />\n\n <Button\n variant={viewMode === \"card\" ? \"default\" : \"subdued\"}\n onClick={() => setViewMode(\"card\")}\n icon=\"autoAwesomeMotion\"\n />\n </InputGroup>\n </Stack>\n </Stack>\n </Stack>\n </>\n )\n}\n","import { Link } from \"@tanstack/react-router\"\nimport { Project } from \"@/server/Project/types/models\"\nimport { Box, ContentHeading } from \"@cloudoperators/juno-ui-components\"\nimport { Trans } from \"@lingui/react/macro\"\n\ntype ProjectCardProps = {\n project: Project\n}\ntype ProjectCardViewProps = {\n projects: Project[] | undefined\n}\n\nexport function ProjectCard({ project }: ProjectCardProps) {\n return (\n <Link to=\"/projects/$projectId\" params={{ projectId: project.id }} className=\"block text-inherit no-underline\">\n <Box className=\"hover:bg-theme-background-lvl-2 min-h-50 rounded-lg p-6 shadow-md\">\n <div className=\"w-full\">\n <ContentHeading className=\"text-theme-accent\">{project.name}</ContentHeading>\n\n <p className=\"mt-4 line-clamp-3 pr-4 leading-relaxed\">{project.description}</p>\n </div>\n </Box>\n </Link>\n )\n}\nexport function ProjectCardView({ projects }: ProjectCardViewProps) {\n return (\n <div className=\"mx-auto h-full w-full max-w-[95vw] px-4\">\n {/* Adaptive Grid: max 3 columns, adjusts on smaller screens */}\n <div className=\"grid grid-cols-1 gap-6 sm:grid-cols-2 lg:grid-cols-3\">\n {projects?.length ? (\n projects.map((project) => <ProjectCard key={project.id} project={project} />)\n ) : (\n <p className=\"text-center\">\n <Trans>No projects available.</Trans>\n </p>\n )}\n </div>\n </div>\n )\n}\n","import { Project } from \"@/server/Project/types/models\"\nimport { DataGrid, DataGridCell, DataGridRow } from \"@cloudoperators/juno-ui-components\"\nimport { Trans } from \"@lingui/react/macro\"\nimport { Link } from \"@tanstack/react-router\"\n\ntype ProjectListViewProps = {\n projects: Project[] | undefined\n}\n\nexport function ProjectListView({ projects }: ProjectListViewProps) {\n return (\n <DataGrid className=\"overflow-hidden\" columns={2}>\n {projects?.length ? (\n projects.map((project) => {\n return (\n <DataGridRow key={project.id}>\n <DataGridCell>\n <Link\n to=\"/projects/$projectId\"\n params={{ projectId: project.id }}\n className=\"text-inherit no-underline\"\n >\n {project.name}\n </Link>\n </DataGridCell>\n <DataGridCell>{project.description}</DataGridCell>\n </DataGridRow>\n )\n })\n ) : (\n <div className=\"py-6 text-center\">\n <Trans>No projects found</Trans>\n </div>\n )}\n </DataGrid>\n )\n}\n","import { createFileRoute } from \"@tanstack/react-router\"\nimport { useState } from \"react\"\nimport { ProjectsOverviewNavBar, ViewMode } from \"@/client/routes/_auth/projects/-components/ProjectOverviewNavBar\"\nimport { ProjectCardView } from \"@/client/routes/_auth/projects/-components/ProjectCardView\"\nimport { ProjectListView } from \"@/client/routes/_auth/projects/-components/ProjectListView\"\nimport { RouteError } from \"@/client/components/Error/RouteError\"\nimport { TRPCClientError } from \"@trpc/client\"\nimport { ContentHeading } from \"@cloudoperators/juno-ui-components\"\nimport { Trans } from \"@lingui/react/macro\"\n\nimport { z } from \"zod\"\n\nconst searchSchema = z.object({\n search: z.string().optional(),\n})\n\nexport const Route = createFileRoute(\"/_auth/projects/\")({\n component: ProjectsOverview,\n errorComponent: ({ error }) => (\n <RouteError error={error} safeErrorMessage={error instanceof TRPCClientError ? error.message : undefined} />\n ),\n notFoundComponent: () => {\n return <p>Projects not found</p>\n },\n\n validateSearch: searchSchema,\n\n loaderDeps: ({ search }) => ({\n search: search.search || \"\",\n }),\n\n loader: async ({ context, deps }) => {\n const allProjects = await context.trpcClient?.project.getAuthProjects.query()\n\n // Filter projects based on search term\n let projects = allProjects\n if (deps.search && deps.search.trim() !== \"\") {\n const searchTermLower = deps.search.toLowerCase()\n projects = allProjects?.filter(\n (project) =>\n project.name?.toLowerCase().includes(searchTermLower) ||\n project.description?.toLowerCase().includes(searchTermLower)\n )\n }\n\n return {\n projects,\n }\n },\n})\n\nfunction ProjectsOverview() {\n const [viewMode, setViewMode] = useState<ViewMode>(\"card\")\n const { projects } = Route.useLoaderData()\n const { search = \"\" } = Route.useSearch()\n const navigate = Route.useNavigate()\n\n const handleSearch = (value: string) => {\n navigate({\n search: { search: value },\n replace: true,\n })\n }\n\n return (\n <div>\n <div className=\"mx-auto h-full w-full max-w-full p-4 lg:w-3/4 xl:w-2/3 2xl:w-5/8\">\n <div className=\"mx-auto w-full\">\n <ContentHeading className=\"px-4 pt-4\">\n <Trans>Projects</Trans>\n </ContentHeading>\n <ProjectsOverviewNavBar\n viewMode={viewMode}\n setViewMode={setViewMode}\n searchTerm={search}\n onSearch={handleSearch}\n />\n <div className=\"w-full pt-5\">\n {viewMode === \"list\" ? <ProjectListView projects={projects} /> : <ProjectCardView projects={projects} />}\n </div>\n </div>\n </div>\n </div>\n )\n}\n"],"mappings":"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"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"projects-jyIHL6DE.mjs","names":["RouteError","TRPCClientError","SplitErrorComponent","error","message","undefined","errorComponent"],"sources":["../../src/client/routes/_auth/projects/index.tsx?tsr-split=errorComponent"],"sourcesContent":["import { createFileRoute } from \"@tanstack/react-router\"\nimport { useState } from \"react\"\nimport { ProjectsOverviewNavBar, ViewMode } from \"@/client/routes/_auth/projects/-components/ProjectOverviewNavBar\"\nimport { ProjectCardView } from \"@/client/routes/_auth/projects/-components/ProjectCardView\"\nimport { ProjectListView } from \"@/client/routes/_auth/projects/-components/ProjectListView\"\nimport { RouteError } from \"@/client/components/Error/RouteError\"\nimport { TRPCClientError } from \"@trpc/client\"\nimport { ContentHeading } from \"@cloudoperators/juno-ui-components\"\nimport { Trans } from \"@lingui/react/macro\"\n\nimport { z } from \"zod\"\n\nconst searchSchema = z.object({\n search: z.string().optional(),\n})\n\nexport const Route = createFileRoute(\"/_auth/projects/\")({\n component: ProjectsOverview,\n errorComponent: ({ error }) => (\n <RouteError error={error} safeErrorMessage={error instanceof TRPCClientError ? error.message : undefined} />\n ),\n notFoundComponent: () => {\n return <p>Projects not found</p>\n },\n\n validateSearch: searchSchema,\n\n loaderDeps: ({ search }) => ({\n search: search.search || \"\",\n }),\n\n loader: async ({ context, deps }) => {\n const allProjects = await context.trpcClient?.project.getAuthProjects.query()\n\n // Filter projects based on search term\n let projects = allProjects\n if (deps.search && deps.search.trim() !== \"\") {\n const searchTermLower = deps.search.toLowerCase()\n projects = allProjects?.filter(\n (project) =>\n project.name?.toLowerCase().includes(searchTermLower) ||\n project.description?.toLowerCase().includes(searchTermLower)\n )\n }\n\n return {\n projects,\n }\n },\n})\n\nfunction ProjectsOverview() {\n const [viewMode, setViewMode] = useState<ViewMode>(\"card\")\n const { projects } = Route.useLoaderData()\n const { search = \"\" } = Route.useSearch()\n const navigate = Route.useNavigate()\n\n const handleSearch = (value: string) => {\n navigate({\n search: { search: value },\n replace: true,\n })\n }\n\n return (\n <div>\n <div className=\"mx-auto h-full w-full max-w-full p-4 lg:w-3/4 xl:w-2/3 2xl:w-5/8\">\n <div className=\"mx-auto w-full\">\n <ContentHeading className=\"px-4 pt-4\">\n <Trans>Projects</Trans>\n </ContentHeading>\n <ProjectsOverviewNavBar\n viewMode={viewMode}\n setViewMode={setViewMode}\n searchTerm={search}\n onSearch={handleSearch}\n />\n <div className=\"w-full pt-5\">\n {viewMode === \"list\" ? <ProjectListView projects={projects} /> : <ProjectCardView projects={projects} />}\n </div>\n </div>\n </div>\n </div>\n )\n}\n"],"mappings":";;;;AAM8C,IAAAE,KAY3B,EAAEC,eACjB,kBAAC,GAAA;CAAkBA;CAAO,kBAAkBA,aAAiBF,IAAkBE,EAAMC,UAAUC,KAAAA"}