@open-mercato/core 0.4.6-develop-e321a4e2a1 → 0.4.6-main-24e64eef39

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (180) hide show
  1. package/AGENTS.md +0 -22
  2. package/dist/modules/api_docs/frontend/docs/api/page.js +1 -1
  3. package/dist/modules/api_docs/frontend/docs/api/page.js.map +2 -2
  4. package/dist/modules/attachments/api/library/[id]/route.js +1 -0
  5. package/dist/modules/attachments/api/library/[id]/route.js.map +2 -2
  6. package/dist/modules/attachments/components/AttachmentLibrary.js +1 -1
  7. package/dist/modules/attachments/components/AttachmentLibrary.js.map +2 -2
  8. package/dist/modules/attachments/lib/partitionEnv.js +1 -1
  9. package/dist/modules/attachments/lib/partitionEnv.js.map +2 -2
  10. package/dist/modules/auth/backend/users/page.js +1 -1
  11. package/dist/modules/auth/backend/users/page.js.map +2 -2
  12. package/dist/modules/auth/cli.js +1 -1
  13. package/dist/modules/auth/cli.js.map +2 -2
  14. package/dist/modules/auth/commands/users.js +1 -1
  15. package/dist/modules/auth/commands/users.js.map +2 -2
  16. package/dist/modules/business_rules/components/utils/formHelpers.js +1 -1
  17. package/dist/modules/business_rules/components/utils/formHelpers.js.map +2 -2
  18. package/dist/modules/catalog/backend/catalog/products/create/page.js +1 -1
  19. package/dist/modules/catalog/backend/catalog/products/create/page.js.map +2 -2
  20. package/dist/modules/catalog/commands/products.js +1 -1
  21. package/dist/modules/catalog/commands/products.js.map +2 -2
  22. package/dist/modules/catalog/commands/shared.js +1 -1
  23. package/dist/modules/catalog/commands/shared.js.map +2 -2
  24. package/dist/modules/catalog/components/PriceKindSettings.js +1 -1
  25. package/dist/modules/catalog/components/PriceKindSettings.js.map +2 -2
  26. package/dist/modules/catalog/components/products/productForm.js +1 -1
  27. package/dist/modules/catalog/components/products/productForm.js.map +2 -2
  28. package/dist/modules/configs/lib/upgrade-actions.js.map +1 -1
  29. package/dist/modules/currencies/services/providers/raiffeisen.js +1 -1
  30. package/dist/modules/currencies/services/providers/raiffeisen.js.map +2 -2
  31. package/dist/modules/customers/backend/customers/companies/page.js +3 -3
  32. package/dist/modules/customers/backend/customers/companies/page.js.map +2 -2
  33. package/dist/modules/customers/backend/customers/deals/page.js +3 -3
  34. package/dist/modules/customers/backend/customers/deals/page.js.map +2 -2
  35. package/dist/modules/customers/backend/customers/people/page.js +3 -3
  36. package/dist/modules/customers/backend/customers/people/page.js.map +2 -2
  37. package/dist/modules/customers/cli.js +2 -2
  38. package/dist/modules/customers/cli.js.map +2 -2
  39. package/dist/modules/customers/lib/detailHelpers.js +1 -1
  40. package/dist/modules/customers/lib/detailHelpers.js.map +2 -2
  41. package/dist/modules/entities/cli.js +1 -1
  42. package/dist/modules/entities/cli.js.map +2 -2
  43. package/dist/modules/entities/lib/field-definitions.js +1 -1
  44. package/dist/modules/entities/lib/field-definitions.js.map +2 -2
  45. package/dist/modules/entities/lib/install-from-ce.js +1 -1
  46. package/dist/modules/entities/lib/install-from-ce.js.map +2 -2
  47. package/dist/modules/inbox_ops/lib/emailParser.js +1 -1
  48. package/dist/modules/inbox_ops/lib/emailParser.js.map +2 -2
  49. package/dist/modules/inbox_ops/widgets/notifications/ProposalCreatedRenderer.js +0 -8
  50. package/dist/modules/inbox_ops/widgets/notifications/ProposalCreatedRenderer.js.map +2 -2
  51. package/dist/modules/perspectives/services/perspectiveService.js +1 -1
  52. package/dist/modules/perspectives/services/perspectiveService.js.map +2 -2
  53. package/dist/modules/planner/backend/planner/availability-rulesets/page.js +3 -3
  54. package/dist/modules/planner/backend/planner/availability-rulesets/page.js.map +2 -2
  55. package/dist/modules/query_index/components/QueryIndexesTable.js +7 -7
  56. package/dist/modules/query_index/components/QueryIndexesTable.js.map +2 -2
  57. package/dist/modules/query_index/lib/engine.js +1 -1
  58. package/dist/modules/query_index/lib/engine.js.map +2 -2
  59. package/dist/modules/resources/backend/resources/resource-types/page.js +2 -2
  60. package/dist/modules/resources/backend/resources/resource-types/page.js.map +2 -2
  61. package/dist/modules/resources/backend/resources/resources/page.js +3 -3
  62. package/dist/modules/resources/backend/resources/resources/page.js.map +2 -2
  63. package/dist/modules/resources/commands/resources.js +1 -1
  64. package/dist/modules/resources/commands/resources.js.map +2 -2
  65. package/dist/modules/resources/lib/seeds.js +1 -1
  66. package/dist/modules/resources/lib/seeds.js.map +2 -2
  67. package/dist/modules/sales/api/dashboard/widgets/new-orders/route.js +1 -1
  68. package/dist/modules/sales/api/dashboard/widgets/new-orders/route.js.map +2 -2
  69. package/dist/modules/sales/api/dashboard/widgets/new-quotes/route.js +1 -1
  70. package/dist/modules/sales/api/dashboard/widgets/new-quotes/route.js.map +2 -2
  71. package/dist/modules/sales/backend/sales/channels/page.js +3 -3
  72. package/dist/modules/sales/backend/sales/channels/page.js.map +2 -2
  73. package/dist/modules/sales/commands/documents.js +2 -2
  74. package/dist/modules/sales/commands/documents.js.map +2 -2
  75. package/dist/modules/sales/components/channels/offerTableUtils.js +2 -3
  76. package/dist/modules/sales/components/channels/offerTableUtils.js.map +2 -2
  77. package/dist/modules/sales/components/documents/SalesDocumentsTable.js +4 -4
  78. package/dist/modules/sales/components/documents/SalesDocumentsTable.js.map +2 -2
  79. package/dist/modules/sales/components/documents/ShipmentDialog.js +1 -1
  80. package/dist/modules/sales/components/documents/ShipmentDialog.js.map +2 -2
  81. package/dist/modules/sales/lib/shipments/snapshots.js.map +1 -1
  82. package/dist/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.js +0 -8
  83. package/dist/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.js.map +2 -2
  84. package/dist/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.js +0 -8
  85. package/dist/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.js.map +2 -2
  86. package/dist/modules/staff/backend/staff/leave-requests/page.js +3 -3
  87. package/dist/modules/staff/backend/staff/leave-requests/page.js.map +2 -2
  88. package/dist/modules/staff/backend/staff/my-leave-requests/page.js +3 -3
  89. package/dist/modules/staff/backend/staff/my-leave-requests/page.js.map +2 -2
  90. package/dist/modules/staff/backend/staff/team-members/page.js +3 -3
  91. package/dist/modules/staff/backend/staff/team-members/page.js.map +2 -2
  92. package/dist/modules/staff/backend/staff/team-roles/page.js +2 -2
  93. package/dist/modules/staff/backend/staff/team-roles/page.js.map +2 -2
  94. package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js +3 -3
  95. package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js.map +2 -2
  96. package/dist/modules/staff/backend/staff/teams/page.js +2 -2
  97. package/dist/modules/staff/backend/staff/teams/page.js.map +2 -2
  98. package/dist/modules/workflows/backend/instances/page.js +2 -2
  99. package/dist/modules/workflows/backend/instances/page.js.map +2 -2
  100. package/dist/modules/workflows/backend/tasks/page.js +1 -1
  101. package/dist/modules/workflows/backend/tasks/page.js.map +2 -2
  102. package/dist/modules/workflows/components/DefinitionTriggersEditor.js +1 -1
  103. package/dist/modules/workflows/components/DefinitionTriggersEditor.js.map +2 -2
  104. package/dist/modules/workflows/lib/graph-utils.js +1 -1
  105. package/dist/modules/workflows/lib/graph-utils.js.map +2 -2
  106. package/package.json +2 -2
  107. package/src/modules/api_docs/frontend/docs/api/page.tsx +1 -1
  108. package/src/modules/attachments/api/library/[id]/route.ts +1 -0
  109. package/src/modules/attachments/components/AttachmentLibrary.tsx +1 -1
  110. package/src/modules/attachments/lib/partitionEnv.ts +1 -1
  111. package/src/modules/auth/backend/users/page.tsx +1 -1
  112. package/src/modules/auth/cli.ts +1 -1
  113. package/src/modules/auth/commands/users.ts +1 -1
  114. package/src/modules/business_rules/components/utils/formHelpers.ts +1 -1
  115. package/src/modules/catalog/backend/catalog/products/create/page.tsx +1 -1
  116. package/src/modules/catalog/commands/products.ts +1 -1
  117. package/src/modules/catalog/commands/shared.ts +1 -1
  118. package/src/modules/catalog/components/PriceKindSettings.tsx +1 -1
  119. package/src/modules/catalog/components/products/productForm.ts +1 -1
  120. package/src/modules/configs/lib/upgrade-actions.ts +1 -1
  121. package/src/modules/currencies/services/providers/raiffeisen.ts +1 -1
  122. package/src/modules/customers/backend/customers/companies/page.tsx +3 -3
  123. package/src/modules/customers/backend/customers/deals/page.tsx +3 -3
  124. package/src/modules/customers/backend/customers/people/page.tsx +3 -3
  125. package/src/modules/customers/cli.ts +2 -2
  126. package/src/modules/customers/lib/detailHelpers.ts +1 -1
  127. package/src/modules/entities/cli.ts +1 -1
  128. package/src/modules/entities/lib/field-definitions.ts +1 -1
  129. package/src/modules/entities/lib/install-from-ce.ts +1 -1
  130. package/src/modules/inbox_ops/lib/emailParser.ts +1 -1
  131. package/src/modules/inbox_ops/widgets/notifications/ProposalCreatedRenderer.tsx +0 -8
  132. package/src/modules/perspectives/services/perspectiveService.ts +1 -1
  133. package/src/modules/planner/backend/planner/availability-rulesets/page.tsx +4 -3
  134. package/src/modules/query_index/components/QueryIndexesTable.tsx +7 -7
  135. package/src/modules/query_index/lib/engine.ts +1 -1
  136. package/src/modules/resources/backend/resources/resource-types/page.tsx +3 -2
  137. package/src/modules/resources/backend/resources/resources/page.tsx +3 -3
  138. package/src/modules/resources/commands/resources.ts +1 -1
  139. package/src/modules/resources/lib/seeds.ts +1 -1
  140. package/src/modules/sales/api/dashboard/widgets/new-orders/route.ts +1 -1
  141. package/src/modules/sales/api/dashboard/widgets/new-quotes/route.ts +1 -1
  142. package/src/modules/sales/backend/sales/channels/page.tsx +3 -3
  143. package/src/modules/sales/commands/documents.ts +2 -2
  144. package/src/modules/sales/components/channels/offerTableUtils.tsx +2 -3
  145. package/src/modules/sales/components/documents/SalesDocumentsTable.tsx +4 -4
  146. package/src/modules/sales/components/documents/ShipmentDialog.tsx +1 -1
  147. package/src/modules/sales/lib/shipments/snapshots.ts +1 -1
  148. package/src/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.tsx +0 -8
  149. package/src/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.tsx +0 -8
  150. package/src/modules/staff/backend/staff/leave-requests/page.tsx +3 -3
  151. package/src/modules/staff/backend/staff/my-leave-requests/page.tsx +3 -3
  152. package/src/modules/staff/backend/staff/team-members/page.tsx +3 -3
  153. package/src/modules/staff/backend/staff/team-roles/page.tsx +2 -2
  154. package/src/modules/staff/backend/staff/teams/[id]/edit/page.tsx +4 -3
  155. package/src/modules/staff/backend/staff/teams/page.tsx +3 -2
  156. package/src/modules/workflows/backend/instances/page.tsx +2 -2
  157. package/src/modules/workflows/backend/tasks/page.tsx +1 -1
  158. package/src/modules/workflows/components/DefinitionTriggersEditor.tsx +1 -1
  159. package/src/modules/workflows/lib/graph-utils.ts +1 -1
  160. package/dist/modules/integrations/acl.js +0 -8
  161. package/dist/modules/integrations/acl.js.map +0 -7
  162. package/dist/modules/integrations/data/enrichers.js +0 -72
  163. package/dist/modules/integrations/data/enrichers.js.map +0 -7
  164. package/dist/modules/integrations/data/entities.js +0 -63
  165. package/dist/modules/integrations/data/entities.js.map +0 -7
  166. package/dist/modules/integrations/index.js +0 -9
  167. package/dist/modules/integrations/index.js.map +0 -7
  168. package/dist/modules/integrations/setup.js +0 -13
  169. package/dist/modules/integrations/setup.js.map +0 -7
  170. package/dist/modules/integrations/widgets/injection/external-ids/widget.client.js +0 -69
  171. package/dist/modules/integrations/widgets/injection/external-ids/widget.client.js.map +0 -7
  172. package/dist/modules/integrations/widgets/injection-table.js +0 -13
  173. package/dist/modules/integrations/widgets/injection-table.js.map +0 -7
  174. package/src/modules/integrations/acl.ts +0 -4
  175. package/src/modules/integrations/data/enrichers.ts +0 -98
  176. package/src/modules/integrations/data/entities.ts +0 -46
  177. package/src/modules/integrations/index.ts +0 -5
  178. package/src/modules/integrations/setup.ts +0 -11
  179. package/src/modules/integrations/widgets/injection/external-ids/widget.client.tsx +0 -94
  180. package/src/modules/integrations/widgets/injection-table.ts +0 -17
package/AGENTS.md CHANGED
@@ -258,28 +258,6 @@ Hosts expose consistent spot ids:
258
258
  - `admin.page:<path>:before|after` — admin pages
259
259
  - `menu:sidebar:main` — main sidebar items/groups
260
260
  - `menu:sidebar:settings` — settings sidebar
261
-
262
- DataTable deep-extension surfaces:
263
- - `data-table:<tableId>:columns`
264
- - `data-table:<tableId>:row-actions`
265
- - `data-table:<tableId>:bulk-actions`
266
- - `data-table:<tableId>:filters`
267
-
268
- CrudForm field-injection surface:
269
- - `crud-form:<entityId>:fields`
270
-
271
- ## API Interceptors
272
-
273
- Define route interceptors in `api/interceptors.ts` and export `interceptors`.
274
- - Keep scope explicit with `targetRoute` + `methods`; use wildcards only when required.
275
- - `before`/`after` hooks must be fail-closed and timeout-safe.
276
- - If `before` rewrites body/query, return a schema-compatible payload (route handler re-validates it).
277
-
278
- ## Component Replacement
279
-
280
- Define component overrides in `widgets/components.ts` and export `componentOverrides`.
281
- - Prefer handle-based targets (`page:*`, `data-table:*`, `crud-form:*`, `section:*`) for deterministic replacement.
282
- - Use wrapper/props-transform modes when possible; replacement mode should preserve props compatibility.
283
261
  - `menu:sidebar:profile` — profile sidebar
284
262
  - `menu:topbar:profile-dropdown` — user/profile dropdown
285
263
  - `menu:topbar:actions` — header action area
@@ -6,7 +6,7 @@ import { resolveApiDocsBaseUrl } from "@open-mercato/core/modules/api_docs/lib/r
6
6
  import { APP_VERSION } from "@open-mercato/shared/lib/version";
