@fluid-app/portal-sdk 0.1.162 → 0.1.164

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 (69) hide show
  1. package/dist/{MySiteScreen-g8Esyofu.mjs → MySiteScreen-CG2DnzQH.mjs} +2 -2
  2. package/dist/{MySiteScreen-g8Esyofu.mjs.map → MySiteScreen-CG2DnzQH.mjs.map} +1 -1
  3. package/dist/{MySiteScreen-Cos0jTa_.cjs → MySiteScreen-DRMrEMEI.cjs} +2 -2
  4. package/dist/{MySiteScreen-Xcg07a3M.cjs → MySiteScreen-Lr2SbrXz.cjs} +2 -2
  5. package/dist/{MySiteScreen-Xcg07a3M.cjs.map → MySiteScreen-Lr2SbrXz.cjs.map} +1 -1
  6. package/dist/{OrdersScreen-DZyOBKmU.cjs → OrdersScreen-B6n41CbG.cjs} +1 -1
  7. package/dist/{OrdersScreen-ZGUm8buk.cjs → OrdersScreen-BKCQdz5A.cjs} +55 -56
  8. package/dist/OrdersScreen-BKCQdz5A.cjs.map +1 -0
  9. package/dist/{OrdersScreen-DeLoyVGI.mjs → OrdersScreen-CFRVfzez.mjs} +55 -56
  10. package/dist/OrdersScreen-CFRVfzez.mjs.map +1 -0
  11. package/dist/{use-portal-shareables-api-DRK9Y5dp.mjs → PortalContentApiProvider-CW0ADhPi.mjs} +285 -94
  12. package/dist/PortalContentApiProvider-CW0ADhPi.mjs.map +1 -0
  13. package/dist/{use-portal-shareables-api-DcjYlAOy.cjs → PortalContentApiProvider-Di5emtYd.cjs} +290 -129
  14. package/dist/PortalContentApiProvider-Di5emtYd.cjs.map +1 -0
  15. package/dist/PortalProductsApiProvider-BCXX9NGK.mjs +780 -0
  16. package/dist/PortalProductsApiProvider-BCXX9NGK.mjs.map +1 -0
  17. package/dist/PortalProductsApiProvider-BquMHwvt.cjs +816 -0
  18. package/dist/PortalProductsApiProvider-BquMHwvt.cjs.map +1 -0
  19. package/dist/{ProductsScreen-B2SKzTE4.cjs → ProductsScreen-C6eNgxjP.cjs} +4 -3
  20. package/dist/{ProductsScreen-D6eoU86k.mjs → ProductsScreen-CVNJudq9.mjs} +39 -44
  21. package/dist/ProductsScreen-CVNJudq9.mjs.map +1 -0
  22. package/dist/{ProductsScreen-DbwSCY7G.cjs → ProductsScreen-KjjhlDGo.cjs} +39 -44
  23. package/dist/ProductsScreen-KjjhlDGo.cjs.map +1 -0
  24. package/dist/{ProductsScreen-BGah2tcD.mjs → ProductsScreen-nHmUftQn.mjs} +4 -3
  25. package/dist/{ShareablesScreen-DRUT-yoi.mjs → ShareablesScreen-CdTyyDRO.mjs} +4 -3
  26. package/dist/{ShareablesScreen-Dy04EXe5.cjs → ShareablesScreen-DUzo8kRi.cjs} +4 -3
  27. package/dist/ShareablesScreen-DpEP_6u0.mjs +132 -0
  28. package/dist/ShareablesScreen-DpEP_6u0.mjs.map +1 -0
  29. package/dist/ShareablesScreen-Dz8w2l3e.cjs +144 -0
  30. package/dist/ShareablesScreen-Dz8w2l3e.cjs.map +1 -0
  31. package/dist/{ShopScreen-CQ48b1c8.mjs → ShopScreen-BH6zQndJ.mjs} +10 -724
  32. package/dist/ShopScreen-BH6zQndJ.mjs.map +1 -0
  33. package/dist/{ShopScreen-CFqoT4IC.cjs → ShopScreen-BUXUtEuj.cjs} +13 -727
  34. package/dist/ShopScreen-BUXUtEuj.cjs.map +1 -0
  35. package/dist/{ShopScreen-B7faQx84.cjs → ShopScreen-BbucUNI7.cjs} +2 -1
  36. package/dist/{SubscriptionsScreen-C2iORyT_.cjs → SubscriptionsScreen-BdGF5OLE.cjs} +408 -409
  37. package/dist/SubscriptionsScreen-BdGF5OLE.cjs.map +1 -0
  38. package/dist/{SubscriptionsScreen-ClWrrqPK.cjs → SubscriptionsScreen-Cwa2lR1D.cjs} +1 -1
  39. package/dist/{SubscriptionsScreen-BfdK8067.mjs → SubscriptionsScreen-Dn3AEUJi.mjs} +408 -409
  40. package/dist/SubscriptionsScreen-Dn3AEUJi.mjs.map +1 -0
  41. package/dist/{dist-lO2OG0T5.cjs → dist-BF_4vk1z.cjs} +1 -1
  42. package/dist/{dist-lO2OG0T5.cjs.map → dist-BF_4vk1z.cjs.map} +1 -1
  43. package/dist/index.cjs +21 -20
  44. package/dist/index.cjs.map +1 -1
  45. package/dist/index.d.cts.map +1 -1
  46. package/dist/index.d.mts.map +1 -1
  47. package/dist/index.mjs +21 -20
  48. package/dist/index.mjs.map +1 -1
  49. package/dist/{sortable.esm-DSrWP4x9.mjs → sortable.esm-E6JdQn7I.mjs} +1 -1
  50. package/dist/{sortable.esm-DSrWP4x9.mjs.map → sortable.esm-E6JdQn7I.mjs.map} +1 -1
  51. package/package.json +17 -17
  52. package/dist/OrdersScreen-DeLoyVGI.mjs.map +0 -1
  53. package/dist/OrdersScreen-ZGUm8buk.cjs.map +0 -1
  54. package/dist/ProductsScreen-D6eoU86k.mjs.map +0 -1
  55. package/dist/ProductsScreen-DbwSCY7G.cjs.map +0 -1
  56. package/dist/ShareablesScreen-DPHZMh-V.cjs +0 -391
  57. package/dist/ShareablesScreen-DPHZMh-V.cjs.map +0 -1
  58. package/dist/ShareablesScreen-DrQDQ1-U.mjs +0 -379
  59. package/dist/ShareablesScreen-DrQDQ1-U.mjs.map +0 -1
  60. package/dist/ShopScreen-CFqoT4IC.cjs.map +0 -1
  61. package/dist/ShopScreen-CQ48b1c8.mjs.map +0 -1
  62. package/dist/SubscriptionsScreen-BfdK8067.mjs.map +0 -1
  63. package/dist/SubscriptionsScreen-C2iORyT_.cjs.map +0 -1
  64. package/dist/use-portal-products-client-BUFD20ZY.mjs +0 -65
  65. package/dist/use-portal-products-client-BUFD20ZY.mjs.map +0 -1
  66. package/dist/use-portal-products-client-DTkFvOal.cjs +0 -71
  67. package/dist/use-portal-products-client-DTkFvOal.cjs.map +0 -1
  68. package/dist/use-portal-shareables-api-DRK9Y5dp.mjs.map +0 -1
  69. package/dist/use-portal-shareables-api-DcjYlAOy.cjs.map +0 -1
