@open-mercato/core 0.6.5-develop.4534.1.b459babe6d → 0.6.5-develop.4559.1.839e136509

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 (644) hide show
  1. package/.turbo/turbo-build.log +1 -1
  2. package/AGENTS.md +5 -0
  3. package/dist/generated/entities/role/index.js +3 -1
  4. package/dist/generated/entities/role/index.js.map +2 -2
  5. package/dist/generated/entities/user/index.js +3 -1
  6. package/dist/generated/entities/user/index.js.map +2 -2
  7. package/dist/generated/entity-fields-registry.js +2 -0
  8. package/dist/generated/entity-fields-registry.js.map +2 -2
  9. package/dist/helpers/integration/communicationChannelsFixtures.js.map +2 -2
  10. package/dist/helpers/integration/dbFixtures.js +2 -1
  11. package/dist/helpers/integration/dbFixtures.js.map +2 -2
  12. package/dist/helpers/integration/optimisticLockUi.js +104 -0
  13. package/dist/helpers/integration/optimisticLockUi.js.map +7 -0
  14. package/dist/helpers/integration/salesFixtures.js +17 -0
  15. package/dist/helpers/integration/salesFixtures.js.map +2 -2
  16. package/dist/modules/api_keys/backend/api-keys/page.js +9 -5
  17. package/dist/modules/api_keys/backend/api-keys/page.js.map +2 -2
  18. package/dist/modules/attachments/components/AttachmentPartitionSettings.js +17 -9
  19. package/dist/modules/attachments/components/AttachmentPartitionSettings.js.map +2 -2
  20. package/dist/modules/auth/api/roles/acl/route.js +32 -13
  21. package/dist/modules/auth/api/roles/acl/route.js.map +2 -2
  22. package/dist/modules/auth/api/roles/route.js +3 -1
  23. package/dist/modules/auth/api/roles/route.js.map +2 -2
  24. package/dist/modules/auth/api/sidebar/preferences/route.js +71 -3
  25. package/dist/modules/auth/api/sidebar/preferences/route.js.map +2 -2
  26. package/dist/modules/auth/api/users/acl/route.js +42 -19
  27. package/dist/modules/auth/api/users/acl/route.js.map +2 -2
  28. package/dist/modules/auth/api/users/route.js +3 -1
  29. package/dist/modules/auth/api/users/route.js.map +2 -2
  30. package/dist/modules/auth/backend/roles/[id]/edit/page.js +24 -4
  31. package/dist/modules/auth/backend/roles/[id]/edit/page.js.map +2 -2
  32. package/dist/modules/auth/backend/roles/page.js +8 -4
  33. package/dist/modules/auth/backend/roles/page.js.map +2 -2
  34. package/dist/modules/auth/backend/users/[id]/edit/page.js +27 -5
  35. package/dist/modules/auth/backend/users/[id]/edit/page.js.map +2 -2
  36. package/dist/modules/auth/backend/users/page.js +6 -2
  37. package/dist/modules/auth/backend/users/page.js.map +2 -2
  38. package/dist/modules/auth/components/AclEditor.js +3 -1
  39. package/dist/modules/auth/components/AclEditor.js.map +2 -2
  40. package/dist/modules/auth/data/entities.js +6 -0
  41. package/dist/modules/auth/data/entities.js.map +2 -2
  42. package/dist/modules/auth/services/sidebarPreferencesService.js +32 -4
  43. package/dist/modules/auth/services/sidebarPreferencesService.js.map +2 -2
  44. package/dist/modules/business_rules/api/rules/route.js +28 -0
  45. package/dist/modules/business_rules/api/rules/route.js.map +2 -2
  46. package/dist/modules/business_rules/api/sets/route.js +28 -0
  47. package/dist/modules/business_rules/api/sets/route.js.map +2 -2
  48. package/dist/modules/business_rules/backend/rules/[id]/page.js +11 -4
  49. package/dist/modules/business_rules/backend/rules/[id]/page.js.map +3 -3
  50. package/dist/modules/business_rules/backend/rules/page.js +20 -11
  51. package/dist/modules/business_rules/backend/rules/page.js.map +2 -2
  52. package/dist/modules/business_rules/backend/sets/[id]/page.js +11 -4
  53. package/dist/modules/business_rules/backend/sets/[id]/page.js.map +2 -2
  54. package/dist/modules/business_rules/backend/sets/page.js +20 -11
  55. package/dist/modules/business_rules/backend/sets/page.js.map +2 -2
  56. package/dist/modules/catalog/api/categories/route.js +2 -0
  57. package/dist/modules/catalog/api/categories/route.js.map +2 -2
  58. package/dist/modules/catalog/api/products/route.js +2 -1
  59. package/dist/modules/catalog/api/products/route.js.map +2 -2
  60. package/dist/modules/catalog/backend/catalog/categories/[id]/edit/page.js +2 -0
  61. package/dist/modules/catalog/backend/catalog/categories/[id]/edit/page.js.map +2 -2
  62. package/dist/modules/catalog/backend/catalog/products/[id]/page.js +94 -40
  63. package/dist/modules/catalog/backend/catalog/products/[id]/page.js.map +2 -2
  64. package/dist/modules/catalog/backend/catalog/products/[productId]/variants/[variantId]/page.js +37 -8
  65. package/dist/modules/catalog/backend/catalog/products/[productId]/variants/[variantId]/page.js.map +2 -2
  66. package/dist/modules/catalog/backend/catalog/products/optionSchemaClient.js.map +2 -2
  67. package/dist/modules/catalog/commands/variants.js +32 -31
  68. package/dist/modules/catalog/commands/variants.js.map +2 -2
  69. package/dist/modules/catalog/components/PriceKindSettings.js +12 -5
  70. package/dist/modules/catalog/components/PriceKindSettings.js.map +2 -2
  71. package/dist/modules/catalog/components/categories/CategoriesDataTable.js.map +2 -2
  72. package/dist/modules/catalog/components/products/ProductMediaManager.js.map +2 -2
  73. package/dist/modules/catalog/components/products/ProductsDataTable.js +5 -3
  74. package/dist/modules/catalog/components/products/ProductsDataTable.js.map +2 -2
  75. package/dist/modules/catalog/components/products/productForm.js.map +2 -2
  76. package/dist/modules/catalog/components/products/variantForm.js +2 -1
  77. package/dist/modules/catalog/components/products/variantForm.js.map +2 -2
  78. package/dist/modules/communication_channels/api/post/test-seed/route.js +23 -2
  79. package/dist/modules/communication_channels/api/post/test-seed/route.js.map +2 -2
  80. package/dist/modules/communication_channels/backend/profile/communication-channels/page.js +5 -0
  81. package/dist/modules/communication_channels/backend/profile/communication-channels/page.js.map +2 -2
  82. package/dist/modules/communication_channels/commands/set-primary-channel.js +2 -1
  83. package/dist/modules/communication_channels/commands/set-primary-channel.js.map +2 -2
  84. package/dist/modules/currencies/backend/currencies/[id]/page.js +6 -3
  85. package/dist/modules/currencies/backend/currencies/[id]/page.js.map +2 -2
  86. package/dist/modules/currencies/backend/currencies/page.js +18 -11
  87. package/dist/modules/currencies/backend/currencies/page.js.map +2 -2
  88. package/dist/modules/currencies/backend/exchange-rates/[id]/page.js +1 -0
  89. package/dist/modules/currencies/backend/exchange-rates/[id]/page.js.map +2 -2
  90. package/dist/modules/currencies/backend/exchange-rates/page.js +10 -6
  91. package/dist/modules/currencies/backend/exchange-rates/page.js.map +2 -2
  92. package/dist/modules/currencies/commands/currencies.js +7 -5
  93. package/dist/modules/currencies/commands/currencies.js.map +2 -2
  94. package/dist/modules/currencies/components/CurrencyFetchingConfig.js +26 -19
  95. package/dist/modules/currencies/components/CurrencyFetchingConfig.js.map +2 -2
  96. package/dist/modules/customer_accounts/api/admin/roles/[id].js +28 -5
  97. package/dist/modules/customer_accounts/api/admin/roles/[id].js.map +2 -2
  98. package/dist/modules/customer_accounts/api/admin/roles.js +4 -2
  99. package/dist/modules/customer_accounts/api/admin/roles.js.map +2 -2
  100. package/dist/modules/customer_accounts/api/admin/users/[id].js +28 -5
  101. package/dist/modules/customer_accounts/api/admin/users/[id].js.map +2 -2
  102. package/dist/modules/customer_accounts/api/admin/users.js +2 -0
  103. package/dist/modules/customer_accounts/api/admin/users.js.map +2 -2
  104. package/dist/modules/customer_accounts/backend/customer_accounts/roles/[id]/page.js +16 -8
  105. package/dist/modules/customer_accounts/backend/customer_accounts/roles/[id]/page.js.map +2 -2
  106. package/dist/modules/customer_accounts/backend/customer_accounts/roles/page.js +8 -4
  107. package/dist/modules/customer_accounts/backend/customer_accounts/roles/page.js.map +2 -2
  108. package/dist/modules/customer_accounts/backend/customer_accounts/settings/domain/page.js +8 -4
  109. package/dist/modules/customer_accounts/backend/customer_accounts/settings/domain/page.js.map +2 -2
  110. package/dist/modules/customer_accounts/backend/customer_accounts/users/[id]/page.js +29 -18
  111. package/dist/modules/customer_accounts/backend/customer_accounts/users/[id]/page.js.map +2 -2
  112. package/dist/modules/customer_accounts/backend/customer_accounts/users/page.js +18 -11
  113. package/dist/modules/customer_accounts/backend/customer_accounts/users/page.js.map +2 -2
  114. package/dist/modules/customers/api/companies/route.js +13 -2
  115. package/dist/modules/customers/api/companies/route.js.map +2 -2
  116. package/dist/modules/customers/api/deals/route.js +2 -0
  117. package/dist/modules/customers/api/deals/route.js.map +2 -2
  118. package/dist/modules/customers/api/people/route.js +11 -2
  119. package/dist/modules/customers/api/people/route.js.map +2 -2
  120. package/dist/modules/customers/api/todos/route.js +1 -0
  121. package/dist/modules/customers/api/todos/route.js.map +2 -2
  122. package/dist/modules/customers/backend/config/customers/deals/page.js.map +2 -2
  123. package/dist/modules/customers/backend/config/customers/pipeline-stages/page.js +34 -21
  124. package/dist/modules/customers/backend/config/customers/pipeline-stages/page.js.map +2 -2
  125. package/dist/modules/customers/backend/customers/companies/[id]/page.js +45 -27
  126. package/dist/modules/customers/backend/customers/companies/[id]/page.js.map +2 -2
  127. package/dist/modules/customers/backend/customers/companies/page.js.map +2 -2
  128. package/dist/modules/customers/backend/customers/companies-v2/[id]/page.js +22 -5
  129. package/dist/modules/customers/backend/customers/companies-v2/[id]/page.js.map +2 -2
  130. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealFormHandlers.js +30 -8
  131. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealFormHandlers.js.map +2 -2
  132. package/dist/modules/customers/backend/customers/deals/[id]/page.js +1 -0
  133. package/dist/modules/customers/backend/customers/deals/[id]/page.js.map +2 -2
  134. package/dist/modules/customers/backend/customers/deals/page.js +16 -6
  135. package/dist/modules/customers/backend/customers/deals/page.js.map +2 -2
  136. package/dist/modules/customers/backend/customers/deals/pipeline/page.js +62 -39
  137. package/dist/modules/customers/backend/customers/deals/pipeline/page.js.map +2 -2
  138. package/dist/modules/customers/backend/customers/people/[id]/page.js +41 -26
  139. package/dist/modules/customers/backend/customers/people/[id]/page.js.map +2 -2
  140. package/dist/modules/customers/backend/customers/people/page.js.map +2 -2
  141. package/dist/modules/customers/backend/customers/people-v2/[id]/page.js +50 -23
  142. package/dist/modules/customers/backend/customers/people-v2/[id]/page.js.map +2 -2
  143. package/dist/modules/customers/commands/addresses.js +16 -14
  144. package/dist/modules/customers/commands/addresses.js.map +2 -2
  145. package/dist/modules/customers/commands/companies.js +1 -1
  146. package/dist/modules/customers/commands/companies.js.map +2 -2
  147. package/dist/modules/customers/commands/interactions.js +41 -4
  148. package/dist/modules/customers/commands/interactions.js.map +2 -2
  149. package/dist/modules/customers/commands/people.js +1 -1
  150. package/dist/modules/customers/commands/people.js.map +2 -2
  151. package/dist/modules/customers/commands/personCompanyLinks.js +8 -5
  152. package/dist/modules/customers/commands/personCompanyLinks.js.map +2 -2
  153. package/dist/modules/customers/commands/pipeline-stages.js +13 -11
  154. package/dist/modules/customers/commands/pipeline-stages.js.map +3 -3
  155. package/dist/modules/customers/components/AddressFormatSettings.js.map +2 -2
  156. package/dist/modules/customers/components/DictionarySettings.js +20 -13
  157. package/dist/modules/customers/components/DictionarySettings.js.map +2 -2
  158. package/dist/modules/customers/components/DictionarySortSettings.js +4 -0
  159. package/dist/modules/customers/components/DictionarySortSettings.js.map +2 -2
  160. package/dist/modules/customers/components/PipelineSettings.js +38 -23
  161. package/dist/modules/customers/components/PipelineSettings.js.map +2 -2
  162. package/dist/modules/customers/components/detail/ActivityTimeline.js +1 -1
  163. package/dist/modules/customers/components/detail/ActivityTimeline.js.map +2 -2
  164. package/dist/modules/customers/components/detail/AddressesSection.js +4 -0
  165. package/dist/modules/customers/components/detail/AddressesSection.js.map +2 -2
  166. package/dist/modules/customers/components/detail/CompanyPeopleSection.js +28 -22
  167. package/dist/modules/customers/components/detail/CompanyPeopleSection.js.map +2 -2
  168. package/dist/modules/customers/components/detail/DealsSection.js +36 -24
  169. package/dist/modules/customers/components/detail/DealsSection.js.map +2 -2
  170. package/dist/modules/customers/components/detail/EmailCardActions.js +5 -0
  171. package/dist/modules/customers/components/detail/EmailCardActions.js.map +2 -2
  172. package/dist/modules/customers/components/detail/EntityTagsDialog.js +7 -0
  173. package/dist/modules/customers/components/detail/EntityTagsDialog.js.map +2 -2
  174. package/dist/modules/customers/components/detail/ManageTagsDialog.js +34 -22
  175. package/dist/modules/customers/components/detail/ManageTagsDialog.js.map +2 -2
  176. package/dist/modules/customers/components/detail/PersonCompaniesSection.js +41 -29
  177. package/dist/modules/customers/components/detail/PersonCompaniesSection.js.map +2 -2
  178. package/dist/modules/customers/components/detail/RoleAssignmentRow.js +14 -8
  179. package/dist/modules/customers/components/detail/RoleAssignmentRow.js.map +2 -2
  180. package/dist/modules/customers/components/detail/ScheduleActivityDialog.js +14 -6
  181. package/dist/modules/customers/components/detail/ScheduleActivityDialog.js.map +2 -2
  182. package/dist/modules/customers/components/detail/hooks/useInteractionMutations.js +29 -13
  183. package/dist/modules/customers/components/detail/hooks/useInteractionMutations.js.map +2 -2
  184. package/dist/modules/customers/components/detail/hooks/useInteractions.js +77 -35
  185. package/dist/modules/customers/components/detail/hooks/useInteractions.js.map +2 -2
  186. package/dist/modules/customers/components/detail/hooks/usePersonTasks.js +25 -17
  187. package/dist/modules/customers/components/detail/hooks/usePersonTasks.js.map +2 -2
  188. package/dist/modules/customers/components/detail/schedule/useScheduleFormState.js.map +2 -2
  189. package/dist/modules/customers/components/formConfig.js.map +2 -2
  190. package/dist/modules/customers/data/guards.js +66 -0
  191. package/dist/modules/customers/data/guards.js.map +7 -0
  192. package/dist/modules/customers/di.js +37 -0
  193. package/dist/modules/customers/di.js.map +2 -2
  194. package/dist/modules/customers/lib/todoCompatibility.js +11 -0
  195. package/dist/modules/customers/lib/todoCompatibility.js.map +2 -2
  196. package/dist/modules/dashboards/components/WidgetVisibilityEditor.js.map +2 -2
  197. package/dist/modules/data_sync/api/options.js +4 -4
  198. package/dist/modules/data_sync/api/options.js.map +2 -2
  199. package/dist/modules/data_sync/api/schedules/route.js +9 -1
  200. package/dist/modules/data_sync/api/schedules/route.js.map +2 -2
  201. package/dist/modules/data_sync/backend/data-sync/page.js +17 -8
  202. package/dist/modules/data_sync/backend/data-sync/page.js.map +2 -2
  203. package/dist/modules/data_sync/components/IntegrationScheduleTab.js +43 -22
  204. package/dist/modules/data_sync/components/IntegrationScheduleTab.js.map +2 -2
  205. package/dist/modules/data_sync/lib/sync-schedule-service.js +9 -0
  206. package/dist/modules/data_sync/lib/sync-schedule-service.js.map +2 -2
  207. package/dist/modules/dictionaries/api/[dictionaryId]/entries/[entryId]/route.js +8 -1
  208. package/dist/modules/dictionaries/api/[dictionaryId]/entries/[entryId]/route.js.map +2 -2
  209. package/dist/modules/dictionaries/api/[dictionaryId]/route.js +17 -1
  210. package/dist/modules/dictionaries/api/[dictionaryId]/route.js.map +2 -2
  211. package/dist/modules/dictionaries/components/DictionariesManager.js +31 -10
  212. package/dist/modules/dictionaries/components/DictionariesManager.js.map +2 -2
  213. package/dist/modules/dictionaries/components/DictionaryEntriesEditor.js +28 -15
  214. package/dist/modules/dictionaries/components/DictionaryEntriesEditor.js.map +2 -2
  215. package/dist/modules/directory/api/organizations/route.js +3 -0
  216. package/dist/modules/directory/api/organizations/route.js.map +2 -2
  217. package/dist/modules/directory/backend/directory/organizations/[id]/edit/page.js +2 -0
  218. package/dist/modules/directory/backend/directory/organizations/[id]/edit/page.js.map +2 -2
  219. package/dist/modules/directory/backend/directory/organizations/page.js +9 -5
  220. package/dist/modules/directory/backend/directory/organizations/page.js.map +2 -2
  221. package/dist/modules/directory/backend/directory/tenants/[id]/edit/page.js +7 -3
  222. package/dist/modules/directory/backend/directory/tenants/[id]/edit/page.js.map +2 -2
  223. package/dist/modules/directory/backend/directory/tenants/page.js +8 -4
  224. package/dist/modules/directory/backend/directory/tenants/page.js.map +2 -2
  225. package/dist/modules/directory/commands/organizations.js +7 -2
  226. package/dist/modules/directory/commands/organizations.js.map +2 -2
  227. package/dist/modules/entities/api/records.js +66 -0
  228. package/dist/modules/entities/api/records.js.map +2 -2
  229. package/dist/modules/entities/backend/entities/user/[entityId]/records/[recordId]/page.js +1 -0
  230. package/dist/modules/entities/backend/entities/user/[entityId]/records/[recordId]/page.js.map +2 -2
  231. package/dist/modules/entities/backend/entities/user/[entityId]/records/page.js +8 -4
  232. package/dist/modules/entities/backend/entities/user/[entityId]/records/page.js.map +2 -2
  233. package/dist/modules/entities/lib/helpers.js +17 -0
  234. package/dist/modules/entities/lib/helpers.js.map +2 -2
  235. package/dist/modules/feature_toggles/api/global/[id]/override/route.js +2 -1
  236. package/dist/modules/feature_toggles/api/global/[id]/override/route.js.map +2 -2
  237. package/dist/modules/feature_toggles/api/overrides/route.js +15 -0
  238. package/dist/modules/feature_toggles/api/overrides/route.js.map +2 -2
  239. package/dist/modules/feature_toggles/backend/feature-toggles/global/[id]/edit/page.js +15 -14
  240. package/dist/modules/feature_toggles/backend/feature-toggles/global/[id]/edit/page.js.map +2 -2
  241. package/dist/modules/feature_toggles/components/FeatureToggleOverrideCard.js +20 -12
  242. package/dist/modules/feature_toggles/components/FeatureToggleOverrideCard.js.map +2 -2
  243. package/dist/modules/feature_toggles/components/FeatureTogglesTable.js +6 -2
  244. package/dist/modules/feature_toggles/components/FeatureTogglesTable.js.map +2 -2
  245. package/dist/modules/feature_toggles/components/formConfig.js +2 -1
  246. package/dist/modules/feature_toggles/components/formConfig.js.map +2 -2
  247. package/dist/modules/feature_toggles/components/overrideFormConfig.js +5 -1
  248. package/dist/modules/feature_toggles/components/overrideFormConfig.js.map +2 -2
  249. package/dist/modules/feature_toggles/data/validators.js +7 -4
  250. package/dist/modules/feature_toggles/data/validators.js.map +2 -2
  251. package/dist/modules/inbox_ops/api/settings/route.js +17 -2
  252. package/dist/modules/inbox_ops/api/settings/route.js.map +2 -2
  253. package/dist/modules/inbox_ops/backend/inbox-ops/settings/page.js +13 -8
  254. package/dist/modules/inbox_ops/backend/inbox-ops/settings/page.js.map +2 -2
  255. package/dist/modules/inbox_ops/components/proposals/EditActionDialog.js +9 -4
  256. package/dist/modules/inbox_ops/components/proposals/EditActionDialog.js.map +2 -2
  257. package/dist/modules/integrations/backend/integrations/bundle/[id]/page.js +18 -11
  258. package/dist/modules/integrations/backend/integrations/bundle/[id]/page.js.map +2 -2
  259. package/dist/modules/integrations/backend/integrations/page.js +12 -8
  260. package/dist/modules/integrations/backend/integrations/page.js.map +2 -2
  261. package/dist/modules/messages/commands/messages.js +13 -10
  262. package/dist/modules/messages/commands/messages.js.map +2 -2
  263. package/dist/modules/perspectives/api/[tableId]/route.js +39 -30
  264. package/dist/modules/perspectives/api/[tableId]/route.js.map +2 -2
  265. package/dist/modules/perspectives/services/perspectiveService.js +7 -0
  266. package/dist/modules/perspectives/services/perspectiveService.js.map +2 -2
  267. package/dist/modules/planner/backend/planner/availability-rulesets/[id]/page.js +6 -14
  268. package/dist/modules/planner/backend/planner/availability-rulesets/[id]/page.js.map +3 -3
  269. package/dist/modules/planner/backend/planner/availability-rulesets/page.js +4 -2
  270. package/dist/modules/planner/backend/planner/availability-rulesets/page.js.map +2 -2
  271. package/dist/modules/planner/components/AvailabilityRuleSetForm.js +2 -0
  272. package/dist/modules/planner/components/AvailabilityRuleSetForm.js.map +2 -2
  273. package/dist/modules/planner/components/AvailabilityRulesEditor.js +36 -11
  274. package/dist/modules/planner/components/AvailabilityRulesEditor.js.map +2 -2
  275. package/dist/modules/planner/components/AvailabilitySchedule.js +9 -5
  276. package/dist/modules/planner/components/AvailabilitySchedule.js.map +2 -2
  277. package/dist/modules/query_index/lib/engine.js +19 -0
  278. package/dist/modules/query_index/lib/engine.js.map +2 -2
  279. package/dist/modules/resources/backend/resources/resource-types/[id]/edit/page.js +1 -0
  280. package/dist/modules/resources/backend/resources/resource-types/[id]/edit/page.js.map +2 -2
  281. package/dist/modules/resources/backend/resources/resource-types/page.js +4 -2
  282. package/dist/modules/resources/backend/resources/resource-types/page.js.map +2 -2
  283. package/dist/modules/resources/backend/resources/resources/[id]/page.js +14 -3
  284. package/dist/modules/resources/backend/resources/resources/[id]/page.js.map +2 -2
  285. package/dist/modules/resources/backend/resources/resources/page.js +8 -4
  286. package/dist/modules/resources/backend/resources/resources/page.js.map +2 -2
  287. package/dist/modules/resources/components/ResourceCrudForm.js +2 -0
  288. package/dist/modules/resources/components/ResourceCrudForm.js.map +2 -2
  289. package/dist/modules/resources/components/ResourceTypeCrudForm.js +1 -0
  290. package/dist/modules/resources/components/ResourceTypeCrudForm.js.map +2 -2
  291. package/dist/modules/sales/api/documents/factory.js +7 -2
  292. package/dist/modules/sales/api/documents/factory.js.map +2 -2
  293. package/dist/modules/sales/backend/sales/channels/[channelId]/edit/page.js +3 -1
  294. package/dist/modules/sales/backend/sales/channels/[channelId]/edit/page.js.map +2 -2
  295. package/dist/modules/sales/backend/sales/channels/offers/page.js +13 -4
  296. package/dist/modules/sales/backend/sales/channels/offers/page.js.map +2 -2
  297. package/dist/modules/sales/backend/sales/channels/page.js +16 -4
  298. package/dist/modules/sales/backend/sales/channels/page.js.map +2 -2
  299. package/dist/modules/sales/backend/sales/documents/[id]/page.js +68 -22
  300. package/dist/modules/sales/backend/sales/documents/[id]/page.js.map +2 -2
  301. package/dist/modules/sales/backend/sales/documents/create/page.js.map +2 -2
  302. package/dist/modules/sales/commands/documentAddresses.js +181 -2
  303. package/dist/modules/sales/commands/documentAddresses.js.map +2 -2
  304. package/dist/modules/sales/commands/documents.js +29 -1
  305. package/dist/modules/sales/commands/documents.js.map +2 -2
  306. package/dist/modules/sales/commands/returns.js +12 -2
  307. package/dist/modules/sales/commands/returns.js.map +2 -2
  308. package/dist/modules/sales/commands/shared.js +15 -0
  309. package/dist/modules/sales/commands/shared.js.map +2 -2
  310. package/dist/modules/sales/commands/shipments.js +4 -1
  311. package/dist/modules/sales/commands/shipments.js.map +2 -2
  312. package/dist/modules/sales/components/AdjustmentKindSettings.js +19 -11
  313. package/dist/modules/sales/components/AdjustmentKindSettings.js.map +2 -2
  314. package/dist/modules/sales/components/DocumentNumberSettings.js.map +2 -2
  315. package/dist/modules/sales/components/OrderEditingSettings.js.map +2 -2
  316. package/dist/modules/sales/components/PaymentMethodsSettings.js +12 -4
  317. package/dist/modules/sales/components/PaymentMethodsSettings.js.map +2 -2
  318. package/dist/modules/sales/components/ShippingMethodsSettings.js +12 -4
  319. package/dist/modules/sales/components/ShippingMethodsSettings.js.map +2 -2
  320. package/dist/modules/sales/components/StatusSettings.js +18 -11
  321. package/dist/modules/sales/components/StatusSettings.js.map +2 -2
  322. package/dist/modules/sales/components/TaxRatesSettings.js +12 -4
  323. package/dist/modules/sales/components/TaxRatesSettings.js.map +2 -2
  324. package/dist/modules/sales/components/channels/ChannelOfferForm.js +47 -16
  325. package/dist/modules/sales/components/channels/ChannelOfferForm.js.map +2 -2
  326. package/dist/modules/sales/components/channels/SalesChannelOffersPanel.js +8 -4
  327. package/dist/modules/sales/components/channels/SalesChannelOffersPanel.js.map +2 -2
  328. package/dist/modules/sales/components/documents/AddressesSection.js +44 -25
  329. package/dist/modules/sales/components/documents/AddressesSection.js.map +2 -2
  330. package/dist/modules/sales/components/documents/AdjustmentsSection.js +43 -23
  331. package/dist/modules/sales/components/documents/AdjustmentsSection.js.map +2 -2
  332. package/dist/modules/sales/components/documents/ItemsSection.js +22 -13
  333. package/dist/modules/sales/components/documents/ItemsSection.js.map +2 -2
  334. package/dist/modules/sales/components/documents/LineItemDialog.js +23 -10
  335. package/dist/modules/sales/components/documents/LineItemDialog.js.map +2 -2
  336. package/dist/modules/sales/components/documents/PaymentDialog.js +29 -14
  337. package/dist/modules/sales/components/documents/PaymentDialog.js.map +2 -2
  338. package/dist/modules/sales/components/documents/PaymentsSection.js +20 -10
  339. package/dist/modules/sales/components/documents/PaymentsSection.js.map +2 -2
  340. package/dist/modules/sales/components/documents/ReturnDialog.js +26 -17
  341. package/dist/modules/sales/components/documents/ReturnDialog.js.map +2 -2
  342. package/dist/modules/sales/components/documents/ReturnsSection.js +3 -1
  343. package/dist/modules/sales/components/documents/ReturnsSection.js.map +2 -2
  344. package/dist/modules/sales/components/documents/SalesDocumentsTable.js +10 -5
  345. package/dist/modules/sales/components/documents/SalesDocumentsTable.js.map +2 -2
  346. package/dist/modules/sales/components/documents/ShipmentDialog.js +21 -7
  347. package/dist/modules/sales/components/documents/ShipmentDialog.js.map +2 -2
  348. package/dist/modules/sales/components/documents/ShipmentsSection.js +19 -10
  349. package/dist/modules/sales/components/documents/ShipmentsSection.js.map +2 -2
  350. package/dist/modules/sales/components/documents/optimisticLock.js +27 -0
  351. package/dist/modules/sales/components/documents/optimisticLock.js.map +7 -0
  352. package/dist/modules/sales/di.js +18 -0
  353. package/dist/modules/sales/di.js.map +2 -2
  354. package/dist/modules/staff/api/job-histories.js +11 -2
  355. package/dist/modules/staff/api/job-histories.js.map +2 -2
  356. package/dist/modules/staff/api/timesheets/time-entries/route.js +11 -4
  357. package/dist/modules/staff/api/timesheets/time-entries/route.js.map +2 -2
  358. package/dist/modules/staff/backend/staff/leave-requests/[id]/page.js +13 -8
  359. package/dist/modules/staff/backend/staff/leave-requests/[id]/page.js.map +2 -2
  360. package/dist/modules/staff/backend/staff/my-leave-requests/[id]/page.js +2 -1
  361. package/dist/modules/staff/backend/staff/my-leave-requests/[id]/page.js.map +2 -2
  362. package/dist/modules/staff/backend/staff/team-members/[id]/page.js +7 -4
  363. package/dist/modules/staff/backend/staff/team-members/[id]/page.js.map +2 -2
  364. package/dist/modules/staff/backend/staff/team-members/page.js +4 -2
  365. package/dist/modules/staff/backend/staff/team-members/page.js.map +2 -2
  366. package/dist/modules/staff/backend/staff/team-roles/[id]/edit/page.js +1 -0
  367. package/dist/modules/staff/backend/staff/team-roles/[id]/edit/page.js.map +2 -2
  368. package/dist/modules/staff/backend/staff/team-roles/page.js +4 -2
  369. package/dist/modules/staff/backend/staff/team-roles/page.js.map +2 -2
  370. package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js +5 -2
  371. package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js.map +2 -2
  372. package/dist/modules/staff/backend/staff/teams/page.js +12 -3
  373. package/dist/modules/staff/backend/staff/teams/page.js.map +2 -2
  374. package/dist/modules/staff/backend/staff/timesheets/page.js +4 -1
  375. package/dist/modules/staff/backend/staff/timesheets/page.js.map +2 -2
  376. package/dist/modules/staff/backend/staff/timesheets/projects/[id]/page.js.map +2 -2
  377. package/dist/modules/staff/backend/staff/timesheets/projects/page.js +12 -3
  378. package/dist/modules/staff/backend/staff/timesheets/projects/page.js.map +2 -2
  379. package/dist/modules/staff/commands/job-histories.js +40 -3
  380. package/dist/modules/staff/commands/job-histories.js.map +2 -2
  381. package/dist/modules/staff/components/LeaveRequestForm.js +1 -0
  382. package/dist/modules/staff/components/LeaveRequestForm.js.map +2 -2
  383. package/dist/modules/staff/components/TeamForm.js +1 -0
  384. package/dist/modules/staff/components/TeamForm.js.map +2 -2
  385. package/dist/modules/staff/components/TeamMemberForm.js +1 -0
  386. package/dist/modules/staff/components/TeamMemberForm.js.map +2 -2
  387. package/dist/modules/staff/components/TeamRoleForm.js +1 -0
  388. package/dist/modules/staff/components/TeamRoleForm.js.map +2 -2
  389. package/dist/modules/staff/components/detail/JobHistorySection.js +20 -7
  390. package/dist/modules/staff/components/detail/JobHistorySection.js.map +2 -2
  391. package/dist/modules/staff/data/validators.js +7 -1
  392. package/dist/modules/staff/data/validators.js.map +2 -2
  393. package/dist/modules/staff/lib/leaveRequestHelpers.js +2 -1
  394. package/dist/modules/staff/lib/leaveRequestHelpers.js.map +2 -2
  395. package/dist/modules/translations/components/TranslationManager.js +12 -8
  396. package/dist/modules/translations/components/TranslationManager.js.map +2 -2
  397. package/dist/modules/workflows/api/definitions/[id]/route.js +106 -0
  398. package/dist/modules/workflows/api/definitions/[id]/route.js.map +2 -2
  399. package/dist/modules/workflows/backend/definitions/[id]/page.js +11 -3
  400. package/dist/modules/workflows/backend/definitions/[id]/page.js.map +2 -2
  401. package/dist/modules/workflows/backend/definitions/page.js +19 -8
  402. package/dist/modules/workflows/backend/definitions/page.js.map +2 -2
  403. package/dist/modules/workflows/backend/definitions/visual-editor/page.js +29 -16
  404. package/dist/modules/workflows/backend/definitions/visual-editor/page.js.map +2 -2
  405. package/dist/modules/workflows/components/formConfig.js +4 -1
  406. package/dist/modules/workflows/components/formConfig.js.map +2 -2
  407. package/dist/modules/workflows/di.js +12 -0
  408. package/dist/modules/workflows/di.js.map +2 -2
  409. package/generated/entities/role/index.ts +1 -0
  410. package/generated/entities/user/index.ts +1 -0
  411. package/generated/entity-fields-registry.ts +2 -0
  412. package/jest.setup.ts +17 -0
  413. package/package.json +8 -7
  414. package/src/helpers/integration/communicationChannelsFixtures.ts +6 -0
  415. package/src/helpers/integration/dbFixtures.ts +1 -1
  416. package/src/helpers/integration/optimisticLockUi.ts +172 -0
  417. package/src/helpers/integration/salesFixtures.ts +29 -0
  418. package/src/modules/api_keys/backend/api-keys/page.tsx +10 -5
  419. package/src/modules/attachments/components/AttachmentPartitionSettings.tsx +19 -9
  420. package/src/modules/auth/api/roles/acl/route.ts +37 -11
  421. package/src/modules/auth/api/roles/route.ts +2 -0
  422. package/src/modules/auth/api/sidebar/preferences/route.ts +73 -0
  423. package/src/modules/auth/api/users/acl/route.ts +46 -18
  424. package/src/modules/auth/api/users/route.ts +2 -0
  425. package/src/modules/auth/backend/roles/[id]/edit/page.tsx +29 -4
  426. package/src/modules/auth/backend/roles/page.tsx +9 -4
  427. package/src/modules/auth/backend/users/[id]/edit/page.tsx +37 -4
  428. package/src/modules/auth/backend/users/page.tsx +7 -2
  429. package/src/modules/auth/components/AclEditor.tsx +10 -1
  430. package/src/modules/auth/data/entities.ts +7 -1
  431. package/src/modules/auth/services/sidebarPreferencesService.ts +38 -4
  432. package/src/modules/business_rules/api/rules/route.ts +30 -0
  433. package/src/modules/business_rules/api/sets/route.ts +30 -0
  434. package/src/modules/business_rules/backend/rules/[id]/page.tsx +16 -4
  435. package/src/modules/business_rules/backend/rules/page.tsx +20 -11
  436. package/src/modules/business_rules/backend/sets/[id]/page.tsx +16 -4
  437. package/src/modules/business_rules/backend/sets/page.tsx +20 -11
  438. package/src/modules/catalog/api/categories/route.ts +3 -0
  439. package/src/modules/catalog/api/products/route.ts +4 -0
  440. package/src/modules/catalog/backend/catalog/categories/[id]/edit/page.tsx +5 -0
  441. package/src/modules/catalog/backend/catalog/products/[id]/page.tsx +112 -35
  442. package/src/modules/catalog/backend/catalog/products/[productId]/variants/[variantId]/page.tsx +56 -7
  443. package/src/modules/catalog/backend/catalog/products/optionSchemaClient.ts +2 -0
  444. package/src/modules/catalog/commands/variants.ts +32 -32
  445. package/src/modules/catalog/components/PriceKindSettings.tsx +20 -7
  446. package/src/modules/catalog/components/categories/CategoriesDataTable.tsx +1 -0
  447. package/src/modules/catalog/components/products/ProductMediaManager.tsx +2 -0
  448. package/src/modules/catalog/components/products/ProductsDataTable.tsx +8 -4
  449. package/src/modules/catalog/components/products/productForm.ts +3 -0
  450. package/src/modules/catalog/components/products/variantForm.ts +9 -0
  451. package/src/modules/communication_channels/api/post/test-seed/route.ts +28 -1
  452. package/src/modules/communication_channels/backend/profile/communication-channels/page.tsx +5 -0
  453. package/src/modules/communication_channels/commands/set-primary-channel.ts +10 -7
  454. package/src/modules/currencies/backend/currencies/[id]/page.tsx +13 -6
  455. package/src/modules/currencies/backend/currencies/page.tsx +18 -11
  456. package/src/modules/currencies/backend/exchange-rates/[id]/page.tsx +3 -0
  457. package/src/modules/currencies/backend/exchange-rates/page.tsx +10 -6
  458. package/src/modules/currencies/commands/currencies.ts +10 -5
  459. package/src/modules/currencies/components/CurrencyFetchingConfig.tsx +31 -21
  460. package/src/modules/customer_accounts/api/admin/roles/[id].ts +35 -5
  461. package/src/modules/customer_accounts/api/admin/roles.ts +2 -0
  462. package/src/modules/customer_accounts/api/admin/users/[id].ts +38 -5
  463. package/src/modules/customer_accounts/api/admin/users.ts +2 -0
  464. package/src/modules/customer_accounts/backend/customer_accounts/roles/[id]/page.tsx +34 -20
  465. package/src/modules/customer_accounts/backend/customer_accounts/roles/page.tsx +9 -4
  466. package/src/modules/customer_accounts/backend/customer_accounts/settings/domain/page.tsx +11 -4
  467. package/src/modules/customer_accounts/backend/customer_accounts/users/[id]/page.tsx +28 -17
  468. package/src/modules/customer_accounts/backend/customer_accounts/users/page.tsx +19 -11
  469. package/src/modules/customers/AGENTS.md +2 -2
  470. package/src/modules/customers/api/companies/route.ts +14 -1
  471. package/src/modules/customers/api/deals/route.ts +3 -0
  472. package/src/modules/customers/api/people/route.ts +12 -1
  473. package/src/modules/customers/api/todos/route.ts +1 -0
  474. package/src/modules/customers/backend/config/customers/deals/page.tsx +1 -0
  475. package/src/modules/customers/backend/config/customers/pipeline-stages/page.tsx +36 -21
  476. package/src/modules/customers/backend/customers/companies/[id]/page.tsx +52 -27
  477. package/src/modules/customers/backend/customers/companies/page.tsx +2 -0
  478. package/src/modules/customers/backend/customers/companies-v2/[id]/page.tsx +27 -5
  479. package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealFormHandlers.ts +39 -7
  480. package/src/modules/customers/backend/customers/deals/[id]/page.tsx +1 -0
  481. package/src/modules/customers/backend/customers/deals/page.tsx +18 -6
  482. package/src/modules/customers/backend/customers/deals/pipeline/page.tsx +64 -39
  483. package/src/modules/customers/backend/customers/people/[id]/page.tsx +46 -26
  484. package/src/modules/customers/backend/customers/people/page.tsx +2 -0
  485. package/src/modules/customers/backend/customers/people-v2/[id]/page.tsx +84 -24
  486. package/src/modules/customers/commands/addresses.ts +16 -14
  487. package/src/modules/customers/commands/companies.ts +3 -1
  488. package/src/modules/customers/commands/interactions.ts +50 -4
  489. package/src/modules/customers/commands/people.ts +2 -1
  490. package/src/modules/customers/commands/personCompanyLinks.ts +8 -5
  491. package/src/modules/customers/commands/pipeline-stages.ts +16 -16
  492. package/src/modules/customers/components/AddressFormatSettings.tsx +1 -0
  493. package/src/modules/customers/components/DictionarySettings.tsx +18 -13
  494. package/src/modules/customers/components/DictionarySortSettings.tsx +4 -0
  495. package/src/modules/customers/components/PipelineSettings.tsx +42 -21
  496. package/src/modules/customers/components/detail/ActivityTimeline.tsx +3 -3
  497. package/src/modules/customers/components/detail/AddressesSection.tsx +4 -0
  498. package/src/modules/customers/components/detail/CompanyPeopleSection.tsx +2 -0
  499. package/src/modules/customers/components/detail/DealsSection.tsx +4 -0
  500. package/src/modules/customers/components/detail/EmailCardActions.tsx +5 -0
  501. package/src/modules/customers/components/detail/EntityTagsDialog.tsx +7 -0
  502. package/src/modules/customers/components/detail/ManageTagsDialog.tsx +4 -0
  503. package/src/modules/customers/components/detail/PersonCompaniesSection.tsx +4 -0
  504. package/src/modules/customers/components/detail/RoleAssignmentRow.tsx +2 -0
  505. package/src/modules/customers/components/detail/ScheduleActivityDialog.tsx +23 -7
  506. package/src/modules/customers/components/detail/hooks/useInteractionMutations.ts +25 -15
  507. package/src/modules/customers/components/detail/hooks/useInteractions.ts +76 -35
  508. package/src/modules/customers/components/detail/hooks/usePersonTasks.ts +30 -17
  509. package/src/modules/customers/components/detail/schedule/useScheduleFormState.ts +2 -0
  510. package/src/modules/customers/components/detail/types.ts +1 -0
  511. package/src/modules/customers/components/formConfig.tsx +2 -0
  512. package/src/modules/customers/data/guards.ts +67 -0
  513. package/src/modules/customers/di.ts +66 -0
  514. package/src/modules/customers/i18n/de.json +2 -0
  515. package/src/modules/customers/i18n/en.json +2 -0
  516. package/src/modules/customers/i18n/es.json +2 -0
  517. package/src/modules/customers/i18n/pl.json +2 -0
  518. package/src/modules/customers/lib/todoCompatibility.ts +14 -0
  519. package/src/modules/dashboards/components/WidgetVisibilityEditor.tsx +2 -0
  520. package/src/modules/data_sync/api/options.ts +7 -4
  521. package/src/modules/data_sync/api/schedules/route.ts +9 -1
  522. package/src/modules/data_sync/backend/data-sync/page.tsx +18 -5
  523. package/src/modules/data_sync/components/IntegrationScheduleTab.tsx +46 -19
  524. package/src/modules/data_sync/lib/sync-schedule-service.ts +11 -0
  525. package/src/modules/dictionaries/api/[dictionaryId]/entries/[entryId]/route.ts +8 -1
  526. package/src/modules/dictionaries/api/[dictionaryId]/route.ts +23 -0
  527. package/src/modules/dictionaries/components/DictionariesManager.tsx +32 -9
  528. package/src/modules/dictionaries/components/DictionaryEntriesEditor.tsx +30 -14
  529. package/src/modules/dictionaries/i18n/de.json +1 -0
  530. package/src/modules/dictionaries/i18n/en.json +1 -0
  531. package/src/modules/dictionaries/i18n/es.json +1 -0
  532. package/src/modules/dictionaries/i18n/pl.json +1 -0
  533. package/src/modules/directory/api/organizations/route.ts +3 -0
  534. package/src/modules/directory/backend/directory/organizations/[id]/edit/page.tsx +8 -0
  535. package/src/modules/directory/backend/directory/organizations/page.tsx +10 -5
  536. package/src/modules/directory/backend/directory/tenants/[id]/edit/page.tsx +16 -5
  537. package/src/modules/directory/backend/directory/tenants/page.tsx +8 -4
  538. package/src/modules/directory/commands/organizations.ts +7 -4
  539. package/src/modules/entities/api/records.ts +99 -0
  540. package/src/modules/entities/backend/entities/user/[entityId]/records/[recordId]/page.tsx +7 -0
  541. package/src/modules/entities/backend/entities/user/[entityId]/records/page.tsx +8 -4
  542. package/src/modules/entities/lib/helpers.ts +17 -0
  543. package/src/modules/feature_toggles/api/global/[id]/override/route.ts +1 -0
  544. package/src/modules/feature_toggles/api/overrides/route.ts +19 -0
  545. package/src/modules/feature_toggles/backend/feature-toggles/global/[id]/edit/page.tsx +19 -13
  546. package/src/modules/feature_toggles/components/FeatureToggleOverrideCard.tsx +22 -12
  547. package/src/modules/feature_toggles/components/FeatureTogglesTable.tsx +7 -2
  548. package/src/modules/feature_toggles/components/formConfig.tsx +2 -1
  549. package/src/modules/feature_toggles/components/overrideFormConfig.tsx +10 -1
  550. package/src/modules/feature_toggles/data/validators.ts +11 -3
  551. package/src/modules/inbox_ops/api/settings/route.ts +18 -0
  552. package/src/modules/inbox_ops/backend/inbox-ops/settings/page.tsx +15 -10
  553. package/src/modules/inbox_ops/components/proposals/EditActionDialog.tsx +9 -4
  554. package/src/modules/integrations/backend/integrations/bundle/[id]/page.tsx +20 -11
  555. package/src/modules/integrations/backend/integrations/page.tsx +13 -8
  556. package/src/modules/messages/commands/messages.ts +27 -15
  557. package/src/modules/perspectives/api/[tableId]/route.ts +11 -2
  558. package/src/modules/perspectives/services/perspectiveService.ts +13 -1
  559. package/src/modules/planner/backend/planner/availability-rulesets/[id]/page.tsx +16 -14
  560. package/src/modules/planner/backend/planner/availability-rulesets/page.tsx +6 -3
  561. package/src/modules/planner/components/AvailabilityRuleSetForm.tsx +3 -0
  562. package/src/modules/planner/components/AvailabilityRulesEditor.tsx +58 -15
  563. package/src/modules/planner/components/AvailabilitySchedule.tsx +22 -7
  564. package/src/modules/query_index/lib/engine.ts +34 -0
  565. package/src/modules/resources/backend/resources/resource-types/[id]/edit/page.tsx +7 -1
  566. package/src/modules/resources/backend/resources/resource-types/page.tsx +6 -3
  567. package/src/modules/resources/backend/resources/resources/[id]/page.tsx +23 -3
  568. package/src/modules/resources/backend/resources/resources/page.tsx +15 -4
  569. package/src/modules/resources/components/ResourceCrudForm.tsx +3 -0
  570. package/src/modules/resources/components/ResourceTypeCrudForm.tsx +2 -0
  571. package/src/modules/sales/api/documents/factory.ts +13 -1
  572. package/src/modules/sales/backend/sales/channels/[channelId]/edit/page.tsx +6 -0
  573. package/src/modules/sales/backend/sales/channels/offers/page.tsx +10 -4
  574. package/src/modules/sales/backend/sales/channels/page.tsx +19 -4
  575. package/src/modules/sales/backend/sales/documents/[id]/page.tsx +73 -20
  576. package/src/modules/sales/backend/sales/documents/create/page.tsx +2 -0
  577. package/src/modules/sales/commands/documentAddresses.ts +226 -4
  578. package/src/modules/sales/commands/documents.ts +28 -0
  579. package/src/modules/sales/commands/returns.ts +12 -3
  580. package/src/modules/sales/commands/shared.ts +36 -0
  581. package/src/modules/sales/commands/shipments.ts +17 -1
  582. package/src/modules/sales/components/AdjustmentKindSettings.tsx +20 -11
  583. package/src/modules/sales/components/DocumentNumberSettings.tsx +1 -0
  584. package/src/modules/sales/components/OrderEditingSettings.tsx +1 -0
  585. package/src/modules/sales/components/PaymentMethodsSettings.tsx +12 -4
  586. package/src/modules/sales/components/ShippingMethodsSettings.tsx +12 -4
  587. package/src/modules/sales/components/StatusSettings.tsx +20 -11
  588. package/src/modules/sales/components/TaxRatesSettings.tsx +12 -5
  589. package/src/modules/sales/components/channels/ChannelOfferForm.tsx +67 -14
  590. package/src/modules/sales/components/channels/SalesChannelOffersPanel.tsx +7 -4
  591. package/src/modules/sales/components/documents/AddressesSection.tsx +35 -25
  592. package/src/modules/sales/components/documents/AdjustmentsSection.tsx +50 -25
  593. package/src/modules/sales/components/documents/ItemsSection.tsx +24 -13
  594. package/src/modules/sales/components/documents/LineItemDialog.tsx +26 -9
  595. package/src/modules/sales/components/documents/PaymentDialog.tsx +33 -14
  596. package/src/modules/sales/components/documents/PaymentsSection.tsx +22 -10
  597. package/src/modules/sales/components/documents/ReturnDialog.tsx +28 -17
  598. package/src/modules/sales/components/documents/ReturnsSection.tsx +4 -1
  599. package/src/modules/sales/components/documents/SalesDocumentsTable.tsx +11 -4
  600. package/src/modules/sales/components/documents/ShipmentDialog.tsx +23 -8
  601. package/src/modules/sales/components/documents/ShipmentsSection.tsx +20 -10
  602. package/src/modules/sales/components/documents/optimisticLock.ts +34 -0
  603. package/src/modules/sales/components/documents/shipmentTypes.ts +1 -0
  604. package/src/modules/sales/di.ts +35 -0
  605. package/src/modules/sales/i18n/de.json +3 -0
  606. package/src/modules/sales/i18n/en.json +3 -0
  607. package/src/modules/sales/i18n/es.json +3 -0
  608. package/src/modules/sales/i18n/pl.json +3 -0
  609. package/src/modules/staff/api/job-histories.ts +12 -2
  610. package/src/modules/staff/api/timesheets/time-entries/route.ts +16 -4
  611. package/src/modules/staff/backend/staff/leave-requests/[id]/page.tsx +12 -7
  612. package/src/modules/staff/backend/staff/my-leave-requests/[id]/page.tsx +2 -0
  613. package/src/modules/staff/backend/staff/team-members/[id]/page.tsx +16 -5
  614. package/src/modules/staff/backend/staff/team-members/page.tsx +6 -2
  615. package/src/modules/staff/backend/staff/team-roles/[id]/edit/page.tsx +8 -0
  616. package/src/modules/staff/backend/staff/team-roles/page.tsx +6 -2
  617. package/src/modules/staff/backend/staff/teams/[id]/edit/page.tsx +13 -3
  618. package/src/modules/staff/backend/staff/teams/page.tsx +9 -3
  619. package/src/modules/staff/backend/staff/timesheets/page.tsx +10 -1
  620. package/src/modules/staff/backend/staff/timesheets/projects/[id]/page.tsx +4 -0
  621. package/src/modules/staff/backend/staff/timesheets/projects/page.tsx +9 -3
  622. package/src/modules/staff/commands/job-histories.ts +42 -3
  623. package/src/modules/staff/components/LeaveRequestForm.tsx +2 -0
  624. package/src/modules/staff/components/TeamForm.tsx +2 -0
  625. package/src/modules/staff/components/TeamMemberForm.tsx +2 -0
  626. package/src/modules/staff/components/TeamRoleForm.tsx +2 -0
  627. package/src/modules/staff/components/detail/JobHistorySection.tsx +28 -6
  628. package/src/modules/staff/data/validators.ts +6 -0
  629. package/src/modules/staff/i18n/de.json +1 -0
  630. package/src/modules/staff/i18n/en.json +1 -0
  631. package/src/modules/staff/i18n/es.json +1 -0
  632. package/src/modules/staff/i18n/pl.json +1 -0
  633. package/src/modules/staff/lib/leaveRequestHelpers.ts +4 -0
  634. package/src/modules/translations/components/TranslationManager.tsx +13 -8
  635. package/src/modules/workflows/api/definitions/[id]/route.ts +112 -0
  636. package/src/modules/workflows/backend/definitions/[id]/page.tsx +20 -4
  637. package/src/modules/workflows/backend/definitions/page.tsx +20 -9
  638. package/src/modules/workflows/backend/definitions/visual-editor/page.tsx +29 -16
  639. package/src/modules/workflows/components/formConfig.tsx +5 -0
  640. package/src/modules/workflows/di.ts +20 -0
  641. package/src/modules/workflows/i18n/de.json +1 -0
  642. package/src/modules/workflows/i18n/en.json +1 -0
  643. package/src/modules/workflows/i18n/es.json +1 -0
  644. package/src/modules/workflows/i18n/pl.json +1 -0
