@fluid-app/portal-sdk 0.1.135 → 0.1.136

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 (31) hide show
  1. package/dist/ProductsScreen-C7cXqLN8.cjs +110 -0
  2. package/dist/ProductsScreen-C7cXqLN8.cjs.map +1 -0
  3. package/dist/{ProductsScreen-CbKT2maZ.mjs → ProductsScreen-CTUejNR0.mjs} +2 -2
  4. package/dist/{ProductsScreen-T1tGuiGy.cjs → ProductsScreen-CcF3f9VD.cjs} +2 -2
  5. package/dist/ProductsScreen-Di6uVQCx.mjs +98 -0
  6. package/dist/ProductsScreen-Di6uVQCx.mjs.map +1 -0
  7. package/dist/ShareablesScreen-B-uZ3B5d.cjs +430 -0
  8. package/dist/ShareablesScreen-B-uZ3B5d.cjs.map +1 -0
  9. package/dist/{ShareablesScreen-DSW36iS7.mjs → ShareablesScreen-BHyxG-oy.mjs} +2 -2
  10. package/dist/{ShareablesScreen-BLbzr0Jb.cjs → ShareablesScreen-DAM01cxf.cjs} +2 -2
  11. package/dist/ShareablesScreen-iBUd6p7M.mjs +412 -0
  12. package/dist/ShareablesScreen-iBUd6p7M.mjs.map +1 -0
  13. package/dist/index.cjs +7 -7
  14. package/dist/index.d.cts.map +1 -1
  15. package/dist/index.d.mts.map +1 -1
  16. package/dist/index.mjs +7 -7
  17. package/dist/{src-BbjxT3sc.mjs → portal-tenant-product-media-adapter-Co9Kw-US.mjs} +980 -4
  18. package/dist/portal-tenant-product-media-adapter-Co9Kw-US.mjs.map +1 -0
  19. package/dist/{src-DmzXXBVt.cjs → portal-tenant-product-media-adapter-DM94zJXA.cjs} +1039 -3
  20. package/dist/portal-tenant-product-media-adapter-DM94zJXA.cjs.map +1 -0
  21. package/package.json +16 -16
  22. package/dist/ProductsScreen-B1OAGNQx.cjs +0 -283
  23. package/dist/ProductsScreen-B1OAGNQx.cjs.map +0 -1
  24. package/dist/ProductsScreen-CL71MQNL.mjs +0 -271
  25. package/dist/ProductsScreen-CL71MQNL.mjs.map +0 -1
  26. package/dist/ShareablesScreen-CecBPoFO.mjs +0 -1360
  27. package/dist/ShareablesScreen-CecBPoFO.mjs.map +0 -1
  28. package/dist/ShareablesScreen-DDutXKpW.cjs +0 -1378
  29. package/dist/ShareablesScreen-DDutXKpW.cjs.map +0 -1
  30. package/dist/src-BbjxT3sc.mjs.map +0 -1
  31. package/dist/src-DmzXXBVt.cjs.map +0 -1
