@fluid-app/portal-sdk 0.1.153 → 0.1.154
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/{FluidProvider-BYRso8Tk.cjs → FluidProvider-CRMe-Enf.cjs} +8 -6
- package/dist/{FluidProvider-BYRso8Tk.cjs.map → FluidProvider-CRMe-Enf.cjs.map} +1 -1
- package/dist/{FluidProvider-BQBX3iae.mjs → FluidProvider-DpET43hN.mjs} +7 -5
- package/dist/{FluidProvider-BQBX3iae.mjs.map → FluidProvider-DpET43hN.mjs.map} +1 -1
- package/dist/{MessagingScreen-CWb3EA6D.cjs → MessagingScreen-ChCBVO4U.cjs} +2 -2
- package/dist/{MessagingScreen-BER99flG.mjs → MessagingScreen-DXgDrzlG.mjs} +2 -2
- package/dist/{MessagingScreen-BER99flG.mjs.map → MessagingScreen-DXgDrzlG.mjs.map} +1 -1
- package/dist/{MessagingScreen-BQnNoNip.cjs → MessagingScreen-cxQADd91.cjs} +2 -2
- package/dist/{MessagingScreen-BQnNoNip.cjs.map → MessagingScreen-cxQADd91.cjs.map} +1 -1
- package/dist/{ProductsScreen-xKstDLgs.cjs → ProductsScreen-BJ2ZUaPi.cjs} +3 -3
- package/dist/{ProductsScreen-CTGeFoGz.cjs → ProductsScreen-CfcVBUZ8.cjs} +2 -2
- package/dist/{ProductsScreen-CTGeFoGz.cjs.map → ProductsScreen-CfcVBUZ8.cjs.map} +1 -1
- package/dist/{ProductsScreen-B2VxoaL-.mjs → ProductsScreen-Dcnp0MG-.mjs} +3 -3
- package/dist/{ProductsScreen-DfUMO1ci.mjs → ProductsScreen-H-8HShV5.mjs} +2 -2
- package/dist/{ProductsScreen-DfUMO1ci.mjs.map → ProductsScreen-H-8HShV5.mjs.map} +1 -1
- package/dist/{ProfileScreen-8DOH9wDg.cjs → ProfileScreen-BRPBvvI7.cjs} +2 -2
- package/dist/{ProfileScreen-DtWrW4RA.mjs → ProfileScreen-CaNQYd_F.mjs} +2 -2
- package/dist/{ProfileScreen-DtWrW4RA.mjs.map → ProfileScreen-CaNQYd_F.mjs.map} +1 -1
- package/dist/{ProfileScreen-Ds4OpUkH.cjs → ProfileScreen-D4OJk9F0.cjs} +2 -2
- package/dist/{ProfileScreen-Ds4OpUkH.cjs.map → ProfileScreen-D4OJk9F0.cjs.map} +1 -1
- package/dist/{ShareablesScreen-Bp8zb0oX.cjs → ShareablesScreen-CL6A1m_P.cjs} +3 -3
- package/dist/{ShareablesScreen-Cqhe5iOS.mjs → ShareablesScreen-CtK4S7FT.mjs} +2 -2
- package/dist/{ShareablesScreen-Cqhe5iOS.mjs.map → ShareablesScreen-CtK4S7FT.mjs.map} +1 -1
- package/dist/{ShareablesScreen-BKbsEK2q.cjs → ShareablesScreen-DLz5tvRS.cjs} +2 -2
- package/dist/{ShareablesScreen-BKbsEK2q.cjs.map → ShareablesScreen-DLz5tvRS.cjs.map} +1 -1
- package/dist/{ShareablesScreen-CQ6ld0ol.mjs → ShareablesScreen-FDOtQ67f.mjs} +3 -3
- package/dist/{ShopScreen-tTRMdHfZ.cjs → ShopScreen-Bj3PjrVi.cjs} +2 -2
- package/dist/{ShopScreen-tTRMdHfZ.cjs.map → ShopScreen-Bj3PjrVi.cjs.map} +1 -1
- package/dist/{ShopScreen-DJTi8rx4.cjs → ShopScreen-CKidhX9F.cjs} +2 -2
- package/dist/{ShopScreen-8Oomx-4R.mjs → ShopScreen-DDscuexZ.mjs} +2 -2
- package/dist/{ShopScreen-8Oomx-4R.mjs.map → ShopScreen-DDscuexZ.mjs.map} +1 -1
- package/dist/index.cjs +85 -49
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +25 -1
- package/dist/index.d.cts.map +1 -1
- package/dist/index.d.mts +25 -1
- package/dist/index.d.mts.map +1 -1
- package/dist/index.mjs +84 -50
- package/dist/index.mjs.map +1 -1
- package/dist/{use-portal-shareables-api-Cr_bve2i.cjs → use-portal-shareables-api-BpQpcPcx.cjs} +2 -2
- package/dist/{use-portal-shareables-api-Cr_bve2i.cjs.map → use-portal-shareables-api-BpQpcPcx.cjs.map} +1 -1
- package/dist/{use-portal-shareables-api-DTk0758D.mjs → use-portal-shareables-api-kcUnux8J.mjs} +2 -2
- package/dist/{use-portal-shareables-api-DTk0758D.mjs.map → use-portal-shareables-api-kcUnux8J.mjs.map} +1 -1
- package/package.json +19 -19
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"FluidProvider-BQBX3iae.mjs","names":["#provider","ApiError","fluidOs.getFluidOSManifest","fluidOs.getFluidOSManifest","portalTenant.app_definition_show","portalTenant.account_show","portalTenant.account_update","portalTenantPay.addresses_list","portalTenantPay.addresses_create","portalTenantPay.addresses_update","portalTenantPay.addresses_destroy","portalTenantPay.payment_methods_list","portalTenantPay.payment_methods_create","portalTenantPay.payment_methods_update","portalTenantPay.payment_methods_destroy","portalTenantPay.payment_methods_vault_show","portalTenantPay.points_ledgers_list","mapMeta","portalTenantStore.countries_list","portalTenantStore.languages_list","portalTenantMysite.mysite_profile_show","portalTenantMysite.mysite_profile_update","portalTenantMysite.mysite_settings_update","portalTenantMysite.mysite_links_list","portalTenantMysite.mysite_links_create","portalTenantMysite.mysite_links_update","portalTenantMysite.mysite_links_destroy","portalTenantMysite.mysite_links_bulk_reorder","portalTenantMysite.mysite_favorites_list","portalTenantMysite.mysite_favorites_create","portalTenantMysite.mysite_favorites_destroy","portalTenantMysite.mysite_favorites_bulk_reorder","portalTenantMysite.mysite_themes_list","createPortalTenantFetchClient"],"sources":["../../../../node_modules/.pnpm/@tanstack+query-core@5.90.12/node_modules/@tanstack/query-core/build/modern/timeoutManager.js","../../../../node_modules/.pnpm/@tanstack+query-core@5.90.12/node_modules/@tanstack/query-core/build/modern/utils.js","../../../../node_modules/.pnpm/@tanstack+query-core@5.90.12/node_modules/@tanstack/query-core/build/modern/notifyManager.js","../../../../node_modules/.pnpm/@tanstack+query-persist-client-core@5.91.11/node_modules/@tanstack/query-persist-client-core/build/modern/createPersister.js","../../../platform/query-persister/src/persister.ts","../../core/src/types/fluidos-api-context.ts","../../core/src/app-definition-api-context.ts","../../core/src/pay-api-context.ts","../../../mysite/core/src/mysite-api-context.ts","../../../store/core/src/languages-api-context.ts","../../../platform/api-client-core/src/fetch-client.ts","../../../api-clients/fluidos/src/namespaces/fluid_os.ts","../src/client/types.ts","../../core/src/theme/types.ts","../../core/src/theme/color-engine.ts","../../core/src/theme/tailwind-overrides.ts","../../core/src/theme/css-generator.ts","../../core/src/theme/defaults.ts","../../core/src/theme/serialisation.ts","../../core/src/theme/transforms.ts","../../core/src/theme/theme-applicator.ts","../src/transforms/screen-transforms.ts","../src/transforms/navigation-transforms.ts","../src/transforms/index.ts","../src/client/fluid-client.ts","../src/adapters/fluidos-api-adapter.ts","../src/adapters/app-definition-api-adapter.ts","../src/adapters/account-api-adapter.ts","../../../api-clients/portal-tenant-pay/src/namespaces/portal_tenant_pay.ts","../src/adapters/pay-api-adapter.ts","../../../api-clients/portal-tenant-store/src/namespaces/portal_tenant_store.ts","../../../store/api-client/src/portal-tenant-countries-adapter.ts","../../../store/api-client/src/portal-tenant-languages-adapter.ts","../src/adapters/countries-api-adapter.ts","../src/adapters/languages-api-adapter.ts","../src/adapters/mysite-api-adapter.ts","../src/providers/FluidThemeProvider.tsx","../../core/src/widget-utils/utils.ts","../../widgets/src/widgets/index.ts","../src/core/default-widget-registry.ts","../src/providers/FluidProvider.tsx"],"sourcesContent":["// src/timeoutManager.ts\nvar defaultTimeoutProvider = {\n // We need the wrapper function syntax below instead of direct references to\n // global setTimeout etc.\n //\n // BAD: `setTimeout: setTimeout`\n // GOOD: `setTimeout: (cb, delay) => setTimeout(cb, delay)`\n //\n // If we use direct references here, then anything that wants to spy on or\n // replace the global setTimeout (like tests) won't work since we'll already\n // have a hard reference to the original implementation at the time when this\n // file was imported.\n setTimeout: (callback, delay) => setTimeout(callback, delay),\n clearTimeout: (timeoutId) => clearTimeout(timeoutId),\n setInterval: (callback, delay) => setInterval(callback, delay),\n clearInterval: (intervalId) => clearInterval(intervalId)\n};\nvar TimeoutManager = class {\n // We cannot have TimeoutManager<T> as we must instantiate it with a concrete\n // type at app boot; and if we leave that type, then any new timer provider\n // would need to support ReturnType<typeof setTimeout>, which is infeasible.\n //\n // We settle for type safety for the TimeoutProvider type, and accept that\n // this class is unsafe internally to allow for extension.\n #provider = defaultTimeoutProvider;\n #providerCalled = false;\n setTimeoutProvider(provider) {\n if (process.env.NODE_ENV !== \"production\") {\n if (this.#providerCalled && provider !== this.#provider) {\n console.error(\n `[timeoutManager]: Switching provider after calls to previous provider might result in unexpected behavior.`,\n { previous: this.#provider, provider }\n );\n }\n }\n this.#provider = provider;\n if (process.env.NODE_ENV !== \"production\") {\n this.#providerCalled = false;\n }\n }\n setTimeout(callback, delay) {\n if (process.env.NODE_ENV !== \"production\") {\n this.#providerCalled = true;\n }\n return this.#provider.setTimeout(callback, delay);\n }\n clearTimeout(timeoutId) {\n this.#provider.clearTimeout(timeoutId);\n }\n setInterval(callback, delay) {\n if (process.env.NODE_ENV !== \"production\") {\n this.#providerCalled = true;\n }\n return this.#provider.setInterval(callback, delay);\n }\n clearInterval(intervalId) {\n this.#provider.clearInterval(intervalId);\n }\n};\nvar timeoutManager = new TimeoutManager();\nfunction systemSetTimeoutZero(callback) {\n setTimeout(callback, 0);\n}\nexport {\n TimeoutManager,\n defaultTimeoutProvider,\n systemSetTimeoutZero,\n timeoutManager\n};\n//# sourceMappingURL=timeoutManager.js.map","// src/utils.ts\nimport { timeoutManager } from \"./timeoutManager.js\";\nvar isServer = typeof window === \"undefined\" || \"Deno\" in globalThis;\nfunction noop() {\n}\nfunction functionalUpdate(updater, input) {\n return typeof updater === \"function\" ? updater(input) : updater;\n}\nfunction isValidTimeout(value) {\n return typeof value === \"number\" && value >= 0 && value !== Infinity;\n}\nfunction timeUntilStale(updatedAt, staleTime) {\n return Math.max(updatedAt + (staleTime || 0) - Date.now(), 0);\n}\nfunction resolveStaleTime(staleTime, query) {\n return typeof staleTime === \"function\" ? staleTime(query) : staleTime;\n}\nfunction resolveEnabled(enabled, query) {\n return typeof enabled === \"function\" ? enabled(query) : enabled;\n}\nfunction matchQuery(filters, query) {\n const {\n type = \"all\",\n exact,\n fetchStatus,\n predicate,\n queryKey,\n stale\n } = filters;\n if (queryKey) {\n if (exact) {\n if (query.queryHash !== hashQueryKeyByOptions(queryKey, query.options)) {\n return false;\n }\n } else if (!partialMatchKey(query.queryKey, queryKey)) {\n return false;\n }\n }\n if (type !== \"all\") {\n const isActive = query.isActive();\n if (type === \"active\" && !isActive) {\n return false;\n }\n if (type === \"inactive\" && isActive) {\n return false;\n }\n }\n if (typeof stale === \"boolean\" && query.isStale() !== stale) {\n return false;\n }\n if (fetchStatus && fetchStatus !== query.state.fetchStatus) {\n return false;\n }\n if (predicate && !predicate(query)) {\n return false;\n }\n return true;\n}\nfunction matchMutation(filters, mutation) {\n const { exact, status, predicate, mutationKey } = filters;\n if (mutationKey) {\n if (!mutation.options.mutationKey) {\n return false;\n }\n if (exact) {\n if (hashKey(mutation.options.mutationKey) !== hashKey(mutationKey)) {\n return false;\n }\n } else if (!partialMatchKey(mutation.options.mutationKey, mutationKey)) {\n return false;\n }\n }\n if (status && mutation.state.status !== status) {\n return false;\n }\n if (predicate && !predicate(mutation)) {\n return false;\n }\n return true;\n}\nfunction hashQueryKeyByOptions(queryKey, options) {\n const hashFn = options?.queryKeyHashFn || hashKey;\n return hashFn(queryKey);\n}\nfunction hashKey(queryKey) {\n return JSON.stringify(\n queryKey,\n (_, val) => isPlainObject(val) ? Object.keys(val).sort().reduce((result, key) => {\n result[key] = val[key];\n return result;\n }, {}) : val\n );\n}\nfunction partialMatchKey(a, b) {\n if (a === b) {\n return true;\n }\n if (typeof a !== typeof b) {\n return false;\n }\n if (a && b && typeof a === \"object\" && typeof b === \"object\") {\n return Object.keys(b).every((key) => partialMatchKey(a[key], b[key]));\n }\n return false;\n}\nvar hasOwn = Object.prototype.hasOwnProperty;\nfunction replaceEqualDeep(a, b) {\n if (a === b) {\n return a;\n }\n const array = isPlainArray(a) && isPlainArray(b);\n if (!array && !(isPlainObject(a) && isPlainObject(b))) return b;\n const aItems = array ? a : Object.keys(a);\n const aSize = aItems.length;\n const bItems = array ? b : Object.keys(b);\n const bSize = bItems.length;\n const copy = array ? new Array(bSize) : {};\n let equalItems = 0;\n for (let i = 0; i < bSize; i++) {\n const key = array ? i : bItems[i];\n const aItem = a[key];\n const bItem = b[key];\n if (aItem === bItem) {\n copy[key] = aItem;\n if (array ? i < aSize : hasOwn.call(a, key)) equalItems++;\n continue;\n }\n if (aItem === null || bItem === null || typeof aItem !== \"object\" || typeof bItem !== \"object\") {\n copy[key] = bItem;\n continue;\n }\n const v = replaceEqualDeep(aItem, bItem);\n copy[key] = v;\n if (v === aItem) equalItems++;\n }\n return aSize === bSize && equalItems === aSize ? a : copy;\n}\nfunction shallowEqualObjects(a, b) {\n if (!b || Object.keys(a).length !== Object.keys(b).length) {\n return false;\n }\n for (const key in a) {\n if (a[key] !== b[key]) {\n return false;\n }\n }\n return true;\n}\nfunction isPlainArray(value) {\n return Array.isArray(value) && value.length === Object.keys(value).length;\n}\nfunction isPlainObject(o) {\n if (!hasObjectPrototype(o)) {\n return false;\n }\n const ctor = o.constructor;\n if (ctor === void 0) {\n return true;\n }\n const prot = ctor.prototype;\n if (!hasObjectPrototype(prot)) {\n return false;\n }\n if (!prot.hasOwnProperty(\"isPrototypeOf\")) {\n return false;\n }\n if (Object.getPrototypeOf(o) !== Object.prototype) {\n return false;\n }\n return true;\n}\nfunction hasObjectPrototype(o) {\n return Object.prototype.toString.call(o) === \"[object Object]\";\n}\nfunction sleep(timeout) {\n return new Promise((resolve) => {\n timeoutManager.setTimeout(resolve, timeout);\n });\n}\nfunction replaceData(prevData, data, options) {\n if (typeof options.structuralSharing === \"function\") {\n return options.structuralSharing(prevData, data);\n } else if (options.structuralSharing !== false) {\n if (process.env.NODE_ENV !== \"production\") {\n try {\n return replaceEqualDeep(prevData, data);\n } catch (error) {\n console.error(\n `Structural sharing requires data to be JSON serializable. To fix this, turn off structuralSharing or return JSON-serializable data from your queryFn. [${options.queryHash}]: ${error}`\n );\n throw error;\n }\n }\n return replaceEqualDeep(prevData, data);\n }\n return data;\n}\nfunction keepPreviousData(previousData) {\n return previousData;\n}\nfunction addToEnd(items, item, max = 0) {\n const newItems = [...items, item];\n return max && newItems.length > max ? newItems.slice(1) : newItems;\n}\nfunction addToStart(items, item, max = 0) {\n const newItems = [item, ...items];\n return max && newItems.length > max ? newItems.slice(0, -1) : newItems;\n}\nvar skipToken = Symbol();\nfunction ensureQueryFn(options, fetchOptions) {\n if (process.env.NODE_ENV !== \"production\") {\n if (options.queryFn === skipToken) {\n console.error(\n `Attempted to invoke queryFn when set to skipToken. This is likely a configuration error. Query hash: '${options.queryHash}'`\n );\n }\n }\n if (!options.queryFn && fetchOptions?.initialPromise) {\n return () => fetchOptions.initialPromise;\n }\n if (!options.queryFn || options.queryFn === skipToken) {\n return () => Promise.reject(new Error(`Missing queryFn: '${options.queryHash}'`));\n }\n return options.queryFn;\n}\nfunction shouldThrowError(throwOnError, params) {\n if (typeof throwOnError === \"function\") {\n return throwOnError(...params);\n }\n return !!throwOnError;\n}\nexport {\n addToEnd,\n addToStart,\n ensureQueryFn,\n functionalUpdate,\n hashKey,\n hashQueryKeyByOptions,\n isPlainArray,\n isPlainObject,\n isServer,\n isValidTimeout,\n keepPreviousData,\n matchMutation,\n matchQuery,\n noop,\n partialMatchKey,\n replaceData,\n replaceEqualDeep,\n resolveEnabled,\n resolveStaleTime,\n shallowEqualObjects,\n shouldThrowError,\n skipToken,\n sleep,\n timeUntilStale\n};\n//# sourceMappingURL=utils.js.map","// src/notifyManager.ts\nimport { systemSetTimeoutZero } from \"./timeoutManager.js\";\nvar defaultScheduler = systemSetTimeoutZero;\nfunction createNotifyManager() {\n let queue = [];\n let transactions = 0;\n let notifyFn = (callback) => {\n callback();\n };\n let batchNotifyFn = (callback) => {\n callback();\n };\n let scheduleFn = defaultScheduler;\n const schedule = (callback) => {\n if (transactions) {\n queue.push(callback);\n } else {\n scheduleFn(() => {\n notifyFn(callback);\n });\n }\n };\n const flush = () => {\n const originalQueue = queue;\n queue = [];\n if (originalQueue.length) {\n scheduleFn(() => {\n batchNotifyFn(() => {\n originalQueue.forEach((callback) => {\n notifyFn(callback);\n });\n });\n });\n }\n };\n return {\n batch: (callback) => {\n let result;\n transactions++;\n try {\n result = callback();\n } finally {\n transactions--;\n if (!transactions) {\n flush();\n }\n }\n return result;\n },\n /**\n * All calls to the wrapped function will be batched.\n */\n batchCalls: (callback) => {\n return (...args) => {\n schedule(() => {\n callback(...args);\n });\n };\n },\n schedule,\n /**\n * Use this method to set a custom notify function.\n * This can be used to for example wrap notifications with `React.act` while running tests.\n */\n setNotifyFunction: (fn) => {\n notifyFn = fn;\n },\n /**\n * Use this method to set a custom function to batch notifications together into a single tick.\n * By default React Query will use the batch function provided by ReactDOM or React Native.\n */\n setBatchNotifyFunction: (fn) => {\n batchNotifyFn = fn;\n },\n setScheduler: (fn) => {\n scheduleFn = fn;\n }\n };\n}\nvar notifyManager = createNotifyManager();\nexport {\n createNotifyManager,\n defaultScheduler,\n notifyManager\n};\n//# sourceMappingURL=notifyManager.js.map","// src/createPersister.ts\nimport {\n hashKey,\n matchQuery,\n notifyManager,\n partialMatchKey\n} from \"@tanstack/query-core\";\nvar PERSISTER_KEY_PREFIX = \"tanstack-query\";\nfunction experimental_createQueryPersister({\n storage,\n buster = \"\",\n maxAge = 1e3 * 60 * 60 * 24,\n serialize = JSON.stringify,\n deserialize = JSON.parse,\n prefix = PERSISTER_KEY_PREFIX,\n refetchOnRestore = true,\n filters\n}) {\n function isExpiredOrBusted(persistedQuery) {\n if (persistedQuery.state.dataUpdatedAt) {\n const queryAge = Date.now() - persistedQuery.state.dataUpdatedAt;\n const expired = queryAge > maxAge;\n const busted = persistedQuery.buster !== buster;\n if (expired || busted) {\n return true;\n }\n return false;\n }\n return true;\n }\n async function retrieveQuery(queryHash, afterRestoreMacroTask) {\n if (storage != null) {\n const storageKey = `${prefix}-${queryHash}`;\n try {\n const storedData = await storage.getItem(storageKey);\n if (storedData) {\n const persistedQuery = await deserialize(storedData);\n if (isExpiredOrBusted(persistedQuery)) {\n await storage.removeItem(storageKey);\n } else {\n if (afterRestoreMacroTask) {\n notifyManager.schedule(\n () => afterRestoreMacroTask(persistedQuery)\n );\n }\n return persistedQuery.state.data;\n }\n }\n } catch (err) {\n if (process.env.NODE_ENV === \"development\") {\n console.error(err);\n console.warn(\n \"Encountered an error attempting to restore query cache from persisted location.\"\n );\n }\n await storage.removeItem(storageKey);\n }\n }\n return;\n }\n async function persistQueryByKey(queryKey, queryClient) {\n if (storage != null) {\n const query = queryClient.getQueryCache().find({ queryKey });\n if (query) {\n await persistQuery(query);\n } else {\n if (process.env.NODE_ENV === \"development\") {\n console.warn(\n \"Could not find query to be persisted. QueryKey:\",\n JSON.stringify(queryKey)\n );\n }\n }\n }\n }\n async function persistQuery(query) {\n if (storage != null) {\n const storageKey = `${prefix}-${query.queryHash}`;\n storage.setItem(\n storageKey,\n await serialize({\n state: query.state,\n queryKey: query.queryKey,\n queryHash: query.queryHash,\n buster\n })\n );\n }\n }\n async function persisterFn(queryFn, ctx, query) {\n const matchesFilter = filters ? matchQuery(filters, query) : true;\n if (matchesFilter && query.state.data === void 0 && storage != null) {\n const restoredData = await retrieveQuery(\n query.queryHash,\n (persistedQuery) => {\n query.setState({\n dataUpdatedAt: persistedQuery.state.dataUpdatedAt,\n errorUpdatedAt: persistedQuery.state.errorUpdatedAt\n });\n if (refetchOnRestore === \"always\" || refetchOnRestore === true && query.isStale()) {\n query.fetch();\n }\n }\n );\n if (restoredData !== void 0) {\n return Promise.resolve(restoredData);\n }\n }\n const queryFnResult = await queryFn(ctx);\n if (matchesFilter && storage != null) {\n notifyManager.schedule(() => {\n persistQuery(query);\n });\n }\n return Promise.resolve(queryFnResult);\n }\n async function persisterGc() {\n if (storage?.entries) {\n const entries = await storage.entries();\n for (const [key, value] of entries) {\n if (key.startsWith(prefix)) {\n const persistedQuery = await deserialize(value);\n if (isExpiredOrBusted(persistedQuery)) {\n await storage.removeItem(key);\n }\n }\n }\n } else if (process.env.NODE_ENV === \"development\") {\n throw new Error(\n \"Provided storage does not implement `entries` method. Garbage collection is not possible without ability to iterate over storage items.\"\n );\n }\n }\n async function restoreQueries(queryClient, filters2 = {}) {\n const { exact, queryKey } = filters2;\n if (storage?.entries) {\n const entries = await storage.entries();\n for (const [key, value] of entries) {\n if (key.startsWith(prefix)) {\n const persistedQuery = await deserialize(value);\n if (isExpiredOrBusted(persistedQuery)) {\n await storage.removeItem(key);\n continue;\n }\n if (queryKey) {\n if (exact) {\n if (persistedQuery.queryHash !== hashKey(queryKey)) {\n continue;\n }\n } else if (!partialMatchKey(persistedQuery.queryKey, queryKey)) {\n continue;\n }\n }\n queryClient.setQueryData(\n persistedQuery.queryKey,\n persistedQuery.state.data,\n {\n updatedAt: persistedQuery.state.dataUpdatedAt\n }\n );\n }\n }\n } else if (process.env.NODE_ENV === \"development\") {\n throw new Error(\n \"Provided storage does not implement `entries` method. Restoration of all stored entries is not possible without ability to iterate over storage items.\"\n );\n }\n }\n return {\n persisterFn,\n persistQuery,\n persistQueryByKey,\n retrieveQuery,\n persisterGc,\n restoreQueries\n };\n}\nexport {\n PERSISTER_KEY_PREFIX,\n experimental_createQueryPersister\n};\n//# sourceMappingURL=createPersister.js.map","import {\n type AsyncStorage,\n experimental_createQueryPersister,\n type PersistedQuery,\n} from \"@tanstack/react-query-persist-client\";\n\nconst DB_NAME = \"fluid_tanstack_query_cache\";\nconst STORE_NAME = \"fluid_queries\";\nconst VERSION = 1;\n\nlet dbPromise: Promise<IDBDatabase> | null = null;\n\n// Exported for testing - allows resetting internal state between tests\nexport function _resetForTesting(): void {\n dbPromise = null;\n}\n\nexport async function deleteDatabase(reason: string = \"error\"): Promise<void> {\n return new Promise((resolve, reject) => {\n console.warn(`[IDB] Deleting database due to ${reason}`);\n const req = indexedDB.deleteDatabase(DB_NAME);\n req.onsuccess = () => {\n console.log(\"[IDB] Database deleted successfully\");\n dbPromise = null; // ensure next getDb() opens a fresh connection\n resolve();\n };\n req.onerror = () => {\n console.error(\"[IDB] Failed to delete database:\", req.error);\n reject(req.error ?? new Error(\"deleteDatabase failed\"));\n };\n req.onblocked = () => {\n console.warn(\"[IDB] Delete blocked: close all tabs using this database\");\n // Resolve anyway so callers (e.g. clearCompanyCache) don't hang\n // indefinitely. The delete will complete once blocking connections close.\n dbPromise = null; // best-effort: allow fresh open on next access\n resolve();\n };\n });\n}\n\nfunction openDatabase(): Promise<IDBDatabase> {\n return new Promise((resolve, reject) => {\n const req = indexedDB.open(DB_NAME, VERSION);\n\n req.onupgradeneeded = () => {\n const upgradeDb = req.result;\n if (!upgradeDb.objectStoreNames.contains(STORE_NAME)) {\n upgradeDb.createObjectStore(STORE_NAME);\n }\n };\n\n req.onsuccess = () => {\n const conn = req.result;\n conn.onversionchange = () => {\n console.trace(\"[IDB] version change – closing connection\");\n conn.close();\n // clear the cached promise so the next getDb call re-opens a connection\n dbPromise = null;\n };\n resolve(conn);\n };\n\n req.onblocked = () => {\n console.warn(\"[IDB] open blocked: another connection is holding the DB\");\n };\n\n req.onerror = () => {\n reject(\n req.error instanceof Error\n ? req.error\n : new Error(`IndexedDB open failed: ${String(req.error)}`),\n );\n };\n });\n}\n\nasync function getDbWithRecovery(): Promise<IDBDatabase> {\n try {\n return await openDatabase();\n } catch (err) {\n // First open failed - try to delete and recreate\n console.warn(\"[IDB] Initial open failed, attempting recovery:\", err);\n try {\n await deleteDatabase();\n console.log(\"[IDB] Retrying database open after deletion\");\n return await openDatabase();\n } catch (retryErr) {\n console.error(\"[IDB] Recovery failed:\", retryErr);\n throw retryErr;\n }\n }\n}\n\nfunction getDb(): Promise<IDBDatabase> {\n if (dbPromise) return dbPromise;\n\n // Create the promise and cache it immediately to prevent race conditions\n // All concurrent calls will share this same promise\n dbPromise = getDbWithRecovery().catch((err) => {\n // On failure, clear the cache so next call can retry fresh\n dbPromise = null;\n throw err;\n });\n\n return dbPromise;\n}\n\n// Exported for testing - allows direct testing of storage operations\nexport const storage: AsyncStorage<PersistedQuery> = {\n async getItem(key: string) {\n try {\n const db = await getDb();\n return new Promise<PersistedQuery | undefined>((res) => {\n try {\n const r = db\n .transaction(STORE_NAME, \"readonly\")\n .objectStore(STORE_NAME)\n .get(key);\n r.onsuccess = () => res(r.result as PersistedQuery);\n r.onerror = () => {\n console.trace(\"[IDB] getItem error:\", r.error);\n res(undefined);\n };\n } catch (txErr) {\n // Transaction creation failed (e.g., stale connection)\n console.trace(\"[IDB] getItem transaction error:\", txErr);\n res(undefined);\n }\n });\n } catch (err) {\n // getDb() failed - try to recover with a fresh connection\n console.trace(\"[IDB] getItem getDb error:\", err);\n try {\n const db = await getDb();\n // Use fresh connection directly - getDb() already handles recovery\n return new Promise<PersistedQuery | undefined>((res) => {\n try {\n const r = db\n .transaction(STORE_NAME, \"readonly\")\n .objectStore(STORE_NAME)\n .get(key);\n r.onsuccess = () => res(r.result as PersistedQuery);\n r.onerror = () => {\n console.trace(\"[IDB] getItem retry error:\", r.error);\n res(undefined);\n };\n } catch (txErr) {\n console.trace(\"[IDB] getItem retry transaction error:\", txErr);\n res(undefined);\n }\n });\n } catch (recoveryErr) {\n console.trace(\"[IDB] getItem recovery failed:\", recoveryErr);\n }\n return undefined;\n }\n },\n\n async setItem(key: string, value: PersistedQuery) {\n // JSON round-trip strips non-cloneable properties (functions, class instances)\n // before IndexedDB's structured clone algorithm runs. Without this, IDB\n // throws DataCloneError. Theme Color objects are now kept out of the cache\n // via the select pattern, but other queries may still have non-cloneable data.\n const cloneableValue = JSON.parse(JSON.stringify(value));\n try {\n const db = await getDb();\n if (!db) return;\n await new Promise<void>((resolve) => {\n try {\n const req = db\n .transaction(STORE_NAME, \"readwrite\")\n .objectStore(STORE_NAME)\n .put(cloneableValue, key);\n req.onsuccess = () => resolve();\n req.onerror = () => {\n console.trace(\"[IDB] setItem error:\", req.error);\n resolve();\n };\n } catch (txErr) {\n console.trace(\"[IDB] setItem transaction error:\", txErr);\n resolve();\n }\n });\n } catch (err) {\n // getDb() failed - try to recover with a fresh connection\n console.trace(\"[IDB] setItem getDb error:\", err);\n try {\n const db = await getDb();\n // Use fresh connection directly - getDb() already handles recovery\n await new Promise<void>((resolve) => {\n try {\n const req = db\n .transaction(STORE_NAME, \"readwrite\")\n .objectStore(STORE_NAME)\n .put(cloneableValue, key);\n req.onsuccess = () => resolve();\n req.onerror = () => {\n console.trace(\"[IDB] setItem retry error:\", req.error);\n resolve();\n };\n } catch (txErr) {\n console.trace(\"[IDB] setItem retry transaction error:\", txErr);\n resolve();\n }\n });\n } catch (recoveryErr) {\n console.trace(\"[IDB] setItem recovery failed:\", recoveryErr);\n }\n }\n },\n\n async removeItem(key: string) {\n try {\n const db = await getDb();\n if (!db) return;\n await new Promise<void>((resolve) => {\n try {\n const req = db\n .transaction(STORE_NAME, \"readwrite\")\n .objectStore(STORE_NAME)\n .delete(key);\n req.onsuccess = () => resolve();\n req.onerror = () => {\n console.trace(\"[IDB] removeItem error:\", req.error);\n resolve();\n };\n } catch (txErr) {\n console.trace(\"[IDB] removeItem transaction error:\", txErr);\n resolve();\n }\n });\n } catch (err) {\n // getDb() failed - try to recover with a fresh connection\n console.trace(\"[IDB] removeItem getDb error:\", err);\n try {\n const db = await getDb();\n // Use fresh connection directly - getDb() already handles recovery\n await new Promise<void>((resolve) => {\n try {\n const req = db\n .transaction(STORE_NAME, \"readwrite\")\n .objectStore(STORE_NAME)\n .delete(key);\n req.onsuccess = () => resolve();\n req.onerror = () => {\n console.trace(\"[IDB] removeItem retry error:\", req.error);\n resolve();\n };\n } catch (txErr) {\n console.trace(\"[IDB] removeItem retry transaction error:\", txErr);\n resolve();\n }\n });\n } catch (recoveryErr) {\n console.trace(\"[IDB] removeItem recovery failed:\", recoveryErr);\n }\n }\n },\n};\n\n// Factory function to create persister - only call this client-side\nexport function createPersister(): ReturnType<\n typeof experimental_createQueryPersister\n> {\n return experimental_createQueryPersister<PersistedQuery>({\n storage,\n serialize: (persistedQuery) => persistedQuery,\n deserialize: (cached) => cached,\n });\n}\n","import { createContext, useContext } from \"react\";\nimport type { FluidOsReadApi, FluidOsBuilderApi } from \"./fluidos-api\";\n\nconst FluidOsReadApiContext = createContext<FluidOsReadApi | null>(null);\nconst FluidOsBuilderApiContext = createContext<FluidOsBuilderApi | null>(null);\n\n/** Provider for read-only manifest fetching (used by SDK). */\nexport const FluidOsApiProvider = FluidOsReadApiContext.Provider;\n\n/** Provider for full CRUD access (used by builder / fluid-admin). */\nexport const FluidOsBuilderApiProvider = FluidOsBuilderApiContext.Provider;\n\n/** Hook for read-only manifest fetching (used by SDK). */\nexport function useFluidOsApi(): FluidOsReadApi {\n const builderApi = useContext(FluidOsBuilderApiContext);\n const readApi = useContext(FluidOsReadApiContext);\n if (builderApi) return builderApi;\n if (!readApi) {\n throw new Error(\n \"useFluidOsApi must be used within a FluidOsApiProvider or FluidOsBuilderApiProvider\",\n );\n }\n return readApi;\n}\n\n/** Hook for full CRUD access (used by builder). */\nexport function useFluidOsBuilderApi(): FluidOsBuilderApi {\n const api = useContext(FluidOsBuilderApiContext);\n if (!api) {\n throw new Error(\n \"useFluidOsBuilderApi must be used within a FluidOsBuilderApiProvider\",\n );\n }\n return api;\n}\n\n/** Optional variant that returns null when no provider is present. */\nexport function useFluidOsApiOptional(): FluidOsReadApi | null {\n const builderApi = useContext(FluidOsBuilderApiContext);\n const readApi = useContext(FluidOsReadApiContext);\n return builderApi ?? readApi;\n}\n","import { createContext, useContext, type Provider } from \"react\";\nimport type { AppDefinitionApi } from \"./app-definition-api\";\n\nconst AppDefinitionApiContext = createContext<AppDefinitionApi | null>(null);\n\nexport const AppDefinitionApiProvider: Provider<AppDefinitionApi | null> =\n AppDefinitionApiContext.Provider;\n\nexport function useAppDefinitionApi(): AppDefinitionApi {\n const api = useContext(AppDefinitionApiContext);\n if (!api) {\n throw new Error(\n \"useAppDefinitionApi must be used within an AppDefinitionApiProvider\",\n );\n }\n return api;\n}\n","import { createContext, useContext } from \"react\";\nimport type { PayApi } from \"./pay-api\";\n\nconst PayApiContext = createContext<PayApi | null>(null);\n\nexport const PayApiProvider = PayApiContext.Provider;\n\nexport function usePayApi(): PayApi {\n const api = useContext(PayApiContext);\n if (!api) {\n throw new Error(\"usePayApi must be used within a PayApiProvider\");\n }\n return api;\n}\n","import { createContext, useContext } from \"react\";\nimport type { MySiteApi } from \"./mysite-api\";\n\nconst MySiteApiContext = createContext<MySiteApi | null>(null);\n\nexport const MySiteApiProvider = MySiteApiContext.Provider;\n\nexport function useMySiteApi(): MySiteApi {\n const api = useContext(MySiteApiContext);\n if (!api) {\n throw new Error(\"useMySiteApi must be used within a MySiteApiProvider\");\n }\n return api;\n}\n","import { createContext, useContext, type Provider } from \"react\";\nimport type { LanguagesApi } from \"./languages-api\";\n\nconst LanguagesApiContext = createContext<LanguagesApi | null>(null);\n\nexport const LanguagesApiProvider: Provider<LanguagesApi | null> =\n LanguagesApiContext.Provider;\n\nexport function useLanguagesApi(): LanguagesApi {\n const api = useContext(LanguagesApiContext);\n if (!api) {\n throw new Error(\n \"useLanguagesApi must be used within a LanguagesApiProvider\",\n );\n }\n return api;\n}\n","/**\n * Minimal, framework-agnostic fetch client for Fluid APIs\n * Compatible with fluid-admin patterns but usable standalone\n */\n\nexport interface FetchClientConfig {\n /**\n * Base URL for all requests (e.g., \"https://api.fluid.app/api\")\n */\n baseUrl: string;\n\n /**\n * Optional function to get auth token\n * Return null/undefined if no token available\n */\n getAuthToken?: () => string | null | Promise<string | null>;\n\n /**\n * Optional callback when 401 auth error occurs\n */\n onAuthError?: () => void;\n\n /**\n * Default headers to include in all requests\n * Example: { \"x-fluid-client\": \"admin\" }\n */\n defaultHeaders?: Record<string, string>;\n\n /**\n * Credentials mode for fetch requests.\n * Set to `\"include\"` for cookie-based (same-origin BFF) authentication.\n * @default undefined (browser default: \"same-origin\")\n */\n credentials?: RequestCredentials;\n}\n\nexport interface RequestOptions {\n method?: \"GET\" | \"POST\" | \"PUT\" | \"DELETE\" | \"PATCH\";\n headers?: Record<string, string>;\n params?: Record<string, unknown>;\n body?: unknown;\n signal?: AbortSignal;\n}\n\n/**\n * API Error class compatible with fluid-admin's ApiError\n */\nexport class ApiError extends Error {\n public readonly status: number;\n public readonly data: unknown;\n\n constructor(message: string, status: number, data?: unknown) {\n super(message);\n this.name = \"ApiError\";\n this.status = status;\n this.data = data;\n\n if (\"captureStackTrace\" in Error) {\n (\n Error as {\n captureStackTrace: (\n target: Error,\n constructor: NewableFunction,\n ) => void;\n }\n ).captureStackTrace(this, ApiError);\n }\n }\n\n toJSON(): { name: string; message: string; status: number; data: unknown } {\n return {\n name: this.name,\n message: this.message,\n status: this.status,\n data: this.data,\n };\n }\n}\n\n/**\n * Type guard for ApiError\n */\nexport function isApiError(error: unknown): error is ApiError {\n return error instanceof ApiError;\n}\n\nexport interface FetchClientInstance {\n request: <TResponse = unknown>(\n endpoint: string,\n options?: RequestOptions,\n ) => Promise<TResponse>;\n requestWithFormData: <TResponse = unknown>(\n endpoint: string,\n formData: FormData,\n options?: Omit<RequestOptions, \"body\" | \"params\"> & {\n method?: \"POST\" | \"PUT\" | \"PATCH\";\n },\n ) => Promise<TResponse>;\n get: <TResponse = unknown>(\n endpoint: string,\n params?: Record<string, unknown>,\n options?: Omit<RequestOptions, \"method\" | \"params\">,\n ) => Promise<TResponse>;\n post: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ) => Promise<TResponse>;\n put: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ) => Promise<TResponse>;\n patch: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ) => Promise<TResponse>;\n delete: <TResponse = unknown>(\n endpoint: string,\n options?: Omit<RequestOptions, \"method\">,\n ) => Promise<TResponse>;\n}\n\n/**\n * Creates a configured fetch client instance\n */\nexport function createFetchClient(\n config: FetchClientConfig,\n): FetchClientInstance {\n const {\n baseUrl,\n getAuthToken,\n onAuthError,\n defaultHeaders = {},\n credentials,\n } = config;\n\n /**\n * Build headers for a request\n */\n async function buildHeaders(\n customHeaders?: Record<string, string>,\n ): Promise<Record<string, string>> {\n const headers: Record<string, string> = {\n Accept: \"application/json\",\n \"Content-Type\": \"application/json\",\n ...defaultHeaders,\n ...customHeaders,\n };\n\n // Add auth token if available\n if (getAuthToken) {\n const token = await getAuthToken();\n if (token) {\n headers.Authorization = `Bearer ${token}`;\n }\n }\n\n return headers;\n }\n\n /**\n * Join baseUrl + endpoint via string concatenation (matches fetchApi).\n * Using `new URL(endpoint, baseUrl)` would strip any path prefix from\n * baseUrl (e.g. \"/api\") when the endpoint starts with \"/\".\n */\n function joinUrl(endpoint: string): string {\n return `${baseUrl}${endpoint}`;\n }\n\n /**\n * Build URL with query parameters for GET requests\n * Compatible with fluid-admin's query param handling\n */\n function buildUrl(\n endpoint: string,\n params?: Record<string, unknown>,\n ): string {\n const fullUrl = joinUrl(endpoint);\n\n if (!params || Object.keys(params).length === 0) {\n return fullUrl;\n }\n\n const queryString = new URLSearchParams();\n\n Object.entries(params).forEach(([key, value]) => {\n if (value === undefined || value === null) {\n return; // Skip undefined/null values\n }\n\n if (Array.isArray(value)) {\n // Handle arrays like Rails expects: key[]\n value.forEach((item) => queryString.append(`${key}[]`, String(item)));\n } else if (typeof value === \"object\") {\n // Handle nested objects: key[subkey]\n Object.entries(value).forEach(([subKey, subValue]) => {\n if (subValue === undefined || subValue === null) {\n return;\n }\n\n if (Array.isArray(subValue)) {\n subValue.forEach((item) =>\n queryString.append(`${key}[${subKey}][]`, String(item)),\n );\n } else {\n queryString.append(`${key}[${subKey}]`, String(subValue));\n }\n });\n } else {\n queryString.append(key, String(value));\n }\n });\n\n const qs = queryString.toString();\n return qs ? `${fullUrl}?${qs}` : fullUrl;\n }\n\n /**\n * Shared response handler for both JSON and FormData requests.\n * Handles auth errors, non-OK responses, 204 No Content, and JSON parsing.\n */\n async function handleResponse<TResponse>(\n response: Response,\n method: string,\n _url: string,\n ): Promise<TResponse> {\n if (response.status === 401 && onAuthError) {\n onAuthError();\n }\n\n if (!response.ok) {\n // Read body as text first to avoid SyntaxError from response.json()\n // when server returns non-JSON bodies with application/json content-type.\n const errorText = await response.text().catch(() => \"\");\n const contentType = response.headers.get(\"content-type\");\n\n if (contentType?.includes(\"application/json\")) {\n let data: Record<string, unknown>;\n try {\n data = JSON.parse(errorText);\n } catch {\n throw new ApiError(\n errorText.slice(0, 200) ||\n `${method} request failed with status ${response.status}`,\n response.status,\n null,\n );\n }\n const msg = (data.message || data.error_message) as string | undefined;\n throw new ApiError(\n msg || `${method} request failed`,\n response.status,\n data.errors || data,\n );\n } else {\n throw new ApiError(\n `${method} request failed with status ${response.status}`,\n response.status,\n null,\n );\n }\n }\n\n if (\n response.status === 204 ||\n response.headers.get(\"content-length\") === \"0\"\n ) {\n return null as TResponse;\n }\n\n const contentType = response.headers.get(\"content-type\");\n\n if (contentType?.includes(\"application/json\")) {\n try {\n const data = await response.json();\n return data as TResponse;\n } catch {\n try {\n // API declared JSON content-type but body isn't valid JSON\n const text = await response.text();\n return text as TResponse;\n } catch {\n return null as TResponse;\n }\n }\n }\n\n // Non-JSON response (text/plain, text/html, etc.)\n return null as TResponse;\n }\n\n /**\n * Main request function\n */\n async function request<TResponse = unknown>(\n endpoint: string,\n options: RequestOptions = {},\n ): Promise<TResponse> {\n const {\n method = \"GET\",\n headers: customHeaders,\n params,\n body,\n signal,\n } = options;\n\n const url = params ? buildUrl(endpoint, params) : joinUrl(endpoint);\n\n const headers = await buildHeaders(customHeaders);\n\n let response: Response;\n\n try {\n const fetchOptions: RequestInit = { method, headers };\n if (credentials) fetchOptions.credentials = credentials;\n const serializedBody =\n body && method !== \"GET\" ? JSON.stringify(body) : null;\n if (serializedBody) fetchOptions.body = serializedBody;\n if (signal) fetchOptions.signal = signal;\n response = await fetch(url, fetchOptions);\n } catch (networkError) {\n throw new ApiError(\n `Network error: ${networkError instanceof Error ? networkError.message : \"Unknown network error\"}`,\n 0,\n null,\n );\n }\n\n return handleResponse<TResponse>(response, method, url);\n }\n\n /**\n * Request with FormData (for file uploads)\n */\n async function requestWithFormData<TResponse = unknown>(\n endpoint: string,\n formData: FormData,\n options: Omit<RequestOptions, \"body\" | \"params\"> & {\n method?: \"POST\" | \"PUT\" | \"PATCH\";\n } = {},\n ): Promise<TResponse> {\n const { method = \"POST\", headers: customHeaders, signal } = options;\n\n const url = joinUrl(endpoint);\n const headers = await buildHeaders(customHeaders);\n\n // Remove Content-Type to let browser set it with boundary\n delete headers[\"Content-Type\"];\n\n let response: Response;\n\n try {\n const fetchOptions: RequestInit = { method, headers, body: formData };\n if (credentials) fetchOptions.credentials = credentials;\n if (signal) fetchOptions.signal = signal;\n response = await fetch(url, fetchOptions);\n } catch (networkError) {\n throw new ApiError(\n `Network error: ${networkError instanceof Error ? networkError.message : \"Unknown network error\"}`,\n 0,\n null,\n );\n }\n\n return handleResponse<TResponse>(response, method, url);\n }\n\n // Return client with convenience methods\n return {\n request: request,\n requestWithFormData: requestWithFormData,\n\n // Convenience methods for common HTTP verbs\n get: <TResponse = unknown>(\n endpoint: string,\n params?: Record<string, unknown>,\n options?: Omit<RequestOptions, \"method\" | \"params\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: \"GET\" as const,\n ...(params && { params }),\n }),\n\n post: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: \"POST\",\n body,\n }),\n\n put: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: \"PUT\",\n body,\n }),\n\n patch: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: \"PATCH\",\n body,\n }),\n\n delete: <TResponse = unknown>(\n endpoint: string,\n options?: Omit<RequestOptions, \"method\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: \"DELETE\",\n }),\n };\n}\n\nexport type FetchClient = FetchClientInstance;\n","/**\n * Generated API client functions for fluid_os\n *\n * DO NOT EDIT THIS FILE DIRECTLY\n * This file is auto-generated. To update:\n * 1. Update the OpenAPI spec file\n * 2. Run: pnpm generate\n */\n\nimport type { FetchClient } from \"../lib/fetch-client\";\nimport type { operations } from \"../generated/fluid_os\";\n\n// ============================================================================\n// Fluid OS - Definitions\n// ============================================================================\n\n/**\n * List Fluid OS definitions\n * Retrieve a list of Fluid OS definitions for the current company\n *\n * @param client - Fetch client instance\n * @param params? - params?\n */\nexport async function listFluidOSDefinitions(\n client: FetchClient,\n params?: operations[\"listFluidOSDefinitions\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"listFluidOSDefinitions\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/company/fluid_os/definitions`, params);\n}\n\n/**\n * Create a Fluid OS definition\n * Create a new Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param body - body\n */\nexport async function createFluidOSDefinition(\n client: FetchClient,\n body: operations[\"createFluidOSDefinition\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"createFluidOSDefinition\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.post(`/api/company/fluid_os/definitions`, body);\n}\n\n/**\n * Get a Fluid OS definition\n * Retrieve a specific Fluid OS definition with all associations\n *\n * @param client - Fetch client instance\n * @param id - id\n */\nexport async function getFluidOSDefinition(\n client: FetchClient,\n id: string | number,\n): Promise<\n operations[\"getFluidOSDefinition\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/company/fluid_os/definitions/${id}`);\n}\n\n/**\n * Update a Fluid OS definition\n * Update an existing Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param id - id\n * @param body - body\n */\nexport async function updateFluidOSDefinition(\n client: FetchClient,\n id: string | number,\n body: operations[\"updateFluidOSDefinition\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"updateFluidOSDefinition\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.put(`/api/company/fluid_os/definitions/${id}`, body);\n}\n\n/**\n * Delete a Fluid OS definition\n * Delete a Definition\n *\n * @param client - Fetch client instance\n * @param id - id\n */\nexport async function deleteFluidOSDefinition(\n client: FetchClient,\n id: string | number,\n): Promise<\n operations[\"deleteFluidOSDefinition\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.delete(`/api/company/fluid_os/definitions/${id}`);\n}\n\n// ============================================================================\n// Fluid OS - Navigation Items\n// ============================================================================\n\n/**\n * List navigation items for a navigation\n * Retrieve a list of navigation items for a specific navigation\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param navigation_id - navigation_id\n */\nexport async function listFluidOSNavigationItems(\n client: FetchClient,\n definition_id: string | number,\n navigation_id: string | number,\n): Promise<\n operations[\"listFluidOSNavigationItems\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/navigations/${navigation_id}/navigation_items`,\n );\n}\n\n/**\n * Create a navigation item\n * Create a new navigation item for a navigation\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param navigation_id - navigation_id\n * @param body - body\n */\nexport async function createFluidOSNavigationItem(\n client: FetchClient,\n definition_id: string | number,\n navigation_id: string | number,\n body: operations[\"createFluidOSNavigationItem\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"createFluidOSNavigationItem\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.post(\n `/api/company/fluid_os/definitions/${definition_id}/navigations/${navigation_id}/navigation_items`,\n body,\n );\n}\n\n/**\n * Get a specific navigation item\n * Retrieve a specific navigation item\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param navigation_id - navigation_id\n * @param id - id\n */\nexport async function getFluidOSNavigationItem(\n client: FetchClient,\n definition_id: string | number,\n navigation_id: string | number,\n id: string | number,\n): Promise<\n operations[\"getFluidOSNavigationItem\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/navigations/${navigation_id}/navigation_items/${id}`,\n );\n}\n\n/**\n * Update a navigation item\n * Update an existing navigation item\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param navigation_id - navigation_id\n * @param id - id\n * @param body - body\n */\nexport async function updateFluidOSNavigationItem(\n client: FetchClient,\n definition_id: string | number,\n navigation_id: string | number,\n id: string | number,\n body: operations[\"updateFluidOSNavigationItem\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"updateFluidOSNavigationItem\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.put(\n `/api/company/fluid_os/definitions/${definition_id}/navigations/${navigation_id}/navigation_items/${id}`,\n body,\n );\n}\n\n/**\n * Delete a navigation item\n * Delete a navigation item\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param navigation_id - navigation_id\n * @param id - id\n */\nexport async function deleteFluidOSNavigationItem(\n client: FetchClient,\n definition_id: string | number,\n navigation_id: string | number,\n id: string | number,\n): Promise<\n operations[\"deleteFluidOSNavigationItem\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.delete(\n `/api/company/fluid_os/definitions/${definition_id}/navigations/${navigation_id}/navigation_items/${id}`,\n );\n}\n\n// ============================================================================\n// Fluid OS - Navigations\n// ============================================================================\n\n/**\n * List navigations for a Fluid OS definition\n * Retrieve a list of navigations for a specific Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param params? - params?\n */\nexport async function listFluidOSNavigations(\n client: FetchClient,\n definition_id: string | number,\n params?: operations[\"listFluidOSNavigations\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"listFluidOSNavigations\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/navigations`,\n params,\n );\n}\n\n/**\n * Create a navigation for a Fluid OS definition\n * Create a new navigation for a Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param body - body\n */\nexport async function createFluidOSNavigation(\n client: FetchClient,\n definition_id: string | number,\n body: operations[\"createFluidOSNavigation\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"createFluidOSNavigation\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.post(\n `/api/company/fluid_os/definitions/${definition_id}/navigations`,\n body,\n );\n}\n\n/**\n * Get a specific navigation\n * Retrieve a specific navigation\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function getFluidOSNavigation(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"getFluidOSNavigation\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/navigations/${id}`,\n );\n}\n\n/**\n * Update a navigation\n * Update an existing navigation\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n * @param body - body\n */\nexport async function updateFluidOSNavigation(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n body: operations[\"updateFluidOSNavigation\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"updateFluidOSNavigation\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.put(\n `/api/company/fluid_os/definitions/${definition_id}/navigations/${id}`,\n body,\n );\n}\n\n/**\n * Delete a navigation\n * Delete a navigation\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function deleteFluidOSNavigation(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"deleteFluidOSNavigation\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.delete(\n `/api/company/fluid_os/definitions/${definition_id}/navigations/${id}`,\n );\n}\n\n// ============================================================================\n// Fluid OS - Profiles\n// ============================================================================\n\n/**\n * List profiles for a Fluid OS definition\n * Retrieve a list of profiles for a specific Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param params? - params?\n */\nexport async function listFluidOSProfiles(\n client: FetchClient,\n definition_id: string | number,\n params?: operations[\"listFluidOSProfiles\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"listFluidOSProfiles\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/profiles`,\n params,\n );\n}\n\n/**\n * Create a profile for a Fluid OS definition\n * Create a new profile for a Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param body - body\n */\nexport async function createFluidOSProfile(\n client: FetchClient,\n definition_id: string | number,\n body: operations[\"createFluidOSProfile\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"createFluidOSProfile\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.post(\n `/api/company/fluid_os/definitions/${definition_id}/profiles`,\n body,\n );\n}\n\n/**\n * Get the default profile\n * Retrieve the default profile for a Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n */\nexport async function getDefaultFluidOSProfile(\n client: FetchClient,\n definition_id: string | number,\n): Promise<\n operations[\"getDefaultFluidOSProfile\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/profiles/default`,\n );\n}\n\n/**\n * Get a specific profile\n * Retrieve a specific profile\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function getFluidOSProfile(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"getFluidOSProfile\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/profiles/${id}`,\n );\n}\n\n/**\n * Update a profile\n * Update an existing profile\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n * @param body - body\n */\nexport async function updateFluidOSProfile(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n body: operations[\"updateFluidOSProfile\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"updateFluidOSProfile\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.put(\n `/api/company/fluid_os/definitions/${definition_id}/profiles/${id}`,\n body,\n );\n}\n\n/**\n * Delete a profile\n * Delete a profile\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function deleteFluidOSProfile(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"deleteFluidOSProfile\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.delete(\n `/api/company/fluid_os/definitions/${definition_id}/profiles/${id}`,\n );\n}\n\n// ============================================================================\n// Fluid OS - Public\n// ============================================================================\n\n/**\n * Get active Fluid OS definition\n * Retrieve the active Fluid OS definition manifest for a specific platform\n *\n * @param client - Fetch client instance\n * @param params - params\n */\nexport async function getFluidOSManifest(\n client: FetchClient,\n params: operations[\"getFluidOSManifest\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"getFluidOSManifest\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/fluid_os/definitions/active`, params);\n}\n\n// ============================================================================\n// Fluid OS - Screens\n// ============================================================================\n\n/**\n * List screens for a Fluid OS definition\n * Retrieve a list of screens for a specific Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param params? - params?\n */\nexport async function listFluidOSScreens(\n client: FetchClient,\n definition_id: string | number,\n params?: operations[\"listFluidOSScreens\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"listFluidOSScreens\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/screens`,\n params,\n );\n}\n\n/**\n * Create a screen for a Fluid OS definition\n * Create a new screen for a Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param body - body\n */\nexport async function createFluidOSScreen(\n client: FetchClient,\n definition_id: string | number,\n body: operations[\"createFluidOSScreen\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"createFluidOSScreen\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.post(\n `/api/company/fluid_os/definitions/${definition_id}/screens`,\n body,\n );\n}\n\n/**\n * Get a specific screen\n * Retrieve a specific screen\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function getFluidOSScreen(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"getFluidOSScreen\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/screens/${id}`,\n );\n}\n\n/**\n * Update a screen\n * Update an existing screen\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n * @param body - body\n */\nexport async function updateFluidOSScreen(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n body: operations[\"updateFluidOSScreen\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"updateFluidOSScreen\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.put(\n `/api/company/fluid_os/definitions/${definition_id}/screens/${id}`,\n body,\n );\n}\n\n/**\n * Delete a screen\n * Delete a screen\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function deleteFluidOSScreen(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"deleteFluidOSScreen\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.delete(\n `/api/company/fluid_os/definitions/${definition_id}/screens/${id}`,\n );\n}\n\n// ============================================================================\n// Fluid OS - Themes\n// ============================================================================\n\n/**\n * List themes for a Fluid OS definition\n * Retrieve a list of themes for a specific Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param params? - params?\n */\nexport async function listFluidOSThemes(\n client: FetchClient,\n definition_id: string | number,\n params?: operations[\"listFluidOSThemes\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"listFluidOSThemes\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/themes`,\n params,\n );\n}\n\n/**\n * Create a theme for a Fluid OS definition\n * Create a new theme for a Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param body - body\n */\nexport async function createFluidOSTheme(\n client: FetchClient,\n definition_id: string | number,\n body: operations[\"createFluidOSTheme\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"createFluidOSTheme\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.post(\n `/api/company/fluid_os/definitions/${definition_id}/themes`,\n body,\n );\n}\n\n/**\n * Get a specific theme\n * Retrieve a specific theme\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function getFluidOSTheme(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"getFluidOSTheme\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/themes/${id}`,\n );\n}\n\n/**\n * Update a theme\n * Update an existing theme\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n * @param body - body\n */\nexport async function updateFluidOSTheme(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n body: operations[\"updateFluidOSTheme\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"updateFluidOSTheme\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.put(\n `/api/company/fluid_os/definitions/${definition_id}/themes/${id}`,\n body,\n );\n}\n\n/**\n * Delete a theme\n * Delete a theme\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function deleteFluidOSTheme(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"deleteFluidOSTheme\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.delete(\n `/api/company/fluid_os/definitions/${definition_id}/themes/${id}`,\n );\n}\n\n// ============================================================================\n// Fluid OS - Versions\n// ============================================================================\n\n/**\n * List versions for a Fluid OS definition\n * Retrieve a list of published versions for a specific Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param params? - params?\n */\nexport async function listFluidOSVersions(\n client: FetchClient,\n definition_id: string | number,\n params?: operations[\"listFluidOSVersions\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"listFluidOSVersions\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/versions`,\n params,\n );\n}\n\n/**\n * Publish a new version of a Fluid OS definition\n * Publish a new version of the Fluid OS definition. This creates a snapshot of the current definition state including all screens, profiles, themes, and navigations. No request body is required - the manifest is built automatically from the current definition state.\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n */\nexport async function createFluidOSVersion(\n client: FetchClient,\n definition_id: string | number,\n): Promise<\n operations[\"createFluidOSVersion\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.post(\n `/api/company/fluid_os/definitions/${definition_id}/versions`,\n );\n}\n\n/**\n * Get a specific version\n * Retrieve a specific published version including its manifest\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function getFluidOSVersion(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"getFluidOSVersion\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/versions/${id}`,\n );\n}\n\n/**\n * Update a version\n * Update a version. Currently only supports activating/deactivating a version.\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n * @param body - body\n */\nexport async function updateFluidOSVersion(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n body: operations[\"updateFluidOSVersion\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"updateFluidOSVersion\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.put(\n `/api/company/fluid_os/definitions/${definition_id}/versions/${id}`,\n body,\n );\n}\n","// ============================================================================\n// HTTP Method Constants\n// ============================================================================\n\n/**\n * HTTP methods supported by the API client.\n * Use `as const` for literal type inference and type safety.\n */\nexport const HTTP_METHODS = {\n GET: \"GET\",\n POST: \"POST\",\n PUT: \"PUT\",\n PATCH: \"PATCH\",\n DELETE: \"DELETE\",\n} as const;\n\n/**\n * Union type of all supported HTTP methods.\n * Derived from HTTP_METHODS constant to avoid duplication.\n */\nexport type HttpMethod = (typeof HTTP_METHODS)[keyof typeof HTTP_METHODS];\n\n// ============================================================================\n// Configuration Types\n// ============================================================================\n\n/**\n * Configuration for the Fluid SDK client.\n * Use Readonly<FluidSDKConfig> when the config should not be modified after creation.\n */\nexport interface FluidSDKConfig {\n /**\n * Base URL for the Fluid API domain (e.g., \"https://api.fluid.app\").\n * Endpoints include their full path from root (e.g., \"/api/reps/me\").\n */\n readonly baseUrl: string;\n\n /**\n * Function to retrieve the authentication token\n * Return null/undefined if no token is available\n */\n readonly getAuthToken?: () => string | null | Promise<string | null>;\n\n /**\n * Callback invoked when a 401 authentication error occurs\n * Use this to trigger re-authentication flows\n */\n readonly onAuthError?: () => void;\n\n /**\n * Default headers to include in all requests\n * Example: { \"x-fluid-client\": \"portal\" }\n */\n readonly defaultHeaders?: Readonly<Record<string, string>>;\n\n /**\n * Filestack API key for messaging file uploads.\n * If not provided, file attachments will be disabled in the messaging composer.\n */\n readonly filestackApiKey?: string;\n\n /**\n * Override WebSocket URL for real-time messaging.\n * Default: derived from baseUrl by replacing trailing /api with /cable\n */\n readonly websocketUrl?: string;\n\n /**\n * ISO country code for the store/merchant (e.g., \"US\", \"CA\", \"GB\").\n * Used to fetch country-specific payment methods.\n * @default \"US\"\n */\n readonly countryIso?: string;\n}\n\n/**\n * Options for individual API requests.\n * Uses HttpMethod type for method to ensure type safety.\n */\nexport interface RequestOptions {\n readonly method?: HttpMethod;\n readonly body?: unknown;\n readonly params?: Readonly<Record<string, unknown>>;\n readonly headers?: Readonly<Record<string, string>>;\n readonly signal?: AbortSignal;\n}\n\n// ============================================================================\n// Pagination Types\n// ============================================================================\n\n/**\n * Cursor-based pagination parameters for list endpoints.\n * Sent as query params `page[cursor]` and `page[limit]`.\n */\nexport interface PaginationParams {\n readonly page?: {\n readonly cursor?: string;\n readonly limit?: number;\n };\n}\n\n// ============================================================================\n// Sort Order - Derive from constant for single source of truth\n// ============================================================================\n\n/**\n * Sort order constant - single source of truth for sort direction values.\n * Use SORT_ORDERS.asc instead of \"asc\" for type-safe comparisons.\n */\nexport const SORT_ORDERS = {\n asc: \"asc\",\n desc: \"desc\",\n} as const;\n\n/**\n * Union type of sort order values, derived from SORT_ORDERS constant.\n * @see deriving-typeof-for-object-keys pattern\n */\nexport type SortOrder = (typeof SORT_ORDERS)[keyof typeof SORT_ORDERS];\n\n/**\n * Common filter parameters for list endpoints\n */\nexport interface BaseListParams extends PaginationParams {\n readonly sort_by?: string;\n readonly sort_order?: SortOrder;\n readonly search?: string;\n}\n","import type Color from \"colorjs.io\";\n\n// Semantic color names - matches portal-widgets tailwind.config.ts and field-types.ts\nexport const SEMANTIC_COLOR_NAMES = [\n \"background\",\n \"foreground\",\n \"primary\",\n \"secondary\",\n \"accent\",\n \"muted\",\n \"destructive\",\n] as const;\nexport type SemanticColorName = (typeof SEMANTIC_COLOR_NAMES)[number];\n\nexport const SHADE_STEPS = [\n 100, 200, 300, 400, 500, 600, 700, 800, 900,\n] as const;\nexport type ShadeStep = (typeof SHADE_STEPS)[number];\n\nexport const FONT_SIZE_KEYS = [\n \"extraSmall\",\n \"small\",\n \"regular\",\n \"large\",\n \"extraLarge\",\n \"giant\",\n] as const;\nexport type FontSizeKey = (typeof FONT_SIZE_KEYS)[number];\n\nexport const FONT_FAMILY_KEYS = [\"header\", \"body\"] as const;\nexport type FontFamilyKey = (typeof FONT_FAMILY_KEYS)[number];\n\nexport const RADIUS_KEYS = [\"small\", \"medium\", \"large\", \"extraLarge\"] as const;\nexport type RadiusKey = (typeof RADIUS_KEYS)[number];\n\n/** Author-time color input (what the user configures) */\nexport interface ThemeColorInput {\n base: Color;\n foreground: Color;\n}\n\n/** Complete theme definition — stored in-memory with Color objects */\nexport interface ThemeDefinition {\n id: string;\n name: string;\n /** Light mode — always fully specified */\n light: Record<SemanticColorName, ThemeColorInput>;\n /**\n * Dark mode — only user-overridden colors.\n * Missing keys are auto-derived from `light` at resolve time.\n */\n dark: Partial<Record<SemanticColorName, Partial<ThemeColorInput>>>;\n fontSizes: Record<FontSizeKey, string>;\n fontFamilies: Record<FontFamilyKey, string>;\n spacing: string;\n radii: Record<RadiusKey, string>;\n /** When true, theme colors are re-derived from brand guidelines on every load */\n syncWithBrandColors?: boolean;\n}\n\n/** Resolved semantic color with generated shade ramp */\nexport interface ResolvedSemanticColor {\n base: Color;\n foreground: Color;\n shades: Record<ShadeStep, Color>;\n}\n\n/** Complete resolved color set for one mode */\nexport type ResolvedColorSet = Record<SemanticColorName, ResolvedSemanticColor>;\n\n/** Fully resolved theme — all colors materialised for both modes */\nexport interface ResolvedTheme {\n id: string;\n name: string;\n light: ResolvedColorSet;\n dark: ResolvedColorSet;\n fontSizes: ThemeDefinition[\"fontSizes\"];\n fontFamilies: ThemeDefinition[\"fontFamilies\"];\n spacing: string;\n radii: ThemeDefinition[\"radii\"];\n}\n\n/** Plain OKLCH triplet for JSON serialisation (no Color dependency) */\nexport interface OklchPlain {\n l: number;\n c: number;\n h: number;\n}\n\n/** Serialised color pair as stored in the backend payload */\nexport interface ThemeColorPlain {\n base: OklchPlain;\n foreground: OklchPlain;\n}\n\n/** Backend payload — plain JSON, no Color objects */\nexport interface ThemePayload {\n [key: string]: unknown;\n id: string;\n name: string;\n light: Record<SemanticColorName, ThemeColorPlain>;\n dark: Partial<\n Record<SemanticColorName, { base?: OklchPlain; foreground?: OklchPlain }>\n >;\n fontSizes: Record<FontSizeKey, string>;\n fontFamilies: Record<FontFamilyKey, string>;\n spacing: string;\n radii: Record<RadiusKey, string>;\n syncWithBrandColors?: boolean;\n}\n","import Color from \"colorjs.io\";\nimport {\n SEMANTIC_COLOR_NAMES,\n SHADE_STEPS,\n type SemanticColorName,\n type ShadeStep,\n type ThemeColorInput,\n type ThemeDefinition,\n type ResolvedColorSet,\n type ResolvedTheme,\n} from \"./types\";\n\n/**\n * Attempt to convert any string into a Color using colorjs.io.\n * If the string is exactly 6 characters it is assumed to be a bare hex value\n * (e.g. \"3b82f6\") and a \"#\" prefix is added before parsing.\n *\n * @returns the parsed Color, or a neutral gray (`oklch(0.5 0 0)`) on failure\n */\nexport function parseColor(value: string): Color {\n if (value.length === 6) {\n value = `#${value}`;\n }\n try {\n return new Color(value);\n } catch (error) {\n console.warn(\"[theme] Failed to parse color:\", value, error);\n return new Color(\"oklch\", [0.5, 0, 0]);\n }\n}\n\n/**\n * Returns either the original foreground or a corrected lightness variant,\n * whichever provides better contrast against `color`.\n * Inversion triggers when the APCA contrast is below 50.\n */\nexport function getForegroundColor(foreground: Color, color: Color): Color {\n if (foreground.oklch.l == null || color.oklch.l == null) {\n return foreground;\n }\n const contrast = color.contrastAPCA(foreground);\n\n if (contrast < 50) {\n return new Color(\"oklch\", [\n color.oklch.l < 0.7 ? 0.95 : 0.15,\n foreground.oklch.c || 0,\n foreground.oklch.h || 0,\n ]);\n }\n return foreground;\n}\n\n/**\n * Generate a 100–900 shade ramp from a base color.\n * Base anchors at 500. Light shades (100–400) step toward white,\n * dark shades (600–900) step toward black. Dark steps use an asymmetric\n * multiplier (1.6×, 1.875×, 3×, 4× of `darkStep`) for a more gradual\n * initial descent. Chroma is nudged per step for perceptually natural ramps.\n */\nexport function generateShades(base: Color): Record<ShadeStep, Color> {\n const l = base.oklch.l ?? 0;\n const c = base.oklch.c ?? 0;\n const h = base.oklch.h ?? 0;\n\n const safeMax = l >= 0.885 ? 0.995 : 0.97;\n const safeMin = l <= 0.33 ? 0 : 0.21;\n\n const lightStep = (safeMax - l) / 5;\n const darkStep = -(l - safeMin) / 8;\n\n const shade = (lDelta: number, cDelta: number): Color => {\n return new Color(\"oklch\", [\n Math.max(0, Math.min(1, l + lDelta)),\n c <= 0.001 ? c : Math.max(0, c + cDelta),\n h,\n ]);\n };\n\n return {\n 100: shade(5 * lightStep, -0.00375),\n 200: shade(4 * lightStep, -0.00375),\n 300: shade(3 * lightStep, -0.00375),\n 400: shade(2 * lightStep, -0.00375),\n 500: new Color(\"oklch\", [l, c, h]),\n 600: shade(1.6 * darkStep, 0.025),\n 700: shade(1.875 * 2 * darkStep, 0.05),\n 800: shade(3 * 2 * darkStep, 0.075),\n 900: shade(4 * 2 * darkStep, 0.1),\n };\n}\n\n// ── Dark Mode Derivation ────────────────────────────────────────────\n//\n// Dark-mode colors are derived from their light counterparts by adjusting\n// OKLCH lightness and optionally scaling chroma. Neutral slots (background,\n// foreground, muted) use fixed lightness values while chromatic slots\n// (primary, secondary, accent, destructive) invert lightness around 0.5.\n\nconst DARK_DERIVATION_CONFIG: Record<\n SemanticColorName,\n {\n baseLightness: number | \"invert\";\n fgLightness: number | \"invert\";\n chromaScale?: number;\n }\n> = {\n background: { baseLightness: 0.15, fgLightness: 0.93 },\n foreground: { baseLightness: 0.93, fgLightness: 0.15 },\n muted: { baseLightness: 0.22, fgLightness: 0.75 },\n primary: { baseLightness: \"invert\", fgLightness: 0.95, chromaScale: 0.9 },\n secondary: { baseLightness: \"invert\", fgLightness: 0.93, chromaScale: 0.85 },\n accent: { baseLightness: \"invert\", fgLightness: 0.95, chromaScale: 0.9 },\n destructive: {\n baseLightness: \"invert\",\n fgLightness: 0.95,\n chromaScale: 0.95,\n },\n};\n\n/** Invert OKLCH lightness (1 - l), clamped to [0.35, 0.75] to avoid extremes. */\nfunction invertLightness(l: number): number {\n const inverted = 1 - l;\n return Math.max(0.35, Math.min(0.75, inverted));\n}\n\n/**\n * Derive a dark-mode ThemeColorInput from its light-mode counterpart.\n */\nexport function deriveDarkVariant(\n name: SemanticColorName,\n light: ThemeColorInput,\n): ThemeColorInput {\n const config = DARK_DERIVATION_CONFIG[name];\n const chromaScale = config.chromaScale ?? 1;\n\n const baseLightness =\n config.baseLightness === \"invert\"\n ? invertLightness(light.base.oklch.l ?? 0)\n : config.baseLightness;\n\n const fgLightness =\n config.fgLightness === \"invert\"\n ? invertLightness(light.foreground.oklch.l ?? 0)\n : config.fgLightness;\n\n return {\n base: new Color(\"oklch\", [\n baseLightness,\n (light.base.oklch.c || 0) * chromaScale,\n light.base.oklch.h || 0,\n ]),\n foreground: new Color(\"oklch\", [\n fgLightness,\n (light.foreground.oklch.c || 0) * chromaScale,\n light.foreground.oklch.h || 0,\n ]),\n };\n}\n\n// ── Dark Mode Merge ─────────────────────────────────────────────────\n\n/**\n * Merge auto-derived dark colors with any user-specified overrides.\n * For each semantic color, if the user has fully overridden both base and\n * foreground those are used; otherwise the missing channels are derived.\n */\nexport function mergeDarkOverrides(\n def: ThemeDefinition,\n): Record<SemanticColorName, ThemeColorInput> {\n const darkColors = {} as Record<SemanticColorName, ThemeColorInput>;\n\n for (const name of SEMANTIC_COLOR_NAMES) {\n const lightInput = def.light[name];\n const darkOverride = def.dark[name];\n\n if (darkOverride?.base && darkOverride?.foreground) {\n darkColors[name] = darkOverride as ThemeColorInput;\n } else if (darkOverride) {\n const base =\n darkOverride.base ?? deriveDarkVariant(name, lightInput).base;\n darkColors[name] = {\n base: base,\n foreground:\n darkOverride.foreground ??\n getForegroundColor(def.light.foreground.base, base),\n };\n } else {\n darkColors[name] = deriveDarkVariant(name, lightInput);\n }\n }\n\n return darkColors;\n}\n\n// ── Theme Resolution ────────────────────────────────────────────────\n\nfunction resolveColorSet(\n colors: Record<SemanticColorName, ThemeColorInput>,\n): ResolvedColorSet {\n const resolved = {} as ResolvedColorSet;\n\n for (const name of SEMANTIC_COLOR_NAMES) {\n const input = colors[name];\n const shades = generateShades(input.base);\n const resolvedShades = {} as Record<ShadeStep, Color>;\n\n for (const step of SHADE_STEPS) {\n resolvedShades[step] = shades[step];\n }\n\n resolved[name] = {\n base: input.base.clone(),\n foreground: input.foreground.clone(),\n shades: resolvedShades,\n };\n }\n\n return resolved;\n}\n\n/**\n * Resolve a ThemeDefinition into a complete ResolvedTheme.\n * Dark mode colors are derived from light where not overridden.\n */\nexport function resolveTheme(def: ThemeDefinition): ResolvedTheme {\n return {\n id: def.id,\n name: def.name,\n light: resolveColorSet(def.light),\n dark: resolveColorSet(mergeDarkOverrides(def)),\n fontSizes: { ...def.fontSizes },\n fontFamilies: { ...def.fontFamilies },\n spacing: def.spacing,\n radii: { ...def.radii },\n };\n}\n","import { SHADE_STEPS, type SemanticColorName, type ShadeStep } from \"./types\";\n\n/**\n * Specific overrides, otherwise all the overrides are generated using emitTailwindOverrides\n */\nconst OVERRIDES: Partial<Record<string, string>> = {\n \"--color-gray-50\": \"var(--color-muted)\",\n \"--color-gray-100\": \"var(--color-muted-600)\",\n \"--color-gray-200\": \"var(--color-border)\",\n} as const;\n\n/**\n * Returns the inverted shade for dark mode foreground colors.\n * In dark mode, light shades (50, 100) should map to dark values (950, 900) and vice versa.\n */\nfunction getInvertedStep(shade: ShadeStep): ShadeStep {\n const shadeIndex = SHADE_STEPS.indexOf(shade);\n const invertedIndex = SHADE_STEPS.length - 1 - shadeIndex;\n return SHADE_STEPS[invertedIndex] || 500;\n}\n\n/**\n * Map semantic colors to Tailwind built-in color names.\n */\nexport function emitTailwindOverrides(darkMode: boolean = false): string[] {\n const TAILWIND_COLOR_MAP: Record<string, SemanticColorName> = {\n gray: \"foreground\",\n red: \"destructive\",\n blue: \"primary\",\n green: \"accent\",\n };\n\n const TAILWIND_SHADES = [\n 50, 100, 200, 300, 400, 500, 600, 700, 800, 900, 950,\n ] as const;\n const SHADE_REMAP: Partial<Record<number, ShadeStep>> = {\n 50: 100,\n 950: 900,\n };\n\n const lines: string[] = [];\n for (const [twName, semantic] of Object.entries(TAILWIND_COLOR_MAP)) {\n for (const shade of TAILWIND_SHADES) {\n const step = (SHADE_REMAP[shade] ?? shade) as ShadeStep;\n const override = OVERRIDES[`--color-${twName}-${shade}`];\n lines.push(\n `--color-${twName}-${shade}: ${override ? override : `var(--color-${semantic}-${semantic === \"foreground\" && darkMode === true ? getInvertedStep(step) : step})`};`,\n );\n }\n }\n\n lines.push(\"--color-white: var(--color-background);\");\n lines.push(\"--color-black: var(--color-foreground);\");\n\n return lines;\n}\n","import { emitTailwindOverrides } from \"./tailwind-overrides\";\nimport {\n SEMANTIC_COLOR_NAMES,\n SHADE_STEPS,\n FONT_SIZE_KEYS,\n FONT_FAMILY_KEYS,\n RADIUS_KEYS,\n type ResolvedColorSet,\n type ResolvedTheme,\n} from \"./types\";\n\nfunction colorToCSS(color: import(\"colorjs.io\").default): string {\n const result = color.toString({ format: \"oklch\" });\n if (result.includes(\"NaN\")) {\n console.warn(\n \"[theme] colorToCSS produced NaN, using neutral fallback:\",\n result,\n );\n return \"oklch(0.5 0 0)\";\n }\n return result;\n}\n\nfunction camelToKebab(str: string): string {\n return str.replace(/([a-z])([A-Z])/g, \"$1-$2\").toLowerCase();\n}\n\n/**\n * Emit --color-{name}, --color-{name}-foreground, --color-{name}-{shade} vars.\n * Uses --color- prefix to match portal-widgets/tailwind.config.ts.\n */\nfunction emitColorVars(colors: ResolvedColorSet): string[] {\n const lines: string[] = [];\n\n for (const name of SEMANTIC_COLOR_NAMES) {\n const color = colors[name];\n lines.push(`--color-${name}: ${colorToCSS(color.base)};`);\n lines.push(`--color-${name}-foreground: ${colorToCSS(color.foreground)};`);\n for (const step of SHADE_STEPS) {\n lines.push(`--color-${name}-${step}: ${colorToCSS(color.shades[step])};`);\n }\n }\n\n return lines;\n}\n\n/**\n * Format a font family value for CSS output.\n * - If the value starts with \"var(\" (legacy), pass through as-is\n * - If the value already contains a comma (has fallback), pass through as-is\n * - Otherwise, wrap in quotes and append a generic sans-serif fallback\n */\nfunction formatFontFamily(value: string): string {\n if (value.startsWith(\"var(\")) return value;\n if (value.includes(\",\")) return value;\n return `'${value}', sans-serif`;\n}\n\n/**\n * Emit non-color CSS variables (font sizes, families, spacing, radii).\n */\nfunction emitNonColorVars(theme: ResolvedTheme): string[] {\n const lines: string[] = [];\n for (const key of FONT_SIZE_KEYS) {\n lines.push(`--font-size-${camelToKebab(key)}: ${theme.fontSizes[key]};`);\n }\n for (const key of FONT_FAMILY_KEYS) {\n lines.push(`--font-${key}: ${formatFontFamily(theme.fontFamilies[key])};`);\n }\n lines.push(`--spacing: ${theme.spacing};`);\n for (const key of RADIUS_KEYS) {\n lines.push(`--radius-${camelToKebab(key)}: ${theme.radii[key]};`);\n }\n return lines;\n}\n\n/**\n * Static CSS alias variables that bridge theme var names to Tailwind/component conventions.\n * These are always emitted and not mode-dependent.\n */\nconst globalCSSOverride = [\n \"--color-background-foreground: var(--color-foreground);\",\n \"--color-foreground-foreground: var(--color-background);\",\n \"--color-contrast: var(--color-foreground);\",\n ...SEMANTIC_COLOR_NAMES.map((value) => `--${value}: var(--color-${value});`),\n ...SEMANTIC_COLOR_NAMES.map(\n (value) => `--${value}-foreground: var(--color-${value}-foreground);`,\n ),\n\n \"--sidebar-ring: var(--color-primary);\",\n \"--sidebar-border: var(--color-border);\",\n \"--sidebar-accent-foreground: var(--color-accent-foreground);\",\n \"--sidebar-accent: var(--color-accent);\",\n \"--sidebar-primary-foreground: var(--color-primary-foreground);\",\n \"--sidebar-primary: var(--color-primary);\",\n \"--sidebar-foreground: var(--color-muted-foreground);\",\n \"--sidebar: var(--color-muted);\",\n \"--border: var(--color-background-600);\",\n \"--ring: var(--color-primary);\",\n \"--popover: var(--color-background);\",\n \"--popover-foreground: var(--color-foreground);\",\n \"--card: var(--color-muted);\",\n \"--card-foreground: var(--color-muted-foreground);\",\n\n \"--radius-sm: var(--radius-small);\",\n \"--radius-md: var(--radius-medium);\",\n \"--radius-lg: var(--radius-large);\",\n \"--radius-xl: var(--radius-extra-large);\",\n \"--text-xs: var(--font-size-extra-small);\",\n \"--text-sm: var(--font-size-small);\",\n \"--text-base: var(--font-size-regular);\",\n \"--text-lg: var(--font-size-large);\",\n \"--text-xl: var(--font-size-extra-large);\",\n \"--text-2xl: var(--font-size-giant);\",\n];\n\n/**\n * Overrides for global tailwindcss for specifically dark mode.\n */\nconst globalDarkCSSOverride = [\"--border: var(--color-background-400);\"];\n\nexport interface GenerateThemeCSSOptions {\n /** Whether or not to allow prefers-color-scheme to choose the theme mode */\n disableAutoTheme?: boolean;\n /** Whether to emit Tailwind built-in color overrides (default true) */\n mapTailwindColors?: boolean;\n}\n\n/**\n * Generate a complete CSS string for a resolved theme.\n * Outputs 2–3 blocks: light default, dark explicit via `[data-theme-mode=\"dark\"]`,\n * and (unless `disableAutoTheme`) a `prefers-color-scheme: dark` media query block.\n */\nexport function generateThemeCSS(\n theme: ResolvedTheme,\n options: GenerateThemeCSSOptions = {},\n): string {\n const sel = `[data-theme=\"${theme.id}\"]`;\n const tw = options.mapTailwindColors ?? true;\n const blocks: string[] = [];\n\n // Light mode (default)\n blocks.push(`${sel} {`);\n blocks.push(...globalCSSOverride);\n blocks.push(...emitNonColorVars(theme));\n blocks.push(...emitColorVars(theme.light));\n if (tw) blocks.push(...emitTailwindOverrides());\n blocks.push(`}`);\n\n // Dark mode: explicit via attribute\n blocks.push(`${sel}[data-theme-mode=\"dark\"] {`);\n blocks.push(...globalDarkCSSOverride);\n blocks.push(...emitColorVars(theme.dark));\n if (tw) blocks.push(...emitTailwindOverrides(true));\n blocks.push(`}`);\n\n // Dark mode: auto via system preference\n if (!options.disableAutoTheme) {\n blocks.push(`@media (prefers-color-scheme: dark) {`);\n blocks.push(`${sel}:not([data-theme-mode]) {`);\n blocks.push(...globalDarkCSSOverride);\n blocks.push(...emitColorVars(theme.dark).map((l) => `${l}`));\n if (tw) blocks.push(...emitTailwindOverrides(true).map((l) => `${l}`));\n blocks.push(`}`);\n blocks.push(`}`);\n }\n\n return blocks.join(\"\\n\");\n}\n","import Color from \"colorjs.io\";\nimport type {\n FontSizeKey,\n FontFamilyKey,\n RadiusKey,\n ThemeDefinition,\n} from \"./types\";\nimport { getForegroundColor } from \"./color-engine\";\n\n// ── Non-color defaults ──────────────────────────────────────────────\n\nexport const DEFAULT_FONT_SIZES: Record<FontSizeKey, string> = {\n extraSmall: \"0.75rem\",\n small: \"0.875rem\",\n regular: \"1rem\",\n large: \"1.125rem\",\n extraLarge: \"1.25rem\",\n giant: \"1.5rem\",\n};\n\nexport const DEFAULT_FONT_FAMILIES: Record<FontFamilyKey, string> = {\n header: \"Inter\",\n body: \"Inter\",\n};\n\nexport const DEFAULT_SPACING = \"0.25rem\";\n\nexport const DEFAULT_RADII: Record<RadiusKey, string> = {\n small: \"0.25rem\",\n medium: \"0.5rem\",\n large: \"0.75rem\",\n extraLarge: \"1rem\",\n};\n\n// ── Default colors (hex) ────────────────────────────────────────────\n\nexport const DEFAULT_COLORS = {\n background: \"#ffffff\",\n foreground: \"#1a1a1a\",\n primary: \"#3b82f6\",\n secondary: \"#6b7280\",\n accent: \"#10b981\",\n muted: \"#f3f4f6\",\n destructive: \"#ef4444\",\n mutedForeground: \"#6b7280\",\n} as const;\n\n// ── Default theme identity ──────────────────────────────────────────\n\nexport const DEFAULT_THEME_ID = \"default\";\nexport const DEFAULT_THEME_NAME = \"Default Theme\";\n\n// ── Factory ─────────────────────────────────────────────────────────\n\n/**\n * Build a fresh ThemeDefinition populated with all defaults.\n * Returns a new object each call because Color instances are mutable — do not cache the result.\n */\nexport function getDefaultThemeDefinition(): ThemeDefinition {\n const bg = new Color(DEFAULT_COLORS.background);\n const fg = new Color(DEFAULT_COLORS.foreground);\n const primary = new Color(DEFAULT_COLORS.primary);\n const secondary = new Color(DEFAULT_COLORS.secondary);\n const accent = new Color(DEFAULT_COLORS.accent);\n const muted = new Color(DEFAULT_COLORS.muted);\n const destructive = new Color(DEFAULT_COLORS.destructive);\n const mutedFg = new Color(DEFAULT_COLORS.mutedForeground);\n\n const darkBg = new Color(\"#0a0a0a\");\n const darkFg = new Color(\"#fafafa\");\n const darkMuted = new Color(\"#171717\");\n const darkMutedForeground = new Color(\"#dddddd\");\n\n return {\n id: DEFAULT_THEME_ID,\n name: DEFAULT_THEME_NAME,\n light: {\n background: { base: bg, foreground: fg },\n foreground: { base: fg, foreground: bg },\n primary: {\n base: primary,\n foreground: getForegroundColor(fg, primary),\n },\n secondary: {\n base: secondary,\n foreground: getForegroundColor(fg, secondary),\n },\n accent: {\n base: accent,\n foreground: getForegroundColor(fg, accent),\n },\n muted: { base: muted, foreground: mutedFg },\n destructive: {\n base: destructive,\n foreground: getForegroundColor(fg, destructive),\n },\n },\n dark: {\n background: { base: darkBg, foreground: darkFg },\n foreground: { base: darkFg, foreground: darkBg },\n muted: { base: darkMuted, foreground: darkMutedForeground },\n },\n fontSizes: { ...DEFAULT_FONT_SIZES },\n fontFamilies: { ...DEFAULT_FONT_FAMILIES },\n spacing: DEFAULT_SPACING,\n radii: { ...DEFAULT_RADII },\n };\n}\n","import Color from \"colorjs.io\";\nimport {\n SEMANTIC_COLOR_NAMES,\n type SemanticColorName,\n type ThemeColorInput,\n type ThemeDefinition,\n type ThemePayload,\n type OklchPlain,\n type FontSizeKey,\n type FontFamilyKey,\n type RadiusKey,\n} from \"./types\";\nimport {\n DEFAULT_FONT_SIZES,\n DEFAULT_FONT_FAMILIES,\n DEFAULT_SPACING,\n DEFAULT_RADII,\n getDefaultThemeDefinition,\n} from \"./defaults\";\n\nfunction colorToPlain(color: Color): OklchPlain {\n return {\n l: color.oklch.l ?? 0,\n c: color.oklch.c ?? 0,\n h: color.oklch.h ?? 0,\n };\n}\n\nfunction plainToColor(plain: OklchPlain): Color {\n return new Color(\"oklch\", [plain.l, plain.c, plain.h]);\n}\n\n/**\n * Serialise a ThemeDefinition (with Color objects) to a plain JSON payload\n * suitable for backend storage.\n */\nexport function serialiseTheme(def: ThemeDefinition): ThemePayload {\n const light = {} as ThemePayload[\"light\"];\n for (const name of SEMANTIC_COLOR_NAMES) {\n light[name] = {\n base: colorToPlain(def.light[name].base),\n foreground: colorToPlain(def.light[name].foreground),\n };\n }\n\n const dark: ThemePayload[\"dark\"] = {};\n for (const [name, value] of Object.entries(def.dark)) {\n if (!value) continue;\n dark[name as SemanticColorName] = {\n ...(value.base ? { base: colorToPlain(value.base) } : {}),\n ...(value.foreground\n ? { foreground: colorToPlain(value.foreground) }\n : {}),\n };\n }\n\n return {\n id: def.id,\n name: def.name,\n light,\n dark,\n fontSizes: { ...def.fontSizes },\n fontFamilies: { ...def.fontFamilies },\n spacing: def.spacing,\n radii: { ...def.radii },\n ...(def.syncWithBrandColors ? { syncWithBrandColors: true } : {}),\n };\n}\n\n/**\n * Deserialise a backend payload into a ThemeDefinition with Color objects.\n * Accepts `Record<string, unknown>` because API data is untyped at the boundary.\n * Falls back to default colors for any missing light-mode entries.\n */\nexport function deserialiseTheme(\n payload: Record<string, unknown>,\n): ThemeDefinition {\n const lightRaw = ((payload.light as Record<string, unknown> | undefined) ??\n {}) as Record<string, { base: OklchPlain; foreground: OklchPlain }>;\n const darkRaw = ((payload.dark as Record<string, unknown> | undefined) ??\n {}) as Record<string, { base?: OklchPlain; foreground?: OklchPlain }>;\n\n const defaults = getDefaultThemeDefinition();\n const light = {} as Record<SemanticColorName, ThemeColorInput>;\n for (const name of SEMANTIC_COLOR_NAMES) {\n const entry = lightRaw[name];\n if (entry) {\n light[name] = {\n base: plainToColor(entry.base),\n foreground: plainToColor(entry.foreground),\n };\n } else {\n console.warn(\n `[theme] deserialiseTheme: missing light color \"${name}\", using default`,\n );\n light[name] = defaults.light[name];\n }\n }\n\n const dark: Partial<Record<SemanticColorName, Partial<ThemeColorInput>>> = {};\n for (const [name, value] of Object.entries(darkRaw)) {\n if (!value) continue;\n dark[name as SemanticColorName] = {\n ...(value.base ? { base: plainToColor(value.base) } : {}),\n ...(value.foreground\n ? { foreground: plainToColor(value.foreground) }\n : {}),\n };\n }\n\n return {\n id: payload.id as string,\n name: payload.name as string,\n light,\n dark,\n fontSizes: (payload.fontSizes ?? DEFAULT_FONT_SIZES) as Record<\n FontSizeKey,\n string\n >,\n fontFamilies: (payload.fontFamilies ?? DEFAULT_FONT_FAMILIES) as Record<\n FontFamilyKey,\n string\n >,\n spacing: (payload.spacing as string) ?? DEFAULT_SPACING,\n radii: (payload.radii ?? DEFAULT_RADII) as Record<RadiusKey, string>,\n ...(payload.syncWithBrandColors === true\n ? { syncWithBrandColors: true }\n : {}),\n };\n}\n","/**\n * Theme Transforms\n * Convert raw API theme objects to ThemeDefinition format.\n * Handles both new structured format (OKLCH) and legacy flat format (hex strings).\n */\n\nimport type { ThemeDefinition } from \"./types\";\nimport { deserialiseTheme } from \"./serialisation\";\nimport { parseColor, getForegroundColor } from \"./color-engine\";\nimport {\n DEFAULT_COLORS,\n DEFAULT_FONT_SIZES,\n DEFAULT_FONT_FAMILIES,\n DEFAULT_SPACING,\n DEFAULT_RADII,\n} from \"./defaults\";\n\n/** Shape of a raw theme from the FluidOS API */\nexport interface RawApiTheme {\n id: number;\n config?: Record<string, unknown> | null;\n active?: boolean | null;\n name?: string | null;\n}\n\n/**\n * Check if a theme config uses the new structured format (has a `light` key\n * that is an object) vs the legacy flat format.\n */\nfunction isNewThemeFormat(config: Record<string, unknown>): boolean {\n return config.light != null && typeof config.light === \"object\";\n}\n\n/**\n * Convert a legacy flat config to a ThemeDefinition.\n * Legacy format: { base: \"#fff\", text: \"#000\", primary: \"oklch(0.6 0.2 250)\", ... }\n */\nfunction legacyConfigToDefinition(\n id: number,\n name: string,\n config: Record<string, string>,\n): ThemeDefinition {\n const bg = parseColor(\n config.base ?? config.background ?? DEFAULT_COLORS.background,\n );\n const fg = parseColor(\n config.text ?? config.foreground ?? DEFAULT_COLORS.foreground,\n );\n const primary = parseColor(config.primary ?? DEFAULT_COLORS.primary);\n const secondary = parseColor(config.secondary ?? DEFAULT_COLORS.secondary);\n const accent = parseColor(config.accent ?? DEFAULT_COLORS.accent);\n const muted = parseColor(config.muted ?? DEFAULT_COLORS.muted);\n const destructive = parseColor(\n config.destructive ?? DEFAULT_COLORS.destructive,\n );\n const mutedFg = parseColor(\n config.mutedForeground ?? DEFAULT_COLORS.mutedForeground,\n );\n\n return {\n id: String(id),\n name,\n light: {\n background: { base: bg, foreground: fg },\n foreground: { base: fg, foreground: bg },\n primary: {\n base: primary,\n foreground: getForegroundColor(fg, primary),\n },\n secondary: {\n base: secondary,\n foreground: getForegroundColor(fg, secondary),\n },\n accent: {\n base: accent,\n foreground: getForegroundColor(fg, accent),\n },\n muted: { base: muted, foreground: mutedFg },\n destructive: {\n base: destructive,\n foreground: getForegroundColor(fg, destructive),\n },\n },\n dark: {},\n fontSizes: {\n extraSmall: config.extraSmall ?? DEFAULT_FONT_SIZES.extraSmall,\n small: config.small ?? DEFAULT_FONT_SIZES.small,\n regular: config.regular ?? DEFAULT_FONT_SIZES.regular,\n large: config.large ?? DEFAULT_FONT_SIZES.large,\n extraLarge: config.extraLarge ?? DEFAULT_FONT_SIZES.extraLarge,\n giant: config.giant ?? DEFAULT_FONT_SIZES.giant,\n },\n fontFamilies: {\n header: config.headerFont ?? DEFAULT_FONT_FAMILIES.header,\n body: config.bodyFont ?? DEFAULT_FONT_FAMILIES.body,\n },\n spacing: config.globalSpacing ?? DEFAULT_SPACING,\n radii: {\n small: config.radiusSmall ?? DEFAULT_RADII.small,\n medium: config.radiusMedium ?? DEFAULT_RADII.medium,\n large: config.radiusLarge ?? DEFAULT_RADII.large,\n extraLarge: config.radiusExtraLarge ?? DEFAULT_RADII.extraLarge,\n },\n };\n}\n\n/**\n * Build a ThemeDefinition from a single API theme object.\n * Handles both new structured format and legacy flat format.\n */\nexport function buildThemeDefinition(theme: RawApiTheme): ThemeDefinition {\n const config = (theme.config ?? {}) as Record<string, unknown>;\n\n if (isNewThemeFormat(config)) {\n return deserialiseTheme({\n ...config,\n id: String(theme.id),\n name: theme.name ?? \"Untitled Theme\",\n });\n }\n\n return legacyConfigToDefinition(\n theme.id,\n theme.name ?? \"Untitled Theme\",\n config as Record<string, string>,\n );\n}\n\n/**\n * Transform raw API themes to ThemeDefinition[].\n * Catches and logs errors per theme (graceful degradation).\n */\nexport function transformThemes(themes: RawApiTheme[]): ThemeDefinition[] {\n return themes.flatMap((theme) => {\n try {\n return [buildThemeDefinition(theme)];\n } catch (error) {\n console.error(`[theme] Failed to build theme id=${theme.id}:`, error);\n return [];\n }\n });\n}\n\n/**\n * Get the active theme ID from a list of raw API themes.\n * Falls back to the first theme if none is marked active.\n */\nexport function getActiveThemeId(themes: RawApiTheme[]): string | undefined {\n const active = themes.find((t) => t.active) ?? themes[0];\n return active ? String(active.id) : undefined;\n}\n","import { generateThemeCSS } from \"./css-generator\";\nimport { FONT_FAMILY_KEYS, type ResolvedTheme } from \"./types\";\nimport type { GenerateThemeCSSOptions } from \"./css-generator\";\n\nconst STYLE_PREFIX = \"theme-style-\";\nconst FONT_LINK_PREFIX = \"theme-font-\";\n\nconst SYSTEM_FONTS = new Set([\n \"sans-serif\",\n \"serif\",\n \"monospace\",\n \"cursive\",\n \"fantasy\",\n \"system-ui\",\n \"ui-sans-serif\",\n \"ui-serif\",\n \"ui-monospace\",\n]);\n\n/** Build a Google Fonts CSS2 URL for a given font family with all weights. */\nfunction buildGoogleFontUrl(family: string): string {\n const encoded = encodeURIComponent(family).replace(/%20/g, \"+\");\n return `https://fonts.googleapis.com/css2?family=${encoded}:wght@100;200;300;400;500;600;700;800;900&display=swap`;\n}\n\n/** Check if a font family value needs to be loaded (i.e. is not a CSS var or system font). */\nfunction isLoadableFont(value: string): boolean {\n if (!value) return false;\n if (value.startsWith(\"var(\")) return false;\n return !SYSTEM_FONTS.has(value.toLowerCase());\n}\n\n/** Deterministic link element ID for a font family. */\nfunction fontLinkId(family: string): string {\n return `${FONT_LINK_PREFIX}${family.replace(/\\s+/g, \"-\").toLowerCase()}`;\n}\n\n/**\n * Inject or update `<link>` elements for Google Fonts used by the theme.\n * Removes links for fonts that are no longer referenced.\n */\nfunction loadThemeFonts(theme: ResolvedTheme): void {\n if (typeof document === \"undefined\") return;\n\n const fontsToLoad = new Set<string>();\n for (const key of FONT_FAMILY_KEYS) {\n const value = theme.fontFamilies[key];\n if (isLoadableFont(value)) {\n fontsToLoad.add(value);\n }\n }\n\n // Remove stale font links owned by this theme\n const existingLinks = document.querySelectorAll(\n `link[id^=\"${FONT_LINK_PREFIX}\"]`,\n );\n existingLinks.forEach((link) => {\n const owners = link.getAttribute(\"data-font-theme-ids\")?.split(\",\") ?? [];\n if (!owners.includes(theme.id)) return;\n const fontName = link.getAttribute(\"data-font-family\");\n if (fontName && !fontsToLoad.has(fontName)) {\n const remaining = owners.filter((id) => id !== theme.id);\n if (remaining.length === 0) {\n link.remove();\n } else {\n link.setAttribute(\"data-font-theme-ids\", remaining.join(\",\"));\n }\n }\n });\n\n // Add new font links (or register this theme as an owner of existing ones)\n for (const family of fontsToLoad) {\n const id = fontLinkId(family);\n const existing = document.getElementById(id);\n if (existing) {\n const owners =\n existing.getAttribute(\"data-font-theme-ids\")?.split(\",\") ?? [];\n if (!owners.includes(theme.id)) {\n existing.setAttribute(\n \"data-font-theme-ids\",\n [...owners, theme.id].join(\",\"),\n );\n }\n } else {\n const link = document.createElement(\"link\");\n link.id = id;\n link.rel = \"stylesheet\";\n link.href = buildGoogleFontUrl(family);\n link.setAttribute(\"data-font-family\", family);\n link.setAttribute(\"data-font-theme-ids\", theme.id);\n document.head.appendChild(link);\n }\n }\n}\n\n/** Remove all font `<link>` elements injected by the theme system. */\nfunction removeAllFontLinks(): void {\n if (typeof document === \"undefined\") return;\n document\n .querySelectorAll(`link[id^=\"${FONT_LINK_PREFIX}\"]`)\n .forEach((el) => el.remove());\n}\n\n/**\n * Inject or update a `<style>` element in `<head>` for the given theme.\n * The element ID is deterministic (`theme-style-{themeId}`) so repeated calls\n * for the same theme are idempotent — the existing element is updated in place.\n * Also loads Google Fonts referenced by the theme's font families.\n * No-op when `document` is unavailable (SSR).\n */\nexport function applyTheme(\n theme: ResolvedTheme,\n options?: GenerateThemeCSSOptions,\n): void {\n if (typeof document === \"undefined\") return;\n\n try {\n loadThemeFonts(theme);\n\n const styleId = `${STYLE_PREFIX}${theme.id}`;\n let el = document.getElementById(styleId) as HTMLStyleElement | null;\n\n if (!el) {\n el = document.createElement(\"style\");\n el.id = styleId;\n document.head.appendChild(el);\n }\n\n el.textContent = generateThemeCSS(theme, options);\n } catch (error) {\n console.error(`[theme] applyTheme failed for \"${theme.id}\":`, error);\n }\n}\n\n/** Remove an injected theme stylesheet and clean up font link ownership. No-op during SSR. */\nexport function removeTheme(themeId: string): void {\n if (typeof document === \"undefined\") return;\n document.getElementById(`${STYLE_PREFIX}${themeId}`)?.remove();\n\n // Remove this theme from font link ownership; delete links with no remaining owners\n document\n .querySelectorAll(`link[id^=\"${FONT_LINK_PREFIX}\"]`)\n .forEach((link) => {\n const owners = link.getAttribute(\"data-font-theme-ids\")?.split(\",\") ?? [];\n const remaining = owners.filter((id) => id !== themeId);\n if (remaining.length === 0) {\n link.remove();\n } else {\n link.setAttribute(\"data-font-theme-ids\", remaining.join(\",\"));\n }\n });\n}\n\n/** Remove all injected theme stylesheets and font links. No-op during SSR. */\nexport function removeAllThemes(): void {\n if (typeof document === \"undefined\") return;\n document\n .querySelectorAll(`style[id^=\"${STYLE_PREFIX}\"]`)\n .forEach((el) => el.remove());\n removeAllFontLinks();\n}\n","/**\n * Screen Transforms\n * Convert FluidOS API screen objects to ScreenDefinition format.\n *\n * Extracted from:\n * - apps/fluid-admin/networking/app-builder/app-screens/types.ts\n * - apps/fluid-admin/networking/reps/screens.api.ts\n */\n\nimport type { WidgetSchema } from \"@fluid-app/portal-core/types\";\nimport type { ScreenDefinition } from \"@fluid-app/portal-core/types\";\n\n/** Raw screen from the FluidOS API */\nexport interface RawApiScreen {\n id: number | string;\n definition_id?: number | string;\n name?: string | null;\n slug?: string | null;\n component_tree?: unknown;\n}\n\n/**\n * Normalize component_tree to always be an array.\n * The API stores component_tree as a hash (object), but the frontend expects an array.\n */\nexport function normalizeComponentTree(componentTree: unknown): WidgetSchema[] {\n if (!componentTree) return [];\n if (Array.isArray(componentTree)) return componentTree as WidgetSchema[];\n if (typeof componentTree === \"object\") {\n return [componentTree as WidgetSchema];\n }\n return [];\n}\n\n/**\n * Convert a raw FluidOS screen to ScreenDefinition.\n * Normalizes component_tree and converts string IDs to numbers.\n */\nexport function toScreenDefinition(screen: RawApiScreen): ScreenDefinition {\n return {\n id: Number(screen.id),\n slug: screen.slug ?? \"\",\n name: screen.name ?? \"\",\n component_tree: normalizeComponentTree(screen.component_tree),\n };\n}\n","/**\n * Navigation Transforms\n * Convert FluidOS API navigation items to NavigationItem format.\n *\n * Extracted from:\n * - apps/fluid-admin/networking/app-builder/app-navigation-items/types.ts\n * - apps/fluid-admin/networking/reps/screens.api.ts\n */\n\nimport type { NavigationItem } from \"@fluid-app/portal-core/types\";\nimport type { FluidOsApiNavigationItem } from \"@fluid-app/portal-core/fluidos-api\";\n\n/** Raw navigation item from the FluidOS API. */\nexport type RawApiNavigationItem = FluidOsApiNavigationItem;\n\n/**\n * Convert a raw FluidOS navigation item to NavigationItem.\n * Recursively transforms children and sorts by position.\n */\nexport function toNavigationItem(item: RawApiNavigationItem): NavigationItem {\n const children = (item.children ?? [])\n .map(toNavigationItem)\n .sort((a, b) => (a.position ?? 0) - (b.position ?? 0));\n\n return {\n id: Number(item.id),\n label: item.label ?? \"Untitled\",\n // Use conditional spread for optional properties (exactOptionalPropertyTypes)\n ...(item.slug != null ? { slug: String(item.slug) } : {}),\n ...(item.icon != null ? { icon: String(item.icon) } : {}),\n ...(item.screen_id != null ? { screen_id: Number(item.screen_id) } : {}),\n ...(item.parent_id != null ? { parent_id: Number(item.parent_id) } : {}),\n ...(item.source != null ? { source: item.source } : {}),\n position: item.position ?? 0,\n children,\n };\n}\n","/**\n * Data Transforms\n * Convert FluidOS API responses to RepAppData format.\n *\n * This is the main entry point for all transforms used by the SDK client\n * and hooks when fetching from the fluidos API.\n */\n\nexport {\n transformThemes,\n buildThemeDefinition,\n getActiveThemeId,\n type RawApiTheme,\n} from \"@fluid-app/portal-core/theme\";\n\nexport {\n normalizeComponentTree,\n toScreenDefinition,\n type RawApiScreen,\n} from \"./screen-transforms\";\n\nexport {\n toNavigationItem,\n type RawApiNavigationItem,\n} from \"./navigation-transforms\";\n\nimport type { RepAppData, RepAppManifest } from \"@fluid-app/portal-core/types\";\nimport {\n transformThemes,\n getActiveThemeId,\n type RawApiTheme,\n} from \"@fluid-app/portal-core/theme\";\nimport { toScreenDefinition, type RawApiScreen } from \"./screen-transforms\";\nimport {\n toNavigationItem,\n type RawApiNavigationItem,\n} from \"./navigation-transforms\";\n\n/** Raw manifest shape from the `/fluid_os/definitions/active` API response */\nexport interface RawManifestResponse {\n manifest: {\n definition_id: number;\n published_version?: number;\n screens?: RawApiScreen[];\n profile?: {\n name?: string;\n definition_id: number;\n themes?: RawApiTheme[];\n navigation?: {\n id: number;\n name?: string;\n definition_id: number;\n navigation_items?: RawApiNavigationItem[];\n };\n mobile_navigation?: {\n id: number;\n name?: string;\n definition_id: number;\n navigation_items?: RawApiNavigationItem[];\n };\n };\n };\n}\n\n/**\n * Convert a raw FluidOS API response to RawManifestResponse.\n *\n * The wire format includes `navigation_items` and full theme objects,\n * but the generated OpenAPI types differ (e.g. `navigation_tree`, `themes: number[]`).\n * This function bridges that gap so callers avoid `as unknown as` casts.\n */\nexport function toRawManifest<T extends { manifest?: object }>(\n raw: T,\n): RawManifestResponse {\n if (!raw.manifest) {\n throw new Error(\"FluidOS API returned empty manifest\");\n }\n // Safe: the wire format matches RawManifestResponse; the generated\n // types just model the schema less precisely for this endpoint.\n return raw as unknown as RawManifestResponse;\n}\n\n/**\n * Transform a raw FluidOS manifest API response into RepAppData.\n *\n * This is the top-level transform used by `FluidClient.app.get()`.\n * It handles:\n * - Theme transformation (legacy and new formats)\n * - Screen normalization (component_tree array wrapping)\n * - Navigation item transformation (recursive with position sorting)\n */\nexport function transformManifestToRepAppData(\n response: RawManifestResponse,\n): RepAppData {\n const manifest = response.manifest;\n const rawProfile = manifest.profile;\n\n const rawThemes: RawApiTheme[] = Array.isArray(rawProfile?.themes)\n ? rawProfile.themes\n : [];\n\n const screens = (manifest.screens ?? []).map((screen) =>\n toScreenDefinition(screen),\n );\n\n const navigationItems = (rawProfile?.navigation?.navigation_items ?? []).map(\n toNavigationItem,\n );\n\n const nav = rawProfile?.navigation;\n const mobileNav = rawProfile?.mobile_navigation;\n const mobileNavigationItems = (mobileNav?.navigation_items ?? []).map(\n toNavigationItem,\n );\n const activeThemeId = getActiveThemeId(rawThemes);\n\n return {\n definition_id: manifest.definition_id,\n published_version: manifest.published_version ?? 0,\n screens,\n profile: {\n name: rawProfile?.name ?? \"Default\",\n definition_id: rawProfile?.definition_id ?? manifest.definition_id,\n themes: transformThemes(rawThemes),\n // Conditional spread for exactOptionalPropertyTypes compliance\n ...(activeThemeId !== undefined ? { activeThemeId } : {}),\n navigation: {\n definition_id: nav?.definition_id ?? manifest.definition_id,\n id: nav?.id ?? 0,\n name: nav?.name ?? \"Main Navigation\",\n navigation_items: navigationItems,\n screens,\n },\n ...(mobileNav\n ? {\n mobile_navigation: {\n definition_id: mobileNav.definition_id ?? manifest.definition_id,\n id: mobileNav.id ?? 0,\n name: mobileNav.name ?? \"Mobile Navigation\",\n navigation_items: mobileNavigationItems,\n screens,\n },\n }\n : {}),\n },\n };\n}\n\n/**\n * Transform a raw manifest response, unwrapping the `{ manifest }` envelope.\n * Convenience wrapper matching the `RepAppManifest` type.\n */\nexport function transformRawManifest(\n response: RawManifestResponse,\n): RepAppManifest {\n return {\n manifest: transformManifestToRepAppData(response),\n };\n}\n","/**\n * Fluid API Client\n * Adapted from: packages/fluidos-api-client/src/lib/fetch-client.ts\n * Provides authenticated API access with domain-specific methods\n */\n\nimport type { Rep, UpdateRepData } from \"../types/rep\";\nimport type { UserPermissions } from \"../types/permissions\";\nimport type { RepAppData } from \"@fluid-app/portal-core/types\";\nimport {\n createFetchClient,\n type FetchClient,\n} from \"@fluid-app/api-client-core\";\nimport { fluidOs } from \"@fluid-app/fluidos-api-client\";\nimport {\n HTTP_METHODS,\n type FluidSDKConfig,\n type RequestOptions,\n} from \"./types\";\nimport {\n transformManifestToRepAppData,\n toRawManifest,\n type RawManifestResponse,\n} from \"../transforms\";\n\n/**\n * API Error class for structured error handling\n */\nexport class ApiError extends Error {\n readonly status: number;\n readonly data: unknown;\n\n constructor(message: string, status: number, data?: unknown) {\n super(message);\n this.name = \"ApiError\";\n this.status = status;\n this.data = data;\n\n // V8-specific stack trace capture (Node.js, Chrome)\n const errorWithCapture = Error as typeof Error & {\n captureStackTrace?: (\n target: object,\n constructor: typeof ApiError,\n ) => void;\n };\n if (errorWithCapture.captureStackTrace) {\n errorWithCapture.captureStackTrace(this, ApiError);\n }\n }\n\n toJSON(): { name: string; message: string; status: number; data: unknown } {\n return {\n name: this.name,\n message: this.message,\n status: this.status,\n data: this.data,\n };\n }\n}\n\n/**\n * Type guard for ApiError\n */\nexport function isApiError(error: unknown): error is ApiError {\n return error instanceof ApiError;\n}\n\n// ============================================================================\n// Discriminated Union for API Responses\n// ============================================================================\n\n/**\n * Discriminated union representing the result of an API call.\n * Use `isApiSuccess` and `isApiFailure` type guards to narrow.\n */\nexport type ApiResult<T> =\n | { readonly success: true; readonly data: T }\n | { readonly success: false; readonly error: ApiError };\n\n/**\n * Type guard for successful API result\n */\nexport function isApiSuccess<T>(\n result: ApiResult<T>,\n): result is { readonly success: true; readonly data: T } {\n return result.success === true;\n}\n\n/**\n * Type guard for failed API result\n */\nexport function isApiFailure<T>(\n result: ApiResult<T>,\n): result is { readonly success: false; readonly error: ApiError } {\n return result.success === false;\n}\n\n/**\n * Type guard to check if a value is a non-null string\n */\nfunction isString(value: unknown): value is string {\n return typeof value === \"string\";\n}\n\n/**\n * Extract error message from API response data using `in` operator narrowing.\n * Checks common error message field names in order of precedence.\n */\nfunction extractErrorMessage(\n data: Readonly<Record<string, unknown>>,\n fallback: string,\n): string {\n // Use `in` operator to narrow and then type guard to validate string type\n if (\"message\" in data && isString(data.message)) {\n return data.message;\n }\n if (\"error_message\" in data && isString(data.error_message)) {\n return data.error_message;\n }\n if (\"error\" in data && isString(data.error)) {\n return data.error;\n }\n return fallback;\n}\n\n// ============================================================================\n// Response Envelope\n// ============================================================================\n\n/**\n * Standard envelope returned by all portal tenant API responses.\n * The `data` key holds the actual resource(s).\n */\ninterface ApiEnvelope<T = unknown> {\n readonly status: number;\n readonly data: T;\n readonly meta: EnvelopeMeta;\n}\n\n/**\n * Metadata included in every API envelope response.\n */\nexport interface EnvelopeMeta {\n readonly request_id: string | null;\n readonly timestamp: string;\n readonly pagination?: CursorPagination;\n}\n\n/**\n * Cursor-based pagination metadata returned in the envelope `meta`.\n */\nexport interface CursorPagination {\n readonly cursor: string | null;\n readonly limit: number;\n readonly next_cursor: string | null;\n readonly prev_cursor: string | null;\n}\n\n/**\n * Result type for paginated list endpoints.\n * Includes both the unwrapped data and cursor-based pagination metadata.\n */\nexport interface PaginatedResult<T> {\n readonly data: T;\n readonly pagination: CursorPagination | undefined;\n}\n\n/**\n * Type guard to detect whether a parsed JSON value is an API envelope.\n * Envelopes always have numeric `status` and a `data` key.\n */\nfunction isApiEnvelope(value: unknown): value is ApiEnvelope {\n return (\n typeof value === \"object\" &&\n value !== null &&\n \"status\" in value &&\n typeof (value as Record<string, unknown>).status === \"number\" &&\n \"data\" in value\n );\n}\n\n// ============================================================================\n// Analytics Types\n// ============================================================================\n\n/** Minimal order shape returned by the analytics dashboard endpoint. */\nexport interface DashboardOrder {\n readonly id: string;\n readonly order_number: string;\n readonly status: string;\n readonly total: number;\n readonly created_at: string;\n}\n\nexport interface DashboardData {\n readonly total_sales: number;\n readonly total_orders: number;\n readonly total_customers: number;\n readonly recent_orders: readonly DashboardOrder[];\n}\n\nexport interface SalesParams {\n readonly date_from?: string;\n readonly date_to?: string;\n readonly group_by?: \"day\" | \"week\" | \"month\";\n}\n\nexport interface SalesData {\n readonly total: number;\n readonly data: readonly SalesDataPoint[];\n}\n\nexport interface SalesDataPoint {\n readonly date: string;\n readonly amount: number;\n readonly orders: number;\n}\n\n// ============================================================================\n// Client Return Type\n// ============================================================================\n\n/**\n * The shape returned by `createFluidClient`.\n * Generic helper methods preserve their type parameters so callers\n * can specify response types at each call site.\n */\nexport interface FluidClientReturn {\n readonly fetchClient: FetchClient;\n\n readonly request: <TResponse = unknown>(\n endpoint: string,\n options?: RequestOptions,\n ) => Promise<TResponse>;\n\n readonly requestNullable: <TResponse>(\n endpoint: string,\n options?: RequestOptions,\n ) => Promise<TResponse | null>;\n\n readonly safeRequest: <TResponse>(\n endpoint: string,\n options?: RequestOptions,\n ) => Promise<ApiResult<TResponse>>;\n\n readonly get: <\n TResponse = unknown,\n TParams extends object = Record<string, unknown>,\n >(\n endpoint: string,\n params?: TParams,\n options?: Omit<RequestOptions, \"method\" | \"params\">,\n ) => Promise<TResponse>;\n\n readonly post: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ) => Promise<TResponse>;\n\n readonly put: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ) => Promise<TResponse>;\n\n readonly patch: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ) => Promise<TResponse>;\n\n readonly delete: <TResponse = unknown>(\n endpoint: string,\n options?: Omit<RequestOptions, \"method\">,\n ) => Promise<TResponse>;\n\n // TODO(portal-tenant): migrate to portal-tenant once BFF endpoint available\n readonly reps: {\n readonly current: () => Promise<Rep>;\n readonly updateProfile: (data: UpdateRepData) => Promise<Rep>;\n };\n\n readonly app: {\n readonly getRaw: () => Promise<RawManifestResponse>;\n readonly get: () => Promise<RepAppData>;\n };\n\n readonly permissions: {\n readonly get: () => Promise<UserPermissions>;\n };\n\n readonly analytics: {\n readonly dashboard: () => Promise<DashboardData>;\n readonly sales: (params?: SalesParams) => Promise<SalesData>;\n };\n}\n\n// ============================================================================\n// Client Implementation\n// ============================================================================\n\n/**\n * Creates a configured Fluid API client instance\n */\nexport function createFluidClient(config: FluidSDKConfig): FluidClientReturn {\n const { baseUrl, getAuthToken, onAuthError, defaultHeaders = {} } = config;\n\n // Create a fluidos FetchClient for structured API calls (manifest, etc.)\n // Conditional spread for exactOptionalPropertyTypes: only include getAuthToken when defined.\n // getAuthToken is still forwarded to the FetchClient for backward compatibility\n // with consumers that provide it (e.g., for non-BFF deployments).\n const fetchClient: FetchClient = createFetchClient({\n baseUrl,\n ...(getAuthToken ? { getAuthToken } : {}),\n onAuthError,\n defaultHeaders,\n credentials: \"include\",\n });\n\n /**\n * Build headers for a request.\n * Auth is handled by session cookies via `credentials: 'include'` on fetch calls.\n */\n function buildHeaders(\n customHeaders?: Record<string, string>,\n ): Record<string, string> {\n return {\n \"Content-Type\": \"application/json\",\n ...defaultHeaders,\n ...customHeaders,\n };\n }\n\n /**\n * Build URL with query parameters (Rails-compatible)\n */\n function buildUrl(\n endpoint: string,\n params?: Record<string, unknown>,\n ): string {\n // Construct URL by concatenating baseUrl + endpoint\n // Using URL constructor with relative paths would strip the baseUrl path (e.g., /api)\n const normalizedBase = baseUrl.endsWith(\"/\")\n ? baseUrl.slice(0, -1)\n : baseUrl;\n const normalizedEndpoint = endpoint.startsWith(\"/\")\n ? endpoint\n : `/${endpoint}`;\n\n // When baseUrl is empty (same-origin), use window.location.origin as the\n // URL base so that `new URL()` can resolve the relative path correctly.\n const url = normalizedBase\n ? new URL(normalizedBase + normalizedEndpoint)\n : new URL(\n normalizedEndpoint,\n typeof window !== \"undefined\"\n ? window.location.origin\n : \"http://localhost\",\n );\n\n if (params) {\n for (const [key, value] of Object.entries(params)) {\n if (value === undefined || value === null) {\n continue;\n }\n\n if (Array.isArray(value)) {\n // Handle arrays like Rails expects: key[]\n for (const item of value) {\n url.searchParams.append(`${key}[]`, String(item));\n }\n } else if (typeof value === \"object\") {\n // Handle nested objects: key[subkey]\n for (const [subKey, subValue] of Object.entries(\n value as Record<string, unknown>,\n )) {\n if (subValue === undefined || subValue === null) {\n continue;\n }\n\n if (Array.isArray(subValue)) {\n for (const item of subValue) {\n url.searchParams.append(`${key}[${subKey}][]`, String(item));\n }\n } else {\n url.searchParams.append(`${key}[${subKey}]`, String(subValue));\n }\n }\n } else {\n url.searchParams.append(key, String(value));\n }\n }\n }\n\n return url.toString();\n }\n\n /**\n * Default request options for type-safe defaults.\n * Uses `satisfies` to validate against RequestOptions while preserving literal types.\n */\n const defaultRequestOptions = {\n method: HTTP_METHODS.GET,\n } as const satisfies Partial<RequestOptions>;\n\n /**\n * Main request function\n */\n async function request<TResponse = unknown>(\n endpoint: string,\n options: RequestOptions = {},\n ): Promise<TResponse> {\n const {\n method = defaultRequestOptions.method,\n headers: customHeaders,\n params,\n body,\n signal,\n } = options;\n\n // Use buildUrl for all requests to ensure baseUrl path is preserved\n const url = buildUrl(\n endpoint,\n method === HTTP_METHODS.GET ? params : undefined,\n );\n\n const headers = buildHeaders(customHeaders);\n\n let response: Response;\n\n try {\n const fetchOptions: RequestInit = {\n method,\n headers,\n credentials: \"include\",\n };\n // Only add signal and body when defined to satisfy exactOptionalPropertyTypes\n // RequestInit expects signal: AbortSignal | null, not undefined\n if (signal !== undefined) {\n fetchOptions.signal = signal;\n }\n if (body && method !== HTTP_METHODS.GET) {\n fetchOptions.body = JSON.stringify(body);\n }\n response = await fetch(url, fetchOptions);\n } catch (networkError) {\n throw new ApiError(\n `Network error: ${networkError instanceof Error ? networkError.message : \"Unknown network error\"}`,\n 0,\n null,\n );\n }\n\n // Handle authentication errors — invoke the handler, then throw so\n // TanStack Query surfaces the error instead of caching `null` as data.\n if (response.status === 401) {\n onAuthError?.();\n throw new ApiError(\"Authentication required\", 401, null);\n }\n\n if (!response.ok) {\n try {\n const contentType = response.headers.get(\"content-type\");\n if (contentType?.includes(\"application/json\")) {\n const data = (await response.json()) as Record<string, unknown>;\n // Use `in` operator narrowing to safely extract error message\n const errorMessage = extractErrorMessage(\n data,\n `${method} request failed`,\n );\n throw new ApiError(\n errorMessage,\n response.status,\n \"errors\" in data ? data.errors : data,\n );\n } else {\n throw new ApiError(\n `${method} request failed with status ${response.status}`,\n response.status,\n null,\n );\n }\n } catch (error) {\n if (isApiError(error)) {\n throw error;\n }\n\n throw new ApiError(\n `${method} request failed with status ${response.status}`,\n response.status,\n null,\n );\n }\n }\n\n // Handle empty responses (204 No Content)\n // Note: Callers expecting nullable responses should use requestNullable<T>\n // which properly types the return as T | null\n if (\n response.status === 204 ||\n response.headers.get(\"content-length\") === \"0\"\n ) {\n // Type assertion required: 204 No Content has no body to parse.\n // This is safe when TResponse is a union including null (e.g., T | null).\n // For fully type-safe nullable handling, prefer requestNullable() or safeRequest().\n return null as TResponse;\n }\n\n try {\n const raw: unknown = await response.json();\n // Runtime check: verify we got an object/array, not a primitive that would be mistyped\n if (raw === null || raw === undefined) {\n throw new ApiError(\n \"Unexpected null/undefined in JSON response\",\n response.status,\n null,\n );\n }\n // Automatically unwrap the standard `{ status, data, meta }` envelope\n // used by portal tenant API responses. Non-envelope responses (e.g.\n // legacy or third-party endpoints) pass through unchanged.\n const unwrapped = isApiEnvelope(raw) ? raw.data : raw;\n // Type assertion required: JSON.parse returns `unknown`, but we've validated\n // the response is not null/undefined. The caller specifies TResponse based on\n // their knowledge of the API endpoint's response schema.\n return unwrapped as TResponse;\n } catch (parseError) {\n if (isApiError(parseError)) {\n throw parseError;\n }\n throw new ApiError(\n \"Failed to parse response as JSON\",\n response.status,\n null,\n );\n }\n }\n\n /**\n * Request function for endpoints that may return null (204 No Content).\n * Properly types the return as T | null.\n */\n async function requestNullable<TResponse>(\n endpoint: string,\n options: RequestOptions = {},\n ): Promise<TResponse | null> {\n return request<TResponse | null>(endpoint, options);\n }\n\n /**\n * Safe request wrapper that returns a discriminated union instead of throwing.\n * Use `isApiSuccess` or `isApiFailure` to narrow the result.\n */\n async function safeRequest<TResponse>(\n endpoint: string,\n options: RequestOptions = {},\n ): Promise<ApiResult<TResponse>> {\n try {\n const data = await request<TResponse>(endpoint, options);\n return { success: true, data };\n } catch (error) {\n if (isApiError(error)) {\n return { success: false, error };\n }\n // Wrap unknown errors in ApiError\n return {\n success: false,\n error: new ApiError(\n error instanceof Error ? error.message : \"Unknown error\",\n 0,\n null,\n ),\n };\n }\n }\n\n /**\n * Helper to safely convert typed params to Record<string, unknown>.\n * Type assertion required: TypeScript's structural typing allows any object\n * to be treated as Record<string, unknown> when we only need to iterate\n * over its entries. This is safe because buildUrl only reads properties.\n */\n function toParams<T extends object>(\n params: T | undefined,\n ): Record<string, unknown> | undefined {\n return params as Record<string, unknown> | undefined;\n }\n\n // Convenience HTTP methods using HTTP_METHODS constant for type safety\n const get = <\n TResponse = unknown,\n TParams extends object = Record<string, unknown>,\n >(\n endpoint: string,\n params?: TParams,\n options?: Omit<RequestOptions, \"method\" | \"params\">,\n ): Promise<TResponse> => {\n // Build request options object, only adding params if defined\n const baseOptions = {\n ...options,\n method: HTTP_METHODS.GET,\n } satisfies RequestOptions;\n\n const convertedParams = toParams(params);\n const requestOptions: RequestOptions =\n convertedParams !== undefined\n ? { ...baseOptions, params: convertedParams }\n : baseOptions;\n\n return request<TResponse>(endpoint, requestOptions);\n };\n\n const post = <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: HTTP_METHODS.POST,\n body,\n } satisfies RequestOptions);\n\n const put = <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: HTTP_METHODS.PUT,\n body,\n } satisfies RequestOptions);\n\n const patch = <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: HTTP_METHODS.PATCH,\n body,\n } satisfies RequestOptions);\n\n const del = <TResponse = unknown>(\n endpoint: string,\n options?: Omit<RequestOptions, \"method\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: HTTP_METHODS.DELETE,\n } satisfies RequestOptions);\n\n // ============================================================================\n // Domain-Specific API Methods\n // ============================================================================\n\n return {\n // Fluidos FetchClient for structured API calls\n fetchClient: fetchClient,\n\n // Low-level methods for custom endpoints\n request: request,\n requestNullable: requestNullable,\n safeRequest: safeRequest,\n get: get,\n post: post,\n put: put,\n patch: patch,\n delete: del,\n\n // TODO(portal-tenant): migrate to portal-tenant once BFF endpoint available\n reps: {\n current: (): Promise<Rep> => get<Rep>(\"/api/reps/me\"),\n updateProfile: (data: UpdateRepData): Promise<Rep> =>\n patch<Rep>(\"/api/reps/me\", data),\n },\n\n // TODO(portal-tenant): replace with BFF app definition once manifest parity achieved\n app: {\n /** Fetch the raw manifest (plain JSON, no Color objects). Cache-safe. */\n getRaw: async (): Promise<RawManifestResponse> => {\n const raw = await fluidOs.getFluidOSManifest(fetchClient, {\n platform: \"browser\",\n });\n return toRawManifest(raw);\n },\n /** Fetch the active app definition with full theme/screen/navigation transforms */\n get: async (): Promise<RepAppData> => {\n const raw = await fluidOs.getFluidOSManifest(fetchClient, {\n platform: \"browser\",\n });\n return transformManifestToRepAppData(toRawManifest(raw));\n },\n },\n\n // TODO(portal-tenant): migrate to portal-tenant once BFF endpoint available\n permissions: {\n get: (): Promise<UserPermissions> =>\n get<UserPermissions>(\"/api/company/roles/my_permissions\"),\n },\n\n // TODO(portal-tenant): migrate to portal-tenant once BFF endpoint available\n analytics: {\n dashboard: (): Promise<DashboardData> =>\n get<DashboardData>(\"/api/analytics/dashboard\"),\n sales: (params?: SalesParams): Promise<SalesData> =>\n get<SalesData, SalesParams>(\"/api/analytics/sales\", params),\n },\n };\n}\n\nexport type FluidClient = ReturnType<typeof createFluidClient>;\n","import type { FluidOsReadApi } from \"@fluid-app/portal-core/fluidos-api\";\nimport type { FetchClient } from \"@fluid-app/api-client-core\";\nimport { fluidOs } from \"@fluid-app/fluidos-api-client\";\n\n/** Create a FluidOsReadApi adapter backed by a FetchClient. */\nexport function createFluidOsReadAdapter(client: FetchClient): FluidOsReadApi {\n return {\n getManifest: (params) => fluidOs.getFluidOSManifest(client, params),\n };\n}\n","import type { AppDefinitionApi } from \"@fluid-app/portal-core/app-definition-api\";\nimport type {\n AppDefinitionResponse,\n AppFluidOsDefinition,\n} from \"@fluid-app/portal-core/app-definition-types\";\nimport type { FetchClient } from \"@fluid-app/portal-tenant-api-client\";\nimport { portalTenant } from \"@fluid-app/portal-tenant-api-client\";\n\nfunction mapDefinition(\n raw: NonNullable<\n Awaited<ReturnType<typeof portalTenant.app_definition_show>>[\"definition\"]\n >,\n): AppFluidOsDefinition {\n return {\n id: raw.id ?? 0,\n name: raw.name ?? \"\",\n version: raw.version ?? null,\n components: raw.components ?? [],\n active: raw.active ?? false,\n } satisfies AppFluidOsDefinition;\n}\n\n/**\n * Creates an AppDefinitionApi adapter backed by the portal-tenant BFF client.\n *\n * Maps the generated portal-tenant `app_definition_show` response to the\n * `AppDefinitionApi` port, applying runtime defaults for optional BFF fields\n * so TypeScript catches schema drift at compile time via `satisfies`.\n */\nexport function createAppDefinitionApiAdapter(\n client: FetchClient,\n): AppDefinitionApi {\n return {\n fetchDefinition: async () => {\n const response = await portalTenant.app_definition_show(client);\n return {\n definition: mapDefinition(response.definition ?? {}),\n meta: {\n request_id: response.meta?.request_id ?? \"\",\n timestamp: response.meta?.timestamp ?? \"\",\n },\n } satisfies AppDefinitionResponse;\n },\n };\n}\n","import type { AccountApi } from \"@fluid-app/portal-core/account-api\";\nimport type {\n AccountMemberType,\n AccountRep,\n AccountResponse,\n UpdateAccountBody,\n} from \"@fluid-app/portal-core/account-types\";\nimport type { FetchClient } from \"@fluid-app/portal-tenant-api-client\";\nimport { portalTenant } from \"@fluid-app/portal-tenant-api-client\";\n\nfunction mapAccount(\n raw: NonNullable<\n Awaited<ReturnType<typeof portalTenant.account_show>>[\"account\"]\n >,\n): AccountRep {\n return {\n id: raw.id ?? 0,\n member_type: (raw.member_type ?? \"rep\") satisfies AccountMemberType,\n first_name: raw.first_name ?? \"\",\n last_name: raw.last_name ?? \"\",\n email: raw.email ?? \"\",\n phone: raw.phone ?? null,\n bio: raw.bio ?? null,\n avatar_url: raw.avatar_url ?? null,\n slug: raw.slug ?? \"\",\n social_links: (raw.social_links ?? null) as Record<string, string> | null,\n } satisfies AccountRep;\n}\n\nexport function createAccountApiAdapter(client: FetchClient): AccountApi {\n return {\n fetchAccount: async () => {\n const response = await portalTenant.account_show(client);\n return {\n account: mapAccount(response.account ?? {}),\n meta: {\n request_id: response.meta?.request_id ?? \"\",\n timestamp: response.meta?.timestamp ?? \"\",\n },\n } satisfies AccountResponse;\n },\n\n updateAccount: async (body: UpdateAccountBody) => {\n const response = await portalTenant.account_update(client, body);\n return {\n account: mapAccount(response.account ?? {}),\n meta: {\n request_id: response.meta?.request_id ?? \"\",\n timestamp: response.meta?.timestamp ?? \"\",\n },\n } satisfies AccountResponse;\n },\n };\n}\n","/**\n * Generated API client functions for portal_tenant_pay\n *\n * DO NOT EDIT THIS FILE DIRECTLY\n * This file is auto-generated. To update:\n * 1. Update the OpenAPI spec file\n * 2. Run: pnpm generate\n */\n\nimport type { FetchClient } from \"../lib/fetch-client\";\nimport type { operations } from \"../generated/portal-tenant-pay\";\n\n// ============================================================================\n// addresses\n// ============================================================================\n\n/**\n * List addresses\n * Returns addresses associated with the member's customer record in this tenant.\n *\n * @param client - Fetch client instance\n * @param [params] - params\n */\nexport async function addresses_list(\n client: FetchClient,\n params?: operations[\"addresses_list\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"addresses_list\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/pay/addresses`, params);\n}\n\n/**\n * Create an address\n * Adds a new address to the member's customer record. If an identical address already exists it is returned instead of creating a duplicate.\n *\n * @param client - Fetch client instance\n * @param body - body\n */\nexport async function addresses_create(\n client: FetchClient,\n body: operations[\"addresses_create\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"addresses_create\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.post(`/api/pay/addresses`, body);\n}\n\n/**\n * Update an address\n * Creates a new address with the merged attributes and discards the old one, preserving references from existing orders.\n *\n * @param client - Fetch client instance\n * @param id - id\n * @param body - body\n */\nexport async function addresses_update(\n client: FetchClient,\n id: string | number,\n body: operations[\"addresses_update\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"addresses_update\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.patch(`/api/pay/addresses/${id}`, body);\n}\n\n/**\n * Delete an address\n * Removes an address from the member's customer record. The default address cannot be removed.\n *\n * @param client - Fetch client instance\n * @param id - id\n */\nexport async function addresses_destroy(\n client: FetchClient,\n id: string | number,\n): Promise<\n operations[\"addresses_destroy\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.delete(`/api/pay/addresses/${id}`);\n}\n\n// ============================================================================\n// payment-methods\n// ============================================================================\n\n/**\n * List payment methods\n * Returns displayable payment methods on the member's customer record, excluding Apple Pay sources.\n *\n * @param client - Fetch client instance\n * @param [params] - params\n */\nexport async function payment_methods_list(\n client: FetchClient,\n params?: operations[\"payment_methods_list\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"payment_methods_list\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/pay/payment_methods`, params);\n}\n\n/**\n * Create a payment method\n * Tokenizes and stores a new payment method via the vault provider. Requires a vault token obtained from the vault credentials endpoint.\n *\n * @param client - Fetch client instance\n * @param body - body\n */\nexport async function payment_methods_create(\n client: FetchClient,\n body: operations[\"payment_methods_create\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"payment_methods_create\"][\"responses\"][201][\"content\"][\"application/json\"]\n> {\n return client.post(`/api/pay/payment_methods`, body);\n}\n\n/**\n * Update a payment method\n * Updates a payment method's attributes. Currently supports setting a payment method as the default.\n *\n * @param client - Fetch client instance\n * @param id - id\n * @param body - body\n */\nexport async function payment_methods_update(\n client: FetchClient,\n id: string | number,\n body: operations[\"payment_methods_update\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"payment_methods_update\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.patch(`/api/pay/payment_methods/${id}`, body);\n}\n\n/**\n * Delete a payment method\n * Removes a payment method from the member's customer record. If the removed method was the default, the default is cleared.\n *\n * @param client - Fetch client instance\n * @param id - id\n */\nexport async function payment_methods_destroy(\n client: FetchClient,\n id: string | number,\n): Promise<\n operations[\"payment_methods_destroy\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.delete(`/api/pay/payment_methods/${id}`);\n}\n\n/**\n * Get vault credentials\n * Returns a short-lived vault token and environment identifier for initializing the client-side payment vault SDK.\n *\n * @param client - Fetch client instance\n \n */\nexport async function payment_methods_vault_show(\n client: FetchClient,\n): Promise<\n operations[\"payment_methods_vault_show\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/pay/payment_methods/vault`);\n}\n\n// ============================================================================\n// points-ledgers\n// ============================================================================\n\n/**\n * List points ledger entries\n * Returns loyalty points ledger entries for the member's customer record in this tenant, ordered by most recent first.\n *\n * @param client - Fetch client instance\n * @param [params] - params\n */\nexport async function points_ledgers_list(\n client: FetchClient,\n params?: operations[\"points_ledgers_list\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"points_ledgers_list\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/pay/points_ledgers`, params);\n}\n","import type { PayApi } from \"@fluid-app/portal-core/pay-api\";\nimport type {\n PayAddress,\n PayPaymentMethod,\n CreateAddressBody,\n CreatePaymentMethodBody,\n UpdatePaymentMethodBody,\n VaultResponse,\n PointsLedgersResponse,\n} from \"@fluid-app/portal-core/pay-types\";\nimport type { FetchClient } from \"@fluid-app/portal-tenant-api-client\";\nimport { portalTenantPay } from \"@fluid-app/portal-tenant-pay-api-client\";\n\n// ---------------------------------------------------------------------------\n// BFF response type helpers\n// ---------------------------------------------------------------------------\n\ntype BffAddress = NonNullable<\n Awaited<ReturnType<typeof portalTenantPay.addresses_list>>[\"addresses\"]\n>[number];\n\ntype BffPaymentMethod = NonNullable<\n Awaited<\n ReturnType<typeof portalTenantPay.payment_methods_list>\n >[\"payment_methods\"]\n>[number];\n\n// ---------------------------------------------------------------------------\n// BFF → port type mappers\n// ---------------------------------------------------------------------------\n\nfunction mapAddress(raw: BffAddress): PayAddress {\n return {\n id: raw.id ?? 0,\n street1: raw.street1 ?? \"\",\n street2: raw.street2 ?? null,\n city: raw.city ?? \"\",\n state: raw.state ?? \"\",\n zip: raw.zip ?? \"\",\n country: raw.country ?? \"\",\n default: raw.default ?? false,\n created_at: raw.created_at ?? null,\n updated_at: raw.updated_at ?? null,\n };\n}\n\nfunction mapPaymentMethod(raw: BffPaymentMethod): PayPaymentMethod {\n return {\n id: raw.id ?? 0,\n type: raw.type ?? \"card\",\n brand: raw.brand ?? null,\n last_four: raw.last_four ?? \"\",\n exp_month: raw.exp_month ?? null,\n exp_year: raw.exp_year ?? null,\n default: raw.default ?? false,\n created_at: raw.created_at ?? null,\n updated_at: raw.updated_at ?? null,\n };\n}\n\n// ---------------------------------------------------------------------------\n// Adapter factory\n// ---------------------------------------------------------------------------\n\nexport function createPortalTenantPayAdapter(client: FetchClient): PayApi {\n return {\n fetchAddresses: async () => {\n const response = await portalTenantPay.addresses_list(client);\n return {\n addresses: (response.addresses ?? []).map(mapAddress),\n meta: {\n request_id: response.meta?.request_id ?? \"\",\n timestamp: response.meta?.timestamp ?? \"\",\n },\n };\n },\n\n createAddress: async (body: CreateAddressBody) => {\n await portalTenantPay.addresses_create(client, body);\n },\n\n updateAddress: async (addressId: number, body: CreateAddressBody) => {\n await portalTenantPay.addresses_update(client, addressId, body);\n },\n\n deleteAddress: async (addressId: number) => {\n await portalTenantPay.addresses_destroy(client, addressId);\n },\n\n fetchPaymentMethods: async () => {\n const response = await portalTenantPay.payment_methods_list(client);\n return {\n payment_methods: (response.payment_methods ?? []).map(mapPaymentMethod),\n meta: {\n request_id: response.meta?.request_id ?? \"\",\n timestamp: response.meta?.timestamp ?? \"\",\n },\n };\n },\n\n createPaymentMethod: async (body: CreatePaymentMethodBody) => {\n await portalTenantPay.payment_methods_create(client, body);\n },\n\n updatePaymentMethod: async (\n paymentMethodId: number,\n body: UpdatePaymentMethodBody,\n ) => {\n await portalTenantPay.payment_methods_update(\n client,\n paymentMethodId,\n body,\n );\n },\n\n deletePaymentMethod: async (paymentMethodId: number) => {\n await portalTenantPay.payment_methods_destroy(client, paymentMethodId);\n },\n\n fetchVaultCredentials: async () => {\n const response = await portalTenantPay.payment_methods_vault_show(client);\n return {\n vault: {\n token: response.vault?.token ?? null,\n environment: response.vault?.environment ?? \"production\",\n },\n meta: {\n request_id: response.meta?.request_id ?? \"\",\n timestamp: response.meta?.timestamp ?? \"\",\n },\n } satisfies VaultResponse;\n },\n\n fetchPointsLedgers: async () => {\n const response = await portalTenantPay.points_ledgers_list(client);\n return {\n points_ledgers: (response.points_ledgers ?? []).map((entry) => ({\n id: entry.id ?? 0,\n amount: entry.amount ?? 0,\n total_balance: entry.total_balance ?? 0,\n metadata: (entry.metadata as Record<string, unknown>) ?? null,\n created_at: entry.created_at ?? \"\",\n })),\n meta: {\n request_id: response.meta?.request_id ?? \"\",\n timestamp: response.meta?.timestamp ?? \"\",\n },\n } satisfies PointsLedgersResponse;\n },\n };\n}\n","/**\n * Generated API client functions for portal_tenant_store\n *\n * DO NOT EDIT THIS FILE DIRECTLY\n * This file is auto-generated. To update:\n * 1. Update the OpenAPI spec file\n * 2. Run: pnpm generate\n */\n\nimport type { FetchClient } from \"../lib/fetch-client\";\nimport type { operations } from \"../generated/portal-tenant-store\";\n\n// ============================================================================\n// countries\n// ============================================================================\n\n/**\n * List available countries\n * Returns countries enabled for the tenant store, each with its ISO 3166 subdivisions (states/provinces).\n *\n * @param client - Fetch client instance\n * @param [params] - params\n */\nexport async function countries_list(\n client: FetchClient,\n params?: operations[\"countries_list\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"countries_list\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/store/countries`, params);\n}\n\n// ============================================================================\n// currencies\n// ============================================================================\n\n/**\n * List supported currencies\n * Returns the distinct set of currencies derived from the tenant company's enabled countries, sorted by name.\n *\n * @param client - Fetch client instance\n \n */\nexport async function currencies_list(\n client: FetchClient,\n): Promise<\n operations[\"currencies_list\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/store/currencies`);\n}\n\n// ============================================================================\n// languages\n// ============================================================================\n\n/**\n * List available languages\n * Returns languages enabled for the tenant store, sorted by name.\n *\n * @param client - Fetch client instance\n * @param [params] - params\n */\nexport async function languages_list(\n client: FetchClient,\n params?: operations[\"languages_list\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"languages_list\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/store/languages`, params);\n}\n","import type {\n CountriesApi,\n countries,\n} from \"@fluid-app/store-core/countries-api\";\nimport type { FetchClient } from \"./lib/fetch-client\";\nimport { portalTenantStore } from \"@fluid-app/portal-tenant-store-api-client\";\n\n/**\n * Maps a BFF state to the port's State shape.\n */\nfunction mapState(\n raw: NonNullable<\n NonNullable<\n Awaited<ReturnType<typeof portalTenantStore.countries_list>>[\"countries\"]\n >[number][\"states\"]\n >[number],\n): countries.State {\n return {\n code: raw.code ?? \"\",\n name: raw.name ?? \"\",\n };\n}\n\n/**\n * Maps a BFF country to the port's Country shape.\n */\nfunction mapCountry(\n raw: NonNullable<\n Awaited<ReturnType<typeof portalTenantStore.countries_list>>[\"countries\"]\n >[number],\n): countries.Country {\n return {\n code: raw.code ?? \"\",\n name: raw.name ?? \"\",\n currency_code: raw.currency_code ?? \"\",\n states: (raw.states ?? []).map(mapState),\n };\n}\n\n/**\n * Maps the BFF meta envelope to the port's ApiMeta shape.\n */\nfunction mapMeta(\n raw: Awaited<ReturnType<typeof portalTenantStore.countries_list>>[\"meta\"],\n): countries.CountriesListResponse[\"meta\"] {\n return {\n request_id: raw?.request_id ?? null,\n timestamp: raw?.timestamp ?? \"\",\n pagination: raw?.pagination\n ? {\n cursor: raw.pagination.cursor ?? null,\n limit: raw.pagination.limit,\n next_cursor: raw.pagination.next_cursor ?? null,\n prev_cursor: raw.pagination.prev_cursor ?? null,\n }\n : undefined,\n };\n}\n\n/**\n * Creates a CountriesApi adapter backed by the portal-tenant store BFF.\n *\n * Maps the generated portal-tenant-store namespace functions to the abstract\n * CountriesApi port, closing over the FetchClient so consumers don't need\n * to pass it per-call.\n */\nexport function createPortalTenantCountriesAdapter(\n client: FetchClient,\n): CountriesApi {\n return {\n listCountries: async (params) => {\n const response = await portalTenantStore.countries_list(client, {\n \"page[cursor]\": params?.cursor,\n \"page[limit]\": params?.limit,\n });\n return {\n countries: (response.countries ?? []).map(mapCountry),\n meta: mapMeta(response.meta),\n };\n },\n };\n}\n","import type {\n LanguagesApi,\n languages,\n} from \"@fluid-app/store-core/languages-api\";\nimport type { FetchClient } from \"./lib/fetch-client\";\nimport { portalTenantStore } from \"@fluid-app/portal-tenant-store-api-client\";\n\n/**\n * Maps a BFF language to the port's Language shape.\n */\nfunction mapLanguage(\n raw: NonNullable<\n Awaited<ReturnType<typeof portalTenantStore.languages_list>>[\"languages\"]\n >[number],\n): languages.Language {\n return {\n code: raw.code ?? \"\",\n name: raw.name ?? \"\",\n };\n}\n\n/**\n * Maps the BFF meta envelope to the port's ApiMeta shape.\n */\nfunction mapMeta(\n raw: Awaited<ReturnType<typeof portalTenantStore.languages_list>>[\"meta\"],\n): languages.LanguagesListResponse[\"meta\"] {\n return {\n request_id: raw?.request_id ?? null,\n timestamp: raw?.timestamp ?? \"\",\n pagination: raw?.pagination\n ? {\n cursor: raw.pagination.cursor ?? null,\n limit: raw.pagination.limit,\n next_cursor: raw.pagination.next_cursor ?? null,\n prev_cursor: raw.pagination.prev_cursor ?? null,\n }\n : undefined,\n };\n}\n\n/**\n * Creates a LanguagesApi adapter backed by the portal-tenant store BFF.\n *\n * Maps the generated portal-tenant-store namespace functions to the abstract\n * LanguagesApi port, closing over the FetchClient so consumers don't need\n * to pass it per-call.\n */\nexport function createPortalTenantLanguagesAdapter(\n client: FetchClient,\n): LanguagesApi {\n return {\n listLanguages: async (params) => {\n const response = await portalTenantStore.languages_list(client, {\n \"page[cursor]\": params?.cursor,\n \"page[limit]\": params?.limit,\n });\n return {\n languages: (response.languages ?? []).map(mapLanguage),\n meta: mapMeta(response.meta),\n };\n },\n };\n}\n","import type { CountriesApi } from \"@fluid-app/store-core/countries-api\";\nimport type { FetchClient } from \"@fluid-app/portal-tenant-api-client\";\nimport { createPortalTenantCountriesAdapter } from \"@fluid-app/store-api-client\";\n\n/**\n * Creates a CountriesApi adapter backed by the portal-tenant store BFF.\n *\n * Delegates to the adapter factory in `@fluid-app/store-api-client` which\n * maps the generated portal-tenant-store namespace functions to the abstract\n * CountriesApi port.\n */\nexport function createCountriesApiAdapter(client: FetchClient): CountriesApi {\n return createPortalTenantCountriesAdapter(client);\n}\n","import type { LanguagesApi } from \"@fluid-app/store-core/languages-api\";\nimport type { FetchClient } from \"@fluid-app/portal-tenant-api-client\";\nimport { createPortalTenantLanguagesAdapter } from \"@fluid-app/store-api-client\";\n\n/**\n * Creates a LanguagesApi adapter backed by the portal-tenant store BFF.\n *\n * Delegates to the adapter factory in `@fluid-app/store-api-client` which\n * maps the generated portal-tenant-store namespace functions to the abstract\n * LanguagesApi port.\n */\nexport function createLanguagesApiAdapter(client: FetchClient): LanguagesApi {\n return createPortalTenantLanguagesAdapter(client);\n}\n","import type { MySiteApi } from \"@fluid-app/mysite-core/mysite-api\";\nimport type {\n MySiteProfile,\n MySiteLink,\n MySiteFavorite,\n MySiteTheme,\n UpdateProfileBody,\n UpdateSettingsBody,\n CreateLinkBody,\n UpdateLinkBody,\n AddFavoriteBody,\n} from \"@fluid-app/mysite-core/mysite-api-types\";\nimport type { FetchClient } from \"@fluid-app/portal-tenant-api-client\";\nimport { portalTenantMysite } from \"@fluid-app/portal-tenant-mysite-api-client\";\n\n// ---------------------------------------------------------------------------\n// BFF response type helpers\n// ---------------------------------------------------------------------------\n\ntype BffProfile = NonNullable<\n Awaited<ReturnType<typeof portalTenantMysite.mysite_profile_show>>[\"profile\"]\n>;\n\ntype BffLink = NonNullable<\n Awaited<ReturnType<typeof portalTenantMysite.mysite_links_list>>[\"links\"]\n>[number];\n\ntype BffFavorite = NonNullable<\n Awaited<\n ReturnType<typeof portalTenantMysite.mysite_favorites_list>\n >[\"favorites\"]\n>[number];\n\ntype BffTheme = NonNullable<\n Awaited<ReturnType<typeof portalTenantMysite.mysite_themes_list>>[\"themes\"]\n>[number];\n\n// ---------------------------------------------------------------------------\n// BFF -> port type mappers\n// ---------------------------------------------------------------------------\n\nfunction mapProfile(raw: BffProfile): MySiteProfile {\n return {\n id: raw.id ?? 0,\n mysite_url: raw.mysite_url ?? null,\n theme_id: raw.theme_id ?? null,\n bio: raw.bio ?? null,\n avatar_url: raw.avatar_url ?? null,\n display_name: raw.display_name ?? null,\n slug: raw.slug ?? null,\n } satisfies MySiteProfile;\n}\n\nfunction mapLink(raw: BffLink): MySiteLink {\n return {\n id: raw.id ?? 0,\n url: raw.url ?? \"\",\n title: raw.title ?? \"\",\n position: raw.position ?? 0,\n } satisfies MySiteLink;\n}\n\nfunction mapFavorite(raw: BffFavorite): MySiteFavorite {\n return {\n id: raw.id ?? 0,\n product_id: raw.product_id ?? 0,\n product_name: raw.product_name ?? null,\n product_image_url: raw.product_image_url ?? null,\n position: raw.position ?? 0,\n created_at: raw.created_at ?? null,\n } satisfies MySiteFavorite;\n}\n\nfunction mapTheme(raw: BffTheme): MySiteTheme {\n return {\n id: raw.id ?? 0,\n name: raw.name ?? \"\",\n preview_url: raw.preview_url ?? null,\n } satisfies MySiteTheme;\n}\n\n// ---------------------------------------------------------------------------\n// Adapter factory\n// ---------------------------------------------------------------------------\n\nexport function createMySiteApiAdapter(client: FetchClient): MySiteApi {\n return {\n fetchProfile: async () => {\n const res = await portalTenantMysite.mysite_profile_show(client);\n return mapProfile((res.profile ?? {}) as BffProfile);\n },\n\n updateProfile: async (body: UpdateProfileBody) => {\n const res = await portalTenantMysite.mysite_profile_update(client, {\n profile: {\n display_name: body.display_name,\n bio: body.bio,\n avatar_url: body.avatar_url,\n },\n });\n return mapProfile((res.profile ?? {}) as BffProfile);\n },\n\n updateSettings: async (body: UpdateSettingsBody) => {\n await Promise.all([\n body.theme_id !== undefined\n ? portalTenantMysite.mysite_settings_update(client, {\n settings: { theme_id: body.theme_id },\n })\n : Promise.resolve(),\n body.slug !== undefined\n ? portalTenantMysite.mysite_profile_update(client, {\n profile: { slug: body.slug },\n })\n : Promise.resolve(),\n ]);\n },\n\n listLinks: async () => {\n const res = await portalTenantMysite.mysite_links_list(client);\n return (res.links ?? []).map(mapLink);\n },\n\n createLink: async (body: CreateLinkBody) => {\n const res = await portalTenantMysite.mysite_links_create(client, {\n link: { title: body.title, url: body.url },\n });\n return mapLink((res.link ?? {}) as BffLink);\n },\n\n updateLink: async (linkId: number, body: UpdateLinkBody) => {\n const res = await portalTenantMysite.mysite_links_update(client, linkId, {\n link: { title: body.title, url: body.url },\n });\n return mapLink((res.link ?? {}) as BffLink);\n },\n\n deleteLink: async (linkId: number) => {\n await portalTenantMysite.mysite_links_destroy(client, linkId);\n },\n\n reorderLinks: async (orderedIds: number[]) => {\n const res = await portalTenantMysite.mysite_links_bulk_reorder(client, {\n ordered_ids: orderedIds,\n });\n return (res.links ?? []).map(mapLink);\n },\n\n listFavorites: async () => {\n const res = await portalTenantMysite.mysite_favorites_list(client);\n return (res.favorites ?? []).map(mapFavorite);\n },\n\n addFavorite: async (body: AddFavoriteBody) => {\n const res = await portalTenantMysite.mysite_favorites_create(client, {\n favorite: { product_id: body.product_id },\n });\n return mapFavorite((res.favorite ?? {}) as BffFavorite);\n },\n\n deleteFavorite: async (favoriteId: number) => {\n await portalTenantMysite.mysite_favorites_destroy(client, favoriteId);\n },\n\n reorderFavorites: async (orderedIds: number[]) => {\n const res = await portalTenantMysite.mysite_favorites_bulk_reorder(\n client,\n { ordered_ids: orderedIds },\n );\n return (res.favorites ?? []).map(mapFavorite);\n },\n\n listThemes: async () => {\n const res = await portalTenantMysite.mysite_themes_list(client);\n return (res.themes ?? []).map(mapTheme);\n },\n };\n}\n","/**\n * Theme Provider for Fluid SDK\n * Handles CSS variable injection using portal-core's theme engine.\n * Accepts ThemeDefinition objects and resolves them into CSS via generateThemeCSS.\n */\n\nimport {\n createContext,\n useContext,\n useState,\n useEffect,\n useMemo,\n useCallback,\n type ReactNode,\n} from \"react\";\nimport type { ThemeDefinition } from \"../types/theme\";\nimport {\n resolveTheme,\n applyTheme as coreApplyTheme,\n removeTheme,\n} from \"../themes\";\n\ntype ThemeMode = \"light\" | \"dark\";\n\n/**\n * Context value for theme management.\n * All properties are readonly since context values should not be mutated by consumers.\n */\ninterface ThemeContextValue {\n /** Currently active theme definition */\n readonly currentTheme: ThemeDefinition | null;\n /** Switch to a different theme */\n readonly setTheme: (theme: ThemeDefinition) => void;\n /** Switch between light and dark mode for the current theme */\n readonly setThemeMode: (mode: ThemeMode) => void;\n /** Current theme mode */\n readonly mode: ThemeMode | undefined;\n}\n\nconst ThemeContext = createContext<ThemeContextValue | null>(null);\n\nexport interface FluidThemeProviderProps {\n children: ReactNode;\n /** Initial theme to apply */\n initialTheme?: ThemeDefinition;\n /** Container element for scoped theme application (defaults to document.documentElement) */\n container?: HTMLElement | null;\n}\n\n/**\n * Apply a theme to the DOM using the shared portal-core pipeline.\n * Also sets data attributes on the container for CSS selector targeting.\n */\nfunction applyThemeToDOM(\n theme: ThemeDefinition,\n mode: ThemeMode | undefined,\n container: HTMLElement | null,\n): void {\n const target = container ?? document.documentElement;\n\n // Use shared resolve + inject pipeline from portal-core\n const resolved = resolveTheme(theme);\n coreApplyTheme(resolved);\n\n // Set data attributes for CSS selector targeting\n target.dataset.theme = theme.id;\n if (mode) {\n target.dataset.themeMode = mode;\n } else {\n delete target.dataset.themeMode;\n }\n}\n\nexport function FluidThemeProvider({\n children,\n initialTheme,\n container,\n}: FluidThemeProviderProps): React.JSX.Element {\n const [currentTheme, setCurrentTheme] = useState<ThemeDefinition | null>(\n initialTheme ?? null,\n );\n const [mode, setMode] = useState<ThemeMode | undefined>(undefined);\n\n // Apply theme CSS when theme or mode changes\n useEffect(() => {\n if (currentTheme) {\n applyThemeToDOM(currentTheme, mode, container ?? null);\n }\n\n return () => {\n if (currentTheme) {\n removeTheme(currentTheme.id);\n }\n };\n }, [currentTheme, mode, container]);\n\n const setTheme = useCallback((theme: ThemeDefinition) => {\n setCurrentTheme(theme);\n }, []);\n\n const setThemeMode = useCallback((newMode: ThemeMode) => {\n setMode(newMode);\n }, []);\n\n const value = useMemo(\n () =>\n ({\n currentTheme,\n setTheme,\n setThemeMode,\n mode,\n }) satisfies ThemeContextValue,\n [currentTheme, setTheme, setThemeMode, mode],\n );\n\n return (\n <ThemeContext.Provider value={value}>{children}</ThemeContext.Provider>\n );\n}\n\n/**\n * Hook to access theme context\n * Must be used within a FluidThemeProvider\n */\nexport function useThemeContext(): ThemeContextValue {\n const context = useContext(ThemeContext);\n if (!context) {\n throw new Error(\"useThemeContext must be used within a FluidThemeProvider\");\n }\n return context;\n}\n","import type { AnyComponent } from \"../types/widget-schema\";\nimport type { WidgetSchema, TypedWidgetSchema } from \"../types\";\nimport type { ShareableItem } from \"../types/shareable-item\";\nimport type { WidgetManifest } from \"../registries/widget-manifest\";\n\nexport function createWidgetRegistry<\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n T extends Record<string, AnyComponent>,\n>(registry: T, plugins?: WidgetManifest[]): T {\n if (!plugins || plugins.length === 0) {\n return registry;\n }\n\n const pluginEntries = Object.fromEntries(\n plugins.map((p) => [p.type, p.component]),\n );\n\n return { ...registry, ...pluginEntries } as T;\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport function createScreen<T extends Record<string, AnyComponent>>(\n registry: T,\n widgets: TypedWidgetSchema<T>[],\n): TypedWidgetSchema<T>[] {\n // Validate that all widget types exist in the registry at runtime\n widgets.forEach((widget) => {\n if (!(widget.type in registry)) {\n throw new Error(\n `Widget type \"${String(widget.type)}\" not found in registry`,\n );\n }\n });\n return widgets;\n}\n\n// Helper to create WidgetSchema from shareable item\nexport function createWidgetFromShareable(item: ShareableItem): WidgetSchema {\n const isVideo = item.kind === \"video\" || !!item.videoUrl;\n\n if (isVideo && item.videoUrl) {\n return {\n type: \"VideoWidget\",\n props: {\n src: item.videoUrl,\n poster: item.imageUrl,\n caption: item.title,\n },\n };\n }\n\n return {\n type: \"ImageWidget\",\n props: {\n src: item.imageUrl,\n alt: item.title || \"Image\",\n objectFit: \"cover\",\n },\n };\n}\n","import type { WidgetPropertySchema } from \"@fluid-app/portal-core/registries\";\n\n// Widget Components\nexport { AlertWidget, alertWidgetPropertySchema } from \"./AlertWidget\";\nexport {\n BulletListWidget,\n bulletListWidgetPropertySchema,\n} from \"./BulletListWidget\";\nexport { CalendarWidget, calendarWidgetPropertySchema } from \"./CalendarWidget\";\nexport { CardWidget, cardWidgetPropertySchema } from \"./CardWidget\";\nexport { CarouselWidget, carouselWidgetPropertySchema } from \"./CarouselWidget\";\nexport { CatchUpWidget, catchUpWidgetPropertySchema } from \"./CatchUpWidget\";\nexport { ChartWidget, chartWidgetPropertySchema } from \"./ChartWidget\";\nexport {\n ContainerWidget,\n containerWidgetPropertySchema,\n} from \"./ContainerWidget\";\nexport { EmbedWidget, embedWidgetPropertySchema } from \"./EmbedWidget\";\nexport { ImageWidget, imageWidgetPropertySchema } from \"./ImageWidget\";\nexport { LayoutWidget, layoutWidgetPropertySchema } from \"./LayoutWidget\";\nexport { LinkWidget, linkWidgetPropertySchema } from \"./LinkWidget\";\nexport { ListWidget, listWidgetPropertySchema } from \"./ListWidget\";\nexport { MySiteWidget, mySiteWidgetPropertySchema } from \"./MySiteWidget\";\nexport { NestedWidget, nestedWidgetPropertySchema } from \"./NestedWidget\";\nexport { PointsWidget, pointsWidgetPropertySchema } from \"./PointsWidget\";\nexport {\n QuickShareWidget,\n quickShareWidgetPropertySchema,\n} from \"./QuickShareWidget\";\nexport {\n RecentActivityWidget,\n recentActivityWidgetPropertySchema,\n} from \"./RecentActivityWidget\";\nexport {\n SeparatorWidget,\n separatorWidgetPropertySchema,\n} from \"./SeparatorWidget\";\nexport { SpacerWidget, spacerWidgetPropertySchema } from \"./SpacerWidget\";\nexport { TableWidget, tableWidgetPropertySchema } from \"./TableWidget\";\nexport { TextWidget, textWidgetPropertySchema } from \"./TextWidget\";\nexport { ToDoWidget, toDoWidgetPropertySchema } from \"./ToDoWidget\";\nexport { VideoWidget, videoWidgetPropertySchema } from \"./VideoWidget\";\n\n// Re-export gapValues from core (used by widgets)\nexport { gapValues } from \"../core/fields\";\n\n// Re-export all property schemas as a collection\nexport const widgetPropertySchemas = {\n AlertWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./AlertWidget\").then((m) => m.alertWidgetPropertySchema),\n BulletListWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./BulletListWidget\").then((m) => m.bulletListWidgetPropertySchema),\n CalendarWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./CalendarWidget\").then((m) => m.calendarWidgetPropertySchema),\n CardWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./CardWidget\").then((m) => m.cardWidgetPropertySchema),\n CarouselWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./CarouselWidget\").then((m) => m.carouselWidgetPropertySchema),\n CatchUpWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./CatchUpWidget\").then((m) => m.catchUpWidgetPropertySchema),\n ChartWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./ChartWidget\").then((m) => m.chartWidgetPropertySchema),\n ContainerWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./ContainerWidget\").then((m) => m.containerWidgetPropertySchema),\n EmbedWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./EmbedWidget\").then((m) => m.embedWidgetPropertySchema),\n ImageWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./ImageWidget\").then((m) => m.imageWidgetPropertySchema),\n LayoutWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./LayoutWidget\").then((m) => m.layoutWidgetPropertySchema),\n LinkWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./LinkWidget\").then((m) => m.linkWidgetPropertySchema),\n ListWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./ListWidget\").then((m) => m.listWidgetPropertySchema),\n MySiteWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./MySiteWidget\").then((m) => m.mySiteWidgetPropertySchema),\n NestedWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./NestedWidget\").then((m) => m.nestedWidgetPropertySchema),\n PointsWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./PointsWidget\").then((m) => m.pointsWidgetPropertySchema),\n QuickShareWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./QuickShareWidget\").then((m) => m.quickShareWidgetPropertySchema),\n RecentActivityWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./RecentActivityWidget\").then(\n (m) => m.recentActivityWidgetPropertySchema,\n ),\n SeparatorWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./SeparatorWidget\").then((m) => m.separatorWidgetPropertySchema),\n SpacerWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./SpacerWidget\").then((m) => m.spacerWidgetPropertySchema),\n TableWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./TableWidget\").then((m) => m.tableWidgetPropertySchema),\n TextWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./TextWidget\").then((m) => m.textWidgetPropertySchema),\n ToDoWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./ToDoWidget\").then((m) => m.toDoWidgetPropertySchema),\n VideoWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./VideoWidget\").then((m) => m.videoWidgetPropertySchema),\n};\n","/**\n * Default widget registry for SDK apps.\n * Maps widget type names to their view-only components from portal-widgets.\n */\n\nimport type { WidgetRegistry } from \"@fluid-app/portal-core/types\";\nimport { createWidgetRegistry } from \"@fluid-app/portal-core/widget-utils\";\nimport {\n AlertWidget,\n BulletListWidget,\n CalendarWidget,\n CardWidget,\n CarouselWidget,\n CatchUpWidget,\n ChartWidget,\n ContainerWidget,\n EmbedWidget,\n ImageWidget,\n LayoutWidget,\n LinkWidget,\n ListWidget,\n MySiteWidget,\n NestedWidget,\n PointsWidget,\n QuickShareWidget,\n RecentActivityWidget,\n SeparatorWidget,\n SpacerWidget,\n TableWidget,\n TextWidget,\n ToDoWidget,\n VideoWidget,\n} from \"@fluid-app/portal-widgets/widgets\";\n\nexport const DEFAULT_SDK_WIDGET_REGISTRY: WidgetRegistry = createWidgetRegistry(\n {\n AlertWidget,\n BulletListWidget,\n CalendarWidget,\n CardWidget,\n CarouselWidget,\n CatchUpWidget,\n ChartWidget,\n ContainerWidget,\n EmbedWidget,\n ImageWidget,\n LayoutWidget,\n LinkWidget,\n ListWidget,\n MySiteWidget,\n NestedWidget,\n PointsWidget,\n QuickShareWidget,\n RecentActivityWidget,\n SeparatorWidget,\n SpacerWidget,\n TableWidget,\n TextWidget,\n ToDoWidget,\n VideoWidget,\n },\n);\n","/**\n * Main Fluid SDK Provider\n * Wraps QueryClientProvider and FluidThemeProvider\n */\n\nimport {\n createContext,\n useContext,\n useCallback,\n useMemo,\n useRef,\n type ReactNode,\n} from \"react\";\nimport { QueryClient, QueryClientProvider } from \"@tanstack/react-query\";\nimport { createPersister } from \"@fluid-app/query-persister\";\nimport { FluidOsApiProvider } from \"@fluid-app/portal-core/fluidos-api-context\";\nimport { AppDefinitionApiProvider } from \"@fluid-app/portal-core/app-definition-api-context\";\nimport { AccountApiProvider } from \"@fluid-app/portal-core/account-api-context\";\nimport { PayApiProvider } from \"@fluid-app/portal-core/pay-api-context\";\nimport { MySiteApiProvider } from \"@fluid-app/mysite-core/mysite-api-context\";\nimport { CountriesApiProvider } from \"@fluid-app/store-core/countries-api-context\";\nimport { LanguagesApiProvider } from \"@fluid-app/store-core/languages-api-context\";\nimport { createFetchClient as createPortalTenantFetchClient } from \"@fluid-app/portal-tenant-api-client\";\nimport { createFluidClient, type FluidClient } from \"../client/fluid-client\";\nimport { createFluidOsReadAdapter } from \"../adapters/fluidos-api-adapter\";\nimport { createAppDefinitionApiAdapter } from \"../adapters/app-definition-api-adapter\";\nimport { createAccountApiAdapter } from \"../adapters/account-api-adapter\";\nimport { createPortalTenantPayAdapter } from \"../adapters/pay-api-adapter\";\nimport { createCountriesApiAdapter } from \"../adapters/countries-api-adapter\";\nimport { createLanguagesApiAdapter } from \"../adapters/languages-api-adapter\";\nimport { createMySiteApiAdapter } from \"../adapters/mysite-api-adapter\";\nimport { FluidThemeProvider } from \"./FluidThemeProvider\";\nimport { PortalTenantClientProvider } from \"./PortalTenantClientProvider\";\nimport { DataSourceRegistryProvider } from \"@fluid-app/portal-react/data-sources/registry-context\";\nimport { RegistryProvider } from \"@fluid-app/portal-widgets/contexts\";\nimport { DEFAULT_SDK_WIDGET_REGISTRY } from \"../core/default-widget-registry\";\nimport type { FluidSDKConfig } from \"../client/types\";\nimport type { ThemeDefinition } from \"../types/theme\";\n\n/**\n * Context value for FluidProvider.\n * All properties are readonly since context values should not be mutated by consumers.\n */\ninterface FluidContextValue {\n /** Configured API client instance */\n readonly client: FluidClient;\n /** SDK configuration */\n readonly config: FluidSDKConfig;\n}\n\nconst FluidContext = createContext<FluidContextValue | null>(null);\n\nexport interface FluidProviderProps {\n /** SDK configuration (baseUrl, auth, etc.) */\n config: FluidSDKConfig;\n /** React children */\n children: ReactNode;\n /** Optional custom QueryClient instance */\n queryClient?: QueryClient;\n /** Optional initial theme */\n initialTheme?: ThemeDefinition;\n /** Optional container for scoped theme application */\n themeContainer?: HTMLElement | null;\n /** Optional custom widget registry (defaults to all built-in widgets) */\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n widgetRegistry?: Record<string, React.ComponentType<any>>;\n /** Dynamic variables for data source endpoint path substitution (e.g., { rep_id: \"123\" }) */\n variables?: Record<string, string>;\n}\n\n/**\n * Main provider for the Fluid Portal SDK\n *\n * @example\n * ```tsx\n * import { FluidProvider } from \"@fluid-app/portal-sdk\";\n *\n * function App() {\n * return (\n * <FluidProvider\n * config={{ baseUrl: \"\" }}\n * >\n * <YourApp />\n * </FluidProvider>\n * );\n * }\n * ```\n */\nexport function FluidProvider({\n config,\n children,\n queryClient,\n initialTheme,\n themeContainer,\n widgetRegistry,\n variables,\n}: FluidProviderProps): React.JSX.Element {\n // Create default QueryClient if none provided\n // Using lazy initialization to ensure it's only created once\n const defaultQueryClient = useMemo(() => {\n const persister =\n typeof window !== \"undefined\" && !import.meta.env?.DEV\n ? createPersister()\n : undefined;\n\n return new QueryClient({\n defaultOptions: {\n queries: {\n staleTime: 1000 * 60, // 1 minute\n retry: 1,\n persister: persister?.persisterFn,\n },\n },\n });\n }, []);\n\n // Keep latest config in a ref so client creation reads current values\n const configRef = useRef(config);\n configRef.current = config;\n\n // Recreate client only when baseUrl changes (primitive dependency).\n // Wrap callbacks in closures so they always read the latest from configRef,\n // avoiding stale captures when parent passes new getAuthToken/onAuthError.\n const client = useMemo(\n () =>\n createFluidClient({\n ...configRef.current,\n getAuthToken: () => configRef.current.getAuthToken?.() ?? null,\n onAuthError: () => configRef.current.onAuthError?.(),\n }),\n // eslint-disable-next-line react-hooks/exhaustive-deps\n [config.baseUrl],\n );\n\n // Create FluidOsReadApi adapter for dependency inversion\n const fluidOsApi = useMemo(\n () => createFluidOsReadAdapter(client.fetchClient),\n [client.fetchClient],\n );\n\n // Read the CSRF token from the <meta name=\"csrf-token\"> tag that Rails\n // embeds in the HTML. Required for non-GET requests to the BFF.\n const csrfToken =\n typeof document !== \"undefined\"\n ? document\n .querySelector('meta[name=\"csrf-token\"]')\n ?.getAttribute(\"content\")\n : null;\n\n const portalTenantClient = useMemo(\n () =>\n createPortalTenantFetchClient({\n baseUrl: configRef.current.baseUrl,\n onAuthError: () => configRef.current.onAuthError?.(),\n credentials: \"include\",\n defaultHeaders: {\n ...(csrfToken ? { \"X-CSRF-Token\": csrfToken } : {}),\n },\n }),\n // eslint-disable-next-line react-hooks/exhaustive-deps\n [config.baseUrl, csrfToken],\n );\n\n // Create AppDefinitionApi adapter for dependency inversion\n const appDefinitionApi = useMemo(\n () => createAppDefinitionApiAdapter(portalTenantClient),\n [portalTenantClient],\n );\n\n // Create AccountApi adapter for dependency inversion\n const accountApi = useMemo(\n () => createAccountApiAdapter(portalTenantClient),\n [portalTenantClient],\n );\n\n // Create PayApi adapter for dependency inversion\n const payApi = useMemo(\n () => createPortalTenantPayAdapter(portalTenantClient),\n [portalTenantClient],\n );\n\n // Create CountriesApi adapter for dependency inversion\n const countriesApi = useMemo(\n () => createCountriesApiAdapter(portalTenantClient),\n [portalTenantClient],\n );\n\n // Create LanguagesApi adapter for dependency inversion\n const languagesApi = useMemo(\n () => createLanguagesApiAdapter(portalTenantClient),\n [portalTenantClient],\n );\n\n // Create MySiteApi adapter for dependency inversion\n const mysiteApi = useMemo(\n () => createMySiteApiAdapter(portalTenantClient),\n [portalTenantClient],\n );\n\n // Context value only changes when client changes\n const contextValue = useMemo(\n () => ({ client, config: configRef.current }) satisfies FluidContextValue,\n [client],\n );\n\n // Stable callback to provide auth headers for data source fetchers.\n // Auth is handled by session cookies via `credentials: 'include'` on fetch calls.\n const getApiHeaders = useCallback((): Record<string, string> => {\n return {\n \"Content-Type\": \"application/json\",\n };\n }, []);\n\n // Data source endpoints (e.g., /company/v1/products) expect the base URL\n // to include /api, matching fluid-admin's NEXT_PUBLIC_API_URL pattern.\n const dataSourceBaseUrl = useMemo(() => {\n const base = config.baseUrl.replace(/\\/+$/, \"\");\n return base.endsWith(\"/api\") ? base : `${base}/api`;\n }, [config.baseUrl]);\n\n // Variables for data source path interpolation (e.g., { rep_id: \"123\" }).\n // Must be passed explicitly via the `variables` prop.\n const effectiveVariables = variables;\n\n // Build theme provider props conditionally to satisfy exactOptionalPropertyTypes\n // We only pass props that are defined to avoid passing `undefined` explicitly\n const themeProviderProps = {\n ...(initialTheme !== undefined && { initialTheme }),\n ...(themeContainer !== undefined && { container: themeContainer }),\n };\n\n // Use provided registry or fall back to the default SDK widget registry\n const registry = widgetRegistry ?? DEFAULT_SDK_WIDGET_REGISTRY;\n\n return (\n <QueryClientProvider client={queryClient ?? defaultQueryClient}>\n <FluidContext.Provider value={contextValue}>\n <PortalTenantClientProvider value={portalTenantClient}>\n <FluidOsApiProvider value={fluidOsApi}>\n <AppDefinitionApiProvider value={appDefinitionApi}>\n <AccountApiProvider value={accountApi}>\n <PayApiProvider value={payApi}>\n <CountriesApiProvider value={countriesApi}>\n <LanguagesApiProvider value={languagesApi}>\n <MySiteApiProvider value={mysiteApi}>\n <ProviderStack\n baseUrl={dataSourceBaseUrl}\n getApiHeaders={getApiHeaders}\n variables={effectiveVariables}\n registry={registry}\n themeProviderProps={themeProviderProps}\n >\n {children}\n </ProviderStack>\n </MySiteApiProvider>\n </LanguagesApiProvider>\n </CountriesApiProvider>\n </PayApiProvider>\n </AccountApiProvider>\n </AppDefinitionApiProvider>\n </FluidOsApiProvider>\n </PortalTenantClientProvider>\n </FluidContext.Provider>\n </QueryClientProvider>\n );\n}\n\n/**\n * Inner component that sets up data source providers and theme.\n * Must be rendered inside FluidContext.Provider + QueryClientProvider.\n */\nfunction ProviderStack({\n baseUrl,\n getApiHeaders,\n variables,\n registry,\n themeProviderProps,\n children,\n}: {\n baseUrl: string;\n getApiHeaders: () => Record<string, string>;\n variables: Record<string, string> | undefined;\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n registry: Record<string, React.ComponentType<any>>;\n themeProviderProps: Record<string, unknown>;\n children: ReactNode;\n}): React.JSX.Element {\n return (\n <DataSourceRegistryProvider\n baseUrl={baseUrl}\n getApiHeaders={getApiHeaders}\n variables={variables}\n >\n <RegistryProvider registry={registry}>\n <FluidThemeProvider {...themeProviderProps}>\n {children}\n </FluidThemeProvider>\n </RegistryProvider>\n </DataSourceRegistryProvider>\n );\n}\n\n/**\n * Hook to access the Fluid context\n * Must be used within a FluidProvider\n */\nexport function useFluidContext(): FluidContextValue {\n const context = useContext(FluidContext);\n if (!context) {\n throw new Error(\"useFluidContext must be used within a FluidProvider\");\n }\n return context;\n}\n"],"x_google_ignoreList":[0,1,2,3],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"FluidProvider-DpET43hN.mjs","names":["#provider","ApiError","fluidOs.getFluidOSManifest","fluidOs.getFluidOSManifest","portalTenant.app_definition_show","portalTenant.account_show","portalTenant.account_update","portalTenantPay.addresses_list","portalTenantPay.addresses_create","portalTenantPay.addresses_update","portalTenantPay.addresses_destroy","portalTenantPay.payment_methods_list","portalTenantPay.payment_methods_create","portalTenantPay.payment_methods_update","portalTenantPay.payment_methods_destroy","portalTenantPay.payment_methods_vault_show","portalTenantPay.points_ledgers_list","mapMeta","portalTenantStore.countries_list","portalTenantStore.languages_list","portalTenantMysite.mysite_profile_show","portalTenantMysite.mysite_profile_update","portalTenantMysite.mysite_settings_update","portalTenantMysite.mysite_links_list","portalTenantMysite.mysite_links_create","portalTenantMysite.mysite_links_update","portalTenantMysite.mysite_links_destroy","portalTenantMysite.mysite_links_bulk_reorder","portalTenantMysite.mysite_favorites_list","portalTenantMysite.mysite_favorites_create","portalTenantMysite.mysite_favorites_destroy","portalTenantMysite.mysite_favorites_bulk_reorder","portalTenantMysite.mysite_themes_list","createPortalTenantFetchClient"],"sources":["../../../../node_modules/.pnpm/@tanstack+query-core@5.90.12/node_modules/@tanstack/query-core/build/modern/timeoutManager.js","../../../../node_modules/.pnpm/@tanstack+query-core@5.90.12/node_modules/@tanstack/query-core/build/modern/utils.js","../../../../node_modules/.pnpm/@tanstack+query-core@5.90.12/node_modules/@tanstack/query-core/build/modern/notifyManager.js","../../../../node_modules/.pnpm/@tanstack+query-persist-client-core@5.91.11/node_modules/@tanstack/query-persist-client-core/build/modern/createPersister.js","../../../platform/query-persister/src/persister.ts","../../core/src/types/fluidos-api-context.ts","../../core/src/app-definition-api-context.ts","../../core/src/pay-api-context.ts","../../../mysite/core/src/mysite-api-context.ts","../../../store/core/src/languages-api-context.ts","../../../platform/api-client-core/src/fetch-client.ts","../../../api-clients/fluidos/src/namespaces/fluid_os.ts","../src/client/types.ts","../../core/src/theme/types.ts","../../core/src/theme/color-engine.ts","../../core/src/theme/tailwind-overrides.ts","../../core/src/theme/css-generator.ts","../../core/src/theme/defaults.ts","../../core/src/theme/serialisation.ts","../../core/src/theme/transforms.ts","../../core/src/theme/theme-applicator.ts","../src/transforms/screen-transforms.ts","../src/transforms/navigation-transforms.ts","../src/transforms/index.ts","../src/client/fluid-client.ts","../src/adapters/fluidos-api-adapter.ts","../src/adapters/app-definition-api-adapter.ts","../src/adapters/account-api-adapter.ts","../../../api-clients/portal-tenant-pay/src/namespaces/portal_tenant_pay.ts","../src/adapters/pay-api-adapter.ts","../../../api-clients/portal-tenant-store/src/namespaces/portal_tenant_store.ts","../../../store/api-client/src/portal-tenant-countries-adapter.ts","../../../store/api-client/src/portal-tenant-languages-adapter.ts","../src/adapters/countries-api-adapter.ts","../src/adapters/languages-api-adapter.ts","../src/adapters/mysite-api-adapter.ts","../src/providers/FluidThemeProvider.tsx","../../core/src/widget-utils/utils.ts","../../widgets/src/widgets/index.ts","../src/core/default-widget-registry.ts","../src/providers/FluidProvider.tsx"],"sourcesContent":["// src/timeoutManager.ts\nvar defaultTimeoutProvider = {\n // We need the wrapper function syntax below instead of direct references to\n // global setTimeout etc.\n //\n // BAD: `setTimeout: setTimeout`\n // GOOD: `setTimeout: (cb, delay) => setTimeout(cb, delay)`\n //\n // If we use direct references here, then anything that wants to spy on or\n // replace the global setTimeout (like tests) won't work since we'll already\n // have a hard reference to the original implementation at the time when this\n // file was imported.\n setTimeout: (callback, delay) => setTimeout(callback, delay),\n clearTimeout: (timeoutId) => clearTimeout(timeoutId),\n setInterval: (callback, delay) => setInterval(callback, delay),\n clearInterval: (intervalId) => clearInterval(intervalId)\n};\nvar TimeoutManager = class {\n // We cannot have TimeoutManager<T> as we must instantiate it with a concrete\n // type at app boot; and if we leave that type, then any new timer provider\n // would need to support ReturnType<typeof setTimeout>, which is infeasible.\n //\n // We settle for type safety for the TimeoutProvider type, and accept that\n // this class is unsafe internally to allow for extension.\n #provider = defaultTimeoutProvider;\n #providerCalled = false;\n setTimeoutProvider(provider) {\n if (process.env.NODE_ENV !== \"production\") {\n if (this.#providerCalled && provider !== this.#provider) {\n console.error(\n `[timeoutManager]: Switching provider after calls to previous provider might result in unexpected behavior.`,\n { previous: this.#provider, provider }\n );\n }\n }\n this.#provider = provider;\n if (process.env.NODE_ENV !== \"production\") {\n this.#providerCalled = false;\n }\n }\n setTimeout(callback, delay) {\n if (process.env.NODE_ENV !== \"production\") {\n this.#providerCalled = true;\n }\n return this.#provider.setTimeout(callback, delay);\n }\n clearTimeout(timeoutId) {\n this.#provider.clearTimeout(timeoutId);\n }\n setInterval(callback, delay) {\n if (process.env.NODE_ENV !== \"production\") {\n this.#providerCalled = true;\n }\n return this.#provider.setInterval(callback, delay);\n }\n clearInterval(intervalId) {\n this.#provider.clearInterval(intervalId);\n }\n};\nvar timeoutManager = new TimeoutManager();\nfunction systemSetTimeoutZero(callback) {\n setTimeout(callback, 0);\n}\nexport {\n TimeoutManager,\n defaultTimeoutProvider,\n systemSetTimeoutZero,\n timeoutManager\n};\n//# sourceMappingURL=timeoutManager.js.map","// src/utils.ts\nimport { timeoutManager } from \"./timeoutManager.js\";\nvar isServer = typeof window === \"undefined\" || \"Deno\" in globalThis;\nfunction noop() {\n}\nfunction functionalUpdate(updater, input) {\n return typeof updater === \"function\" ? updater(input) : updater;\n}\nfunction isValidTimeout(value) {\n return typeof value === \"number\" && value >= 0 && value !== Infinity;\n}\nfunction timeUntilStale(updatedAt, staleTime) {\n return Math.max(updatedAt + (staleTime || 0) - Date.now(), 0);\n}\nfunction resolveStaleTime(staleTime, query) {\n return typeof staleTime === \"function\" ? staleTime(query) : staleTime;\n}\nfunction resolveEnabled(enabled, query) {\n return typeof enabled === \"function\" ? enabled(query) : enabled;\n}\nfunction matchQuery(filters, query) {\n const {\n type = \"all\",\n exact,\n fetchStatus,\n predicate,\n queryKey,\n stale\n } = filters;\n if (queryKey) {\n if (exact) {\n if (query.queryHash !== hashQueryKeyByOptions(queryKey, query.options)) {\n return false;\n }\n } else if (!partialMatchKey(query.queryKey, queryKey)) {\n return false;\n }\n }\n if (type !== \"all\") {\n const isActive = query.isActive();\n if (type === \"active\" && !isActive) {\n return false;\n }\n if (type === \"inactive\" && isActive) {\n return false;\n }\n }\n if (typeof stale === \"boolean\" && query.isStale() !== stale) {\n return false;\n }\n if (fetchStatus && fetchStatus !== query.state.fetchStatus) {\n return false;\n }\n if (predicate && !predicate(query)) {\n return false;\n }\n return true;\n}\nfunction matchMutation(filters, mutation) {\n const { exact, status, predicate, mutationKey } = filters;\n if (mutationKey) {\n if (!mutation.options.mutationKey) {\n return false;\n }\n if (exact) {\n if (hashKey(mutation.options.mutationKey) !== hashKey(mutationKey)) {\n return false;\n }\n } else if (!partialMatchKey(mutation.options.mutationKey, mutationKey)) {\n return false;\n }\n }\n if (status && mutation.state.status !== status) {\n return false;\n }\n if (predicate && !predicate(mutation)) {\n return false;\n }\n return true;\n}\nfunction hashQueryKeyByOptions(queryKey, options) {\n const hashFn = options?.queryKeyHashFn || hashKey;\n return hashFn(queryKey);\n}\nfunction hashKey(queryKey) {\n return JSON.stringify(\n queryKey,\n (_, val) => isPlainObject(val) ? Object.keys(val).sort().reduce((result, key) => {\n result[key] = val[key];\n return result;\n }, {}) : val\n );\n}\nfunction partialMatchKey(a, b) {\n if (a === b) {\n return true;\n }\n if (typeof a !== typeof b) {\n return false;\n }\n if (a && b && typeof a === \"object\" && typeof b === \"object\") {\n return Object.keys(b).every((key) => partialMatchKey(a[key], b[key]));\n }\n return false;\n}\nvar hasOwn = Object.prototype.hasOwnProperty;\nfunction replaceEqualDeep(a, b) {\n if (a === b) {\n return a;\n }\n const array = isPlainArray(a) && isPlainArray(b);\n if (!array && !(isPlainObject(a) && isPlainObject(b))) return b;\n const aItems = array ? a : Object.keys(a);\n const aSize = aItems.length;\n const bItems = array ? b : Object.keys(b);\n const bSize = bItems.length;\n const copy = array ? new Array(bSize) : {};\n let equalItems = 0;\n for (let i = 0; i < bSize; i++) {\n const key = array ? i : bItems[i];\n const aItem = a[key];\n const bItem = b[key];\n if (aItem === bItem) {\n copy[key] = aItem;\n if (array ? i < aSize : hasOwn.call(a, key)) equalItems++;\n continue;\n }\n if (aItem === null || bItem === null || typeof aItem !== \"object\" || typeof bItem !== \"object\") {\n copy[key] = bItem;\n continue;\n }\n const v = replaceEqualDeep(aItem, bItem);\n copy[key] = v;\n if (v === aItem) equalItems++;\n }\n return aSize === bSize && equalItems === aSize ? a : copy;\n}\nfunction shallowEqualObjects(a, b) {\n if (!b || Object.keys(a).length !== Object.keys(b).length) {\n return false;\n }\n for (const key in a) {\n if (a[key] !== b[key]) {\n return false;\n }\n }\n return true;\n}\nfunction isPlainArray(value) {\n return Array.isArray(value) && value.length === Object.keys(value).length;\n}\nfunction isPlainObject(o) {\n if (!hasObjectPrototype(o)) {\n return false;\n }\n const ctor = o.constructor;\n if (ctor === void 0) {\n return true;\n }\n const prot = ctor.prototype;\n if (!hasObjectPrototype(prot)) {\n return false;\n }\n if (!prot.hasOwnProperty(\"isPrototypeOf\")) {\n return false;\n }\n if (Object.getPrototypeOf(o) !== Object.prototype) {\n return false;\n }\n return true;\n}\nfunction hasObjectPrototype(o) {\n return Object.prototype.toString.call(o) === \"[object Object]\";\n}\nfunction sleep(timeout) {\n return new Promise((resolve) => {\n timeoutManager.setTimeout(resolve, timeout);\n });\n}\nfunction replaceData(prevData, data, options) {\n if (typeof options.structuralSharing === \"function\") {\n return options.structuralSharing(prevData, data);\n } else if (options.structuralSharing !== false) {\n if (process.env.NODE_ENV !== \"production\") {\n try {\n return replaceEqualDeep(prevData, data);\n } catch (error) {\n console.error(\n `Structural sharing requires data to be JSON serializable. To fix this, turn off structuralSharing or return JSON-serializable data from your queryFn. [${options.queryHash}]: ${error}`\n );\n throw error;\n }\n }\n return replaceEqualDeep(prevData, data);\n }\n return data;\n}\nfunction keepPreviousData(previousData) {\n return previousData;\n}\nfunction addToEnd(items, item, max = 0) {\n const newItems = [...items, item];\n return max && newItems.length > max ? newItems.slice(1) : newItems;\n}\nfunction addToStart(items, item, max = 0) {\n const newItems = [item, ...items];\n return max && newItems.length > max ? newItems.slice(0, -1) : newItems;\n}\nvar skipToken = Symbol();\nfunction ensureQueryFn(options, fetchOptions) {\n if (process.env.NODE_ENV !== \"production\") {\n if (options.queryFn === skipToken) {\n console.error(\n `Attempted to invoke queryFn when set to skipToken. This is likely a configuration error. Query hash: '${options.queryHash}'`\n );\n }\n }\n if (!options.queryFn && fetchOptions?.initialPromise) {\n return () => fetchOptions.initialPromise;\n }\n if (!options.queryFn || options.queryFn === skipToken) {\n return () => Promise.reject(new Error(`Missing queryFn: '${options.queryHash}'`));\n }\n return options.queryFn;\n}\nfunction shouldThrowError(throwOnError, params) {\n if (typeof throwOnError === \"function\") {\n return throwOnError(...params);\n }\n return !!throwOnError;\n}\nexport {\n addToEnd,\n addToStart,\n ensureQueryFn,\n functionalUpdate,\n hashKey,\n hashQueryKeyByOptions,\n isPlainArray,\n isPlainObject,\n isServer,\n isValidTimeout,\n keepPreviousData,\n matchMutation,\n matchQuery,\n noop,\n partialMatchKey,\n replaceData,\n replaceEqualDeep,\n resolveEnabled,\n resolveStaleTime,\n shallowEqualObjects,\n shouldThrowError,\n skipToken,\n sleep,\n timeUntilStale\n};\n//# sourceMappingURL=utils.js.map","// src/notifyManager.ts\nimport { systemSetTimeoutZero } from \"./timeoutManager.js\";\nvar defaultScheduler = systemSetTimeoutZero;\nfunction createNotifyManager() {\n let queue = [];\n let transactions = 0;\n let notifyFn = (callback) => {\n callback();\n };\n let batchNotifyFn = (callback) => {\n callback();\n };\n let scheduleFn = defaultScheduler;\n const schedule = (callback) => {\n if (transactions) {\n queue.push(callback);\n } else {\n scheduleFn(() => {\n notifyFn(callback);\n });\n }\n };\n const flush = () => {\n const originalQueue = queue;\n queue = [];\n if (originalQueue.length) {\n scheduleFn(() => {\n batchNotifyFn(() => {\n originalQueue.forEach((callback) => {\n notifyFn(callback);\n });\n });\n });\n }\n };\n return {\n batch: (callback) => {\n let result;\n transactions++;\n try {\n result = callback();\n } finally {\n transactions--;\n if (!transactions) {\n flush();\n }\n }\n return result;\n },\n /**\n * All calls to the wrapped function will be batched.\n */\n batchCalls: (callback) => {\n return (...args) => {\n schedule(() => {\n callback(...args);\n });\n };\n },\n schedule,\n /**\n * Use this method to set a custom notify function.\n * This can be used to for example wrap notifications with `React.act` while running tests.\n */\n setNotifyFunction: (fn) => {\n notifyFn = fn;\n },\n /**\n * Use this method to set a custom function to batch notifications together into a single tick.\n * By default React Query will use the batch function provided by ReactDOM or React Native.\n */\n setBatchNotifyFunction: (fn) => {\n batchNotifyFn = fn;\n },\n setScheduler: (fn) => {\n scheduleFn = fn;\n }\n };\n}\nvar notifyManager = createNotifyManager();\nexport {\n createNotifyManager,\n defaultScheduler,\n notifyManager\n};\n//# sourceMappingURL=notifyManager.js.map","// src/createPersister.ts\nimport {\n hashKey,\n matchQuery,\n notifyManager,\n partialMatchKey\n} from \"@tanstack/query-core\";\nvar PERSISTER_KEY_PREFIX = \"tanstack-query\";\nfunction experimental_createQueryPersister({\n storage,\n buster = \"\",\n maxAge = 1e3 * 60 * 60 * 24,\n serialize = JSON.stringify,\n deserialize = JSON.parse,\n prefix = PERSISTER_KEY_PREFIX,\n refetchOnRestore = true,\n filters\n}) {\n function isExpiredOrBusted(persistedQuery) {\n if (persistedQuery.state.dataUpdatedAt) {\n const queryAge = Date.now() - persistedQuery.state.dataUpdatedAt;\n const expired = queryAge > maxAge;\n const busted = persistedQuery.buster !== buster;\n if (expired || busted) {\n return true;\n }\n return false;\n }\n return true;\n }\n async function retrieveQuery(queryHash, afterRestoreMacroTask) {\n if (storage != null) {\n const storageKey = `${prefix}-${queryHash}`;\n try {\n const storedData = await storage.getItem(storageKey);\n if (storedData) {\n const persistedQuery = await deserialize(storedData);\n if (isExpiredOrBusted(persistedQuery)) {\n await storage.removeItem(storageKey);\n } else {\n if (afterRestoreMacroTask) {\n notifyManager.schedule(\n () => afterRestoreMacroTask(persistedQuery)\n );\n }\n return persistedQuery.state.data;\n }\n }\n } catch (err) {\n if (process.env.NODE_ENV === \"development\") {\n console.error(err);\n console.warn(\n \"Encountered an error attempting to restore query cache from persisted location.\"\n );\n }\n await storage.removeItem(storageKey);\n }\n }\n return;\n }\n async function persistQueryByKey(queryKey, queryClient) {\n if (storage != null) {\n const query = queryClient.getQueryCache().find({ queryKey });\n if (query) {\n await persistQuery(query);\n } else {\n if (process.env.NODE_ENV === \"development\") {\n console.warn(\n \"Could not find query to be persisted. QueryKey:\",\n JSON.stringify(queryKey)\n );\n }\n }\n }\n }\n async function persistQuery(query) {\n if (storage != null) {\n const storageKey = `${prefix}-${query.queryHash}`;\n storage.setItem(\n storageKey,\n await serialize({\n state: query.state,\n queryKey: query.queryKey,\n queryHash: query.queryHash,\n buster\n })\n );\n }\n }\n async function persisterFn(queryFn, ctx, query) {\n const matchesFilter = filters ? matchQuery(filters, query) : true;\n if (matchesFilter && query.state.data === void 0 && storage != null) {\n const restoredData = await retrieveQuery(\n query.queryHash,\n (persistedQuery) => {\n query.setState({\n dataUpdatedAt: persistedQuery.state.dataUpdatedAt,\n errorUpdatedAt: persistedQuery.state.errorUpdatedAt\n });\n if (refetchOnRestore === \"always\" || refetchOnRestore === true && query.isStale()) {\n query.fetch();\n }\n }\n );\n if (restoredData !== void 0) {\n return Promise.resolve(restoredData);\n }\n }\n const queryFnResult = await queryFn(ctx);\n if (matchesFilter && storage != null) {\n notifyManager.schedule(() => {\n persistQuery(query);\n });\n }\n return Promise.resolve(queryFnResult);\n }\n async function persisterGc() {\n if (storage?.entries) {\n const entries = await storage.entries();\n for (const [key, value] of entries) {\n if (key.startsWith(prefix)) {\n const persistedQuery = await deserialize(value);\n if (isExpiredOrBusted(persistedQuery)) {\n await storage.removeItem(key);\n }\n }\n }\n } else if (process.env.NODE_ENV === \"development\") {\n throw new Error(\n \"Provided storage does not implement `entries` method. Garbage collection is not possible without ability to iterate over storage items.\"\n );\n }\n }\n async function restoreQueries(queryClient, filters2 = {}) {\n const { exact, queryKey } = filters2;\n if (storage?.entries) {\n const entries = await storage.entries();\n for (const [key, value] of entries) {\n if (key.startsWith(prefix)) {\n const persistedQuery = await deserialize(value);\n if (isExpiredOrBusted(persistedQuery)) {\n await storage.removeItem(key);\n continue;\n }\n if (queryKey) {\n if (exact) {\n if (persistedQuery.queryHash !== hashKey(queryKey)) {\n continue;\n }\n } else if (!partialMatchKey(persistedQuery.queryKey, queryKey)) {\n continue;\n }\n }\n queryClient.setQueryData(\n persistedQuery.queryKey,\n persistedQuery.state.data,\n {\n updatedAt: persistedQuery.state.dataUpdatedAt\n }\n );\n }\n }\n } else if (process.env.NODE_ENV === \"development\") {\n throw new Error(\n \"Provided storage does not implement `entries` method. Restoration of all stored entries is not possible without ability to iterate over storage items.\"\n );\n }\n }\n return {\n persisterFn,\n persistQuery,\n persistQueryByKey,\n retrieveQuery,\n persisterGc,\n restoreQueries\n };\n}\nexport {\n PERSISTER_KEY_PREFIX,\n experimental_createQueryPersister\n};\n//# sourceMappingURL=createPersister.js.map","import {\n type AsyncStorage,\n experimental_createQueryPersister,\n type PersistedQuery,\n} from \"@tanstack/react-query-persist-client\";\n\nconst DB_NAME = \"fluid_tanstack_query_cache\";\nconst STORE_NAME = \"fluid_queries\";\nconst VERSION = 1;\n\nlet dbPromise: Promise<IDBDatabase> | null = null;\n\n// Exported for testing - allows resetting internal state between tests\nexport function _resetForTesting(): void {\n dbPromise = null;\n}\n\nexport async function deleteDatabase(reason: string = \"error\"): Promise<void> {\n return new Promise((resolve, reject) => {\n console.warn(`[IDB] Deleting database due to ${reason}`);\n const req = indexedDB.deleteDatabase(DB_NAME);\n req.onsuccess = () => {\n console.log(\"[IDB] Database deleted successfully\");\n dbPromise = null; // ensure next getDb() opens a fresh connection\n resolve();\n };\n req.onerror = () => {\n console.error(\"[IDB] Failed to delete database:\", req.error);\n reject(req.error ?? new Error(\"deleteDatabase failed\"));\n };\n req.onblocked = () => {\n console.warn(\"[IDB] Delete blocked: close all tabs using this database\");\n // Resolve anyway so callers (e.g. clearCompanyCache) don't hang\n // indefinitely. The delete will complete once blocking connections close.\n dbPromise = null; // best-effort: allow fresh open on next access\n resolve();\n };\n });\n}\n\nfunction openDatabase(): Promise<IDBDatabase> {\n return new Promise((resolve, reject) => {\n const req = indexedDB.open(DB_NAME, VERSION);\n\n req.onupgradeneeded = () => {\n const upgradeDb = req.result;\n if (!upgradeDb.objectStoreNames.contains(STORE_NAME)) {\n upgradeDb.createObjectStore(STORE_NAME);\n }\n };\n\n req.onsuccess = () => {\n const conn = req.result;\n conn.onversionchange = () => {\n console.trace(\"[IDB] version change – closing connection\");\n conn.close();\n // clear the cached promise so the next getDb call re-opens a connection\n dbPromise = null;\n };\n resolve(conn);\n };\n\n req.onblocked = () => {\n console.warn(\"[IDB] open blocked: another connection is holding the DB\");\n };\n\n req.onerror = () => {\n reject(\n req.error instanceof Error\n ? req.error\n : new Error(`IndexedDB open failed: ${String(req.error)}`),\n );\n };\n });\n}\n\nasync function getDbWithRecovery(): Promise<IDBDatabase> {\n try {\n return await openDatabase();\n } catch (err) {\n // First open failed - try to delete and recreate\n console.warn(\"[IDB] Initial open failed, attempting recovery:\", err);\n try {\n await deleteDatabase();\n console.log(\"[IDB] Retrying database open after deletion\");\n return await openDatabase();\n } catch (retryErr) {\n console.error(\"[IDB] Recovery failed:\", retryErr);\n throw retryErr;\n }\n }\n}\n\nfunction getDb(): Promise<IDBDatabase> {\n if (dbPromise) return dbPromise;\n\n // Create the promise and cache it immediately to prevent race conditions\n // All concurrent calls will share this same promise\n dbPromise = getDbWithRecovery().catch((err) => {\n // On failure, clear the cache so next call can retry fresh\n dbPromise = null;\n throw err;\n });\n\n return dbPromise;\n}\n\n// Exported for testing - allows direct testing of storage operations\nexport const storage: AsyncStorage<PersistedQuery> = {\n async getItem(key: string) {\n try {\n const db = await getDb();\n return new Promise<PersistedQuery | undefined>((res) => {\n try {\n const r = db\n .transaction(STORE_NAME, \"readonly\")\n .objectStore(STORE_NAME)\n .get(key);\n r.onsuccess = () => res(r.result as PersistedQuery);\n r.onerror = () => {\n console.trace(\"[IDB] getItem error:\", r.error);\n res(undefined);\n };\n } catch (txErr) {\n // Transaction creation failed (e.g., stale connection)\n console.trace(\"[IDB] getItem transaction error:\", txErr);\n res(undefined);\n }\n });\n } catch (err) {\n // getDb() failed - try to recover with a fresh connection\n console.trace(\"[IDB] getItem getDb error:\", err);\n try {\n const db = await getDb();\n // Use fresh connection directly - getDb() already handles recovery\n return new Promise<PersistedQuery | undefined>((res) => {\n try {\n const r = db\n .transaction(STORE_NAME, \"readonly\")\n .objectStore(STORE_NAME)\n .get(key);\n r.onsuccess = () => res(r.result as PersistedQuery);\n r.onerror = () => {\n console.trace(\"[IDB] getItem retry error:\", r.error);\n res(undefined);\n };\n } catch (txErr) {\n console.trace(\"[IDB] getItem retry transaction error:\", txErr);\n res(undefined);\n }\n });\n } catch (recoveryErr) {\n console.trace(\"[IDB] getItem recovery failed:\", recoveryErr);\n }\n return undefined;\n }\n },\n\n async setItem(key: string, value: PersistedQuery) {\n // JSON round-trip strips non-cloneable properties (functions, class instances)\n // before IndexedDB's structured clone algorithm runs. Without this, IDB\n // throws DataCloneError. Theme Color objects are now kept out of the cache\n // via the select pattern, but other queries may still have non-cloneable data.\n const cloneableValue = JSON.parse(JSON.stringify(value));\n try {\n const db = await getDb();\n if (!db) return;\n await new Promise<void>((resolve) => {\n try {\n const req = db\n .transaction(STORE_NAME, \"readwrite\")\n .objectStore(STORE_NAME)\n .put(cloneableValue, key);\n req.onsuccess = () => resolve();\n req.onerror = () => {\n console.trace(\"[IDB] setItem error:\", req.error);\n resolve();\n };\n } catch (txErr) {\n console.trace(\"[IDB] setItem transaction error:\", txErr);\n resolve();\n }\n });\n } catch (err) {\n // getDb() failed - try to recover with a fresh connection\n console.trace(\"[IDB] setItem getDb error:\", err);\n try {\n const db = await getDb();\n // Use fresh connection directly - getDb() already handles recovery\n await new Promise<void>((resolve) => {\n try {\n const req = db\n .transaction(STORE_NAME, \"readwrite\")\n .objectStore(STORE_NAME)\n .put(cloneableValue, key);\n req.onsuccess = () => resolve();\n req.onerror = () => {\n console.trace(\"[IDB] setItem retry error:\", req.error);\n resolve();\n };\n } catch (txErr) {\n console.trace(\"[IDB] setItem retry transaction error:\", txErr);\n resolve();\n }\n });\n } catch (recoveryErr) {\n console.trace(\"[IDB] setItem recovery failed:\", recoveryErr);\n }\n }\n },\n\n async removeItem(key: string) {\n try {\n const db = await getDb();\n if (!db) return;\n await new Promise<void>((resolve) => {\n try {\n const req = db\n .transaction(STORE_NAME, \"readwrite\")\n .objectStore(STORE_NAME)\n .delete(key);\n req.onsuccess = () => resolve();\n req.onerror = () => {\n console.trace(\"[IDB] removeItem error:\", req.error);\n resolve();\n };\n } catch (txErr) {\n console.trace(\"[IDB] removeItem transaction error:\", txErr);\n resolve();\n }\n });\n } catch (err) {\n // getDb() failed - try to recover with a fresh connection\n console.trace(\"[IDB] removeItem getDb error:\", err);\n try {\n const db = await getDb();\n // Use fresh connection directly - getDb() already handles recovery\n await new Promise<void>((resolve) => {\n try {\n const req = db\n .transaction(STORE_NAME, \"readwrite\")\n .objectStore(STORE_NAME)\n .delete(key);\n req.onsuccess = () => resolve();\n req.onerror = () => {\n console.trace(\"[IDB] removeItem retry error:\", req.error);\n resolve();\n };\n } catch (txErr) {\n console.trace(\"[IDB] removeItem retry transaction error:\", txErr);\n resolve();\n }\n });\n } catch (recoveryErr) {\n console.trace(\"[IDB] removeItem recovery failed:\", recoveryErr);\n }\n }\n },\n};\n\n// Factory function to create persister - only call this client-side\nexport function createPersister(): ReturnType<\n typeof experimental_createQueryPersister\n> {\n return experimental_createQueryPersister<PersistedQuery>({\n storage,\n serialize: (persistedQuery) => persistedQuery,\n deserialize: (cached) => cached,\n });\n}\n","import { createContext, useContext } from \"react\";\nimport type { FluidOsReadApi, FluidOsBuilderApi } from \"./fluidos-api\";\n\nconst FluidOsReadApiContext = createContext<FluidOsReadApi | null>(null);\nconst FluidOsBuilderApiContext = createContext<FluidOsBuilderApi | null>(null);\n\n/** Provider for read-only manifest fetching (used by SDK). */\nexport const FluidOsApiProvider = FluidOsReadApiContext.Provider;\n\n/** Provider for full CRUD access (used by builder / fluid-admin). */\nexport const FluidOsBuilderApiProvider = FluidOsBuilderApiContext.Provider;\n\n/** Hook for read-only manifest fetching (used by SDK). */\nexport function useFluidOsApi(): FluidOsReadApi {\n const builderApi = useContext(FluidOsBuilderApiContext);\n const readApi = useContext(FluidOsReadApiContext);\n if (builderApi) return builderApi;\n if (!readApi) {\n throw new Error(\n \"useFluidOsApi must be used within a FluidOsApiProvider or FluidOsBuilderApiProvider\",\n );\n }\n return readApi;\n}\n\n/** Hook for full CRUD access (used by builder). */\nexport function useFluidOsBuilderApi(): FluidOsBuilderApi {\n const api = useContext(FluidOsBuilderApiContext);\n if (!api) {\n throw new Error(\n \"useFluidOsBuilderApi must be used within a FluidOsBuilderApiProvider\",\n );\n }\n return api;\n}\n\n/** Optional variant that returns null when no provider is present. */\nexport function useFluidOsApiOptional(): FluidOsReadApi | null {\n const builderApi = useContext(FluidOsBuilderApiContext);\n const readApi = useContext(FluidOsReadApiContext);\n return builderApi ?? readApi;\n}\n","import { createContext, useContext, type Provider } from \"react\";\nimport type { AppDefinitionApi } from \"./app-definition-api\";\n\nconst AppDefinitionApiContext = createContext<AppDefinitionApi | null>(null);\n\nexport const AppDefinitionApiProvider: Provider<AppDefinitionApi | null> =\n AppDefinitionApiContext.Provider;\n\nexport function useAppDefinitionApi(): AppDefinitionApi {\n const api = useContext(AppDefinitionApiContext);\n if (!api) {\n throw new Error(\n \"useAppDefinitionApi must be used within an AppDefinitionApiProvider\",\n );\n }\n return api;\n}\n","import { createContext, useContext } from \"react\";\nimport type { PayApi } from \"./pay-api\";\n\nconst PayApiContext = createContext<PayApi | null>(null);\n\nexport const PayApiProvider = PayApiContext.Provider;\n\nexport function usePayApi(): PayApi {\n const api = useContext(PayApiContext);\n if (!api) {\n throw new Error(\"usePayApi must be used within a PayApiProvider\");\n }\n return api;\n}\n","import { createContext, useContext } from \"react\";\nimport type { MySiteApi } from \"./mysite-api\";\n\nconst MySiteApiContext = createContext<MySiteApi | null>(null);\n\nexport const MySiteApiProvider = MySiteApiContext.Provider;\n\nexport function useMySiteApi(): MySiteApi {\n const api = useContext(MySiteApiContext);\n if (!api) {\n throw new Error(\"useMySiteApi must be used within a MySiteApiProvider\");\n }\n return api;\n}\n","import { createContext, useContext, type Provider } from \"react\";\nimport type { LanguagesApi } from \"./languages-api\";\n\nconst LanguagesApiContext = createContext<LanguagesApi | null>(null);\n\nexport const LanguagesApiProvider: Provider<LanguagesApi | null> =\n LanguagesApiContext.Provider;\n\nexport function useLanguagesApi(): LanguagesApi {\n const api = useContext(LanguagesApiContext);\n if (!api) {\n throw new Error(\n \"useLanguagesApi must be used within a LanguagesApiProvider\",\n );\n }\n return api;\n}\n","/**\n * Minimal, framework-agnostic fetch client for Fluid APIs\n * Compatible with fluid-admin patterns but usable standalone\n */\n\nexport interface FetchClientConfig {\n /**\n * Base URL for all requests (e.g., \"https://api.fluid.app/api\")\n */\n baseUrl: string;\n\n /**\n * Optional function to get auth token\n * Return null/undefined if no token available\n */\n getAuthToken?: () => string | null | Promise<string | null>;\n\n /**\n * Optional callback when 401 auth error occurs\n */\n onAuthError?: () => void;\n\n /**\n * Default headers to include in all requests\n * Example: { \"x-fluid-client\": \"admin\" }\n */\n defaultHeaders?: Record<string, string>;\n\n /**\n * Credentials mode for fetch requests.\n * Set to `\"include\"` for cookie-based (same-origin BFF) authentication.\n * @default undefined (browser default: \"same-origin\")\n */\n credentials?: RequestCredentials;\n}\n\nexport interface RequestOptions {\n method?: \"GET\" | \"POST\" | \"PUT\" | \"DELETE\" | \"PATCH\";\n headers?: Record<string, string>;\n params?: Record<string, unknown>;\n body?: unknown;\n signal?: AbortSignal;\n}\n\n/**\n * API Error class compatible with fluid-admin's ApiError\n */\nexport class ApiError extends Error {\n public readonly status: number;\n public readonly data: unknown;\n\n constructor(message: string, status: number, data?: unknown) {\n super(message);\n this.name = \"ApiError\";\n this.status = status;\n this.data = data;\n\n if (\"captureStackTrace\" in Error) {\n (\n Error as {\n captureStackTrace: (\n target: Error,\n constructor: NewableFunction,\n ) => void;\n }\n ).captureStackTrace(this, ApiError);\n }\n }\n\n toJSON(): { name: string; message: string; status: number; data: unknown } {\n return {\n name: this.name,\n message: this.message,\n status: this.status,\n data: this.data,\n };\n }\n}\n\n/**\n * Type guard for ApiError\n */\nexport function isApiError(error: unknown): error is ApiError {\n return error instanceof ApiError;\n}\n\nexport interface FetchClientInstance {\n request: <TResponse = unknown>(\n endpoint: string,\n options?: RequestOptions,\n ) => Promise<TResponse>;\n requestWithFormData: <TResponse = unknown>(\n endpoint: string,\n formData: FormData,\n options?: Omit<RequestOptions, \"body\" | \"params\"> & {\n method?: \"POST\" | \"PUT\" | \"PATCH\";\n },\n ) => Promise<TResponse>;\n get: <TResponse = unknown>(\n endpoint: string,\n params?: Record<string, unknown>,\n options?: Omit<RequestOptions, \"method\" | \"params\">,\n ) => Promise<TResponse>;\n post: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ) => Promise<TResponse>;\n put: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ) => Promise<TResponse>;\n patch: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ) => Promise<TResponse>;\n delete: <TResponse = unknown>(\n endpoint: string,\n options?: Omit<RequestOptions, \"method\">,\n ) => Promise<TResponse>;\n}\n\n/**\n * Creates a configured fetch client instance\n */\nexport function createFetchClient(\n config: FetchClientConfig,\n): FetchClientInstance {\n const {\n baseUrl,\n getAuthToken,\n onAuthError,\n defaultHeaders = {},\n credentials,\n } = config;\n\n /**\n * Build headers for a request\n */\n async function buildHeaders(\n customHeaders?: Record<string, string>,\n ): Promise<Record<string, string>> {\n const headers: Record<string, string> = {\n Accept: \"application/json\",\n \"Content-Type\": \"application/json\",\n ...defaultHeaders,\n ...customHeaders,\n };\n\n // Add auth token if available\n if (getAuthToken) {\n const token = await getAuthToken();\n if (token) {\n headers.Authorization = `Bearer ${token}`;\n }\n }\n\n return headers;\n }\n\n /**\n * Join baseUrl + endpoint via string concatenation (matches fetchApi).\n * Using `new URL(endpoint, baseUrl)` would strip any path prefix from\n * baseUrl (e.g. \"/api\") when the endpoint starts with \"/\".\n */\n function joinUrl(endpoint: string): string {\n return `${baseUrl}${endpoint}`;\n }\n\n /**\n * Build URL with query parameters for GET requests\n * Compatible with fluid-admin's query param handling\n */\n function buildUrl(\n endpoint: string,\n params?: Record<string, unknown>,\n ): string {\n const fullUrl = joinUrl(endpoint);\n\n if (!params || Object.keys(params).length === 0) {\n return fullUrl;\n }\n\n const queryString = new URLSearchParams();\n\n Object.entries(params).forEach(([key, value]) => {\n if (value === undefined || value === null) {\n return; // Skip undefined/null values\n }\n\n if (Array.isArray(value)) {\n // Handle arrays like Rails expects: key[]\n value.forEach((item) => queryString.append(`${key}[]`, String(item)));\n } else if (typeof value === \"object\") {\n // Handle nested objects: key[subkey]\n Object.entries(value).forEach(([subKey, subValue]) => {\n if (subValue === undefined || subValue === null) {\n return;\n }\n\n if (Array.isArray(subValue)) {\n subValue.forEach((item) =>\n queryString.append(`${key}[${subKey}][]`, String(item)),\n );\n } else {\n queryString.append(`${key}[${subKey}]`, String(subValue));\n }\n });\n } else {\n queryString.append(key, String(value));\n }\n });\n\n const qs = queryString.toString();\n return qs ? `${fullUrl}?${qs}` : fullUrl;\n }\n\n /**\n * Shared response handler for both JSON and FormData requests.\n * Handles auth errors, non-OK responses, 204 No Content, and JSON parsing.\n */\n async function handleResponse<TResponse>(\n response: Response,\n method: string,\n _url: string,\n ): Promise<TResponse> {\n if (response.status === 401 && onAuthError) {\n onAuthError();\n }\n\n if (!response.ok) {\n // Read body as text first to avoid SyntaxError from response.json()\n // when server returns non-JSON bodies with application/json content-type.\n const errorText = await response.text().catch(() => \"\");\n const contentType = response.headers.get(\"content-type\");\n\n if (contentType?.includes(\"application/json\")) {\n let data: Record<string, unknown>;\n try {\n data = JSON.parse(errorText);\n } catch {\n throw new ApiError(\n errorText.slice(0, 200) ||\n `${method} request failed with status ${response.status}`,\n response.status,\n null,\n );\n }\n const msg = (data.message || data.error_message) as string | undefined;\n throw new ApiError(\n msg || `${method} request failed`,\n response.status,\n data.errors || data,\n );\n } else {\n throw new ApiError(\n `${method} request failed with status ${response.status}`,\n response.status,\n null,\n );\n }\n }\n\n if (\n response.status === 204 ||\n response.headers.get(\"content-length\") === \"0\"\n ) {\n return null as TResponse;\n }\n\n const contentType = response.headers.get(\"content-type\");\n\n if (contentType?.includes(\"application/json\")) {\n try {\n const data = await response.json();\n return data as TResponse;\n } catch {\n try {\n // API declared JSON content-type but body isn't valid JSON\n const text = await response.text();\n return text as TResponse;\n } catch {\n return null as TResponse;\n }\n }\n }\n\n // Non-JSON response (text/plain, text/html, etc.)\n return null as TResponse;\n }\n\n /**\n * Main request function\n */\n async function request<TResponse = unknown>(\n endpoint: string,\n options: RequestOptions = {},\n ): Promise<TResponse> {\n const {\n method = \"GET\",\n headers: customHeaders,\n params,\n body,\n signal,\n } = options;\n\n const url = params ? buildUrl(endpoint, params) : joinUrl(endpoint);\n\n const headers = await buildHeaders(customHeaders);\n\n let response: Response;\n\n try {\n const fetchOptions: RequestInit = { method, headers };\n if (credentials) fetchOptions.credentials = credentials;\n const serializedBody =\n body && method !== \"GET\" ? JSON.stringify(body) : null;\n if (serializedBody) fetchOptions.body = serializedBody;\n if (signal) fetchOptions.signal = signal;\n response = await fetch(url, fetchOptions);\n } catch (networkError) {\n throw new ApiError(\n `Network error: ${networkError instanceof Error ? networkError.message : \"Unknown network error\"}`,\n 0,\n null,\n );\n }\n\n return handleResponse<TResponse>(response, method, url);\n }\n\n /**\n * Request with FormData (for file uploads)\n */\n async function requestWithFormData<TResponse = unknown>(\n endpoint: string,\n formData: FormData,\n options: Omit<RequestOptions, \"body\" | \"params\"> & {\n method?: \"POST\" | \"PUT\" | \"PATCH\";\n } = {},\n ): Promise<TResponse> {\n const { method = \"POST\", headers: customHeaders, signal } = options;\n\n const url = joinUrl(endpoint);\n const headers = await buildHeaders(customHeaders);\n\n // Remove Content-Type to let browser set it with boundary\n delete headers[\"Content-Type\"];\n\n let response: Response;\n\n try {\n const fetchOptions: RequestInit = { method, headers, body: formData };\n if (credentials) fetchOptions.credentials = credentials;\n if (signal) fetchOptions.signal = signal;\n response = await fetch(url, fetchOptions);\n } catch (networkError) {\n throw new ApiError(\n `Network error: ${networkError instanceof Error ? networkError.message : \"Unknown network error\"}`,\n 0,\n null,\n );\n }\n\n return handleResponse<TResponse>(response, method, url);\n }\n\n // Return client with convenience methods\n return {\n request: request,\n requestWithFormData: requestWithFormData,\n\n // Convenience methods for common HTTP verbs\n get: <TResponse = unknown>(\n endpoint: string,\n params?: Record<string, unknown>,\n options?: Omit<RequestOptions, \"method\" | \"params\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: \"GET\" as const,\n ...(params && { params }),\n }),\n\n post: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: \"POST\",\n body,\n }),\n\n put: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: \"PUT\",\n body,\n }),\n\n patch: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: \"PATCH\",\n body,\n }),\n\n delete: <TResponse = unknown>(\n endpoint: string,\n options?: Omit<RequestOptions, \"method\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: \"DELETE\",\n }),\n };\n}\n\nexport type FetchClient = FetchClientInstance;\n","/**\n * Generated API client functions for fluid_os\n *\n * DO NOT EDIT THIS FILE DIRECTLY\n * This file is auto-generated. To update:\n * 1. Update the OpenAPI spec file\n * 2. Run: pnpm generate\n */\n\nimport type { FetchClient } from \"../lib/fetch-client\";\nimport type { operations } from \"../generated/fluid_os\";\n\n// ============================================================================\n// Fluid OS - Definitions\n// ============================================================================\n\n/**\n * List Fluid OS definitions\n * Retrieve a list of Fluid OS definitions for the current company\n *\n * @param client - Fetch client instance\n * @param params? - params?\n */\nexport async function listFluidOSDefinitions(\n client: FetchClient,\n params?: operations[\"listFluidOSDefinitions\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"listFluidOSDefinitions\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/company/fluid_os/definitions`, params);\n}\n\n/**\n * Create a Fluid OS definition\n * Create a new Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param body - body\n */\nexport async function createFluidOSDefinition(\n client: FetchClient,\n body: operations[\"createFluidOSDefinition\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"createFluidOSDefinition\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.post(`/api/company/fluid_os/definitions`, body);\n}\n\n/**\n * Get a Fluid OS definition\n * Retrieve a specific Fluid OS definition with all associations\n *\n * @param client - Fetch client instance\n * @param id - id\n */\nexport async function getFluidOSDefinition(\n client: FetchClient,\n id: string | number,\n): Promise<\n operations[\"getFluidOSDefinition\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/company/fluid_os/definitions/${id}`);\n}\n\n/**\n * Update a Fluid OS definition\n * Update an existing Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param id - id\n * @param body - body\n */\nexport async function updateFluidOSDefinition(\n client: FetchClient,\n id: string | number,\n body: operations[\"updateFluidOSDefinition\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"updateFluidOSDefinition\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.put(`/api/company/fluid_os/definitions/${id}`, body);\n}\n\n/**\n * Delete a Fluid OS definition\n * Delete a Definition\n *\n * @param client - Fetch client instance\n * @param id - id\n */\nexport async function deleteFluidOSDefinition(\n client: FetchClient,\n id: string | number,\n): Promise<\n operations[\"deleteFluidOSDefinition\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.delete(`/api/company/fluid_os/definitions/${id}`);\n}\n\n// ============================================================================\n// Fluid OS - Navigation Items\n// ============================================================================\n\n/**\n * List navigation items for a navigation\n * Retrieve a list of navigation items for a specific navigation\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param navigation_id - navigation_id\n */\nexport async function listFluidOSNavigationItems(\n client: FetchClient,\n definition_id: string | number,\n navigation_id: string | number,\n): Promise<\n operations[\"listFluidOSNavigationItems\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/navigations/${navigation_id}/navigation_items`,\n );\n}\n\n/**\n * Create a navigation item\n * Create a new navigation item for a navigation\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param navigation_id - navigation_id\n * @param body - body\n */\nexport async function createFluidOSNavigationItem(\n client: FetchClient,\n definition_id: string | number,\n navigation_id: string | number,\n body: operations[\"createFluidOSNavigationItem\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"createFluidOSNavigationItem\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.post(\n `/api/company/fluid_os/definitions/${definition_id}/navigations/${navigation_id}/navigation_items`,\n body,\n );\n}\n\n/**\n * Get a specific navigation item\n * Retrieve a specific navigation item\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param navigation_id - navigation_id\n * @param id - id\n */\nexport async function getFluidOSNavigationItem(\n client: FetchClient,\n definition_id: string | number,\n navigation_id: string | number,\n id: string | number,\n): Promise<\n operations[\"getFluidOSNavigationItem\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/navigations/${navigation_id}/navigation_items/${id}`,\n );\n}\n\n/**\n * Update a navigation item\n * Update an existing navigation item\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param navigation_id - navigation_id\n * @param id - id\n * @param body - body\n */\nexport async function updateFluidOSNavigationItem(\n client: FetchClient,\n definition_id: string | number,\n navigation_id: string | number,\n id: string | number,\n body: operations[\"updateFluidOSNavigationItem\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"updateFluidOSNavigationItem\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.put(\n `/api/company/fluid_os/definitions/${definition_id}/navigations/${navigation_id}/navigation_items/${id}`,\n body,\n );\n}\n\n/**\n * Delete a navigation item\n * Delete a navigation item\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param navigation_id - navigation_id\n * @param id - id\n */\nexport async function deleteFluidOSNavigationItem(\n client: FetchClient,\n definition_id: string | number,\n navigation_id: string | number,\n id: string | number,\n): Promise<\n operations[\"deleteFluidOSNavigationItem\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.delete(\n `/api/company/fluid_os/definitions/${definition_id}/navigations/${navigation_id}/navigation_items/${id}`,\n );\n}\n\n// ============================================================================\n// Fluid OS - Navigations\n// ============================================================================\n\n/**\n * List navigations for a Fluid OS definition\n * Retrieve a list of navigations for a specific Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param params? - params?\n */\nexport async function listFluidOSNavigations(\n client: FetchClient,\n definition_id: string | number,\n params?: operations[\"listFluidOSNavigations\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"listFluidOSNavigations\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/navigations`,\n params,\n );\n}\n\n/**\n * Create a navigation for a Fluid OS definition\n * Create a new navigation for a Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param body - body\n */\nexport async function createFluidOSNavigation(\n client: FetchClient,\n definition_id: string | number,\n body: operations[\"createFluidOSNavigation\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"createFluidOSNavigation\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.post(\n `/api/company/fluid_os/definitions/${definition_id}/navigations`,\n body,\n );\n}\n\n/**\n * Get a specific navigation\n * Retrieve a specific navigation\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function getFluidOSNavigation(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"getFluidOSNavigation\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/navigations/${id}`,\n );\n}\n\n/**\n * Update a navigation\n * Update an existing navigation\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n * @param body - body\n */\nexport async function updateFluidOSNavigation(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n body: operations[\"updateFluidOSNavigation\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"updateFluidOSNavigation\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.put(\n `/api/company/fluid_os/definitions/${definition_id}/navigations/${id}`,\n body,\n );\n}\n\n/**\n * Delete a navigation\n * Delete a navigation\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function deleteFluidOSNavigation(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"deleteFluidOSNavigation\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.delete(\n `/api/company/fluid_os/definitions/${definition_id}/navigations/${id}`,\n );\n}\n\n// ============================================================================\n// Fluid OS - Profiles\n// ============================================================================\n\n/**\n * List profiles for a Fluid OS definition\n * Retrieve a list of profiles for a specific Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param params? - params?\n */\nexport async function listFluidOSProfiles(\n client: FetchClient,\n definition_id: string | number,\n params?: operations[\"listFluidOSProfiles\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"listFluidOSProfiles\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/profiles`,\n params,\n );\n}\n\n/**\n * Create a profile for a Fluid OS definition\n * Create a new profile for a Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param body - body\n */\nexport async function createFluidOSProfile(\n client: FetchClient,\n definition_id: string | number,\n body: operations[\"createFluidOSProfile\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"createFluidOSProfile\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.post(\n `/api/company/fluid_os/definitions/${definition_id}/profiles`,\n body,\n );\n}\n\n/**\n * Get the default profile\n * Retrieve the default profile for a Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n */\nexport async function getDefaultFluidOSProfile(\n client: FetchClient,\n definition_id: string | number,\n): Promise<\n operations[\"getDefaultFluidOSProfile\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/profiles/default`,\n );\n}\n\n/**\n * Get a specific profile\n * Retrieve a specific profile\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function getFluidOSProfile(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"getFluidOSProfile\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/profiles/${id}`,\n );\n}\n\n/**\n * Update a profile\n * Update an existing profile\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n * @param body - body\n */\nexport async function updateFluidOSProfile(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n body: operations[\"updateFluidOSProfile\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"updateFluidOSProfile\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.put(\n `/api/company/fluid_os/definitions/${definition_id}/profiles/${id}`,\n body,\n );\n}\n\n/**\n * Delete a profile\n * Delete a profile\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function deleteFluidOSProfile(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"deleteFluidOSProfile\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.delete(\n `/api/company/fluid_os/definitions/${definition_id}/profiles/${id}`,\n );\n}\n\n// ============================================================================\n// Fluid OS - Public\n// ============================================================================\n\n/**\n * Get active Fluid OS definition\n * Retrieve the active Fluid OS definition manifest for a specific platform\n *\n * @param client - Fetch client instance\n * @param params - params\n */\nexport async function getFluidOSManifest(\n client: FetchClient,\n params: operations[\"getFluidOSManifest\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"getFluidOSManifest\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/fluid_os/definitions/active`, params);\n}\n\n// ============================================================================\n// Fluid OS - Screens\n// ============================================================================\n\n/**\n * List screens for a Fluid OS definition\n * Retrieve a list of screens for a specific Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param params? - params?\n */\nexport async function listFluidOSScreens(\n client: FetchClient,\n definition_id: string | number,\n params?: operations[\"listFluidOSScreens\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"listFluidOSScreens\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/screens`,\n params,\n );\n}\n\n/**\n * Create a screen for a Fluid OS definition\n * Create a new screen for a Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param body - body\n */\nexport async function createFluidOSScreen(\n client: FetchClient,\n definition_id: string | number,\n body: operations[\"createFluidOSScreen\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"createFluidOSScreen\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.post(\n `/api/company/fluid_os/definitions/${definition_id}/screens`,\n body,\n );\n}\n\n/**\n * Get a specific screen\n * Retrieve a specific screen\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function getFluidOSScreen(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"getFluidOSScreen\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/screens/${id}`,\n );\n}\n\n/**\n * Update a screen\n * Update an existing screen\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n * @param body - body\n */\nexport async function updateFluidOSScreen(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n body: operations[\"updateFluidOSScreen\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"updateFluidOSScreen\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.put(\n `/api/company/fluid_os/definitions/${definition_id}/screens/${id}`,\n body,\n );\n}\n\n/**\n * Delete a screen\n * Delete a screen\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function deleteFluidOSScreen(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"deleteFluidOSScreen\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.delete(\n `/api/company/fluid_os/definitions/${definition_id}/screens/${id}`,\n );\n}\n\n// ============================================================================\n// Fluid OS - Themes\n// ============================================================================\n\n/**\n * List themes for a Fluid OS definition\n * Retrieve a list of themes for a specific Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param params? - params?\n */\nexport async function listFluidOSThemes(\n client: FetchClient,\n definition_id: string | number,\n params?: operations[\"listFluidOSThemes\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"listFluidOSThemes\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/themes`,\n params,\n );\n}\n\n/**\n * Create a theme for a Fluid OS definition\n * Create a new theme for a Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param body - body\n */\nexport async function createFluidOSTheme(\n client: FetchClient,\n definition_id: string | number,\n body: operations[\"createFluidOSTheme\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"createFluidOSTheme\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.post(\n `/api/company/fluid_os/definitions/${definition_id}/themes`,\n body,\n );\n}\n\n/**\n * Get a specific theme\n * Retrieve a specific theme\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function getFluidOSTheme(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"getFluidOSTheme\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/themes/${id}`,\n );\n}\n\n/**\n * Update a theme\n * Update an existing theme\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n * @param body - body\n */\nexport async function updateFluidOSTheme(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n body: operations[\"updateFluidOSTheme\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"updateFluidOSTheme\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.put(\n `/api/company/fluid_os/definitions/${definition_id}/themes/${id}`,\n body,\n );\n}\n\n/**\n * Delete a theme\n * Delete a theme\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function deleteFluidOSTheme(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"deleteFluidOSTheme\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.delete(\n `/api/company/fluid_os/definitions/${definition_id}/themes/${id}`,\n );\n}\n\n// ============================================================================\n// Fluid OS - Versions\n// ============================================================================\n\n/**\n * List versions for a Fluid OS definition\n * Retrieve a list of published versions for a specific Fluid OS definition\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param params? - params?\n */\nexport async function listFluidOSVersions(\n client: FetchClient,\n definition_id: string | number,\n params?: operations[\"listFluidOSVersions\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"listFluidOSVersions\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/versions`,\n params,\n );\n}\n\n/**\n * Publish a new version of a Fluid OS definition\n * Publish a new version of the Fluid OS definition. This creates a snapshot of the current definition state including all screens, profiles, themes, and navigations. No request body is required - the manifest is built automatically from the current definition state.\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n */\nexport async function createFluidOSVersion(\n client: FetchClient,\n definition_id: string | number,\n): Promise<\n operations[\"createFluidOSVersion\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.post(\n `/api/company/fluid_os/definitions/${definition_id}/versions`,\n );\n}\n\n/**\n * Get a specific version\n * Retrieve a specific published version including its manifest\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n */\nexport async function getFluidOSVersion(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n): Promise<\n operations[\"getFluidOSVersion\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(\n `/api/company/fluid_os/definitions/${definition_id}/versions/${id}`,\n );\n}\n\n/**\n * Update a version\n * Update a version. Currently only supports activating/deactivating a version.\n *\n * @param client - Fetch client instance\n * @param definition_id - definition_id\n * @param id - id\n * @param body - body\n */\nexport async function updateFluidOSVersion(\n client: FetchClient,\n definition_id: string | number,\n id: string | number,\n body: operations[\"updateFluidOSVersion\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"updateFluidOSVersion\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.put(\n `/api/company/fluid_os/definitions/${definition_id}/versions/${id}`,\n body,\n );\n}\n","// ============================================================================\n// HTTP Method Constants\n// ============================================================================\n\n/**\n * HTTP methods supported by the API client.\n * Use `as const` for literal type inference and type safety.\n */\nexport const HTTP_METHODS = {\n GET: \"GET\",\n POST: \"POST\",\n PUT: \"PUT\",\n PATCH: \"PATCH\",\n DELETE: \"DELETE\",\n} as const;\n\n/**\n * Union type of all supported HTTP methods.\n * Derived from HTTP_METHODS constant to avoid duplication.\n */\nexport type HttpMethod = (typeof HTTP_METHODS)[keyof typeof HTTP_METHODS];\n\n// ============================================================================\n// Configuration Types\n// ============================================================================\n\n/**\n * Configuration for the Fluid SDK client.\n * Use Readonly<FluidSDKConfig> when the config should not be modified after creation.\n */\nexport interface FluidSDKConfig {\n /**\n * Base URL for the Fluid API domain (e.g., \"https://api.fluid.app\").\n * Endpoints include their full path from root (e.g., \"/api/reps/me\").\n */\n readonly baseUrl: string;\n\n /**\n * Function to retrieve the authentication token\n * Return null/undefined if no token is available\n */\n readonly getAuthToken?: () => string | null | Promise<string | null>;\n\n /**\n * Callback invoked when a 401 authentication error occurs\n * Use this to trigger re-authentication flows\n */\n readonly onAuthError?: () => void;\n\n /**\n * Default headers to include in all requests\n * Example: { \"x-fluid-client\": \"portal\" }\n */\n readonly defaultHeaders?: Readonly<Record<string, string>>;\n\n /**\n * Filestack API key for messaging file uploads.\n * If not provided, file attachments will be disabled in the messaging composer.\n */\n readonly filestackApiKey?: string;\n\n /**\n * Override WebSocket URL for real-time messaging.\n * Default: derived from baseUrl by replacing trailing /api with /cable\n */\n readonly websocketUrl?: string;\n\n /**\n * ISO country code for the store/merchant (e.g., \"US\", \"CA\", \"GB\").\n * Used to fetch country-specific payment methods.\n * @default \"US\"\n */\n readonly countryIso?: string;\n}\n\n/**\n * Options for individual API requests.\n * Uses HttpMethod type for method to ensure type safety.\n */\nexport interface RequestOptions {\n readonly method?: HttpMethod;\n readonly body?: unknown;\n readonly params?: Readonly<Record<string, unknown>>;\n readonly headers?: Readonly<Record<string, string>>;\n readonly signal?: AbortSignal;\n}\n\n// ============================================================================\n// Pagination Types\n// ============================================================================\n\n/**\n * Cursor-based pagination parameters for list endpoints.\n * Sent as query params `page[cursor]` and `page[limit]`.\n */\nexport interface PaginationParams {\n readonly page?: {\n readonly cursor?: string;\n readonly limit?: number;\n };\n}\n\n// ============================================================================\n// Sort Order - Derive from constant for single source of truth\n// ============================================================================\n\n/**\n * Sort order constant - single source of truth for sort direction values.\n * Use SORT_ORDERS.asc instead of \"asc\" for type-safe comparisons.\n */\nexport const SORT_ORDERS = {\n asc: \"asc\",\n desc: \"desc\",\n} as const;\n\n/**\n * Union type of sort order values, derived from SORT_ORDERS constant.\n * @see deriving-typeof-for-object-keys pattern\n */\nexport type SortOrder = (typeof SORT_ORDERS)[keyof typeof SORT_ORDERS];\n\n/**\n * Common filter parameters for list endpoints\n */\nexport interface BaseListParams extends PaginationParams {\n readonly sort_by?: string;\n readonly sort_order?: SortOrder;\n readonly search?: string;\n}\n","import type Color from \"colorjs.io\";\n\n// Semantic color names - matches portal-widgets tailwind.config.ts and field-types.ts\nexport const SEMANTIC_COLOR_NAMES = [\n \"background\",\n \"foreground\",\n \"primary\",\n \"secondary\",\n \"accent\",\n \"muted\",\n \"destructive\",\n] as const;\nexport type SemanticColorName = (typeof SEMANTIC_COLOR_NAMES)[number];\n\nexport const SHADE_STEPS = [\n 100, 200, 300, 400, 500, 600, 700, 800, 900,\n] as const;\nexport type ShadeStep = (typeof SHADE_STEPS)[number];\n\nexport const FONT_SIZE_KEYS = [\n \"extraSmall\",\n \"small\",\n \"regular\",\n \"large\",\n \"extraLarge\",\n \"giant\",\n] as const;\nexport type FontSizeKey = (typeof FONT_SIZE_KEYS)[number];\n\nexport const FONT_FAMILY_KEYS = [\"header\", \"body\"] as const;\nexport type FontFamilyKey = (typeof FONT_FAMILY_KEYS)[number];\n\nexport const RADIUS_KEYS = [\"small\", \"medium\", \"large\", \"extraLarge\"] as const;\nexport type RadiusKey = (typeof RADIUS_KEYS)[number];\n\n/** Author-time color input (what the user configures) */\nexport interface ThemeColorInput {\n base: Color;\n foreground: Color;\n}\n\n/** Complete theme definition — stored in-memory with Color objects */\nexport interface ThemeDefinition {\n id: string;\n name: string;\n /** Light mode — always fully specified */\n light: Record<SemanticColorName, ThemeColorInput>;\n /**\n * Dark mode — only user-overridden colors.\n * Missing keys are auto-derived from `light` at resolve time.\n */\n dark: Partial<Record<SemanticColorName, Partial<ThemeColorInput>>>;\n fontSizes: Record<FontSizeKey, string>;\n fontFamilies: Record<FontFamilyKey, string>;\n spacing: string;\n radii: Record<RadiusKey, string>;\n /** When true, theme colors are re-derived from brand guidelines on every load */\n syncWithBrandColors?: boolean;\n}\n\n/** Resolved semantic color with generated shade ramp */\nexport interface ResolvedSemanticColor {\n base: Color;\n foreground: Color;\n shades: Record<ShadeStep, Color>;\n}\n\n/** Complete resolved color set for one mode */\nexport type ResolvedColorSet = Record<SemanticColorName, ResolvedSemanticColor>;\n\n/** Fully resolved theme — all colors materialised for both modes */\nexport interface ResolvedTheme {\n id: string;\n name: string;\n light: ResolvedColorSet;\n dark: ResolvedColorSet;\n fontSizes: ThemeDefinition[\"fontSizes\"];\n fontFamilies: ThemeDefinition[\"fontFamilies\"];\n spacing: string;\n radii: ThemeDefinition[\"radii\"];\n}\n\n/** Plain OKLCH triplet for JSON serialisation (no Color dependency) */\nexport interface OklchPlain {\n l: number;\n c: number;\n h: number;\n}\n\n/** Serialised color pair as stored in the backend payload */\nexport interface ThemeColorPlain {\n base: OklchPlain;\n foreground: OklchPlain;\n}\n\n/** Backend payload — plain JSON, no Color objects */\nexport interface ThemePayload {\n [key: string]: unknown;\n id: string;\n name: string;\n light: Record<SemanticColorName, ThemeColorPlain>;\n dark: Partial<\n Record<SemanticColorName, { base?: OklchPlain; foreground?: OklchPlain }>\n >;\n fontSizes: Record<FontSizeKey, string>;\n fontFamilies: Record<FontFamilyKey, string>;\n spacing: string;\n radii: Record<RadiusKey, string>;\n syncWithBrandColors?: boolean;\n}\n","import Color from \"colorjs.io\";\nimport {\n SEMANTIC_COLOR_NAMES,\n SHADE_STEPS,\n type SemanticColorName,\n type ShadeStep,\n type ThemeColorInput,\n type ThemeDefinition,\n type ResolvedColorSet,\n type ResolvedTheme,\n} from \"./types\";\n\n/**\n * Attempt to convert any string into a Color using colorjs.io.\n * If the string is exactly 6 characters it is assumed to be a bare hex value\n * (e.g. \"3b82f6\") and a \"#\" prefix is added before parsing.\n *\n * @returns the parsed Color, or a neutral gray (`oklch(0.5 0 0)`) on failure\n */\nexport function parseColor(value: string): Color {\n if (value.length === 6) {\n value = `#${value}`;\n }\n try {\n return new Color(value);\n } catch (error) {\n console.warn(\"[theme] Failed to parse color:\", value, error);\n return new Color(\"oklch\", [0.5, 0, 0]);\n }\n}\n\n/**\n * Returns either the original foreground or a corrected lightness variant,\n * whichever provides better contrast against `color`.\n * Inversion triggers when the APCA contrast is below 50.\n */\nexport function getForegroundColor(foreground: Color, color: Color): Color {\n if (foreground.oklch.l == null || color.oklch.l == null) {\n return foreground;\n }\n const contrast = color.contrastAPCA(foreground);\n\n if (contrast < 50) {\n return new Color(\"oklch\", [\n color.oklch.l < 0.7 ? 0.95 : 0.15,\n foreground.oklch.c || 0,\n foreground.oklch.h || 0,\n ]);\n }\n return foreground;\n}\n\n/**\n * Generate a 100–900 shade ramp from a base color.\n * Base anchors at 500. Light shades (100–400) step toward white,\n * dark shades (600–900) step toward black. Dark steps use an asymmetric\n * multiplier (1.6×, 1.875×, 3×, 4× of `darkStep`) for a more gradual\n * initial descent. Chroma is nudged per step for perceptually natural ramps.\n */\nexport function generateShades(base: Color): Record<ShadeStep, Color> {\n const l = base.oklch.l ?? 0;\n const c = base.oklch.c ?? 0;\n const h = base.oklch.h ?? 0;\n\n const safeMax = l >= 0.885 ? 0.995 : 0.97;\n const safeMin = l <= 0.33 ? 0 : 0.21;\n\n const lightStep = (safeMax - l) / 5;\n const darkStep = -(l - safeMin) / 8;\n\n const shade = (lDelta: number, cDelta: number): Color => {\n return new Color(\"oklch\", [\n Math.max(0, Math.min(1, l + lDelta)),\n c <= 0.001 ? c : Math.max(0, c + cDelta),\n h,\n ]);\n };\n\n return {\n 100: shade(5 * lightStep, -0.00375),\n 200: shade(4 * lightStep, -0.00375),\n 300: shade(3 * lightStep, -0.00375),\n 400: shade(2 * lightStep, -0.00375),\n 500: new Color(\"oklch\", [l, c, h]),\n 600: shade(1.6 * darkStep, 0.025),\n 700: shade(1.875 * 2 * darkStep, 0.05),\n 800: shade(3 * 2 * darkStep, 0.075),\n 900: shade(4 * 2 * darkStep, 0.1),\n };\n}\n\n// ── Dark Mode Derivation ────────────────────────────────────────────\n//\n// Dark-mode colors are derived from their light counterparts by adjusting\n// OKLCH lightness and optionally scaling chroma. Neutral slots (background,\n// foreground, muted) use fixed lightness values while chromatic slots\n// (primary, secondary, accent, destructive) invert lightness around 0.5.\n\nconst DARK_DERIVATION_CONFIG: Record<\n SemanticColorName,\n {\n baseLightness: number | \"invert\";\n fgLightness: number | \"invert\";\n chromaScale?: number;\n }\n> = {\n background: { baseLightness: 0.15, fgLightness: 0.93 },\n foreground: { baseLightness: 0.93, fgLightness: 0.15 },\n muted: { baseLightness: 0.22, fgLightness: 0.75 },\n primary: { baseLightness: \"invert\", fgLightness: 0.95, chromaScale: 0.9 },\n secondary: { baseLightness: \"invert\", fgLightness: 0.93, chromaScale: 0.85 },\n accent: { baseLightness: \"invert\", fgLightness: 0.95, chromaScale: 0.9 },\n destructive: {\n baseLightness: \"invert\",\n fgLightness: 0.95,\n chromaScale: 0.95,\n },\n};\n\n/** Invert OKLCH lightness (1 - l), clamped to [0.35, 0.75] to avoid extremes. */\nfunction invertLightness(l: number): number {\n const inverted = 1 - l;\n return Math.max(0.35, Math.min(0.75, inverted));\n}\n\n/**\n * Derive a dark-mode ThemeColorInput from its light-mode counterpart.\n */\nexport function deriveDarkVariant(\n name: SemanticColorName,\n light: ThemeColorInput,\n): ThemeColorInput {\n const config = DARK_DERIVATION_CONFIG[name];\n const chromaScale = config.chromaScale ?? 1;\n\n const baseLightness =\n config.baseLightness === \"invert\"\n ? invertLightness(light.base.oklch.l ?? 0)\n : config.baseLightness;\n\n const fgLightness =\n config.fgLightness === \"invert\"\n ? invertLightness(light.foreground.oklch.l ?? 0)\n : config.fgLightness;\n\n return {\n base: new Color(\"oklch\", [\n baseLightness,\n (light.base.oklch.c || 0) * chromaScale,\n light.base.oklch.h || 0,\n ]),\n foreground: new Color(\"oklch\", [\n fgLightness,\n (light.foreground.oklch.c || 0) * chromaScale,\n light.foreground.oklch.h || 0,\n ]),\n };\n}\n\n// ── Dark Mode Merge ─────────────────────────────────────────────────\n\n/**\n * Merge auto-derived dark colors with any user-specified overrides.\n * For each semantic color, if the user has fully overridden both base and\n * foreground those are used; otherwise the missing channels are derived.\n */\nexport function mergeDarkOverrides(\n def: ThemeDefinition,\n): Record<SemanticColorName, ThemeColorInput> {\n const darkColors = {} as Record<SemanticColorName, ThemeColorInput>;\n\n for (const name of SEMANTIC_COLOR_NAMES) {\n const lightInput = def.light[name];\n const darkOverride = def.dark[name];\n\n if (darkOverride?.base && darkOverride?.foreground) {\n darkColors[name] = darkOverride as ThemeColorInput;\n } else if (darkOverride) {\n const base =\n darkOverride.base ?? deriveDarkVariant(name, lightInput).base;\n darkColors[name] = {\n base: base,\n foreground:\n darkOverride.foreground ??\n getForegroundColor(def.light.foreground.base, base),\n };\n } else {\n darkColors[name] = deriveDarkVariant(name, lightInput);\n }\n }\n\n return darkColors;\n}\n\n// ── Theme Resolution ────────────────────────────────────────────────\n\nfunction resolveColorSet(\n colors: Record<SemanticColorName, ThemeColorInput>,\n): ResolvedColorSet {\n const resolved = {} as ResolvedColorSet;\n\n for (const name of SEMANTIC_COLOR_NAMES) {\n const input = colors[name];\n const shades = generateShades(input.base);\n const resolvedShades = {} as Record<ShadeStep, Color>;\n\n for (const step of SHADE_STEPS) {\n resolvedShades[step] = shades[step];\n }\n\n resolved[name] = {\n base: input.base.clone(),\n foreground: input.foreground.clone(),\n shades: resolvedShades,\n };\n }\n\n return resolved;\n}\n\n/**\n * Resolve a ThemeDefinition into a complete ResolvedTheme.\n * Dark mode colors are derived from light where not overridden.\n */\nexport function resolveTheme(def: ThemeDefinition): ResolvedTheme {\n return {\n id: def.id,\n name: def.name,\n light: resolveColorSet(def.light),\n dark: resolveColorSet(mergeDarkOverrides(def)),\n fontSizes: { ...def.fontSizes },\n fontFamilies: { ...def.fontFamilies },\n spacing: def.spacing,\n radii: { ...def.radii },\n };\n}\n","import { SHADE_STEPS, type SemanticColorName, type ShadeStep } from \"./types\";\n\n/**\n * Specific overrides, otherwise all the overrides are generated using emitTailwindOverrides\n */\nconst OVERRIDES: Partial<Record<string, string>> = {\n \"--color-gray-50\": \"var(--color-muted)\",\n \"--color-gray-100\": \"var(--color-muted-600)\",\n \"--color-gray-200\": \"var(--color-border)\",\n} as const;\n\n/**\n * Returns the inverted shade for dark mode foreground colors.\n * In dark mode, light shades (50, 100) should map to dark values (950, 900) and vice versa.\n */\nfunction getInvertedStep(shade: ShadeStep): ShadeStep {\n const shadeIndex = SHADE_STEPS.indexOf(shade);\n const invertedIndex = SHADE_STEPS.length - 1 - shadeIndex;\n return SHADE_STEPS[invertedIndex] || 500;\n}\n\n/**\n * Map semantic colors to Tailwind built-in color names.\n */\nexport function emitTailwindOverrides(darkMode: boolean = false): string[] {\n const TAILWIND_COLOR_MAP: Record<string, SemanticColorName> = {\n gray: \"foreground\",\n red: \"destructive\",\n blue: \"primary\",\n green: \"accent\",\n };\n\n const TAILWIND_SHADES = [\n 50, 100, 200, 300, 400, 500, 600, 700, 800, 900, 950,\n ] as const;\n const SHADE_REMAP: Partial<Record<number, ShadeStep>> = {\n 50: 100,\n 950: 900,\n };\n\n const lines: string[] = [];\n for (const [twName, semantic] of Object.entries(TAILWIND_COLOR_MAP)) {\n for (const shade of TAILWIND_SHADES) {\n const step = (SHADE_REMAP[shade] ?? shade) as ShadeStep;\n const override = OVERRIDES[`--color-${twName}-${shade}`];\n lines.push(\n `--color-${twName}-${shade}: ${override ? override : `var(--color-${semantic}-${semantic === \"foreground\" && darkMode === true ? getInvertedStep(step) : step})`};`,\n );\n }\n }\n\n lines.push(\"--color-white: var(--color-background);\");\n lines.push(\"--color-black: var(--color-foreground);\");\n\n return lines;\n}\n","import { emitTailwindOverrides } from \"./tailwind-overrides\";\nimport {\n SEMANTIC_COLOR_NAMES,\n SHADE_STEPS,\n FONT_SIZE_KEYS,\n FONT_FAMILY_KEYS,\n RADIUS_KEYS,\n type ResolvedColorSet,\n type ResolvedTheme,\n} from \"./types\";\n\nfunction colorToCSS(color: import(\"colorjs.io\").default): string {\n const result = color.toString({ format: \"oklch\" });\n if (result.includes(\"NaN\")) {\n console.warn(\n \"[theme] colorToCSS produced NaN, using neutral fallback:\",\n result,\n );\n return \"oklch(0.5 0 0)\";\n }\n return result;\n}\n\nfunction camelToKebab(str: string): string {\n return str.replace(/([a-z])([A-Z])/g, \"$1-$2\").toLowerCase();\n}\n\n/**\n * Emit --color-{name}, --color-{name}-foreground, --color-{name}-{shade} vars.\n * Uses --color- prefix to match portal-widgets/tailwind.config.ts.\n */\nfunction emitColorVars(colors: ResolvedColorSet): string[] {\n const lines: string[] = [];\n\n for (const name of SEMANTIC_COLOR_NAMES) {\n const color = colors[name];\n lines.push(`--color-${name}: ${colorToCSS(color.base)};`);\n lines.push(`--color-${name}-foreground: ${colorToCSS(color.foreground)};`);\n for (const step of SHADE_STEPS) {\n lines.push(`--color-${name}-${step}: ${colorToCSS(color.shades[step])};`);\n }\n }\n\n return lines;\n}\n\n/**\n * Format a font family value for CSS output.\n * - If the value starts with \"var(\" (legacy), pass through as-is\n * - If the value already contains a comma (has fallback), pass through as-is\n * - Otherwise, wrap in quotes and append a generic sans-serif fallback\n */\nfunction formatFontFamily(value: string): string {\n if (value.startsWith(\"var(\")) return value;\n if (value.includes(\",\")) return value;\n return `'${value}', sans-serif`;\n}\n\n/**\n * Emit non-color CSS variables (font sizes, families, spacing, radii).\n */\nfunction emitNonColorVars(theme: ResolvedTheme): string[] {\n const lines: string[] = [];\n for (const key of FONT_SIZE_KEYS) {\n lines.push(`--font-size-${camelToKebab(key)}: ${theme.fontSizes[key]};`);\n }\n for (const key of FONT_FAMILY_KEYS) {\n lines.push(`--font-${key}: ${formatFontFamily(theme.fontFamilies[key])};`);\n }\n lines.push(`--spacing: ${theme.spacing};`);\n for (const key of RADIUS_KEYS) {\n lines.push(`--radius-${camelToKebab(key)}: ${theme.radii[key]};`);\n }\n return lines;\n}\n\n/**\n * Static CSS alias variables that bridge theme var names to Tailwind/component conventions.\n * These are always emitted and not mode-dependent.\n */\nconst globalCSSOverride = [\n \"--color-background-foreground: var(--color-foreground);\",\n \"--color-foreground-foreground: var(--color-background);\",\n \"--color-contrast: var(--color-foreground);\",\n ...SEMANTIC_COLOR_NAMES.map((value) => `--${value}: var(--color-${value});`),\n ...SEMANTIC_COLOR_NAMES.map(\n (value) => `--${value}-foreground: var(--color-${value}-foreground);`,\n ),\n\n \"--sidebar-ring: var(--color-primary);\",\n \"--sidebar-border: var(--color-border);\",\n \"--sidebar-accent-foreground: var(--color-accent-foreground);\",\n \"--sidebar-accent: var(--color-accent);\",\n \"--sidebar-primary-foreground: var(--color-primary-foreground);\",\n \"--sidebar-primary: var(--color-primary);\",\n \"--sidebar-foreground: var(--color-muted-foreground);\",\n \"--sidebar: var(--color-muted);\",\n \"--border: var(--color-background-600);\",\n \"--ring: var(--color-primary);\",\n \"--popover: var(--color-background);\",\n \"--popover-foreground: var(--color-foreground);\",\n \"--card: var(--color-muted);\",\n \"--card-foreground: var(--color-muted-foreground);\",\n\n \"--radius-sm: var(--radius-small);\",\n \"--radius-md: var(--radius-medium);\",\n \"--radius-lg: var(--radius-large);\",\n \"--radius-xl: var(--radius-extra-large);\",\n \"--text-xs: var(--font-size-extra-small);\",\n \"--text-sm: var(--font-size-small);\",\n \"--text-base: var(--font-size-regular);\",\n \"--text-lg: var(--font-size-large);\",\n \"--text-xl: var(--font-size-extra-large);\",\n \"--text-2xl: var(--font-size-giant);\",\n];\n\n/**\n * Overrides for global tailwindcss for specifically dark mode.\n */\nconst globalDarkCSSOverride = [\"--border: var(--color-background-400);\"];\n\nexport interface GenerateThemeCSSOptions {\n /** Whether or not to allow prefers-color-scheme to choose the theme mode */\n disableAutoTheme?: boolean;\n /** Whether to emit Tailwind built-in color overrides (default true) */\n mapTailwindColors?: boolean;\n}\n\n/**\n * Generate a complete CSS string for a resolved theme.\n * Outputs 2–3 blocks: light default, dark explicit via `[data-theme-mode=\"dark\"]`,\n * and (unless `disableAutoTheme`) a `prefers-color-scheme: dark` media query block.\n */\nexport function generateThemeCSS(\n theme: ResolvedTheme,\n options: GenerateThemeCSSOptions = {},\n): string {\n const sel = `[data-theme=\"${theme.id}\"]`;\n const tw = options.mapTailwindColors ?? true;\n const blocks: string[] = [];\n\n // Light mode (default)\n blocks.push(`${sel} {`);\n blocks.push(...globalCSSOverride);\n blocks.push(...emitNonColorVars(theme));\n blocks.push(...emitColorVars(theme.light));\n if (tw) blocks.push(...emitTailwindOverrides());\n blocks.push(`}`);\n\n // Dark mode: explicit via attribute\n blocks.push(`${sel}[data-theme-mode=\"dark\"] {`);\n blocks.push(...globalDarkCSSOverride);\n blocks.push(...emitColorVars(theme.dark));\n if (tw) blocks.push(...emitTailwindOverrides(true));\n blocks.push(`}`);\n\n // Dark mode: auto via system preference\n if (!options.disableAutoTheme) {\n blocks.push(`@media (prefers-color-scheme: dark) {`);\n blocks.push(`${sel}:not([data-theme-mode]) {`);\n blocks.push(...globalDarkCSSOverride);\n blocks.push(...emitColorVars(theme.dark).map((l) => `${l}`));\n if (tw) blocks.push(...emitTailwindOverrides(true).map((l) => `${l}`));\n blocks.push(`}`);\n blocks.push(`}`);\n }\n\n return blocks.join(\"\\n\");\n}\n","import Color from \"colorjs.io\";\nimport type {\n FontSizeKey,\n FontFamilyKey,\n RadiusKey,\n ThemeDefinition,\n} from \"./types\";\nimport { getForegroundColor } from \"./color-engine\";\n\n// ── Non-color defaults ──────────────────────────────────────────────\n\nexport const DEFAULT_FONT_SIZES: Record<FontSizeKey, string> = {\n extraSmall: \"0.75rem\",\n small: \"0.875rem\",\n regular: \"1rem\",\n large: \"1.125rem\",\n extraLarge: \"1.25rem\",\n giant: \"1.5rem\",\n};\n\nexport const DEFAULT_FONT_FAMILIES: Record<FontFamilyKey, string> = {\n header: \"Inter\",\n body: \"Inter\",\n};\n\nexport const DEFAULT_SPACING = \"0.25rem\";\n\nexport const DEFAULT_RADII: Record<RadiusKey, string> = {\n small: \"0.25rem\",\n medium: \"0.5rem\",\n large: \"0.75rem\",\n extraLarge: \"1rem\",\n};\n\n// ── Default colors (hex) ────────────────────────────────────────────\n\nexport const DEFAULT_COLORS = {\n background: \"#ffffff\",\n foreground: \"#1a1a1a\",\n primary: \"#3b82f6\",\n secondary: \"#6b7280\",\n accent: \"#10b981\",\n muted: \"#f3f4f6\",\n destructive: \"#ef4444\",\n mutedForeground: \"#6b7280\",\n} as const;\n\n// ── Default theme identity ──────────────────────────────────────────\n\nexport const DEFAULT_THEME_ID = \"default\";\nexport const DEFAULT_THEME_NAME = \"Default Theme\";\n\n// ── Factory ─────────────────────────────────────────────────────────\n\n/**\n * Build a fresh ThemeDefinition populated with all defaults.\n * Returns a new object each call because Color instances are mutable — do not cache the result.\n */\nexport function getDefaultThemeDefinition(): ThemeDefinition {\n const bg = new Color(DEFAULT_COLORS.background);\n const fg = new Color(DEFAULT_COLORS.foreground);\n const primary = new Color(DEFAULT_COLORS.primary);\n const secondary = new Color(DEFAULT_COLORS.secondary);\n const accent = new Color(DEFAULT_COLORS.accent);\n const muted = new Color(DEFAULT_COLORS.muted);\n const destructive = new Color(DEFAULT_COLORS.destructive);\n const mutedFg = new Color(DEFAULT_COLORS.mutedForeground);\n\n const darkBg = new Color(\"#0a0a0a\");\n const darkFg = new Color(\"#fafafa\");\n const darkMuted = new Color(\"#171717\");\n const darkMutedForeground = new Color(\"#dddddd\");\n\n return {\n id: DEFAULT_THEME_ID,\n name: DEFAULT_THEME_NAME,\n light: {\n background: { base: bg, foreground: fg },\n foreground: { base: fg, foreground: bg },\n primary: {\n base: primary,\n foreground: getForegroundColor(fg, primary),\n },\n secondary: {\n base: secondary,\n foreground: getForegroundColor(fg, secondary),\n },\n accent: {\n base: accent,\n foreground: getForegroundColor(fg, accent),\n },\n muted: { base: muted, foreground: mutedFg },\n destructive: {\n base: destructive,\n foreground: getForegroundColor(fg, destructive),\n },\n },\n dark: {\n background: { base: darkBg, foreground: darkFg },\n foreground: { base: darkFg, foreground: darkBg },\n muted: { base: darkMuted, foreground: darkMutedForeground },\n },\n fontSizes: { ...DEFAULT_FONT_SIZES },\n fontFamilies: { ...DEFAULT_FONT_FAMILIES },\n spacing: DEFAULT_SPACING,\n radii: { ...DEFAULT_RADII },\n };\n}\n","import Color from \"colorjs.io\";\nimport {\n SEMANTIC_COLOR_NAMES,\n type SemanticColorName,\n type ThemeColorInput,\n type ThemeDefinition,\n type ThemePayload,\n type OklchPlain,\n type FontSizeKey,\n type FontFamilyKey,\n type RadiusKey,\n} from \"./types\";\nimport {\n DEFAULT_FONT_SIZES,\n DEFAULT_FONT_FAMILIES,\n DEFAULT_SPACING,\n DEFAULT_RADII,\n getDefaultThemeDefinition,\n} from \"./defaults\";\n\nfunction colorToPlain(color: Color): OklchPlain {\n return {\n l: color.oklch.l ?? 0,\n c: color.oklch.c ?? 0,\n h: color.oklch.h ?? 0,\n };\n}\n\nfunction plainToColor(plain: OklchPlain): Color {\n return new Color(\"oklch\", [plain.l, plain.c, plain.h]);\n}\n\n/**\n * Serialise a ThemeDefinition (with Color objects) to a plain JSON payload\n * suitable for backend storage.\n */\nexport function serialiseTheme(def: ThemeDefinition): ThemePayload {\n const light = {} as ThemePayload[\"light\"];\n for (const name of SEMANTIC_COLOR_NAMES) {\n light[name] = {\n base: colorToPlain(def.light[name].base),\n foreground: colorToPlain(def.light[name].foreground),\n };\n }\n\n const dark: ThemePayload[\"dark\"] = {};\n for (const [name, value] of Object.entries(def.dark)) {\n if (!value) continue;\n dark[name as SemanticColorName] = {\n ...(value.base ? { base: colorToPlain(value.base) } : {}),\n ...(value.foreground\n ? { foreground: colorToPlain(value.foreground) }\n : {}),\n };\n }\n\n return {\n id: def.id,\n name: def.name,\n light,\n dark,\n fontSizes: { ...def.fontSizes },\n fontFamilies: { ...def.fontFamilies },\n spacing: def.spacing,\n radii: { ...def.radii },\n ...(def.syncWithBrandColors ? { syncWithBrandColors: true } : {}),\n };\n}\n\n/**\n * Deserialise a backend payload into a ThemeDefinition with Color objects.\n * Accepts `Record<string, unknown>` because API data is untyped at the boundary.\n * Falls back to default colors for any missing light-mode entries.\n */\nexport function deserialiseTheme(\n payload: Record<string, unknown>,\n): ThemeDefinition {\n const lightRaw = ((payload.light as Record<string, unknown> | undefined) ??\n {}) as Record<string, { base: OklchPlain; foreground: OklchPlain }>;\n const darkRaw = ((payload.dark as Record<string, unknown> | undefined) ??\n {}) as Record<string, { base?: OklchPlain; foreground?: OklchPlain }>;\n\n const defaults = getDefaultThemeDefinition();\n const light = {} as Record<SemanticColorName, ThemeColorInput>;\n for (const name of SEMANTIC_COLOR_NAMES) {\n const entry = lightRaw[name];\n if (entry) {\n light[name] = {\n base: plainToColor(entry.base),\n foreground: plainToColor(entry.foreground),\n };\n } else {\n console.warn(\n `[theme] deserialiseTheme: missing light color \"${name}\", using default`,\n );\n light[name] = defaults.light[name];\n }\n }\n\n const dark: Partial<Record<SemanticColorName, Partial<ThemeColorInput>>> = {};\n for (const [name, value] of Object.entries(darkRaw)) {\n if (!value) continue;\n dark[name as SemanticColorName] = {\n ...(value.base ? { base: plainToColor(value.base) } : {}),\n ...(value.foreground\n ? { foreground: plainToColor(value.foreground) }\n : {}),\n };\n }\n\n return {\n id: payload.id as string,\n name: payload.name as string,\n light,\n dark,\n fontSizes: (payload.fontSizes ?? DEFAULT_FONT_SIZES) as Record<\n FontSizeKey,\n string\n >,\n fontFamilies: (payload.fontFamilies ?? DEFAULT_FONT_FAMILIES) as Record<\n FontFamilyKey,\n string\n >,\n spacing: (payload.spacing as string) ?? DEFAULT_SPACING,\n radii: (payload.radii ?? DEFAULT_RADII) as Record<RadiusKey, string>,\n ...(payload.syncWithBrandColors === true\n ? { syncWithBrandColors: true }\n : {}),\n };\n}\n","/**\n * Theme Transforms\n * Convert raw API theme objects to ThemeDefinition format.\n * Handles both new structured format (OKLCH) and legacy flat format (hex strings).\n */\n\nimport type { ThemeDefinition } from \"./types\";\nimport { deserialiseTheme } from \"./serialisation\";\nimport { parseColor, getForegroundColor } from \"./color-engine\";\nimport {\n DEFAULT_COLORS,\n DEFAULT_FONT_SIZES,\n DEFAULT_FONT_FAMILIES,\n DEFAULT_SPACING,\n DEFAULT_RADII,\n} from \"./defaults\";\n\n/** Shape of a raw theme from the FluidOS API */\nexport interface RawApiTheme {\n id: number;\n config?: Record<string, unknown> | null;\n active?: boolean | null;\n name?: string | null;\n}\n\n/**\n * Check if a theme config uses the new structured format (has a `light` key\n * that is an object) vs the legacy flat format.\n */\nfunction isNewThemeFormat(config: Record<string, unknown>): boolean {\n return config.light != null && typeof config.light === \"object\";\n}\n\n/**\n * Convert a legacy flat config to a ThemeDefinition.\n * Legacy format: { base: \"#fff\", text: \"#000\", primary: \"oklch(0.6 0.2 250)\", ... }\n */\nfunction legacyConfigToDefinition(\n id: number,\n name: string,\n config: Record<string, string>,\n): ThemeDefinition {\n const bg = parseColor(\n config.base ?? config.background ?? DEFAULT_COLORS.background,\n );\n const fg = parseColor(\n config.text ?? config.foreground ?? DEFAULT_COLORS.foreground,\n );\n const primary = parseColor(config.primary ?? DEFAULT_COLORS.primary);\n const secondary = parseColor(config.secondary ?? DEFAULT_COLORS.secondary);\n const accent = parseColor(config.accent ?? DEFAULT_COLORS.accent);\n const muted = parseColor(config.muted ?? DEFAULT_COLORS.muted);\n const destructive = parseColor(\n config.destructive ?? DEFAULT_COLORS.destructive,\n );\n const mutedFg = parseColor(\n config.mutedForeground ?? DEFAULT_COLORS.mutedForeground,\n );\n\n return {\n id: String(id),\n name,\n light: {\n background: { base: bg, foreground: fg },\n foreground: { base: fg, foreground: bg },\n primary: {\n base: primary,\n foreground: getForegroundColor(fg, primary),\n },\n secondary: {\n base: secondary,\n foreground: getForegroundColor(fg, secondary),\n },\n accent: {\n base: accent,\n foreground: getForegroundColor(fg, accent),\n },\n muted: { base: muted, foreground: mutedFg },\n destructive: {\n base: destructive,\n foreground: getForegroundColor(fg, destructive),\n },\n },\n dark: {},\n fontSizes: {\n extraSmall: config.extraSmall ?? DEFAULT_FONT_SIZES.extraSmall,\n small: config.small ?? DEFAULT_FONT_SIZES.small,\n regular: config.regular ?? DEFAULT_FONT_SIZES.regular,\n large: config.large ?? DEFAULT_FONT_SIZES.large,\n extraLarge: config.extraLarge ?? DEFAULT_FONT_SIZES.extraLarge,\n giant: config.giant ?? DEFAULT_FONT_SIZES.giant,\n },\n fontFamilies: {\n header: config.headerFont ?? DEFAULT_FONT_FAMILIES.header,\n body: config.bodyFont ?? DEFAULT_FONT_FAMILIES.body,\n },\n spacing: config.globalSpacing ?? DEFAULT_SPACING,\n radii: {\n small: config.radiusSmall ?? DEFAULT_RADII.small,\n medium: config.radiusMedium ?? DEFAULT_RADII.medium,\n large: config.radiusLarge ?? DEFAULT_RADII.large,\n extraLarge: config.radiusExtraLarge ?? DEFAULT_RADII.extraLarge,\n },\n };\n}\n\n/**\n * Build a ThemeDefinition from a single API theme object.\n * Handles both new structured format and legacy flat format.\n */\nexport function buildThemeDefinition(theme: RawApiTheme): ThemeDefinition {\n const config = (theme.config ?? {}) as Record<string, unknown>;\n\n if (isNewThemeFormat(config)) {\n return deserialiseTheme({\n ...config,\n id: String(theme.id),\n name: theme.name ?? \"Untitled Theme\",\n });\n }\n\n return legacyConfigToDefinition(\n theme.id,\n theme.name ?? \"Untitled Theme\",\n config as Record<string, string>,\n );\n}\n\n/**\n * Transform raw API themes to ThemeDefinition[].\n * Catches and logs errors per theme (graceful degradation).\n */\nexport function transformThemes(themes: RawApiTheme[]): ThemeDefinition[] {\n return themes.flatMap((theme) => {\n try {\n return [buildThemeDefinition(theme)];\n } catch (error) {\n console.error(`[theme] Failed to build theme id=${theme.id}:`, error);\n return [];\n }\n });\n}\n\n/**\n * Get the active theme ID from a list of raw API themes.\n * Falls back to the first theme if none is marked active.\n */\nexport function getActiveThemeId(themes: RawApiTheme[]): string | undefined {\n const active = themes.find((t) => t.active) ?? themes[0];\n return active ? String(active.id) : undefined;\n}\n","import { generateThemeCSS } from \"./css-generator\";\nimport { FONT_FAMILY_KEYS, type ResolvedTheme } from \"./types\";\nimport type { GenerateThemeCSSOptions } from \"./css-generator\";\n\nconst STYLE_PREFIX = \"theme-style-\";\nconst FONT_LINK_PREFIX = \"theme-font-\";\n\nconst SYSTEM_FONTS = new Set([\n \"sans-serif\",\n \"serif\",\n \"monospace\",\n \"cursive\",\n \"fantasy\",\n \"system-ui\",\n \"ui-sans-serif\",\n \"ui-serif\",\n \"ui-monospace\",\n]);\n\n/** Build a Google Fonts CSS2 URL for a given font family with all weights. */\nfunction buildGoogleFontUrl(family: string): string {\n const encoded = encodeURIComponent(family).replace(/%20/g, \"+\");\n return `https://fonts.googleapis.com/css2?family=${encoded}:wght@100;200;300;400;500;600;700;800;900&display=swap`;\n}\n\n/** Check if a font family value needs to be loaded (i.e. is not a CSS var or system font). */\nfunction isLoadableFont(value: string): boolean {\n if (!value) return false;\n if (value.startsWith(\"var(\")) return false;\n return !SYSTEM_FONTS.has(value.toLowerCase());\n}\n\n/** Deterministic link element ID for a font family. */\nfunction fontLinkId(family: string): string {\n return `${FONT_LINK_PREFIX}${family.replace(/\\s+/g, \"-\").toLowerCase()}`;\n}\n\n/**\n * Inject or update `<link>` elements for Google Fonts used by the theme.\n * Removes links for fonts that are no longer referenced.\n */\nfunction loadThemeFonts(theme: ResolvedTheme): void {\n if (typeof document === \"undefined\") return;\n\n const fontsToLoad = new Set<string>();\n for (const key of FONT_FAMILY_KEYS) {\n const value = theme.fontFamilies[key];\n if (isLoadableFont(value)) {\n fontsToLoad.add(value);\n }\n }\n\n // Remove stale font links owned by this theme\n const existingLinks = document.querySelectorAll(\n `link[id^=\"${FONT_LINK_PREFIX}\"]`,\n );\n existingLinks.forEach((link) => {\n const owners = link.getAttribute(\"data-font-theme-ids\")?.split(\",\") ?? [];\n if (!owners.includes(theme.id)) return;\n const fontName = link.getAttribute(\"data-font-family\");\n if (fontName && !fontsToLoad.has(fontName)) {\n const remaining = owners.filter((id) => id !== theme.id);\n if (remaining.length === 0) {\n link.remove();\n } else {\n link.setAttribute(\"data-font-theme-ids\", remaining.join(\",\"));\n }\n }\n });\n\n // Add new font links (or register this theme as an owner of existing ones)\n for (const family of fontsToLoad) {\n const id = fontLinkId(family);\n const existing = document.getElementById(id);\n if (existing) {\n const owners =\n existing.getAttribute(\"data-font-theme-ids\")?.split(\",\") ?? [];\n if (!owners.includes(theme.id)) {\n existing.setAttribute(\n \"data-font-theme-ids\",\n [...owners, theme.id].join(\",\"),\n );\n }\n } else {\n const link = document.createElement(\"link\");\n link.id = id;\n link.rel = \"stylesheet\";\n link.href = buildGoogleFontUrl(family);\n link.setAttribute(\"data-font-family\", family);\n link.setAttribute(\"data-font-theme-ids\", theme.id);\n document.head.appendChild(link);\n }\n }\n}\n\n/** Remove all font `<link>` elements injected by the theme system. */\nfunction removeAllFontLinks(): void {\n if (typeof document === \"undefined\") return;\n document\n .querySelectorAll(`link[id^=\"${FONT_LINK_PREFIX}\"]`)\n .forEach((el) => el.remove());\n}\n\n/**\n * Inject or update a `<style>` element in `<head>` for the given theme.\n * The element ID is deterministic (`theme-style-{themeId}`) so repeated calls\n * for the same theme are idempotent — the existing element is updated in place.\n * Also loads Google Fonts referenced by the theme's font families.\n * No-op when `document` is unavailable (SSR).\n */\nexport function applyTheme(\n theme: ResolvedTheme,\n options?: GenerateThemeCSSOptions,\n): void {\n if (typeof document === \"undefined\") return;\n\n try {\n loadThemeFonts(theme);\n\n const styleId = `${STYLE_PREFIX}${theme.id}`;\n let el = document.getElementById(styleId) as HTMLStyleElement | null;\n\n if (!el) {\n el = document.createElement(\"style\");\n el.id = styleId;\n document.head.appendChild(el);\n }\n\n el.textContent = generateThemeCSS(theme, options);\n } catch (error) {\n console.error(`[theme] applyTheme failed for \"${theme.id}\":`, error);\n }\n}\n\n/** Remove an injected theme stylesheet and clean up font link ownership. No-op during SSR. */\nexport function removeTheme(themeId: string): void {\n if (typeof document === \"undefined\") return;\n document.getElementById(`${STYLE_PREFIX}${themeId}`)?.remove();\n\n // Remove this theme from font link ownership; delete links with no remaining owners\n document\n .querySelectorAll(`link[id^=\"${FONT_LINK_PREFIX}\"]`)\n .forEach((link) => {\n const owners = link.getAttribute(\"data-font-theme-ids\")?.split(\",\") ?? [];\n const remaining = owners.filter((id) => id !== themeId);\n if (remaining.length === 0) {\n link.remove();\n } else {\n link.setAttribute(\"data-font-theme-ids\", remaining.join(\",\"));\n }\n });\n}\n\n/** Remove all injected theme stylesheets and font links. No-op during SSR. */\nexport function removeAllThemes(): void {\n if (typeof document === \"undefined\") return;\n document\n .querySelectorAll(`style[id^=\"${STYLE_PREFIX}\"]`)\n .forEach((el) => el.remove());\n removeAllFontLinks();\n}\n","/**\n * Screen Transforms\n * Convert FluidOS API screen objects to ScreenDefinition format.\n *\n * Extracted from:\n * - apps/fluid-admin/networking/app-builder/app-screens/types.ts\n * - apps/fluid-admin/networking/reps/screens.api.ts\n */\n\nimport type { WidgetSchema } from \"@fluid-app/portal-core/types\";\nimport type { ScreenDefinition } from \"@fluid-app/portal-core/types\";\n\n/** Raw screen from the FluidOS API */\nexport interface RawApiScreen {\n id: number | string;\n definition_id?: number | string;\n name?: string | null;\n slug?: string | null;\n component_tree?: unknown;\n}\n\n/**\n * Normalize component_tree to always be an array.\n * The API stores component_tree as a hash (object), but the frontend expects an array.\n */\nexport function normalizeComponentTree(componentTree: unknown): WidgetSchema[] {\n if (!componentTree) return [];\n if (Array.isArray(componentTree)) return componentTree as WidgetSchema[];\n if (typeof componentTree === \"object\") {\n return [componentTree as WidgetSchema];\n }\n return [];\n}\n\n/**\n * Convert a raw FluidOS screen to ScreenDefinition.\n * Normalizes component_tree and converts string IDs to numbers.\n */\nexport function toScreenDefinition(screen: RawApiScreen): ScreenDefinition {\n return {\n id: Number(screen.id),\n slug: screen.slug ?? \"\",\n name: screen.name ?? \"\",\n component_tree: normalizeComponentTree(screen.component_tree),\n };\n}\n","/**\n * Navigation Transforms\n * Convert FluidOS API navigation items to NavigationItem format.\n *\n * Extracted from:\n * - apps/fluid-admin/networking/app-builder/app-navigation-items/types.ts\n * - apps/fluid-admin/networking/reps/screens.api.ts\n */\n\nimport type { NavigationItem } from \"@fluid-app/portal-core/types\";\nimport type { FluidOsApiNavigationItem } from \"@fluid-app/portal-core/fluidos-api\";\n\n/** Raw navigation item from the FluidOS API. */\nexport type RawApiNavigationItem = FluidOsApiNavigationItem;\n\n/**\n * Convert a raw FluidOS navigation item to NavigationItem.\n * Recursively transforms children and sorts by position.\n */\nexport function toNavigationItem(item: RawApiNavigationItem): NavigationItem {\n const children = (item.children ?? [])\n .map(toNavigationItem)\n .sort((a, b) => (a.position ?? 0) - (b.position ?? 0));\n\n return {\n id: Number(item.id),\n label: item.label ?? \"Untitled\",\n // Use conditional spread for optional properties (exactOptionalPropertyTypes)\n ...(item.slug != null ? { slug: String(item.slug) } : {}),\n ...(item.icon != null ? { icon: String(item.icon) } : {}),\n ...(item.screen_id != null ? { screen_id: Number(item.screen_id) } : {}),\n ...(item.parent_id != null ? { parent_id: Number(item.parent_id) } : {}),\n ...(item.source != null ? { source: item.source } : {}),\n position: item.position ?? 0,\n children,\n };\n}\n","/**\n * Data Transforms\n * Convert FluidOS API responses to RepAppData format.\n *\n * This is the main entry point for all transforms used by the SDK client\n * and hooks when fetching from the fluidos API.\n */\n\nexport {\n transformThemes,\n buildThemeDefinition,\n getActiveThemeId,\n type RawApiTheme,\n} from \"@fluid-app/portal-core/theme\";\n\nexport {\n normalizeComponentTree,\n toScreenDefinition,\n type RawApiScreen,\n} from \"./screen-transforms\";\n\nexport {\n toNavigationItem,\n type RawApiNavigationItem,\n} from \"./navigation-transforms\";\n\nimport type { RepAppData, RepAppManifest } from \"@fluid-app/portal-core/types\";\nimport {\n transformThemes,\n getActiveThemeId,\n type RawApiTheme,\n} from \"@fluid-app/portal-core/theme\";\nimport { toScreenDefinition, type RawApiScreen } from \"./screen-transforms\";\nimport {\n toNavigationItem,\n type RawApiNavigationItem,\n} from \"./navigation-transforms\";\n\n/** Raw manifest shape from the `/fluid_os/definitions/active` API response */\nexport interface RawManifestResponse {\n manifest: {\n definition_id: number;\n published_version?: number;\n screens?: RawApiScreen[];\n profile?: {\n name?: string;\n definition_id: number;\n themes?: RawApiTheme[];\n navigation?: {\n id: number;\n name?: string;\n definition_id: number;\n navigation_items?: RawApiNavigationItem[];\n };\n mobile_navigation?: {\n id: number;\n name?: string;\n definition_id: number;\n navigation_items?: RawApiNavigationItem[];\n };\n };\n };\n}\n\n/**\n * Convert a raw FluidOS API response to RawManifestResponse.\n *\n * The wire format includes `navigation_items` and full theme objects,\n * but the generated OpenAPI types differ (e.g. `navigation_tree`, `themes: number[]`).\n * This function bridges that gap so callers avoid `as unknown as` casts.\n */\nexport function toRawManifest<T extends { manifest?: object }>(\n raw: T,\n): RawManifestResponse {\n if (!raw.manifest) {\n throw new Error(\"FluidOS API returned empty manifest\");\n }\n // Safe: the wire format matches RawManifestResponse; the generated\n // types just model the schema less precisely for this endpoint.\n return raw as unknown as RawManifestResponse;\n}\n\n/**\n * Transform a raw FluidOS manifest API response into RepAppData.\n *\n * This is the top-level transform used by `FluidClient.app.get()`.\n * It handles:\n * - Theme transformation (legacy and new formats)\n * - Screen normalization (component_tree array wrapping)\n * - Navigation item transformation (recursive with position sorting)\n */\nexport function transformManifestToRepAppData(\n response: RawManifestResponse,\n): RepAppData {\n const manifest = response.manifest;\n const rawProfile = manifest.profile;\n\n const rawThemes: RawApiTheme[] = Array.isArray(rawProfile?.themes)\n ? rawProfile.themes\n : [];\n\n const screens = (manifest.screens ?? []).map((screen) =>\n toScreenDefinition(screen),\n );\n\n const navigationItems = (rawProfile?.navigation?.navigation_items ?? []).map(\n toNavigationItem,\n );\n\n const nav = rawProfile?.navigation;\n const mobileNav = rawProfile?.mobile_navigation;\n const mobileNavigationItems = (mobileNav?.navigation_items ?? []).map(\n toNavigationItem,\n );\n const activeThemeId = getActiveThemeId(rawThemes);\n\n return {\n definition_id: manifest.definition_id,\n published_version: manifest.published_version ?? 0,\n screens,\n profile: {\n name: rawProfile?.name ?? \"Default\",\n definition_id: rawProfile?.definition_id ?? manifest.definition_id,\n themes: transformThemes(rawThemes),\n // Conditional spread for exactOptionalPropertyTypes compliance\n ...(activeThemeId !== undefined ? { activeThemeId } : {}),\n navigation: {\n definition_id: nav?.definition_id ?? manifest.definition_id,\n id: nav?.id ?? 0,\n name: nav?.name ?? \"Main Navigation\",\n navigation_items: navigationItems,\n screens,\n },\n ...(mobileNav\n ? {\n mobile_navigation: {\n definition_id: mobileNav.definition_id ?? manifest.definition_id,\n id: mobileNav.id ?? 0,\n name: mobileNav.name ?? \"Mobile Navigation\",\n navigation_items: mobileNavigationItems,\n screens,\n },\n }\n : {}),\n },\n };\n}\n\n/**\n * Transform a raw manifest response, unwrapping the `{ manifest }` envelope.\n * Convenience wrapper matching the `RepAppManifest` type.\n */\nexport function transformRawManifest(\n response: RawManifestResponse,\n): RepAppManifest {\n return {\n manifest: transformManifestToRepAppData(response),\n };\n}\n","/**\n * Fluid API Client\n * Adapted from: packages/fluidos-api-client/src/lib/fetch-client.ts\n * Provides authenticated API access with domain-specific methods\n */\n\nimport type { Rep, UpdateRepData } from \"../types/rep\";\nimport type { UserPermissions } from \"../types/permissions\";\nimport type { RepAppData } from \"@fluid-app/portal-core/types\";\nimport {\n createFetchClient,\n type FetchClient,\n} from \"@fluid-app/api-client-core\";\nimport { fluidOs } from \"@fluid-app/fluidos-api-client\";\nimport {\n HTTP_METHODS,\n type FluidSDKConfig,\n type RequestOptions,\n} from \"./types\";\nimport {\n transformManifestToRepAppData,\n toRawManifest,\n type RawManifestResponse,\n} from \"../transforms\";\n\n/**\n * API Error class for structured error handling\n */\nexport class ApiError extends Error {\n readonly status: number;\n readonly data: unknown;\n\n constructor(message: string, status: number, data?: unknown) {\n super(message);\n this.name = \"ApiError\";\n this.status = status;\n this.data = data;\n\n // V8-specific stack trace capture (Node.js, Chrome)\n const errorWithCapture = Error as typeof Error & {\n captureStackTrace?: (\n target: object,\n constructor: typeof ApiError,\n ) => void;\n };\n if (errorWithCapture.captureStackTrace) {\n errorWithCapture.captureStackTrace(this, ApiError);\n }\n }\n\n toJSON(): { name: string; message: string; status: number; data: unknown } {\n return {\n name: this.name,\n message: this.message,\n status: this.status,\n data: this.data,\n };\n }\n}\n\n/**\n * Type guard for ApiError\n */\nexport function isApiError(error: unknown): error is ApiError {\n return error instanceof ApiError;\n}\n\n// ============================================================================\n// Discriminated Union for API Responses\n// ============================================================================\n\n/**\n * Discriminated union representing the result of an API call.\n * Use `isApiSuccess` and `isApiFailure` type guards to narrow.\n */\nexport type ApiResult<T> =\n | { readonly success: true; readonly data: T }\n | { readonly success: false; readonly error: ApiError };\n\n/**\n * Type guard for successful API result\n */\nexport function isApiSuccess<T>(\n result: ApiResult<T>,\n): result is { readonly success: true; readonly data: T } {\n return result.success === true;\n}\n\n/**\n * Type guard for failed API result\n */\nexport function isApiFailure<T>(\n result: ApiResult<T>,\n): result is { readonly success: false; readonly error: ApiError } {\n return result.success === false;\n}\n\n/**\n * Type guard to check if a value is a non-null string\n */\nfunction isString(value: unknown): value is string {\n return typeof value === \"string\";\n}\n\n/**\n * Extract error message from API response data using `in` operator narrowing.\n * Checks common error message field names in order of precedence.\n */\nfunction extractErrorMessage(\n data: Readonly<Record<string, unknown>>,\n fallback: string,\n): string {\n // Use `in` operator to narrow and then type guard to validate string type\n if (\"message\" in data && isString(data.message)) {\n return data.message;\n }\n if (\"error_message\" in data && isString(data.error_message)) {\n return data.error_message;\n }\n if (\"error\" in data && isString(data.error)) {\n return data.error;\n }\n return fallback;\n}\n\n// ============================================================================\n// Response Envelope\n// ============================================================================\n\n/**\n * Standard envelope returned by all portal tenant API responses.\n * The `data` key holds the actual resource(s).\n */\ninterface ApiEnvelope<T = unknown> {\n readonly status: number;\n readonly data: T;\n readonly meta: EnvelopeMeta;\n}\n\n/**\n * Metadata included in every API envelope response.\n */\nexport interface EnvelopeMeta {\n readonly request_id: string | null;\n readonly timestamp: string;\n readonly pagination?: CursorPagination;\n}\n\n/**\n * Cursor-based pagination metadata returned in the envelope `meta`.\n */\nexport interface CursorPagination {\n readonly cursor: string | null;\n readonly limit: number;\n readonly next_cursor: string | null;\n readonly prev_cursor: string | null;\n}\n\n/**\n * Result type for paginated list endpoints.\n * Includes both the unwrapped data and cursor-based pagination metadata.\n */\nexport interface PaginatedResult<T> {\n readonly data: T;\n readonly pagination: CursorPagination | undefined;\n}\n\n/**\n * Type guard to detect whether a parsed JSON value is an API envelope.\n * Envelopes always have numeric `status` and a `data` key.\n */\nfunction isApiEnvelope(value: unknown): value is ApiEnvelope {\n return (\n typeof value === \"object\" &&\n value !== null &&\n \"status\" in value &&\n typeof (value as Record<string, unknown>).status === \"number\" &&\n \"data\" in value\n );\n}\n\n// ============================================================================\n// Analytics Types\n// ============================================================================\n\n/** Minimal order shape returned by the analytics dashboard endpoint. */\nexport interface DashboardOrder {\n readonly id: string;\n readonly order_number: string;\n readonly status: string;\n readonly total: number;\n readonly created_at: string;\n}\n\nexport interface DashboardData {\n readonly total_sales: number;\n readonly total_orders: number;\n readonly total_customers: number;\n readonly recent_orders: readonly DashboardOrder[];\n}\n\nexport interface SalesParams {\n readonly date_from?: string;\n readonly date_to?: string;\n readonly group_by?: \"day\" | \"week\" | \"month\";\n}\n\nexport interface SalesData {\n readonly total: number;\n readonly data: readonly SalesDataPoint[];\n}\n\nexport interface SalesDataPoint {\n readonly date: string;\n readonly amount: number;\n readonly orders: number;\n}\n\n// ============================================================================\n// Client Return Type\n// ============================================================================\n\n/**\n * The shape returned by `createFluidClient`.\n * Generic helper methods preserve their type parameters so callers\n * can specify response types at each call site.\n */\nexport interface FluidClientReturn {\n readonly fetchClient: FetchClient;\n\n readonly request: <TResponse = unknown>(\n endpoint: string,\n options?: RequestOptions,\n ) => Promise<TResponse>;\n\n readonly requestNullable: <TResponse>(\n endpoint: string,\n options?: RequestOptions,\n ) => Promise<TResponse | null>;\n\n readonly safeRequest: <TResponse>(\n endpoint: string,\n options?: RequestOptions,\n ) => Promise<ApiResult<TResponse>>;\n\n readonly get: <\n TResponse = unknown,\n TParams extends object = Record<string, unknown>,\n >(\n endpoint: string,\n params?: TParams,\n options?: Omit<RequestOptions, \"method\" | \"params\">,\n ) => Promise<TResponse>;\n\n readonly post: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ) => Promise<TResponse>;\n\n readonly put: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ) => Promise<TResponse>;\n\n readonly patch: <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ) => Promise<TResponse>;\n\n readonly delete: <TResponse = unknown>(\n endpoint: string,\n options?: Omit<RequestOptions, \"method\">,\n ) => Promise<TResponse>;\n\n // TODO(portal-tenant): migrate to portal-tenant once BFF endpoint available\n readonly reps: {\n readonly current: () => Promise<Rep>;\n readonly updateProfile: (data: UpdateRepData) => Promise<Rep>;\n };\n\n readonly app: {\n readonly getRaw: () => Promise<RawManifestResponse>;\n readonly get: () => Promise<RepAppData>;\n };\n\n readonly permissions: {\n readonly get: () => Promise<UserPermissions>;\n };\n\n readonly analytics: {\n readonly dashboard: () => Promise<DashboardData>;\n readonly sales: (params?: SalesParams) => Promise<SalesData>;\n };\n}\n\n// ============================================================================\n// Client Implementation\n// ============================================================================\n\n/**\n * Creates a configured Fluid API client instance\n */\nexport function createFluidClient(config: FluidSDKConfig): FluidClientReturn {\n const { baseUrl, getAuthToken, onAuthError, defaultHeaders = {} } = config;\n\n // Create a fluidos FetchClient for structured API calls (manifest, etc.)\n // Conditional spread for exactOptionalPropertyTypes: only include getAuthToken when defined.\n // getAuthToken is still forwarded to the FetchClient for backward compatibility\n // with consumers that provide it (e.g., for non-BFF deployments).\n const fetchClient: FetchClient = createFetchClient({\n baseUrl,\n ...(getAuthToken ? { getAuthToken } : {}),\n onAuthError,\n defaultHeaders,\n credentials: \"include\",\n });\n\n /**\n * Build headers for a request.\n * Auth is handled by session cookies via `credentials: 'include'` on fetch calls.\n */\n function buildHeaders(\n customHeaders?: Record<string, string>,\n ): Record<string, string> {\n return {\n \"Content-Type\": \"application/json\",\n ...defaultHeaders,\n ...customHeaders,\n };\n }\n\n /**\n * Build URL with query parameters (Rails-compatible)\n */\n function buildUrl(\n endpoint: string,\n params?: Record<string, unknown>,\n ): string {\n // Construct URL by concatenating baseUrl + endpoint\n // Using URL constructor with relative paths would strip the baseUrl path (e.g., /api)\n const normalizedBase = baseUrl.endsWith(\"/\")\n ? baseUrl.slice(0, -1)\n : baseUrl;\n const normalizedEndpoint = endpoint.startsWith(\"/\")\n ? endpoint\n : `/${endpoint}`;\n\n // When baseUrl is empty (same-origin), use window.location.origin as the\n // URL base so that `new URL()` can resolve the relative path correctly.\n const url = normalizedBase\n ? new URL(normalizedBase + normalizedEndpoint)\n : new URL(\n normalizedEndpoint,\n typeof window !== \"undefined\"\n ? window.location.origin\n : \"http://localhost\",\n );\n\n if (params) {\n for (const [key, value] of Object.entries(params)) {\n if (value === undefined || value === null) {\n continue;\n }\n\n if (Array.isArray(value)) {\n // Handle arrays like Rails expects: key[]\n for (const item of value) {\n url.searchParams.append(`${key}[]`, String(item));\n }\n } else if (typeof value === \"object\") {\n // Handle nested objects: key[subkey]\n for (const [subKey, subValue] of Object.entries(\n value as Record<string, unknown>,\n )) {\n if (subValue === undefined || subValue === null) {\n continue;\n }\n\n if (Array.isArray(subValue)) {\n for (const item of subValue) {\n url.searchParams.append(`${key}[${subKey}][]`, String(item));\n }\n } else {\n url.searchParams.append(`${key}[${subKey}]`, String(subValue));\n }\n }\n } else {\n url.searchParams.append(key, String(value));\n }\n }\n }\n\n return url.toString();\n }\n\n /**\n * Default request options for type-safe defaults.\n * Uses `satisfies` to validate against RequestOptions while preserving literal types.\n */\n const defaultRequestOptions = {\n method: HTTP_METHODS.GET,\n } as const satisfies Partial<RequestOptions>;\n\n /**\n * Main request function\n */\n async function request<TResponse = unknown>(\n endpoint: string,\n options: RequestOptions = {},\n ): Promise<TResponse> {\n const {\n method = defaultRequestOptions.method,\n headers: customHeaders,\n params,\n body,\n signal,\n } = options;\n\n // Use buildUrl for all requests to ensure baseUrl path is preserved\n const url = buildUrl(\n endpoint,\n method === HTTP_METHODS.GET ? params : undefined,\n );\n\n const headers = buildHeaders(customHeaders);\n\n let response: Response;\n\n try {\n const fetchOptions: RequestInit = {\n method,\n headers,\n credentials: \"include\",\n };\n // Only add signal and body when defined to satisfy exactOptionalPropertyTypes\n // RequestInit expects signal: AbortSignal | null, not undefined\n if (signal !== undefined) {\n fetchOptions.signal = signal;\n }\n if (body && method !== HTTP_METHODS.GET) {\n fetchOptions.body = JSON.stringify(body);\n }\n response = await fetch(url, fetchOptions);\n } catch (networkError) {\n throw new ApiError(\n `Network error: ${networkError instanceof Error ? networkError.message : \"Unknown network error\"}`,\n 0,\n null,\n );\n }\n\n // Handle authentication errors — invoke the handler, then throw so\n // TanStack Query surfaces the error instead of caching `null` as data.\n if (response.status === 401) {\n onAuthError?.();\n throw new ApiError(\"Authentication required\", 401, null);\n }\n\n if (!response.ok) {\n try {\n const contentType = response.headers.get(\"content-type\");\n if (contentType?.includes(\"application/json\")) {\n const data = (await response.json()) as Record<string, unknown>;\n // Use `in` operator narrowing to safely extract error message\n const errorMessage = extractErrorMessage(\n data,\n `${method} request failed`,\n );\n throw new ApiError(\n errorMessage,\n response.status,\n \"errors\" in data ? data.errors : data,\n );\n } else {\n throw new ApiError(\n `${method} request failed with status ${response.status}`,\n response.status,\n null,\n );\n }\n } catch (error) {\n if (isApiError(error)) {\n throw error;\n }\n\n throw new ApiError(\n `${method} request failed with status ${response.status}`,\n response.status,\n null,\n );\n }\n }\n\n // Handle empty responses (204 No Content)\n // Note: Callers expecting nullable responses should use requestNullable<T>\n // which properly types the return as T | null\n if (\n response.status === 204 ||\n response.headers.get(\"content-length\") === \"0\"\n ) {\n // Type assertion required: 204 No Content has no body to parse.\n // This is safe when TResponse is a union including null (e.g., T | null).\n // For fully type-safe nullable handling, prefer requestNullable() or safeRequest().\n return null as TResponse;\n }\n\n try {\n const raw: unknown = await response.json();\n // Runtime check: verify we got an object/array, not a primitive that would be mistyped\n if (raw === null || raw === undefined) {\n throw new ApiError(\n \"Unexpected null/undefined in JSON response\",\n response.status,\n null,\n );\n }\n // Automatically unwrap the standard `{ status, data, meta }` envelope\n // used by portal tenant API responses. Non-envelope responses (e.g.\n // legacy or third-party endpoints) pass through unchanged.\n const unwrapped = isApiEnvelope(raw) ? raw.data : raw;\n // Type assertion required: JSON.parse returns `unknown`, but we've validated\n // the response is not null/undefined. The caller specifies TResponse based on\n // their knowledge of the API endpoint's response schema.\n return unwrapped as TResponse;\n } catch (parseError) {\n if (isApiError(parseError)) {\n throw parseError;\n }\n throw new ApiError(\n \"Failed to parse response as JSON\",\n response.status,\n null,\n );\n }\n }\n\n /**\n * Request function for endpoints that may return null (204 No Content).\n * Properly types the return as T | null.\n */\n async function requestNullable<TResponse>(\n endpoint: string,\n options: RequestOptions = {},\n ): Promise<TResponse | null> {\n return request<TResponse | null>(endpoint, options);\n }\n\n /**\n * Safe request wrapper that returns a discriminated union instead of throwing.\n * Use `isApiSuccess` or `isApiFailure` to narrow the result.\n */\n async function safeRequest<TResponse>(\n endpoint: string,\n options: RequestOptions = {},\n ): Promise<ApiResult<TResponse>> {\n try {\n const data = await request<TResponse>(endpoint, options);\n return { success: true, data };\n } catch (error) {\n if (isApiError(error)) {\n return { success: false, error };\n }\n // Wrap unknown errors in ApiError\n return {\n success: false,\n error: new ApiError(\n error instanceof Error ? error.message : \"Unknown error\",\n 0,\n null,\n ),\n };\n }\n }\n\n /**\n * Helper to safely convert typed params to Record<string, unknown>.\n * Type assertion required: TypeScript's structural typing allows any object\n * to be treated as Record<string, unknown> when we only need to iterate\n * over its entries. This is safe because buildUrl only reads properties.\n */\n function toParams<T extends object>(\n params: T | undefined,\n ): Record<string, unknown> | undefined {\n return params as Record<string, unknown> | undefined;\n }\n\n // Convenience HTTP methods using HTTP_METHODS constant for type safety\n const get = <\n TResponse = unknown,\n TParams extends object = Record<string, unknown>,\n >(\n endpoint: string,\n params?: TParams,\n options?: Omit<RequestOptions, \"method\" | \"params\">,\n ): Promise<TResponse> => {\n // Build request options object, only adding params if defined\n const baseOptions = {\n ...options,\n method: HTTP_METHODS.GET,\n } satisfies RequestOptions;\n\n const convertedParams = toParams(params);\n const requestOptions: RequestOptions =\n convertedParams !== undefined\n ? { ...baseOptions, params: convertedParams }\n : baseOptions;\n\n return request<TResponse>(endpoint, requestOptions);\n };\n\n const post = <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: HTTP_METHODS.POST,\n body,\n } satisfies RequestOptions);\n\n const put = <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: HTTP_METHODS.PUT,\n body,\n } satisfies RequestOptions);\n\n const patch = <TResponse = unknown>(\n endpoint: string,\n body?: unknown,\n options?: Omit<RequestOptions, \"method\" | \"body\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: HTTP_METHODS.PATCH,\n body,\n } satisfies RequestOptions);\n\n const del = <TResponse = unknown>(\n endpoint: string,\n options?: Omit<RequestOptions, \"method\">,\n ): Promise<TResponse> =>\n request<TResponse>(endpoint, {\n ...options,\n method: HTTP_METHODS.DELETE,\n } satisfies RequestOptions);\n\n // ============================================================================\n // Domain-Specific API Methods\n // ============================================================================\n\n return {\n // Fluidos FetchClient for structured API calls\n fetchClient: fetchClient,\n\n // Low-level methods for custom endpoints\n request: request,\n requestNullable: requestNullable,\n safeRequest: safeRequest,\n get: get,\n post: post,\n put: put,\n patch: patch,\n delete: del,\n\n // TODO(portal-tenant): migrate to portal-tenant once BFF endpoint available\n reps: {\n current: (): Promise<Rep> => get<Rep>(\"/api/reps/me\"),\n updateProfile: (data: UpdateRepData): Promise<Rep> =>\n patch<Rep>(\"/api/reps/me\", data),\n },\n\n // TODO(portal-tenant): replace with BFF app definition once manifest parity achieved\n app: {\n /** Fetch the raw manifest (plain JSON, no Color objects). Cache-safe. */\n getRaw: async (): Promise<RawManifestResponse> => {\n const raw = await fluidOs.getFluidOSManifest(fetchClient, {\n platform: \"browser\",\n });\n return toRawManifest(raw);\n },\n /** Fetch the active app definition with full theme/screen/navigation transforms */\n get: async (): Promise<RepAppData> => {\n const raw = await fluidOs.getFluidOSManifest(fetchClient, {\n platform: \"browser\",\n });\n return transformManifestToRepAppData(toRawManifest(raw));\n },\n },\n\n // TODO(portal-tenant): migrate to portal-tenant once BFF endpoint available\n permissions: {\n get: (): Promise<UserPermissions> =>\n get<UserPermissions>(\"/api/company/roles/my_permissions\"),\n },\n\n // TODO(portal-tenant): migrate to portal-tenant once BFF endpoint available\n analytics: {\n dashboard: (): Promise<DashboardData> =>\n get<DashboardData>(\"/api/analytics/dashboard\"),\n sales: (params?: SalesParams): Promise<SalesData> =>\n get<SalesData, SalesParams>(\"/api/analytics/sales\", params),\n },\n };\n}\n\nexport type FluidClient = ReturnType<typeof createFluidClient>;\n","import type { FluidOsReadApi } from \"@fluid-app/portal-core/fluidos-api\";\nimport type { FetchClient } from \"@fluid-app/api-client-core\";\nimport { fluidOs } from \"@fluid-app/fluidos-api-client\";\n\n/** Create a FluidOsReadApi adapter backed by a FetchClient. */\nexport function createFluidOsReadAdapter(client: FetchClient): FluidOsReadApi {\n return {\n getManifest: (params) => fluidOs.getFluidOSManifest(client, params),\n };\n}\n","import type { AppDefinitionApi } from \"@fluid-app/portal-core/app-definition-api\";\nimport type {\n AppDefinitionResponse,\n AppFluidOsDefinition,\n} from \"@fluid-app/portal-core/app-definition-types\";\nimport type { FetchClient } from \"@fluid-app/portal-tenant-api-client\";\nimport { portalTenant } from \"@fluid-app/portal-tenant-api-client\";\n\nfunction mapDefinition(\n raw: NonNullable<\n Awaited<ReturnType<typeof portalTenant.app_definition_show>>[\"definition\"]\n >,\n): AppFluidOsDefinition {\n return {\n id: raw.id ?? 0,\n name: raw.name ?? \"\",\n version: raw.version ?? null,\n components: raw.components ?? [],\n active: raw.active ?? false,\n } satisfies AppFluidOsDefinition;\n}\n\n/**\n * Creates an AppDefinitionApi adapter backed by the portal-tenant BFF client.\n *\n * Maps the generated portal-tenant `app_definition_show` response to the\n * `AppDefinitionApi` port, applying runtime defaults for optional BFF fields\n * so TypeScript catches schema drift at compile time via `satisfies`.\n */\nexport function createAppDefinitionApiAdapter(\n client: FetchClient,\n): AppDefinitionApi {\n return {\n fetchDefinition: async () => {\n const response = await portalTenant.app_definition_show(client);\n return {\n definition: mapDefinition(response.definition ?? {}),\n meta: {\n request_id: response.meta?.request_id ?? \"\",\n timestamp: response.meta?.timestamp ?? \"\",\n },\n } satisfies AppDefinitionResponse;\n },\n };\n}\n","import type { AccountApi } from \"@fluid-app/portal-core/account-api\";\nimport type {\n AccountMemberType,\n AccountRep,\n AccountResponse,\n UpdateAccountBody,\n} from \"@fluid-app/portal-core/account-types\";\nimport type { FetchClient } from \"@fluid-app/portal-tenant-api-client\";\nimport { portalTenant } from \"@fluid-app/portal-tenant-api-client\";\n\nfunction mapAccount(\n raw: NonNullable<\n Awaited<ReturnType<typeof portalTenant.account_show>>[\"account\"]\n >,\n): AccountRep {\n return {\n id: raw.id ?? 0,\n member_type: (raw.member_type ?? \"rep\") satisfies AccountMemberType,\n first_name: raw.first_name ?? \"\",\n last_name: raw.last_name ?? \"\",\n email: raw.email ?? \"\",\n phone: raw.phone ?? null,\n bio: raw.bio ?? null,\n avatar_url: raw.avatar_url ?? null,\n slug: raw.slug ?? \"\",\n social_links: (raw.social_links ?? null) as Record<string, string> | null,\n } satisfies AccountRep;\n}\n\nexport function createAccountApiAdapter(client: FetchClient): AccountApi {\n return {\n fetchAccount: async () => {\n const response = await portalTenant.account_show(client);\n return {\n account: mapAccount(response.account ?? {}),\n meta: {\n request_id: response.meta?.request_id ?? \"\",\n timestamp: response.meta?.timestamp ?? \"\",\n },\n } satisfies AccountResponse;\n },\n\n updateAccount: async (body: UpdateAccountBody) => {\n const response = await portalTenant.account_update(client, body);\n return {\n account: mapAccount(response.account ?? {}),\n meta: {\n request_id: response.meta?.request_id ?? \"\",\n timestamp: response.meta?.timestamp ?? \"\",\n },\n } satisfies AccountResponse;\n },\n };\n}\n","/**\n * Generated API client functions for portal_tenant_pay\n *\n * DO NOT EDIT THIS FILE DIRECTLY\n * This file is auto-generated. To update:\n * 1. Update the OpenAPI spec file\n * 2. Run: pnpm generate\n */\n\nimport type { FetchClient } from \"../lib/fetch-client\";\nimport type { operations } from \"../generated/portal-tenant-pay\";\n\n// ============================================================================\n// addresses\n// ============================================================================\n\n/**\n * List addresses\n * Returns addresses associated with the member's customer record in this tenant.\n *\n * @param client - Fetch client instance\n * @param [params] - params\n */\nexport async function addresses_list(\n client: FetchClient,\n params?: operations[\"addresses_list\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"addresses_list\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/pay/addresses`, params);\n}\n\n/**\n * Create an address\n * Adds a new address to the member's customer record. If an identical address already exists it is returned instead of creating a duplicate.\n *\n * @param client - Fetch client instance\n * @param body - body\n */\nexport async function addresses_create(\n client: FetchClient,\n body: operations[\"addresses_create\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"addresses_create\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.post(`/api/pay/addresses`, body);\n}\n\n/**\n * Update an address\n * Creates a new address with the merged attributes and discards the old one, preserving references from existing orders.\n *\n * @param client - Fetch client instance\n * @param id - id\n * @param body - body\n */\nexport async function addresses_update(\n client: FetchClient,\n id: string | number,\n body: operations[\"addresses_update\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"addresses_update\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.patch(`/api/pay/addresses/${id}`, body);\n}\n\n/**\n * Delete an address\n * Removes an address from the member's customer record. The default address cannot be removed.\n *\n * @param client - Fetch client instance\n * @param id - id\n */\nexport async function addresses_destroy(\n client: FetchClient,\n id: string | number,\n): Promise<\n operations[\"addresses_destroy\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.delete(`/api/pay/addresses/${id}`);\n}\n\n// ============================================================================\n// payment-methods\n// ============================================================================\n\n/**\n * List payment methods\n * Returns displayable payment methods on the member's customer record, excluding Apple Pay sources.\n *\n * @param client - Fetch client instance\n * @param [params] - params\n */\nexport async function payment_methods_list(\n client: FetchClient,\n params?: operations[\"payment_methods_list\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"payment_methods_list\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/pay/payment_methods`, params);\n}\n\n/**\n * Create a payment method\n * Tokenizes and stores a new payment method via the vault provider. Requires a vault token obtained from the vault credentials endpoint.\n *\n * @param client - Fetch client instance\n * @param body - body\n */\nexport async function payment_methods_create(\n client: FetchClient,\n body: operations[\"payment_methods_create\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"payment_methods_create\"][\"responses\"][201][\"content\"][\"application/json\"]\n> {\n return client.post(`/api/pay/payment_methods`, body);\n}\n\n/**\n * Update a payment method\n * Updates a payment method's attributes. Currently supports setting a payment method as the default.\n *\n * @param client - Fetch client instance\n * @param id - id\n * @param body - body\n */\nexport async function payment_methods_update(\n client: FetchClient,\n id: string | number,\n body: operations[\"payment_methods_update\"][\"requestBody\"][\"content\"][\"application/json\"],\n): Promise<\n operations[\"payment_methods_update\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.patch(`/api/pay/payment_methods/${id}`, body);\n}\n\n/**\n * Delete a payment method\n * Removes a payment method from the member's customer record. If the removed method was the default, the default is cleared.\n *\n * @param client - Fetch client instance\n * @param id - id\n */\nexport async function payment_methods_destroy(\n client: FetchClient,\n id: string | number,\n): Promise<\n operations[\"payment_methods_destroy\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.delete(`/api/pay/payment_methods/${id}`);\n}\n\n/**\n * Get vault credentials\n * Returns a short-lived vault token and environment identifier for initializing the client-side payment vault SDK.\n *\n * @param client - Fetch client instance\n \n */\nexport async function payment_methods_vault_show(\n client: FetchClient,\n): Promise<\n operations[\"payment_methods_vault_show\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/pay/payment_methods/vault`);\n}\n\n// ============================================================================\n// points-ledgers\n// ============================================================================\n\n/**\n * List points ledger entries\n * Returns loyalty points ledger entries for the member's customer record in this tenant, ordered by most recent first.\n *\n * @param client - Fetch client instance\n * @param [params] - params\n */\nexport async function points_ledgers_list(\n client: FetchClient,\n params?: operations[\"points_ledgers_list\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"points_ledgers_list\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/pay/points_ledgers`, params);\n}\n","import type { PayApi } from \"@fluid-app/portal-core/pay-api\";\nimport type {\n PayAddress,\n PayPaymentMethod,\n CreateAddressBody,\n CreatePaymentMethodBody,\n UpdatePaymentMethodBody,\n VaultResponse,\n PointsLedgersResponse,\n} from \"@fluid-app/portal-core/pay-types\";\nimport type { FetchClient } from \"@fluid-app/portal-tenant-api-client\";\nimport { portalTenantPay } from \"@fluid-app/portal-tenant-pay-api-client\";\n\n// ---------------------------------------------------------------------------\n// BFF response type helpers\n// ---------------------------------------------------------------------------\n\ntype BffAddress = NonNullable<\n Awaited<ReturnType<typeof portalTenantPay.addresses_list>>[\"addresses\"]\n>[number];\n\ntype BffPaymentMethod = NonNullable<\n Awaited<\n ReturnType<typeof portalTenantPay.payment_methods_list>\n >[\"payment_methods\"]\n>[number];\n\n// ---------------------------------------------------------------------------\n// BFF → port type mappers\n// ---------------------------------------------------------------------------\n\nfunction mapAddress(raw: BffAddress): PayAddress {\n return {\n id: raw.id ?? 0,\n street1: raw.street1 ?? \"\",\n street2: raw.street2 ?? null,\n city: raw.city ?? \"\",\n state: raw.state ?? \"\",\n zip: raw.zip ?? \"\",\n country: raw.country ?? \"\",\n default: raw.default ?? false,\n created_at: raw.created_at ?? null,\n updated_at: raw.updated_at ?? null,\n };\n}\n\nfunction mapPaymentMethod(raw: BffPaymentMethod): PayPaymentMethod {\n return {\n id: raw.id ?? 0,\n type: raw.type ?? \"card\",\n brand: raw.brand ?? null,\n last_four: raw.last_four ?? \"\",\n exp_month: raw.exp_month ?? null,\n exp_year: raw.exp_year ?? null,\n default: raw.default ?? false,\n created_at: raw.created_at ?? null,\n updated_at: raw.updated_at ?? null,\n };\n}\n\n// ---------------------------------------------------------------------------\n// Adapter factory\n// ---------------------------------------------------------------------------\n\nexport function createPortalTenantPayAdapter(client: FetchClient): PayApi {\n return {\n fetchAddresses: async () => {\n const response = await portalTenantPay.addresses_list(client);\n return {\n addresses: (response.addresses ?? []).map(mapAddress),\n meta: {\n request_id: response.meta?.request_id ?? \"\",\n timestamp: response.meta?.timestamp ?? \"\",\n },\n };\n },\n\n createAddress: async (body: CreateAddressBody) => {\n await portalTenantPay.addresses_create(client, body);\n },\n\n updateAddress: async (addressId: number, body: CreateAddressBody) => {\n await portalTenantPay.addresses_update(client, addressId, body);\n },\n\n deleteAddress: async (addressId: number) => {\n await portalTenantPay.addresses_destroy(client, addressId);\n },\n\n fetchPaymentMethods: async () => {\n const response = await portalTenantPay.payment_methods_list(client);\n return {\n payment_methods: (response.payment_methods ?? []).map(mapPaymentMethod),\n meta: {\n request_id: response.meta?.request_id ?? \"\",\n timestamp: response.meta?.timestamp ?? \"\",\n },\n };\n },\n\n createPaymentMethod: async (body: CreatePaymentMethodBody) => {\n await portalTenantPay.payment_methods_create(client, body);\n },\n\n updatePaymentMethod: async (\n paymentMethodId: number,\n body: UpdatePaymentMethodBody,\n ) => {\n await portalTenantPay.payment_methods_update(\n client,\n paymentMethodId,\n body,\n );\n },\n\n deletePaymentMethod: async (paymentMethodId: number) => {\n await portalTenantPay.payment_methods_destroy(client, paymentMethodId);\n },\n\n fetchVaultCredentials: async () => {\n const response = await portalTenantPay.payment_methods_vault_show(client);\n return {\n vault: {\n token: response.vault?.token ?? null,\n environment: response.vault?.environment ?? \"production\",\n },\n meta: {\n request_id: response.meta?.request_id ?? \"\",\n timestamp: response.meta?.timestamp ?? \"\",\n },\n } satisfies VaultResponse;\n },\n\n fetchPointsLedgers: async () => {\n const response = await portalTenantPay.points_ledgers_list(client);\n return {\n points_ledgers: (response.points_ledgers ?? []).map((entry) => ({\n id: entry.id ?? 0,\n amount: entry.amount ?? 0,\n total_balance: entry.total_balance ?? 0,\n metadata: (entry.metadata as Record<string, unknown>) ?? null,\n created_at: entry.created_at ?? \"\",\n })),\n meta: {\n request_id: response.meta?.request_id ?? \"\",\n timestamp: response.meta?.timestamp ?? \"\",\n },\n } satisfies PointsLedgersResponse;\n },\n };\n}\n","/**\n * Generated API client functions for portal_tenant_store\n *\n * DO NOT EDIT THIS FILE DIRECTLY\n * This file is auto-generated. To update:\n * 1. Update the OpenAPI spec file\n * 2. Run: pnpm generate\n */\n\nimport type { FetchClient } from \"../lib/fetch-client\";\nimport type { operations } from \"../generated/portal-tenant-store\";\n\n// ============================================================================\n// countries\n// ============================================================================\n\n/**\n * List available countries\n * Returns countries enabled for the tenant store, each with its ISO 3166 subdivisions (states/provinces).\n *\n * @param client - Fetch client instance\n * @param [params] - params\n */\nexport async function countries_list(\n client: FetchClient,\n params?: operations[\"countries_list\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"countries_list\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/store/countries`, params);\n}\n\n// ============================================================================\n// currencies\n// ============================================================================\n\n/**\n * List supported currencies\n * Returns the distinct set of currencies derived from the tenant company's enabled countries, sorted by name.\n *\n * @param client - Fetch client instance\n \n */\nexport async function currencies_list(\n client: FetchClient,\n): Promise<\n operations[\"currencies_list\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/store/currencies`);\n}\n\n// ============================================================================\n// languages\n// ============================================================================\n\n/**\n * List available languages\n * Returns languages enabled for the tenant store, sorted by name.\n *\n * @param client - Fetch client instance\n * @param [params] - params\n */\nexport async function languages_list(\n client: FetchClient,\n params?: operations[\"languages_list\"][\"parameters\"][\"query\"],\n): Promise<\n operations[\"languages_list\"][\"responses\"][200][\"content\"][\"application/json\"]\n> {\n return client.get(`/api/store/languages`, params);\n}\n","import type {\n CountriesApi,\n countries,\n} from \"@fluid-app/store-core/countries-api\";\nimport type { FetchClient } from \"./lib/fetch-client\";\nimport { portalTenantStore } from \"@fluid-app/portal-tenant-store-api-client\";\n\n/**\n * Maps a BFF state to the port's State shape.\n */\nfunction mapState(\n raw: NonNullable<\n NonNullable<\n Awaited<ReturnType<typeof portalTenantStore.countries_list>>[\"countries\"]\n >[number][\"states\"]\n >[number],\n): countries.State {\n return {\n code: raw.code ?? \"\",\n name: raw.name ?? \"\",\n };\n}\n\n/**\n * Maps a BFF country to the port's Country shape.\n */\nfunction mapCountry(\n raw: NonNullable<\n Awaited<ReturnType<typeof portalTenantStore.countries_list>>[\"countries\"]\n >[number],\n): countries.Country {\n return {\n code: raw.code ?? \"\",\n name: raw.name ?? \"\",\n currency_code: raw.currency_code ?? \"\",\n states: (raw.states ?? []).map(mapState),\n };\n}\n\n/**\n * Maps the BFF meta envelope to the port's ApiMeta shape.\n */\nfunction mapMeta(\n raw: Awaited<ReturnType<typeof portalTenantStore.countries_list>>[\"meta\"],\n): countries.CountriesListResponse[\"meta\"] {\n return {\n request_id: raw?.request_id ?? null,\n timestamp: raw?.timestamp ?? \"\",\n pagination: raw?.pagination\n ? {\n cursor: raw.pagination.cursor ?? null,\n limit: raw.pagination.limit,\n next_cursor: raw.pagination.next_cursor ?? null,\n prev_cursor: raw.pagination.prev_cursor ?? null,\n }\n : undefined,\n };\n}\n\n/**\n * Creates a CountriesApi adapter backed by the portal-tenant store BFF.\n *\n * Maps the generated portal-tenant-store namespace functions to the abstract\n * CountriesApi port, closing over the FetchClient so consumers don't need\n * to pass it per-call.\n */\nexport function createPortalTenantCountriesAdapter(\n client: FetchClient,\n): CountriesApi {\n return {\n listCountries: async (params) => {\n const response = await portalTenantStore.countries_list(client, {\n \"page[cursor]\": params?.cursor,\n \"page[limit]\": params?.limit,\n });\n return {\n countries: (response.countries ?? []).map(mapCountry),\n meta: mapMeta(response.meta),\n };\n },\n };\n}\n","import type {\n LanguagesApi,\n languages,\n} from \"@fluid-app/store-core/languages-api\";\nimport type { FetchClient } from \"./lib/fetch-client\";\nimport { portalTenantStore } from \"@fluid-app/portal-tenant-store-api-client\";\n\n/**\n * Maps a BFF language to the port's Language shape.\n */\nfunction mapLanguage(\n raw: NonNullable<\n Awaited<ReturnType<typeof portalTenantStore.languages_list>>[\"languages\"]\n >[number],\n): languages.Language {\n return {\n code: raw.code ?? \"\",\n name: raw.name ?? \"\",\n };\n}\n\n/**\n * Maps the BFF meta envelope to the port's ApiMeta shape.\n */\nfunction mapMeta(\n raw: Awaited<ReturnType<typeof portalTenantStore.languages_list>>[\"meta\"],\n): languages.LanguagesListResponse[\"meta\"] {\n return {\n request_id: raw?.request_id ?? null,\n timestamp: raw?.timestamp ?? \"\",\n pagination: raw?.pagination\n ? {\n cursor: raw.pagination.cursor ?? null,\n limit: raw.pagination.limit,\n next_cursor: raw.pagination.next_cursor ?? null,\n prev_cursor: raw.pagination.prev_cursor ?? null,\n }\n : undefined,\n };\n}\n\n/**\n * Creates a LanguagesApi adapter backed by the portal-tenant store BFF.\n *\n * Maps the generated portal-tenant-store namespace functions to the abstract\n * LanguagesApi port, closing over the FetchClient so consumers don't need\n * to pass it per-call.\n */\nexport function createPortalTenantLanguagesAdapter(\n client: FetchClient,\n): LanguagesApi {\n return {\n listLanguages: async (params) => {\n const response = await portalTenantStore.languages_list(client, {\n \"page[cursor]\": params?.cursor,\n \"page[limit]\": params?.limit,\n });\n return {\n languages: (response.languages ?? []).map(mapLanguage),\n meta: mapMeta(response.meta),\n };\n },\n };\n}\n","import type { CountriesApi } from \"@fluid-app/store-core/countries-api\";\nimport type { FetchClient } from \"@fluid-app/portal-tenant-api-client\";\nimport { createPortalTenantCountriesAdapter } from \"@fluid-app/store-api-client\";\n\n/**\n * Creates a CountriesApi adapter backed by the portal-tenant store BFF.\n *\n * Delegates to the adapter factory in `@fluid-app/store-api-client` which\n * maps the generated portal-tenant-store namespace functions to the abstract\n * CountriesApi port.\n */\nexport function createCountriesApiAdapter(client: FetchClient): CountriesApi {\n return createPortalTenantCountriesAdapter(client);\n}\n","import type { LanguagesApi } from \"@fluid-app/store-core/languages-api\";\nimport type { FetchClient } from \"@fluid-app/portal-tenant-api-client\";\nimport { createPortalTenantLanguagesAdapter } from \"@fluid-app/store-api-client\";\n\n/**\n * Creates a LanguagesApi adapter backed by the portal-tenant store BFF.\n *\n * Delegates to the adapter factory in `@fluid-app/store-api-client` which\n * maps the generated portal-tenant-store namespace functions to the abstract\n * LanguagesApi port.\n */\nexport function createLanguagesApiAdapter(client: FetchClient): LanguagesApi {\n return createPortalTenantLanguagesAdapter(client);\n}\n","import type { MySiteApi } from \"@fluid-app/mysite-core/mysite-api\";\nimport type {\n MySiteProfile,\n MySiteLink,\n MySiteFavorite,\n MySiteTheme,\n UpdateProfileBody,\n UpdateSettingsBody,\n CreateLinkBody,\n UpdateLinkBody,\n AddFavoriteBody,\n} from \"@fluid-app/mysite-core/mysite-api-types\";\nimport type { FetchClient } from \"@fluid-app/portal-tenant-api-client\";\nimport { portalTenantMysite } from \"@fluid-app/portal-tenant-mysite-api-client\";\n\n// ---------------------------------------------------------------------------\n// BFF response type helpers\n// ---------------------------------------------------------------------------\n\ntype BffProfile = NonNullable<\n Awaited<ReturnType<typeof portalTenantMysite.mysite_profile_show>>[\"profile\"]\n>;\n\ntype BffLink = NonNullable<\n Awaited<ReturnType<typeof portalTenantMysite.mysite_links_list>>[\"links\"]\n>[number];\n\ntype BffFavorite = NonNullable<\n Awaited<\n ReturnType<typeof portalTenantMysite.mysite_favorites_list>\n >[\"favorites\"]\n>[number];\n\ntype BffTheme = NonNullable<\n Awaited<ReturnType<typeof portalTenantMysite.mysite_themes_list>>[\"themes\"]\n>[number];\n\n// ---------------------------------------------------------------------------\n// BFF -> port type mappers\n// ---------------------------------------------------------------------------\n\nfunction mapProfile(raw: BffProfile): MySiteProfile {\n return {\n id: raw.id ?? 0,\n mysite_url: raw.mysite_url ?? null,\n theme_id: raw.theme_id ?? null,\n bio: raw.bio ?? null,\n avatar_url: raw.avatar_url ?? null,\n display_name: raw.display_name ?? null,\n slug: raw.slug ?? null,\n } satisfies MySiteProfile;\n}\n\nfunction mapLink(raw: BffLink): MySiteLink {\n return {\n id: raw.id ?? 0,\n url: raw.url ?? \"\",\n title: raw.title ?? \"\",\n position: raw.position ?? 0,\n } satisfies MySiteLink;\n}\n\nfunction mapFavorite(raw: BffFavorite): MySiteFavorite {\n return {\n id: raw.id ?? 0,\n product_id: raw.product_id ?? 0,\n product_name: raw.product_name ?? null,\n product_image_url: raw.product_image_url ?? null,\n position: raw.position ?? 0,\n created_at: raw.created_at ?? null,\n } satisfies MySiteFavorite;\n}\n\nfunction mapTheme(raw: BffTheme): MySiteTheme {\n return {\n id: raw.id ?? 0,\n name: raw.name ?? \"\",\n preview_url: raw.preview_url ?? null,\n } satisfies MySiteTheme;\n}\n\n// ---------------------------------------------------------------------------\n// Adapter factory\n// ---------------------------------------------------------------------------\n\nexport function createMySiteApiAdapter(client: FetchClient): MySiteApi {\n return {\n fetchProfile: async () => {\n const res = await portalTenantMysite.mysite_profile_show(client);\n return mapProfile((res.profile ?? {}) as BffProfile);\n },\n\n updateProfile: async (body: UpdateProfileBody) => {\n const res = await portalTenantMysite.mysite_profile_update(client, {\n profile: {\n display_name: body.display_name,\n bio: body.bio,\n avatar_url: body.avatar_url,\n },\n });\n return mapProfile((res.profile ?? {}) as BffProfile);\n },\n\n updateSettings: async (body: UpdateSettingsBody) => {\n await Promise.all([\n body.theme_id !== undefined\n ? portalTenantMysite.mysite_settings_update(client, {\n settings: { theme_id: body.theme_id },\n })\n : Promise.resolve(),\n body.slug !== undefined\n ? portalTenantMysite.mysite_profile_update(client, {\n profile: { slug: body.slug },\n })\n : Promise.resolve(),\n ]);\n },\n\n listLinks: async () => {\n const res = await portalTenantMysite.mysite_links_list(client);\n return (res.links ?? []).map(mapLink);\n },\n\n createLink: async (body: CreateLinkBody) => {\n const res = await portalTenantMysite.mysite_links_create(client, {\n link: { title: body.title, url: body.url },\n });\n return mapLink((res.link ?? {}) as BffLink);\n },\n\n updateLink: async (linkId: number, body: UpdateLinkBody) => {\n const res = await portalTenantMysite.mysite_links_update(client, linkId, {\n link: { title: body.title, url: body.url },\n });\n return mapLink((res.link ?? {}) as BffLink);\n },\n\n deleteLink: async (linkId: number) => {\n await portalTenantMysite.mysite_links_destroy(client, linkId);\n },\n\n reorderLinks: async (orderedIds: number[]) => {\n const res = await portalTenantMysite.mysite_links_bulk_reorder(client, {\n ordered_ids: orderedIds,\n });\n return (res.links ?? []).map(mapLink);\n },\n\n listFavorites: async () => {\n const res = await portalTenantMysite.mysite_favorites_list(client);\n return (res.favorites ?? []).map(mapFavorite);\n },\n\n addFavorite: async (body: AddFavoriteBody) => {\n const res = await portalTenantMysite.mysite_favorites_create(client, {\n favorite: { product_id: body.product_id },\n });\n return mapFavorite((res.favorite ?? {}) as BffFavorite);\n },\n\n deleteFavorite: async (favoriteId: number) => {\n await portalTenantMysite.mysite_favorites_destroy(client, favoriteId);\n },\n\n reorderFavorites: async (orderedIds: number[]) => {\n const res = await portalTenantMysite.mysite_favorites_bulk_reorder(\n client,\n { ordered_ids: orderedIds },\n );\n return (res.favorites ?? []).map(mapFavorite);\n },\n\n listThemes: async () => {\n const res = await portalTenantMysite.mysite_themes_list(client);\n return (res.themes ?? []).map(mapTheme);\n },\n };\n}\n","/**\n * Theme Provider for Fluid SDK\n * Handles CSS variable injection using portal-core's theme engine.\n * Accepts ThemeDefinition objects and resolves them into CSS via generateThemeCSS.\n */\n\nimport {\n createContext,\n useContext,\n useState,\n useEffect,\n useMemo,\n useCallback,\n type ReactNode,\n} from \"react\";\nimport type { ThemeDefinition } from \"../types/theme\";\nimport {\n resolveTheme,\n applyTheme as coreApplyTheme,\n removeTheme,\n} from \"../themes\";\n\ntype ThemeMode = \"light\" | \"dark\";\n\n/**\n * Context value for theme management.\n * All properties are readonly since context values should not be mutated by consumers.\n */\ninterface ThemeContextValue {\n /** Currently active theme definition */\n readonly currentTheme: ThemeDefinition | null;\n /** Switch to a different theme */\n readonly setTheme: (theme: ThemeDefinition) => void;\n /** Switch between light and dark mode for the current theme */\n readonly setThemeMode: (mode: ThemeMode) => void;\n /** Current theme mode */\n readonly mode: ThemeMode | undefined;\n}\n\nconst ThemeContext = createContext<ThemeContextValue | null>(null);\n\nexport interface FluidThemeProviderProps {\n children: ReactNode;\n /** Initial theme to apply */\n initialTheme?: ThemeDefinition;\n /** Container element for scoped theme application (defaults to document.documentElement) */\n container?: HTMLElement | null;\n}\n\n/**\n * Apply a theme to the DOM using the shared portal-core pipeline.\n * Also sets data attributes on the container for CSS selector targeting.\n */\nfunction applyThemeToDOM(\n theme: ThemeDefinition,\n mode: ThemeMode | undefined,\n container: HTMLElement | null,\n): void {\n const target = container ?? document.documentElement;\n\n // Use shared resolve + inject pipeline from portal-core\n const resolved = resolveTheme(theme);\n coreApplyTheme(resolved);\n\n // Set data attributes for CSS selector targeting\n target.dataset.theme = theme.id;\n if (mode) {\n target.dataset.themeMode = mode;\n } else {\n delete target.dataset.themeMode;\n }\n}\n\nexport function FluidThemeProvider({\n children,\n initialTheme,\n container,\n}: FluidThemeProviderProps): React.JSX.Element {\n const [currentTheme, setCurrentTheme] = useState<ThemeDefinition | null>(\n initialTheme ?? null,\n );\n const [mode, setMode] = useState<ThemeMode | undefined>(undefined);\n\n // Apply theme CSS when theme or mode changes\n useEffect(() => {\n if (currentTheme) {\n applyThemeToDOM(currentTheme, mode, container ?? null);\n }\n\n return () => {\n if (currentTheme) {\n removeTheme(currentTheme.id);\n }\n };\n }, [currentTheme, mode, container]);\n\n const setTheme = useCallback((theme: ThemeDefinition) => {\n setCurrentTheme(theme);\n }, []);\n\n const setThemeMode = useCallback((newMode: ThemeMode) => {\n setMode(newMode);\n }, []);\n\n const value = useMemo(\n () =>\n ({\n currentTheme,\n setTheme,\n setThemeMode,\n mode,\n }) satisfies ThemeContextValue,\n [currentTheme, setTheme, setThemeMode, mode],\n );\n\n return (\n <ThemeContext.Provider value={value}>{children}</ThemeContext.Provider>\n );\n}\n\n/**\n * Hook to access theme context\n * Must be used within a FluidThemeProvider\n */\nexport function useThemeContext(): ThemeContextValue {\n const context = useContext(ThemeContext);\n if (!context) {\n throw new Error(\"useThemeContext must be used within a FluidThemeProvider\");\n }\n return context;\n}\n","import type { AnyComponent } from \"../types/widget-schema\";\nimport type { WidgetSchema, TypedWidgetSchema } from \"../types\";\nimport type { ShareableItem } from \"../types/shareable-item\";\nimport type { WidgetManifest } from \"../registries/widget-manifest\";\n\nexport function createWidgetRegistry<\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n T extends Record<string, AnyComponent>,\n>(registry: T, plugins?: WidgetManifest[]): T {\n if (!plugins || plugins.length === 0) {\n return registry;\n }\n\n const pluginEntries = Object.fromEntries(\n plugins.map((p) => [p.type, p.component]),\n );\n\n return { ...registry, ...pluginEntries } as T;\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport function createScreen<T extends Record<string, AnyComponent>>(\n registry: T,\n widgets: TypedWidgetSchema<T>[],\n): TypedWidgetSchema<T>[] {\n // Validate that all widget types exist in the registry at runtime\n widgets.forEach((widget) => {\n if (!(widget.type in registry)) {\n throw new Error(\n `Widget type \"${String(widget.type)}\" not found in registry`,\n );\n }\n });\n return widgets;\n}\n\n// Helper to create WidgetSchema from shareable item\nexport function createWidgetFromShareable(item: ShareableItem): WidgetSchema {\n const isVideo = item.kind === \"video\" || !!item.videoUrl;\n\n if (isVideo && item.videoUrl) {\n return {\n type: \"VideoWidget\",\n props: {\n src: item.videoUrl,\n poster: item.imageUrl,\n caption: item.title,\n },\n };\n }\n\n return {\n type: \"ImageWidget\",\n props: {\n src: item.imageUrl,\n alt: item.title || \"Image\",\n objectFit: \"cover\",\n },\n };\n}\n","import type { WidgetPropertySchema } from \"@fluid-app/portal-core/registries\";\n\n// Widget Components\nexport { AlertWidget, alertWidgetPropertySchema } from \"./AlertWidget\";\nexport {\n BulletListWidget,\n bulletListWidgetPropertySchema,\n} from \"./BulletListWidget\";\nexport { CalendarWidget, calendarWidgetPropertySchema } from \"./CalendarWidget\";\nexport { CardWidget, cardWidgetPropertySchema } from \"./CardWidget\";\nexport { CarouselWidget, carouselWidgetPropertySchema } from \"./CarouselWidget\";\nexport { CatchUpWidget, catchUpWidgetPropertySchema } from \"./CatchUpWidget\";\nexport { ChartWidget, chartWidgetPropertySchema } from \"./ChartWidget\";\nexport {\n ContainerWidget,\n containerWidgetPropertySchema,\n} from \"./ContainerWidget\";\nexport { EmbedWidget, embedWidgetPropertySchema } from \"./EmbedWidget\";\nexport { ImageWidget, imageWidgetPropertySchema } from \"./ImageWidget\";\nexport { LayoutWidget, layoutWidgetPropertySchema } from \"./LayoutWidget\";\nexport { LinkWidget, linkWidgetPropertySchema } from \"./LinkWidget\";\nexport { ListWidget, listWidgetPropertySchema } from \"./ListWidget\";\nexport { MySiteWidget, mySiteWidgetPropertySchema } from \"./MySiteWidget\";\nexport { NestedWidget, nestedWidgetPropertySchema } from \"./NestedWidget\";\nexport { PointsWidget, pointsWidgetPropertySchema } from \"./PointsWidget\";\nexport {\n QuickShareWidget,\n quickShareWidgetPropertySchema,\n} from \"./QuickShareWidget\";\nexport {\n RecentActivityWidget,\n recentActivityWidgetPropertySchema,\n} from \"./RecentActivityWidget\";\nexport {\n SeparatorWidget,\n separatorWidgetPropertySchema,\n} from \"./SeparatorWidget\";\nexport { SpacerWidget, spacerWidgetPropertySchema } from \"./SpacerWidget\";\nexport { TableWidget, tableWidgetPropertySchema } from \"./TableWidget\";\nexport { TextWidget, textWidgetPropertySchema } from \"./TextWidget\";\nexport { ToDoWidget, toDoWidgetPropertySchema } from \"./ToDoWidget\";\nexport { VideoWidget, videoWidgetPropertySchema } from \"./VideoWidget\";\n\n// Re-export gapValues from core (used by widgets)\nexport { gapValues } from \"../core/fields\";\n\n// Re-export all property schemas as a collection\nexport const widgetPropertySchemas = {\n AlertWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./AlertWidget\").then((m) => m.alertWidgetPropertySchema),\n BulletListWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./BulletListWidget\").then((m) => m.bulletListWidgetPropertySchema),\n CalendarWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./CalendarWidget\").then((m) => m.calendarWidgetPropertySchema),\n CardWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./CardWidget\").then((m) => m.cardWidgetPropertySchema),\n CarouselWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./CarouselWidget\").then((m) => m.carouselWidgetPropertySchema),\n CatchUpWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./CatchUpWidget\").then((m) => m.catchUpWidgetPropertySchema),\n ChartWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./ChartWidget\").then((m) => m.chartWidgetPropertySchema),\n ContainerWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./ContainerWidget\").then((m) => m.containerWidgetPropertySchema),\n EmbedWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./EmbedWidget\").then((m) => m.embedWidgetPropertySchema),\n ImageWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./ImageWidget\").then((m) => m.imageWidgetPropertySchema),\n LayoutWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./LayoutWidget\").then((m) => m.layoutWidgetPropertySchema),\n LinkWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./LinkWidget\").then((m) => m.linkWidgetPropertySchema),\n ListWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./ListWidget\").then((m) => m.listWidgetPropertySchema),\n MySiteWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./MySiteWidget\").then((m) => m.mySiteWidgetPropertySchema),\n NestedWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./NestedWidget\").then((m) => m.nestedWidgetPropertySchema),\n PointsWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./PointsWidget\").then((m) => m.pointsWidgetPropertySchema),\n QuickShareWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./QuickShareWidget\").then((m) => m.quickShareWidgetPropertySchema),\n RecentActivityWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./RecentActivityWidget\").then(\n (m) => m.recentActivityWidgetPropertySchema,\n ),\n SeparatorWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./SeparatorWidget\").then((m) => m.separatorWidgetPropertySchema),\n SpacerWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./SpacerWidget\").then((m) => m.spacerWidgetPropertySchema),\n TableWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./TableWidget\").then((m) => m.tableWidgetPropertySchema),\n TextWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./TextWidget\").then((m) => m.textWidgetPropertySchema),\n ToDoWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./ToDoWidget\").then((m) => m.toDoWidgetPropertySchema),\n VideoWidget: (): Promise<WidgetPropertySchema> =>\n import(\"./VideoWidget\").then((m) => m.videoWidgetPropertySchema),\n};\n","/**\n * Default widget registry for SDK apps.\n * Maps widget type names to their view-only components from portal-widgets.\n */\n\nimport type { WidgetRegistry } from \"@fluid-app/portal-core/types\";\nimport { createWidgetRegistry } from \"@fluid-app/portal-core/widget-utils\";\nimport {\n AlertWidget,\n BulletListWidget,\n CalendarWidget,\n CardWidget,\n CarouselWidget,\n CatchUpWidget,\n ChartWidget,\n ContainerWidget,\n EmbedWidget,\n ImageWidget,\n LayoutWidget,\n LinkWidget,\n ListWidget,\n MySiteWidget,\n NestedWidget,\n PointsWidget,\n QuickShareWidget,\n RecentActivityWidget,\n SeparatorWidget,\n SpacerWidget,\n TableWidget,\n TextWidget,\n ToDoWidget,\n VideoWidget,\n} from \"@fluid-app/portal-widgets/widgets\";\n\nexport const DEFAULT_SDK_WIDGET_REGISTRY: WidgetRegistry = createWidgetRegistry(\n {\n AlertWidget,\n BulletListWidget,\n CalendarWidget,\n CardWidget,\n CarouselWidget,\n CatchUpWidget,\n ChartWidget,\n ContainerWidget,\n EmbedWidget,\n ImageWidget,\n LayoutWidget,\n LinkWidget,\n ListWidget,\n MySiteWidget,\n NestedWidget,\n PointsWidget,\n QuickShareWidget,\n RecentActivityWidget,\n SeparatorWidget,\n SpacerWidget,\n TableWidget,\n TextWidget,\n ToDoWidget,\n VideoWidget,\n },\n);\n","/**\n * Main Fluid SDK Provider\n * Wraps QueryClientProvider and FluidThemeProvider\n */\n\nimport {\n createContext,\n useContext,\n useCallback,\n useMemo,\n useRef,\n type ReactNode,\n} from \"react\";\nimport { QueryClient, QueryClientProvider } from \"@tanstack/react-query\";\nimport { createPersister } from \"@fluid-app/query-persister\";\nimport { FluidOsApiProvider } from \"@fluid-app/portal-core/fluidos-api-context\";\nimport { AppDefinitionApiProvider } from \"@fluid-app/portal-core/app-definition-api-context\";\nimport { AccountApiProvider } from \"@fluid-app/portal-core/account-api-context\";\nimport { PayApiProvider } from \"@fluid-app/portal-core/pay-api-context\";\nimport { MySiteApiProvider } from \"@fluid-app/mysite-core/mysite-api-context\";\nimport { CountriesApiProvider } from \"@fluid-app/store-core/countries-api-context\";\nimport { LanguagesApiProvider } from \"@fluid-app/store-core/languages-api-context\";\nimport { createFetchClient as createPortalTenantFetchClient } from \"@fluid-app/portal-tenant-api-client\";\nimport { createFluidClient, type FluidClient } from \"../client/fluid-client\";\nimport { createFluidOsReadAdapter } from \"../adapters/fluidos-api-adapter\";\nimport { createAppDefinitionApiAdapter } from \"../adapters/app-definition-api-adapter\";\nimport { createAccountApiAdapter } from \"../adapters/account-api-adapter\";\nimport { createPortalTenantPayAdapter } from \"../adapters/pay-api-adapter\";\nimport { createCountriesApiAdapter } from \"../adapters/countries-api-adapter\";\nimport { createLanguagesApiAdapter } from \"../adapters/languages-api-adapter\";\nimport { createMySiteApiAdapter } from \"../adapters/mysite-api-adapter\";\nimport { FluidThemeProvider } from \"./FluidThemeProvider\";\nimport { PortalTenantClientProvider } from \"./PortalTenantClientProvider\";\nimport { DataSourceRegistryProvider } from \"@fluid-app/portal-react/data-sources/registry-context\";\nimport { RegistryProvider } from \"@fluid-app/portal-widgets/contexts\";\nimport { DEFAULT_SDK_WIDGET_REGISTRY } from \"../core/default-widget-registry\";\nimport type { FluidSDKConfig } from \"../client/types\";\nimport type { ThemeDefinition } from \"../types/theme\";\n\n/**\n * Context value for FluidProvider.\n * All properties are readonly since context values should not be mutated by consumers.\n */\ninterface FluidContextValue {\n /** Configured API client instance */\n readonly client: FluidClient;\n /** SDK configuration */\n readonly config: FluidSDKConfig;\n}\n\nconst FluidContext = createContext<FluidContextValue | null>(null);\n\nexport interface FluidProviderProps {\n /** SDK configuration (baseUrl, auth, etc.) */\n config: FluidSDKConfig;\n /** React children */\n children: ReactNode;\n /** Optional custom QueryClient instance */\n queryClient?: QueryClient;\n /** Optional initial theme */\n initialTheme?: ThemeDefinition;\n /** Optional container for scoped theme application */\n themeContainer?: HTMLElement | null;\n /** Optional custom widget registry (defaults to all built-in widgets) */\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n widgetRegistry?: Record<string, React.ComponentType<any>>;\n /** Dynamic variables for data source endpoint path substitution (e.g., { rep_id: \"123\" }) */\n variables?: Record<string, string>;\n}\n\n/**\n * Main provider for the Fluid Portal SDK\n *\n * @example\n * ```tsx\n * import { FluidProvider } from \"@fluid-app/portal-sdk\";\n *\n * function App() {\n * return (\n * <FluidProvider\n * config={{ baseUrl: \"\" }}\n * >\n * <YourApp />\n * </FluidProvider>\n * );\n * }\n * ```\n */\nexport function FluidProvider({\n config,\n children,\n queryClient,\n initialTheme,\n themeContainer,\n widgetRegistry,\n variables,\n}: FluidProviderProps): React.JSX.Element {\n // Create default QueryClient if none provided\n // Using lazy initialization to ensure it's only created once\n const defaultQueryClient = useMemo(() => {\n const persister =\n typeof window !== \"undefined\" && !import.meta.env?.DEV\n ? createPersister()\n : undefined;\n\n return new QueryClient({\n defaultOptions: {\n queries: {\n staleTime: 1000 * 60, // 1 minute\n retry: 1,\n persister: persister?.persisterFn,\n },\n },\n });\n }, []);\n\n // Keep latest config in a ref so client creation reads current values\n const configRef = useRef(config);\n configRef.current = config;\n\n // Recreate client only when baseUrl changes (primitive dependency).\n // Wrap callbacks in closures so they always read the latest from configRef,\n // avoiding stale captures when parent passes new getAuthToken/onAuthError.\n const client = useMemo(\n () =>\n createFluidClient({\n ...configRef.current,\n getAuthToken: () => configRef.current.getAuthToken?.() ?? null,\n onAuthError: () => configRef.current.onAuthError?.(),\n }),\n // eslint-disable-next-line react-hooks/exhaustive-deps\n [config.baseUrl],\n );\n\n // Create FluidOsReadApi adapter for dependency inversion\n const fluidOsApi = useMemo(\n () => createFluidOsReadAdapter(client.fetchClient),\n [client.fetchClient],\n );\n\n // Read the CSRF token from the <meta name=\"csrf-token\"> tag that Rails\n // embeds in the HTML. Required for non-GET requests to the BFF.\n const csrfToken =\n typeof document !== \"undefined\"\n ? document\n .querySelector('meta[name=\"csrf-token\"]')\n ?.getAttribute(\"content\")\n : null;\n\n const portalTenantClient = useMemo(\n () =>\n createPortalTenantFetchClient({\n baseUrl: configRef.current.baseUrl,\n onAuthError: () => configRef.current.onAuthError?.(),\n credentials: \"include\",\n defaultHeaders: {\n ...(csrfToken ? { \"X-CSRF-Token\": csrfToken } : {}),\n },\n }),\n // eslint-disable-next-line react-hooks/exhaustive-deps\n [config.baseUrl, csrfToken],\n );\n\n // Create AppDefinitionApi adapter for dependency inversion\n const appDefinitionApi = useMemo(\n () => createAppDefinitionApiAdapter(portalTenantClient),\n [portalTenantClient],\n );\n\n // Create AccountApi adapter for dependency inversion\n const accountApi = useMemo(\n () => createAccountApiAdapter(portalTenantClient),\n [portalTenantClient],\n );\n\n // Create PayApi adapter for dependency inversion\n const payApi = useMemo(\n () => createPortalTenantPayAdapter(portalTenantClient),\n [portalTenantClient],\n );\n\n // Create CountriesApi adapter for dependency inversion\n const countriesApi = useMemo(\n () => createCountriesApiAdapter(portalTenantClient),\n [portalTenantClient],\n );\n\n // Create LanguagesApi adapter for dependency inversion\n const languagesApi = useMemo(\n () => createLanguagesApiAdapter(portalTenantClient),\n [portalTenantClient],\n );\n\n // Create MySiteApi adapter for dependency inversion\n const mysiteApi = useMemo(\n () => createMySiteApiAdapter(portalTenantClient),\n [portalTenantClient],\n );\n\n // Context value only changes when client changes\n const contextValue = useMemo(\n () => ({ client, config: configRef.current }) satisfies FluidContextValue,\n [client],\n );\n\n // Stable callback to provide auth headers for data source fetchers.\n // Auth is handled by session cookies via `credentials: 'include'` on fetch calls.\n const getApiHeaders = useCallback((): Record<string, string> => {\n return {\n \"Content-Type\": \"application/json\",\n };\n }, []);\n\n // Data source endpoints (e.g., /company/v1/products) expect the base URL\n // to include /api, matching fluid-admin's NEXT_PUBLIC_API_URL pattern.\n const dataSourceBaseUrl = useMemo(() => {\n const base = config.baseUrl.replace(/\\/+$/, \"\");\n return base.endsWith(\"/api\") ? base : `${base}/api`;\n }, [config.baseUrl]);\n\n // Variables for data source path interpolation (e.g., { rep_id: \"123\" }).\n // Must be passed explicitly via the `variables` prop.\n const effectiveVariables = variables;\n\n // Build theme provider props conditionally to satisfy exactOptionalPropertyTypes\n // We only pass props that are defined to avoid passing `undefined` explicitly\n const themeProviderProps = {\n ...(initialTheme !== undefined && { initialTheme }),\n ...(themeContainer !== undefined && { container: themeContainer }),\n };\n\n // Use provided registry or fall back to the default SDK widget registry\n const registry = widgetRegistry ?? DEFAULT_SDK_WIDGET_REGISTRY;\n\n return (\n <QueryClientProvider client={queryClient ?? defaultQueryClient}>\n <FluidContext.Provider value={contextValue}>\n <PortalTenantClientProvider value={portalTenantClient}>\n <FluidOsApiProvider value={fluidOsApi}>\n <AppDefinitionApiProvider value={appDefinitionApi}>\n <AccountApiProvider value={accountApi}>\n <PayApiProvider value={payApi}>\n <CountriesApiProvider value={countriesApi}>\n <LanguagesApiProvider value={languagesApi}>\n <MySiteApiProvider value={mysiteApi}>\n <ProviderStack\n baseUrl={dataSourceBaseUrl}\n getApiHeaders={getApiHeaders}\n variables={effectiveVariables}\n registry={registry}\n themeProviderProps={themeProviderProps}\n >\n {children}\n </ProviderStack>\n </MySiteApiProvider>\n </LanguagesApiProvider>\n </CountriesApiProvider>\n </PayApiProvider>\n </AccountApiProvider>\n </AppDefinitionApiProvider>\n </FluidOsApiProvider>\n </PortalTenantClientProvider>\n </FluidContext.Provider>\n </QueryClientProvider>\n );\n}\n\n/**\n * Inner component that sets up data source providers and theme.\n * Must be rendered inside FluidContext.Provider + QueryClientProvider.\n */\nfunction ProviderStack({\n baseUrl,\n getApiHeaders,\n variables,\n registry,\n themeProviderProps,\n children,\n}: {\n baseUrl: string;\n getApiHeaders: () => Record<string, string>;\n variables: Record<string, string> | undefined;\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n registry: Record<string, React.ComponentType<any>>;\n themeProviderProps: Record<string, unknown>;\n children: ReactNode;\n}): React.JSX.Element {\n return (\n <DataSourceRegistryProvider\n baseUrl={baseUrl}\n getApiHeaders={getApiHeaders}\n variables={variables}\n >\n <RegistryProvider registry={registry}>\n <FluidThemeProvider {...themeProviderProps}>\n {children}\n </FluidThemeProvider>\n </RegistryProvider>\n </DataSourceRegistryProvider>\n );\n}\n\n/**\n * Hook to access the Fluid context\n * Must be used within a FluidProvider\n */\nexport function useFluidContext(): FluidContextValue {\n const context = useContext(FluidContext);\n if (!context) {\n throw new Error(\"useFluidContext must be used within a FluidProvider\");\n }\n return context;\n}\n"],"x_google_ignoreList":[0,1,2,3],"mappings":"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"}
|