@@ -26,6 +26,7 @@ type ReturnRow = {
26
26
  type SalesReturnsSectionProps = {
27
27
  orderId: string
28
28
  currencyCode?: string | null
29
+ documentUpdatedAt?: string | null
29
30
  }
30
31
 
31
32
  function formatDisplayDate(value: string | null | undefined): string | null {
@@ -35,7 +36,7 @@ function formatDisplayDate(value: string | null | undefined): string | null {
35
36
  return new Intl.DateTimeFormat(undefined, { dateStyle: 'medium' }).format(date)
36
37
  }
37
38
 
38
- export function SalesReturnsSection({ orderId, currencyCode }: SalesReturnsSectionProps) {
39
+ export function SalesReturnsSection({ orderId, currencyCode, documentUpdatedAt }: SalesReturnsSectionProps) {
39
40
  const t = useT()
40
41
  const [returns, setReturns] = React.useState<ReturnRow[]>([])
41
42
  const [lines, setLines] = React.useState<ReturnOrderLine[]>([])
@@ -208,6 +209,7 @@ export function SalesReturnsSection({ orderId, currencyCode }: SalesReturnsSecti
208
209
  open={dialogOpen}
209
210
  orderId={orderId}
210
211
  lines={lines}
212
+ documentUpdatedAt={documentUpdatedAt}
211
213
  onClose={() => setDialogOpen(false)}
212
214
  onSaved={async () => {
213
215
  emitSalesDocumentTotalsRefresh({ documentId: orderId, kind: 'order' })
@@ -258,6 +260,7 @@ export function SalesReturnsSection({ orderId, currencyCode }: SalesReturnsSecti
258
260
  open={dialogOpen}
259
261
  orderId={orderId}
260
262
  lines={lines}
263
+ documentUpdatedAt={documentUpdatedAt}
261
264
  onClose={() => setDialogOpen(false)}
262
265
  onSaved={async () => {
263
266
  emitSalesDocumentTotalsRefresh({ documentId: orderId, kind: 'order' })
@@ -9,7 +9,8 @@ import { DataTable, type DataTableExportFormat, withDataTableNamespaces } from '
9
9
  import type { FilterDef, FilterValues } from '@open-mercato/ui/backend/FilterBar'
10
10
  import { RowActions } from '@open-mercato/ui/backend/RowActions'
11
11
  import { Button } from '@open-mercato/ui/primitives/button'
12
- import { apiCall } from '@open-mercato/ui/backend/utils/apiCall'
12
+ import { apiCall, withScopedApiRequestHeaders } from '@open-mercato/ui/backend/utils/apiCall'
13
+ import { buildOptimisticLockHeader } from '@open-mercato/ui/backend/utils/optimisticLock'
13
14
  import { buildCrudExportUrl, deleteCrud } from '@open-mercato/ui/backend/utils/crud'
14
15
  import { flash } from '@open-mercato/ui/backend/FlashMessages'
15
16
  import { useOrganizationScopeVersion } from '@open-mercato/shared/lib/frontend/useOrganizationScope'
@@ -76,6 +77,7 @@ type SalesDocumentRow = {
76
77
  totalGross?: number | null
77
78
  currency?: string | null
78
79
  date?: string | null
80
+ updatedAt?: string | null
79
81
  }
80
82
 
81
83
  const PAGE_SIZE = 20
@@ -464,6 +466,7 @@ export function SalesDocumentsTable({ kind }: { kind: SalesDocumentKind }) {
464
466
  totalGross,
465
467
  currency: doc.currencyCode ?? null,
466
468
  date,
469
+ updatedAt: doc.updatedAt ?? null,
467
470
  }, item)
468
471
  },
469
472
  [kind]
@@ -540,9 +543,13 @@ export function SalesDocumentsTable({ kind }: { kind: SalesDocumentKind }) {
540
543
  })
541
544
  if (!confirmed) return
542
545
  try {
543
- const result = await deleteCrud(`sales/${resource}`, row.id, {
544
- errorMessage: t('sales.documents.list.table.deleteError', 'Failed to delete document.'),
545
- })
546
+ const result = await withScopedApiRequestHeaders(
547
+ buildOptimisticLockHeader(row.updatedAt),
548
+ () =>
549
+ deleteCrud(`sales/${resource}`, row.id, {
550
+ errorMessage: t('sales.documents.list.table.deleteError', 'Failed to delete document.'),
551
+ }),
552
+ )
546
553
  if (result.ok) {
547
554
  flash(
548
555
  kind === 'order'
@@ -10,10 +10,12 @@ import { Label } from '@open-mercato/ui/primitives/label'
10
10
  import { Switch } from '@open-mercato/ui/primitives/switch'
11
11
  import { CrudForm, type CrudCustomFieldRenderProps, type CrudField, type CrudFormGroup } from '@open-mercato/ui/backend/CrudForm'
12
12
  import { LookupSelect, type LookupSelectItem } from '@open-mercato/ui/backend/inputs'
13
- import { apiCall } from '@open-mercato/ui/backend/utils/apiCall'
13
+ import { apiCall, withScopedApiRequestHeaders } from '@open-mercato/ui/backend/utils/apiCall'
14
+ import { buildOptimisticLockHeader } from '@open-mercato/ui/backend/utils/optimisticLock'
14
15
  import { createCrud, updateCrud } from '@open-mercato/ui/backend/utils/crud'
15
16
  import { collectCustomFieldValues } from '@open-mercato/ui/backend/utils/customFieldValues'
16
17
  import { createCrudFormError } from '@open-mercato/ui/backend/utils/serverErrors'
18
+ import { handleSectionMutationError, rowOptimisticVersion } from './optimisticLock'
17
19
  import { flash } from '@open-mercato/ui/backend/FlashMessages'
18
20
  import { cn } from '@open-mercato/shared/lib/utils'
19
21
  import { E } from '#generated/entities.ids.generated'
@@ -973,13 +975,25 @@ export function ShipmentDialog({
973
975
  }
974
976
 
975
977
  const action = shipment?.id ? updateCrud : createCrud
976
- const result = await action(
977
- 'sales/shipments',
978
- shipment?.id ? { id: shipment.id, ...payload } : payload,
979
- {
980
- errorMessage: t('sales.documents.shipments.errorSave', 'Failed to save shipment.'),
981
- },
982
- )
978
+ let result
979
+ try {
980
+ result = await withScopedApiRequestHeaders(
981
+ buildOptimisticLockHeader(shipment?.id ? rowOptimisticVersion(shipment) : undefined),
982
+ () =>
983
+ action(
984
+ 'sales/shipments',
985
+ shipment?.id ? { id: shipment.id, ...payload } : payload,
986
+ {
987
+ errorMessage: t('sales.documents.shipments.errorSave', 'Failed to save shipment.'),
988
+ },
989
+ ),
990
+ )
991
+ } catch (err) {
992
+ if (handleSectionMutationError(err, t, () => void onSaved())) {
993
+ return
994
+ }
995
+ throw err
996
+ }
983
997
  if (result.ok) {
984
998
  const shipmentId = ((result.result as any)?.id as string | undefined) ?? shipment?.id ?? null
985
999
  const shouldAddShippingAdjustment =
@@ -1101,6 +1115,7 @@ export function ShipmentDialog({
1101
1115
  orderId,
1102
1116
  organizationId,
1103
1117
  shipment?.id,
1118
+ shipment?.updatedAt,
1104
1119
  addressOptions,
1105
1120
  addressOptionsMap,
1106
1121
  shippingMethods,
@@ -6,7 +6,8 @@ import { Button } from '@open-mercato/ui/primitives/button'
6
6
  import { Badge } from '@open-mercato/ui/primitives/badge'
7
7
  import { ErrorMessage, LoadingMessage, TabEmptyState } from '@open-mercato/ui/backend/detail'
8
8
  import { flash } from '@open-mercato/ui/backend/FlashMessages'
9
- import { apiCall } from '@open-mercato/ui/backend/utils/apiCall'
9
+ import { apiCall, withScopedApiRequestHeaders } from '@open-mercato/ui/backend/utils/apiCall'
10
+ import { buildOptimisticLockHeader } from '@open-mercato/ui/backend/utils/optimisticLock'
10
11
  import { deleteCrud } from '@open-mercato/ui/backend/utils/crud'
11
12
  import { useOrganizationScopeDetail } from '@open-mercato/shared/lib/frontend/useOrganizationScope'
12
13
  import { useT } from '@open-mercato/shared/lib/i18n/context'
@@ -19,6 +20,7 @@ import type { SectionAction } from '@open-mercato/core/modules/customers/compone
19
20
  import { generateTempId } from '@open-mercato/core/modules/customers/lib/detailHelpers'
20
21
  import { formatAddressString, type AddressValue } from '@open-mercato/core/modules/customers/utils/addressFormat'
21
22
  import { ShipmentDialog } from './ShipmentDialog'
23
+ import { handleSectionMutationError, readRowUpdatedAt, rowOptimisticVersion } from './optimisticLock'
22
24
  import { extractCustomFieldValues } from './customFieldHelpers'
23
25
  import type { OrderLine, ShipmentRow, ShipmentItem } from './shipmentTypes'
24
26
 
@@ -267,6 +269,7 @@ export function SalesShipmentsSection({
267
269
  const customValues = extractCustomFieldValues(item as Record<string, unknown>)
268
270
  return {
269
271
  id,
272
+ updatedAt: readRowUpdatedAt(item),
270
273
  shipmentNumber:
271
274
  typeof (item as any).shipment_number === 'string'
272
275
  ? (item as any).shipment_number
@@ -397,20 +400,27 @@ export function SalesShipmentsSection({
397
400
  })
398
401
  if (!confirmed) return
399
402
  try {
400
- const result = await deleteCrud('sales/shipments', {
401
- body: {
402
- id: shipment.id,
403
- orderId,
404
- organizationId: resolvedOrganizationId,
405
- tenantId: resolvedTenantId,
406
- },
407
- errorMessage: t('sales.documents.shipments.errorDelete', 'Failed to delete shipment.'),
408
- })
403
+ const result = await withScopedApiRequestHeaders(
404
+ buildOptimisticLockHeader(rowOptimisticVersion(shipment)),
405
+ () =>
406
+ deleteCrud('sales/shipments', {
407
+ body: {
408
+ id: shipment.id,
409
+ orderId,
410
+ organizationId: resolvedOrganizationId,
411
+ tenantId: resolvedTenantId,
412
+ },
413
+ errorMessage: t('sales.documents.shipments.errorDelete', 'Failed to delete shipment.'),
414
+ })
415
+ )
409
416
  if (result.ok) {
410
417
  await loadShipments()
411
418
  emitSalesDocumentTotalsRefresh({ documentId: orderId, kind: 'order' })
412
419
  }
413
420
  } catch (err) {
421
+ if (handleSectionMutationError(err, t, () => void loadShipments())) {
422
+ return
423
+ }
414
424
  console.error('sales.shipments.delete', err)
415
425
  flash(t('sales.documents.shipments.errorDelete', 'Failed to delete shipment.'), 'error')
416
426
  }
@@ -0,0 +1,34 @@
1
+ import { surfaceRecordConflict } from '@open-mercato/ui/backend/conflicts'
2
+
3
+ type Translate = (key: string, fallback?: string) => string
4
+
5
+ export function rowOptimisticVersion(row: { updatedAt?: string | null } | null | undefined): string | undefined {
6
+ const value = row?.updatedAt
7
+ return typeof value === 'string' && value.length ? value : undefined
8
+ }
9
+
10
+ /**
11
+ * Read a row's `updated_at` / `updatedAt` version from an untyped API record
12
+ * without `any`. Returns `null` when neither key holds a non-empty string.
13
+ */
14
+ export function readRowUpdatedAt(source: unknown): string | null {
15
+ if (!source || typeof source !== 'object') return null
16
+ const record = source as Record<string, unknown>
17
+ const snake = record.updated_at
18
+ if (typeof snake === 'string' && snake.length) return snake
19
+ const camel = record.updatedAt
20
+ if (typeof camel === 'string' && camel.length) return camel
21
+ return null
22
+ }
23
+
24
+ export function handleSectionMutationError(
25
+ err: unknown,
26
+ t: Translate,
27
+ refresh: () => void,
28
+ ): boolean {
29
+ if (surfaceRecordConflict(err, t, { onRefresh: refresh })) {
30
+ refresh()
31
+ return true
32
+ }
33
+ return false
34
+ }
@@ -9,6 +9,7 @@ export type ShipmentItem = {
9
9
 
10
10
  export type ShipmentRow = {
11
11
  id: string
12
+ updatedAt: string | null
12
13
  shipmentNumber: string | null
13
14
  shippingMethodId: string | null
14
15
  shippingMethodCode: string | null
@@ -2,6 +2,8 @@ import { asFunction, asValue } from 'awilix'
2
2
  import type { EntityManager } from '@mikro-orm/postgresql'
3
3
  import type { EventBus } from '@open-mercato/events'
4
4
  import type { AppContainer } from '@open-mercato/shared/lib/di/container'
5
+ import type { OptimisticLockCurrentReader } from '@open-mercato/shared/lib/crud/optimistic-lock'
6
+ import { registerOptimisticLockReaders } from '@open-mercato/shared/lib/crud/optimistic-lock-store'
5
7
  import { DefaultSalesCalculationService } from './services/salesCalculationService'
6
8
  import { DefaultTaxCalculationService } from './services/taxCalculationService'
7
9
  import { SalesDocumentNumberGenerator } from './services/salesDocumentNumberGenerator'
@@ -38,6 +40,34 @@ type AppCradle = AppContainer['cradle'] & {
38
40
  eventBus?: EventBus | null
39
41
  }
40
42
 
43
+ const RESOURCE_KIND_ORDER = 'sales.order'
44
+
45
+ const readSalesOrderUpdatedAt: OptimisticLockCurrentReader = async (
46
+ em: EntityManager,
47
+ { resourceId, tenantId, organizationId },
48
+ ) => {
49
+ const row = await em.findOne(
50
+ SalesOrder,
51
+ {
52
+ id: resourceId,
53
+ tenantId,
54
+ ...(organizationId ? { organizationId } : {}),
55
+ deletedAt: null,
56
+ },
57
+ { fields: ['updatedAt'] as const },
58
+ )
59
+ return row?.updatedAt instanceof Date ? row.updatedAt.toISOString() : null
60
+ }
61
+
62
+ // Hand-wired sales.order reader registered at module-load time so the
63
+ // `customer_entities`-style polymorphic-table override pattern stays
64
+ // observable for downstream modules. Functionally identical to the
65
+ // auto-registered generic reader; kept here as a reference example. The
66
+ // guard's mode check short-circuits when `OM_OPTIMISTIC_LOCK=off`.
67
+ registerOptimisticLockReaders({
68
+ [RESOURCE_KIND_ORDER]: readSalesOrderUpdatedAt,
69
+ })
70
+
41
71
  export function register(container: AppContainer) {
42
72
  container.register({
43
73
  salesCalculationService: asFunction(({ eventBus }: AppCradle) => {
@@ -81,4 +111,9 @@ export function register(container: AppContainer) {
81
111
  SalesPaymentMethod: asValue(SalesPaymentMethod),
82
112
  SalesTaxRate: asValue(SalesTaxRate),
83
113
  })
114
+
115
+ // `crudMutationGuardService` is registered platform-wide in the shared
116
+ // DI bootstrap (`packages/shared/src/lib/di/container.ts`). The
117
+ // hand-wired sales.order reader above already lives in the global store,
118
+ // so this module no longer needs its own DI binding.
84
119
  }
@@ -15,6 +15,9 @@
15
15
  "sales.audit.delivery-windows.create": "Create delivery window",
16
16
  "sales.audit.delivery-windows.delete": "Delete delivery window",
17
17
  "sales.audit.delivery-windows.update": "Update delivery window",
18
+ "sales.audit.document_addresses.create": "Dokumentadresse hinzufügen",
19
+ "sales.audit.document_addresses.delete": "Dokumentadresse entfernen",
20
+ "sales.audit.document_addresses.update": "Dokumentadresse aktualisieren",
18
21
  "sales.audit.invoices.create": "Rechnung erstellen",
19
22
  "sales.audit.invoices.delete": "Rechnung löschen",
20
23
  "sales.audit.invoices.update": "Rechnung aktualisieren",
@@ -15,6 +15,9 @@
15
15
  "sales.audit.delivery-windows.create": "Create delivery window",
16
16
  "sales.audit.delivery-windows.delete": "Delete delivery window",
17
17
  "sales.audit.delivery-windows.update": "Update delivery window",
18
+ "sales.audit.document_addresses.create": "Add document address",
19
+ "sales.audit.document_addresses.delete": "Remove document address",
20
+ "sales.audit.document_addresses.update": "Update document address",
18
21
  "sales.audit.invoices.create": "Create invoice",
19
22
  "sales.audit.invoices.delete": "Delete invoice",
20
23
  "sales.audit.invoices.update": "Update invoice",
@@ -15,6 +15,9 @@
15
15
  "sales.audit.delivery-windows.create": "Create delivery window",
16
16
  "sales.audit.delivery-windows.delete": "Delete delivery window",
17
17
  "sales.audit.delivery-windows.update": "Update delivery window",
18
+ "sales.audit.document_addresses.create": "Agregar dirección del documento",
19
+ "sales.audit.document_addresses.delete": "Eliminar dirección del documento",
20
+ "sales.audit.document_addresses.update": "Actualizar dirección del documento",
18
21
  "sales.audit.invoices.create": "Crear factura",
19
22
  "sales.audit.invoices.delete": "Eliminar factura",
20
23
  "sales.audit.invoices.update": "Actualizar factura",
@@ -15,6 +15,9 @@
15
15
  "sales.audit.delivery-windows.create": "Create delivery window",
16
16
  "sales.audit.delivery-windows.delete": "Delete delivery window",
17
17
  "sales.audit.delivery-windows.update": "Update delivery window",
18
+ "sales.audit.document_addresses.create": "Dodaj adres dokumentu",
19
+ "sales.audit.document_addresses.delete": "Usuń adres dokumentu",
20
+ "sales.audit.document_addresses.update": "Zaktualizuj adres dokumentu",
18
21
  "sales.audit.invoices.create": "Utwórz fakturę",
19
22
  "sales.audit.invoices.delete": "Usuń fakturę",
20
23
  "sales.audit.invoices.update": "Zaktualizuj fakturę",
@@ -4,6 +4,7 @@ import { resolveTranslations } from '@open-mercato/shared/lib/i18n/server'
4
4
  import { resolveCrudRecordId, parseScopedCommandInput } from '@open-mercato/shared/lib/api/scoped'
5
5
  import { StaffTeamMemberJobHistory } from '../data/entities'
6
6
  import {
7
+ optimisticUpdatedAtSchema,
7
8
  staffTeamMemberJobHistoryCreateSchema,
8
9
  staffTeamMemberJobHistoryUpdateSchema,
9
10
  } from '../data/validators'
@@ -12,6 +13,11 @@ import { E } from '#generated/entities.ids.generated'
12
13
 
13
14
  const rawBodySchema = z.object({}).passthrough()
14
15
 
16
+ const optimisticDeleteSchema = z.object({
17
+ id: z.string().uuid(),
18
+ updatedAt: optimisticUpdatedAtSchema.optional(),
19
+ })
20
+
15
21
  const listSchema = z
16
22
  .object({
17
23
  page: z.coerce.number().min(1).default(1),
@@ -94,7 +100,11 @@ const crud = makeCrudRoute({
94
100
  mapInput: async ({ parsed, ctx }) => {
95
101
  const { translate } = await resolveTranslations()
96
102
  const id = resolveCrudRecordId(parsed, ctx, translate)
97
- return { id }
103
+ const parsedBody = (parsed as { body?: Record<string, unknown> })?.body
104
+ return {
105
+ id,
106
+ updatedAt: typeof parsedBody?.updatedAt === 'string' ? parsedBody.updatedAt : undefined,
107
+ }
98
108
  },
99
109
  response: () => ({ ok: true }),
100
110
  },
@@ -137,7 +147,7 @@ export const openApi = createStaffCrudOpenApi({
137
147
  description: 'Updates a team member job history entry.',
138
148
  },
139
149
  del: {
140
- schema: z.object({ id: z.string().uuid() }),
150
+ schema: optimisticDeleteSchema,
141
151
  responseSchema: defaultOkResponseSchema,
142
152
  description: 'Deletes a team member job history entry.',
143
153
  },
@@ -37,13 +37,25 @@ export const metadata = routeMetadata
37
37
 
38
38
  const rawBodySchema = z.object({}).passthrough()
39
39
 
40
+ const isParseableDateFilter = (value: string): boolean => {
41
+ const trimmed = value.trim()
42
+ if (trimmed.length === 0) return true
43
+ if (!/^\d{4}-\d{2}-\d{2}([T ].*)?$/.test(trimmed)) return false
44
+ return !Number.isNaN(new Date(trimmed).getTime())
45
+ }
46
+
47
+ const dateFilterSchema = z
48
+ .string()
49
+ .refine(isParseableDateFilter, { message: 'Invalid date' })
50
+ .optional()
51
+
40
52
  const listSchema = z
41
53
  .object({
42
54
  page: z.coerce.number().min(1).default(1),
43
55
  pageSize: z.coerce.number().min(1).max(100).default(50),
44
56
  staffMemberId: z.string().uuid().optional(),
45
- from: z.string().optional(),
46
- to: z.string().optional(),
57
+ from: dateFilterSchema,
58
+ to: dateFilterSchema,
47
59
  projectId: z.string().uuid().optional(),
48
60
  ids: z.string().optional(),
49
61
  sortField: z.string().optional(),
@@ -102,10 +114,10 @@ const crud = makeCrudRoute({
102
114
  if (typeof query.staffMemberId === 'string' && query.staffMemberId.length > 0) {
103
115
  filters[F.staff_member_id] = query.staffMemberId
104
116
  }
105
- if (typeof query.from === 'string' && query.from.length > 0) {
117
+ if (typeof query.from === 'string' && query.from.length > 0 && isParseableDateFilter(query.from)) {
106
118
  filters[F.date] = { ...((filters[F.date] as Record<string, unknown>) ?? {}), $gte: query.from }
107
119
  }
108
- if (typeof query.to === 'string' && query.to.length > 0) {
120
+ if (typeof query.to === 'string' && query.to.length > 0 && isParseableDateFilter(query.to)) {
109
121
  filters[F.date] = { ...((filters[F.date] as Record<string, unknown>) ?? {}), $lte: query.to }
110
122
  }
111
123
  if (typeof query.projectId === 'string' && query.projectId.length > 0) {
@@ -8,7 +8,8 @@ import { Button } from '@open-mercato/ui/primitives/button'
8
8
  import { Textarea } from '@open-mercato/ui/primitives/textarea'
9
9
  import { LoadingMessage, ErrorMessage, RecordNotFoundState } from '@open-mercato/ui/backend/detail'
10
10
  import { SendObjectMessageDialog } from '@open-mercato/ui/backend/messages'
11
- import { apiCallOrThrow, readApiResultOrThrow } from '@open-mercato/ui/backend/utils/apiCall'
11
+ import { apiCallOrThrow, readApiResultOrThrow, withScopedApiRequestHeaders } from '@open-mercato/ui/backend/utils/apiCall'
12
+ import { buildOptimisticLockHeader } from '@open-mercato/ui/backend/utils/optimisticLock'
12
13
  import { updateCrud } from '@open-mercato/ui/backend/utils/crud'
13
14
  import { flash } from '@open-mercato/ui/backend/FlashMessages'
14
15
  import { useT } from '@open-mercato/shared/lib/i18n/context'
@@ -89,6 +90,7 @@ export default function StaffLeaveRequestDetailPage({ params }: { params?: { id?
89
90
  unavailabilityReasonEntryId: record?.unavailabilityReasonEntryId ?? null,
90
91
  unavailabilityReasonValue: record?.unavailabilityReasonValue ?? null,
91
92
  note: record?.note ?? null,
93
+ updatedAt: record?.updatedAt ?? null,
92
94
  }), [record, memberLabel])
93
95
 
94
96
  const handleSubmit = React.useCallback(async (values: LeaveRequestFormValues) => {
@@ -104,11 +106,14 @@ const handleSubmit = React.useCallback(async (values: LeaveRequestFormValues) =>
104
106
  const handleDecision = React.useCallback(async (action: 'accept' | 'reject') => {
105
107
  if (!record?.id) return
106
108
  const endpoint = action === 'accept' ? '/api/staff/leave-requests/accept' : '/api/staff/leave-requests/reject'
107
- await apiCallOrThrow(endpoint, {
108
- method: 'POST',
109
- headers: { 'content-type': 'application/json' },
110
- body: JSON.stringify({ id: record.id, decisionComment: decisionComment || null }),
111
- })
109
+ await withScopedApiRequestHeaders(
110
+ buildOptimisticLockHeader(record.updatedAt),
111
+ () => apiCallOrThrow(endpoint, {
112
+ method: 'POST',
113
+ headers: { 'content-type': 'application/json' },
114
+ body: JSON.stringify({ id: record.id, decisionComment: decisionComment || null }),
115
+ }),
116
+ )
112
117
  flash(
113
118
  action === 'accept'
114
119
  ? t('staff.leaveRequests.messages.accepted', 'Leave request approved.')
@@ -116,7 +121,7 @@ const handleSubmit = React.useCallback(async (values: LeaveRequestFormValues) =>
116
121
  'success',
117
122
  )
118
123
  router.refresh()
119
- }, [decisionComment, record?.id, router, t])
124
+ }, [decisionComment, record?.id, record?.updatedAt, router, t])
120
125
 
121
126
  if (isLoading) {
122
127
  return (
@@ -74,6 +74,7 @@ export default function StaffMyLeaveRequestDetailPage({ params }: { params?: { i
74
74
 
75
75
  const status = record?.status ?? 'pending'
76
76
  const memberLabel = record?.member?.displayName ?? null
77
+ // optimistic-lock: LeaveRequestForm forwards optimisticLockUpdatedAt from initialValues.updatedAt (wrapper auto-derives the header on save).
77
78
  const initialValues = React.useMemo<LeaveRequestFormValues>(() => ({
78
79
  id: record?.id,
79
80
  memberId: record?.memberId ?? null,
@@ -84,6 +85,7 @@ export default function StaffMyLeaveRequestDetailPage({ params }: { params?: { i
84
85
  unavailabilityReasonEntryId: record?.unavailabilityReasonEntryId ?? null,
85
86
  unavailabilityReasonValue: record?.unavailabilityReasonValue ?? null,
86
87
  note: record?.note ?? null,
88
+ updatedAt: record?.updatedAt ?? null,
87
89
  }), [record, memberLabel])
88
90
  const dateSummary = formatDateRange(record?.startDate, record?.endDate)
89
91
  const handleSubmit = React.useCallback(async (values: LeaveRequestFormValues) => {
@@ -5,9 +5,10 @@ import Link from 'next/link'
5
5
  import { useRouter, useSearchParams } from 'next/navigation'
6
6
  import { Page, PageBody } from '@open-mercato/ui/backend/Page'
7
7
  import { Button } from '@open-mercato/ui/primitives/button'
8
- import { readApiResultOrThrow } from '@open-mercato/ui/backend/utils/apiCall'
8
+ import { readApiResultOrThrow, withScopedApiRequestHeaders } from '@open-mercato/ui/backend/utils/apiCall'
9
9
  import { extractCustomFieldEntries } from '@open-mercato/shared/lib/crud/custom-fields-client'
10
10
  import { updateCrud, deleteCrud } from '@open-mercato/ui/backend/utils/crud'
11
+ import { buildOptimisticLockHeader } from '@open-mercato/ui/backend/utils/optimisticLock'
11
12
  import { flash } from '@open-mercato/ui/backend/FlashMessages'
12
13
  import { useT } from '@open-mercato/shared/lib/i18n/context'
13
14
  import { createTranslatorWithFallback } from '@open-mercato/shared/lib/i18n/translate'
@@ -54,6 +55,8 @@ type TeamMemberRecord = {
54
55
  is_active?: boolean
55
56
  availabilityRuleSetId?: string | null
56
57
  availability_rule_set_id?: string | null
58
+ updatedAt?: string | null
59
+ updated_at?: string | null
57
60
  user?: { id?: string; email?: string | null } | null
58
61
  team?: { id?: string; name?: string | null } | null
59
62
  customFields?: Record<string, unknown> | null
@@ -226,6 +229,11 @@ export default function StaffTeamMemberDetailPage({ params }: { params?: { id?:
226
229
  roleIds: normalizedRoleIds,
227
230
  tags: normalizeStringList(record.tags),
228
231
  isActive: record.isActive ?? record.is_active ?? true,
232
+ updatedAt: typeof record.updatedAt === 'string'
233
+ ? record.updatedAt
234
+ : typeof record.updated_at === 'string'
235
+ ? record.updated_at
236
+ : null,
229
237
  ...customFields,
230
238
  })
231
239
  setMemberRecord(record)
@@ -289,12 +297,15 @@ export default function StaffTeamMemberDetailPage({ params }: { params?: { id?:
289
297
 
290
298
  const handleRulesetChange = React.useCallback(async (nextId: string | null) => {
291
299
  if (!memberId) return
292
- await updateCrud('staff/team-members', { id: memberId, availabilityRuleSetId: nextId }, {
293
- errorMessage: t('staff.teamMembers.availability.ruleset.updateError', 'Failed to update schedule.'),
294
- })
300
+ const headers = buildOptimisticLockHeader(initialValues?.updatedAt)
301
+ await withScopedApiRequestHeaders(headers, () => (
302
+ updateCrud('staff/team-members', { id: memberId, availabilityRuleSetId: nextId }, {
303
+ errorMessage: t('staff.teamMembers.availability.ruleset.updateError', 'Failed to update schedule.'),
304
+ })
305
+ ))
295
306
  setAvailabilityRuleSetId(nextId)
296
307
  flash(t('staff.teamMembers.availability.ruleset.updateSuccess', 'Schedule updated.'), 'success')
297
- }, [memberId, t])
308
+ }, [initialValues?.updatedAt, memberId, t])
298
309
 
299
310
  const panelTabs = React.useMemo(() => ([
300
311
  { id: 'details' as const, label: t('staff.teamMembers.detail.tabs.details', 'Details') },
@@ -9,8 +9,9 @@ import { DataTable, withDataTableNamespaces } from '@open-mercato/ui/backend/Dat
9
9
  import { RowActions } from '@open-mercato/ui/backend/RowActions'
10
10
  import { Button } from '@open-mercato/ui/primitives/button'
11
11
  import { BooleanIcon } from '@open-mercato/ui/backend/ValueIcons'
12
- import { readApiResultOrThrow, apiCall } from '@open-mercato/ui/backend/utils/apiCall'
12
+ import { readApiResultOrThrow, apiCall, withScopedApiRequestHeaders } from '@open-mercato/ui/backend/utils/apiCall'
13
13
  import { deleteCrud } from '@open-mercato/ui/backend/utils/crud'
14
+ import { buildOptimisticLockHeader } from '@open-mercato/ui/backend/utils/optimisticLock'
14
15
  import { flash } from '@open-mercato/ui/backend/FlashMessages'
15
16
  import { useConfirmDialog } from '@open-mercato/ui/backend/confirm-dialog'
16
17
  import type { FilterDef, FilterValues } from '@open-mercato/ui/backend/FilterBar'
@@ -391,7 +392,10 @@ export default function StaffTeamMembersPage() {
391
392
  })
392
393
  if (!confirmed) return
393
394
  try {
394
- await deleteCrud('staff/team-members', entry.id, { errorMessage: labels.errors.delete })
395
+ const headers = buildOptimisticLockHeader(entry.updatedAt)
396
+ await withScopedApiRequestHeaders(headers, () => (
397
+ deleteCrud('staff/team-members', entry.id, { errorMessage: labels.errors.delete })
398
+ ))
395
399
  flash(labels.messages.deleted, 'success')
396
400
  handleRefresh()
397
401
  } catch (error) {
@@ -23,6 +23,8 @@ type TeamRoleRecord = {
23
23
  appearanceColor?: string | null
24
24
  appearance_icon?: string | null
25
25
  appearance_color?: string | null
26
+ updatedAt?: string | null
27
+ updated_at?: string | null
26
28
  } & Record<string, unknown>
27
29
 
28
30
  type TeamRoleResponse = {
@@ -38,6 +40,7 @@ export default function StaffTeamRoleEditPage({ params }: { params?: { id?: stri
38
40
  const t = useT()
39
41
  const router = useRouter()
40
42
  const scopeVersion = useOrganizationScopeVersion()
43
+ // optimistic-lock: TeamRoleForm forwards optimisticLockUpdatedAt from initialValues.updatedAt (wrapper auto-derives the header on save + delete).
41
44
  const [initialValues, setInitialValues] = React.useState<TeamRoleFormValues | null>(null)
42
45
  const [error, setError] = React.useState<string | null>(null)
43
46
  const [isNotFound, setIsNotFound] = React.useState(false)
@@ -86,6 +89,11 @@ export default function StaffTeamRoleEditPage({ params }: { params?: { id?: stri
86
89
  name: record.name ?? '',
87
90
  description: record.description ?? '',
88
91
  appearance: { icon: appearanceIcon, color: appearanceColor },
92
+ updatedAt: typeof record.updatedAt === 'string'
93
+ ? record.updatedAt
94
+ : typeof record.updated_at === 'string'
95
+ ? record.updated_at
96
+ : null,
89
97
  ...customFields,
90
98
  })
91
99
  }
@@ -9,8 +9,9 @@ import { DataTable, withDataTableNamespaces } from '@open-mercato/ui/backend/Dat
9
9
  import { RowActions } from '@open-mercato/ui/backend/RowActions'
10
10
  import { MarkdownPreview } from '@open-mercato/ui/backend/markdown/MarkdownContent'
11
11
  import { Button } from '@open-mercato/ui/primitives/button'
12
- import { readApiResultOrThrow, apiCall } from '@open-mercato/ui/backend/utils/apiCall'
12
+ import { readApiResultOrThrow, apiCall, withScopedApiRequestHeaders } from '@open-mercato/ui/backend/utils/apiCall'
13
13
  import { deleteCrud } from '@open-mercato/ui/backend/utils/crud'
14
+ import { buildOptimisticLockHeader } from '@open-mercato/ui/backend/utils/optimisticLock'
14
15
  import { flash } from '@open-mercato/ui/backend/FlashMessages'
15
16
  import { useConfirmDialog } from '@open-mercato/ui/backend/confirm-dialog'
16
17
  import { Pencil, Users } from 'lucide-react'
@@ -277,7 +278,10 @@ export default function StaffTeamRolesPage() {
277
278
  })
278
279
  if (!confirmed) return
279
280
  try {
280
- await deleteCrud('staff/team-roles', entry.id, { errorMessage: labels.errors.delete })
281
+ const headers = buildOptimisticLockHeader(entry.updatedAt)
282
+ await withScopedApiRequestHeaders(headers, () => (
283
+ deleteCrud('staff/team-roles', entry.id, { errorMessage: labels.errors.delete })
284
+ ))
281
285
  flash(labels.messages.deleted, 'success')
282
286
  handleRefresh()
283
287
  } catch (error) {