7
7
  function collectOperations(doc) {
8
8
  const operations = [];
9
- const paths = Object.keys(doc.paths ?? {}).sort((a, b) => a.localeCompare(b));
9
+ const paths = Object.keys(doc.paths ?? {}).sort();
10
10
  for (const path of paths) {
11
11
  const methodEntries = Object.entries(doc.paths[path] ?? {});
12
12
  for (const [method, operation] of methodEntries) {
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../../src/modules/api_docs/frontend/docs/api/page.tsx"],
4
- "sourcesContent": ["import ApiDocsExplorer from './Explorer'\nimport { getModules } from '@open-mercato/shared/lib/i18n/server'\nimport { buildOpenApiDocument } from '@open-mercato/shared/lib/openapi'\nimport { resolveApiDocsBaseUrl } from '@open-mercato/core/modules/api_docs/lib/resources'\nimport { APP_VERSION } from '@open-mercato/shared/lib/version'\n\ntype ExplorerOperation = {\n id: string\n path: string\n method: string\n tag: string\n summary?: string\n description?: string\n operation: any\n}\n\nfunction collectOperations(doc: any): ExplorerOperation[] {\n const operations: ExplorerOperation[] = []\n const paths = Object.keys(doc.paths ?? {}).sort((a, b) => a.localeCompare(b))\n for (const path of paths) {\n const methodEntries = Object.entries(doc.paths[path] ?? {})\n for (const [method, operation] of methodEntries) {\n const methodUpper = method.toUpperCase()\n const op = (operation ?? {}) as { summary?: unknown; description?: unknown; tags?: unknown }\n const summary: string | undefined = typeof op.summary === 'string' ? op.summary : undefined\n const description: string | undefined = typeof op.description === 'string' ? op.description : undefined\n const tag = Array.isArray(op.tags) && typeof op.tags[0] === 'string' ? op.tags[0] : 'General'\n operations.push({\n id: `${methodUpper}-${path.replace(/[^\\w]+/g, '-')}`,\n path,\n method: methodUpper,\n tag,\n summary,\n description,\n operation,\n })\n }\n }\n return operations\n}\n\nfunction buildTagOrder(doc: any, operations: ExplorerOperation[]): string[] {\n const fromDoc = Array.isArray(doc.tags) ? doc.tags.map((tag: any) => tag?.name).filter(Boolean) : []\n const fromOps = Array.from(new Set(operations.map((operation) => operation.tag)))\n const order: string[] = []\n for (const tag of [...fromDoc, ...fromOps]) {\n if (typeof tag !== 'string') continue\n if (!order.includes(tag)) order.push(tag)\n }\n return order\n}\n\nexport default async function ApiDocsViewerPage() {\n const baseUrl = resolveApiDocsBaseUrl()\n const modules = getModules()\n const doc = buildOpenApiDocument(modules, {\n title: 'Open Mercato API',\n version: APP_VERSION,\n description: 'Auto-generated OpenAPI definition for all enabled modules.',\n servers: [{ url: baseUrl, description: 'Default environment' }],\n baseUrlForExamples: baseUrl,\n defaultSecurity: ['bearerAuth'],\n })\n\n const operations = collectOperations(doc)\n const tagOrder = buildTagOrder(doc, operations)\n\n return (\n <ApiDocsExplorer\n title={doc.info?.title ?? 'Open Mercato API'}\n version={doc.info?.version ?? APP_VERSION}\n description={doc.info?.description}\n operations={operations}\n tagOrder={tagOrder}\n servers={doc.servers ?? []}\n docsUrl=\"https://docs.openmercato.com\"\n jsonSpecUrl=\"/api/docs/openapi\"\n markdownSpecUrl=\"/api/docs/markdown\"\n />\n )\n}\n"],
5
- "mappings": "AAoEI;AApEJ,OAAO,qBAAqB;AAC5B,SAAS,kBAAkB;AAC3B,SAAS,4BAA4B;AACrC,SAAS,6BAA6B;AACtC,SAAS,mBAAmB;AAY5B,SAAS,kBAAkB,KAA+B;AACxD,QAAM,aAAkC,CAAC;AACzC,QAAM,QAAQ,OAAO,KAAK,IAAI,SAAS,CAAC,CAAC,EAAE,KAAK,CAAC,GAAG,MAAM,EAAE,cAAc,CAAC,CAAC;AAC5E,aAAW,QAAQ,OAAO;AACxB,UAAM,gBAAgB,OAAO,QAAQ,IAAI,MAAM,IAAI,KAAK,CAAC,CAAC;AAC1D,eAAW,CAAC,QAAQ,SAAS,KAAK,eAAe;AAC/C,YAAM,cAAc,OAAO,YAAY;AACvC,YAAM,KAAM,aAAa,CAAC;AAC1B,YAAM,UAA8B,OAAO,GAAG,YAAY,WAAW,GAAG,UAAU;AAClF,YAAM,cAAkC,OAAO,GAAG,gBAAgB,WAAW,GAAG,cAAc;AAC9F,YAAM,MAAM,MAAM,QAAQ,GAAG,IAAI,KAAK,OAAO,GAAG,KAAK,CAAC,MAAM,WAAW,GAAG,KAAK,CAAC,IAAI;AACpF,iBAAW,KAAK;AAAA,QACd,IAAI,GAAG,WAAW,IAAI,KAAK,QAAQ,WAAW,GAAG,CAAC;AAAA,QAClD;AAAA,QACA,QAAQ;AAAA,QACR;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,MACF,CAAC;AAAA,IACH;AAAA,EACF;AACA,SAAO;AACT;AAEA,SAAS,cAAc,KAAU,YAA2C;AAC1E,QAAM,UAAU,MAAM,QAAQ,IAAI,IAAI,IAAI,IAAI,KAAK,IAAI,CAAC,QAAa,KAAK,IAAI,EAAE,OAAO,OAAO,IAAI,CAAC;AACnG,QAAM,UAAU,MAAM,KAAK,IAAI,IAAI,WAAW,IAAI,CAAC,cAAc,UAAU,GAAG,CAAC,CAAC;AAChF,QAAM,QAAkB,CAAC;AACzB,aAAW,OAAO,CAAC,GAAG,SAAS,GAAG,OAAO,GAAG;AAC1C,QAAI,OAAO,QAAQ,SAAU;AAC7B,QAAI,CAAC,MAAM,SAAS,GAAG,EAAG,OAAM,KAAK,GAAG;AAAA,EAC1C;AACA,SAAO;AACT;AAEA,eAAO,oBAA2C;AAChD,QAAM,UAAU,sBAAsB;AACtC,QAAM,UAAU,WAAW;AAC3B,QAAM,MAAM,qBAAqB,SAAS;AAAA,IACxC,OAAO;AAAA,IACP,SAAS;AAAA,IACT,aAAa;AAAA,IACb,SAAS,CAAC,EAAE,KAAK,SAAS,aAAa,sBAAsB,CAAC;AAAA,IAC9D,oBAAoB;AAAA,IACpB,iBAAiB,CAAC,YAAY;AAAA,EAChC,CAAC;AAED,QAAM,aAAa,kBAAkB,GAAG;AACxC,QAAM,WAAW,cAAc,KAAK,UAAU;AAE9C,SACE;AAAA,IAAC;AAAA;AAAA,MACC,OAAO,IAAI,MAAM,SAAS;AAAA,MAC1B,SAAS,IAAI,MAAM,WAAW;AAAA,MAC9B,aAAa,IAAI,MAAM;AAAA,MACvB;AAAA,MACA;AAAA,MACA,SAAS,IAAI,WAAW,CAAC;AAAA,MACzB,SAAQ;AAAA,MACR,aAAY;AAAA,MACZ,iBAAgB;AAAA;AAAA,EAClB;AAEJ;",
4
+ "sourcesContent": ["import ApiDocsExplorer from './Explorer'\nimport { getModules } from '@open-mercato/shared/lib/i18n/server'\nimport { buildOpenApiDocument } from '@open-mercato/shared/lib/openapi'\nimport { resolveApiDocsBaseUrl } from '@open-mercato/core/modules/api_docs/lib/resources'\nimport { APP_VERSION } from '@open-mercato/shared/lib/version'\n\ntype ExplorerOperation = {\n id: string\n path: string\n method: string\n tag: string\n summary?: string\n description?: string\n operation: any\n}\n\nfunction collectOperations(doc: any): ExplorerOperation[] {\n const operations: ExplorerOperation[] = []\n const paths = Object.keys(doc.paths ?? {}).sort()\n for (const path of paths) {\n const methodEntries = Object.entries(doc.paths[path] ?? {})\n for (const [method, operation] of methodEntries) {\n const methodUpper = method.toUpperCase()\n const op = (operation ?? {}) as { summary?: unknown; description?: unknown; tags?: unknown }\n const summary: string | undefined = typeof op.summary === 'string' ? op.summary : undefined\n const description: string | undefined = typeof op.description === 'string' ? op.description : undefined\n const tag = Array.isArray(op.tags) && typeof op.tags[0] === 'string' ? op.tags[0] : 'General'\n operations.push({\n id: `${methodUpper}-${path.replace(/[^\\w]+/g, '-')}`,\n path,\n method: methodUpper,\n tag,\n summary,\n description,\n operation,\n })\n }\n }\n return operations\n}\n\nfunction buildTagOrder(doc: any, operations: ExplorerOperation[]): string[] {\n const fromDoc = Array.isArray(doc.tags) ? doc.tags.map((tag: any) => tag?.name).filter(Boolean) : []\n const fromOps = Array.from(new Set(operations.map((operation) => operation.tag)))\n const order: string[] = []\n for (const tag of [...fromDoc, ...fromOps]) {\n if (typeof tag !== 'string') continue\n if (!order.includes(tag)) order.push(tag)\n }\n return order\n}\n\nexport default async function ApiDocsViewerPage() {\n const baseUrl = resolveApiDocsBaseUrl()\n const modules = getModules()\n const doc = buildOpenApiDocument(modules, {\n title: 'Open Mercato API',\n version: APP_VERSION,\n description: 'Auto-generated OpenAPI definition for all enabled modules.',\n servers: [{ url: baseUrl, description: 'Default environment' }],\n baseUrlForExamples: baseUrl,\n defaultSecurity: ['bearerAuth'],\n })\n\n const operations = collectOperations(doc)\n const tagOrder = buildTagOrder(doc, operations)\n\n return (\n <ApiDocsExplorer\n title={doc.info?.title ?? 'Open Mercato API'}\n version={doc.info?.version ?? APP_VERSION}\n description={doc.info?.description}\n operations={operations}\n tagOrder={tagOrder}\n servers={doc.servers ?? []}\n docsUrl=\"https://docs.openmercato.com\"\n jsonSpecUrl=\"/api/docs/openapi\"\n markdownSpecUrl=\"/api/docs/markdown\"\n />\n )\n}\n"],
5
+ "mappings": "AAoEI;AApEJ,OAAO,qBAAqB;AAC5B,SAAS,kBAAkB;AAC3B,SAAS,4BAA4B;AACrC,SAAS,6BAA6B;AACtC,SAAS,mBAAmB;AAY5B,SAAS,kBAAkB,KAA+B;AACxD,QAAM,aAAkC,CAAC;AACzC,QAAM,QAAQ,OAAO,KAAK,IAAI,SAAS,CAAC,CAAC,EAAE,KAAK;AAChD,aAAW,QAAQ,OAAO;AACxB,UAAM,gBAAgB,OAAO,QAAQ,IAAI,MAAM,IAAI,KAAK,CAAC,CAAC;AAC1D,eAAW,CAAC,QAAQ,SAAS,KAAK,eAAe;AAC/C,YAAM,cAAc,OAAO,YAAY;AACvC,YAAM,KAAM,aAAa,CAAC;AAC1B,YAAM,UAA8B,OAAO,GAAG,YAAY,WAAW,GAAG,UAAU;AAClF,YAAM,cAAkC,OAAO,GAAG,gBAAgB,WAAW,GAAG,cAAc;AAC9F,YAAM,MAAM,MAAM,QAAQ,GAAG,IAAI,KAAK,OAAO,GAAG,KAAK,CAAC,MAAM,WAAW,GAAG,KAAK,CAAC,IAAI;AACpF,iBAAW,KAAK;AAAA,QACd,IAAI,GAAG,WAAW,IAAI,KAAK,QAAQ,WAAW,GAAG,CAAC;AAAA,QAClD;AAAA,QACA,QAAQ;AAAA,QACR;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,MACF,CAAC;AAAA,IACH;AAAA,EACF;AACA,SAAO;AACT;AAEA,SAAS,cAAc,KAAU,YAA2C;AAC1E,QAAM,UAAU,MAAM,QAAQ,IAAI,IAAI,IAAI,IAAI,KAAK,IAAI,CAAC,QAAa,KAAK,IAAI,EAAE,OAAO,OAAO,IAAI,CAAC;AACnG,QAAM,UAAU,MAAM,KAAK,IAAI,IAAI,WAAW,IAAI,CAAC,cAAc,UAAU,GAAG,CAAC,CAAC;AAChF,QAAM,QAAkB,CAAC;AACzB,aAAW,OAAO,CAAC,GAAG,SAAS,GAAG,OAAO,GAAG;AAC1C,QAAI,OAAO,QAAQ,SAAU;AAC7B,QAAI,CAAC,MAAM,SAAS,GAAG,EAAG,OAAM,KAAK,GAAG;AAAA,EAC1C;AACA,SAAO;AACT;AAEA,eAAO,oBAA2C;AAChD,QAAM,UAAU,sBAAsB;AACtC,QAAM,UAAU,WAAW;AAC3B,QAAM,MAAM,qBAAqB,SAAS;AAAA,IACxC,OAAO;AAAA,IACP,SAAS;AAAA,IACT,aAAa;AAAA,IACb,SAAS,CAAC,EAAE,KAAK,SAAS,aAAa,sBAAsB,CAAC;AAAA,IAC9D,oBAAoB;AAAA,IACpB,iBAAiB,CAAC,YAAY;AAAA,EAChC,CAAC;AAED,QAAM,aAAa,kBAAkB,GAAG;AACxC,QAAM,WAAW,cAAc,KAAK,UAAU;AAE9C,SACE;AAAA,IAAC;AAAA;AAAA,MACC,OAAO,IAAI,MAAM,SAAS;AAAA,MAC1B,SAAS,IAAI,MAAM,WAAW;AAAA,MAC9B,aAAa,IAAI,MAAM;AAAA,MACvB;AAAA,MACA;AAAA,MACA,SAAS,IAAI,WAAW,CAAC;AAAA,MACzB,SAAQ;AAAA,MACR,aAAY;AAAA,MACZ,iBAAgB;AAAA;AAAA,EAClB;AAEJ;",
6
6
  "names": []
7
7
  }
@@ -39,6 +39,7 @@ const metadata = {
39
39
  };
40
40
  async function resolveAttachmentId(ctx) {
41
41
  const params = ctx?.params;
42
+ if (!params) return null;
42
43
  try {
43
44
  const { id } = await params;
44
45
  if (typeof id === "string" && id.trim().length) {
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../../src/modules/attachments/api/library/%5Bid%5D/route.ts"],
4
- "sourcesContent": ["import { NextRequest, NextResponse } from 'next/server'\nimport { z } from 'zod'\nimport type { OpenApiRouteDoc } from '@open-mercato/shared/lib/openapi'\nimport { getAuthFromRequest } from '@open-mercato/shared/lib/auth/server'\nimport { createRequestContainer } from '@open-mercato/shared/lib/di/container'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport { Attachment, AttachmentPartition } from '../../../data/entities'\nimport {\n mergeAttachmentMetadata,\n normalizeAttachmentAssignments,\n normalizeAttachmentTags,\n readAttachmentMetadata,\n} from '../../../lib/metadata'\nimport { deletePartitionFile } from '../../../lib/storage'\nimport { splitCustomFieldPayload, loadCustomFieldValues } from '@open-mercato/shared/lib/crud/custom-fields'\nimport { emitCrudSideEffects, setCustomFieldsIfAny } from '@open-mercato/shared/lib/commands/helpers'\nimport { normalizeCustomFieldResponse } from '@open-mercato/shared/lib/custom-fields/normalize'\nimport { E } from '#generated/entities.ids.generated'\nimport type { QueryEngine } from '@open-mercato/shared/lib/query/types'\nimport type { DataEngine } from '@open-mercato/shared/lib/data/engine'\nimport { attachmentCrudEvents, attachmentCrudIndexer } from '../../../lib/crud'\nimport { applyAssignmentEnrichments, resolveAssignmentEnrichments } from '../../../lib/assignmentDetails'\nimport {\n attachmentsTag,\n attachmentDetailResponseSchema,\n attachmentErrorSchema,\n} from '../../openapi'\n\nconst updateSchema = z.object({\n tags: z.array(z.string()).optional(),\n assignments: z\n .array(\n z.object({\n type: z.string().min(1),\n id: z.string().min(1),\n href: z.string().nullable().optional(),\n label: z.string().nullable().optional(),\n }),\n )\n .optional(),\n})\n\nexport const metadata = {\n GET: { requireAuth: true, requireFeatures: ['attachments.view'] },\n PATCH: { requireAuth: true, requireFeatures: ['attachments.manage'] },\n DELETE: { requireAuth: true, requireFeatures: ['attachments.manage'] },\n}\n\ntype RouteParams = { id: string }\ntype RouteContext = { params: Promise<RouteParams> }\n\nasync function resolveAttachmentId(ctx: RouteContext): Promise<string | null> {\n const params = ctx?.params\n try {\n const { id } = await params\n if (typeof id === 'string' && id.trim().length) {\n return id\n }\n return null\n } catch {\n return null\n }\n}\n\nexport async function GET(req: NextRequest, ctx: RouteContext) {\n const auth = await getAuthFromRequest(req)\n if (!auth || !auth.tenantId || (!auth.orgId && !auth.isSuperAdmin)) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })\n }\n const attachmentId = await resolveAttachmentId(ctx)\n if (!attachmentId) {\n return NextResponse.json({ error: 'Attachment id is required' }, { status: 400 })\n }\n const { resolve } = await createRequestContainer()\n const em = resolve('em') as EntityManager\n let queryEngine: QueryEngine | null = null\n try {\n queryEngine = resolve('queryEngine') as QueryEngine\n } catch {\n queryEngine = null\n }\n const findFilter: Record<string, unknown> = {\n id: attachmentId,\n tenantId: auth.tenantId,\n }\n if (auth.orgId) {\n findFilter.organizationId = auth.orgId\n }\n const record = await em.findOne(Attachment, findFilter)\n if (!record) {\n return NextResponse.json({ error: 'Attachment not found' }, { status: 404 })\n }\n const metadata = readAttachmentMetadata(record.storageMetadata)\n const partition = record.partitionCode\n ? await em.findOne(AttachmentPartition, { code: record.partitionCode })\n : null\n const customFieldValues = await loadCustomFieldValues({\n em,\n entityId: E.attachments.attachment,\n recordIds: [record.id],\n tenantIdByRecord: { [record.id]: record.tenantId ?? auth.tenantId ?? null },\n organizationIdByRecord: { [record.id]: record.organizationId ?? auth.orgId ?? null },\n tenantFallbacks: [auth.tenantId ?? null].filter((value): value is string => !!value),\n })\n const customFields = normalizeCustomFieldResponse(customFieldValues[record.id])\n const assignments = metadata.assignments ?? []\n const enrichments = await resolveAssignmentEnrichments(assignments, {\n queryEngine,\n tenantId: auth.tenantId,\n organizationId: auth.orgId,\n })\n const enrichedAssignments = applyAssignmentEnrichments(assignments, enrichments)\n return NextResponse.json({\n item: {\n id: record.id,\n fileName: record.fileName,\n fileSize: record.fileSize,\n mimeType: record.mimeType,\n partitionCode: record.partitionCode,\n partitionTitle: partition?.title ?? null,\n tags: metadata.tags ?? [],\n assignments: enrichedAssignments,\n content: record.content ?? null,\n customFields,\n },\n })\n}\n\nexport async function PATCH(req: NextRequest, ctx: RouteContext) {\n const auth = await getAuthFromRequest(req)\n if (!auth || !auth.tenantId || !auth.orgId) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })\n }\n const attachmentId = await resolveAttachmentId(ctx)\n if (!attachmentId) {\n return NextResponse.json({ error: 'Attachment id is required' }, { status: 400 })\n }\n const rawBody = await req.json().catch(() => null)\n const { base, custom } = splitCustomFieldPayload(rawBody)\n const parsed = updateSchema.safeParse(base)\n if (!parsed.success) {\n return NextResponse.json({ error: 'Invalid payload' }, { status: 400 })\n }\n const { resolve } = await createRequestContainer()\n const em = resolve('em') as EntityManager\n let queryEngine: QueryEngine | null = null\n try {\n queryEngine = resolve('queryEngine') as QueryEngine\n } catch {\n queryEngine = null\n }\n const dataEngine = resolve('dataEngine') as DataEngine\n const patchFilter: Record<string, unknown> = {\n id: attachmentId,\n tenantId: auth.tenantId,\n organizationId: auth.orgId,\n }\n const record = await em.findOne(Attachment, patchFilter)\n if (!record) {\n return NextResponse.json({ error: 'Attachment not found' }, { status: 404 })\n }\n const patch: Record<string, unknown> = {}\n if (parsed.data.tags) patch.tags = normalizeAttachmentTags(parsed.data.tags)\n if (parsed.data.assignments) patch.assignments = normalizeAttachmentAssignments(parsed.data.assignments)\n record.storageMetadata = mergeAttachmentMetadata(record.storageMetadata, patch)\n await em.flush()\n\n if (dataEngine && custom && Object.keys(custom).length) {\n try {\n await setCustomFieldsIfAny({\n dataEngine,\n entityId: E.attachments.attachment,\n recordId: record.id,\n tenantId: record.tenantId ?? auth.tenantId ?? null,\n organizationId: record.organizationId ?? auth.orgId ?? null,\n values: custom,\n })\n } catch (error) {\n console.error('[attachments] failed to persist custom attributes', error)\n return NextResponse.json({ error: 'Failed to save attachment attributes.' }, { status: 500 })\n }\n }\n\n if (dataEngine) {\n await emitCrudSideEffects({\n dataEngine,\n action: 'updated',\n entity: record,\n identifiers: {\n id: record.id,\n organizationId: record.organizationId ?? auth.orgId ?? null,\n tenantId: record.tenantId ?? auth.tenantId ?? null,\n },\n events: attachmentCrudEvents,\n indexer: attachmentCrudIndexer,\n })\n await dataEngine.flushOrmEntityChanges()\n }\n\n const metadata = readAttachmentMetadata(record.storageMetadata)\n const assignments = metadata.assignments ?? []\n const enrichments = await resolveAssignmentEnrichments(assignments, {\n queryEngine,\n tenantId: auth.tenantId,\n organizationId: auth.orgId,\n })\n const enrichedAssignments = applyAssignmentEnrichments(assignments, enrichments)\n return NextResponse.json({\n ok: true,\n item: {\n id: record.id,\n tags: metadata.tags ?? [],\n assignments: enrichedAssignments,\n customFields: normalizeCustomFieldResponse(custom ?? null),\n },\n })\n}\n\nexport async function DELETE(req: NextRequest, ctx: RouteContext) {\n const auth = await getAuthFromRequest(req)\n if (!auth || !auth.tenantId || !auth.orgId) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })\n }\n const attachmentId = await resolveAttachmentId(ctx)\n if (!attachmentId) {\n return NextResponse.json({ error: 'Attachment id is required' }, { status: 400 })\n }\n const { resolve } = await createRequestContainer()\n const em = resolve('em') as EntityManager\n const dataEngine = resolve('dataEngine') as DataEngine\n const deleteFilter: Record<string, unknown> = {\n id: attachmentId,\n tenantId: auth.tenantId,\n organizationId: auth.orgId,\n }\n const record = await em.findOne(Attachment, deleteFilter)\n if (!record) {\n return NextResponse.json({ error: 'Attachment not found' }, { status: 404 })\n }\n\n await deletePartitionFile(record.partitionCode, record.storagePath, record.storageDriver)\n await em.removeAndFlush(record)\n\n if (dataEngine) {\n await emitCrudSideEffects({\n dataEngine,\n action: 'deleted',\n entity: record,\n identifiers: {\n id: record.id,\n organizationId: record.organizationId ?? auth.orgId ?? null,\n tenantId: record.tenantId ?? auth.tenantId ?? null,\n },\n events: attachmentCrudEvents,\n indexer: attachmentCrudIndexer,\n })\n await dataEngine.flushOrmEntityChanges()\n }\n\n return NextResponse.json({ ok: true })\n}\n\nexport const openApi: OpenApiRouteDoc = {\n tag: attachmentsTag,\n summary: 'Attachment detail management',\n methods: {\n GET: {\n summary: 'Get attachment details',\n description: 'Returns complete details of an attachment including metadata, tags, assignments, and custom fields.',\n responses: [\n { status: 200, description: 'Attachment details', schema: attachmentDetailResponseSchema },\n ],\n errors: [\n { status: 400, description: 'Invalid attachment ID', schema: attachmentErrorSchema },\n { status: 401, description: 'Unauthorized', schema: attachmentErrorSchema },\n { status: 404, description: 'Attachment not found', schema: attachmentErrorSchema },\n ],\n },\n PATCH: {\n summary: 'Update attachment metadata',\n description: 'Updates attachment tags, assignments, and custom fields. Emits CRUD side effects for indexing and events.',\n requestBody: {\n contentType: 'application/json',\n schema: updateSchema,\n },\n responses: [\n { status: 200, description: 'Attachment updated successfully', schema: z.object({ ok: z.literal(true), item: z.any() }) },\n ],\n errors: [\n { status: 400, description: 'Invalid payload or attachment ID', schema: attachmentErrorSchema },\n { status: 401, description: 'Unauthorized', schema: attachmentErrorSchema },\n { status: 404, description: 'Attachment not found', schema: attachmentErrorSchema },\n { status: 500, description: 'Failed to save attributes', schema: attachmentErrorSchema },\n ],\n },\n DELETE: {\n summary: 'Delete attachment',\n description: 'Permanently deletes an attachment file from storage and database. Emits CRUD side effects.',\n responses: [\n { status: 200, description: 'Attachment deleted successfully', schema: z.object({ ok: z.literal(true) }) },\n ],\n errors: [\n { status: 400, description: 'Invalid attachment ID', schema: attachmentErrorSchema },\n { status: 401, description: 'Unauthorized', schema: attachmentErrorSchema },\n { status: 404, description: 'Attachment not found', schema: attachmentErrorSchema },\n ],\n },\n },\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import { NextRequest, NextResponse } from 'next/server'\nimport { z } from 'zod'\nimport type { OpenApiRouteDoc } from '@open-mercato/shared/lib/openapi'\nimport { getAuthFromRequest } from '@open-mercato/shared/lib/auth/server'\nimport { createRequestContainer } from '@open-mercato/shared/lib/di/container'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport { Attachment, AttachmentPartition } from '../../../data/entities'\nimport {\n mergeAttachmentMetadata,\n normalizeAttachmentAssignments,\n normalizeAttachmentTags,\n readAttachmentMetadata,\n} from '../../../lib/metadata'\nimport { deletePartitionFile } from '../../../lib/storage'\nimport { splitCustomFieldPayload, loadCustomFieldValues } from '@open-mercato/shared/lib/crud/custom-fields'\nimport { emitCrudSideEffects, setCustomFieldsIfAny } from '@open-mercato/shared/lib/commands/helpers'\nimport { normalizeCustomFieldResponse } from '@open-mercato/shared/lib/custom-fields/normalize'\nimport { E } from '#generated/entities.ids.generated'\nimport type { QueryEngine } from '@open-mercato/shared/lib/query/types'\nimport type { DataEngine } from '@open-mercato/shared/lib/data/engine'\nimport { attachmentCrudEvents, attachmentCrudIndexer } from '../../../lib/crud'\nimport { applyAssignmentEnrichments, resolveAssignmentEnrichments } from '../../../lib/assignmentDetails'\nimport {\n attachmentsTag,\n attachmentDetailResponseSchema,\n attachmentErrorSchema,\n} from '../../openapi'\n\nconst updateSchema = z.object({\n tags: z.array(z.string()).optional(),\n assignments: z\n .array(\n z.object({\n type: z.string().min(1),\n id: z.string().min(1),\n href: z.string().nullable().optional(),\n label: z.string().nullable().optional(),\n }),\n )\n .optional(),\n})\n\nexport const metadata = {\n GET: { requireAuth: true, requireFeatures: ['attachments.view'] },\n PATCH: { requireAuth: true, requireFeatures: ['attachments.manage'] },\n DELETE: { requireAuth: true, requireFeatures: ['attachments.manage'] },\n}\n\ntype RouteParams = { id: string }\ntype RouteContext = { params: Promise<RouteParams> }\n\nasync function resolveAttachmentId(ctx: RouteContext): Promise<string | null> {\n const params = ctx?.params\n if (!params) return null\n try {\n const { id } = await params\n if (typeof id === 'string' && id.trim().length) {\n return id\n }\n return null\n } catch {\n return null\n }\n}\n\nexport async function GET(req: NextRequest, ctx: RouteContext) {\n const auth = await getAuthFromRequest(req)\n if (!auth || !auth.tenantId || (!auth.orgId && !auth.isSuperAdmin)) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })\n }\n const attachmentId = await resolveAttachmentId(ctx)\n if (!attachmentId) {\n return NextResponse.json({ error: 'Attachment id is required' }, { status: 400 })\n }\n const { resolve } = await createRequestContainer()\n const em = resolve('em') as EntityManager\n let queryEngine: QueryEngine | null = null\n try {\n queryEngine = resolve('queryEngine') as QueryEngine\n } catch {\n queryEngine = null\n }\n const findFilter: Record<string, unknown> = {\n id: attachmentId,\n tenantId: auth.tenantId,\n }\n if (auth.orgId) {\n findFilter.organizationId = auth.orgId\n }\n const record = await em.findOne(Attachment, findFilter)\n if (!record) {\n return NextResponse.json({ error: 'Attachment not found' }, { status: 404 })\n }\n const metadata = readAttachmentMetadata(record.storageMetadata)\n const partition = record.partitionCode\n ? await em.findOne(AttachmentPartition, { code: record.partitionCode })\n : null\n const customFieldValues = await loadCustomFieldValues({\n em,\n entityId: E.attachments.attachment,\n recordIds: [record.id],\n tenantIdByRecord: { [record.id]: record.tenantId ?? auth.tenantId ?? null },\n organizationIdByRecord: { [record.id]: record.organizationId ?? auth.orgId ?? null },\n tenantFallbacks: [auth.tenantId ?? null].filter((value): value is string => !!value),\n })\n const customFields = normalizeCustomFieldResponse(customFieldValues[record.id])\n const assignments = metadata.assignments ?? []\n const enrichments = await resolveAssignmentEnrichments(assignments, {\n queryEngine,\n tenantId: auth.tenantId,\n organizationId: auth.orgId,\n })\n const enrichedAssignments = applyAssignmentEnrichments(assignments, enrichments)\n return NextResponse.json({\n item: {\n id: record.id,\n fileName: record.fileName,\n fileSize: record.fileSize,\n mimeType: record.mimeType,\n partitionCode: record.partitionCode,\n partitionTitle: partition?.title ?? null,\n tags: metadata.tags ?? [],\n assignments: enrichedAssignments,\n content: record.content ?? null,\n customFields,\n },\n })\n}\n\nexport async function PATCH(req: NextRequest, ctx: RouteContext) {\n const auth = await getAuthFromRequest(req)\n if (!auth || !auth.tenantId || !auth.orgId) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })\n }\n const attachmentId = await resolveAttachmentId(ctx)\n if (!attachmentId) {\n return NextResponse.json({ error: 'Attachment id is required' }, { status: 400 })\n }\n const rawBody = await req.json().catch(() => null)\n const { base, custom } = splitCustomFieldPayload(rawBody)\n const parsed = updateSchema.safeParse(base)\n if (!parsed.success) {\n return NextResponse.json({ error: 'Invalid payload' }, { status: 400 })\n }\n const { resolve } = await createRequestContainer()\n const em = resolve('em') as EntityManager\n let queryEngine: QueryEngine | null = null\n try {\n queryEngine = resolve('queryEngine') as QueryEngine\n } catch {\n queryEngine = null\n }\n const dataEngine = resolve('dataEngine') as DataEngine\n const patchFilter: Record<string, unknown> = {\n id: attachmentId,\n tenantId: auth.tenantId,\n organizationId: auth.orgId,\n }\n const record = await em.findOne(Attachment, patchFilter)\n if (!record) {\n return NextResponse.json({ error: 'Attachment not found' }, { status: 404 })\n }\n const patch: Record<string, unknown> = {}\n if (parsed.data.tags) patch.tags = normalizeAttachmentTags(parsed.data.tags)\n if (parsed.data.assignments) patch.assignments = normalizeAttachmentAssignments(parsed.data.assignments)\n record.storageMetadata = mergeAttachmentMetadata(record.storageMetadata, patch)\n await em.flush()\n\n if (dataEngine && custom && Object.keys(custom).length) {\n try {\n await setCustomFieldsIfAny({\n dataEngine,\n entityId: E.attachments.attachment,\n recordId: record.id,\n tenantId: record.tenantId ?? auth.tenantId ?? null,\n organizationId: record.organizationId ?? auth.orgId ?? null,\n values: custom,\n })\n } catch (error) {\n console.error('[attachments] failed to persist custom attributes', error)\n return NextResponse.json({ error: 'Failed to save attachment attributes.' }, { status: 500 })\n }\n }\n\n if (dataEngine) {\n await emitCrudSideEffects({\n dataEngine,\n action: 'updated',\n entity: record,\n identifiers: {\n id: record.id,\n organizationId: record.organizationId ?? auth.orgId ?? null,\n tenantId: record.tenantId ?? auth.tenantId ?? null,\n },\n events: attachmentCrudEvents,\n indexer: attachmentCrudIndexer,\n })\n await dataEngine.flushOrmEntityChanges()\n }\n\n const metadata = readAttachmentMetadata(record.storageMetadata)\n const assignments = metadata.assignments ?? []\n const enrichments = await resolveAssignmentEnrichments(assignments, {\n queryEngine,\n tenantId: auth.tenantId,\n organizationId: auth.orgId,\n })\n const enrichedAssignments = applyAssignmentEnrichments(assignments, enrichments)\n return NextResponse.json({\n ok: true,\n item: {\n id: record.id,\n tags: metadata.tags ?? [],\n assignments: enrichedAssignments,\n customFields: normalizeCustomFieldResponse(custom ?? null),\n },\n })\n}\n\nexport async function DELETE(req: NextRequest, ctx: RouteContext) {\n const auth = await getAuthFromRequest(req)\n if (!auth || !auth.tenantId || !auth.orgId) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })\n }\n const attachmentId = await resolveAttachmentId(ctx)\n if (!attachmentId) {\n return NextResponse.json({ error: 'Attachment id is required' }, { status: 400 })\n }\n const { resolve } = await createRequestContainer()\n const em = resolve('em') as EntityManager\n const dataEngine = resolve('dataEngine') as DataEngine\n const deleteFilter: Record<string, unknown> = {\n id: attachmentId,\n tenantId: auth.tenantId,\n organizationId: auth.orgId,\n }\n const record = await em.findOne(Attachment, deleteFilter)\n if (!record) {\n return NextResponse.json({ error: 'Attachment not found' }, { status: 404 })\n }\n\n await deletePartitionFile(record.partitionCode, record.storagePath, record.storageDriver)\n await em.removeAndFlush(record)\n\n if (dataEngine) {\n await emitCrudSideEffects({\n dataEngine,\n action: 'deleted',\n entity: record,\n identifiers: {\n id: record.id,\n organizationId: record.organizationId ?? auth.orgId ?? null,\n tenantId: record.tenantId ?? auth.tenantId ?? null,\n },\n events: attachmentCrudEvents,\n indexer: attachmentCrudIndexer,\n })\n await dataEngine.flushOrmEntityChanges()\n }\n\n return NextResponse.json({ ok: true })\n}\n\nexport const openApi: OpenApiRouteDoc = {\n tag: attachmentsTag,\n summary: 'Attachment detail management',\n methods: {\n GET: {\n summary: 'Get attachment details',\n description: 'Returns complete details of an attachment including metadata, tags, assignments, and custom fields.',\n responses: [\n { status: 200, description: 'Attachment details', schema: attachmentDetailResponseSchema },\n ],\n errors: [\n { status: 400, description: 'Invalid attachment ID', schema: attachmentErrorSchema },\n { status: 401, description: 'Unauthorized', schema: attachmentErrorSchema },\n { status: 404, description: 'Attachment not found', schema: attachmentErrorSchema },\n ],\n },\n PATCH: {\n summary: 'Update attachment metadata',\n description: 'Updates attachment tags, assignments, and custom fields. Emits CRUD side effects for indexing and events.',\n requestBody: {\n contentType: 'application/json',\n schema: updateSchema,\n },\n responses: [\n { status: 200, description: 'Attachment updated successfully', schema: z.object({ ok: z.literal(true), item: z.any() }) },\n ],\n errors: [\n { status: 400, description: 'Invalid payload or attachment ID', schema: attachmentErrorSchema },\n { status: 401, description: 'Unauthorized', schema: attachmentErrorSchema },\n { status: 404, description: 'Attachment not found', schema: attachmentErrorSchema },\n { status: 500, description: 'Failed to save attributes', schema: attachmentErrorSchema },\n ],\n },\n DELETE: {\n summary: 'Delete attachment',\n description: 'Permanently deletes an attachment file from storage and database. Emits CRUD side effects.',\n responses: [\n { status: 200, description: 'Attachment deleted successfully', schema: z.object({ ok: z.literal(true) }) },\n ],\n errors: [\n { status: 400, description: 'Invalid attachment ID', schema: attachmentErrorSchema },\n { status: 401, description: 'Unauthorized', schema: attachmentErrorSchema },\n { status: 404, description: 'Attachment not found', schema: attachmentErrorSchema },\n ],\n },\n },\n}\n"],
5
+ "mappings": "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",
6
6
  "names": ["metadata"]
7
7
  }
@@ -44,7 +44,7 @@ function humanDate(value, locale) {
44
44
  return date.toLocaleString(locale ?? void 0);
45
45
  }
46
46
  function buildFilterSignature(values) {
47
- return JSON.stringify(values, Object.keys(values).sort((a, b) => a.localeCompare(b)));
47
+ return JSON.stringify(values, Object.keys(values).sort());
48
48
  }
49
49
  function resolveAbsoluteUrl(path) {
50
50
  if (!path) return path;
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../src/modules/attachments/components/AttachmentLibrary.tsx"],
4
- "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport type { ColumnDef, SortingState } from '@tanstack/react-table'\nimport { useQuery, useQueryClient } from '@tanstack/react-query'\nimport { DataTable } from '@open-mercato/ui/backend/DataTable'\nimport { RowActions } from '@open-mercato/ui/backend/RowActions'\nimport type { FilterDef, FilterValues } from '@open-mercato/ui/backend/FilterBar'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { Badge } from '@open-mercato/ui/primitives/badge'\nimport { Dialog, DialogContent, DialogHeader, DialogTitle } from '@open-mercato/ui/primitives/dialog'\nimport { Input } from '@open-mercato/ui/primitives/input'\nimport { TagsInput } from '@open-mercato/ui/backend/inputs/TagsInput'\nimport { Spinner } from '@open-mercato/ui/primitives/spinner'\nimport { CrudForm, type CrudField, type CrudFormGroup, type CrudCustomFieldRenderProps } from '@open-mercato/ui/backend/CrudForm'\nimport { collectCustomFieldValues } from '@open-mercato/ui/backend/utils/customFieldValues'\nimport { apiCall } from '@open-mercato/ui/backend/utils/apiCall'\nimport { flash } from '@open-mercato/ui/backend/FlashMessages'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { z } from 'zod'\nimport { E } from '#generated/entities.ids.generated'\nimport type { LucideIcon } from 'lucide-react'\nimport { Download, Plus, Upload, Trash2, File, FileText, FileSpreadsheet, FileArchive, FileAudio, FileVideo, FileCode } from 'lucide-react'\nimport { buildAttachmentFileUrl, buildAttachmentImageUrl, slugifyAttachmentFileName } from '@open-mercato/core/modules/attachments/lib/imageUrls'\nimport { cn } from '@open-mercato/shared/lib/utils'\nimport { AttachmentDeleteDialog, AttachmentMetadataDialog, type AttachmentItem, type AttachmentMetadataSavePayload, type AssignmentDraft } from '@open-mercato/ui/backend/detail'\n\ntype AttachmentAssignment = {\n type: string\n id: string\n href?: string | null\n label?: string | null\n}\n\ntype AttachmentRow = AttachmentItem\n\ntype AttachmentLibraryResponse = {\n items: AttachmentRow[]\n page: number\n pageSize: number\n total: number\n totalPages: number\n availableTags: string[]\n partitions: Array<{ code: string; title: string; description?: string | null; isPublic?: boolean }>\n error?: string\n}\n\nconst PAGE_SIZE = 25\nconst ENV_APP_URL = (process.env.NEXT_PUBLIC_APP_URL || '').replace(/\\/$/, '')\nconst LIBRARY_ENTITY_ID = 'attachments:library'\n\nfunction filterLibraryAssignments(assignments?: AttachmentAssignment[] | null): AttachmentAssignment[] {\n return (assignments ?? []).filter((assignment) => assignment.type !== LIBRARY_ENTITY_ID)\n}\n\nfunction formatFileSize(value: number): string {\n if (!Number.isFinite(value)) return '\u2014'\n if (value <= 0) return '0 B'\n const units = ['B', 'KB', 'MB', 'GB', 'TB']\n let idx = 0\n let current = value\n while (current >= 1024 && idx < units.length - 1) {\n current /= 1024\n idx += 1\n }\n return `${current.toFixed(idx === 0 ? 0 : 1)} ${units[idx]}`\n}\n\nfunction humanDate(value: string, locale?: string): string {\n const date = new Date(value)\n if (Number.isNaN(date.getTime())) return value\n return date.toLocaleString(locale ?? undefined)\n}\n\nfunction buildFilterSignature(values: FilterValues): string {\n return JSON.stringify(values, Object.keys(values).sort((a, b) => a.localeCompare(b)))\n}\n\nfunction resolveAbsoluteUrl(path: string): string {\n if (!path) return path\n if (/^https?:\\/\\//i.test(path)) return path\n const base =\n ENV_APP_URL ||\n (typeof window !== 'undefined' && window.location?.origin ? window.location.origin : '')\n if (!base) return path\n const normalizedBase = base.replace(/\\/$/, '')\n return `${normalizedBase}${path.startsWith('/') ? path : `/${path}`}`\n}\n\nfunction resolveFileExtension(fileName?: string | null): string {\n if (!fileName) return ''\n const normalized = fileName.trim()\n if (!normalized) return ''\n const lastDot = normalized.lastIndexOf('.')\n if (lastDot === -1 || lastDot === normalized.length - 1) return ''\n return normalized.slice(lastDot + 1).toLowerCase()\n}\n\nconst EXTENSION_ICON_MAP: Record<string, LucideIcon> = {\n pdf: FileText,\n doc: FileText,\n docx: FileText,\n txt: FileText,\n md: FileText,\n rtf: FileText,\n xls: FileSpreadsheet,\n xlsx: FileSpreadsheet,\n csv: FileSpreadsheet,\n ods: FileSpreadsheet,\n ppt: FileText,\n pptx: FileText,\n zip: FileArchive,\n gz: FileArchive,\n rar: FileArchive,\n tgz: FileArchive,\n '7z': FileArchive,\n tar: FileArchive,\n json: FileCode,\n js: FileCode,\n ts: FileCode,\n jsx: FileCode,\n tsx: FileCode,\n html: FileCode,\n css: FileCode,\n xml: FileCode,\n yaml: FileCode,\n yml: FileCode,\n mp3: FileAudio,\n wav: FileAudio,\n flac: FileAudio,\n ogg: FileAudio,\n mp4: FileVideo,\n mov: FileVideo,\n avi: FileVideo,\n webm: FileVideo,\n}\n\nconst MIME_FALLBACK_ICONS: Record<string, LucideIcon> = {\n audio: FileAudio,\n video: FileVideo,\n text: FileText,\n application: FileText,\n}\n\nfunction resolveAttachmentPlaceholder(mimeType?: string | null, fileName?: string | null): { icon: LucideIcon; label: string } {\n const extension = resolveFileExtension(fileName)\n const normalizedMime = typeof mimeType === 'string' ? mimeType.toLowerCase() : ''\n if (extension && EXTENSION_ICON_MAP[extension]) {\n return { icon: EXTENSION_ICON_MAP[extension], label: extension.toUpperCase() }\n }\n if (!extension && normalizedMime.includes('pdf')) {\n return { icon: FileText, label: 'PDF' }\n }\n if (!extension && normalizedMime.includes('zip')) {\n return { icon: FileArchive, label: 'ZIP' }\n }\n if (!extension && normalizedMime.includes('json')) {\n return { icon: FileCode, label: 'JSON' }\n }\n const mimeRoot = normalizedMime.split('/')[0] || ''\n if (mimeRoot && MIME_FALLBACK_ICONS[mimeRoot]) {\n return { icon: MIME_FALLBACK_ICONS[mimeRoot], label: mimeRoot.toUpperCase() }\n }\n const fallbackSource = extension || mimeRoot || 'file'\n const fallbackLabel = fallbackSource.slice(0, 6).toUpperCase()\n return { icon: File, label: fallbackLabel }\n}\n\nfunction normalizeCustomFieldSubmitValue(value: unknown): unknown {\n if (Array.isArray(value)) {\n return value.filter((entry) => entry !== undefined)\n }\n if (value === undefined) return null\n return value\n}\n\n\ntype AttachmentUploadFormValues = {\n files: File[]\n partitionCode?: string\n tags?: string[]\n assignments?: AssignmentDraft[]\n} & Record<string, unknown>\n\ntype AssignmentsEditorProps = {\n value: AssignmentDraft[]\n onChange: (next: AssignmentDraft[]) => void\n labels: {\n title: string\n description: string\n type: string\n id: string\n href: string\n label?: string\n add: string\n remove: string\n }\n disabled?: boolean\n}\n\nfunction AttachmentAssignmentsEditor({ value, onChange, labels, disabled }: AssignmentsEditorProps) {\n const handleChange = React.useCallback(\n (index: number, patch: Partial<AssignmentDraft>) => {\n onChange(value.map((entry, idx) => (idx === index ? { ...entry, ...patch } : entry)))\n },\n [onChange, value],\n )\n\n const handleRemove = React.useCallback(\n (index: number) => {\n onChange(value.filter((_, idx) => idx !== index))\n },\n [onChange, value],\n )\n\n const handleAdd = React.useCallback(() => {\n onChange([...value, { type: '', id: '', href: '', label: '' }])\n }, [onChange, value])\n\n return (\n <div className=\"space-y-2\">\n <div>\n <div className=\"text-sm font-medium\">{labels.title}</div>\n <div className=\"text-xs text-muted-foreground\">{labels.description}</div>\n </div>\n <div className=\"space-y-3\">\n {value.length === 0 ? (\n <div className=\"text-xs text-muted-foreground\">No assignments yet.</div>\n ) : (\n value.map((entry, index) => (\n <div key={`${index}-${entry.type}-${entry.id}`} className=\"rounded border p-3 space-y-2\">\n <div className=\"grid gap-2 sm:grid-cols-2\">\n <div className=\"space-y-1\">\n <label className=\"text-xs font-medium\">{labels.type}</label>\n <input\n className=\"w-full rounded border px-2 py-1 text-sm\"\n value={entry.type}\n disabled={disabled}\n onChange={(event) => handleChange(index, { type: event.target.value })}\n />\n </div>\n <div className=\"space-y-1\">\n <label className=\"text-xs font-medium\">{labels.id}</label>\n <input\n className=\"w-full rounded border px-2 py-1 text-sm\"\n value={entry.id}\n disabled={disabled}\n onChange={(event) => handleChange(index, { id: event.target.value })}\n />\n </div>\n </div>\n <div className=\"grid gap-2 sm:grid-cols-2\">\n <div className=\"space-y-1\">\n <label className=\"text-xs font-medium\">{labels.href}</label>\n <input\n className=\"w-full rounded border px-2 py-1 text-sm\"\n value={entry.href ?? ''}\n disabled={disabled}\n onChange={(event) => handleChange(index, { href: event.target.value })}\n />\n </div>\n {labels.label ? (\n <div className=\"space-y-1\">\n <label className=\"text-xs font-medium\">{labels.label}</label>\n <input\n className=\"w-full rounded border px-2 py-1 text-sm\"\n value={entry.label ?? ''}\n disabled={disabled}\n onChange={(event) => handleChange(index, { label: event.target.value })}\n />\n </div>\n ) : null}\n </div>\n <div className=\"flex justify-end\">\n <Button\n type=\"button\"\n size=\"sm\"\n variant=\"outline\"\n disabled={disabled}\n onClick={() => handleRemove(index)}\n className=\"inline-flex items-center gap-1 text-muted-foreground\"\n >\n <Trash2 className=\"h-4 w-4\" />\n {labels.remove}\n </Button>\n </div>\n </div>\n ))\n )}\n </div>\n <Button type=\"button\" variant=\"outline\" size=\"sm\" disabled={disabled} onClick={handleAdd} className=\"inline-flex items-center gap-1\">\n <Plus className=\"h-4 w-4\" />\n {labels.add}\n </Button>\n </div>\n )\n}\n\ntype AttachmentFilesFieldProps = CrudCustomFieldRenderProps & {\n labels: {\n dropHint: string\n choose: string\n uploading: string\n empty: string\n }\n uploading: boolean\n}\n\nfunction AttachmentFilesField({\n value,\n setValue,\n disabled,\n error,\n labels,\n uploading,\n}: AttachmentFilesFieldProps) {\n const files = React.useMemo(() => (Array.isArray(value) ? (value as File[]) : []), [value])\n const [isDragOver, setDragOver] = React.useState(false)\n const fileInputRef = React.useRef<HTMLInputElement | null>(null)\n\n const acceptFiles = React.useCallback(\n (list: FileList | null) => {\n if (!list?.length) return\n const dedupe = new Map<string, File>(files.map((file) => [`${file.name}:${file.size}`, file]))\n Array.from(list).forEach((file) => {\n dedupe.set(`${file.name}:${file.size}`, file)\n })\n setValue(Array.from(dedupe.values()))\n },\n [files, setValue],\n )\n\n const handleDrop = React.useCallback(\n (event: React.DragEvent<HTMLDivElement>) => {\n if (disabled || uploading) return\n event.preventDefault()\n event.stopPropagation()\n setDragOver(false)\n acceptFiles(event.dataTransfer?.files ?? null)\n },\n [acceptFiles, disabled, uploading],\n )\n\n const handleDragOver = React.useCallback(\n (event: React.DragEvent<HTMLDivElement>) => {\n if (disabled || uploading) return\n event.preventDefault()\n event.stopPropagation()\n setDragOver(true)\n },\n [disabled, uploading],\n )\n\n const handleDragLeave = React.useCallback((event: React.DragEvent<HTMLDivElement>) => {\n event.preventDefault()\n event.stopPropagation()\n setDragOver(false)\n }, [])\n\n const removeFile = React.useCallback(\n (name: string, size: number) => {\n if (disabled || uploading) return\n setValue(files.filter((file) => !(file.name === name && file.size === size)))\n },\n [disabled, files, setValue, uploading],\n )\n\n const pickFiles = React.useCallback(() => {\n if (disabled || uploading) return\n fileInputRef.current?.click()\n }, [disabled, uploading])\n\n const renderFileList = () => {\n if (!files.length) {\n return <p className=\"text-xs text-muted-foreground\">{labels.empty}</p>\n }\n return (\n <div className=\"space-y-2\">\n {files.map((candidate) => (\n <div key={`${candidate.name}-${candidate.size}`} className=\"flex items-center justify-between rounded border px-3 py-2 text-sm\">\n <div>\n <div className=\"font-medium\">{candidate.name}</div>\n <div className=\"text-xs text-muted-foreground\">{formatFileSize(candidate.size)}</div>\n </div>\n <Button\n type=\"button\"\n variant=\"ghost\"\n size=\"icon\"\n onClick={() => removeFile(candidate.name, candidate.size)}\n disabled={disabled || uploading}\n >\n <Trash2 className=\"h-4 w-4\" />\n </Button>\n </div>\n ))}\n </div>\n )\n }\n\n return (\n <div className=\"space-y-2\">\n <div\n className={cn(\n 'flex flex-col items-center justify-center rounded-lg border border-dashed p-6 text-center transition-colors',\n isDragOver ? 'border-primary bg-primary/5' : 'border-muted-foreground/30',\n disabled || uploading ? 'opacity-70' : '',\n )}\n onDrop={handleDrop}\n onDragOver={handleDragOver}\n onDragLeave={handleDragLeave}\n role=\"presentation\"\n >\n <Upload className=\"mx-auto h-6 w-6 text-muted-foreground\" />\n <p className=\"mt-2 text-sm text-muted-foreground\">{labels.dropHint}</p>\n <Button type=\"button\" variant=\"outline\" size=\"sm\" className=\"mt-4\" onClick={pickFiles} disabled={disabled || uploading}>\n {uploading ? labels.uploading : labels.choose}\n </Button>\n <input\n ref={fileInputRef}\n type=\"file\"\n className=\"hidden\"\n multiple\n onChange={(event) => {\n acceptFiles(event.target.files)\n event.currentTarget.value = ''\n }}\n disabled={disabled || uploading}\n />\n </div>\n {renderFileList()}\n {error ? <p className=\"text-xs font-medium text-red-600\">{error}</p> : null}\n </div>\n )\n}\n\ntype AttachmentUploadFormProps = {\n partitions: Array<{ code: string; title: string }>\n availableTags: string[]\n onUploaded: () => void\n onCancel: () => void\n}\n\nfunction AttachmentUploadForm({ partitions, availableTags, onUploaded, onCancel }: AttachmentUploadFormProps) {\n const t = useT()\n const [isUploading, setIsUploading] = React.useState(false)\n const [uploadProgress, setUploadProgress] = React.useState<{ completed: number; total: number }>({ completed: 0, total: 0 })\n\n const partitionOptions = React.useMemo(\n () =>\n partitions.map((entry) => ({\n value: entry.code,\n label: entry.title || entry.code,\n })),\n [partitions],\n )\n\n const assignmentLabels = React.useMemo(\n () => ({\n title: t('attachments.library.upload.assignments.title', 'Assignments'),\n description: t(\n 'attachments.library.upload.assignments.description',\n 'Optionally link this file to existing records now or add them later.',\n ),\n type: t('attachments.library.upload.assignments.type', 'Type'),\n id: t('attachments.library.upload.assignments.id', 'Record ID'),\n href: t('attachments.library.upload.assignments.href', 'Link'),\n label: t('attachments.library.upload.assignments.label', 'Label'),\n add: t('attachments.library.upload.assignments.add', 'Add assignment'),\n remove: t('attachments.library.upload.assignments.remove', 'Remove'),\n }),\n [t],\n )\n\n const formSchema = React.useMemo(\n () =>\n z\n .object({\n files: z.array(z.any()).min(1, { message: t('attachments.library.upload.fileRequired', 'Select at least one file to upload.') }),\n partitionCode: z.string().optional(),\n tags: z.array(z.string()).optional(),\n assignments: z\n .array(\n z.object({\n type: z.string().min(1),\n id: z.string().min(1),\n href: z.string().optional(),\n label: z.string().optional(),\n }),\n )\n .optional(),\n })\n .passthrough(),\n [t],\n )\n\n const fields = React.useMemo<CrudField[]>(() => {\n return [\n {\n id: 'files',\n label: t('attachments.library.upload.file', 'Files'),\n type: 'custom',\n component: (props) => (\n <AttachmentFilesField\n {...props}\n uploading={isUploading}\n labels={{\n dropHint: t('attachments.library.upload.dropHint', 'Drag and drop files here or click to upload.'),\n choose: t('attachments.library.upload.choose', 'Choose files'),\n uploading: t('attachments.library.upload.submitting', 'Uploading\u2026'),\n empty: t('attachments.library.upload.noFiles', 'No files selected yet.'),\n }}\n />\n ),\n },\n {\n id: 'partitionCode',\n label: t('attachments.library.upload.partition', 'Partition'),\n type: 'select',\n options: [\n { value: '', label: t('attachments.library.upload.partitionDefault', 'Default (private)') },\n ...partitionOptions,\n ],\n },\n {\n id: 'tags',\n label: t('attachments.library.table.tags', 'Tags'),\n type: 'custom',\n component: ({ value, setValue, disabled }) => (\n <TagsInput\n value={Array.isArray(value) ? (value as string[]) : []}\n onChange={(next) => setValue(next)}\n suggestions={availableTags}\n placeholder={t('attachments.library.upload.tagsPlaceholder', 'Add tags')}\n disabled={Boolean(disabled) || isUploading}\n />\n ),\n },\n {\n id: 'assignments',\n label: '',\n type: 'custom',\n component: ({ value, setValue, disabled }) => (\n <AttachmentAssignmentsEditor\n value={Array.isArray(value) ? (value as AssignmentDraft[]) : []}\n onChange={(next) => setValue(next)}\n labels={assignmentLabels}\n disabled={Boolean(disabled) || isUploading}\n />\n ),\n },\n ]\n }, [assignmentLabels, availableTags, isUploading, partitionOptions, t])\n\n const groups = React.useMemo<CrudFormGroup[]>(() => {\n return [\n {\n id: 'details',\n title: t('attachments.library.upload.title', 'Upload attachment'),\n column: 1,\n fields: ['files', 'partitionCode', 'tags', 'assignments'],\n },\n {\n id: 'customFields',\n title: t('entities.customFields.title', 'Custom attributes'),\n column: 2,\n kind: 'customFields',\n },\n ]\n }, [t])\n\n const uploadPercentage = uploadProgress.total\n ? Math.min(100, Math.round((uploadProgress.completed / uploadProgress.total) * 100))\n : 0\n\n const handleSubmit = React.useCallback(\n async (values: AttachmentUploadFormValues) => {\n const files = Array.isArray(values.files) ? values.files : []\n if (!files.length) {\n throw new Error(t('attachments.library.upload.fileRequired', 'Select at least one file to upload.'))\n }\n setUploadProgress({ completed: 0, total: files.length })\n setIsUploading(true)\n try {\n const tags = Array.isArray(values.tags)\n ? values.tags\n .map((tag) => (typeof tag === 'string' ? tag.trim() : ''))\n .filter((tag) => tag.length > 0)\n : []\n const cleanedAssignments =\n Array.isArray(values.assignments) && values.assignments.length\n ? values.assignments\n .map((assignment) => ({\n type: assignment.type?.trim() ?? '',\n id: assignment.id?.trim() ?? '',\n href: assignment.href?.trim() || undefined,\n label: assignment.label?.trim() || undefined,\n }))\n .filter((assignment) => assignment.type && assignment.id)\n : []\n const customFields = collectCustomFieldValues(values, {\n transform: (value) => normalizeCustomFieldSubmitValue(value),\n })\n let completed = 0\n for (const file of files) {\n const fd = new FormData()\n fd.set('entityId', LIBRARY_ENTITY_ID)\n fd.set(\n 'recordId',\n typeof crypto !== 'undefined' && typeof crypto.randomUUID === 'function' ? crypto.randomUUID() : String(Date.now()),\n )\n fd.set('file', file)\n if (typeof values.partitionCode === 'string' && values.partitionCode.trim().length) {\n fd.set('partitionCode', values.partitionCode.trim())\n }\n if (tags.length) fd.set('tags', JSON.stringify(tags))\n if (cleanedAssignments.length) fd.set('assignments', JSON.stringify(cleanedAssignments))\n if (Object.keys(customFields).length) fd.set('customFields', JSON.stringify(customFields))\n const call = await apiCall<{ error?: string }>('/api/attachments', {\n method: 'POST',\n body: fd,\n })\n if (!call.ok) {\n const message = call.result?.error || t('attachments.library.upload.failed', 'Upload failed.')\n throw new Error(message)\n }\n completed += 1\n setUploadProgress({ completed, total: files.length })\n }\n flash(t('attachments.library.upload.success', 'Attachment uploaded.'), 'success')\n onUploaded()\n onCancel()\n } catch (err: any) {\n const message = err?.message || t('attachments.library.upload.failed', 'Upload failed.')\n flash(message, 'error')\n throw new Error(message)\n } finally {\n setIsUploading(false)\n }\n },\n [onCancel, onUploaded, t],\n )\n\n return (\n <div className=\"relative\">\n <CrudForm<AttachmentUploadFormValues>\n embedded\n schema={formSchema}\n entityId={E.attachments.attachment}\n fields={fields}\n groups={groups}\n initialValues={{ files: [], tags: [], assignments: [], partitionCode: '' }}\n submitLabel={\n isUploading\n ? t('attachments.library.upload.submitting', 'Uploading\u2026')\n : t('attachments.library.upload.submit', 'Upload')\n }\n extraActions={\n <Button type=\"button\" variant=\"outline\" onClick={onCancel} disabled={isUploading}>\n {t('attachments.library.upload.cancel', 'Cancel')}\n </Button>\n }\n onSubmit={handleSubmit}\n />\n {isUploading ? (\n <div className=\"pointer-events-none absolute inset-0 z-20 flex items-center justify-center bg-background/90 px-6 text-center backdrop-blur\">\n <div className=\"flex w-full max-w-sm flex-col items-center gap-4 rounded-xl border border-border/50 bg-card/95 px-6 py-8 shadow-2xl\">\n <Spinner size=\"lg\" className=\"border-primary/50 border-t-primary\" />\n <div className=\"w-full space-y-3\">\n <p className=\"text-base font-semibold\">\n {t('attachments.library.upload.progressLabel', 'Uploading files')}\n </p>\n {uploadProgress.total > 0 ? (\n <>\n <p className=\"text-sm text-muted-foreground\">\n {uploadProgress.completed}/{uploadProgress.total}\n </p>\n <div className=\"h-2 w-full rounded bg-muted\">\n <div\n className=\"h-2 rounded bg-primary transition-all\"\n style={{\n width: `${uploadPercentage}%`,\n }}\n />\n </div>\n </>\n ) : null}\n </div>\n </div>\n </div>\n ) : null}\n </div>\n )\n}\ntype UploadDialogProps = {\n open: boolean\n onOpenChange: (next: boolean) => void\n partitions: Array<{ code: string; title: string }>\n availableTags: string[]\n onUploaded: () => void\n}\n\nfunction AttachmentUploadDialog({ open, onOpenChange, partitions, availableTags, onUploaded }: UploadDialogProps) {\n const t = useT()\n const [formResetKey, setFormResetKey] = React.useState(0)\n const previousOpen = React.useRef(open)\n\n React.useEffect(() => {\n if (previousOpen.current && !open) {\n setFormResetKey((prev) => prev + 1)\n }\n previousOpen.current = open\n }, [open])\n\n const handleDialogChange = React.useCallback(\n (next: boolean) => {\n onOpenChange(next)\n },\n [onOpenChange],\n )\n\n const handleUploaded = React.useCallback(() => {\n onUploaded()\n }, [onUploaded])\n\n return (\n <Dialog open={open} onOpenChange={handleDialogChange}>\n <DialogContent className=\"sm:max-w-[54.6rem]\">\n <DialogHeader>\n <DialogTitle>{t('attachments.library.upload.title', 'Upload attachment')}</DialogTitle>\n </DialogHeader>\n <AttachmentUploadForm\n key={formResetKey}\n partitions={partitions}\n availableTags={availableTags}\n onUploaded={handleUploaded}\n onCancel={() => handleDialogChange(false)}\n />\n </DialogContent>\n </Dialog>\n )\n}\n\nexport function AttachmentLibrary() {\n const t = useT()\n const queryClient = useQueryClient()\n const [page, setPage] = React.useState(1)\n const [search, setSearch] = React.useState('')\n const [filterValues, setFilterValues] = React.useState<FilterValues>({})\n const [sorting, setSorting] = React.useState<SortingState>([{ id: 'createdAt', desc: true }])\n const [metadataDialogOpen, setMetadataDialogOpen] = React.useState(false)\n const [selectedRow, setSelectedRow] = React.useState<AttachmentRow | null>(null)\n const [uploadDialogOpen, setUploadDialogOpen] = React.useState(false)\n const [deleteDialogOpen, setDeleteDialogOpen] = React.useState(false)\n const [deleteTarget, setDeleteTarget] = React.useState<AttachmentRow | null>(null)\n const [deletePending, setDeletePending] = React.useState(false)\n const filterSignature = React.useMemo(() => buildFilterSignature(filterValues), [filterValues])\n const sortingSignature = React.useMemo(() => JSON.stringify(sorting), [sorting])\n\n const { data, isLoading, error, refetch } = useQuery({\n queryKey: ['attachments-library', page, search, filterSignature, sortingSignature],\n queryFn: async () => {\n const params = new URLSearchParams()\n params.set('page', String(page))\n params.set('pageSize', String(PAGE_SIZE))\n if (search.trim().length > 0) params.set('search', search.trim())\n const partition = typeof filterValues.partition === 'string' ? filterValues.partition : ''\n if (partition) params.set('partition', partition)\n const tags = Array.isArray(filterValues.tags) ? filterValues.tags : []\n if (tags.length > 0) params.set('tags', tags.join(','))\n if (sorting.length > 0) {\n const primary = sorting[0]\n params.set('sortField', primary.id)\n params.set('sortDir', primary.desc ? 'desc' : 'asc')\n }\n const call = await apiCall<AttachmentLibraryResponse>(`/api/attachments/library?${params.toString()}`)\n if (!call.ok || !call.result) {\n const message = call.result?.error || t('attachments.library.errors.load', 'Failed to load attachments.')\n throw new Error(message)\n }\n return call.result\n },\n })\n\n const partitions = data?.partitions ?? []\n const availableTags = data?.availableTags ?? []\n\n const filters = React.useMemo<FilterDef[]>(() => {\n const partitionOptions = partitions.map((entry) => ({\n value: entry.code,\n label: entry.title || entry.code,\n }))\n return [\n {\n id: 'partition',\n label: t('attachments.library.filters.partition', 'Partition'),\n type: 'select',\n options: partitionOptions,\n },\n {\n id: 'tags',\n label: t('attachments.library.filters.tags', 'Tags'),\n type: 'tags',\n placeholder: t('attachments.library.filters.tagsPlaceholder', 'Filter by tag'),\n options: availableTags.map((tag) => ({ value: tag, label: tag })),\n },\n ]\n }, [availableTags, partitions, t])\n\n const items = data?.items ?? []\n\n const columns = React.useMemo<ColumnDef<AttachmentRow>[]>(() => {\n return [\n {\n id: 'preview',\n header: '',\n enableSorting: false,\n cell: ({ row }) => {\n const value = row.original\n if (value.thumbnailUrl) {\n return (\n <div className=\"h-16 w-16 overflow-hidden rounded border bg-muted\">\n <img\n src={value.thumbnailUrl}\n alt={value.fileName}\n className=\"h-full w-full object-cover\"\n loading=\"lazy\"\n />\n </div>\n )\n }\n const placeholder = resolveAttachmentPlaceholder(value.mimeType, value.fileName)\n const PlaceholderIcon = placeholder.icon\n return (\n <div className=\"flex h-16 w-16 flex-col items-center justify-center rounded border bg-muted text-[10px] font-semibold uppercase text-muted-foreground\">\n <PlaceholderIcon className=\"mb-1 h-5 w-5 text-muted-foreground\" aria-hidden />\n {placeholder.label}\n </div>\n )\n },\n },\n {\n id: 'fileName',\n accessorKey: 'fileName',\n header: t('attachments.library.table.file', 'File'),\n cell: ({ row }) => {\n const value = row.original\n return (\n <div className=\"space-y-1 min-w-0 max-w-[280px]\">\n <div className=\"font-medium truncate\" title={value.fileName}>\n {value.fileName}\n </div>\n <div className=\"text-xs text-muted-foreground\">\n {formatFileSize(value.fileSize)} \u2022 {value.mimeType || 'application/octet-stream'}\n </div>\n <div className=\"text-xs text-muted-foreground line-clamp-2\">\n {value.content?.trim()\n ? value.content\n : t('attachments.library.metadata.noContent', 'No text extracted')}\n </div>\n </div>\n )\n },\n },\n {\n id: 'tags',\n accessorKey: 'tags',\n header: t('attachments.library.table.tags', 'Tags'),\n enableSorting: false,\n cell: ({ row }) => {\n const tags = row.original.tags ?? []\n if (!tags.length) return <span className=\"text-xs text-muted-foreground\">\u2014</span>\n return (\n <div className=\"flex flex-wrap gap-1\">\n {tags.map((tag) => (\n <Badge key={tag} variant=\"outline\">\n {tag}\n </Badge>\n ))}\n </div>\n )\n },\n },\n {\n id: 'assignments',\n accessorKey: 'assignments',\n header: t('attachments.library.table.assignments', 'Assignments'),\n enableSorting: false,\n cell: ({ row }) => {\n const assignments = filterLibraryAssignments(row.original.assignments)\n if (!assignments.length) return <span className=\"text-xs text-muted-foreground\">\u2014</span>\n return (\n <div className=\"flex flex-col gap-1\">\n {assignments.map((assignment) => {\n const label = assignment.label?.trim() || assignment.id\n const hideType =\n assignment.type === (E as any).catalog?.catalog_product ||\n assignment.type === (E as any).catalog?.catalog_product_variant\n const content = hideType ? label : `${assignment.type}: ${label}`\n return assignment.href ? (\n <a\n key={`${assignment.type}-${assignment.id}-${assignment.href}`}\n href={assignment.href}\n className=\"text-sm text-blue-600 underline\"\n target=\"_blank\"\n rel=\"noreferrer\"\n >\n {content}\n </a>\n ) : (\n <div key={`${assignment.type}-${assignment.id}`} className=\"text-sm\">\n {content}\n </div>\n )\n })}\n </div>\n )\n },\n },\n {\n id: 'partitionCode',\n accessorKey: 'partitionCode',\n header: t('attachments.library.table.partition', 'Partition'),\n cell: ({ row }) => (\n <div className=\"text-sm text-muted-foreground\">\n {row.original.partitionTitle ?? row.original.partitionCode}\n </div>\n ),\n },\n {\n id: 'createdAt',\n accessorKey: 'createdAt',\n header: t('attachments.library.table.created', 'Created'),\n cell: ({ row }) => {\n const createdAt = row.original.createdAt\n return (\n <div className=\"text-sm text-muted-foreground\">\n {createdAt ? humanDate(createdAt) : '\u2014'}\n </div>\n )\n },\n },\n {\n id: 'download',\n header: t('attachments.library.table.download', 'Download'),\n enableSorting: false,\n cell: ({ row }) => {\n const downloadPath = buildAttachmentFileUrl(row.original.id, { download: true })\n const absolute = resolveAbsoluteUrl(downloadPath)\n return (\n <Button variant=\"ghost\" size=\"icon\" asChild>\n <a href={absolute} download aria-label={t('attachments.library.table.download', 'Download')}>\n <Download className=\"h-4 w-4\" />\n </a>\n </Button>\n )\n },\n },\n ]\n }, [t])\n\n const openMetadataDialog = React.useCallback((row: AttachmentRow) => {\n setSelectedRow(row)\n setMetadataDialogOpen(true)\n }, [])\n\n const openDeleteDialog = React.useCallback((row: AttachmentRow) => {\n setDeleteTarget(row)\n setDeleteDialogOpen(true)\n }, [])\n\n const handleMetadataSave = React.useCallback(\n async (id: string, payload: AttachmentMetadataSavePayload) => {\n try {\n const body: Record<string, unknown> = {\n tags: payload.tags,\n assignments: payload.assignments,\n }\n if (payload.customFields && Object.keys(payload.customFields).length) {\n body.customFields = payload.customFields\n }\n const call = await apiCall<{ error?: string }>(`/api/attachments/library/${encodeURIComponent(id)}`, {\n method: 'PATCH',\n headers: { 'content-type': 'application/json' },\n body: JSON.stringify(body),\n })\n if (!call.ok) {\n const message =\n call.result?.error || t('attachments.library.metadata.error', 'Failed to update metadata.')\n flash(message, 'error')\n return\n }\n flash(t('attachments.library.metadata.success', 'Attachment updated.'), 'success')\n await queryClient.invalidateQueries({ queryKey: ['attachments-library'], exact: false })\n setMetadataDialogOpen(false)\n } catch (err: any) {\n flash(err?.message || t('attachments.library.metadata.error', 'Failed to update metadata.'), 'error')\n }\n },\n [queryClient, t],\n )\n\n const handleUploadCompleted = React.useCallback(async () => {\n await queryClient.invalidateQueries({ queryKey: ['attachments-library'], exact: false })\n }, [queryClient])\n\n const handleDelete = React.useCallback(async () => {\n if (!deleteTarget) return\n try {\n setDeletePending(true)\n const call = await apiCall<{ error?: string }>(\n `/api/attachments/library/${encodeURIComponent(deleteTarget.id)}`,\n { method: 'DELETE' },\n )\n if (!call.ok) {\n const message =\n call.result?.error || t('attachments.library.errors.delete', 'Failed to delete attachment.')\n flash(message, 'error')\n return\n }\n flash(t('attachments.library.messages.deleted', 'Attachment removed.'), 'success')\n if (selectedRow?.id === deleteTarget.id) {\n setSelectedRow(null)\n setMetadataDialogOpen(false)\n }\n setDeleteDialogOpen(false)\n setDeleteTarget(null)\n await queryClient.invalidateQueries({ queryKey: ['attachments-library'], exact: false })\n } catch (err: any) {\n flash(err?.message || t('attachments.library.errors.delete', 'Failed to delete attachment.'), 'error')\n } finally {\n setDeletePending(false)\n }\n }, [deleteTarget, queryClient, selectedRow, t])\n\n const total = data?.total ?? 0\n const totalPages = data?.totalPages ?? 1\n return (\n <>\n <DataTable<AttachmentRow>\n title={t('attachments.library.title', 'Attachments')}\n refreshButton={{\n label: t('attachments.library.actions.refresh', 'Refresh'),\n onRefresh: () => { void refetch() },\n isRefreshing: isLoading,\n }}\n actions={(\n <Button onClick={() => setUploadDialogOpen(true)}>\n {t('attachments.library.actions.upload', 'Upload')}\n </Button>\n )}\n columns={columns}\n data={items}\n sorting={sorting}\n onSortingChange={setSorting}\n rowActions={(row) => (\n <RowActions\n items={[\n {\n id: 'open',\n label: t('attachments.library.actions.open', 'Open'),\n onSelect: () => {\n if (!row.url) return\n window.open(row.url, '_blank', 'noopener,noreferrer')\n },\n },\n {\n id: 'edit',\n label: t('attachments.library.actions.edit', 'Edit metadata'),\n onSelect: () => openMetadataDialog(row),\n },\n {\n id: 'copy-url',\n label: t('attachments.library.actions.copyUrl', 'Copy URL'),\n onSelect: () => {\n if (!row.url) {\n flash(t('attachments.library.actions.copyError', 'Unable to copy link.'), 'error')\n return\n }\n const absolute = resolveAbsoluteUrl(row.url)\n navigator.clipboard\n .writeText(absolute)\n .then(() =>\n flash(\n t('attachments.library.actions.copied', 'Link copied.'),\n 'success',\n ),\n )\n .catch(() =>\n flash(\n t('attachments.library.actions.copyError', 'Unable to copy link.'),\n 'error',\n ),\n )\n },\n },\n {\n id: 'delete',\n label: t('attachments.library.actions.delete', 'Delete'),\n destructive: true,\n onSelect: () => openDeleteDialog(row),\n },\n ]}\n />\n )}\n onRowClick={(row) => openMetadataDialog(row)}\n isLoading={isLoading}\n error={error?.message}\n emptyState={\n <div className=\"py-10 text-center text-sm text-muted-foreground\">\n {t('attachments.library.table.empty', 'No attachments found.')}\n </div>\n }\n searchValue={search}\n onSearchChange={(value) => {\n setPage(1)\n setSearch(value)\n }}\n searchPlaceholder={t('attachments.library.table.search', 'Search files\u2026')}\n filters={filters}\n filterValues={filterValues}\n onFiltersApply={(values) => {\n setFilterValues(values)\n setPage(1)\n }}\n onFiltersClear={() => {\n setFilterValues({})\n setPage(1)\n }}\n pagination={{\n page,\n pageSize: PAGE_SIZE,\n total,\n totalPages,\n onPageChange: (next) => setPage(next),\n }}\n />\n <AttachmentMetadataDialog\n open={metadataDialogOpen}\n onOpenChange={setMetadataDialogOpen}\n item={selectedRow}\n availableTags={availableTags}\n onSave={handleMetadataSave}\n />\n <AttachmentDeleteDialog\n open={deleteDialogOpen}\n onOpenChange={setDeleteDialogOpen}\n fileName={deleteTarget?.fileName}\n onConfirm={handleDelete}\n isDeleting={deletePending}\n />\n <AttachmentUploadDialog\n open={uploadDialogOpen}\n onOpenChange={setUploadDialogOpen}\n partitions={partitions}\n availableTags={availableTags}\n onUploaded={handleUploadCompleted}\n />\n </>\n )\n}\n"],
5
- "mappings": ";AA6NM,SAmcU,UAlcR,KADF;AA3NN,YAAY,WAAW;AAEvB,SAAS,UAAU,sBAAsB;AACzC,SAAS,iBAAiB;AAC1B,SAAS,kBAAkB;AAE3B,SAAS,cAAc;AACvB,SAAS,aAAa;AACtB,SAAS,QAAQ,eAAe,cAAc,mBAAmB;AAEjE,SAAS,iBAAiB;AAC1B,SAAS,eAAe;AACxB,SAAS,gBAAqF;AAC9F,SAAS,gCAAgC;AACzC,SAAS,eAAe;AACxB,SAAS,aAAa;AACtB,SAAS,YAAY;AACrB,SAAS,SAAS;AAClB,SAAS,SAAS;AAElB,SAAS,UAAU,MAAM,QAAQ,QAAQ,MAAM,UAAU,iBAAiB,aAAa,WAAW,WAAW,gBAAgB;AAC7H,SAAS,8BAAkF;AAC3F,SAAS,UAAU;AACnB,SAAS,wBAAwB,gCAA+G;AAsBhJ,MAAM,YAAY;AAClB,MAAM,eAAe,QAAQ,IAAI,uBAAuB,IAAI,QAAQ,OAAO,EAAE;AAC7E,MAAM,oBAAoB;AAE1B,SAAS,yBAAyB,aAAqE;AACrG,UAAQ,eAAe,CAAC,GAAG,OAAO,CAAC,eAAe,WAAW,SAAS,iBAAiB;AACzF;AAEA,SAAS,eAAe,OAAuB;AAC7C,MAAI,CAAC,OAAO,SAAS,KAAK,EAAG,QAAO;AACpC,MAAI,SAAS,EAAG,QAAO;AACvB,QAAM,QAAQ,CAAC,KAAK,MAAM,MAAM,MAAM,IAAI;AAC1C,MAAI,MAAM;AACV,MAAI,UAAU;AACd,SAAO,WAAW,QAAQ,MAAM,MAAM,SAAS,GAAG;AAChD,eAAW;AACX,WAAO;AAAA,EACT;AACA,SAAO,GAAG,QAAQ,QAAQ,QAAQ,IAAI,IAAI,CAAC,CAAC,IAAI,MAAM,GAAG,CAAC;AAC5D;AAEA,SAAS,UAAU,OAAe,QAAyB;AACzD,QAAM,OAAO,IAAI,KAAK,KAAK;AAC3B,MAAI,OAAO,MAAM,KAAK,QAAQ,CAAC,EAAG,QAAO;AACzC,SAAO,KAAK,eAAe,UAAU,MAAS;AAChD;AAEA,SAAS,qBAAqB,QAA8B;AAC1D,SAAO,KAAK,UAAU,QAAQ,OAAO,KAAK,MAAM,EAAE,KAAK,CAAC,GAAG,MAAM,EAAE,cAAc,CAAC,CAAC,CAAC;AACtF;AAEA,SAAS,mBAAmB,MAAsB;AAChD,MAAI,CAAC,KAAM,QAAO;AAClB,MAAI,gBAAgB,KAAK,IAAI,EAAG,QAAO;AACvC,QAAM,OACJ,gBACC,OAAO,WAAW,eAAe,OAAO,UAAU,SAAS,OAAO,SAAS,SAAS;AACvF,MAAI,CAAC,KAAM,QAAO;AAClB,QAAM,iBAAiB,KAAK,QAAQ,OAAO,EAAE;AAC7C,SAAO,GAAG,cAAc,GAAG,KAAK,WAAW,GAAG,IAAI,OAAO,IAAI,IAAI,EAAE;AACrE;AAEA,SAAS,qBAAqB,UAAkC;AAC9D,MAAI,CAAC,SAAU,QAAO;AACtB,QAAM,aAAa,SAAS,KAAK;AACjC,MAAI,CAAC,WAAY,QAAO;AACxB,QAAM,UAAU,WAAW,YAAY,GAAG;AAC1C,MAAI,YAAY,MAAM,YAAY,WAAW,SAAS,EAAG,QAAO;AAChE,SAAO,WAAW,MAAM,UAAU,CAAC,EAAE,YAAY;AACnD;AAEA,MAAM,qBAAiD;AAAA,EACrD,KAAK;AAAA,EACL,KAAK;AAAA,EACL,MAAM;AAAA,EACN,KAAK;AAAA,EACL,IAAI;AAAA,EACJ,KAAK;AAAA,EACL,KAAK;AAAA,EACL,MAAM;AAAA,EACN,KAAK;AAAA,EACL,KAAK;AAAA,EACL,KAAK;AAAA,EACL,MAAM;AAAA,EACN,KAAK;AAAA,EACL,IAAI;AAAA,EACJ,KAAK;AAAA,EACL,KAAK;AAAA,EACL,MAAM;AAAA,EACN,KAAK;AAAA,EACL,MAAM;AAAA,EACN,IAAI;AAAA,EACJ,IAAI;AAAA,EACJ,KAAK;AAAA,EACL,KAAK;AAAA,EACL,MAAM;AAAA,EACN,KAAK;AAAA,EACL,KAAK;AAAA,EACL,MAAM;AAAA,EACN,KAAK;AAAA,EACL,KAAK;AAAA,EACL,KAAK;AAAA,EACL,MAAM;AAAA,EACN,KAAK;AAAA,EACL,KAAK;AAAA,EACL,KAAK;AAAA,EACL,KAAK;AAAA,EACL,MAAM;AACR;AAEA,MAAM,sBAAkD;AAAA,EACtD,OAAO;AAAA,EACP,OAAO;AAAA,EACP,MAAM;AAAA,EACN,aAAa;AACf;AAEA,SAAS,6BAA6B,UAA0B,UAA+D;AAC7H,QAAM,YAAY,qBAAqB,QAAQ;AAC/C,QAAM,iBAAiB,OAAO,aAAa,WAAW,SAAS,YAAY,IAAI;AAC/E,MAAI,aAAa,mBAAmB,SAAS,GAAG;AAC9C,WAAO,EAAE,MAAM,mBAAmB,SAAS,GAAG,OAAO,UAAU,YAAY,EAAE;AAAA,EAC/E;AACA,MAAI,CAAC,aAAa,eAAe,SAAS,KAAK,GAAG;AAChD,WAAO,EAAE,MAAM,UAAU,OAAO,MAAM;AAAA,EACxC;AACA,MAAI,CAAC,aAAa,eAAe,SAAS,KAAK,GAAG;AAChD,WAAO,EAAE,MAAM,aAAa,OAAO,MAAM;AAAA,EAC3C;AACA,MAAI,CAAC,aAAa,eAAe,SAAS,MAAM,GAAG;AACjD,WAAO,EAAE,MAAM,UAAU,OAAO,OAAO;AAAA,EACzC;AACA,QAAM,WAAW,eAAe,MAAM,GAAG,EAAE,CAAC,KAAK;AACjD,MAAI,YAAY,oBAAoB,QAAQ,GAAG;AAC7C,WAAO,EAAE,MAAM,oBAAoB,QAAQ,GAAG,OAAO,SAAS,YAAY,EAAE;AAAA,EAC9E;AACA,QAAM,iBAAiB,aAAa,YAAY;AAChD,QAAM,gBAAgB,eAAe,MAAM,GAAG,CAAC,EAAE,YAAY;AAC7D,SAAO,EAAE,MAAM,MAAM,OAAO,cAAc;AAC5C;AAEA,SAAS,gCAAgC,OAAyB;AAChE,MAAI,MAAM,QAAQ,KAAK,GAAG;AACxB,WAAO,MAAM,OAAO,CAAC,UAAU,UAAU,MAAS;AAAA,EACpD;AACA,MAAI,UAAU,OAAW,QAAO;AAChC,SAAO;AACT;AA0BA,SAAS,4BAA4B,EAAE,OAAO,UAAU,QAAQ,SAAS,GAA2B;AAClG,QAAM,eAAe,MAAM;AAAA,IACzB,CAAC,OAAe,UAAoC;AAClD,eAAS,MAAM,IAAI,CAAC,OAAO,QAAS,QAAQ,QAAQ,EAAE,GAAG,OAAO,GAAG,MAAM,IAAI,KAAM,CAAC;AAAA,IACtF;AAAA,IACA,CAAC,UAAU,KAAK;AAAA,EAClB;AAEA,QAAM,eAAe,MAAM;AAAA,IACzB,CAAC,UAAkB;AACjB,eAAS,MAAM,OAAO,CAAC,GAAG,QAAQ,QAAQ,KAAK,CAAC;AAAA,IAClD;AAAA,IACA,CAAC,UAAU,KAAK;AAAA,EAClB;AAEA,QAAM,YAAY,MAAM,YAAY,MAAM;AACxC,aAAS,CAAC,GAAG,OAAO,EAAE,MAAM,IAAI,IAAI,IAAI,MAAM,IAAI,OAAO,GAAG,CAAC,CAAC;AAAA,EAChE,GAAG,CAAC,UAAU,KAAK,CAAC;AAEpB,SACE,qBAAC,SAAI,WAAU,aACb;AAAA,yBAAC,SACC;AAAA,0BAAC,SAAI,WAAU,uBAAuB,iBAAO,OAAM;AAAA,MACnD,oBAAC,SAAI,WAAU,iCAAiC,iBAAO,aAAY;AAAA,OACrE;AAAA,IACA,oBAAC,SAAI,WAAU,aACZ,gBAAM,WAAW,IAChB,oBAAC,SAAI,WAAU,iCAAgC,iCAAmB,IAElE,MAAM,IAAI,CAAC,OAAO,UAChB,qBAAC,SAA+C,WAAU,gCACxD;AAAA,2BAAC,SAAI,WAAU,6BACb;AAAA,6BAAC,SAAI,WAAU,aACb;AAAA,8BAAC,WAAM,WAAU,uBAAuB,iBAAO,MAAK;AAAA,UACpD;AAAA,YAAC;AAAA;AAAA,cACC,WAAU;AAAA,cACV,OAAO,MAAM;AAAA,cACb;AAAA,cACA,UAAU,CAAC,UAAU,aAAa,OAAO,EAAE,MAAM,MAAM,OAAO,MAAM,CAAC;AAAA;AAAA,UACvE;AAAA,WACF;AAAA,QACA,qBAAC,SAAI,WAAU,aACb;AAAA,8BAAC,WAAM,WAAU,uBAAuB,iBAAO,IAAG;AAAA,UAClD;AAAA,YAAC;AAAA;AAAA,cACC,WAAU;AAAA,cACV,OAAO,MAAM;AAAA,cACb;AAAA,cACA,UAAU,CAAC,UAAU,aAAa,OAAO,EAAE,IAAI,MAAM,OAAO,MAAM,CAAC;AAAA;AAAA,UACrE;AAAA,WACF;AAAA,SACF;AAAA,MACA,qBAAC,SAAI,WAAU,6BACb;AAAA,6BAAC,SAAI,WAAU,aACb;AAAA,8BAAC,WAAM,WAAU,uBAAuB,iBAAO,MAAK;AAAA,UACpD;AAAA,YAAC;AAAA;AAAA,cACC,WAAU;AAAA,cACV,OAAO,MAAM,QAAQ;AAAA,cACrB;AAAA,cACA,UAAU,CAAC,UAAU,aAAa,OAAO,EAAE,MAAM,MAAM,OAAO,MAAM,CAAC;AAAA;AAAA,UACvE;AAAA,WACF;AAAA,QACC,OAAO,QACN,qBAAC,SAAI,WAAU,aACb;AAAA,8BAAC,WAAM,WAAU,uBAAuB,iBAAO,OAAM;AAAA,UACrD;AAAA,YAAC;AAAA;AAAA,cACC,WAAU;AAAA,cACV,OAAO,MAAM,SAAS;AAAA,cACtB;AAAA,cACA,UAAU,CAAC,UAAU,aAAa,OAAO,EAAE,OAAO,MAAM,OAAO,MAAM,CAAC;AAAA;AAAA,UACxE;AAAA,WACF,IACE;AAAA,SACN;AAAA,MACA,oBAAC,SAAI,WAAU,oBACb;AAAA,QAAC;AAAA;AAAA,UACC,MAAK;AAAA,UACL,MAAK;AAAA,UACL,SAAQ;AAAA,UACR;AAAA,UACA,SAAS,MAAM,aAAa,KAAK;AAAA,UACjC,WAAU;AAAA,UAEV;AAAA,gCAAC,UAAO,WAAU,WAAU;AAAA,YAC3B,OAAO;AAAA;AAAA;AAAA,MACV,GACF;AAAA,SAvDQ,GAAG,KAAK,IAAI,MAAM,IAAI,IAAI,MAAM,EAAE,EAwD5C,CACD,GAEL;AAAA,IACA,qBAAC,UAAO,MAAK,UAAS,SAAQ,WAAU,MAAK,MAAK,UAAoB,SAAS,WAAW,WAAU,kCAClG;AAAA,0BAAC,QAAK,WAAU,WAAU;AAAA,MACzB,OAAO;AAAA,OACV;AAAA,KACF;AAEJ;AAYA,SAAS,qBAAqB;AAAA,EAC5B;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF,GAA8B;AAC5B,QAAM,QAAQ,MAAM,QAAQ,MAAO,MAAM,QAAQ,KAAK,IAAK,QAAmB,CAAC,GAAI,CAAC,KAAK,CAAC;AAC1F,QAAM,CAAC,YAAY,WAAW,IAAI,MAAM,SAAS,KAAK;AACtD,QAAM,eAAe,MAAM,OAAgC,IAAI;AAE/D,QAAM,cAAc,MAAM;AAAA,IACxB,CAAC,SAA0B;AACzB,UAAI,CAAC,MAAM,OAAQ;AACnB,YAAM,SAAS,IAAI,IAAkB,MAAM,IAAI,CAAC,SAAS,CAAC,GAAG,KAAK,IAAI,IAAI,KAAK,IAAI,IAAI,IAAI,CAAC,CAAC;AAC7F,YAAM,KAAK,IAAI,EAAE,QAAQ,CAAC,SAAS;AACjC,eAAO,IAAI,GAAG,KAAK,IAAI,IAAI,KAAK,IAAI,IAAI,IAAI;AAAA,MAC9C,CAAC;AACD,eAAS,MAAM,KAAK,OAAO,OAAO,CAAC,CAAC;AAAA,IACtC;AAAA,IACA,CAAC,OAAO,QAAQ;AAAA,EAClB;AAEA,QAAM,aAAa,MAAM;AAAA,IACvB,CAAC,UAA2C;AAC1C,UAAI,YAAY,UAAW;AAC3B,YAAM,eAAe;AACrB,YAAM,gBAAgB;AACtB,kBAAY,KAAK;AACjB,kBAAY,MAAM,cAAc,SAAS,IAAI;AAAA,IAC/C;AAAA,IACA,CAAC,aAAa,UAAU,SAAS;AAAA,EACnC;AAEA,QAAM,iBAAiB,MAAM;AAAA,IAC3B,CAAC,UAA2C;AAC1C,UAAI,YAAY,UAAW;AAC3B,YAAM,eAAe;AACrB,YAAM,gBAAgB;AACtB,kBAAY,IAAI;AAAA,IAClB;AAAA,IACA,CAAC,UAAU,SAAS;AAAA,EACtB;AAEA,QAAM,kBAAkB,MAAM,YAAY,CAAC,UAA2C;AACpF,UAAM,eAAe;AACrB,UAAM,gBAAgB;AACtB,gBAAY,KAAK;AAAA,EACnB,GAAG,CAAC,CAAC;AAEL,QAAM,aAAa,MAAM;AAAA,IACvB,CAAC,MAAc,SAAiB;AAC9B,UAAI,YAAY,UAAW;AAC3B,eAAS,MAAM,OAAO,CAAC,SAAS,EAAE,KAAK,SAAS,QAAQ,KAAK,SAAS,KAAK,CAAC;AAAA,IAC9E;AAAA,IACA,CAAC,UAAU,OAAO,UAAU,SAAS;AAAA,EACvC;AAEA,QAAM,YAAY,MAAM,YAAY,MAAM;AACxC,QAAI,YAAY,UAAW;AAC3B,iBAAa,SAAS,MAAM;AAAA,EAC9B,GAAG,CAAC,UAAU,SAAS,CAAC;AAExB,QAAM,iBAAiB,MAAM;AAC3B,QAAI,CAAC,MAAM,QAAQ;AACjB,aAAO,oBAAC,OAAE,WAAU,iCAAiC,iBAAO,OAAM;AAAA,IACpE;AACA,WACE,oBAAC,SAAI,WAAU,aACZ,gBAAM,IAAI,CAAC,cACV,qBAAC,SAAgD,WAAU,sEACzD;AAAA,2BAAC,SACC;AAAA,4BAAC,SAAI,WAAU,eAAe,oBAAU,MAAK;AAAA,QAC7C,oBAAC,SAAI,WAAU,iCAAiC,yBAAe,UAAU,IAAI,GAAE;AAAA,SACjF;AAAA,MACA;AAAA,QAAC;AAAA;AAAA,UACC,MAAK;AAAA,UACL,SAAQ;AAAA,UACR,MAAK;AAAA,UACL,SAAS,MAAM,WAAW,UAAU,MAAM,UAAU,IAAI;AAAA,UACxD,UAAU,YAAY;AAAA,UAEtB,8BAAC,UAAO,WAAU,WAAU;AAAA;AAAA,MAC9B;AAAA,SAbQ,GAAG,UAAU,IAAI,IAAI,UAAU,IAAI,EAc7C,CACD,GACH;AAAA,EAEJ;AAEA,SACE,qBAAC,SAAI,WAAU,aACb;AAAA;AAAA,MAAC;AAAA;AAAA,QACC,WAAW;AAAA,UACT;AAAA,UACA,aAAa,gCAAgC;AAAA,UAC7C,YAAY,YAAY,eAAe;AAAA,QACzC;AAAA,QACA,QAAQ;AAAA,QACR,YAAY;AAAA,QACZ,aAAa;AAAA,QACb,MAAK;AAAA,QAEL;AAAA,8BAAC,UAAO,WAAU,yCAAwC;AAAA,UAC1D,oBAAC,OAAE,WAAU,sCAAsC,iBAAO,UAAS;AAAA,UACnE,oBAAC,UAAO,MAAK,UAAS,SAAQ,WAAU,MAAK,MAAK,WAAU,QAAO,SAAS,WAAW,UAAU,YAAY,WAC1G,sBAAY,OAAO,YAAY,OAAO,QACzC;AAAA,UACA;AAAA,YAAC;AAAA;AAAA,cACC,KAAK;AAAA,cACL,MAAK;AAAA,cACL,WAAU;AAAA,cACV,UAAQ;AAAA,cACR,UAAU,CAAC,UAAU;AACnB,4BAAY,MAAM,OAAO,KAAK;AAC9B,sBAAM,cAAc,QAAQ;AAAA,cAC9B;AAAA,cACA,UAAU,YAAY;AAAA;AAAA,UACxB;AAAA;AAAA;AAAA,IACF;AAAA,IACC,eAAe;AAAA,IACf,QAAQ,oBAAC,OAAE,WAAU,oCAAoC,iBAAM,IAAO;AAAA,KACzE;AAEJ;AASA,SAAS,qBAAqB,EAAE,YAAY,eAAe,YAAY,SAAS,GAA8B;AAC5G,QAAM,IAAI,KAAK;AACf,QAAM,CAAC,aAAa,cAAc,IAAI,MAAM,SAAS,KAAK;AAC1D,QAAM,CAAC,gBAAgB,iBAAiB,IAAI,MAAM,SAA+C,EAAE,WAAW,GAAG,OAAO,EAAE,CAAC;AAE3H,QAAM,mBAAmB,MAAM;AAAA,IAC7B,MACE,WAAW,IAAI,CAAC,WAAW;AAAA,MACzB,OAAO,MAAM;AAAA,MACb,OAAO,MAAM,SAAS,MAAM;AAAA,IAC9B,EAAE;AAAA,IACJ,CAAC,UAAU;AAAA,EACb;AAEA,QAAM,mBAAmB,MAAM;AAAA,IAC7B,OAAO;AAAA,MACL,OAAO,EAAE,gDAAgD,aAAa;AAAA,MACtE,aAAa;AAAA,QACX;AAAA,QACA;AAAA,MACF;AAAA,MACA,MAAM,EAAE,+CAA+C,MAAM;AAAA,MAC7D,IAAI,EAAE,6CAA6C,WAAW;AAAA,MAC9D,MAAM,EAAE,+CAA+C,MAAM;AAAA,MAC7D,OAAO,EAAE,gDAAgD,OAAO;AAAA,MAChE,KAAK,EAAE,8CAA8C,gBAAgB;AAAA,MACrE,QAAQ,EAAE,iDAAiD,QAAQ;AAAA,IACrE;AAAA,IACA,CAAC,CAAC;AAAA,EACJ;AAEA,QAAM,aAAa,MAAM;AAAA,IACvB,MACE,EACG,OAAO;AAAA,MACN,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,IAAI,GAAG,EAAE,SAAS,EAAE,2CAA2C,qCAAqC,EAAE,CAAC;AAAA,MAC/H,eAAe,EAAE,OAAO,EAAE,SAAS;AAAA,MACnC,MAAM,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,SAAS;AAAA,MACnC,aAAa,EACV;AAAA,QACC,EAAE,OAAO;AAAA,UACP,MAAM,EAAE,OAAO,EAAE,IAAI,CAAC;AAAA,UACtB,IAAI,EAAE,OAAO,EAAE,IAAI,CAAC;AAAA,UACpB,MAAM,EAAE,OAAO,EAAE,SAAS;AAAA,UAC1B,OAAO,EAAE,OAAO,EAAE,SAAS;AAAA,QAC7B,CAAC;AAAA,MACH,EACC,SAAS;AAAA,IACd,CAAC,EACA,YAAY;AAAA,IACjB,CAAC,CAAC;AAAA,EACJ;AAEA,QAAM,SAAS,MAAM,QAAqB,MAAM;AAC9C,WAAO;AAAA,MACL;AAAA,QACE,IAAI;AAAA,QACJ,OAAO,EAAE,mCAAmC,OAAO;AAAA,QACnD,MAAM;AAAA,QACN,WAAW,CAAC,UACV;AAAA,UAAC;AAAA;AAAA,YACE,GAAG;AAAA,YACJ,WAAW;AAAA,YACX,QAAQ;AAAA,cACN,UAAU,EAAE,uCAAuC,8CAA8C;AAAA,cACjG,QAAQ,EAAE,qCAAqC,cAAc;AAAA,cAC7D,WAAW,EAAE,yCAAyC,iBAAY;AAAA,cAClE,OAAO,EAAE,sCAAsC,wBAAwB;AAAA,YACzE;AAAA;AAAA,QACF;AAAA,MAEJ;AAAA,MACA;AAAA,QACE,IAAI;AAAA,QACJ,OAAO,EAAE,wCAAwC,WAAW;AAAA,QAC5D,MAAM;AAAA,QACN,SAAS;AAAA,UACP,EAAE,OAAO,IAAI,OAAO,EAAE,+CAA+C,mBAAmB,EAAE;AAAA,UAC1F,GAAG;AAAA,QACL;AAAA,MACF;AAAA,MACA;AAAA,QACE,IAAI;AAAA,QACJ,OAAO,EAAE,kCAAkC,MAAM;AAAA,QACjD,MAAM;AAAA,QACN,WAAW,CAAC,EAAE,OAAO,UAAU,SAAS,MACtC;AAAA,UAAC;AAAA;AAAA,YACC,OAAO,MAAM,QAAQ,KAAK,IAAK,QAAqB,CAAC;AAAA,YACrD,UAAU,CAAC,SAAS,SAAS,IAAI;AAAA,YACjC,aAAa;AAAA,YACb,aAAa,EAAE,8CAA8C,UAAU;AAAA,YACvE,UAAU,QAAQ,QAAQ,KAAK;AAAA;AAAA,QACjC;AAAA,MAEJ;AAAA,MACA;AAAA,QACE,IAAI;AAAA,QACJ,OAAO;AAAA,QACP,MAAM;AAAA,QACN,WAAW,CAAC,EAAE,OAAO,UAAU,SAAS,MACtC;AAAA,UAAC;AAAA;AAAA,YACC,OAAO,MAAM,QAAQ,KAAK,IAAK,QAA8B,CAAC;AAAA,YAC9D,UAAU,CAAC,SAAS,SAAS,IAAI;AAAA,YACjC,QAAQ;AAAA,YACR,UAAU,QAAQ,QAAQ,KAAK;AAAA;AAAA,QACjC;AAAA,MAEJ;AAAA,IACF;AAAA,EACF,GAAG,CAAC,kBAAkB,eAAe,aAAa,kBAAkB,CAAC,CAAC;AAEtE,QAAM,SAAS,MAAM,QAAyB,MAAM;AAClD,WAAO;AAAA,MACL;AAAA,QACE,IAAI;AAAA,QACJ,OAAO,EAAE,oCAAoC,mBAAmB;AAAA,QAChE,QAAQ;AAAA,QACR,QAAQ,CAAC,SAAS,iBAAiB,QAAQ,aAAa;AAAA,MAC1D;AAAA,MACA;AAAA,QACE,IAAI;AAAA,QACJ,OAAO,EAAE,+BAA+B,mBAAmB;AAAA,QAC3D,QAAQ;AAAA,QACR,MAAM;AAAA,MACR;AAAA,IACF;AAAA,EACF,GAAG,CAAC,CAAC,CAAC;AAEN,QAAM,mBAAmB,eAAe,QACpC,KAAK,IAAI,KAAK,KAAK,MAAO,eAAe,YAAY,eAAe,QAAS,GAAG,CAAC,IACjF;AAEJ,QAAM,eAAe,MAAM;AAAA,IACzB,OAAO,WAAuC;AAC5C,YAAM,QAAQ,MAAM,QAAQ,OAAO,KAAK,IAAI,OAAO,QAAQ,CAAC;AAC5D,UAAI,CAAC,MAAM,QAAQ;AACjB,cAAM,IAAI,MAAM,EAAE,2CAA2C,qCAAqC,CAAC;AAAA,MACrG;AACA,wBAAkB,EAAE,WAAW,GAAG,OAAO,MAAM,OAAO,CAAC;AACvD,qBAAe,IAAI;AACnB,UAAI;AACF,cAAM,OAAO,MAAM,QAAQ,OAAO,IAAI,IAClC,OAAO,KACJ,IAAI,CAAC,QAAS,OAAO,QAAQ,WAAW,IAAI,KAAK,IAAI,EAAG,EACxD,OAAO,CAAC,QAAQ,IAAI,SAAS,CAAC,IACjC,CAAC;AACL,cAAM,qBACJ,MAAM,QAAQ,OAAO,WAAW,KAAK,OAAO,YAAY,SACpD,OAAO,YACJ,IAAI,CAAC,gBAAgB;AAAA,UACpB,MAAM,WAAW,MAAM,KAAK,KAAK;AAAA,UACjC,IAAI,WAAW,IAAI,KAAK,KAAK;AAAA,UAC7B,MAAM,WAAW,MAAM,KAAK,KAAK;AAAA,UACjC,OAAO,WAAW,OAAO,KAAK,KAAK;AAAA,QACrC,EAAE,EACD,OAAO,CAAC,eAAe,WAAW,QAAQ,WAAW,EAAE,IAC1D,CAAC;AACP,cAAM,eAAe,yBAAyB,QAAQ;AAAA,UACpD,WAAW,CAAC,UAAU,gCAAgC,KAAK;AAAA,QAC7D,CAAC;AACD,YAAI,YAAY;AAChB,mBAAW,QAAQ,OAAO;AACxB,gBAAM,KAAK,IAAI,SAAS;AACxB,aAAG,IAAI,YAAY,iBAAiB;AACpC,aAAG;AAAA,YACD;AAAA,YACA,OAAO,WAAW,eAAe,OAAO,OAAO,eAAe,aAAa,OAAO,WAAW,IAAI,OAAO,KAAK,IAAI,CAAC;AAAA,UACpH;AACA,aAAG,IAAI,QAAQ,IAAI;AACnB,cAAI,OAAO,OAAO,kBAAkB,YAAY,OAAO,cAAc,KAAK,EAAE,QAAQ;AAClF,eAAG,IAAI,iBAAiB,OAAO,cAAc,KAAK,CAAC;AAAA,UACrD;AACA,cAAI,KAAK,OAAQ,IAAG,IAAI,QAAQ,KAAK,UAAU,IAAI,CAAC;AACpD,cAAI,mBAAmB,OAAQ,IAAG,IAAI,eAAe,KAAK,UAAU,kBAAkB,CAAC;AACvF,cAAI,OAAO,KAAK,YAAY,EAAE,OAAQ,IAAG,IAAI,gBAAgB,KAAK,UAAU,YAAY,CAAC;AACzF,gBAAM,OAAO,MAAM,QAA4B,oBAAoB;AAAA,YACjE,QAAQ;AAAA,YACR,MAAM;AAAA,UACR,CAAC;AACD,cAAI,CAAC,KAAK,IAAI;AACZ,kBAAM,UAAU,KAAK,QAAQ,SAAS,EAAE,qCAAqC,gBAAgB;AAC7F,kBAAM,IAAI,MAAM,OAAO;AAAA,UACzB;AACA,uBAAa;AACb,4BAAkB,EAAE,WAAW,OAAO,MAAM,OAAO,CAAC;AAAA,QACtD;AACA,cAAM,EAAE,sCAAsC,sBAAsB,GAAG,SAAS;AAChF,mBAAW;AACX,iBAAS;AAAA,MACX,SAAS,KAAU;AACjB,cAAM,UAAU,KAAK,WAAW,EAAE,qCAAqC,gBAAgB;AACvF,cAAM,SAAS,OAAO;AACtB,cAAM,IAAI,MAAM,OAAO;AAAA,MACzB,UAAE;AACA,uBAAe,KAAK;AAAA,MACtB;AAAA,IACF;AAAA,IACA,CAAC,UAAU,YAAY,CAAC;AAAA,EAC1B;AAEA,SACE,qBAAC,SAAI,WAAU,YACb;AAAA;AAAA,MAAC;AAAA;AAAA,QACC,UAAQ;AAAA,QACR,QAAQ;AAAA,QACR,UAAU,EAAE,YAAY;AAAA,QACxB;AAAA,QACA;AAAA,QACA,eAAe,EAAE,OAAO,CAAC,GAAG,MAAM,CAAC,GAAG,aAAa,CAAC,GAAG,eAAe,GAAG;AAAA,QACzE,aACE,cACI,EAAE,yCAAyC,iBAAY,IACvD,EAAE,qCAAqC,QAAQ;AAAA,QAErD,cACE,oBAAC,UAAO,MAAK,UAAS,SAAQ,WAAU,SAAS,UAAU,UAAU,aAClE,YAAE,qCAAqC,QAAQ,GAClD;AAAA,QAEF,UAAU;AAAA;AAAA,IACZ;AAAA,IACC,cACC,oBAAC,SAAI,WAAU,8HACb,+BAAC,SAAI,WAAU,uHACb;AAAA,0BAAC,WAAQ,MAAK,MAAK,WAAU,sCAAqC;AAAA,MAClE,qBAAC,SAAI,WAAU,oBACb;AAAA,4BAAC,OAAE,WAAU,2BACV,YAAE,4CAA4C,iBAAiB,GAClE;AAAA,QACC,eAAe,QAAQ,IACtB,iCACE;AAAA,+BAAC,OAAE,WAAU,iCACV;AAAA,2BAAe;AAAA,YAAU;AAAA,YAAE,eAAe;AAAA,aAC7C;AAAA,UACA,oBAAC,SAAI,WAAU,+BACb;AAAA,YAAC;AAAA;AAAA,cACC,WAAU;AAAA,cACV,OAAO;AAAA,gBACL,OAAO,GAAG,gBAAgB;AAAA,cAC5B;AAAA;AAAA,UACF,GACF;AAAA,WACF,IACE;AAAA,SACN;AAAA,OACF,GACF,IACE;AAAA,KACN;AAEJ;AASA,SAAS,uBAAuB,EAAE,MAAM,cAAc,YAAY,eAAe,WAAW,GAAsB;AAChH,QAAM,IAAI,KAAK;AACf,QAAM,CAAC,cAAc,eAAe,IAAI,MAAM,SAAS,CAAC;AACxD,QAAM,eAAe,MAAM,OAAO,IAAI;AAEtC,QAAM,UAAU,MAAM;AACpB,QAAI,aAAa,WAAW,CAAC,MAAM;AACjC,sBAAgB,CAAC,SAAS,OAAO,CAAC;AAAA,IACpC;AACA,iBAAa,UAAU;AAAA,EACzB,GAAG,CAAC,IAAI,CAAC;AAET,QAAM,qBAAqB,MAAM;AAAA,IAC/B,CAAC,SAAkB;AACjB,mBAAa,IAAI;AAAA,IACnB;AAAA,IACA,CAAC,YAAY;AAAA,EACf;AAEA,QAAM,iBAAiB,MAAM,YAAY,MAAM;AAC7C,eAAW;AAAA,EACb,GAAG,CAAC,UAAU,CAAC;AAEf,SACE,oBAAC,UAAO,MAAY,cAAc,oBAChC,+BAAC,iBAAc,WAAU,sBACvB;AAAA,wBAAC,gBACC,8BAAC,eAAa,YAAE,oCAAoC,mBAAmB,GAAE,GAC3E;AAAA,IACA;AAAA,MAAC;AAAA;AAAA,QAEC;AAAA,QACA;AAAA,QACA,YAAY;AAAA,QACZ,UAAU,MAAM,mBAAmB,KAAK;AAAA;AAAA,MAJnC;AAAA,IAKP;AAAA,KACF,GACF;AAEJ;AAEO,SAAS,oBAAoB;AAClC,QAAM,IAAI,KAAK;AACf,QAAM,cAAc,eAAe;AACnC,QAAM,CAAC,MAAM,OAAO,IAAI,MAAM,SAAS,CAAC;AACxC,QAAM,CAAC,QAAQ,SAAS,IAAI,MAAM,SAAS,EAAE;AAC7C,QAAM,CAAC,cAAc,eAAe,IAAI,MAAM,SAAuB,CAAC,CAAC;AACvE,QAAM,CAAC,SAAS,UAAU,IAAI,MAAM,SAAuB,CAAC,EAAE,IAAI,aAAa,MAAM,KAAK,CAAC,CAAC;AAC5F,QAAM,CAAC,oBAAoB,qBAAqB,IAAI,MAAM,SAAS,KAAK;AACxE,QAAM,CAAC,aAAa,cAAc,IAAI,MAAM,SAA+B,IAAI;AAC/E,QAAM,CAAC,kBAAkB,mBAAmB,IAAI,MAAM,SAAS,KAAK;AACpE,QAAM,CAAC,kBAAkB,mBAAmB,IAAI,MAAM,SAAS,KAAK;AACpE,QAAM,CAAC,cAAc,eAAe,IAAI,MAAM,SAA+B,IAAI;AACjF,QAAM,CAAC,eAAe,gBAAgB,IAAI,MAAM,SAAS,KAAK;AAC9D,QAAM,kBAAkB,MAAM,QAAQ,MAAM,qBAAqB,YAAY,GAAG,CAAC,YAAY,CAAC;AAC9F,QAAM,mBAAmB,MAAM,QAAQ,MAAM,KAAK,UAAU,OAAO,GAAG,CAAC,OAAO,CAAC;AAE/E,QAAM,EAAE,MAAM,WAAW,OAAO,QAAQ,IAAI,SAAS;AAAA,IACnD,UAAU,CAAC,uBAAuB,MAAM,QAAQ,iBAAiB,gBAAgB;AAAA,IACjF,SAAS,YAAY;AACnB,YAAM,SAAS,IAAI,gBAAgB;AACnC,aAAO,IAAI,QAAQ,OAAO,IAAI,CAAC;AAC/B,aAAO,IAAI,YAAY,OAAO,SAAS,CAAC;AACxC,UAAI,OAAO,KAAK,EAAE,SAAS,EAAG,QAAO,IAAI,UAAU,OAAO,KAAK,CAAC;AAChE,YAAM,YAAY,OAAO,aAAa,cAAc,WAAW,aAAa,YAAY;AACxF,UAAI,UAAW,QAAO,IAAI,aAAa,SAAS;AAChD,YAAM,OAAO,MAAM,QAAQ,aAAa,IAAI,IAAI,aAAa,OAAO,CAAC;AACrE,UAAI,KAAK,SAAS,EAAG,QAAO,IAAI,QAAQ,KAAK,KAAK,GAAG,CAAC;AACtD,UAAI,QAAQ,SAAS,GAAG;AACtB,cAAM,UAAU,QAAQ,CAAC;AACzB,eAAO,IAAI,aAAa,QAAQ,EAAE;AAClC,eAAO,IAAI,WAAW,QAAQ,OAAO,SAAS,KAAK;AAAA,MACrD;AACA,YAAM,OAAO,MAAM,QAAmC,4BAA4B,OAAO,SAAS,CAAC,EAAE;AACrG,UAAI,CAAC,KAAK,MAAM,CAAC,KAAK,QAAQ;AAC5B,cAAM,UAAU,KAAK,QAAQ,SAAS,EAAE,mCAAmC,6BAA6B;AACxG,cAAM,IAAI,MAAM,OAAO;AAAA,MACzB;AACA,aAAO,KAAK;AAAA,IACd;AAAA,EACF,CAAC;AAED,QAAM,aAAa,MAAM,cAAc,CAAC;AACxC,QAAM,gBAAgB,MAAM,iBAAiB,CAAC;AAE9C,QAAM,UAAU,MAAM,QAAqB,MAAM;AAC/C,UAAM,mBAAmB,WAAW,IAAI,CAAC,WAAW;AAAA,MAClD,OAAO,MAAM;AAAA,MACb,OAAO,MAAM,SAAS,MAAM;AAAA,IAC9B,EAAE;AACF,WAAO;AAAA,MACL;AAAA,QACE,IAAI;AAAA,QACJ,OAAO,EAAE,yCAAyC,WAAW;AAAA,QAC7D,MAAM;AAAA,QACN,SAAS;AAAA,MACX;AAAA,MACA;AAAA,QACE,IAAI;AAAA,QACJ,OAAO,EAAE,oCAAoC,MAAM;AAAA,QACnD,MAAM;AAAA,QACN,aAAa,EAAE,+CAA+C,eAAe;AAAA,QAC7E,SAAS,cAAc,IAAI,CAAC,SAAS,EAAE,OAAO,KAAK,OAAO,IAAI,EAAE;AAAA,MAClE;AAAA,IACF;AAAA,EACF,GAAG,CAAC,eAAe,YAAY,CAAC,CAAC;AAEjC,QAAM,QAAQ,MAAM,SAAS,CAAC;AAE9B,QAAM,UAAU,MAAM,QAAoC,MAAM;AAC9D,WAAO;AAAA,MACL;AAAA,QACE,IAAI;AAAA,QACJ,QAAQ;AAAA,QACR,eAAe;AAAA,QACf,MAAM,CAAC,EAAE,IAAI,MAAM;AACjB,gBAAM,QAAQ,IAAI;AAClB,cAAI,MAAM,cAAc;AACtB,mBACE,oBAAC,SAAI,WAAU,qDACb;AAAA,cAAC;AAAA;AAAA,gBACC,KAAK,MAAM;AAAA,gBACX,KAAK,MAAM;AAAA,gBACX,WAAU;AAAA,gBACV,SAAQ;AAAA;AAAA,YACV,GACF;AAAA,UAEJ;AACA,gBAAM,cAAc,6BAA6B,MAAM,UAAU,MAAM,QAAQ;AAC/E,gBAAM,kBAAkB,YAAY;AACpC,iBACE,qBAAC,SAAI,WAAU,yIACb;AAAA,gCAAC,mBAAgB,WAAU,sCAAqC,eAAW,MAAC;AAAA,YAC3E,YAAY;AAAA,aACf;AAAA,QAEJ;AAAA,MACF;AAAA,MACA;AAAA,QACE,IAAI;AAAA,QACJ,aAAa;AAAA,QACb,QAAQ,EAAE,kCAAkC,MAAM;AAAA,QAClD,MAAM,CAAC,EAAE,IAAI,MAAM;AACjB,gBAAM,QAAQ,IAAI;AAClB,iBACE,qBAAC,SAAI,WAAU,mCACb;AAAA,gCAAC,SAAI,WAAU,wBAAuB,OAAO,MAAM,UAChD,gBAAM,UACT;AAAA,YACA,qBAAC,SAAI,WAAU,iCACZ;AAAA,6BAAe,MAAM,QAAQ;AAAA,cAAE;AAAA,cAAI,MAAM,YAAY;AAAA,eACxD;AAAA,YACA,oBAAC,SAAI,WAAU,8CACZ,gBAAM,SAAS,KAAK,IACjB,MAAM,UACN,EAAE,0CAA0C,mBAAmB,GACrE;AAAA,aACF;AAAA,QAEJ;AAAA,MACF;AAAA,MACA;AAAA,QACE,IAAI;AAAA,QACJ,aAAa;AAAA,QACb,QAAQ,EAAE,kCAAkC,MAAM;AAAA,QAClD,eAAe;AAAA,QACf,MAAM,CAAC,EAAE,IAAI,MAAM;AACjB,gBAAM,OAAO,IAAI,SAAS,QAAQ,CAAC;AACnC,cAAI,CAAC,KAAK,OAAQ,QAAO,oBAAC,UAAK,WAAU,iCAAgC,oBAAC;AAC1E,iBACE,oBAAC,SAAI,WAAU,wBACZ,eAAK,IAAI,CAAC,QACT,oBAAC,SAAgB,SAAQ,WACtB,iBADS,GAEZ,CACD,GACH;AAAA,QAEJ;AAAA,MACF;AAAA,MACA;AAAA,QACE,IAAI;AAAA,QACJ,aAAa;AAAA,QACb,QAAQ,EAAE,yCAAyC,aAAa;AAAA,QAChE,eAAe;AAAA,QACf,MAAM,CAAC,EAAE,IAAI,MAAM;AACjB,gBAAM,cAAc,yBAAyB,IAAI,SAAS,WAAW;AACrE,cAAI,CAAC,YAAY,OAAQ,QAAO,oBAAC,UAAK,WAAU,iCAAgC,oBAAC;AACjF,iBACE,oBAAC,SAAI,WAAU,uBACZ,sBAAY,IAAI,CAAC,eAAe;AAC/B,kBAAM,QAAQ,WAAW,OAAO,KAAK,KAAK,WAAW;AACrD,kBAAM,WACJ,WAAW,SAAU,EAAU,SAAS,mBACxC,WAAW,SAAU,EAAU,SAAS;AAC1C,kBAAM,UAAU,WAAW,QAAQ,GAAG,WAAW,IAAI,KAAK,KAAK;AAC/D,mBAAO,WAAW,OAChB;AAAA,cAAC;AAAA;AAAA,gBAEC,MAAM,WAAW;AAAA,gBACjB,WAAU;AAAA,gBACV,QAAO;AAAA,gBACP,KAAI;AAAA,gBAEH;AAAA;AAAA,cANI,GAAG,WAAW,IAAI,IAAI,WAAW,EAAE,IAAI,WAAW,IAAI;AAAA,YAO7D,IAEA,oBAAC,SAAgD,WAAU,WACxD,qBADO,GAAG,WAAW,IAAI,IAAI,WAAW,EAAE,EAE7C;AAAA,UAEJ,CAAC,GACH;AAAA,QAEJ;AAAA,MACF;AAAA,MACA;AAAA,QACE,IAAI;AAAA,QACJ,aAAa;AAAA,QACb,QAAQ,EAAE,uCAAuC,WAAW;AAAA,QAC5D,MAAM,CAAC,EAAE,IAAI,MACX,oBAAC,SAAI,WAAU,iCACZ,cAAI,SAAS,kBAAkB,IAAI,SAAS,eAC/C;AAAA,MAEJ;AAAA,MACA;AAAA,QACE,IAAI;AAAA,QACJ,aAAa;AAAA,QACb,QAAQ,EAAE,qCAAqC,SAAS;AAAA,QACxD,MAAM,CAAC,EAAE,IAAI,MAAM;AACjB,gBAAM,YAAY,IAAI,SAAS;AAC/B,iBACE,oBAAC,SAAI,WAAU,iCACZ,sBAAY,UAAU,SAAS,IAAI,UACtC;AAAA,QAEJ;AAAA,MACF;AAAA,MACA;AAAA,QACE,IAAI;AAAA,QACJ,QAAQ,EAAE,sCAAsC,UAAU;AAAA,QAC1D,eAAe;AAAA,QACf,MAAM,CAAC,EAAE,IAAI,MAAM;AACjB,gBAAM,eAAe,uBAAuB,IAAI,SAAS,IAAI,EAAE,UAAU,KAAK,CAAC;AAC/E,gBAAM,WAAW,mBAAmB,YAAY;AAChD,iBACE,oBAAC,UAAO,SAAQ,SAAQ,MAAK,QAAO,SAAO,MACzC,8BAAC,OAAE,MAAM,UAAU,UAAQ,MAAC,cAAY,EAAE,sCAAsC,UAAU,GACxF,8BAAC,YAAS,WAAU,WAAU,GAChC,GACF;AAAA,QAEJ;AAAA,MACF;AAAA,IACF;AAAA,EACF,GAAG,CAAC,CAAC,CAAC;AAEN,QAAM,qBAAqB,MAAM,YAAY,CAAC,QAAuB;AACnE,mBAAe,GAAG;AAClB,0BAAsB,IAAI;AAAA,EAC5B,GAAG,CAAC,CAAC;AAEL,QAAM,mBAAmB,MAAM,YAAY,CAAC,QAAuB;AACjE,oBAAgB,GAAG;AACnB,wBAAoB,IAAI;AAAA,EAC1B,GAAG,CAAC,CAAC;AAEL,QAAM,qBAAqB,MAAM;AAAA,IAC/B,OAAO,IAAY,YAA2C;AAC5D,UAAI;AACF,cAAM,OAAgC;AAAA,UACpC,MAAM,QAAQ;AAAA,UACd,aAAa,QAAQ;AAAA,QACvB;AACA,YAAI,QAAQ,gBAAgB,OAAO,KAAK,QAAQ,YAAY,EAAE,QAAQ;AACpE,eAAK,eAAe,QAAQ;AAAA,QAC9B;AACA,cAAM,OAAO,MAAM,QAA4B,4BAA4B,mBAAmB,EAAE,CAAC,IAAI;AAAA,UACnG,QAAQ;AAAA,UACR,SAAS,EAAE,gBAAgB,mBAAmB;AAAA,UAC9C,MAAM,KAAK,UAAU,IAAI;AAAA,QAC3B,CAAC;AACD,YAAI,CAAC,KAAK,IAAI;AACZ,gBAAM,UACJ,KAAK,QAAQ,SAAS,EAAE,sCAAsC,4BAA4B;AAC5F,gBAAM,SAAS,OAAO;AACtB;AAAA,QACF;AACA,cAAM,EAAE,wCAAwC,qBAAqB,GAAG,SAAS;AACjF,cAAM,YAAY,kBAAkB,EAAE,UAAU,CAAC,qBAAqB,GAAG,OAAO,MAAM,CAAC;AACvF,8BAAsB,KAAK;AAAA,MAC7B,SAAS,KAAU;AACjB,cAAM,KAAK,WAAW,EAAE,sCAAsC,4BAA4B,GAAG,OAAO;AAAA,MACtG;AAAA,IACF;AAAA,IACA,CAAC,aAAa,CAAC;AAAA,EACjB;AAEA,QAAM,wBAAwB,MAAM,YAAY,YAAY;AAC1D,UAAM,YAAY,kBAAkB,EAAE,UAAU,CAAC,qBAAqB,GAAG,OAAO,MAAM,CAAC;AAAA,EACzF,GAAG,CAAC,WAAW,CAAC;AAEhB,QAAM,eAAe,MAAM,YAAY,YAAY;AACjD,QAAI,CAAC,aAAc;AACnB,QAAI;AACF,uBAAiB,IAAI;AACrB,YAAM,OAAO,MAAM;AAAA,QACjB,4BAA4B,mBAAmB,aAAa,EAAE,CAAC;AAAA,QAC/D,EAAE,QAAQ,SAAS;AAAA,MACrB;AACA,UAAI,CAAC,KAAK,IAAI;AACZ,cAAM,UACJ,KAAK,QAAQ,SAAS,EAAE,qCAAqC,8BAA8B;AAC7F,cAAM,SAAS,OAAO;AACtB;AAAA,MACF;AACA,YAAM,EAAE,wCAAwC,qBAAqB,GAAG,SAAS;AACjF,UAAI,aAAa,OAAO,aAAa,IAAI;AACvC,uBAAe,IAAI;AACnB,8BAAsB,KAAK;AAAA,MAC7B;AACA,0BAAoB,KAAK;AACzB,sBAAgB,IAAI;AACpB,YAAM,YAAY,kBAAkB,EAAE,UAAU,CAAC,qBAAqB,GAAG,OAAO,MAAM,CAAC;AAAA,IACzF,SAAS,KAAU;AACjB,YAAM,KAAK,WAAW,EAAE,qCAAqC,8BAA8B,GAAG,OAAO;AAAA,IACvG,UAAE;AACA,uBAAiB,KAAK;AAAA,IACxB;AAAA,EACF,GAAG,CAAC,cAAc,aAAa,aAAa,CAAC,CAAC;AAE9C,QAAM,QAAQ,MAAM,SAAS;AAC7B,QAAM,aAAa,MAAM,cAAc;AACvC,SACE,iCACE;AAAA;AAAA,MAAC;AAAA;AAAA,QACC,OAAO,EAAE,6BAA6B,aAAa;AAAA,QACnD,eAAe;AAAA,UACb,OAAO,EAAE,uCAAuC,SAAS;AAAA,UACzD,WAAW,MAAM;AAAE,iBAAK,QAAQ;AAAA,UAAE;AAAA,UAClC,cAAc;AAAA,QAChB;AAAA,QACA,SACE,oBAAC,UAAO,SAAS,MAAM,oBAAoB,IAAI,GAC5C,YAAE,sCAAsC,QAAQ,GACnD;AAAA,QAEF;AAAA,QACA,MAAM;AAAA,QACN;AAAA,QACA,iBAAiB;AAAA,QACjB,YAAY,CAAC,QACX;AAAA,UAAC;AAAA;AAAA,YACC,OAAO;AAAA,cACL;AAAA,gBACE,IAAI;AAAA,gBACJ,OAAO,EAAE,oCAAoC,MAAM;AAAA,gBACnD,UAAU,MAAM;AACd,sBAAI,CAAC,IAAI,IAAK;AACd,yBAAO,KAAK,IAAI,KAAK,UAAU,qBAAqB;AAAA,gBACtD;AAAA,cACF;AAAA,cACA;AAAA,gBACE,IAAI;AAAA,gBACJ,OAAO,EAAE,oCAAoC,eAAe;AAAA,gBAC5D,UAAU,MAAM,mBAAmB,GAAG;AAAA,cACxC;AAAA,cACA;AAAA,gBACE,IAAI;AAAA,gBACJ,OAAO,EAAE,uCAAuC,UAAU;AAAA,gBAC1D,UAAU,MAAM;AACd,sBAAI,CAAC,IAAI,KAAK;AACZ,0BAAM,EAAE,yCAAyC,sBAAsB,GAAG,OAAO;AACjF;AAAA,kBACF;AACA,wBAAM,WAAW,mBAAmB,IAAI,GAAG;AAC3C,4BAAU,UACP,UAAU,QAAQ,EAClB;AAAA,oBAAK,MACJ;AAAA,sBACE,EAAE,sCAAsC,cAAc;AAAA,sBACtD;AAAA,oBACF;AAAA,kBACF,EACC;AAAA,oBAAM,MACL;AAAA,sBACE,EAAE,yCAAyC,sBAAsB;AAAA,sBACjE;AAAA,oBACF;AAAA,kBACF;AAAA,gBACJ;AAAA,cACF;AAAA,cACA;AAAA,gBACE,IAAI;AAAA,gBACJ,OAAO,EAAE,sCAAsC,QAAQ;AAAA,gBACvD,aAAa;AAAA,gBACb,UAAU,MAAM,iBAAiB,GAAG;AAAA,cACtC;AAAA,YACF;AAAA;AAAA,QACF;AAAA,QAEF,YAAY,CAAC,QAAQ,mBAAmB,GAAG;AAAA,QAC3C;AAAA,QACA,OAAO,OAAO;AAAA,QACd,YACE,oBAAC,SAAI,WAAU,mDACZ,YAAE,mCAAmC,uBAAuB,GAC/D;AAAA,QAEF,aAAa;AAAA,QACb,gBAAgB,CAAC,UAAU;AACzB,kBAAQ,CAAC;AACT,oBAAU,KAAK;AAAA,QACjB;AAAA,QACA,mBAAmB,EAAE,oCAAoC,oBAAe;AAAA,QACxE;AAAA,QACA;AAAA,QACA,gBAAgB,CAAC,WAAW;AAC1B,0BAAgB,MAAM;AACtB,kBAAQ,CAAC;AAAA,QACX;AAAA,QACA,gBAAgB,MAAM;AACpB,0BAAgB,CAAC,CAAC;AAClB,kBAAQ,CAAC;AAAA,QACX;AAAA,QACA,YAAY;AAAA,UACV;AAAA,UACA,UAAU;AAAA,UACV;AAAA,UACA;AAAA,UACA,cAAc,CAAC,SAAS,QAAQ,IAAI;AAAA,QACtC;AAAA;AAAA,IACF;AAAA,IACA;AAAA,MAAC;AAAA;AAAA,QACC,MAAM;AAAA,QACN,cAAc;AAAA,QACd,MAAM;AAAA,QACN;AAAA,QACA,QAAQ;AAAA;AAAA,IACV;AAAA,IACA;AAAA,MAAC;AAAA;AAAA,QACC,MAAM;AAAA,QACN,cAAc;AAAA,QACd,UAAU,cAAc;AAAA,QACxB,WAAW;AAAA,QACX,YAAY;AAAA;AAAA,IACd;AAAA,IACA;AAAA,MAAC;AAAA;AAAA,QACC,MAAM;AAAA,QACN,cAAc;AAAA,QACd;AAAA,QACA;AAAA,QACA,YAAY;AAAA;AAAA,IACd;AAAA,KACF;AAEJ;",
4
+ "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport type { ColumnDef, SortingState } from '@tanstack/react-table'\nimport { useQuery, useQueryClient } from '@tanstack/react-query'\nimport { DataTable } from '@open-mercato/ui/backend/DataTable'\nimport { RowActions } from '@open-mercato/ui/backend/RowActions'\nimport type { FilterDef, FilterValues } from '@open-mercato/ui/backend/FilterBar'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { Badge } from '@open-mercato/ui/primitives/badge'\nimport { Dialog, DialogContent, DialogHeader, DialogTitle } from '@open-mercato/ui/primitives/dialog'\nimport { Input } from '@open-mercato/ui/primitives/input'\nimport { TagsInput } from '@open-mercato/ui/backend/inputs/TagsInput'\nimport { Spinner } from '@open-mercato/ui/primitives/spinner'\nimport { CrudForm, type CrudField, type CrudFormGroup, type CrudCustomFieldRenderProps } from '@open-mercato/ui/backend/CrudForm'\nimport { collectCustomFieldValues } from '@open-mercato/ui/backend/utils/customFieldValues'\nimport { apiCall } from '@open-mercato/ui/backend/utils/apiCall'\nimport { flash } from '@open-mercato/ui/backend/FlashMessages'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { z } from 'zod'\nimport { E } from '#generated/entities.ids.generated'\nimport type { LucideIcon } from 'lucide-react'\nimport { Download, Plus, Upload, Trash2, File, FileText, FileSpreadsheet, FileArchive, FileAudio, FileVideo, FileCode } from 'lucide-react'\nimport { buildAttachmentFileUrl, buildAttachmentImageUrl, slugifyAttachmentFileName } from '@open-mercato/core/modules/attachments/lib/imageUrls'\nimport { cn } from '@open-mercato/shared/lib/utils'\nimport { AttachmentDeleteDialog, AttachmentMetadataDialog, type AttachmentItem, type AttachmentMetadataSavePayload, type AssignmentDraft } from '@open-mercato/ui/backend/detail'\n\ntype AttachmentAssignment = {\n type: string\n id: string\n href?: string | null\n label?: string | null\n}\n\ntype AttachmentRow = AttachmentItem\n\ntype AttachmentLibraryResponse = {\n items: AttachmentRow[]\n page: number\n pageSize: number\n total: number\n totalPages: number\n availableTags: string[]\n partitions: Array<{ code: string; title: string; description?: string | null; isPublic?: boolean }>\n error?: string\n}\n\nconst PAGE_SIZE = 25\nconst ENV_APP_URL = (process.env.NEXT_PUBLIC_APP_URL || '').replace(/\\/$/, '')\nconst LIBRARY_ENTITY_ID = 'attachments:library'\n\nfunction filterLibraryAssignments(assignments?: AttachmentAssignment[] | null): AttachmentAssignment[] {\n return (assignments ?? []).filter((assignment) => assignment.type !== LIBRARY_ENTITY_ID)\n}\n\nfunction formatFileSize(value: number): string {\n if (!Number.isFinite(value)) return '\u2014'\n if (value <= 0) return '0 B'\n const units = ['B', 'KB', 'MB', 'GB', 'TB']\n let idx = 0\n let current = value\n while (current >= 1024 && idx < units.length - 1) {\n current /= 1024\n idx += 1\n }\n return `${current.toFixed(idx === 0 ? 0 : 1)} ${units[idx]}`\n}\n\nfunction humanDate(value: string, locale?: string): string {\n const date = new Date(value)\n if (Number.isNaN(date.getTime())) return value\n return date.toLocaleString(locale ?? undefined)\n}\n\nfunction buildFilterSignature(values: FilterValues): string {\n return JSON.stringify(values, Object.keys(values).sort())\n}\n\nfunction resolveAbsoluteUrl(path: string): string {\n if (!path) return path\n if (/^https?:\\/\\//i.test(path)) return path\n const base =\n ENV_APP_URL ||\n (typeof window !== 'undefined' && window.location?.origin ? window.location.origin : '')\n if (!base) return path\n const normalizedBase = base.replace(/\\/$/, '')\n return `${normalizedBase}${path.startsWith('/') ? path : `/${path}`}`\n}\n\nfunction resolveFileExtension(fileName?: string | null): string {\n if (!fileName) return ''\n const normalized = fileName.trim()\n if (!normalized) return ''\n const lastDot = normalized.lastIndexOf('.')\n if (lastDot === -1 || lastDot === normalized.length - 1) return ''\n return normalized.slice(lastDot + 1).toLowerCase()\n}\n\nconst EXTENSION_ICON_MAP: Record<string, LucideIcon> = {\n pdf: FileText,\n doc: FileText,\n docx: FileText,\n txt: FileText,\n md: FileText,\n rtf: FileText,\n xls: FileSpreadsheet,\n xlsx: FileSpreadsheet,\n csv: FileSpreadsheet,\n ods: FileSpreadsheet,\n ppt: FileText,\n pptx: FileText,\n zip: FileArchive,\n gz: FileArchive,\n rar: FileArchive,\n tgz: FileArchive,\n '7z': FileArchive,\n tar: FileArchive,\n json: FileCode,\n js: FileCode,\n ts: FileCode,\n jsx: FileCode,\n tsx: FileCode,\n html: FileCode,\n css: FileCode,\n xml: FileCode,\n yaml: FileCode,\n yml: FileCode,\n mp3: FileAudio,\n wav: FileAudio,\n flac: FileAudio,\n ogg: FileAudio,\n mp4: FileVideo,\n mov: FileVideo,\n avi: FileVideo,\n webm: FileVideo,\n}\n\nconst MIME_FALLBACK_ICONS: Record<string, LucideIcon> = {\n audio: FileAudio,\n video: FileVideo,\n text: FileText,\n application: FileText,\n}\n\nfunction resolveAttachmentPlaceholder(mimeType?: string | null, fileName?: string | null): { icon: LucideIcon; label: string } {\n const extension = resolveFileExtension(fileName)\n const normalizedMime = typeof mimeType === 'string' ? mimeType.toLowerCase() : ''\n if (extension && EXTENSION_ICON_MAP[extension]) {\n return { icon: EXTENSION_ICON_MAP[extension], label: extension.toUpperCase() }\n }\n if (!extension && normalizedMime.includes('pdf')) {\n return { icon: FileText, label: 'PDF' }\n }\n if (!extension && normalizedMime.includes('zip')) {\n return { icon: FileArchive, label: 'ZIP' }\n }\n if (!extension && normalizedMime.includes('json')) {\n return { icon: FileCode, label: 'JSON' }\n }\n const mimeRoot = normalizedMime.split('/')[0] || ''\n if (mimeRoot && MIME_FALLBACK_ICONS[mimeRoot]) {\n return { icon: MIME_FALLBACK_ICONS[mimeRoot], label: mimeRoot.toUpperCase() }\n }\n const fallbackSource = extension || mimeRoot || 'file'\n const fallbackLabel = fallbackSource.slice(0, 6).toUpperCase()\n return { icon: File, label: fallbackLabel }\n}\n\nfunction normalizeCustomFieldSubmitValue(value: unknown): unknown {\n if (Array.isArray(value)) {\n return value.filter((entry) => entry !== undefined)\n }\n if (value === undefined) return null\n return value\n}\n\n\ntype AttachmentUploadFormValues = {\n files: File[]\n partitionCode?: string\n tags?: string[]\n assignments?: AssignmentDraft[]\n} & Record<string, unknown>\n\ntype AssignmentsEditorProps = {\n value: AssignmentDraft[]\n onChange: (next: AssignmentDraft[]) => void\n labels: {\n title: string\n description: string\n type: string\n id: string\n href: string\n label?: string\n add: string\n remove: string\n }\n disabled?: boolean\n}\n\nfunction AttachmentAssignmentsEditor({ value, onChange, labels, disabled }: AssignmentsEditorProps) {\n const handleChange = React.useCallback(\n (index: number, patch: Partial<AssignmentDraft>) => {\n onChange(value.map((entry, idx) => (idx === index ? { ...entry, ...patch } : entry)))\n },\n [onChange, value],\n )\n\n const handleRemove = React.useCallback(\n (index: number) => {\n onChange(value.filter((_, idx) => idx !== index))\n },\n [onChange, value],\n )\n\n const handleAdd = React.useCallback(() => {\n onChange([...value, { type: '', id: '', href: '', label: '' }])\n }, [onChange, value])\n\n return (\n <div className=\"space-y-2\">\n <div>\n <div className=\"text-sm font-medium\">{labels.title}</div>\n <div className=\"text-xs text-muted-foreground\">{labels.description}</div>\n </div>\n <div className=\"space-y-3\">\n {value.length === 0 ? (\n <div className=\"text-xs text-muted-foreground\">No assignments yet.</div>\n ) : (\n value.map((entry, index) => (\n <div key={`${index}-${entry.type}-${entry.id}`} className=\"rounded border p-3 space-y-2\">\n <div className=\"grid gap-2 sm:grid-cols-2\">\n <div className=\"space-y-1\">\n <label className=\"text-xs font-medium\">{labels.type}</label>\n <input\n className=\"w-full rounded border px-2 py-1 text-sm\"\n value={entry.type}\n disabled={disabled}\n onChange={(event) => handleChange(index, { type: event.target.value })}\n />\n </div>\n <div className=\"space-y-1\">\n <label className=\"text-xs font-medium\">{labels.id}</label>\n <input\n className=\"w-full rounded border px-2 py-1 text-sm\"\n value={entry.id}\n disabled={disabled}\n onChange={(event) => handleChange(index, { id: event.target.value })}\n />\n </div>\n </div>\n <div className=\"grid gap-2 sm:grid-cols-2\">\n <div className=\"space-y-1\">\n <label className=\"text-xs font-medium\">{labels.href}</label>\n <input\n className=\"w-full rounded border px-2 py-1 text-sm\"\n value={entry.href ?? ''}\n disabled={disabled}\n onChange={(event) => handleChange(index, { href: event.target.value })}\n />\n </div>\n {labels.label ? (\n <div className=\"space-y-1\">\n <label className=\"text-xs font-medium\">{labels.label}</label>\n <input\n className=\"w-full rounded border px-2 py-1 text-sm\"\n value={entry.label ?? ''}\n disabled={disabled}\n onChange={(event) => handleChange(index, { label: event.target.value })}\n />\n </div>\n ) : null}\n </div>\n <div className=\"flex justify-end\">\n <Button\n type=\"button\"\n size=\"sm\"\n variant=\"outline\"\n disabled={disabled}\n onClick={() => handleRemove(index)}\n className=\"inline-flex items-center gap-1 text-muted-foreground\"\n >\n <Trash2 className=\"h-4 w-4\" />\n {labels.remove}\n </Button>\n </div>\n </div>\n ))\n )}\n </div>\n <Button type=\"button\" variant=\"outline\" size=\"sm\" disabled={disabled} onClick={handleAdd} className=\"inline-flex items-center gap-1\">\n <Plus className=\"h-4 w-4\" />\n {labels.add}\n </Button>\n </div>\n )\n}\n\ntype AttachmentFilesFieldProps = CrudCustomFieldRenderProps & {\n labels: {\n dropHint: string\n choose: string\n uploading: string\n empty: string\n }\n uploading: boolean\n}\n\nfunction AttachmentFilesField({\n value,\n setValue,\n disabled,\n error,\n labels,\n uploading,\n}: AttachmentFilesFieldProps) {\n const files = React.useMemo(() => (Array.isArray(value) ? (value as File[]) : []), [value])\n const [isDragOver, setDragOver] = React.useState(false)\n const fileInputRef = React.useRef<HTMLInputElement | null>(null)\n\n const acceptFiles = React.useCallback(\n (list: FileList | null) => {\n if (!list?.length) return\n const dedupe = new Map<string, File>(files.map((file) => [`${file.name}:${file.size}`, file]))\n Array.from(list).forEach((file) => {\n dedupe.set(`${file.name}:${file.size}`, file)\n })\n setValue(Array.from(dedupe.values()))\n },\n [files, setValue],\n )\n\n const handleDrop = React.useCallback(\n (event: React.DragEvent<HTMLDivElement>) => {\n if (disabled || uploading) return\n event.preventDefault()\n event.stopPropagation()\n setDragOver(false)\n acceptFiles(event.dataTransfer?.files ?? null)\n },\n [acceptFiles, disabled, uploading],\n )\n\n const handleDragOver = React.useCallback(\n (event: React.DragEvent<HTMLDivElement>) => {\n if (disabled || uploading) return\n event.preventDefault()\n event.stopPropagation()\n setDragOver(true)\n },\n [disabled, uploading],\n )\n\n const handleDragLeave = React.useCallback((event: React.DragEvent<HTMLDivElement>) => {\n event.preventDefault()\n event.stopPropagation()\n setDragOver(false)\n }, [])\n\n const removeFile = React.useCallback(\n (name: string, size: number) => {\n if (disabled || uploading) return\n setValue(files.filter((file) => !(file.name === name && file.size === size)))\n },\n [disabled, files, setValue, uploading],\n )\n\n const pickFiles = React.useCallback(() => {\n if (disabled || uploading) return\n fileInputRef.current?.click()\n }, [disabled, uploading])\n\n const renderFileList = () => {\n if (!files.length) {\n return <p className=\"text-xs text-muted-foreground\">{labels.empty}</p>\n }\n return (\n <div className=\"space-y-2\">\n {files.map((candidate) => (\n <div key={`${candidate.name}-${candidate.size}`} className=\"flex items-center justify-between rounded border px-3 py-2 text-sm\">\n <div>\n <div className=\"font-medium\">{candidate.name}</div>\n <div className=\"text-xs text-muted-foreground\">{formatFileSize(candidate.size)}</div>\n </div>\n <Button\n type=\"button\"\n variant=\"ghost\"\n size=\"icon\"\n onClick={() => removeFile(candidate.name, candidate.size)}\n disabled={disabled || uploading}\n >\n <Trash2 className=\"h-4 w-4\" />\n </Button>\n </div>\n ))}\n </div>\n )\n }\n\n return (\n <div className=\"space-y-2\">\n <div\n className={cn(\n 'flex flex-col items-center justify-center rounded-lg border border-dashed p-6 text-center transition-colors',\n isDragOver ? 'border-primary bg-primary/5' : 'border-muted-foreground/30',\n disabled || uploading ? 'opacity-70' : '',\n )}\n onDrop={handleDrop}\n onDragOver={handleDragOver}\n onDragLeave={handleDragLeave}\n role=\"presentation\"\n >\n <Upload className=\"mx-auto h-6 w-6 text-muted-foreground\" />\n <p className=\"mt-2 text-sm text-muted-foreground\">{labels.dropHint}</p>\n <Button type=\"button\" variant=\"outline\" size=\"sm\" className=\"mt-4\" onClick={pickFiles} disabled={disabled || uploading}>\n {uploading ? labels.uploading : labels.choose}\n </Button>\n <input\n ref={fileInputRef}\n type=\"file\"\n className=\"hidden\"\n multiple\n onChange={(event) => {\n acceptFiles(event.target.files)\n event.currentTarget.value = ''\n }}\n disabled={disabled || uploading}\n />\n </div>\n {renderFileList()}\n {error ? <p className=\"text-xs font-medium text-red-600\">{error}</p> : null}\n </div>\n )\n}\n\ntype AttachmentUploadFormProps = {\n partitions: Array<{ code: string; title: string }>\n availableTags: string[]\n onUploaded: () => void\n onCancel: () => void\n}\n\nfunction AttachmentUploadForm({ partitions, availableTags, onUploaded, onCancel }: AttachmentUploadFormProps) {\n const t = useT()\n const [isUploading, setIsUploading] = React.useState(false)\n const [uploadProgress, setUploadProgress] = React.useState<{ completed: number; total: number }>({ completed: 0, total: 0 })\n\n const partitionOptions = React.useMemo(\n () =>\n partitions.map((entry) => ({\n value: entry.code,\n label: entry.title || entry.code,\n })),\n [partitions],\n )\n\n const assignmentLabels = React.useMemo(\n () => ({\n title: t('attachments.library.upload.assignments.title', 'Assignments'),\n description: t(\n 'attachments.library.upload.assignments.description',\n 'Optionally link this file to existing records now or add them later.',\n ),\n type: t('attachments.library.upload.assignments.type', 'Type'),\n id: t('attachments.library.upload.assignments.id', 'Record ID'),\n href: t('attachments.library.upload.assignments.href', 'Link'),\n label: t('attachments.library.upload.assignments.label', 'Label'),\n add: t('attachments.library.upload.assignments.add', 'Add assignment'),\n remove: t('attachments.library.upload.assignments.remove', 'Remove'),\n }),\n [t],\n )\n\n const formSchema = React.useMemo(\n () =>\n z\n .object({\n files: z.array(z.any()).min(1, { message: t('attachments.library.upload.fileRequired', 'Select at least one file to upload.') }),\n partitionCode: z.string().optional(),\n tags: z.array(z.string()).optional(),\n assignments: z\n .array(\n z.object({\n type: z.string().min(1),\n id: z.string().min(1),\n href: z.string().optional(),\n label: z.string().optional(),\n }),\n )\n .optional(),\n })\n .passthrough(),\n [t],\n )\n\n const fields = React.useMemo<CrudField[]>(() => {\n return [\n {\n id: 'files',\n label: t('attachments.library.upload.file', 'Files'),\n type: 'custom',\n component: (props) => (\n <AttachmentFilesField\n {...props}\n uploading={isUploading}\n labels={{\n dropHint: t('attachments.library.upload.dropHint', 'Drag and drop files here or click to upload.'),\n choose: t('attachments.library.upload.choose', 'Choose files'),\n uploading: t('attachments.library.upload.submitting', 'Uploading\u2026'),\n empty: t('attachments.library.upload.noFiles', 'No files selected yet.'),\n }}\n />\n ),\n },\n {\n id: 'partitionCode',\n label: t('attachments.library.upload.partition', 'Partition'),\n type: 'select',\n options: [\n { value: '', label: t('attachments.library.upload.partitionDefault', 'Default (private)') },\n ...partitionOptions,\n ],\n },\n {\n id: 'tags',\n label: t('attachments.library.table.tags', 'Tags'),\n type: 'custom',\n component: ({ value, setValue, disabled }) => (\n <TagsInput\n value={Array.isArray(value) ? (value as string[]) : []}\n onChange={(next) => setValue(next)}\n suggestions={availableTags}\n placeholder={t('attachments.library.upload.tagsPlaceholder', 'Add tags')}\n disabled={Boolean(disabled) || isUploading}\n />\n ),\n },\n {\n id: 'assignments',\n label: '',\n type: 'custom',\n component: ({ value, setValue, disabled }) => (\n <AttachmentAssignmentsEditor\n value={Array.isArray(value) ? (value as AssignmentDraft[]) : []}\n onChange={(next) => setValue(next)}\n labels={assignmentLabels}\n disabled={Boolean(disabled) || isUploading}\n />\n ),\n },\n ]\n }, [assignmentLabels, availableTags, isUploading, partitionOptions, t])\n\n const groups = React.useMemo<CrudFormGroup[]>(() => {\n return [\n {\n id: 'details',\n title: t('attachments.library.upload.title', 'Upload attachment'),\n column: 1,\n fields: ['files', 'partitionCode', 'tags', 'assignments'],\n },\n {\n id: 'customFields',\n title: t('entities.customFields.title', 'Custom attributes'),\n column: 2,\n kind: 'customFields',\n },\n ]\n }, [t])\n\n const uploadPercentage = uploadProgress.total\n ? Math.min(100, Math.round((uploadProgress.completed / uploadProgress.total) * 100))\n : 0\n\n const handleSubmit = React.useCallback(\n async (values: AttachmentUploadFormValues) => {\n const files = Array.isArray(values.files) ? values.files : []\n if (!files.length) {\n throw new Error(t('attachments.library.upload.fileRequired', 'Select at least one file to upload.'))\n }\n setUploadProgress({ completed: 0, total: files.length })\n setIsUploading(true)\n try {\n const tags = Array.isArray(values.tags)\n ? values.tags\n .map((tag) => (typeof tag === 'string' ? tag.trim() : ''))\n .filter((tag) => tag.length > 0)\n : []\n const cleanedAssignments =\n Array.isArray(values.assignments) && values.assignments.length\n ? values.assignments\n .map((assignment) => ({\n type: assignment.type?.trim() ?? '',\n id: assignment.id?.trim() ?? '',\n href: assignment.href?.trim() || undefined,\n label: assignment.label?.trim() || undefined,\n }))\n .filter((assignment) => assignment.type && assignment.id)\n : []\n const customFields = collectCustomFieldValues(values, {\n transform: (value) => normalizeCustomFieldSubmitValue(value),\n })\n let completed = 0\n for (const file of files) {\n const fd = new FormData()\n fd.set('entityId', LIBRARY_ENTITY_ID)\n fd.set(\n 'recordId',\n typeof crypto !== 'undefined' && typeof crypto.randomUUID === 'function' ? crypto.randomUUID() : String(Date.now()),\n )\n fd.set('file', file)\n if (typeof values.partitionCode === 'string' && values.partitionCode.trim().length) {\n fd.set('partitionCode', values.partitionCode.trim())\n }\n if (tags.length) fd.set('tags', JSON.stringify(tags))\n if (cleanedAssignments.length) fd.set('assignments', JSON.stringify(cleanedAssignments))\n if (Object.keys(customFields).length) fd.set('customFields', JSON.stringify(customFields))\n const call = await apiCall<{ error?: string }>('/api/attachments', {\n method: 'POST',\n body: fd,\n })\n if (!call.ok) {\n const message = call.result?.error || t('attachments.library.upload.failed', 'Upload failed.')\n throw new Error(message)\n }\n completed += 1\n setUploadProgress({ completed, total: files.length })\n }\n flash(t('attachments.library.upload.success', 'Attachment uploaded.'), 'success')\n onUploaded()\n onCancel()\n } catch (err: any) {\n const message = err?.message || t('attachments.library.upload.failed', 'Upload failed.')\n flash(message, 'error')\n throw new Error(message)\n } finally {\n setIsUploading(false)\n }\n },\n [onCancel, onUploaded, t],\n )\n\n return (\n <div className=\"relative\">\n <CrudForm<AttachmentUploadFormValues>\n embedded\n schema={formSchema}\n entityId={E.attachments.attachment}\n fields={fields}\n groups={groups}\n initialValues={{ files: [], tags: [], assignments: [], partitionCode: '' }}\n submitLabel={\n isUploading\n ? t('attachments.library.upload.submitting', 'Uploading\u2026')\n : t('attachments.library.upload.submit', 'Upload')\n }\n extraActions={\n <Button type=\"button\" variant=\"outline\" onClick={onCancel} disabled={isUploading}>\n {t('attachments.library.upload.cancel', 'Cancel')}\n </Button>\n }\n onSubmit={handleSubmit}\n />\n {isUploading ? (\n <div className=\"pointer-events-none absolute inset-0 z-20 flex items-center justify-center bg-background/90 px-6 text-center backdrop-blur\">\n <div className=\"flex w-full max-w-sm flex-col items-center gap-4 rounded-xl border border-border/50 bg-card/95 px-6 py-8 shadow-2xl\">\n <Spinner size=\"lg\" className=\"border-primary/50 border-t-primary\" />\n <div className=\"w-full space-y-3\">\n <p className=\"text-base font-semibold\">\n {t('attachments.library.upload.progressLabel', 'Uploading files')}\n </p>\n {uploadProgress.total > 0 ? (\n <>\n <p className=\"text-sm text-muted-foreground\">\n {uploadProgress.completed}/{uploadProgress.total}\n </p>\n <div className=\"h-2 w-full rounded bg-muted\">\n <div\n className=\"h-2 rounded bg-primary transition-all\"\n style={{\n width: `${uploadPercentage}%`,\n }}\n />\n </div>\n </>\n ) : null}\n </div>\n </div>\n </div>\n ) : null}\n </div>\n )\n}\ntype UploadDialogProps = {\n open: boolean\n onOpenChange: (next: boolean) => void\n partitions: Array<{ code: string; title: string }>\n availableTags: string[]\n onUploaded: () => void\n}\n\nfunction AttachmentUploadDialog({ open, onOpenChange, partitions, availableTags, onUploaded }: UploadDialogProps) {\n const t = useT()\n const [formResetKey, setFormResetKey] = React.useState(0)\n const previousOpen = React.useRef(open)\n\n React.useEffect(() => {\n if (previousOpen.current && !open) {\n setFormResetKey((prev) => prev + 1)\n }\n previousOpen.current = open\n }, [open])\n\n const handleDialogChange = React.useCallback(\n (next: boolean) => {\n onOpenChange(next)\n },\n [onOpenChange],\n )\n\n const handleUploaded = React.useCallback(() => {\n onUploaded()\n }, [onUploaded])\n\n return (\n <Dialog open={open} onOpenChange={handleDialogChange}>\n <DialogContent className=\"sm:max-w-[54.6rem]\">\n <DialogHeader>\n <DialogTitle>{t('attachments.library.upload.title', 'Upload attachment')}</DialogTitle>\n </DialogHeader>\n <AttachmentUploadForm\n key={formResetKey}\n partitions={partitions}\n availableTags={availableTags}\n onUploaded={handleUploaded}\n onCancel={() => handleDialogChange(false)}\n />\n </DialogContent>\n </Dialog>\n )\n}\n\nexport function AttachmentLibrary() {\n const t = useT()\n const queryClient = useQueryClient()\n const [page, setPage] = React.useState(1)\n const [search, setSearch] = React.useState('')\n const [filterValues, setFilterValues] = React.useState<FilterValues>({})\n const [sorting, setSorting] = React.useState<SortingState>([{ id: 'createdAt', desc: true }])\n const [metadataDialogOpen, setMetadataDialogOpen] = React.useState(false)\n const [selectedRow, setSelectedRow] = React.useState<AttachmentRow | null>(null)\n const [uploadDialogOpen, setUploadDialogOpen] = React.useState(false)\n const [deleteDialogOpen, setDeleteDialogOpen] = React.useState(false)\n const [deleteTarget, setDeleteTarget] = React.useState<AttachmentRow | null>(null)\n const [deletePending, setDeletePending] = React.useState(false)\n const filterSignature = React.useMemo(() => buildFilterSignature(filterValues), [filterValues])\n const sortingSignature = React.useMemo(() => JSON.stringify(sorting), [sorting])\n\n const { data, isLoading, error, refetch } = useQuery({\n queryKey: ['attachments-library', page, search, filterSignature, sortingSignature],\n queryFn: async () => {\n const params = new URLSearchParams()\n params.set('page', String(page))\n params.set('pageSize', String(PAGE_SIZE))\n if (search.trim().length > 0) params.set('search', search.trim())\n const partition = typeof filterValues.partition === 'string' ? filterValues.partition : ''\n if (partition) params.set('partition', partition)\n const tags = Array.isArray(filterValues.tags) ? filterValues.tags : []\n if (tags.length > 0) params.set('tags', tags.join(','))\n if (sorting.length > 0) {\n const primary = sorting[0]\n params.set('sortField', primary.id)\n params.set('sortDir', primary.desc ? 'desc' : 'asc')\n }\n const call = await apiCall<AttachmentLibraryResponse>(`/api/attachments/library?${params.toString()}`)\n if (!call.ok || !call.result) {\n const message = call.result?.error || t('attachments.library.errors.load', 'Failed to load attachments.')\n throw new Error(message)\n }\n return call.result\n },\n })\n\n const partitions = data?.partitions ?? []\n const availableTags = data?.availableTags ?? []\n\n const filters = React.useMemo<FilterDef[]>(() => {\n const partitionOptions = partitions.map((entry) => ({\n value: entry.code,\n label: entry.title || entry.code,\n }))\n return [\n {\n id: 'partition',\n label: t('attachments.library.filters.partition', 'Partition'),\n type: 'select',\n options: partitionOptions,\n },\n {\n id: 'tags',\n label: t('attachments.library.filters.tags', 'Tags'),\n type: 'tags',\n placeholder: t('attachments.library.filters.tagsPlaceholder', 'Filter by tag'),\n options: availableTags.map((tag) => ({ value: tag, label: tag })),\n },\n ]\n }, [availableTags, partitions, t])\n\n const items = data?.items ?? []\n\n const columns = React.useMemo<ColumnDef<AttachmentRow>[]>(() => {\n return [\n {\n id: 'preview',\n header: '',\n enableSorting: false,\n cell: ({ row }) => {\n const value = row.original\n if (value.thumbnailUrl) {\n return (\n <div className=\"h-16 w-16 overflow-hidden rounded border bg-muted\">\n <img\n src={value.thumbnailUrl}\n alt={value.fileName}\n className=\"h-full w-full object-cover\"\n loading=\"lazy\"\n />\n </div>\n )\n }\n const placeholder = resolveAttachmentPlaceholder(value.mimeType, value.fileName)\n const PlaceholderIcon = placeholder.icon\n return (\n <div className=\"flex h-16 w-16 flex-col items-center justify-center rounded border bg-muted text-[10px] font-semibold uppercase text-muted-foreground\">\n <PlaceholderIcon className=\"mb-1 h-5 w-5 text-muted-foreground\" aria-hidden />\n {placeholder.label}\n </div>\n )\n },\n },\n {\n id: 'fileName',\n accessorKey: 'fileName',\n header: t('attachments.library.table.file', 'File'),\n cell: ({ row }) => {\n const value = row.original\n return (\n <div className=\"space-y-1 min-w-0 max-w-[280px]\">\n <div className=\"font-medium truncate\" title={value.fileName}>\n {value.fileName}\n </div>\n <div className=\"text-xs text-muted-foreground\">\n {formatFileSize(value.fileSize)} \u2022 {value.mimeType || 'application/octet-stream'}\n </div>\n <div className=\"text-xs text-muted-foreground line-clamp-2\">\n {value.content?.trim()\n ? value.content\n : t('attachments.library.metadata.noContent', 'No text extracted')}\n </div>\n </div>\n )\n },\n },\n {\n id: 'tags',\n accessorKey: 'tags',\n header: t('attachments.library.table.tags', 'Tags'),\n enableSorting: false,\n cell: ({ row }) => {\n const tags = row.original.tags ?? []\n if (!tags.length) return <span className=\"text-xs text-muted-foreground\">\u2014</span>\n return (\n <div className=\"flex flex-wrap gap-1\">\n {tags.map((tag) => (\n <Badge key={tag} variant=\"outline\">\n {tag}\n </Badge>\n ))}\n </div>\n )\n },\n },\n {\n id: 'assignments',\n accessorKey: 'assignments',\n header: t('attachments.library.table.assignments', 'Assignments'),\n enableSorting: false,\n cell: ({ row }) => {\n const assignments = filterLibraryAssignments(row.original.assignments)\n if (!assignments.length) return <span className=\"text-xs text-muted-foreground\">\u2014</span>\n return (\n <div className=\"flex flex-col gap-1\">\n {assignments.map((assignment) => {\n const label = assignment.label?.trim() || assignment.id\n const hideType =\n assignment.type === (E as any).catalog?.catalog_product ||\n assignment.type === (E as any).catalog?.catalog_product_variant\n const content = hideType ? label : `${assignment.type}: ${label}`\n return assignment.href ? (\n <a\n key={`${assignment.type}-${assignment.id}-${assignment.href}`}\n href={assignment.href}\n className=\"text-sm text-blue-600 underline\"\n target=\"_blank\"\n rel=\"noreferrer\"\n >\n {content}\n </a>\n ) : (\n <div key={`${assignment.type}-${assignment.id}`} className=\"text-sm\">\n {content}\n </div>\n )\n })}\n </div>\n )\n },\n },\n {\n id: 'partitionCode',\n accessorKey: 'partitionCode',\n header: t('attachments.library.table.partition', 'Partition'),\n cell: ({ row }) => (\n <div className=\"text-sm text-muted-foreground\">\n {row.original.partitionTitle ?? row.original.partitionCode}\n </div>\n ),\n },\n {\n id: 'createdAt',\n accessorKey: 'createdAt',\n header: t('attachments.library.table.created', 'Created'),\n cell: ({ row }) => {\n const createdAt = row.original.createdAt\n return (\n <div className=\"text-sm text-muted-foreground\">\n {createdAt ? humanDate(createdAt) : '\u2014'}\n </div>\n )\n },\n },\n {\n id: 'download',\n header: t('attachments.library.table.download', 'Download'),\n enableSorting: false,\n cell: ({ row }) => {\n const downloadPath = buildAttachmentFileUrl(row.original.id, { download: true })\n const absolute = resolveAbsoluteUrl(downloadPath)\n return (\n <Button variant=\"ghost\" size=\"icon\" asChild>\n <a href={absolute} download aria-label={t('attachments.library.table.download', 'Download')}>\n <Download className=\"h-4 w-4\" />\n </a>\n </Button>\n )\n },\n },\n ]\n }, [t])\n\n const openMetadataDialog = React.useCallback((row: AttachmentRow) => {\n setSelectedRow(row)\n setMetadataDialogOpen(true)\n }, [])\n\n const openDeleteDialog = React.useCallback((row: AttachmentRow) => {\n setDeleteTarget(row)\n setDeleteDialogOpen(true)\n }, [])\n\n const handleMetadataSave = React.useCallback(\n async (id: string, payload: AttachmentMetadataSavePayload) => {\n try {\n const body: Record<string, unknown> = {\n tags: payload.tags,\n assignments: payload.assignments,\n }\n if (payload.customFields && Object.keys(payload.customFields).length) {\n body.customFields = payload.customFields\n }\n const call = await apiCall<{ error?: string }>(`/api/attachments/library/${encodeURIComponent(id)}`, {\n method: 'PATCH',\n headers: { 'content-type': 'application/json' },\n body: JSON.stringify(body),\n })\n if (!call.ok) {\n const message =\n call.result?.error || t('attachments.library.metadata.error', 'Failed to update metadata.')\n flash(message, 'error')\n return\n }\n flash(t('attachments.library.metadata.success', 'Attachment updated.'), 'success')\n await queryClient.invalidateQueries({ queryKey: ['attachments-library'], exact: false })\n setMetadataDialogOpen(false)\n } catch (err: any) {\n flash(err?.message || t('attachments.library.metadata.error', 'Failed to update metadata.'), 'error')\n }\n },\n [queryClient, t],\n )\n\n const handleUploadCompleted = React.useCallback(async () => {\n await queryClient.invalidateQueries({ queryKey: ['attachments-library'], exact: false })\n }, [queryClient])\n\n const handleDelete = React.useCallback(async () => {\n if (!deleteTarget) return\n try {\n setDeletePending(true)\n const call = await apiCall<{ error?: string }>(\n `/api/attachments/library/${encodeURIComponent(deleteTarget.id)}`,\n { method: 'DELETE' },\n )\n if (!call.ok) {\n const message =\n call.result?.error || t('attachments.library.errors.delete', 'Failed to delete attachment.')\n flash(message, 'error')\n return\n }\n flash(t('attachments.library.messages.deleted', 'Attachment removed.'), 'success')\n if (selectedRow?.id === deleteTarget.id) {\n setSelectedRow(null)\n setMetadataDialogOpen(false)\n }\n setDeleteDialogOpen(false)\n setDeleteTarget(null)\n await queryClient.invalidateQueries({ queryKey: ['attachments-library'], exact: false })\n } catch (err: any) {\n flash(err?.message || t('attachments.library.errors.delete', 'Failed to delete attachment.'), 'error')\n } finally {\n setDeletePending(false)\n }\n }, [deleteTarget, queryClient, selectedRow, t])\n\n const total = data?.total ?? 0\n const totalPages = data?.totalPages ?? 1\n return (\n <>\n <DataTable<AttachmentRow>\n title={t('attachments.library.title', 'Attachments')}\n refreshButton={{\n label: t('attachments.library.actions.refresh', 'Refresh'),\n onRefresh: () => { void refetch() },\n isRefreshing: isLoading,\n }}\n actions={(\n <Button onClick={() => setUploadDialogOpen(true)}>\n {t('attachments.library.actions.upload', 'Upload')}\n </Button>\n )}\n columns={columns}\n data={items}\n sorting={sorting}\n onSortingChange={setSorting}\n rowActions={(row) => (\n <RowActions\n items={[\n {\n id: 'open',\n label: t('attachments.library.actions.open', 'Open'),\n onSelect: () => {\n if (!row.url) return\n window.open(row.url, '_blank', 'noopener,noreferrer')\n },\n },\n {\n id: 'edit',\n label: t('attachments.library.actions.edit', 'Edit metadata'),\n onSelect: () => openMetadataDialog(row),\n },\n {\n id: 'copy-url',\n label: t('attachments.library.actions.copyUrl', 'Copy URL'),\n onSelect: () => {\n if (!row.url) {\n flash(t('attachments.library.actions.copyError', 'Unable to copy link.'), 'error')\n return\n }\n const absolute = resolveAbsoluteUrl(row.url)\n navigator.clipboard\n .writeText(absolute)\n .then(() =>\n flash(\n t('attachments.library.actions.copied', 'Link copied.'),\n 'success',\n ),\n )\n .catch(() =>\n flash(\n t('attachments.library.actions.copyError', 'Unable to copy link.'),\n 'error',\n ),\n )\n },\n },\n {\n id: 'delete',\n label: t('attachments.library.actions.delete', 'Delete'),\n destructive: true,\n onSelect: () => openDeleteDialog(row),\n },\n ]}\n />\n )}\n onRowClick={(row) => openMetadataDialog(row)}\n isLoading={isLoading}\n error={error?.message}\n emptyState={\n <div className=\"py-10 text-center text-sm text-muted-foreground\">\n {t('attachments.library.table.empty', 'No attachments found.')}\n </div>\n }\n searchValue={search}\n onSearchChange={(value) => {\n setPage(1)\n setSearch(value)\n }}\n searchPlaceholder={t('attachments.library.table.search', 'Search files\u2026')}\n filters={filters}\n filterValues={filterValues}\n onFiltersApply={(values) => {\n setFilterValues(values)\n setPage(1)\n }}\n onFiltersClear={() => {\n setFilterValues({})\n setPage(1)\n }}\n pagination={{\n page,\n pageSize: PAGE_SIZE,\n total,\n totalPages,\n onPageChange: (next) => setPage(next),\n }}\n />\n <AttachmentMetadataDialog\n open={metadataDialogOpen}\n onOpenChange={setMetadataDialogOpen}\n item={selectedRow}\n availableTags={availableTags}\n onSave={handleMetadataSave}\n />\n <AttachmentDeleteDialog\n open={deleteDialogOpen}\n onOpenChange={setDeleteDialogOpen}\n fileName={deleteTarget?.fileName}\n onConfirm={handleDelete}\n isDeleting={deletePending}\n />\n <AttachmentUploadDialog\n open={uploadDialogOpen}\n onOpenChange={setUploadDialogOpen}\n partitions={partitions}\n availableTags={availableTags}\n onUploaded={handleUploadCompleted}\n />\n </>\n )\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -1,7 +1,7 @@
1
1
  const PARTITION_ENV_PREFIX = "ATTACHMENTS_PARTITION_";
2
2
  const ENV_SUFFIX_ROOT = "_ROOT";
3
3
  function toEnvFragment(code) {
4
- return code.replace(/([a-z0-9])([A-Z])/g, "$1_$2").replace(/[^a-zA-Z0-9]+/g, "_").replace(/_{2,}/g, "_").replace(/(?:^_|_$)/g, "").toUpperCase();
4
+ return code.replace(/([a-z0-9])([A-Z])/g, "$1_$2").replace(/[^a-zA-Z0-9]+/g, "_").replace(/_{2,}/g, "_").replace(/^_|_$/g, "").toUpperCase();
5
5
  }
6
6
  function resolvePartitionEnvKey(code) {
7
7
  const fragment = toEnvFragment(code);
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../src/modules/attachments/lib/partitionEnv.ts"],
4
- "sourcesContent": ["const PARTITION_ENV_PREFIX = 'ATTACHMENTS_PARTITION_'\nconst ENV_SUFFIX_ROOT = '_ROOT'\n\nfunction toEnvFragment(code: string): string {\n return code\n .replace(/([a-z0-9])([A-Z])/g, '$1_$2')\n .replace(/[^a-zA-Z0-9]+/g, '_')\n .replace(/_{2,}/g, '_')\n .replace(/(?:^_|_$)/g, '')\n .toUpperCase()\n}\n\nexport function resolvePartitionEnvKey(code: string): string {\n const fragment = toEnvFragment(code)\n return `${PARTITION_ENV_PREFIX}${fragment}${ENV_SUFFIX_ROOT}`\n}\n"],
5
- "mappings": "AAAA,MAAM,uBAAuB;AAC7B,MAAM,kBAAkB;AAExB,SAAS,cAAc,MAAsB;AAC3C,SAAO,KACJ,QAAQ,sBAAsB,OAAO,EACrC,QAAQ,kBAAkB,GAAG,EAC7B,QAAQ,UAAU,GAAG,EACrB,QAAQ,cAAc,EAAE,EACxB,YAAY;AACjB;AAEO,SAAS,uBAAuB,MAAsB;AAC3D,QAAM,WAAW,cAAc,IAAI;AACnC,SAAO,GAAG,oBAAoB,GAAG,QAAQ,GAAG,eAAe;AAC7D;",
4
+ "sourcesContent": ["const PARTITION_ENV_PREFIX = 'ATTACHMENTS_PARTITION_'\nconst ENV_SUFFIX_ROOT = '_ROOT'\n\nfunction toEnvFragment(code: string): string {\n return code\n .replace(/([a-z0-9])([A-Z])/g, '$1_$2')\n .replace(/[^a-zA-Z0-9]+/g, '_')\n .replace(/_{2,}/g, '_')\n .replace(/^_|_$/g, '')\n .toUpperCase()\n}\n\nexport function resolvePartitionEnvKey(code: string): string {\n const fragment = toEnvFragment(code)\n return `${PARTITION_ENV_PREFIX}${fragment}${ENV_SUFFIX_ROOT}`\n}\n"],
5
+ "mappings": "AAAA,MAAM,uBAAuB;AAC7B,MAAM,kBAAkB;AAExB,SAAS,cAAc,MAAsB;AAC3C,SAAO,KACJ,QAAQ,sBAAsB,OAAO,EACrC,QAAQ,kBAAkB,GAAG,EAC7B,QAAQ,UAAU,GAAG,EACrB,QAAQ,UAAU,EAAE,EACpB,YAAY;AACjB;AAEO,SAAS,uBAAuB,MAAsB;AAC3D,QAAM,WAAW,cAAc,IAAI;AACnC,SAAO,GAAG,oBAAoB,GAAG,QAAQ,GAAG,eAAe;AAC7D;",
6
6
  "names": []
7
7
  }
@@ -233,7 +233,7 @@ function UsersListPage() {
233
233
  const normalizedRoleIds = React.useMemo(() => {
234
234
  if (!effectiveRoleIds.length) return [];
235
235
  const unique = Array.from(new Set(effectiveRoleIds));
236
- unique.sort((a, b) => a.localeCompare(b));
236
+ unique.sort();
237
237
  return unique;
238
238
  }, [effectiveRoleIds]);
239
239
  const handleSearchChange = React.useCallback((value) => {