@@ -1,391 +0,0 @@
1
- require("./chunk-9hOWP6kD.cjs");
2
- const require_PortalTenantClientProvider = require("./PortalTenantClientProvider-C4Gnq5tJ.cjs");
3
- const require_use_account = require("./use-account-CrgtI83z.cjs");
4
- const require_AppNavigationContext = require("./AppNavigationContext-BiYNDj_X.cjs");
5
- const require_use_portal_shareables_api = require("./use-portal-shareables-api-DcjYlAOy.cjs");
6
- let react = require("react");
7
- let react_jsx_runtime = require("react/jsx-runtime");
8
- let zod = require("zod");
9
- //#region ../../file-picker/core/src/schemas/dam.ts
10
- const damVariantSchema = zod.z.object({
11
- id: zod.z.string(),
12
- url: zod.z.string().nullable(),
13
- file_name: zod.z.string(),
14
- mime_type: zod.z.string(),
15
- content: zod.z.any().nullable(),
16
- created_at: zod.z.string(),
17
- updated_at: zod.z.string(),
18
- default: zod.z.boolean(),
19
- is_original: zod.z.boolean(),
20
- is_text: zod.z.boolean(),
21
- media_type: zod.z.string(),
22
- processing_status: zod.z.string(),
23
- tags: zod.z.array(zod.z.string())
24
- });
25
- const damAssetSchema = zod.z.object({
26
- id: zod.z.number(),
27
- canonical_path: zod.z.string(),
28
- category: zod.z.string(),
29
- code: zod.z.string(),
30
- company: zod.z.string(),
31
- created_at: zod.z.string(),
32
- default_variant_id: zod.z.string(),
33
- default_variant_url: zod.z.string().optional(),
34
- description: zod.z.string(),
35
- name: zod.z.string(),
36
- updated_at: zod.z.string(),
37
- variants: zod.z.array(damVariantSchema).optional()
38
- });
39
- const damTreeFolderNodeSchema = zod.z.object({
40
- asset_code: zod.z.union([zod.z.string(), zod.z.record(zod.z.string(), zod.z.unknown())]).optional(),
41
- name: zod.z.union([zod.z.string(), zod.z.record(zod.z.string(), zod.z.unknown())]).optional(),
42
- category: zod.z.union([zod.z.string(), zod.z.record(zod.z.string(), zod.z.unknown())]).optional(),
43
- variants: zod.z.union([zod.z.array(zod.z.unknown()), zod.z.record(zod.z.string(), zod.z.unknown())]).optional()
44
- }).passthrough();
45
- const damTreeSchema = zod.z.record(zod.z.string(), zod.z.union([
46
- zod.z.lazy(() => damTreeSchema),
47
- damAssetSchema,
48
- damTreeFolderNodeSchema
49
- ]));
50
- const damQueryResponseSchema = zod.z.object({
51
- path: zod.z.string(),
52
- tree: damTreeSchema,
53
- meta: zod.z.object({ next_cursor: zod.z.string().optional() }).optional()
54
- });
55
- zod.z.object({ asset: zod.z.object({
56
- file: zod.z.any(),
57
- name: zod.z.string(),
58
- description: zod.z.string().optional(),
59
- tags: zod.z.string().optional()
60
- }) });
61
- zod.z.object({
62
- asset: damAssetSchema,
63
- meta: zod.z.object({
64
- request_id: zod.z.string(),
65
- timestamp: zod.z.string()
66
- })
67
- });
68
- zod.z.object({
69
- placeholder_asset: zod.z.object({
70
- mime_type: zod.z.string(),
71
- name: zod.z.string().optional(),
72
- description: zod.z.string().optional()
73
- }),
74
- skip_autotagging: zod.z.boolean().optional()
75
- });
76
- zod.z.object({
77
- asset: zod.z.object({
78
- file: zod.z.any(),
79
- name: zod.z.string(),
80
- description: zod.z.string().optional(),
81
- tags: zod.z.string().optional()
82
- }).optional(),
83
- text_asset: zod.z.object({
84
- file_name: zod.z.string(),
85
- mime_type: zod.z.string(),
86
- text: zod.z.string(),
87
- name: zod.z.string().optional(),
88
- description: zod.z.string().optional(),
89
- tags: zod.z.string().optional()
90
- }).optional(),
91
- placeholder_asset: zod.z.object({
92
- mime_type: zod.z.string(),
93
- name: zod.z.string().optional(),
94
- description: zod.z.string().optional()
95
- }).optional(),
96
- skip_autotagging: zod.z.boolean().optional()
97
- });
98
- zod.z.object({
99
- asset: zod.z.object({
100
- id: zod.z.number(),
101
- canonical_path: zod.z.string(),
102
- name: zod.z.string()
103
- }),
104
- meta: zod.z.object({
105
- request_id: zod.z.string(),
106
- timestamp: zod.z.string()
107
- })
108
- });
109
- //#endregion
110
- //#region ../../file-picker/api-client/src/api/url-proxy.ts
111
- const urlProxyResponseSchema = zod.z.object({
112
- data: zod.z.string(),
113
- contentType: zod.z.string(),
114
- size: zod.z.number()
115
- });
116
- /**
117
- * Proxy a URL fetch through the backend to bypass CORS restrictions.
118
- * The backend fetches the file and returns it as base64-encoded data.
119
- *
120
- * @param url - The URL to fetch
121
- * @param proxyEndpoint - The proxy endpoint (defaults to "/api/proxy-url")
122
- */
123
- async function proxyUrlFetch(url, proxyEndpoint = "/api/proxy-url") {
124
- const response = await fetch(proxyEndpoint, {
125
- method: "POST",
126
- headers: { "Content-Type": "application/json" },
127
- body: JSON.stringify({ url })
128
- });
129
- if (!response.ok) {
130
- const errorData = await response.json().catch(() => ({ error: "Failed to proxy URL fetch" }));
131
- throw new Error(errorData.error || `HTTP ${response.status}`);
132
- }
133
- const data = await response.json();
134
- return urlProxyResponseSchema.parse(data);
135
- }
136
- //#endregion
137
- //#region ../../file-picker/api-client/src/portal-tenant-adapter.ts
138
- /**
139
- * Maps a BFF DAM asset to the file-picker port's DamAssetCreateResponse shape.
140
- *
141
- * The BFF response includes nullable meta.request_id (from Api::Response),
142
- * while the port schema requires a string. We coalesce to empty string.
143
- */
144
- function mapCreateResponse(response) {
145
- const raw = response.asset ?? {};
146
- return {
147
- asset: damAssetSchema.parse({
148
- ...raw,
149
- canonical_path: raw.canonical_path ?? "",
150
- category: raw.category ?? "",
151
- company: raw.company ?? "",
152
- description: raw.description ?? "",
153
- default_variant_id: raw.default_variant_id ?? ""
154
- }),
155
- meta: {
156
- request_id: response.meta?.request_id ?? "",
157
- timestamp: response.meta?.timestamp ?? (/* @__PURE__ */ new Date()).toISOString()
158
- }
159
- };
160
- }
161
- /**
162
- * Maps a BFF asset path response to the file-picker port's shape.
163
- */
164
- function mapAssetPathCreateResponse(response) {
165
- const assetPath = response.asset_path ?? {};
166
- return {
167
- asset: {
168
- id: assetPath.id ?? 0,
169
- canonical_path: assetPath.path ?? "",
170
- name: assetPath.asset_code ?? ""
171
- },
172
- meta: {
173
- request_id: response.meta?.request_id ?? "",
174
- timestamp: response.meta?.timestamp ?? (/* @__PURE__ */ new Date()).toISOString()
175
- }
176
- };
177
- }
178
- /**
179
- * Creates a FilePickerApi adapter backed by the portal-tenant BFF's
180
- * `/api/content/dam/*` endpoints, using cookie-based auth via the
181
- * provided FetchClient.
182
- *
183
- * Unsplash search is not available through the BFF — callers that need
184
- * Unsplash should use the legacy adapter or provide their own
185
- * implementation.
186
- *
187
- * The `onProgress` callback in `createDamAsset` is not supported — the
188
- * underlying `fetch` API does not expose upload progress events.
189
- *
190
- * URL proxy delegates to the same-origin `/api/proxy-url` endpoint
191
- * (served by the hosting app, not the BFF) for CORS bypass, identical
192
- * to the legacy adapter behaviour.
193
- */
194
- function createPortalTenantFilePickerApiAdapter(client) {
195
- return {
196
- createDamAsset: async (params) => {
197
- const formData = new FormData();
198
- formData.append("asset[file]", params.file);
199
- formData.append("asset[name]", params.name);
200
- if (params.description) formData.append("asset[description]", params.description);
201
- if (params.tags && params.tags.length > 0) formData.append("asset[tags]", params.tags.join(","));
202
- return mapCreateResponse(await client.requestWithFormData("/api/content/dam/assets", formData, { method: "POST" }));
203
- },
204
- queryDamAssets: async (params) => {
205
- const response = await require_use_portal_shareables_api.dam_query(client, params);
206
- return damQueryResponseSchema.parse({
207
- ...response,
208
- meta: response.meta ? { next_cursor: response.meta.pagination?.next_cursor ?? void 0 } : void 0
209
- });
210
- },
211
- deleteDamAsset: async (code) => {
212
- return require_use_portal_shareables_api.dam_assets_destroy(client, code);
213
- },
214
- discardDamAsset: async (code) => {
215
- return require_use_portal_shareables_api.dam_assets_discard(client, code);
216
- },
217
- createDamAssetPathForAssets: async (params) => {
218
- return mapAssetPathCreateResponse(await require_use_portal_shareables_api.dam_asset_paths_create(client, params.code, { asset_path: { path: params.asset_paths.join(",") } }));
219
- },
220
- searchUnsplash: async (_query, _page, _perPage) => {
221
- 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.");
222
- },
223
- proxyUrlFetch: (url) => proxyUrlFetch(url)
224
- };
225
- }
226
- //#endregion
227
- //#region src/shareables/use-portal-file-picker-api.ts
228
- /**
229
- * Returns a FilePickerApi adapter backed by the portal-tenant BFF client.
230
- * Screens consume this hook instead of importing the adapter factory directly.
231
- */
232
- function usePortalFilePickerApi() {
233
- const client = require_PortalTenantClientProvider.usePortalTenantClient();
234
- return (0, react.useMemo)(() => createPortalTenantFilePickerApiAdapter(client), [client]);
235
- }
236
- //#endregion
237
- //#region src/shareables/use-portal-playlists-api.ts
238
- /**
239
- * Returns the port-level ContentPlaylistsApi adapter backed by the
240
- * portal-tenant BFF client. Used for standalone playlist operations
241
- * (e.g. deletePlaylist) that operate outside ShareablesApiProvider.
242
- */
243
- function usePortalPlaylistsApi() {
244
- const client = require_PortalTenantClientProvider.usePortalTenantClient();
245
- return (0, react.useMemo)(() => require_use_portal_shareables_api.createPortalTenantPlaylistsAdapter(client), [client]);
246
- }
247
- //#endregion
248
- //#region src/screens/ShareablesScreen.tsx
249
- /**
250
- * Parse the current shareables sub-route from the full slug.
251
- *
252
- * System nav slugs are "share/products", "share/media", "share/playlists".
253
- * Detail pages append an ID: "share/products/123", "share/media/456".
254
- *
255
- * "share/products" → screen="products", detailId=null
256
- * "share/products/123" → screen="products", detailId="123"
257
- * "share/media/456" → screen="media", detailId="456"
258
- * "share/playlists" → screen="playlists", detailId=null
259
- * "share/playlists/789" → screen="playlists", detailId="789"
260
- * "share/files" → screen="files", detailId=null
261
- * "share" → screen=null (default to products)
262
- */
263
- function parseShareablesRoute(currentSlug) {
264
- const slugWithoutPrefix = currentSlug.replace(/^share\/?/, "");
265
- if (!slugWithoutPrefix) return {
266
- screen: null,
267
- detailId: null,
268
- action: null
269
- };
270
- const parts = slugWithoutPrefix.split("/");
271
- return {
272
- screen: parts[0] || null,
273
- detailId: parts[1] || null,
274
- action: parts[2] || null
275
- };
276
- }
277
- function ShareablesScreen({ background, textColor, accentColor, padding, borderRadius, ...divProps }) {
278
- const shareablesApi = require_use_portal_shareables_api.usePortalShareablesApi();
279
- const { productsApi: portalProductsApi } = shareablesApi;
280
- const filePickerApi = usePortalFilePickerApi();
281
- const playlistsAdapter = usePortalPlaylistsApi();
282
- const { data: account } = require_use_account.useAccount();
283
- const { currentSlug, navigate } = require_AppNavigationContext.useAppNavigation();
284
- const isCustomer = account?.member_type === "customer";
285
- const fetchProducts = (0, react.useCallback)(async (search, cursor, limit) => {
286
- if (search) return portalProductsApi.searchProducts(search, {
287
- cursor,
288
- limit
289
- });
290
- return portalProductsApi.listProducts({
291
- cursor,
292
- limit
293
- });
294
- }, [portalProductsApi]);
295
- const fetchProduct = (0, react.useCallback)(async (id) => portalProductsApi.getProduct(id), [portalProductsApi]);
296
- const { screen, detailId, action } = parseShareablesRoute(currentSlug);
297
- const handleNavigate = (0, react.useCallback)((subScreen, id) => {
298
- navigate(id ? `share/${subScreen}/${id}` : `share/${subScreen}`);
299
- }, [navigate]);
300
- const handleBack = (0, react.useCallback)(() => {
301
- if (detailId && screen) navigate(`share/${screen}`);
302
- else navigate("share/products");
303
- }, [
304
- navigate,
305
- detailId,
306
- screen
307
- ]);
308
- const coreConfig = (0, react.useMemo)(() => ({
309
- user: account ? { id: account.id } : null,
310
- repContext: true
311
- }), [account]);
312
- const uiConfig = (0, react.useMemo)(() => ({
313
- user: account ? {
314
- id: account.id,
315
- company: null
316
- } : void 0,
317
- affiliateId: null,
318
- basePath: "",
319
- navigate: (path) => {
320
- const cleanPath = path.replace(/^\//, "");
321
- navigate(cleanPath.startsWith("share/") ? cleanPath : `share/${cleanPath}`);
322
- },
323
- showToast: (opts) => {
324
- console.warn(`[Shareables] ${opts.type}: ${opts.title}`);
325
- },
326
- filePickerApi,
327
- onToggleFavorite: void 0,
328
- onDeletePlaylist: isCustomer ? void 0 : async (playlistId) => {
329
- await playlistsAdapter.deletePlaylist(playlistId);
330
- },
331
- readOnly: isCustomer
332
- }), [
333
- account,
334
- navigate,
335
- filePickerApi,
336
- isCustomer,
337
- playlistsAdapter
338
- ]);
339
- return /* @__PURE__ */ (0, react_jsx_runtime.jsx)("div", {
340
- ...divProps,
341
- className: `h-full ${divProps.className ?? ""}`,
342
- children: /* @__PURE__ */ (0, react_jsx_runtime.jsx)(require_use_portal_shareables_api.ShareablesCoreProvider, {
343
- config: coreConfig,
344
- children: /* @__PURE__ */ (0, react_jsx_runtime.jsx)(require_use_portal_shareables_api.ShareablesApiProvider, {
345
- media: shareablesApi.media,
346
- playlists: shareablesApi.playlists,
347
- fileResources: shareablesApi.fileResources,
348
- share: shareablesApi.share,
349
- productMedia: shareablesApi.productMedia,
350
- children: /* @__PURE__ */ (0, react_jsx_runtime.jsx)(require_use_portal_shareables_api.ShareablesUIProvider, {
351
- config: uiConfig,
352
- children: /* @__PURE__ */ (0, react_jsx_runtime.jsx)(require_use_portal_shareables_api.ShareablesApp, {
353
- screen,
354
- detailId,
355
- action,
356
- companyLogoUrl: void 0,
357
- countryCode: void 0,
358
- fetchProducts,
359
- fetchProduct,
360
- onNavigate: handleNavigate,
361
- onBack: handleBack
362
- })
363
- })
364
- })
365
- })
366
- });
367
- }
368
- const shareablesScreenPropertySchema = {
369
- widgetType: "ShareablesScreen",
370
- displayName: "Shareables Screen",
371
- tabsConfig: [{
372
- id: "styling",
373
- label: "Styling"
374
- }],
375
- fields: []
376
- };
377
- //#endregion
378
- Object.defineProperty(exports, "ShareablesScreen", {
379
- enumerable: true,
380
- get: function() {
381
- return ShareablesScreen;
382
- }
383
- });
384
- Object.defineProperty(exports, "shareablesScreenPropertySchema", {
385
- enumerable: true,
386
- get: function() {
387
- return shareablesScreenPropertySchema;
388
- }
389
- });
390
-
391
- //# sourceMappingURL=ShareablesScreen-DPHZMh-V.cjs.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"ShareablesScreen-DPHZMh-V.cjs","names":["z","z","usePortalTenantClient","usePortalTenantClient","createPortalTenantPlaylistsAdapter","usePortalShareablesApi","useAccount","useAppNavigation","ShareablesCoreProvider","ShareablesApiProvider","ShareablesUIProvider","ShareablesApp"],"sources":["../../../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/shareables/use-portal-file-picker-api.ts","../src/shareables/use-portal-playlists-api.ts","../src/screens/ShareablesScreen.tsx"],"sourcesContent":["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 { useMemo } from \"react\";\nimport type { FilePickerApi } from \"@fluid-app/file-picker-core\";\nimport { createPortalTenantFilePickerApiAdapter } from \"@fluid-app/file-picker-api-client\";\nimport { usePortalTenantClient } from \"../providers/PortalTenantClientProvider\";\n\n/**\n * Returns a FilePickerApi adapter backed by the portal-tenant BFF client.\n * Screens consume this hook instead of importing the adapter factory directly.\n */\nexport function usePortalFilePickerApi(): FilePickerApi {\n const client = usePortalTenantClient();\n return useMemo(\n () => createPortalTenantFilePickerApiAdapter(client),\n [client],\n );\n}\n","import { useMemo } from \"react\";\nimport type { ContentPlaylistsApi } from \"@fluid-app/shareables-core/playlists-api\";\nimport { createPortalTenantPlaylistsAdapter } from \"../adapters/shareables-api-adapter\";\nimport { usePortalTenantClient } from \"../providers/PortalTenantClientProvider\";\n\n/**\n * Returns the port-level ContentPlaylistsApi adapter backed by the\n * portal-tenant BFF client. Used for standalone playlist operations\n * (e.g. deletePlaylist) that operate outside ShareablesApiProvider.\n */\nexport function usePortalPlaylistsApi(): ContentPlaylistsApi {\n const client = usePortalTenantClient();\n return useMemo(() => createPortalTenantPlaylistsAdapter(client), [client]);\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 { useAppNavigation } from \"../shell/AppNavigationContext\";\nimport { useAccount } from \"../hooks/use-account\";\nimport { usePortalShareablesApi } from \"../shareables/use-portal-shareables-api\";\nimport { usePortalFilePickerApi } from \"../shareables/use-portal-file-picker-api\";\nimport { usePortalPlaylistsApi } from \"../shareables/use-portal-playlists-api\";\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 shareablesApi = usePortalShareablesApi();\n const { productsApi: portalProductsApi } = shareablesApi;\n const filePickerApi = usePortalFilePickerApi();\n const playlistsAdapter = usePortalPlaylistsApi();\n const { data: account } = useAccount();\n const { currentSlug, navigate } = useAppNavigation();\n const isCustomer = account?.member_type === \"customer\";\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(`share/${screen}`);\n } else {\n navigate(\"share/products\");\n }\n }, [navigate, detailId, screen]);\n\n const coreConfig = useMemo(\n () => ({\n user: account ? { id: account.id } : null,\n repContext: true,\n }),\n [account],\n );\n\n const uiConfig = useMemo(\n () => ({\n user: account\n ? {\n id: account.id,\n company: null, // TODO(portal-tenant): company branding not available from /api/account\n }\n : undefined,\n // TODO(portal-tenant): affiliate_id not available from /api/account\n affiliateId: null,\n basePath: \"\",\n navigate: (path: string) => {\n const cleanPath = path.replace(/^\\//, \"\");\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 // TODO(portal-tenant): affiliate_id not available from /api/account; favorites disabled\n onToggleFavorite: undefined,\n onDeletePlaylist: isCustomer\n ? undefined\n : async (playlistId: number) => {\n await playlistsAdapter.deletePlaylist(playlistId);\n },\n readOnly: isCustomer,\n }),\n [account, navigate, filePickerApi, isCustomer, playlistsAdapter],\n );\n\n return (\n <div {...divProps} className={`h-full ${divProps.className ?? \"\"}`}>\n <ShareablesCoreProvider config={coreConfig}>\n <ShareablesApiProvider\n media={shareablesApi.media}\n playlists={shareablesApi.playlists}\n fileResources={shareablesApi.fileResources}\n share={shareablesApi.share}\n productMedia={shareablesApi.productMedia}\n >\n <ShareablesUIProvider config={uiConfig}>\n <ShareablesApp\n screen={screen}\n detailId={detailId}\n action={action}\n companyLogoUrl={undefined}\n countryCode={undefined}\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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"}