@@ -0,0 +1,412 @@
1
+ import { et as USER_TYPES, n as useFluidContext, u as usePortalTenantClient, yt as createFetchClient } from "./FluidProvider-Dd2M4mMV.mjs";
2
+ import { t as useFluidAuth } from "./use-fluid-auth-CU2oy-C8.mjs";
3
+ import { n as useCurrentUser } from "./use-current-user-DxFplFVa.mjs";
4
+ import { n as useAppNavigation } from "./AppNavigationContext-Du3Qq0yc.mjs";
5
+ import { i as useSdkClient } from "./use-account-clients-Csy11_yq.mjs";
6
+ import { a as createPortalTenantPlaylistsShareablesAdapter, c as dam_assets_destroy, f as ShareablesApp, h as ShareablesCoreProvider, i as createPortalTenantPlaylistsAdapter, l as dam_assets_discard, m as ShareablesApiProvider, n as createPortalTenantSharesShareablesAdapter, o as createPortalTenantMediaShareablesAdapter, p as ShareablesUIProvider, r as createPortalTenantFilesShareablesAdapter, s as dam_asset_paths_create, t as createPortalTenantProductMediaShareablesAdapter, u as dam_query } from "./portal-tenant-product-media-adapter-Co9Kw-US.mjs";
7
+ import { t as usePortalProductsClient } from "./use-portal-products-client-BUTk8CR4.mjs";
8
+ import { useCallback, useMemo } from "react";
9
+ import { jsx } from "react/jsx-runtime";
10
+ import { z } from "zod";
11
+ //#region src/hooks/use-user-type.ts
12
+ /**
13
+ * Convenience hook for user-type checks in the portal SDK.
14
+ */
15
+ function useUserType() {
16
+ const { user } = useFluidAuth();
17
+ return useMemo(() => {
18
+ const userType = user?.user_type ?? null;
19
+ return {
20
+ userType,
21
+ isCustomer: userType === USER_TYPES.customer,
22
+ isRep: userType === USER_TYPES.rep,
23
+ isAdmin: userType === USER_TYPES.admin || userType === USER_TYPES.root_admin
24
+ };
25
+ }, [user?.user_type]);
26
+ }
27
+ //#endregion
28
+ //#region ../../file-picker/core/src/schemas/dam.ts
29
+ const damVariantSchema = z.object({
30
+ id: z.string(),
31
+ url: z.string().nullable(),
32
+ file_name: z.string(),
33
+ mime_type: z.string(),
34
+ content: z.any().nullable(),
35
+ created_at: z.string(),
36
+ updated_at: z.string(),
37
+ default: z.boolean(),
38
+ is_original: z.boolean(),
39
+ is_text: z.boolean(),
40
+ media_type: z.string(),
41
+ processing_status: z.string(),
42
+ tags: z.array(z.string())
43
+ });
44
+ const damAssetSchema = z.object({
45
+ id: z.number(),
46
+ canonical_path: z.string(),
47
+ category: z.string(),
48
+ code: z.string(),
49
+ company: z.string(),
50
+ created_at: z.string(),
51
+ default_variant_id: z.string(),
52
+ default_variant_url: z.string().optional(),
53
+ description: z.string(),
54
+ name: z.string(),
55
+ updated_at: z.string(),
56
+ variants: z.array(damVariantSchema).optional()
57
+ });
58
+ const damTreeFolderNodeSchema = z.object({
59
+ asset_code: z.union([z.string(), z.record(z.string(), z.unknown())]).optional(),
60
+ name: z.union([z.string(), z.record(z.string(), z.unknown())]).optional(),
61
+ category: z.union([z.string(), z.record(z.string(), z.unknown())]).optional(),
62
+ variants: z.union([z.array(z.unknown()), z.record(z.string(), z.unknown())]).optional()
63
+ }).passthrough();
64
+ const damTreeSchema = z.record(z.string(), z.union([
65
+ z.lazy(() => damTreeSchema),
66
+ damAssetSchema,
67
+ damTreeFolderNodeSchema
68
+ ]));
69
+ const damQueryResponseSchema = z.object({
70
+ path: z.string(),
71
+ tree: damTreeSchema,
72
+ meta: z.object({ next_cursor: z.string().optional() }).optional()
73
+ });
74
+ z.object({ asset: z.object({
75
+ file: z.any(),
76
+ name: z.string(),
77
+ description: z.string().optional(),
78
+ tags: z.string().optional()
79
+ }) });
80
+ z.object({
81
+ asset: damAssetSchema,
82
+ meta: z.object({
83
+ request_id: z.string(),
84
+ timestamp: z.string()
85
+ })
86
+ });
87
+ z.object({
88
+ placeholder_asset: z.object({
89
+ mime_type: z.string(),
90
+ name: z.string().optional(),
91
+ description: z.string().optional()
92
+ }),
93
+ skip_autotagging: z.boolean().optional()
94
+ });
95
+ z.object({
96
+ asset: z.object({
97
+ file: z.any(),
98
+ name: z.string(),
99
+ description: z.string().optional(),
100
+ tags: z.string().optional()
101
+ }).optional(),
102
+ text_asset: z.object({
103
+ file_name: z.string(),
104
+ mime_type: z.string(),
105
+ text: z.string(),
106
+ name: z.string().optional(),
107
+ description: z.string().optional(),
108
+ tags: z.string().optional()
109
+ }).optional(),
110
+ placeholder_asset: z.object({
111
+ mime_type: z.string(),
112
+ name: z.string().optional(),
113
+ description: z.string().optional()
114
+ }).optional(),
115
+ skip_autotagging: z.boolean().optional()
116
+ });
117
+ z.object({
118
+ asset: z.object({
119
+ id: z.number(),
120
+ canonical_path: z.string(),
121
+ name: z.string()
122
+ }),
123
+ meta: z.object({
124
+ request_id: z.string(),
125
+ timestamp: z.string()
126
+ })
127
+ });
128
+ //#endregion
129
+ //#region ../../file-picker/api-client/src/api/url-proxy.ts
130
+ const urlProxyResponseSchema = z.object({
131
+ data: z.string(),
132
+ contentType: z.string(),
133
+ size: z.number()
134
+ });
135
+ /**
136
+ * Proxy a URL fetch through the backend to bypass CORS restrictions.
137
+ * The backend fetches the file and returns it as base64-encoded data.
138
+ *
139
+ * @param url - The URL to fetch
140
+ * @param proxyEndpoint - The proxy endpoint (defaults to "/api/proxy-url")
141
+ */
142
+ async function proxyUrlFetch(url, proxyEndpoint = "/api/proxy-url") {
143
+ const response = await fetch(proxyEndpoint, {
144
+ method: "POST",
145
+ headers: { "Content-Type": "application/json" },
146
+ body: JSON.stringify({ url })
147
+ });
148
+ if (!response.ok) {
149
+ const errorData = await response.json().catch(() => ({ error: "Failed to proxy URL fetch" }));
150
+ throw new Error(errorData.error || `HTTP ${response.status}`);
151
+ }
152
+ const data = await response.json();
153
+ return urlProxyResponseSchema.parse(data);
154
+ }
155
+ //#endregion
156
+ //#region ../../file-picker/api-client/src/portal-tenant-adapter.ts
157
+ /**
158
+ * Maps a BFF DAM asset to the file-picker port's DamAssetCreateResponse shape.
159
+ *
160
+ * The BFF response includes nullable meta.request_id (from Api::Response),
161
+ * while the port schema requires a string. We coalesce to empty string.
162
+ */
163
+ function mapCreateResponse(response) {
164
+ const raw = response.asset ?? {};
165
+ return {
166
+ asset: damAssetSchema.parse({
167
+ ...raw,
168
+ canonical_path: raw.canonical_path ?? "",
169
+ category: raw.category ?? "",
170
+ company: raw.company ?? "",
171
+ description: raw.description ?? "",
172
+ default_variant_id: raw.default_variant_id ?? ""
173
+ }),
174
+ meta: {
175
+ request_id: response.meta?.request_id ?? "",
176
+ timestamp: response.meta?.timestamp ?? (/* @__PURE__ */ new Date()).toISOString()
177
+ }
178
+ };
179
+ }
180
+ /**
181
+ * Maps a BFF asset path response to the file-picker port's shape.
182
+ */
183
+ function mapAssetPathCreateResponse(response) {
184
+ const assetPath = response.asset_path ?? {};
185
+ return {
186
+ asset: {
187
+ id: assetPath.id ?? 0,
188
+ canonical_path: assetPath.path ?? "",
189
+ name: assetPath.asset_code ?? ""
190
+ },
191
+ meta: {
192
+ request_id: response.meta?.request_id ?? "",
193
+ timestamp: response.meta?.timestamp ?? (/* @__PURE__ */ new Date()).toISOString()
194
+ }
195
+ };
196
+ }
197
+ /**
198
+ * Creates a FilePickerApi adapter backed by the portal-tenant BFF's
199
+ * `/api/content/dam/*` endpoints, using cookie-based auth via the
200
+ * provided FetchClient.
201
+ *
202
+ * Unsplash search is not available through the BFF — callers that need
203
+ * Unsplash should use the legacy adapter or provide their own
204
+ * implementation.
205
+ *
206
+ * The `onProgress` callback in `createDamAsset` is not supported — the
207
+ * underlying `fetch` API does not expose upload progress events.
208
+ *
209
+ * URL proxy delegates to the same-origin `/api/proxy-url` endpoint
210
+ * (served by the hosting app, not the BFF) for CORS bypass, identical
211
+ * to the legacy adapter behaviour.
212
+ */
213
+ function createPortalTenantFilePickerApiAdapter(client) {
214
+ return {
215
+ createDamAsset: async (params) => {
216
+ const formData = new FormData();
217
+ formData.append("asset[file]", params.file);
218
+ formData.append("asset[name]", params.name);
219
+ if (params.description) formData.append("asset[description]", params.description);
220
+ if (params.tags && params.tags.length > 0) formData.append("asset[tags]", params.tags.join(","));
221
+ return mapCreateResponse(await client.requestWithFormData("/api/content/dam/assets", formData, { method: "POST" }));
222
+ },
223
+ queryDamAssets: async (params) => {
224
+ const response = await dam_query(client, params);
225
+ return damQueryResponseSchema.parse({
226
+ ...response,
227
+ meta: response.meta ? { next_cursor: response.meta.pagination?.next_cursor ?? void 0 } : void 0
228
+ });
229
+ },
230
+ deleteDamAsset: async (code) => {
231
+ return dam_assets_destroy(client, code);
232
+ },
233
+ discardDamAsset: async (code) => {
234
+ return dam_assets_discard(client, code);
235
+ },
236
+ createDamAssetPathForAssets: async (params) => {
237
+ return mapAssetPathCreateResponse(await dam_asset_paths_create(client, params.code, { asset_path: { path: params.asset_paths.join(",") } }));
238
+ },
239
+ searchUnsplash: async (_query, _page, _perPage) => {
240
+ throw new Error("Unsplash search is not available through the portal-tenant BFF. Configure an Unsplash access key and use the standard FilePickerApi adapter instead.");
241
+ },
242
+ proxyUrlFetch: (url) => proxyUrlFetch(url)
243
+ };
244
+ }
245
+ //#endregion
246
+ //#region src/screens/ShareablesScreen.tsx
247
+ /**
248
+ * Parse the current shareables sub-route from the full slug.
249
+ *
250
+ * System nav slugs are "share/products", "share/media", "share/playlists".
251
+ * Detail pages append an ID: "share/products/123", "share/media/456".
252
+ *
253
+ * "share/products" → screen="products", detailId=null
254
+ * "share/products/123" → screen="products", detailId="123"
255
+ * "share/media/456" → screen="media", detailId="456"
256
+ * "share/playlists" → screen="playlists", detailId=null
257
+ * "share/playlists/789" → screen="playlists", detailId="789"
258
+ * "share/files" → screen="files", detailId=null
259
+ * "share" → screen=null (default to products)
260
+ */
261
+ function parseShareablesRoute(currentSlug) {
262
+ const slugWithoutPrefix = currentSlug.replace(/^share\/?/, "");
263
+ if (!slugWithoutPrefix) return {
264
+ screen: null,
265
+ detailId: null,
266
+ action: null
267
+ };
268
+ const parts = slugWithoutPrefix.split("/");
269
+ return {
270
+ screen: parts[0] || null,
271
+ detailId: parts[1] || null,
272
+ action: parts[2] || null
273
+ };
274
+ }
275
+ function ShareablesScreen({ background, textColor, accentColor, padding, borderRadius, ...divProps }) {
276
+ const domainClient = useSdkClient();
277
+ const portalProductsApi = usePortalProductsClient();
278
+ const portalTenantClient = usePortalTenantClient();
279
+ const { config } = useFluidContext();
280
+ const { data: userData } = useCurrentUser();
281
+ const { currentSlug, navigate } = useAppNavigation();
282
+ const { isCustomer } = useUserType();
283
+ const fetchProducts = useCallback(async (search, cursor, limit) => {
284
+ if (search) return portalProductsApi.searchProducts(search, {
285
+ cursor,
286
+ limit
287
+ });
288
+ return portalProductsApi.listProducts({
289
+ cursor,
290
+ limit
291
+ });
292
+ }, [portalProductsApi]);
293
+ const fetchProduct = useCallback(async (id) => portalProductsApi.getProduct(id), [portalProductsApi]);
294
+ const { screen, detailId, action } = parseShareablesRoute(currentSlug);
295
+ const handleNavigate = useCallback((subScreen, id) => {
296
+ navigate(id ? `share/${subScreen}/${id}` : `share/${subScreen}`);
297
+ }, [navigate]);
298
+ const handleBack = useCallback(() => {
299
+ if (detailId && screen) navigate(`share/${screen}`);
300
+ else navigate("share/products");
301
+ }, [
302
+ navigate,
303
+ detailId,
304
+ screen
305
+ ]);
306
+ const coreConfig = useMemo(() => ({
307
+ client: domainClient,
308
+ user: userData ? { id: userData.id } : null,
309
+ repContext: true
310
+ }), [domainClient, userData]);
311
+ const playlistsAdapter = useMemo(() => createPortalTenantPlaylistsAdapter(portalTenantClient), [portalTenantClient]);
312
+ const mediaAdapter = useMemo(() => createPortalTenantMediaShareablesAdapter(portalTenantClient), [portalTenantClient]);
313
+ const playlistsShareablesAdapter = useMemo(() => createPortalTenantPlaylistsShareablesAdapter(portalTenantClient), [portalTenantClient]);
314
+ const filesAdapter = useMemo(() => createPortalTenantFilesShareablesAdapter(portalTenantClient), [portalTenantClient]);
315
+ const sharesAdapter = useMemo(() => createPortalTenantSharesShareablesAdapter(portalTenantClient), [portalTenantClient]);
316
+ const productMediaAdapter = useMemo(() => createPortalTenantProductMediaShareablesAdapter(portalProductsApi), [portalProductsApi]);
317
+ const filePickerApi = useMemo(() => {
318
+ const baseUrl = config.baseUrl.replace(/\/+$/, "").replace(/\/api$/, "");
319
+ const csrfToken = typeof document !== "undefined" ? document.querySelector("meta[name=\"csrf-token\"]")?.getAttribute("content") : null;
320
+ return createPortalTenantFilePickerApiAdapter(createFetchClient({
321
+ baseUrl,
322
+ getAuthToken: config.getAuthToken,
323
+ onAuthError: config.onAuthError,
324
+ credentials: "include",
325
+ defaultHeaders: {
326
+ ...config.defaultHeaders,
327
+ ...csrfToken ? { "X-CSRF-Token": csrfToken } : {}
328
+ }
329
+ }));
330
+ }, [
331
+ config.baseUrl,
332
+ config.getAuthToken,
333
+ config.onAuthError,
334
+ config.defaultHeaders
335
+ ]);
336
+ const uiConfig = useMemo(() => ({
337
+ user: userData ? {
338
+ id: userData.id,
339
+ company: userData.company ? { logo_url: userData.company.logo_url } : null
340
+ } : void 0,
341
+ affiliateId: userData?.affiliate_id ?? null,
342
+ basePath: "",
343
+ navigate: (path) => {
344
+ const cleanPath = path.replace(/^\//, "");
345
+ navigate(cleanPath.startsWith("share/") ? cleanPath : `share/${cleanPath}`);
346
+ },
347
+ showToast: (opts) => {
348
+ console.warn(`[Shareables] ${opts.type}: ${opts.title}`);
349
+ },
350
+ filePickerApi,
351
+ onToggleFavorite: async (params) => {
352
+ const affiliateId = userData?.affiliate_id;
353
+ if (!affiliateId) throw new Error("No affiliate ID");
354
+ return domainClient.post(`/user_companies/${affiliateId}/favorites/toggle.json`, {
355
+ favoriteable_id: params.favoriteableId,
356
+ favoriteable_type: params.favoriteableType
357
+ });
358
+ },
359
+ onDeletePlaylist: isCustomer ? void 0 : async (playlistId) => {
360
+ await playlistsAdapter.deletePlaylist(playlistId);
361
+ },
362
+ readOnly: isCustomer
363
+ }), [
364
+ userData,
365
+ navigate,
366
+ filePickerApi,
367
+ domainClient,
368
+ isCustomer,
369
+ playlistsAdapter
370
+ ]);
371
+ return /* @__PURE__ */ jsx("div", {
372
+ ...divProps,
373
+ className: `h-full ${divProps.className ?? ""}`,
374
+ children: /* @__PURE__ */ jsx(ShareablesCoreProvider, {
375
+ config: coreConfig,
376
+ children: /* @__PURE__ */ jsx(ShareablesApiProvider, {
377
+ media: mediaAdapter,
378
+ playlists: playlistsShareablesAdapter,
379
+ fileResources: filesAdapter,
380
+ share: sharesAdapter,
381
+ productMedia: productMediaAdapter,
382
+ children: /* @__PURE__ */ jsx(ShareablesUIProvider, {
383
+ config: uiConfig,
384
+ children: /* @__PURE__ */ jsx(ShareablesApp, {
385
+ screen,
386
+ detailId,
387
+ action,
388
+ companyLogoUrl: userData?.company?.logo_url,
389
+ countryCode: userData?.country?.iso,
390
+ fetchProducts,
391
+ fetchProduct,
392
+ onNavigate: handleNavigate,
393
+ onBack: handleBack
394
+ })
395
+ })
396
+ })
397
+ })
398
+ });
399
+ }
400
+ const shareablesScreenPropertySchema = {
401
+ widgetType: "ShareablesScreen",
402
+ displayName: "Shareables Screen",
403
+ tabsConfig: [{
404
+ id: "styling",
405
+ label: "Styling"
406
+ }],
407
+ fields: []
408
+ };
409
+ //#endregion
410
+ export { shareablesScreenPropertySchema as n, useUserType as r, ShareablesScreen as t };
411
+
412
+ //# sourceMappingURL=ShareablesScreen-iBUd6p7M.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ShareablesScreen-iBUd6p7M.mjs","names":["portalTenantContent.dam_query","portalTenantContent.dam_assets_destroy","portalTenantContent.dam_assets_discard","portalTenantContent.dam_asset_paths_create"],"sources":["../src/hooks/use-user-type.ts","../../../file-picker/core/src/schemas/dam.ts","../../../file-picker/api-client/src/api/url-proxy.ts","../../../file-picker/api-client/src/portal-tenant-adapter.ts","../src/screens/ShareablesScreen.tsx"],"sourcesContent":["import { useMemo } from \"react\";\nimport { useFluidAuth } from \"./use-fluid-auth\";\nimport { USER_TYPES, type UserType } from \"../auth/types\";\n\nexport interface UseUserTypeResult {\n userType: UserType | null;\n isCustomer: boolean;\n isRep: boolean;\n isAdmin: boolean;\n}\n\n/**\n * Convenience hook for user-type checks in the portal SDK.\n */\nexport function useUserType(): UseUserTypeResult {\n const { user } = useFluidAuth();\n\n return useMemo(() => {\n const userType = user?.user_type ?? null;\n return {\n userType,\n isCustomer: userType === USER_TYPES.customer,\n isRep: userType === USER_TYPES.rep,\n isAdmin:\n userType === USER_TYPES.admin || userType === USER_TYPES.root_admin,\n };\n }, [user?.user_type]);\n}\n","import { z } from \"zod\";\n\ntype DamVariantApi = {\n id: string;\n url: string | null;\n file_name: string;\n mime_type: string;\n content: any;\n created_at: string;\n updated_at: string;\n default: boolean;\n is_original: boolean;\n is_text: boolean;\n media_type: string;\n processing_status: string;\n tags: string[];\n};\n\nexport const damVariantSchema: z.ZodType<DamVariantApi> = z.object({\n id: z.string(),\n url: z.string().nullable(),\n file_name: z.string(),\n mime_type: z.string(),\n content: z.any().nullable(),\n created_at: z.string(),\n updated_at: z.string(),\n default: z.boolean(),\n is_original: z.boolean(),\n is_text: z.boolean(),\n media_type: z.string(),\n processing_status: z.string(),\n tags: z.array(z.string()),\n});\n\ntype DamAssetApi = {\n id: number;\n canonical_path: string;\n category: string;\n code: string;\n company: string;\n created_at: string;\n default_variant_id: string;\n default_variant_url?: string;\n description: string;\n name: string;\n updated_at: string;\n variants?: DamVariantApi[];\n};\n\nexport const damAssetSchema: z.ZodType<DamAssetApi> = z.object({\n id: z.number(),\n canonical_path: z.string(),\n category: z.string(),\n code: z.string(),\n company: z.string(),\n created_at: z.string(),\n default_variant_id: z.string(),\n default_variant_url: z.string().optional(),\n description: z.string(),\n name: z.string(),\n updated_at: z.string(),\n variants: z.array(damVariantSchema).optional(),\n});\n\ntype DamTreeFolderNode = {\n asset_code?: string | Record<string, unknown>;\n name?: string | Record<string, unknown>;\n category?: string | Record<string, unknown>;\n variants?: unknown[] | Record<string, unknown>;\n [key: string]: unknown;\n};\n\nexport const damTreeFolderNodeSchema: z.ZodType<DamTreeFolderNode> = z\n .object({\n asset_code: z\n .union([z.string(), z.record(z.string(), z.unknown())])\n .optional(),\n name: z.union([z.string(), z.record(z.string(), z.unknown())]).optional(),\n category: z\n .union([z.string(), z.record(z.string(), z.unknown())])\n .optional(),\n variants: z\n .union([z.array(z.unknown()), z.record(z.string(), z.unknown())])\n .optional(),\n })\n .passthrough();\n\nexport const damTreeSchema: z.ZodType<Record<string, any>> = z.record(\n z.string(),\n z.union([\n z.lazy(() => damTreeSchema),\n damAssetSchema,\n damTreeFolderNodeSchema,\n ]),\n);\n\ntype DamQueryResponse = {\n path: string;\n tree: Record<string, any>;\n meta?: { next_cursor?: string };\n};\n\nexport const damQueryResponseSchema: z.ZodType<DamQueryResponse> = z.object({\n path: z.string(),\n tree: damTreeSchema,\n meta: z\n .object({\n next_cursor: z.string().optional(),\n })\n .optional(),\n});\n\ntype DamAssetCreateRequest = {\n asset: {\n file: any;\n name: string;\n description?: string;\n tags?: string;\n };\n};\n\nexport const damAssetCreateRequestSchema: z.ZodType<DamAssetCreateRequest> =\n z.object({\n asset: z.object({\n file: z.any(),\n name: z.string(),\n description: z.string().optional(),\n tags: z.string().optional(),\n }),\n });\n\ntype DamAssetCreateResponse = {\n asset: DamAssetApi;\n meta: { request_id: string; timestamp: string };\n};\n\nexport const damAssetCreateResponseSchema: z.ZodType<DamAssetCreateResponse> =\n z.object({\n asset: damAssetSchema,\n meta: z.object({\n request_id: z.string(),\n timestamp: z.string(),\n }),\n });\n\n// Schema for creating asset with placeholder (for ImageKit direct upload)\ntype DamAssetCreateWithPlaceholderRequest = {\n placeholder_asset: {\n mime_type: string;\n name?: string;\n description?: string;\n };\n skip_autotagging?: boolean;\n};\n\nexport const damAssetCreateWithPlaceholderRequestSchema: z.ZodType<DamAssetCreateWithPlaceholderRequest> =\n z.object({\n placeholder_asset: z.object({\n mime_type: z.string(),\n name: z.string().optional(),\n description: z.string().optional(),\n }),\n skip_autotagging: z.boolean().optional(),\n });\n\n// Schema for creating asset path without file upload (legacy)\ntype DamAssetPathCreateRequest = {\n asset?: {\n file: any;\n name: string;\n description?: string;\n tags?: string;\n };\n text_asset?: {\n file_name: string;\n mime_type: string;\n text: string;\n name?: string;\n description?: string;\n tags?: string;\n };\n placeholder_asset?: {\n mime_type: string;\n name?: string;\n description?: string;\n };\n skip_autotagging?: boolean;\n};\n\nexport const damAssetPathCreateRequestSchema: z.ZodType<DamAssetPathCreateRequest> =\n z.object({\n asset: z\n .object({\n file: z.any(),\n name: z.string(),\n description: z.string().optional(),\n tags: z.string().optional(),\n })\n .optional(),\n text_asset: z\n .object({\n file_name: z.string(),\n mime_type: z.string(),\n text: z.string(),\n name: z.string().optional(),\n description: z.string().optional(),\n tags: z.string().optional(),\n })\n .optional(),\n placeholder_asset: z\n .object({\n mime_type: z.string(),\n name: z.string().optional(),\n description: z.string().optional(),\n })\n .optional(),\n skip_autotagging: z.boolean().optional(),\n });\n\ntype DamAssetPathCreateResponse = {\n asset: { id: number; canonical_path: string; name: string };\n meta: { request_id: string; timestamp: string };\n};\n\nexport const damAssetPathCreateResponseSchema: z.ZodType<DamAssetPathCreateResponse> =\n z.object({\n asset: z.object({\n id: z.number(),\n canonical_path: z.string(),\n name: z.string(),\n }),\n meta: z.object({\n request_id: z.string(),\n timestamp: z.string(),\n }),\n });\n\nexport type { DamVariantApi, DamAssetApi };\nexport type DamTreeApi = z.infer<typeof damTreeSchema>;\nexport type { DamQueryResponse };\nexport type { DamAssetCreateRequest };\nexport type { DamAssetCreateResponse };\nexport type { DamAssetCreateWithPlaceholderRequest };\nexport type { DamAssetPathCreateRequest };\nexport type { DamAssetPathCreateResponse };\n","import { z } from \"zod\";\nimport type { UrlProxyResponse } from \"@fluid-app/file-picker-core\";\n\nexport type { UrlProxyResponse };\n\nconst urlProxyResponseSchema: z.ZodType<UrlProxyResponse> = z.object({\n data: z.string(),\n contentType: z.string(),\n size: z.number(),\n});\n\n/**\n * Proxy a URL fetch through the backend to bypass CORS restrictions.\n * The backend fetches the file and returns it as base64-encoded data.\n *\n * @param url - The URL to fetch\n * @param proxyEndpoint - The proxy endpoint (defaults to \"/api/proxy-url\")\n */\nexport async function proxyUrlFetch(\n url: string,\n proxyEndpoint: string = \"/api/proxy-url\",\n): Promise<UrlProxyResponse> {\n const response = await fetch(proxyEndpoint, {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify({ url }),\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({\n error: \"Failed to proxy URL fetch\",\n }));\n throw new Error(\n (errorData as { error?: string }).error || `HTTP ${response.status}`,\n );\n }\n\n const data: unknown = await response.json();\n return urlProxyResponseSchema.parse(data);\n}\n","import type { FetchClientInstance } from \"@fluid-app/api-client-core\";\nimport {\n damAssetSchema,\n damQueryResponseSchema,\n type CreateDamAssetParams,\n type CreateDamAssetPathForAssetsParams,\n type DamAssetCreateResponse,\n type DamAssetPathCreateResponse,\n type DamQueryParams,\n type DamQueryResponse,\n type FilePickerApi,\n type UnsplashSearchResponse,\n} from \"@fluid-app/file-picker-core\";\nimport {\n portalTenantContent,\n type operations,\n} from \"@fluid-app/portal-tenant-content-api-client\";\nimport { proxyUrlFetch } from \"./api/url-proxy\";\n\n/** BFF create response — derived from the generated OpenAPI types. */\ntype DamAssetCreateBffResponse = Awaited<\n ReturnType<typeof portalTenantContent.dam_assets_create>\n>;\n\n/** BFF asset path create response — derived from the generated OpenAPI types. */\ntype DamAssetPathCreateBffResponse = Awaited<\n ReturnType<typeof portalTenantContent.dam_asset_paths_create>\n>;\n\n/**\n * Maps a BFF DAM asset to the file-picker port's DamAssetCreateResponse shape.\n *\n * The BFF response includes nullable meta.request_id (from Api::Response),\n * while the port schema requires a string. We coalesce to empty string.\n */\nfunction mapCreateResponse(\n response: DamAssetCreateBffResponse,\n): DamAssetCreateResponse {\n const raw = response.asset ?? {};\n return {\n asset: damAssetSchema.parse({\n ...raw,\n canonical_path: raw.canonical_path ?? \"\",\n category: raw.category ?? \"\",\n company: raw.company ?? \"\",\n description: raw.description ?? \"\",\n default_variant_id: raw.default_variant_id ?? \"\",\n }),\n meta: {\n request_id: response.meta?.request_id ?? \"\",\n timestamp: response.meta?.timestamp ?? new Date().toISOString(),\n },\n };\n}\n\n/**\n * Maps a BFF asset path response to the file-picker port's shape.\n */\nfunction mapAssetPathCreateResponse(\n response: DamAssetPathCreateBffResponse,\n): DamAssetPathCreateResponse {\n const assetPath = response.asset_path ?? {};\n return {\n asset: {\n id: assetPath.id ?? 0,\n canonical_path: assetPath.path ?? \"\",\n name: assetPath.asset_code ?? \"\",\n },\n meta: {\n request_id: response.meta?.request_id ?? \"\",\n timestamp: response.meta?.timestamp ?? new Date().toISOString(),\n },\n };\n}\n\n/**\n * Creates a FilePickerApi adapter backed by the portal-tenant BFF's\n * `/api/content/dam/*` endpoints, using cookie-based auth via the\n * provided FetchClient.\n *\n * Unsplash search is not available through the BFF — callers that need\n * Unsplash should use the legacy adapter or provide their own\n * implementation.\n *\n * The `onProgress` callback in `createDamAsset` is not supported — the\n * underlying `fetch` API does not expose upload progress events.\n *\n * URL proxy delegates to the same-origin `/api/proxy-url` endpoint\n * (served by the hosting app, not the BFF) for CORS bypass, identical\n * to the legacy adapter behaviour.\n */\nexport function createPortalTenantFilePickerApiAdapter(\n client: FetchClientInstance,\n): FilePickerApi {\n return {\n createDamAsset: async (\n params: CreateDamAssetParams,\n ): Promise<DamAssetCreateResponse> => {\n // The generated dam_assets_create uses client.post (JSON body), but\n // file uploads require multipart/form-data. Use requestWithFormData\n // directly, typing the response from the generated operation type.\n const formData = new FormData();\n formData.append(\"asset[file]\", params.file);\n formData.append(\"asset[name]\", params.name);\n\n if (params.description) {\n formData.append(\"asset[description]\", params.description);\n }\n\n if (params.tags && params.tags.length > 0) {\n formData.append(\"asset[tags]\", params.tags.join(\",\"));\n }\n\n const response =\n await client.requestWithFormData<DamAssetCreateBffResponse>(\n \"/api/content/dam/assets\",\n formData,\n { method: \"POST\" },\n );\n\n return mapCreateResponse(response);\n },\n\n queryDamAssets: async (\n params: DamQueryParams,\n ): Promise<DamQueryResponse> => {\n const response = await portalTenantContent.dam_query(\n client,\n params satisfies operations[\"dam_query\"][\"requestBody\"][\"content\"][\"application/json\"],\n );\n return damQueryResponseSchema.parse({\n ...response,\n meta: response.meta\n ? { next_cursor: response.meta.pagination?.next_cursor ?? undefined }\n : undefined,\n });\n },\n\n deleteDamAsset: async (code: string): Promise<unknown> => {\n return portalTenantContent.dam_assets_destroy(client, code);\n },\n\n discardDamAsset: async (code: string): Promise<unknown> => {\n return portalTenantContent.dam_assets_discard(client, code);\n },\n\n createDamAssetPathForAssets: async (\n params: CreateDamAssetPathForAssetsParams,\n ): Promise<DamAssetPathCreateResponse> => {\n const response = await portalTenantContent.dam_asset_paths_create(\n client,\n params.code,\n { asset_path: { path: params.asset_paths.join(\",\") } },\n );\n return mapAssetPathCreateResponse(response);\n },\n\n searchUnsplash: async (\n _query: string,\n _page?: number,\n _perPage?: number,\n ): Promise<UnsplashSearchResponse> => {\n throw new Error(\n \"Unsplash search is not available through the portal-tenant BFF. \" +\n \"Configure an Unsplash access key and use the standard FilePickerApi adapter instead.\",\n );\n },\n\n proxyUrlFetch: (url: string) => proxyUrlFetch(url),\n };\n}\n","import { useCallback, useMemo, type ComponentProps } from \"react\";\nimport type {\n BackgroundValue,\n BorderRadiusOptions,\n ColorOptions,\n PaddingOptions,\n} from \"../types\";\nimport type { WidgetPropertySchema } from \"../registries/property-schema-types\";\nimport {\n ShareablesCoreProvider,\n ShareablesApiProvider,\n} from \"@fluid-app/shareables-core\";\nimport { ShareablesUIProvider, ShareablesApp } from \"@fluid-app/shareables-ui\";\nimport {\n createPortalTenantPlaylistsAdapter,\n createPortalTenantMediaShareablesAdapter,\n createPortalTenantPlaylistsShareablesAdapter,\n createPortalTenantFilesShareablesAdapter,\n createPortalTenantSharesShareablesAdapter,\n createPortalTenantProductMediaShareablesAdapter,\n} from \"@fluid-app/shareables-api-client\";\nimport { createPortalTenantFilePickerApiAdapter } from \"@fluid-app/file-picker-api-client\";\nimport { createFetchClient } from \"@fluid-app/api-client-core\";\nimport { useCurrentUser } from \"../hooks/use-current-user\";\nimport { useAppNavigation } from \"../shell/AppNavigationContext\";\nimport { useSdkClient } from \"../account/use-account-clients\";\nimport { usePortalTenantClient } from \"../providers/PortalTenantClientProvider\";\nimport { useFluidContext } from \"../providers/FluidProvider\";\nimport { useUserType } from \"../hooks/use-user-type\";\nimport { usePortalProductsClient } from \"../products/use-portal-products-client\";\n\ntype ShareablesScreenProps = ComponentProps<\"div\"> & {\n background?: BackgroundValue;\n textColor?: ColorOptions;\n accentColor?: ColorOptions;\n padding?: PaddingOptions;\n borderRadius?: BorderRadiusOptions;\n};\n\n/**\n * Parse the current shareables sub-route from the full slug.\n *\n * System nav slugs are \"share/products\", \"share/media\", \"share/playlists\".\n * Detail pages append an ID: \"share/products/123\", \"share/media/456\".\n *\n * \"share/products\" → screen=\"products\", detailId=null\n * \"share/products/123\" → screen=\"products\", detailId=\"123\"\n * \"share/media/456\" → screen=\"media\", detailId=\"456\"\n * \"share/playlists\" → screen=\"playlists\", detailId=null\n * \"share/playlists/789\" → screen=\"playlists\", detailId=\"789\"\n * \"share/files\" → screen=\"files\", detailId=null\n * \"share\" → screen=null (default to products)\n */\nfunction parseShareablesRoute(currentSlug: string): {\n screen: string | null;\n detailId: string | null;\n action: string | null;\n} {\n // Strip the \"share\" prefix\n const slugWithoutPrefix = currentSlug.replace(/^share\\/?/, \"\");\n if (!slugWithoutPrefix) {\n return { screen: null, detailId: null, action: null };\n }\n\n const parts = slugWithoutPrefix.split(\"/\");\n const screen = parts[0] || null;\n const detailId = parts[1] || null;\n const action = parts[2] || null;\n return { screen, detailId, action };\n}\n\nexport function ShareablesScreen({\n /* eslint-disable @typescript-eslint/no-unused-vars -- destructured to exclude from divProps spread */\n background,\n textColor,\n accentColor,\n padding,\n borderRadius,\n /* eslint-enable @typescript-eslint/no-unused-vars */\n ...divProps\n}: ShareablesScreenProps): React.JSX.Element {\n const domainClient = useSdkClient();\n const portalProductsApi = usePortalProductsClient();\n const portalTenantClient = usePortalTenantClient();\n const { config } = useFluidContext();\n const { data: userData } = useCurrentUser();\n const { currentSlug, navigate } = useAppNavigation();\n const { isCustomer } = useUserType();\n\n const fetchProducts = useCallback(\n async (search: string, cursor?: string, limit?: number) => {\n if (search) {\n return portalProductsApi.searchProducts(search, { cursor, limit });\n }\n return portalProductsApi.listProducts({ cursor, limit });\n },\n [portalProductsApi],\n );\n\n const fetchProduct = useCallback(\n async (id: string | number) => portalProductsApi.getProduct(id),\n [portalProductsApi],\n );\n\n const { screen, detailId, action } = parseShareablesRoute(currentSlug);\n\n const handleNavigate = useCallback(\n (subScreen: string, id?: string) => {\n const path = id ? `share/${subScreen}/${id}` : `share/${subScreen}`;\n navigate(path);\n },\n [navigate],\n );\n\n const handleBack = useCallback(() => {\n if (detailId && screen) {\n // Navigate back to the listing for the current screen\n navigate(`share/${screen}`);\n } else {\n // Navigate to the default shareables screen\n navigate(\"share/products\");\n }\n }, [navigate, detailId, screen]);\n\n const coreConfig = useMemo(\n () => ({\n client: domainClient,\n user: userData ? { id: userData.id } : null,\n repContext: true,\n }),\n [domainClient, userData],\n );\n\n // Create the port-level playlists adapter for the standalone deletePlaylist\n // callback (which uses the port interface directly, not ShareablesApi).\n const playlistsAdapter = useMemo(\n () => createPortalTenantPlaylistsAdapter(portalTenantClient),\n [portalTenantClient],\n );\n\n const mediaAdapter = useMemo(\n () => createPortalTenantMediaShareablesAdapter(portalTenantClient),\n [portalTenantClient],\n );\n const playlistsShareablesAdapter = useMemo(\n () => createPortalTenantPlaylistsShareablesAdapter(portalTenantClient),\n [portalTenantClient],\n );\n const filesAdapter = useMemo(\n () => createPortalTenantFilesShareablesAdapter(portalTenantClient),\n [portalTenantClient],\n );\n const sharesAdapter = useMemo(\n () => createPortalTenantSharesShareablesAdapter(portalTenantClient),\n [portalTenantClient],\n );\n const productMediaAdapter = useMemo(\n () => createPortalTenantProductMediaShareablesAdapter(portalProductsApi),\n [portalProductsApi],\n );\n\n const filePickerApi = useMemo(() => {\n const baseUrl = config.baseUrl.replace(/\\/+$/, \"\").replace(/\\/api$/, \"\");\n const csrfToken =\n typeof document !== \"undefined\"\n ? document\n .querySelector('meta[name=\"csrf-token\"]')\n ?.getAttribute(\"content\")\n : null;\n\n const bffClient = createFetchClient({\n baseUrl,\n getAuthToken: config.getAuthToken,\n onAuthError: config.onAuthError,\n credentials: \"include\",\n defaultHeaders: {\n ...config.defaultHeaders,\n ...(csrfToken ? { \"X-CSRF-Token\": csrfToken } : {}),\n },\n });\n\n return createPortalTenantFilePickerApiAdapter(bffClient);\n }, [\n config.baseUrl,\n config.getAuthToken,\n config.onAuthError,\n config.defaultHeaders,\n ]);\n\n const uiConfig = useMemo(\n () => ({\n user: userData\n ? {\n id: userData.id,\n company: userData.company\n ? { logo_url: userData.company.logo_url }\n : null,\n }\n : undefined,\n affiliateId:\n (userData as { affiliate_id?: number } | undefined)?.affiliate_id ??\n null,\n basePath: \"\",\n navigate: (path: string) => {\n // Strip leading slash — cards generate paths like \"/share/product/123\"\n const cleanPath = path.replace(/^\\//, \"\");\n // Ensure share/ prefix — screen components pass relative paths like \"media/new\"\n const prefixed = cleanPath.startsWith(\"share/\")\n ? cleanPath\n : `share/${cleanPath}`;\n navigate(prefixed);\n },\n showToast: (opts: {\n title: string;\n type: \"success\" | \"error\" | \"warning\";\n }) => {\n console.warn(`[Shareables] ${opts.type}: ${opts.title}`);\n },\n filePickerApi,\n onToggleFavorite: async (params: {\n favoriteableId: number;\n favoriteableType: string;\n }) => {\n const affiliateId = (userData as { affiliate_id?: number } | undefined)\n ?.affiliate_id;\n if (!affiliateId) throw new Error(\"No affiliate ID\");\n return domainClient.post<{ is_favorited: boolean }>(\n `/user_companies/${affiliateId}/favorites/toggle.json`,\n {\n favoriteable_id: params.favoriteableId,\n favoriteable_type: params.favoriteableType,\n },\n );\n },\n onDeletePlaylist: isCustomer\n ? undefined\n : async (playlistId: number) => {\n await playlistsAdapter.deletePlaylist(playlistId);\n },\n readOnly: isCustomer,\n }),\n [\n userData,\n navigate,\n filePickerApi,\n domainClient,\n isCustomer,\n playlistsAdapter,\n ],\n );\n\n return (\n <div {...divProps} className={`h-full ${divProps.className ?? \"\"}`}>\n <ShareablesCoreProvider config={coreConfig}>\n <ShareablesApiProvider\n media={mediaAdapter}\n playlists={playlistsShareablesAdapter}\n fileResources={filesAdapter}\n share={sharesAdapter}\n productMedia={productMediaAdapter}\n >\n <ShareablesUIProvider config={uiConfig}>\n <ShareablesApp\n screen={screen}\n detailId={detailId}\n action={action}\n companyLogoUrl={userData?.company?.logo_url}\n countryCode={userData?.country?.iso}\n fetchProducts={fetchProducts}\n fetchProduct={fetchProduct}\n onNavigate={handleNavigate}\n onBack={handleBack}\n />\n </ShareablesUIProvider>\n </ShareablesApiProvider>\n </ShareablesCoreProvider>\n </div>\n );\n}\n\nexport const shareablesScreenPropertySchema: WidgetPropertySchema = {\n widgetType: \"ShareablesScreen\",\n displayName: \"Shareables Screen\",\n tabsConfig: [{ id: \"styling\", label: \"Styling\" }],\n fields: [],\n} as const satisfies WidgetPropertySchema;\n"],"mappings":"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"}
package/dist/index.cjs CHANGED
@@ -38,7 +38,7 @@ const require_use_fluid_auth = require("./use-fluid-auth-DCaxdBCx.cjs");
38
38
  const require_ScreenHeaderContext = require("./ScreenHeaderContext-DpVOhFzT.cjs");
39
39
  const require_use_fluid_api = require("./use-fluid-api-GKNBxMzS.cjs");
40
40
  const require_use_current_user = require("./use-current-user-C4oXzxPk.cjs");
41
- const require_ShareablesScreen = require("./ShareablesScreen-DDutXKpW.cjs");
41
+ const require_ShareablesScreen = require("./ShareablesScreen-B-uZ3B5d.cjs");
42
42
  const require_CustomersScreen = require("./CustomersScreen-9f10-_AZ.cjs");
43
43
  const require_AppNavigationContext = require("./AppNavigationContext-Agp0UkCQ.cjs");
44
44
  const require_MessagingScreen = require("./MessagingScreen-C56qX8iT.cjs");
@@ -51,9 +51,9 @@ require("./src-ClULVolb.cjs");
51
51
  require("./dist-DkYXiQQ1.cjs");
52
52
  const require_OrdersScreen = require("./OrdersScreen-DrDgxVab.cjs");
53
53
  const require_SubscriptionsScreen = require("./SubscriptionsScreen-hLqmanJ5.cjs");
54
- require("./src-DmzXXBVt.cjs");
54
+ require("./portal-tenant-product-media-adapter-DM94zJXA.cjs");
55
55
  require("./dist-BSbac3uh.cjs");
56
- const require_ProductsScreen = require("./ProductsScreen-B1OAGNQx.cjs");
56
+ const require_ProductsScreen = require("./ProductsScreen-C7cXqLN8.cjs");
57
57
  const require_MySiteScreen = require("./MySiteScreen-D_MIuyR_.cjs");
58
58
  const require_ShopScreen = require("./ShopScreen-BWySUJaH.cjs");
59
59
  require("./UpgradeScreen-B7G9xtTu.cjs");
@@ -3007,8 +3007,8 @@ const MessagingScreen$1 = (0, react.lazy)(() => Promise.resolve().then(() => req
3007
3007
  const ContactsScreen$1 = (0, react.lazy)(() => Promise.resolve().then(() => require("./ContactsScreen-BWyCXc_1.cjs")).then((m) => ({ default: m.ContactsScreen })));
3008
3008
  const ShopScreen$1 = (0, react.lazy)(() => Promise.resolve().then(() => require("./ShopScreen-DoXTlY1X.cjs")).then((m) => ({ default: m.ShopScreen })));
3009
3009
  const CustomersScreen$1 = (0, react.lazy)(() => Promise.resolve().then(() => require("./CustomersScreen-9f10-_AZ.cjs")).then((n) => n.CustomersScreen_exports).then((m) => ({ default: m.CustomersScreen })));
3010
- const ProductsScreen$1 = (0, react.lazy)(() => Promise.resolve().then(() => require("./ProductsScreen-T1tGuiGy.cjs")).then((m) => ({ default: m.ProductsScreen })));
3011
- const ShareablesScreen$1 = (0, react.lazy)(() => Promise.resolve().then(() => require("./ShareablesScreen-BLbzr0Jb.cjs")).then((m) => ({ default: m.ShareablesScreen })));
3010
+ const ProductsScreen$1 = (0, react.lazy)(() => Promise.resolve().then(() => require("./ProductsScreen-CcF3f9VD.cjs")).then((m) => ({ default: m.ProductsScreen })));
3011
+ const ShareablesScreen$1 = (0, react.lazy)(() => Promise.resolve().then(() => require("./ShareablesScreen-DAM01cxf.cjs")).then((m) => ({ default: m.ShareablesScreen })));
3012
3012
  const MySiteScreen$1 = (0, react.lazy)(() => Promise.resolve().then(() => require("./MySiteScreen-oACgdq20.cjs")).then((m) => ({ default: m.MySiteScreen })));
3013
3013
  const UpgradeScreen = (0, react.lazy)(() => Promise.resolve().then(() => require("./UpgradeScreen-BbbMIjwQ.cjs")).then((m) => ({ default: m.UpgradeScreen })));
3014
3014
  const AppDownloadScreen = (0, react.lazy)(() => Promise.resolve().then(() => require("./AppDownloadScreen-DWiRvbFR.cjs")).then((m) => ({ default: m.AppDownloadScreen })));
@@ -4482,9 +4482,9 @@ const screenPropertySchemas = {
4482
4482
  OrdersScreen: () => Promise.resolve().then(() => require("./OrdersScreen-2bdFEwkk.cjs")).then((m) => m.ordersScreenPropertySchema),
4483
4483
  SubscriptionsScreen: () => Promise.resolve().then(() => require("./SubscriptionsScreen-BzOtTg8X.cjs")).then((m) => m.subscriptionsScreenPropertySchema),
4484
4484
  CustomersScreen: () => Promise.resolve().then(() => require("./CustomersScreen-9f10-_AZ.cjs")).then((n) => n.CustomersScreen_exports).then((m) => m.customersScreenPropertySchema),
4485
- ProductsScreen: () => Promise.resolve().then(() => require("./ProductsScreen-T1tGuiGy.cjs")).then((m) => m.productsScreenPropertySchema),
4485
+ ProductsScreen: () => Promise.resolve().then(() => require("./ProductsScreen-CcF3f9VD.cjs")).then((m) => m.productsScreenPropertySchema),
4486
4486
  MySiteScreen: () => Promise.resolve().then(() => require("./MySiteScreen-oACgdq20.cjs")).then((m) => m.mySiteScreenPropertySchema),
4487
- ShareablesScreen: () => Promise.resolve().then(() => require("./ShareablesScreen-BLbzr0Jb.cjs")).then((m) => m.shareablesScreenPropertySchema),
4487
+ ShareablesScreen: () => Promise.resolve().then(() => require("./ShareablesScreen-DAM01cxf.cjs")).then((m) => m.shareablesScreenPropertySchema),
4488
4488
  ShopScreen: () => Promise.resolve().then(() => require("./ShopScreen-DoXTlY1X.cjs")).then((m) => m.shopScreenPropertySchema),
4489
4489
  UpgradeScreen: () => Promise.resolve().then(() => require("./UpgradeScreen-BbbMIjwQ.cjs")).then((m) => m.upgradeScreenPropertySchema),
4490
4490
  AppDownloadScreen: () => Promise.resolve().then(() => require("./AppDownloadScreen-DWiRvbFR.cjs")).then((m) => m.appDownloadScreenPropertySchema)