@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
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../../../src/modules/staff/backend/staff/timesheets/projects/page.tsx"],
4
- "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport Link from 'next/link'\nimport { useRouter, useSearchParams } from 'next/navigation'\nimport type { ColumnDef, SortingState } from '@tanstack/react-table'\nimport { Page, PageBody } from '@open-mercato/ui/backend/Page'\nimport { DataTable, withDataTableNamespaces } from '@open-mercato/ui/backend/DataTable'\nimport type { FilterDef, FilterValues } from '@open-mercato/ui/backend/FilterOverlay'\nimport { RowActions } from '@open-mercato/ui/backend/RowActions'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { readApiResultOrThrow } from '@open-mercato/ui/backend/utils/apiCall'\nimport { deleteCrud } from '@open-mercato/ui/backend/utils/crud'\nimport { flash } from '@open-mercato/ui/backend/FlashMessages'\nimport { useConfirmDialog } from '@open-mercato/ui/backend/confirm-dialog'\nimport { useOrganizationScopeVersion } from '@open-mercato/shared/lib/frontend/useOrganizationScope'\nimport { useT, type TranslateFn } from '@open-mercato/shared/lib/i18n/context'\nimport { formatDateTime } from '@open-mercato/shared/lib/time'\nimport { ProjectColorDot } from '../../../../lib/timesheets-ui/ProjectColorDot'\nimport { resolveProjectColorHex } from '../../../../lib/timesheets-ui/colors'\nimport {\n ProjectsKpiStrip,\n type PmKpis,\n type CollabKpis,\n} from '../../../../lib/timesheets-projects-ui/ProjectsKpiStrip'\nimport { SavedViewTabs } from '../../../../lib/timesheets-projects-ui/SavedViewTabs'\nimport { HoursSparkline } from '../../../../lib/timesheets-projects-ui/HoursSparkline'\nimport {\n ProjectMembersAvatarStack,\n type AvatarMember,\n} from '../../../../lib/timesheets-projects-ui/ProjectMembersAvatarStack'\nimport { ViewModeToggle } from '../../../../lib/timesheets-projects-ui/ViewModeToggle'\nimport {\n useProjectsViewMode,\n type ProjectsViewMode,\n} from '../../../../lib/timesheets-projects-ui/useProjectsViewMode'\nimport {\n ProjectCard,\n type ProjectCardData,\n type ProjectCardLabels,\n} from '../../../../lib/timesheets-projects-ui/ProjectCard'\n\nconst PAGE_SIZE = 50\nconst INCLUDE_FIELDS = 'hoursWeek,hoursTrend,members,myRole'\n\ntype StaffEnrichment = {\n hoursWeek?: number\n hoursTrend?: number[]\n myRole?: string | null\n members?: AvatarMember[]\n memberCount?: number\n}\n\ntype ProjectRow = {\n id: string\n name: string\n code: string | null\n customerId: string | null\n customerName: string | null\n status: string\n projectType: string | null\n color: string | null\n startDate: string | null\n updatedAt: string | null\n hoursWeek: number\n hoursTrend: number[]\n myRole: string | null\n members: AvatarMember[]\n memberCount: number\n}\n\ntype ProjectsResponse = {\n items?: Array<Record<string, unknown>>\n total?: number\n totalPages?: number\n}\n\ntype KpisResponse = PmKpis | CollabKpis\n\nfunction formatRelativeTime(iso: string | null, fallback: string, t: TranslateFn): string {\n if (!iso) return fallback\n const parsed = new Date(iso)\n if (Number.isNaN(parsed.getTime())) return fallback\n const diffMs = Date.now() - parsed.getTime()\n const minutes = Math.round(diffMs / 60000)\n if (minutes < 1) return t('staff.timesheets.projects.portfolio.relativeTime.justNow', 'just now')\n if (minutes < 60) return t('staff.timesheets.projects.portfolio.relativeTime.minutesAgo', '{minutes}m ago', { minutes })\n const hours = Math.round(minutes / 60)\n if (hours < 24) return t('staff.timesheets.projects.portfolio.relativeTime.hoursAgo', '{hours}h ago', { hours })\n const days = Math.round(hours / 24)\n if (days < 14) return t('staff.timesheets.projects.portfolio.relativeTime.daysAgo', '{days}d ago', { days })\n return formatDateTime(iso) ?? fallback\n}\n\nfunction mapApiProject(item: Record<string, unknown>): ProjectRow {\n const id = typeof item.id === 'string' ? item.id : ''\n const name = typeof item.name === 'string' ? item.name : id\n const code = typeof item.code === 'string' && item.code.trim().length ? item.code.trim() : null\n const customerId =\n typeof item.customerId === 'string'\n ? item.customerId\n : typeof item.customer_id === 'string'\n ? item.customer_id\n : null\n const customerName =\n typeof item.customerName === 'string'\n ? item.customerName\n : typeof item.customer_name === 'string'\n ? item.customer_name\n : null\n const status = typeof item.status === 'string' ? item.status : 'active'\n const projectType =\n typeof item.projectType === 'string'\n ? item.projectType\n : typeof item.project_type === 'string'\n ? item.project_type\n : null\n const color = typeof item.color === 'string' ? item.color : null\n const startDate =\n typeof item.startDate === 'string'\n ? item.startDate\n : typeof item.start_date === 'string'\n ? item.start_date\n : null\n const updatedAt =\n typeof item.updatedAt === 'string'\n ? item.updatedAt\n : typeof item.updated_at === 'string'\n ? item.updated_at\n : null\n const enrichment =\n (item as { _staff?: StaffEnrichment })._staff ?? ({} as StaffEnrichment)\n const hoursWeek = typeof enrichment.hoursWeek === 'number' ? enrichment.hoursWeek : 0\n const hoursTrend = Array.isArray(enrichment.hoursTrend)\n ? enrichment.hoursTrend.filter((v): v is number => typeof v === 'number')\n : []\n const members = Array.isArray(enrichment.members)\n ? enrichment.members\n .filter((m): m is AvatarMember => !!m && typeof m === 'object' && typeof (m as AvatarMember).id === 'string')\n .map((m) => ({\n id: m.id,\n name: m.name ?? '',\n initials: m.initials ?? '',\n avatarUrl: m.avatarUrl ?? null,\n }))\n : []\n const memberCount = typeof enrichment.memberCount === 'number' ? enrichment.memberCount : members.length\n const myRole = typeof enrichment.myRole === 'string' ? enrichment.myRole : null\n\n return withDataTableNamespaces(\n {\n id,\n name,\n code,\n customerId,\n customerName,\n status,\n projectType,\n color,\n startDate,\n updatedAt,\n hoursWeek,\n hoursTrend,\n myRole,\n members,\n memberCount,\n },\n item,\n )\n}\n\nexport default function TimesheetProjectsPage() {\n const t = useT()\n const router = useRouter()\n const searchParams = useSearchParams()\n const scopeVersion = useOrganizationScopeVersion()\n const { confirm, ConfirmDialogElement } = useConfirmDialog()\n\n const [rows, setRows] = React.useState<ProjectRow[]>([])\n const [page, setPage] = React.useState(1)\n const [total, setTotal] = React.useState(0)\n const [totalPages, setTotalPages] = React.useState(1)\n const [sorting, setSorting] = React.useState<SortingState>([{ id: 'updatedAt', desc: true }])\n const [search, setSearch] = React.useState('')\n const [filterValues, setFilterValues] = React.useState<FilterValues>({})\n const [isLoading, setIsLoading] = React.useState(true)\n const [isRefreshing, setIsRefreshing] = React.useState(false)\n const hasLoadedOnceRef = React.useRef(false)\n const [reloadToken, setReloadToken] = React.useState(0)\n const [kpis, setKpis] = React.useState<KpisResponse | null>(null)\n const [isLoadingKpis, setIsLoadingKpis] = React.useState(true)\n\n const activeTab = searchParams.get('tab') ?? 'all'\n const urlViewMode = searchParams.get('view')\n const [viewMode, setViewMode] = useProjectsViewMode({\n userKey: null,\n urlOverride: urlViewMode,\n })\n\n const labels = React.useMemo(\n () => ({\n title: t('staff.timesheets.projects.page.title', 'Projects'),\n table: {\n name: t('staff.timesheets.projects.table.name', 'Project'),\n status: t('staff.timesheets.projects.table.status', 'Status'),\n type: t('staff.timesheets.projects.table.type', 'Type'),\n updatedAt: t('staff.timesheets.projects.table.updatedAt', 'Updated'),\n empty: t('staff.timesheets.projects.table.empty', 'No projects yet.'),\n search: t('staff.timesheets.projects.table.search', 'Search projects...'),\n team: t('staff.timesheets.projects.portfolio.team', 'Team'),\n myRole: t('staff.timesheets.projects.portfolio.myRole', 'My role'),\n hoursWeek: t('staff.timesheets.projects.portfolio.hoursWeek', 'Hours / week'),\n myHoursWeek: t('staff.timesheets.projects.portfolio.myHoursWeek', 'My hours / week'),\n },\n actions: {\n add: t('staff.timesheets.projects.actions.add', 'Add Project'),\n addFirst: t('staff.timesheets.projects.actions.addFirst', '+ Add first project'),\n viewDetails: t('staff.timesheets.projects.actions.viewDetails', 'View Details'),\n delete: t('staff.timesheets.projects.actions.delete', 'Delete'),\n deleteConfirm: t('staff.timesheets.projects.actions.deleteConfirm', 'Delete project \"{{name}}\"?'),\n refresh: t('staff.timesheets.projects.actions.refresh', 'Refresh'),\n },\n messages: {\n deleted: t('staff.timesheets.projects.messages.deleted', 'Project deleted.'),\n },\n errors: {\n load: t('staff.timesheets.projects.errors.load', 'Failed to load projects.'),\n delete: t('staff.timesheets.projects.errors.delete', 'Failed to delete project.'),\n },\n statuses: {\n all: t('staff.timesheets.projects.statuses.all', 'All'),\n active: t('staff.timesheets.projects.statuses.active', 'Active'),\n on_hold: t('staff.timesheets.projects.statuses.onHold', 'On Hold'),\n completed: t('staff.timesheets.projects.statuses.completed', 'Completed'),\n },\n tabs: {\n all: t('staff.timesheets.projects.portfolio.tabs.all', 'All'),\n active: t('staff.timesheets.projects.portfolio.tabs.active', 'Active'),\n onHold: t('staff.timesheets.projects.portfolio.tabs.onHold', 'On Hold'),\n completed: t('staff.timesheets.projects.portfolio.tabs.completed', 'Completed'),\n mine: t('staff.timesheets.projects.portfolio.tabs.mine', 'Mine'),\n },\n viewMode: {\n table: t('staff.timesheets.projects.portfolio.viewMode.table', 'Table'),\n cards: t('staff.timesheets.projects.portfolio.viewMode.cards', 'Cards'),\n },\n kpi: {\n totalProjects: t('staff.timesheets.projects.portfolio.kpi.totalProjects', 'Total Projects'),\n hoursWeek: t('staff.timesheets.projects.portfolio.kpi.hoursWeek', 'Hours this week'),\n hoursWeekSub: t('staff.timesheets.projects.portfolio.kpi.hoursWeekSub', 'vs previous week'),\n assignedToMe: t('staff.timesheets.projects.portfolio.kpi.assignedToMe', 'Assigned to me'),\n hoursMonth: t('staff.timesheets.projects.portfolio.kpi.hoursMonth', 'Hours this month'),\n hoursMonthSub: t('staff.timesheets.projects.portfolio.kpi.hoursMonthSub', 'vs previous month'),\n teamActive: t('staff.timesheets.projects.portfolio.kpi.teamActive', 'Active team'),\n teamActiveSub: t('staff.timesheets.projects.portfolio.kpi.teamActiveSub', 'Members with entries this month'),\n myProjects: t('staff.timesheets.projects.portfolio.kpi.myProjects', 'My projects'),\n myHoursWeek: t('staff.timesheets.projects.portfolio.kpi.myHoursWeek', 'My hours this week'),\n myHoursMonth: t('staff.timesheets.projects.portfolio.kpi.myHoursMonth', 'My hours this month'),\n deltaFlat: t('staff.timesheets.projects.portfolio.kpi.deltaFlat', 'no change'),\n noPrevious: t('staff.timesheets.projects.portfolio.kpi.noPrevious', 'no previous data'),\n },\n card: {\n hoursPanelPm: t('staff.timesheets.projects.portfolio.card.hoursPanelPm', 'Team hours \u00B7 last 7w'),\n hoursPanelCollab: t('staff.timesheets.projects.portfolio.card.hoursPanelCollab', 'My hours \u00B7 last 7w'),\n sparklineAria: t('staff.timesheets.projects.portfolio.sparkline.ariaLabel', 'Hours per week, last 7 weeks'),\n role: t('staff.timesheets.projects.portfolio.card.role', 'Role'),\n },\n emptyState: {\n noProjects: t('staff.timesheets.projects.portfolio.emptyState.noProjects', 'No projects yet.'),\n noAssignments: t(\n 'staff.timesheets.projects.portfolio.emptyState.noAssignments',\n \"You aren't assigned to any projects yet. Ask a PM to add you.\",\n ),\n noMatches: t('staff.timesheets.projects.portfolio.emptyState.noMatches', 'No projects match these filters.'),\n },\n }),\n [t],\n )\n\n const kpiLabels = React.useMemo(\n () => ({\n totalProjects: labels.kpi.totalProjects,\n totalProjectsSub: ({ active, onHold }: { active: number; onHold: number }) =>\n `${active} ${labels.statuses.active.toLowerCase()} \u00B7 ${onHold} ${labels.statuses.on_hold.toLowerCase()}`,\n hoursWeek: labels.kpi.hoursWeek,\n hoursWeekSub: labels.kpi.hoursWeekSub,\n assignedToMe: labels.kpi.assignedToMe,\n assignedToMeSub: (active: number) => `${active} ${labels.statuses.active.toLowerCase()}`,\n hoursMonth: labels.kpi.hoursMonth,\n hoursMonthSub: labels.kpi.hoursMonthSub,\n teamActive: labels.kpi.teamActive,\n teamActiveSub: labels.kpi.teamActiveSub,\n myProjects: labels.kpi.myProjects,\n myProjectsSub: (active: number) => `${active} ${labels.statuses.active.toLowerCase()}`,\n myHoursWeek: labels.kpi.myHoursWeek,\n myHoursMonth: labels.kpi.myHoursMonth,\n deltaUp: (pct: number) => `up ${pct}%`,\n deltaDown: (pct: number) => `down ${pct}%`,\n deltaFlat: labels.kpi.deltaFlat,\n noPrevious: labels.kpi.noPrevious,\n }),\n [labels],\n )\n\n const isPmRole = kpis?.role === 'pm'\n\n const filters = React.useMemo<FilterDef[]>(\n () => [\n {\n id: 'status',\n label: labels.table.status,\n type: 'select',\n options: [\n { value: 'active', label: labels.statuses.active },\n { value: 'on_hold', label: labels.statuses.on_hold },\n { value: 'completed', label: labels.statuses.completed },\n ],\n },\n ],\n [labels.table.status, labels.statuses],\n )\n\n const tabs = React.useMemo(() => {\n const base = [\n { id: 'all', label: labels.tabs.all },\n { id: 'active', label: labels.tabs.active },\n ]\n if (isPmRole) {\n base.push({ id: 'on_hold', label: labels.tabs.onHold })\n }\n base.push({ id: 'completed', label: labels.tabs.completed })\n if (isPmRole) {\n base.push({ id: 'mine', label: labels.tabs.mine })\n }\n return base\n }, [labels.tabs, isPmRole])\n\n const statusFromTab = (tabId: string): string | null => {\n if (tabId === 'active' || tabId === 'on_hold' || tabId === 'completed') return tabId\n return null\n }\n const mineFromTab = (tabId: string): boolean => tabId === 'mine' || !isPmRole\n\n const loadKpis = React.useCallback(async () => {\n setIsLoadingKpis(true)\n try {\n const payload = await readApiResultOrThrow<KpisResponse>(\n '/api/staff/timesheets/projects/kpis',\n undefined,\n {\n errorMessage: labels.errors.load,\n fallback: null as unknown as KpisResponse,\n },\n )\n setKpis(payload)\n } catch {\n setKpis(null)\n } finally {\n setIsLoadingKpis(false)\n }\n }, [labels.errors.load])\n\n const loadProjects = React.useCallback(async () => {\n if (hasLoadedOnceRef.current) {\n setIsRefreshing(true)\n } else {\n setIsLoading(true)\n }\n try {\n const params = new URLSearchParams({\n page: String(page),\n pageSize: String(PAGE_SIZE),\n include: INCLUDE_FIELDS,\n })\n const sort = sorting[0]\n if (sort?.id) {\n params.set('sortField', sort.id)\n params.set('sortDir', sort.desc ? 'desc' : 'asc')\n }\n if (search.trim()) params.set('q', search.trim())\n const tabStatus = statusFromTab(activeTab)\n if (tabStatus) params.set('status', tabStatus)\n else if (typeof filterValues.status === 'string' && filterValues.status.length > 0) {\n params.set('status', filterValues.status)\n }\n if (mineFromTab(activeTab)) params.set('mine', '1')\n\n const payload = await readApiResultOrThrow<ProjectsResponse>(\n `/api/staff/timesheets/time-projects?${params.toString()}`,\n undefined,\n { errorMessage: labels.errors.load, fallback: { items: [], total: 0, totalPages: 1 } },\n )\n const items = Array.isArray(payload.items) ? payload.items : []\n setRows(items.map(mapApiProject))\n setTotal(typeof payload.total === 'number' ? payload.total : items.length)\n setTotalPages(\n typeof payload.totalPages === 'number'\n ? payload.totalPages\n : Math.max(1, Math.ceil(items.length / PAGE_SIZE)),\n )\n } catch (error) {\n console.error('staff.timesheets.projects.list', error)\n flash(labels.errors.load, 'error')\n } finally {\n setIsLoading(false)\n setIsRefreshing(false)\n hasLoadedOnceRef.current = true\n }\n }, [labels.errors.load, page, search, sorting, filterValues.status, activeTab, isPmRole])\n\n React.useEffect(() => {\n void loadKpis()\n }, [loadKpis, scopeVersion, reloadToken])\n\n React.useEffect(() => {\n void loadProjects()\n }, [loadProjects, scopeVersion, reloadToken])\n\n const handleTabSelect = React.useCallback(\n (id: string) => {\n const params = new URLSearchParams(searchParams.toString())\n if (id === 'all') params.delete('tab')\n else params.set('tab', id)\n router.replace(`?${params.toString()}`)\n setPage(1)\n },\n [router, searchParams],\n )\n\n const handleViewModeChange = React.useCallback(\n (next: ProjectsViewMode) => {\n setViewMode(next)\n const params = new URLSearchParams(searchParams.toString())\n if (next === 'table') params.delete('view')\n else params.set('view', next)\n router.replace(`?${params.toString()}`)\n },\n [router, searchParams, setViewMode],\n )\n\n const handleSearchChange = React.useCallback((value: string) => {\n setSearch(value)\n setPage(1)\n }, [])\n\n const handleFiltersApply = React.useCallback((values: FilterValues) => {\n setFilterValues(values)\n setPage(1)\n }, [])\n\n const handleFiltersClear = React.useCallback(() => {\n setFilterValues({})\n setPage(1)\n }, [])\n\n const handleRefresh = React.useCallback(() => {\n setReloadToken((token) => token + 1)\n }, [])\n\n const handleDelete = React.useCallback(\n async (entry: ProjectRow) => {\n const message = labels.actions.deleteConfirm.replace('{{name}}', entry.name)\n const confirmed = await confirm({\n title: labels.actions.delete,\n text: message,\n variant: 'destructive',\n })\n if (!confirmed) return\n try {\n await deleteCrud('staff/timesheets/time-projects', entry.id, { errorMessage: labels.errors.delete })\n flash(labels.messages.deleted, 'success')\n handleRefresh()\n } catch (error) {\n console.error('staff.timesheets.projects.delete', error)\n flash(labels.errors.delete, 'error')\n }\n },\n [confirm, handleRefresh, labels.actions.deleteConfirm, labels.actions.delete, labels.errors.delete, labels.messages.deleted],\n )\n\n const columns = React.useMemo<ColumnDef<ProjectRow>[]>(\n () => [\n {\n accessorKey: 'name',\n header: labels.table.name,\n meta: { priority: 1, sticky: true },\n cell: ({ row }) => (\n <div className=\"flex items-center gap-2.5\">\n <ProjectColorDot colorKey={row.original.color} projectName={row.original.name} size=\"sm\" />\n <div className=\"flex min-w-0 flex-col\">\n <span className=\"truncate text-sm font-medium text-foreground\">{row.original.name}</span>\n <span className=\"truncate font-mono text-[11px] text-muted-foreground\">\n {row.original.code ?? '\u2014'}\n {row.original.customerName ? ` \u00B7 ${row.original.customerName}` : ''}\n </span>\n </div>\n </div>\n ),\n },\n {\n accessorKey: 'status',\n header: labels.table.status,\n meta: { priority: 2 },\n cell: ({ row }) => {\n const badgeClass =\n row.original.status === 'active'\n ? 'bg-lime-100 text-lime-800 dark:bg-lime-900/30 dark:text-lime-300'\n : row.original.status === 'on_hold'\n ? 'bg-amber-100 text-amber-800 dark:bg-amber-900/30 dark:text-amber-300'\n : 'bg-muted text-muted-foreground'\n const statusLabel =\n labels.statuses[row.original.status as keyof typeof labels.statuses] ?? row.original.status\n return (\n <span\n className={`inline-flex items-center rounded-full px-2 py-0.5 text-[11px] font-medium ${badgeClass}`}\n >\n {statusLabel}\n </span>\n )\n },\n },\n {\n accessorKey: 'projectType',\n header: labels.table.type,\n meta: { priority: 3 },\n cell: ({ row }) =>\n row.original.projectType ? (\n <span className=\"text-sm text-foreground\">{row.original.projectType}</span>\n ) : (\n <span className=\"text-xs text-muted-foreground/70\">\u2014</span>\n ),\n },\n isPmRole\n ? {\n accessorKey: 'members',\n id: 'members',\n header: labels.table.team,\n enableSorting: false,\n meta: { priority: 4 },\n cell: ({ row }) => (\n <ProjectMembersAvatarStack\n members={row.original.members}\n total={row.original.memberCount}\n peopleCountLabel={`${row.original.memberCount}`}\n />\n ),\n }\n : {\n accessorKey: 'myRole',\n header: labels.table.myRole,\n enableSorting: false,\n meta: { priority: 4 },\n cell: ({ row }) =>\n row.original.myRole ? (\n <span className=\"text-sm text-foreground\">{row.original.myRole}</span>\n ) : (\n <span className=\"text-xs text-muted-foreground/70\">\u2014</span>\n ),\n },\n {\n accessorKey: 'hoursWeek',\n header: isPmRole ? labels.table.hoursWeek : labels.table.myHoursWeek,\n enableSorting: false,\n meta: { priority: 5 },\n cell: ({ row }) => {\n const stripe = resolveProjectColorHex(row.original.color, row.original.name)\n return (\n <div className=\"flex items-center justify-end gap-2\">\n <HoursSparkline\n values={row.original.hoursTrend}\n color={stripe}\n ariaLabel={labels.card.sparklineAria}\n />\n <span className=\"text-xs font-medium tabular-nums text-foreground\">\n {row.original.hoursWeek > 0 ? `${row.original.hoursWeek}h` : '\u2014'}\n </span>\n </div>\n )\n },\n },\n {\n accessorKey: 'updatedAt',\n header: labels.table.updatedAt,\n meta: { priority: 6 },\n cell: ({ row }) => (\n <span className=\"text-xs text-muted-foreground\">\n {formatRelativeTime(row.original.updatedAt, '\u2014', t)}\n </span>\n ),\n },\n ],\n [labels.table, labels.statuses, labels.card.sparklineAria, isPmRole],\n )\n\n const cardLabels = React.useMemo<ProjectCardLabels>(\n () => ({\n hoursPanelPm: labels.card.hoursPanelPm,\n hoursPanelCollab: labels.card.hoursPanelCollab,\n sparklineAria: labels.card.sparklineAria,\n peopleCount: (count: number) => `${count}`,\n role: labels.card.role,\n noCustomer: '\u2014',\n statuses: labels.statuses,\n }),\n [labels.card, labels.statuses],\n )\n\n const canManage = isPmRole\n\n const emptyStateCopy = React.useMemo(() => {\n const hasFiltersApplied = activeTab !== 'all' || search.trim().length > 0 || Object.values(filterValues).some(Boolean)\n if (hasFiltersApplied) return labels.emptyState.noMatches\n if (!canManage) return labels.emptyState.noAssignments\n return labels.emptyState.noProjects\n }, [activeTab, search, filterValues, canManage, labels.emptyState])\n\n const cardsData: ProjectCardData[] = rows.map((row) => ({\n id: row.id,\n name: row.name,\n code: row.code,\n customerName: row.customerName,\n color: row.color,\n status: row.status,\n hoursWeek: row.hoursWeek,\n hoursTrend: row.hoursTrend,\n members: row.members,\n memberCount: row.memberCount,\n myRole: row.myRole,\n updatedAt: row.updatedAt,\n }))\n\n return (\n <Page>\n <PageBody>\n <div className=\"mb-4\">\n <ProjectsKpiStrip kpis={kpis} labels={kpiLabels} isLoading={isLoadingKpis} />\n </div>\n\n <div className=\"mb-3 flex flex-wrap items-center justify-between gap-3\">\n <SavedViewTabs\n tabs={tabs}\n activeId={activeTab}\n onSelect={handleTabSelect}\n ariaLabel={t('staff.timesheets.projects.portfolio.savedViews.ariaLabel', 'Saved views')}\n />\n <ViewModeToggle\n mode={viewMode}\n onChange={handleViewModeChange}\n tableLabel={labels.viewMode.table}\n cardsLabel={labels.viewMode.cards}\n ariaLabel={t('staff.timesheets.projects.portfolio.viewMode.ariaLabel', 'View mode')}\n />\n </div>\n\n {viewMode === 'cards' ? (\n <div>\n {isLoading ? (\n <div className=\"grid grid-cols-1 gap-3 md:grid-cols-2 lg:grid-cols-3\">\n {Array.from({ length: 6 }).map((_, idx) => (\n <div key={idx} className=\"h-48 animate-pulse rounded-lg border border-border bg-muted/40\" />\n ))}\n </div>\n ) : rows.length === 0 ? (\n <div className=\"rounded-lg border border-dashed border-border bg-card p-10 text-center\">\n <p className=\"text-sm text-muted-foreground\">{emptyStateCopy}</p>\n {canManage ? (\n <div className=\"mt-3\">\n <Button asChild size=\"sm\">\n <Link href=\"/backend/staff/timesheets/projects/create\">\n {labels.actions.addFirst}\n </Link>\n </Button>\n </div>\n ) : null}\n </div>\n ) : (\n <div className=\"grid grid-cols-1 gap-3 md:grid-cols-2 lg:grid-cols-3\">\n {cardsData.map((card) => (\n <ProjectCard\n key={card.id}\n data={card}\n labels={cardLabels}\n showTeam={isPmRole}\n href={`/backend/staff/timesheets/projects/${card.id}`}\n />\n ))}\n </div>\n )}\n </div>\n ) : (\n <DataTable<ProjectRow>\n title={labels.title}\n data={rows}\n columns={columns}\n isLoading={isLoading}\n searchValue={search}\n onSearchChange={handleSearchChange}\n searchPlaceholder={labels.table.search}\n filters={filters}\n filterValues={filterValues}\n onFiltersApply={handleFiltersApply}\n onFiltersClear={handleFiltersClear}\n emptyState={\n <div className=\"py-12 text-center\">\n <p className=\"text-sm text-muted-foreground mb-4\">{emptyStateCopy}</p>\n {canManage ? (\n <Button asChild size=\"sm\">\n <Link href=\"/backend/staff/timesheets/projects/create\">{labels.actions.addFirst}</Link>\n </Button>\n ) : null}\n </div>\n }\n actions={\n canManage ? (\n <Button asChild size=\"sm\">\n <Link href=\"/backend/staff/timesheets/projects/create\">{labels.actions.add}</Link>\n </Button>\n ) : undefined\n }\n refreshButton={{\n label: labels.actions.refresh,\n onRefresh: handleRefresh,\n isRefreshing: isLoading || isRefreshing,\n }}\n sortable\n sorting={sorting}\n onSortingChange={setSorting}\n pagination={{\n page,\n pageSize: PAGE_SIZE,\n total,\n totalPages,\n onPageChange: setPage,\n }}\n rowActions={(row) => (\n <RowActions\n items={[\n {\n id: 'view',\n label: labels.actions.viewDetails,\n href: `/backend/staff/timesheets/projects/${row.id}`,\n },\n ...(canManage\n ? [\n {\n id: 'delete',\n label: labels.actions.delete,\n destructive: true,\n onSelect: () => {\n void handleDelete(row)\n },\n },\n ]\n : []),\n ]}\n />\n )}\n onRowClick={(row) => router.push(`/backend/staff/timesheets/projects/${row.id}`)}\n />\n )}\n </PageBody>\n {ConfirmDialogElement}\n </Page>\n )\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport Link from 'next/link'\nimport { useRouter, useSearchParams } from 'next/navigation'\nimport type { ColumnDef, SortingState } from '@tanstack/react-table'\nimport { Page, PageBody } from '@open-mercato/ui/backend/Page'\nimport { DataTable, withDataTableNamespaces } from '@open-mercato/ui/backend/DataTable'\nimport type { FilterDef, FilterValues } from '@open-mercato/ui/backend/FilterOverlay'\nimport { RowActions } from '@open-mercato/ui/backend/RowActions'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { readApiResultOrThrow, withScopedApiRequestHeaders } from '@open-mercato/ui/backend/utils/apiCall'\nimport { buildOptimisticLockHeader } from '@open-mercato/ui/backend/utils/optimisticLock'\nimport { surfaceRecordConflict } from '@open-mercato/ui/backend/conflicts'\nimport { deleteCrud } from '@open-mercato/ui/backend/utils/crud'\nimport { flash } from '@open-mercato/ui/backend/FlashMessages'\nimport { useConfirmDialog } from '@open-mercato/ui/backend/confirm-dialog'\nimport { useOrganizationScopeVersion } from '@open-mercato/shared/lib/frontend/useOrganizationScope'\nimport { useT, type TranslateFn } from '@open-mercato/shared/lib/i18n/context'\nimport { formatDateTime } from '@open-mercato/shared/lib/time'\nimport { ProjectColorDot } from '../../../../lib/timesheets-ui/ProjectColorDot'\nimport { resolveProjectColorHex } from '../../../../lib/timesheets-ui/colors'\nimport {\n ProjectsKpiStrip,\n type PmKpis,\n type CollabKpis,\n} from '../../../../lib/timesheets-projects-ui/ProjectsKpiStrip'\nimport { SavedViewTabs } from '../../../../lib/timesheets-projects-ui/SavedViewTabs'\nimport { HoursSparkline } from '../../../../lib/timesheets-projects-ui/HoursSparkline'\nimport {\n ProjectMembersAvatarStack,\n type AvatarMember,\n} from '../../../../lib/timesheets-projects-ui/ProjectMembersAvatarStack'\nimport { ViewModeToggle } from '../../../../lib/timesheets-projects-ui/ViewModeToggle'\nimport {\n useProjectsViewMode,\n type ProjectsViewMode,\n} from '../../../../lib/timesheets-projects-ui/useProjectsViewMode'\nimport {\n ProjectCard,\n type ProjectCardData,\n type ProjectCardLabels,\n} from '../../../../lib/timesheets-projects-ui/ProjectCard'\n\nconst PAGE_SIZE = 50\nconst INCLUDE_FIELDS = 'hoursWeek,hoursTrend,members,myRole'\n\ntype StaffEnrichment = {\n hoursWeek?: number\n hoursTrend?: number[]\n myRole?: string | null\n members?: AvatarMember[]\n memberCount?: number\n}\n\ntype ProjectRow = {\n id: string\n name: string\n code: string | null\n customerId: string | null\n customerName: string | null\n status: string\n projectType: string | null\n color: string | null\n startDate: string | null\n updatedAt: string | null\n hoursWeek: number\n hoursTrend: number[]\n myRole: string | null\n members: AvatarMember[]\n memberCount: number\n}\n\ntype ProjectsResponse = {\n items?: Array<Record<string, unknown>>\n total?: number\n totalPages?: number\n}\n\ntype KpisResponse = PmKpis | CollabKpis\n\nfunction formatRelativeTime(iso: string | null, fallback: string, t: TranslateFn): string {\n if (!iso) return fallback\n const parsed = new Date(iso)\n if (Number.isNaN(parsed.getTime())) return fallback\n const diffMs = Date.now() - parsed.getTime()\n const minutes = Math.round(diffMs / 60000)\n if (minutes < 1) return t('staff.timesheets.projects.portfolio.relativeTime.justNow', 'just now')\n if (minutes < 60) return t('staff.timesheets.projects.portfolio.relativeTime.minutesAgo', '{minutes}m ago', { minutes })\n const hours = Math.round(minutes / 60)\n if (hours < 24) return t('staff.timesheets.projects.portfolio.relativeTime.hoursAgo', '{hours}h ago', { hours })\n const days = Math.round(hours / 24)\n if (days < 14) return t('staff.timesheets.projects.portfolio.relativeTime.daysAgo', '{days}d ago', { days })\n return formatDateTime(iso) ?? fallback\n}\n\nfunction mapApiProject(item: Record<string, unknown>): ProjectRow {\n const id = typeof item.id === 'string' ? item.id : ''\n const name = typeof item.name === 'string' ? item.name : id\n const code = typeof item.code === 'string' && item.code.trim().length ? item.code.trim() : null\n const customerId =\n typeof item.customerId === 'string'\n ? item.customerId\n : typeof item.customer_id === 'string'\n ? item.customer_id\n : null\n const customerName =\n typeof item.customerName === 'string'\n ? item.customerName\n : typeof item.customer_name === 'string'\n ? item.customer_name\n : null\n const status = typeof item.status === 'string' ? item.status : 'active'\n const projectType =\n typeof item.projectType === 'string'\n ? item.projectType\n : typeof item.project_type === 'string'\n ? item.project_type\n : null\n const color = typeof item.color === 'string' ? item.color : null\n const startDate =\n typeof item.startDate === 'string'\n ? item.startDate\n : typeof item.start_date === 'string'\n ? item.start_date\n : null\n const updatedAt =\n typeof item.updatedAt === 'string'\n ? item.updatedAt\n : typeof item.updated_at === 'string'\n ? item.updated_at\n : null\n const enrichment =\n (item as { _staff?: StaffEnrichment })._staff ?? ({} as StaffEnrichment)\n const hoursWeek = typeof enrichment.hoursWeek === 'number' ? enrichment.hoursWeek : 0\n const hoursTrend = Array.isArray(enrichment.hoursTrend)\n ? enrichment.hoursTrend.filter((v): v is number => typeof v === 'number')\n : []\n const members = Array.isArray(enrichment.members)\n ? enrichment.members\n .filter((m): m is AvatarMember => !!m && typeof m === 'object' && typeof (m as AvatarMember).id === 'string')\n .map((m) => ({\n id: m.id,\n name: m.name ?? '',\n initials: m.initials ?? '',\n avatarUrl: m.avatarUrl ?? null,\n }))\n : []\n const memberCount = typeof enrichment.memberCount === 'number' ? enrichment.memberCount : members.length\n const myRole = typeof enrichment.myRole === 'string' ? enrichment.myRole : null\n\n return withDataTableNamespaces(\n {\n id,\n name,\n code,\n customerId,\n customerName,\n status,\n projectType,\n color,\n startDate,\n updatedAt,\n hoursWeek,\n hoursTrend,\n myRole,\n members,\n memberCount,\n },\n item,\n )\n}\n\nexport default function TimesheetProjectsPage() {\n const t = useT()\n const router = useRouter()\n const searchParams = useSearchParams()\n const scopeVersion = useOrganizationScopeVersion()\n const { confirm, ConfirmDialogElement } = useConfirmDialog()\n\n const [rows, setRows] = React.useState<ProjectRow[]>([])\n const [page, setPage] = React.useState(1)\n const [total, setTotal] = React.useState(0)\n const [totalPages, setTotalPages] = React.useState(1)\n const [sorting, setSorting] = React.useState<SortingState>([{ id: 'updatedAt', desc: true }])\n const [search, setSearch] = React.useState('')\n const [filterValues, setFilterValues] = React.useState<FilterValues>({})\n const [isLoading, setIsLoading] = React.useState(true)\n const [isRefreshing, setIsRefreshing] = React.useState(false)\n const hasLoadedOnceRef = React.useRef(false)\n const [reloadToken, setReloadToken] = React.useState(0)\n const [kpis, setKpis] = React.useState<KpisResponse | null>(null)\n const [isLoadingKpis, setIsLoadingKpis] = React.useState(true)\n\n const activeTab = searchParams.get('tab') ?? 'all'\n const urlViewMode = searchParams.get('view')\n const [viewMode, setViewMode] = useProjectsViewMode({\n userKey: null,\n urlOverride: urlViewMode,\n })\n\n const labels = React.useMemo(\n () => ({\n title: t('staff.timesheets.projects.page.title', 'Projects'),\n table: {\n name: t('staff.timesheets.projects.table.name', 'Project'),\n status: t('staff.timesheets.projects.table.status', 'Status'),\n type: t('staff.timesheets.projects.table.type', 'Type'),\n updatedAt: t('staff.timesheets.projects.table.updatedAt', 'Updated'),\n empty: t('staff.timesheets.projects.table.empty', 'No projects yet.'),\n search: t('staff.timesheets.projects.table.search', 'Search projects...'),\n team: t('staff.timesheets.projects.portfolio.team', 'Team'),\n myRole: t('staff.timesheets.projects.portfolio.myRole', 'My role'),\n hoursWeek: t('staff.timesheets.projects.portfolio.hoursWeek', 'Hours / week'),\n myHoursWeek: t('staff.timesheets.projects.portfolio.myHoursWeek', 'My hours / week'),\n },\n actions: {\n add: t('staff.timesheets.projects.actions.add', 'Add Project'),\n addFirst: t('staff.timesheets.projects.actions.addFirst', '+ Add first project'),\n viewDetails: t('staff.timesheets.projects.actions.viewDetails', 'View Details'),\n delete: t('staff.timesheets.projects.actions.delete', 'Delete'),\n deleteConfirm: t('staff.timesheets.projects.actions.deleteConfirm', 'Delete project \"{{name}}\"?'),\n refresh: t('staff.timesheets.projects.actions.refresh', 'Refresh'),\n },\n messages: {\n deleted: t('staff.timesheets.projects.messages.deleted', 'Project deleted.'),\n },\n errors: {\n load: t('staff.timesheets.projects.errors.load', 'Failed to load projects.'),\n delete: t('staff.timesheets.projects.errors.delete', 'Failed to delete project.'),\n },\n statuses: {\n all: t('staff.timesheets.projects.statuses.all', 'All'),\n active: t('staff.timesheets.projects.statuses.active', 'Active'),\n on_hold: t('staff.timesheets.projects.statuses.onHold', 'On Hold'),\n completed: t('staff.timesheets.projects.statuses.completed', 'Completed'),\n },\n tabs: {\n all: t('staff.timesheets.projects.portfolio.tabs.all', 'All'),\n active: t('staff.timesheets.projects.portfolio.tabs.active', 'Active'),\n onHold: t('staff.timesheets.projects.portfolio.tabs.onHold', 'On Hold'),\n completed: t('staff.timesheets.projects.portfolio.tabs.completed', 'Completed'),\n mine: t('staff.timesheets.projects.portfolio.tabs.mine', 'Mine'),\n },\n viewMode: {\n table: t('staff.timesheets.projects.portfolio.viewMode.table', 'Table'),\n cards: t('staff.timesheets.projects.portfolio.viewMode.cards', 'Cards'),\n },\n kpi: {\n totalProjects: t('staff.timesheets.projects.portfolio.kpi.totalProjects', 'Total Projects'),\n hoursWeek: t('staff.timesheets.projects.portfolio.kpi.hoursWeek', 'Hours this week'),\n hoursWeekSub: t('staff.timesheets.projects.portfolio.kpi.hoursWeekSub', 'vs previous week'),\n assignedToMe: t('staff.timesheets.projects.portfolio.kpi.assignedToMe', 'Assigned to me'),\n hoursMonth: t('staff.timesheets.projects.portfolio.kpi.hoursMonth', 'Hours this month'),\n hoursMonthSub: t('staff.timesheets.projects.portfolio.kpi.hoursMonthSub', 'vs previous month'),\n teamActive: t('staff.timesheets.projects.portfolio.kpi.teamActive', 'Active team'),\n teamActiveSub: t('staff.timesheets.projects.portfolio.kpi.teamActiveSub', 'Members with entries this month'),\n myProjects: t('staff.timesheets.projects.portfolio.kpi.myProjects', 'My projects'),\n myHoursWeek: t('staff.timesheets.projects.portfolio.kpi.myHoursWeek', 'My hours this week'),\n myHoursMonth: t('staff.timesheets.projects.portfolio.kpi.myHoursMonth', 'My hours this month'),\n deltaFlat: t('staff.timesheets.projects.portfolio.kpi.deltaFlat', 'no change'),\n noPrevious: t('staff.timesheets.projects.portfolio.kpi.noPrevious', 'no previous data'),\n },\n card: {\n hoursPanelPm: t('staff.timesheets.projects.portfolio.card.hoursPanelPm', 'Team hours \u00B7 last 7w'),\n hoursPanelCollab: t('staff.timesheets.projects.portfolio.card.hoursPanelCollab', 'My hours \u00B7 last 7w'),\n sparklineAria: t('staff.timesheets.projects.portfolio.sparkline.ariaLabel', 'Hours per week, last 7 weeks'),\n role: t('staff.timesheets.projects.portfolio.card.role', 'Role'),\n },\n emptyState: {\n noProjects: t('staff.timesheets.projects.portfolio.emptyState.noProjects', 'No projects yet.'),\n noAssignments: t(\n 'staff.timesheets.projects.portfolio.emptyState.noAssignments',\n \"You aren't assigned to any projects yet. Ask a PM to add you.\",\n ),\n noMatches: t('staff.timesheets.projects.portfolio.emptyState.noMatches', 'No projects match these filters.'),\n },\n }),\n [t],\n )\n\n const kpiLabels = React.useMemo(\n () => ({\n totalProjects: labels.kpi.totalProjects,\n totalProjectsSub: ({ active, onHold }: { active: number; onHold: number }) =>\n `${active} ${labels.statuses.active.toLowerCase()} \u00B7 ${onHold} ${labels.statuses.on_hold.toLowerCase()}`,\n hoursWeek: labels.kpi.hoursWeek,\n hoursWeekSub: labels.kpi.hoursWeekSub,\n assignedToMe: labels.kpi.assignedToMe,\n assignedToMeSub: (active: number) => `${active} ${labels.statuses.active.toLowerCase()}`,\n hoursMonth: labels.kpi.hoursMonth,\n hoursMonthSub: labels.kpi.hoursMonthSub,\n teamActive: labels.kpi.teamActive,\n teamActiveSub: labels.kpi.teamActiveSub,\n myProjects: labels.kpi.myProjects,\n myProjectsSub: (active: number) => `${active} ${labels.statuses.active.toLowerCase()}`,\n myHoursWeek: labels.kpi.myHoursWeek,\n myHoursMonth: labels.kpi.myHoursMonth,\n deltaUp: (pct: number) => `up ${pct}%`,\n deltaDown: (pct: number) => `down ${pct}%`,\n deltaFlat: labels.kpi.deltaFlat,\n noPrevious: labels.kpi.noPrevious,\n }),\n [labels],\n )\n\n const isPmRole = kpis?.role === 'pm'\n\n const filters = React.useMemo<FilterDef[]>(\n () => [\n {\n id: 'status',\n label: labels.table.status,\n type: 'select',\n options: [\n { value: 'active', label: labels.statuses.active },\n { value: 'on_hold', label: labels.statuses.on_hold },\n { value: 'completed', label: labels.statuses.completed },\n ],\n },\n ],\n [labels.table.status, labels.statuses],\n )\n\n const tabs = React.useMemo(() => {\n const base = [\n { id: 'all', label: labels.tabs.all },\n { id: 'active', label: labels.tabs.active },\n ]\n if (isPmRole) {\n base.push({ id: 'on_hold', label: labels.tabs.onHold })\n }\n base.push({ id: 'completed', label: labels.tabs.completed })\n if (isPmRole) {\n base.push({ id: 'mine', label: labels.tabs.mine })\n }\n return base\n }, [labels.tabs, isPmRole])\n\n const statusFromTab = (tabId: string): string | null => {\n if (tabId === 'active' || tabId === 'on_hold' || tabId === 'completed') return tabId\n return null\n }\n const mineFromTab = (tabId: string): boolean => tabId === 'mine' || !isPmRole\n\n const loadKpis = React.useCallback(async () => {\n setIsLoadingKpis(true)\n try {\n const payload = await readApiResultOrThrow<KpisResponse>(\n '/api/staff/timesheets/projects/kpis',\n undefined,\n {\n errorMessage: labels.errors.load,\n fallback: null as unknown as KpisResponse,\n },\n )\n setKpis(payload)\n } catch {\n setKpis(null)\n } finally {\n setIsLoadingKpis(false)\n }\n }, [labels.errors.load])\n\n const loadProjects = React.useCallback(async () => {\n if (hasLoadedOnceRef.current) {\n setIsRefreshing(true)\n } else {\n setIsLoading(true)\n }\n try {\n const params = new URLSearchParams({\n page: String(page),\n pageSize: String(PAGE_SIZE),\n include: INCLUDE_FIELDS,\n })\n const sort = sorting[0]\n if (sort?.id) {\n params.set('sortField', sort.id)\n params.set('sortDir', sort.desc ? 'desc' : 'asc')\n }\n if (search.trim()) params.set('q', search.trim())\n const tabStatus = statusFromTab(activeTab)\n if (tabStatus) params.set('status', tabStatus)\n else if (typeof filterValues.status === 'string' && filterValues.status.length > 0) {\n params.set('status', filterValues.status)\n }\n if (mineFromTab(activeTab)) params.set('mine', '1')\n\n const payload = await readApiResultOrThrow<ProjectsResponse>(\n `/api/staff/timesheets/time-projects?${params.toString()}`,\n undefined,\n { errorMessage: labels.errors.load, fallback: { items: [], total: 0, totalPages: 1 } },\n )\n const items = Array.isArray(payload.items) ? payload.items : []\n setRows(items.map(mapApiProject))\n setTotal(typeof payload.total === 'number' ? payload.total : items.length)\n setTotalPages(\n typeof payload.totalPages === 'number'\n ? payload.totalPages\n : Math.max(1, Math.ceil(items.length / PAGE_SIZE)),\n )\n } catch (error) {\n console.error('staff.timesheets.projects.list', error)\n flash(labels.errors.load, 'error')\n } finally {\n setIsLoading(false)\n setIsRefreshing(false)\n hasLoadedOnceRef.current = true\n }\n }, [labels.errors.load, page, search, sorting, filterValues.status, activeTab, isPmRole])\n\n React.useEffect(() => {\n void loadKpis()\n }, [loadKpis, scopeVersion, reloadToken])\n\n React.useEffect(() => {\n void loadProjects()\n }, [loadProjects, scopeVersion, reloadToken])\n\n const handleTabSelect = React.useCallback(\n (id: string) => {\n const params = new URLSearchParams(searchParams.toString())\n if (id === 'all') params.delete('tab')\n else params.set('tab', id)\n router.replace(`?${params.toString()}`)\n setPage(1)\n },\n [router, searchParams],\n )\n\n const handleViewModeChange = React.useCallback(\n (next: ProjectsViewMode) => {\n setViewMode(next)\n const params = new URLSearchParams(searchParams.toString())\n if (next === 'table') params.delete('view')\n else params.set('view', next)\n router.replace(`?${params.toString()}`)\n },\n [router, searchParams, setViewMode],\n )\n\n const handleSearchChange = React.useCallback((value: string) => {\n setSearch(value)\n setPage(1)\n }, [])\n\n const handleFiltersApply = React.useCallback((values: FilterValues) => {\n setFilterValues(values)\n setPage(1)\n }, [])\n\n const handleFiltersClear = React.useCallback(() => {\n setFilterValues({})\n setPage(1)\n }, [])\n\n const handleRefresh = React.useCallback(() => {\n setReloadToken((token) => token + 1)\n }, [])\n\n const handleDelete = React.useCallback(\n async (entry: ProjectRow) => {\n const message = labels.actions.deleteConfirm.replace('{{name}}', entry.name)\n const confirmed = await confirm({\n title: labels.actions.delete,\n text: message,\n variant: 'destructive',\n })\n if (!confirmed) return\n try {\n await withScopedApiRequestHeaders(\n buildOptimisticLockHeader(entry.updatedAt),\n () => deleteCrud('staff/timesheets/time-projects', entry.id, { errorMessage: labels.errors.delete }),\n )\n flash(labels.messages.deleted, 'success')\n handleRefresh()\n } catch (error) {\n if (surfaceRecordConflict(error, t)) { handleRefresh(); return }\n console.error('staff.timesheets.projects.delete', error)\n flash(labels.errors.delete, 'error')\n }\n },\n [confirm, handleRefresh, t, labels.actions.deleteConfirm, labels.actions.delete, labels.errors.delete, labels.messages.deleted],\n )\n\n const columns = React.useMemo<ColumnDef<ProjectRow>[]>(\n () => [\n {\n accessorKey: 'name',\n header: labels.table.name,\n meta: { priority: 1, sticky: true },\n cell: ({ row }) => (\n <div className=\"flex items-center gap-2.5\">\n <ProjectColorDot colorKey={row.original.color} projectName={row.original.name} size=\"sm\" />\n <div className=\"flex min-w-0 flex-col\">\n <span className=\"truncate text-sm font-medium text-foreground\">{row.original.name}</span>\n <span className=\"truncate font-mono text-[11px] text-muted-foreground\">\n {row.original.code ?? '\u2014'}\n {row.original.customerName ? ` \u00B7 ${row.original.customerName}` : ''}\n </span>\n </div>\n </div>\n ),\n },\n {\n accessorKey: 'status',\n header: labels.table.status,\n meta: { priority: 2 },\n cell: ({ row }) => {\n const badgeClass =\n row.original.status === 'active'\n ? 'bg-lime-100 text-lime-800 dark:bg-lime-900/30 dark:text-lime-300'\n : row.original.status === 'on_hold'\n ? 'bg-amber-100 text-amber-800 dark:bg-amber-900/30 dark:text-amber-300'\n : 'bg-muted text-muted-foreground'\n const statusLabel =\n labels.statuses[row.original.status as keyof typeof labels.statuses] ?? row.original.status\n return (\n <span\n className={`inline-flex items-center rounded-full px-2 py-0.5 text-[11px] font-medium ${badgeClass}`}\n >\n {statusLabel}\n </span>\n )\n },\n },\n {\n accessorKey: 'projectType',\n header: labels.table.type,\n meta: { priority: 3 },\n cell: ({ row }) =>\n row.original.projectType ? (\n <span className=\"text-sm text-foreground\">{row.original.projectType}</span>\n ) : (\n <span className=\"text-xs text-muted-foreground/70\">\u2014</span>\n ),\n },\n isPmRole\n ? {\n accessorKey: 'members',\n id: 'members',\n header: labels.table.team,\n enableSorting: false,\n meta: { priority: 4 },\n cell: ({ row }) => (\n <ProjectMembersAvatarStack\n members={row.original.members}\n total={row.original.memberCount}\n peopleCountLabel={`${row.original.memberCount}`}\n />\n ),\n }\n : {\n accessorKey: 'myRole',\n header: labels.table.myRole,\n enableSorting: false,\n meta: { priority: 4 },\n cell: ({ row }) =>\n row.original.myRole ? (\n <span className=\"text-sm text-foreground\">{row.original.myRole}</span>\n ) : (\n <span className=\"text-xs text-muted-foreground/70\">\u2014</span>\n ),\n },\n {\n accessorKey: 'hoursWeek',\n header: isPmRole ? labels.table.hoursWeek : labels.table.myHoursWeek,\n enableSorting: false,\n meta: { priority: 5 },\n cell: ({ row }) => {\n const stripe = resolveProjectColorHex(row.original.color, row.original.name)\n return (\n <div className=\"flex items-center justify-end gap-2\">\n <HoursSparkline\n values={row.original.hoursTrend}\n color={stripe}\n ariaLabel={labels.card.sparklineAria}\n />\n <span className=\"text-xs font-medium tabular-nums text-foreground\">\n {row.original.hoursWeek > 0 ? `${row.original.hoursWeek}h` : '\u2014'}\n </span>\n </div>\n )\n },\n },\n {\n accessorKey: 'updatedAt',\n header: labels.table.updatedAt,\n meta: { priority: 6 },\n cell: ({ row }) => (\n <span className=\"text-xs text-muted-foreground\">\n {formatRelativeTime(row.original.updatedAt, '\u2014', t)}\n </span>\n ),\n },\n ],\n [labels.table, labels.statuses, labels.card.sparklineAria, isPmRole],\n )\n\n const cardLabels = React.useMemo<ProjectCardLabels>(\n () => ({\n hoursPanelPm: labels.card.hoursPanelPm,\n hoursPanelCollab: labels.card.hoursPanelCollab,\n sparklineAria: labels.card.sparklineAria,\n peopleCount: (count: number) => `${count}`,\n role: labels.card.role,\n noCustomer: '\u2014',\n statuses: labels.statuses,\n }),\n [labels.card, labels.statuses],\n )\n\n const canManage = isPmRole\n\n const emptyStateCopy = React.useMemo(() => {\n const hasFiltersApplied = activeTab !== 'all' || search.trim().length > 0 || Object.values(filterValues).some(Boolean)\n if (hasFiltersApplied) return labels.emptyState.noMatches\n if (!canManage) return labels.emptyState.noAssignments\n return labels.emptyState.noProjects\n }, [activeTab, search, filterValues, canManage, labels.emptyState])\n\n const cardsData: ProjectCardData[] = rows.map((row) => ({\n id: row.id,\n name: row.name,\n code: row.code,\n customerName: row.customerName,\n color: row.color,\n status: row.status,\n hoursWeek: row.hoursWeek,\n hoursTrend: row.hoursTrend,\n members: row.members,\n memberCount: row.memberCount,\n myRole: row.myRole,\n updatedAt: row.updatedAt,\n }))\n\n return (\n <Page>\n <PageBody>\n <div className=\"mb-4\">\n <ProjectsKpiStrip kpis={kpis} labels={kpiLabels} isLoading={isLoadingKpis} />\n </div>\n\n <div className=\"mb-3 flex flex-wrap items-center justify-between gap-3\">\n <SavedViewTabs\n tabs={tabs}\n activeId={activeTab}\n onSelect={handleTabSelect}\n ariaLabel={t('staff.timesheets.projects.portfolio.savedViews.ariaLabel', 'Saved views')}\n />\n <ViewModeToggle\n mode={viewMode}\n onChange={handleViewModeChange}\n tableLabel={labels.viewMode.table}\n cardsLabel={labels.viewMode.cards}\n ariaLabel={t('staff.timesheets.projects.portfolio.viewMode.ariaLabel', 'View mode')}\n />\n </div>\n\n {viewMode === 'cards' ? (\n <div>\n {isLoading ? (\n <div className=\"grid grid-cols-1 gap-3 md:grid-cols-2 lg:grid-cols-3\">\n {Array.from({ length: 6 }).map((_, idx) => (\n <div key={idx} className=\"h-48 animate-pulse rounded-lg border border-border bg-muted/40\" />\n ))}\n </div>\n ) : rows.length === 0 ? (\n <div className=\"rounded-lg border border-dashed border-border bg-card p-10 text-center\">\n <p className=\"text-sm text-muted-foreground\">{emptyStateCopy}</p>\n {canManage ? (\n <div className=\"mt-3\">\n <Button asChild size=\"sm\">\n <Link href=\"/backend/staff/timesheets/projects/create\">\n {labels.actions.addFirst}\n </Link>\n </Button>\n </div>\n ) : null}\n </div>\n ) : (\n <div className=\"grid grid-cols-1 gap-3 md:grid-cols-2 lg:grid-cols-3\">\n {cardsData.map((card) => (\n <ProjectCard\n key={card.id}\n data={card}\n labels={cardLabels}\n showTeam={isPmRole}\n href={`/backend/staff/timesheets/projects/${card.id}`}\n />\n ))}\n </div>\n )}\n </div>\n ) : (\n <DataTable<ProjectRow>\n title={labels.title}\n data={rows}\n columns={columns}\n isLoading={isLoading}\n searchValue={search}\n onSearchChange={handleSearchChange}\n searchPlaceholder={labels.table.search}\n filters={filters}\n filterValues={filterValues}\n onFiltersApply={handleFiltersApply}\n onFiltersClear={handleFiltersClear}\n emptyState={\n <div className=\"py-12 text-center\">\n <p className=\"text-sm text-muted-foreground mb-4\">{emptyStateCopy}</p>\n {canManage ? (\n <Button asChild size=\"sm\">\n <Link href=\"/backend/staff/timesheets/projects/create\">{labels.actions.addFirst}</Link>\n </Button>\n ) : null}\n </div>\n }\n actions={\n canManage ? (\n <Button asChild size=\"sm\">\n <Link href=\"/backend/staff/timesheets/projects/create\">{labels.actions.add}</Link>\n </Button>\n ) : undefined\n }\n refreshButton={{\n label: labels.actions.refresh,\n onRefresh: handleRefresh,\n isRefreshing: isLoading || isRefreshing,\n }}\n sortable\n sorting={sorting}\n onSortingChange={setSorting}\n pagination={{\n page,\n pageSize: PAGE_SIZE,\n total,\n totalPages,\n onPageChange: setPage,\n }}\n rowActions={(row) => (\n <RowActions\n items={[\n {\n id: 'view',\n label: labels.actions.viewDetails,\n href: `/backend/staff/timesheets/projects/${row.id}`,\n },\n ...(canManage\n ? [\n {\n id: 'delete',\n label: labels.actions.delete,\n destructive: true,\n onSelect: () => {\n void handleDelete(row)\n },\n },\n ]\n : []),\n ]}\n />\n )}\n onRowClick={(row) => router.push(`/backend/staff/timesheets/projects/${row.id}`)}\n />\n )}\n </PageBody>\n {ConfirmDialogElement}\n </Page>\n )\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -14,7 +14,12 @@ import {
14
14
  } from "./shared.js";
15
15
  import { resolveTranslations } from "@open-mercato/shared/lib/i18n/server";
16
16
  import { CrudHttpError } from "@open-mercato/shared/lib/crud/errors";
17
+ import {
18
+ enforceCommandOptimisticLock,
19
+ enforceRecordGoneIsConflict
20
+ } from "@open-mercato/shared/lib/crud/optimistic-lock-command";
17
21
  import { E } from "../../../generated/entities.ids.generated.js";
22
+ const JOB_HISTORY_LOCK_RESOURCE_KIND = "staff.jobHistory";
18
23
  const jobHistoryCrudIndexer = {
19
24
  entityType: E.staff.staff_team_member_job_history
20
25
  };
@@ -30,7 +35,8 @@ async function loadJobHistorySnapshot(em, id) {
30
35
  companyName: record.companyName ?? null,
31
36
  description: record.description ?? null,
32
37
  startDate: record.startDate.toISOString(),
33
- endDate: record.endDate ? record.endDate.toISOString() : null
38
+ endDate: record.endDate ? record.endDate.toISOString() : null,
39
+ updatedAt: record.updatedAt.toISOString()
34
40
  };
35
41
  }
36
42
  const createJobHistoryCommand = {
@@ -118,9 +124,24 @@ const updateJobHistoryCommand = {
118
124
  const parsed = staffTeamMemberJobHistoryUpdateSchema.parse(rawInput);
119
125
  const em = ctx.container.resolve("em").fork();
120
126
  const record = await em.findOne(StaffTeamMemberJobHistory, { id: parsed.id });
121
- if (!record) throw new CrudHttpError(404, { error: "Job history entry not found" });
127
+ if (!record) {
128
+ enforceRecordGoneIsConflict({
129
+ resourceKind: JOB_HISTORY_LOCK_RESOURCE_KIND,
130
+ resourceId: parsed.id,
131
+ expected: parsed.updatedAt ?? null,
132
+ request: ctx.request ?? null
133
+ });
134
+ throw new CrudHttpError(404, { error: "Job history entry not found" });
135
+ }
122
136
  ensureTenantScope(ctx, record.tenantId);
123
137
  ensureOrganizationScope(ctx, record.organizationId);
138
+ enforceCommandOptimisticLock({
139
+ resourceKind: JOB_HISTORY_LOCK_RESOURCE_KIND,
140
+ resourceId: record.id,
141
+ current: record.updatedAt ?? null,
142
+ expected: parsed.updatedAt ?? null,
143
+ request: ctx.request ?? null
144
+ });
124
145
  if (parsed.entityId !== void 0) {
125
146
  const member = await requireTeamMember(em, parsed.entityId, "Team member not found");
126
147
  ensureTenantScope(ctx, member.tenantId);
@@ -237,11 +258,27 @@ const deleteJobHistoryCommand = {
237
258
  },
238
259
  async execute(input, ctx) {
239
260
  const id = requireId(input, "Job history id required");
261
+ const expectedUpdatedAt = typeof input.updatedAt === "string" ? input.updatedAt : void 0;
240
262
  const em = ctx.container.resolve("em").fork();
241
263
  const record = await em.findOne(StaffTeamMemberJobHistory, { id });
242
- if (!record) throw new CrudHttpError(404, { error: "Job history entry not found" });
264
+ if (!record) {
265
+ enforceRecordGoneIsConflict({
266
+ resourceKind: JOB_HISTORY_LOCK_RESOURCE_KIND,
267
+ resourceId: id,
268
+ expected: expectedUpdatedAt ?? null,
269
+ request: ctx.request ?? null
270
+ });
271
+ throw new CrudHttpError(404, { error: "Job history entry not found" });
272
+ }
243
273
  ensureTenantScope(ctx, record.tenantId);
244
274
  ensureOrganizationScope(ctx, record.organizationId);
275
+ enforceCommandOptimisticLock({
276
+ resourceKind: JOB_HISTORY_LOCK_RESOURCE_KIND,
277
+ resourceId: record.id,
278
+ current: record.updatedAt ?? null,
279
+ expected: expectedUpdatedAt ?? null,
280
+ request: ctx.request ?? null
281
+ });
245
282
  em.remove(record);
246
283
  await em.flush();
247
284
  const de = ctx.container.resolve("dataEngine");
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../src/modules/staff/commands/job-histories.ts"],
4
- "sourcesContent": ["import { registerCommand } from '@open-mercato/shared/lib/commands'\nimport type { CommandHandler } from '@open-mercato/shared/lib/commands'\nimport { emitCrudSideEffects, emitCrudUndoSideEffects, buildChanges, requireId } from '@open-mercato/shared/lib/commands/helpers'\nimport type { DataEngine } from '@open-mercato/shared/lib/data/engine'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport { StaffTeamMemberJobHistory } from '../data/entities'\nimport {\n staffTeamMemberJobHistoryCreateSchema,\n staffTeamMemberJobHistoryUpdateSchema,\n type StaffTeamMemberJobHistoryCreateInput,\n type StaffTeamMemberJobHistoryUpdateInput,\n} from '../data/validators'\nimport { staffTeamMemberJobHistoryCrudEvents } from '../lib/crud'\nimport {\n ensureOrganizationScope,\n ensureTenantScope,\n extractUndoPayload,\n requireTeamMember,\n} from './shared'\nimport { resolveTranslations } from '@open-mercato/shared/lib/i18n/server'\nimport { CrudHttpError } from '@open-mercato/shared/lib/crud/errors'\nimport type { CrudIndexerConfig } from '@open-mercato/shared/lib/crud/types'\nimport { E } from '#generated/entities.ids.generated'\n\nconst jobHistoryCrudIndexer: CrudIndexerConfig<StaffTeamMemberJobHistory> = {\n entityType: E.staff.staff_team_member_job_history,\n}\n\ntype JobHistorySnapshot = {\n id: string\n organizationId: string\n tenantId: string\n memberId: string\n name: string\n companyName: string | null\n description: string | null\n startDate: string\n endDate: string | null\n}\n\ntype JobHistoryUndoPayload = {\n before?: JobHistorySnapshot | null\n after?: JobHistorySnapshot | null\n}\n\nasync function loadJobHistorySnapshot(em: EntityManager, id: string): Promise<JobHistorySnapshot | null> {\n const record = await em.findOne(StaffTeamMemberJobHistory, { id })\n if (!record) return null\n return {\n id: record.id,\n organizationId: record.organizationId,\n tenantId: record.tenantId,\n memberId: typeof record.member === 'string' ? record.member : record.member.id,\n name: record.name,\n companyName: record.companyName ?? null,\n description: record.description ?? null,\n startDate: record.startDate.toISOString(),\n endDate: record.endDate ? record.endDate.toISOString() : null,\n }\n}\n\nconst createJobHistoryCommand: CommandHandler<StaffTeamMemberJobHistoryCreateInput, { jobHistoryId: string }> = {\n id: 'staff.team-member-job-histories.create',\n async execute(rawInput, ctx) {\n const parsed = staffTeamMemberJobHistoryCreateSchema.parse(rawInput)\n ensureTenantScope(ctx, parsed.tenantId)\n ensureOrganizationScope(ctx, parsed.organizationId)\n\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const member = await requireTeamMember(em, parsed.entityId, 'Team member not found')\n ensureTenantScope(ctx, member.tenantId)\n ensureOrganizationScope(ctx, member.organizationId)\n\n const record = em.create(StaffTeamMemberJobHistory, {\n organizationId: parsed.organizationId,\n tenantId: parsed.tenantId,\n member,\n name: parsed.name,\n companyName: parsed.companyName ?? null,\n description: parsed.description ?? null,\n startDate: parsed.startDate,\n endDate: parsed.endDate ?? null,\n createdAt: new Date(),\n updatedAt: new Date(),\n })\n em.persist(record)\n await em.flush()\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'created',\n entity: record,\n identifiers: {\n id: record.id,\n organizationId: record.organizationId,\n tenantId: record.tenantId,\n },\n events: staffTeamMemberJobHistoryCrudEvents,\n indexer: jobHistoryCrudIndexer,\n })\n\n return { jobHistoryId: record.id }\n },\n captureAfter: async (_input, result, ctx) => {\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n return await loadJobHistorySnapshot(em, result.jobHistoryId)\n },\n buildLog: async ({ result, snapshots }) => {\n const { translate } = await resolveTranslations()\n const snapshot = snapshots.after as JobHistorySnapshot | undefined\n return {\n actionLabel: translate('staff.audit.teamMemberJobHistories.create', 'Create job history entry'),\n resourceKind: 'staff.team_member_job_history',\n resourceId: result.jobHistoryId,\n parentResourceKind: 'staff.teamMember',\n parentResourceId: snapshot?.memberId ?? null,\n tenantId: snapshot?.tenantId ?? null,\n organizationId: snapshot?.organizationId ?? null,\n snapshotAfter: snapshot ?? null,\n payload: {\n undo: {\n after: snapshot ?? null,\n } satisfies JobHistoryUndoPayload,\n },\n }\n },\n undo: async ({ logEntry, ctx }) => {\n const jobHistoryId = logEntry?.resourceId ?? null\n if (!jobHistoryId) return\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const record = await em.findOne(StaffTeamMemberJobHistory, { id: jobHistoryId })\n if (record) {\n em.remove(record)\n await em.flush()\n }\n },\n}\n\nconst updateJobHistoryCommand: CommandHandler<StaffTeamMemberJobHistoryUpdateInput, { jobHistoryId: string }> = {\n id: 'staff.team-member-job-histories.update',\n async prepare(rawInput, ctx) {\n const parsed = staffTeamMemberJobHistoryUpdateSchema.parse(rawInput)\n const em = (ctx.container.resolve('em') as EntityManager)\n const snapshot = await loadJobHistorySnapshot(em, parsed.id)\n return snapshot ? { before: snapshot } : {}\n },\n async execute(rawInput, ctx) {\n const parsed = staffTeamMemberJobHistoryUpdateSchema.parse(rawInput)\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const record = await em.findOne(StaffTeamMemberJobHistory, { id: parsed.id })\n if (!record) throw new CrudHttpError(404, { error: 'Job history entry not found' })\n ensureTenantScope(ctx, record.tenantId)\n ensureOrganizationScope(ctx, record.organizationId)\n\n if (parsed.entityId !== undefined) {\n const member = await requireTeamMember(em, parsed.entityId, 'Team member not found')\n ensureTenantScope(ctx, member.tenantId)\n ensureOrganizationScope(ctx, member.organizationId)\n record.member = member\n }\n if (parsed.name !== undefined) record.name = parsed.name\n if (parsed.companyName !== undefined) record.companyName = parsed.companyName ?? null\n if (parsed.description !== undefined) record.description = parsed.description ?? null\n if (parsed.startDate !== undefined) record.startDate = parsed.startDate\n if (parsed.endDate !== undefined) record.endDate = parsed.endDate ?? null\n\n await em.flush()\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'updated',\n entity: record,\n identifiers: {\n id: record.id,\n organizationId: record.organizationId,\n tenantId: record.tenantId,\n },\n events: staffTeamMemberJobHistoryCrudEvents,\n indexer: jobHistoryCrudIndexer,\n })\n\n return { jobHistoryId: record.id }\n },\n captureAfter: async (_input, result, ctx) => {\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n return await loadJobHistorySnapshot(em, result.jobHistoryId)\n },\n buildLog: async ({ snapshots }) => {\n const { translate } = await resolveTranslations()\n const before = snapshots.before as JobHistorySnapshot | undefined\n if (!before) return null\n const afterSnapshot = snapshots.after as JobHistorySnapshot | undefined\n const changes =\n afterSnapshot && before\n ? buildChanges(\n before as unknown as Record<string, unknown>,\n afterSnapshot as unknown as Record<string, unknown>,\n ['memberId', 'name', 'companyName', 'description', 'startDate', 'endDate'],\n )\n : {}\n return {\n actionLabel: translate('staff.audit.teamMemberJobHistories.update', 'Update job history entry'),\n resourceKind: 'staff.team_member_job_history',\n resourceId: before.id,\n parentResourceKind: 'staff.teamMember',\n parentResourceId: before.memberId ?? null,\n tenantId: before.tenantId,\n organizationId: before.organizationId,\n snapshotBefore: before,\n snapshotAfter: afterSnapshot ?? null,\n changes,\n payload: {\n undo: {\n before,\n after: afterSnapshot ?? null,\n } satisfies JobHistoryUndoPayload,\n },\n }\n },\n undo: async ({ logEntry, ctx }) => {\n const payload = extractUndoPayload<JobHistoryUndoPayload>(logEntry)\n const before = payload?.before\n if (!before) return\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n let record = await em.findOne(StaffTeamMemberJobHistory, { id: before.id })\n const member = await requireTeamMember(em, before.memberId, 'Team member not found')\n\n if (!record) {\n record = em.create(StaffTeamMemberJobHistory, {\n id: before.id,\n organizationId: before.organizationId,\n tenantId: before.tenantId,\n member,\n name: before.name,\n companyName: before.companyName,\n description: before.description,\n startDate: new Date(before.startDate),\n endDate: before.endDate ? new Date(before.endDate) : null,\n createdAt: new Date(),\n updatedAt: new Date(),\n })\n em.persist(record)\n } else {\n record.member = member\n record.name = before.name\n record.companyName = before.companyName\n record.description = before.description\n record.startDate = new Date(before.startDate)\n record.endDate = before.endDate ? new Date(before.endDate) : null\n }\n await em.flush()\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudUndoSideEffects({\n dataEngine: de,\n action: 'updated',\n entity: record,\n identifiers: {\n id: record.id,\n organizationId: record.organizationId,\n tenantId: record.tenantId,\n },\n events: staffTeamMemberJobHistoryCrudEvents,\n indexer: jobHistoryCrudIndexer,\n })\n },\n}\n\nconst deleteJobHistoryCommand: CommandHandler<{ body?: Record<string, unknown>; query?: Record<string, unknown> }, { jobHistoryId: string }> =\n {\n id: 'staff.team-member-job-histories.delete',\n async prepare(input, ctx) {\n const id = requireId(input, 'Job history id required')\n const em = (ctx.container.resolve('em') as EntityManager)\n const snapshot = await loadJobHistorySnapshot(em, id)\n return snapshot ? { before: snapshot } : {}\n },\n async execute(input, ctx) {\n const id = requireId(input, 'Job history id required')\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const record = await em.findOne(StaffTeamMemberJobHistory, { id })\n if (!record) throw new CrudHttpError(404, { error: 'Job history entry not found' })\n ensureTenantScope(ctx, record.tenantId)\n ensureOrganizationScope(ctx, record.organizationId)\n em.remove(record)\n await em.flush()\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'deleted',\n entity: record,\n identifiers: {\n id: record.id,\n organizationId: record.organizationId,\n tenantId: record.tenantId,\n },\n events: staffTeamMemberJobHistoryCrudEvents,\n indexer: jobHistoryCrudIndexer,\n })\n return { jobHistoryId: record.id }\n },\n buildLog: async ({ snapshots }) => {\n const before = snapshots.before as JobHistorySnapshot | undefined\n if (!before) return null\n const { translate } = await resolveTranslations()\n return {\n actionLabel: translate('staff.audit.teamMemberJobHistories.delete', 'Delete job history entry'),\n resourceKind: 'staff.team_member_job_history',\n resourceId: before.id,\n parentResourceKind: 'staff.teamMember',\n parentResourceId: before.memberId ?? null,\n tenantId: before.tenantId,\n organizationId: before.organizationId,\n snapshotBefore: before,\n payload: {\n undo: {\n before,\n } satisfies JobHistoryUndoPayload,\n },\n }\n },\n undo: async ({ logEntry, ctx }) => {\n const payload = extractUndoPayload<JobHistoryUndoPayload>(logEntry)\n const before = payload?.before\n if (!before) return\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const member = await requireTeamMember(em, before.memberId, 'Team member not found')\n let record = await em.findOne(StaffTeamMemberJobHistory, { id: before.id })\n if (!record) {\n record = em.create(StaffTeamMemberJobHistory, {\n id: before.id,\n organizationId: before.organizationId,\n tenantId: before.tenantId,\n member,\n name: before.name,\n companyName: before.companyName,\n description: before.description,\n startDate: new Date(before.startDate),\n endDate: before.endDate ? new Date(before.endDate) : null,\n createdAt: new Date(),\n updatedAt: new Date(),\n })\n em.persist(record)\n } else {\n record.member = member\n record.name = before.name\n record.companyName = before.companyName\n record.description = before.description\n record.startDate = new Date(before.startDate)\n record.endDate = before.endDate ? new Date(before.endDate) : null\n }\n await em.flush()\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudUndoSideEffects({\n dataEngine: de,\n action: 'created',\n entity: record,\n identifiers: {\n id: record.id,\n organizationId: record.organizationId,\n tenantId: record.tenantId,\n },\n events: staffTeamMemberJobHistoryCrudEvents,\n indexer: jobHistoryCrudIndexer,\n })\n },\n }\n\nregisterCommand(createJobHistoryCommand)\nregisterCommand(updateJobHistoryCommand)\nregisterCommand(deleteJobHistoryCommand)\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import { registerCommand } from '@open-mercato/shared/lib/commands'\nimport type { CommandHandler } from '@open-mercato/shared/lib/commands'\nimport { emitCrudSideEffects, emitCrudUndoSideEffects, buildChanges, requireId } from '@open-mercato/shared/lib/commands/helpers'\nimport type { DataEngine } from '@open-mercato/shared/lib/data/engine'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport { StaffTeamMemberJobHistory } from '../data/entities'\nimport {\n staffTeamMemberJobHistoryCreateSchema,\n staffTeamMemberJobHistoryUpdateSchema,\n type StaffTeamMemberJobHistoryCreateInput,\n type StaffTeamMemberJobHistoryUpdateInput,\n} from '../data/validators'\nimport { staffTeamMemberJobHistoryCrudEvents } from '../lib/crud'\nimport {\n ensureOrganizationScope,\n ensureTenantScope,\n extractUndoPayload,\n requireTeamMember,\n} from './shared'\nimport { resolveTranslations } from '@open-mercato/shared/lib/i18n/server'\nimport { CrudHttpError } from '@open-mercato/shared/lib/crud/errors'\nimport {\n enforceCommandOptimisticLock,\n enforceRecordGoneIsConflict,\n} from '@open-mercato/shared/lib/crud/optimistic-lock-command'\nimport type { CrudIndexerConfig } from '@open-mercato/shared/lib/crud/types'\nimport { E } from '#generated/entities.ids.generated'\n\nconst JOB_HISTORY_LOCK_RESOURCE_KIND = 'staff.jobHistory'\n\nconst jobHistoryCrudIndexer: CrudIndexerConfig<StaffTeamMemberJobHistory> = {\n entityType: E.staff.staff_team_member_job_history,\n}\n\ntype JobHistorySnapshot = {\n id: string\n organizationId: string\n tenantId: string\n memberId: string\n name: string\n companyName: string | null\n description: string | null\n startDate: string\n endDate: string | null\n updatedAt: string\n}\n\ntype JobHistoryUndoPayload = {\n before?: JobHistorySnapshot | null\n after?: JobHistorySnapshot | null\n}\n\nasync function loadJobHistorySnapshot(em: EntityManager, id: string): Promise<JobHistorySnapshot | null> {\n const record = await em.findOne(StaffTeamMemberJobHistory, { id })\n if (!record) return null\n return {\n id: record.id,\n organizationId: record.organizationId,\n tenantId: record.tenantId,\n memberId: typeof record.member === 'string' ? record.member : record.member.id,\n name: record.name,\n companyName: record.companyName ?? null,\n description: record.description ?? null,\n startDate: record.startDate.toISOString(),\n endDate: record.endDate ? record.endDate.toISOString() : null,\n updatedAt: record.updatedAt.toISOString(),\n }\n}\n\nconst createJobHistoryCommand: CommandHandler<StaffTeamMemberJobHistoryCreateInput, { jobHistoryId: string }> = {\n id: 'staff.team-member-job-histories.create',\n async execute(rawInput, ctx) {\n const parsed = staffTeamMemberJobHistoryCreateSchema.parse(rawInput)\n ensureTenantScope(ctx, parsed.tenantId)\n ensureOrganizationScope(ctx, parsed.organizationId)\n\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const member = await requireTeamMember(em, parsed.entityId, 'Team member not found')\n ensureTenantScope(ctx, member.tenantId)\n ensureOrganizationScope(ctx, member.organizationId)\n\n const record = em.create(StaffTeamMemberJobHistory, {\n organizationId: parsed.organizationId,\n tenantId: parsed.tenantId,\n member,\n name: parsed.name,\n companyName: parsed.companyName ?? null,\n description: parsed.description ?? null,\n startDate: parsed.startDate,\n endDate: parsed.endDate ?? null,\n createdAt: new Date(),\n updatedAt: new Date(),\n })\n em.persist(record)\n await em.flush()\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'created',\n entity: record,\n identifiers: {\n id: record.id,\n organizationId: record.organizationId,\n tenantId: record.tenantId,\n },\n events: staffTeamMemberJobHistoryCrudEvents,\n indexer: jobHistoryCrudIndexer,\n })\n\n return { jobHistoryId: record.id }\n },\n captureAfter: async (_input, result, ctx) => {\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n return await loadJobHistorySnapshot(em, result.jobHistoryId)\n },\n buildLog: async ({ result, snapshots }) => {\n const { translate } = await resolveTranslations()\n const snapshot = snapshots.after as JobHistorySnapshot | undefined\n return {\n actionLabel: translate('staff.audit.teamMemberJobHistories.create', 'Create job history entry'),\n resourceKind: 'staff.team_member_job_history',\n resourceId: result.jobHistoryId,\n parentResourceKind: 'staff.teamMember',\n parentResourceId: snapshot?.memberId ?? null,\n tenantId: snapshot?.tenantId ?? null,\n organizationId: snapshot?.organizationId ?? null,\n snapshotAfter: snapshot ?? null,\n payload: {\n undo: {\n after: snapshot ?? null,\n } satisfies JobHistoryUndoPayload,\n },\n }\n },\n undo: async ({ logEntry, ctx }) => {\n const jobHistoryId = logEntry?.resourceId ?? null\n if (!jobHistoryId) return\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const record = await em.findOne(StaffTeamMemberJobHistory, { id: jobHistoryId })\n if (record) {\n em.remove(record)\n await em.flush()\n }\n },\n}\n\nconst updateJobHistoryCommand: CommandHandler<StaffTeamMemberJobHistoryUpdateInput, { jobHistoryId: string }> = {\n id: 'staff.team-member-job-histories.update',\n async prepare(rawInput, ctx) {\n const parsed = staffTeamMemberJobHistoryUpdateSchema.parse(rawInput)\n const em = (ctx.container.resolve('em') as EntityManager)\n const snapshot = await loadJobHistorySnapshot(em, parsed.id)\n return snapshot ? { before: snapshot } : {}\n },\n async execute(rawInput, ctx) {\n const parsed = staffTeamMemberJobHistoryUpdateSchema.parse(rawInput)\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const record = await em.findOne(StaffTeamMemberJobHistory, { id: parsed.id })\n if (!record) {\n enforceRecordGoneIsConflict({\n resourceKind: JOB_HISTORY_LOCK_RESOURCE_KIND,\n resourceId: parsed.id,\n expected: parsed.updatedAt ?? null,\n request: ctx.request ?? null,\n })\n throw new CrudHttpError(404, { error: 'Job history entry not found' })\n }\n ensureTenantScope(ctx, record.tenantId)\n ensureOrganizationScope(ctx, record.organizationId)\n enforceCommandOptimisticLock({\n resourceKind: JOB_HISTORY_LOCK_RESOURCE_KIND,\n resourceId: record.id,\n current: record.updatedAt ?? null,\n expected: parsed.updatedAt ?? null,\n request: ctx.request ?? null,\n })\n\n if (parsed.entityId !== undefined) {\n const member = await requireTeamMember(em, parsed.entityId, 'Team member not found')\n ensureTenantScope(ctx, member.tenantId)\n ensureOrganizationScope(ctx, member.organizationId)\n record.member = member\n }\n if (parsed.name !== undefined) record.name = parsed.name\n if (parsed.companyName !== undefined) record.companyName = parsed.companyName ?? null\n if (parsed.description !== undefined) record.description = parsed.description ?? null\n if (parsed.startDate !== undefined) record.startDate = parsed.startDate\n if (parsed.endDate !== undefined) record.endDate = parsed.endDate ?? null\n\n await em.flush()\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'updated',\n entity: record,\n identifiers: {\n id: record.id,\n organizationId: record.organizationId,\n tenantId: record.tenantId,\n },\n events: staffTeamMemberJobHistoryCrudEvents,\n indexer: jobHistoryCrudIndexer,\n })\n\n return { jobHistoryId: record.id }\n },\n captureAfter: async (_input, result, ctx) => {\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n return await loadJobHistorySnapshot(em, result.jobHistoryId)\n },\n buildLog: async ({ snapshots }) => {\n const { translate } = await resolveTranslations()\n const before = snapshots.before as JobHistorySnapshot | undefined\n if (!before) return null\n const afterSnapshot = snapshots.after as JobHistorySnapshot | undefined\n const changes =\n afterSnapshot && before\n ? buildChanges(\n before as unknown as Record<string, unknown>,\n afterSnapshot as unknown as Record<string, unknown>,\n ['memberId', 'name', 'companyName', 'description', 'startDate', 'endDate'],\n )\n : {}\n return {\n actionLabel: translate('staff.audit.teamMemberJobHistories.update', 'Update job history entry'),\n resourceKind: 'staff.team_member_job_history',\n resourceId: before.id,\n parentResourceKind: 'staff.teamMember',\n parentResourceId: before.memberId ?? null,\n tenantId: before.tenantId,\n organizationId: before.organizationId,\n snapshotBefore: before,\n snapshotAfter: afterSnapshot ?? null,\n changes,\n payload: {\n undo: {\n before,\n after: afterSnapshot ?? null,\n } satisfies JobHistoryUndoPayload,\n },\n }\n },\n undo: async ({ logEntry, ctx }) => {\n const payload = extractUndoPayload<JobHistoryUndoPayload>(logEntry)\n const before = payload?.before\n if (!before) return\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n let record = await em.findOne(StaffTeamMemberJobHistory, { id: before.id })\n const member = await requireTeamMember(em, before.memberId, 'Team member not found')\n\n if (!record) {\n record = em.create(StaffTeamMemberJobHistory, {\n id: before.id,\n organizationId: before.organizationId,\n tenantId: before.tenantId,\n member,\n name: before.name,\n companyName: before.companyName,\n description: before.description,\n startDate: new Date(before.startDate),\n endDate: before.endDate ? new Date(before.endDate) : null,\n createdAt: new Date(),\n updatedAt: new Date(),\n })\n em.persist(record)\n } else {\n record.member = member\n record.name = before.name\n record.companyName = before.companyName\n record.description = before.description\n record.startDate = new Date(before.startDate)\n record.endDate = before.endDate ? new Date(before.endDate) : null\n }\n await em.flush()\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudUndoSideEffects({\n dataEngine: de,\n action: 'updated',\n entity: record,\n identifiers: {\n id: record.id,\n organizationId: record.organizationId,\n tenantId: record.tenantId,\n },\n events: staffTeamMemberJobHistoryCrudEvents,\n indexer: jobHistoryCrudIndexer,\n })\n },\n}\n\nconst deleteJobHistoryCommand: CommandHandler<{ id?: string; updatedAt?: string; body?: Record<string, unknown>; query?: Record<string, unknown> }, { jobHistoryId: string }> =\n {\n id: 'staff.team-member-job-histories.delete',\n async prepare(input, ctx) {\n const id = requireId(input, 'Job history id required')\n const em = (ctx.container.resolve('em') as EntityManager)\n const snapshot = await loadJobHistorySnapshot(em, id)\n return snapshot ? { before: snapshot } : {}\n },\n async execute(input, ctx) {\n const id = requireId(input, 'Job history id required')\n const expectedUpdatedAt = typeof input.updatedAt === 'string' ? input.updatedAt : undefined\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const record = await em.findOne(StaffTeamMemberJobHistory, { id })\n if (!record) {\n enforceRecordGoneIsConflict({\n resourceKind: JOB_HISTORY_LOCK_RESOURCE_KIND,\n resourceId: id,\n expected: expectedUpdatedAt ?? null,\n request: ctx.request ?? null,\n })\n throw new CrudHttpError(404, { error: 'Job history entry not found' })\n }\n ensureTenantScope(ctx, record.tenantId)\n ensureOrganizationScope(ctx, record.organizationId)\n enforceCommandOptimisticLock({\n resourceKind: JOB_HISTORY_LOCK_RESOURCE_KIND,\n resourceId: record.id,\n current: record.updatedAt ?? null,\n expected: expectedUpdatedAt ?? null,\n request: ctx.request ?? null,\n })\n em.remove(record)\n await em.flush()\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'deleted',\n entity: record,\n identifiers: {\n id: record.id,\n organizationId: record.organizationId,\n tenantId: record.tenantId,\n },\n events: staffTeamMemberJobHistoryCrudEvents,\n indexer: jobHistoryCrudIndexer,\n })\n return { jobHistoryId: record.id }\n },\n buildLog: async ({ snapshots }) => {\n const before = snapshots.before as JobHistorySnapshot | undefined\n if (!before) return null\n const { translate } = await resolveTranslations()\n return {\n actionLabel: translate('staff.audit.teamMemberJobHistories.delete', 'Delete job history entry'),\n resourceKind: 'staff.team_member_job_history',\n resourceId: before.id,\n parentResourceKind: 'staff.teamMember',\n parentResourceId: before.memberId ?? null,\n tenantId: before.tenantId,\n organizationId: before.organizationId,\n snapshotBefore: before,\n payload: {\n undo: {\n before,\n } satisfies JobHistoryUndoPayload,\n },\n }\n },\n undo: async ({ logEntry, ctx }) => {\n const payload = extractUndoPayload<JobHistoryUndoPayload>(logEntry)\n const before = payload?.before\n if (!before) return\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const member = await requireTeamMember(em, before.memberId, 'Team member not found')\n let record = await em.findOne(StaffTeamMemberJobHistory, { id: before.id })\n if (!record) {\n record = em.create(StaffTeamMemberJobHistory, {\n id: before.id,\n organizationId: before.organizationId,\n tenantId: before.tenantId,\n member,\n name: before.name,\n companyName: before.companyName,\n description: before.description,\n startDate: new Date(before.startDate),\n endDate: before.endDate ? new Date(before.endDate) : null,\n createdAt: new Date(),\n updatedAt: new Date(),\n })\n em.persist(record)\n } else {\n record.member = member\n record.name = before.name\n record.companyName = before.companyName\n record.description = before.description\n record.startDate = new Date(before.startDate)\n record.endDate = before.endDate ? new Date(before.endDate) : null\n }\n await em.flush()\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudUndoSideEffects({\n dataEngine: de,\n action: 'created',\n entity: record,\n identifiers: {\n id: record.id,\n organizationId: record.organizationId,\n tenantId: record.tenantId,\n },\n events: staffTeamMemberJobHistoryCrudEvents,\n indexer: jobHistoryCrudIndexer,\n })\n },\n }\n\nregisterCommand(createJobHistoryCommand)\nregisterCommand(updateJobHistoryCommand)\nregisterCommand(deleteJobHistoryCommand)\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -257,6 +257,7 @@ function LeaveRequestForm(props) {
257
257
  title,
258
258
  fields,
259
259
  initialValues: normalizedInitialValues,
260
+ optimisticLockUpdatedAt: normalizedInitialValues.updatedAt,
260
261
  submitLabel,
261
262
  backHref,
262
263
  cancelHref,
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../src/modules/staff/components/LeaveRequestForm.tsx"],
4
- "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { CrudForm, type CrudField } from '@open-mercato/ui/backend/CrudForm'\nimport { LookupSelect, type LookupSelectItem } from '@open-mercato/ui/backend/inputs'\nimport { DictionaryEntrySelect, type DictionarySelectLabels } from '@open-mercato/core/modules/dictionaries/components/DictionaryEntrySelect'\nimport {\n createUnavailabilityReasonEntry,\n loadUnavailabilityReasonEntries,\n type UnavailabilityReasonEntry,\n} from '@open-mercato/core/modules/planner/components/unavailabilityReasons'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { apiCall } from '@open-mercato/ui/backend/utils/apiCall'\nimport { useOrganizationScopeVersion } from '@open-mercato/shared/lib/frontend/useOrganizationScope'\n\ntype TeamMemberResponse = {\n items?: Array<Record<string, unknown>>\n}\n\ntype FeatureCheckResponse = {\n ok?: boolean\n granted?: string[]\n}\n\nexport type LeaveRequestFormValues = {\n id?: string\n memberId?: string | null\n memberLabel?: string | null\n startDate?: string | Date | null\n endDate?: string | Date | null\n timezone?: string | null\n unavailabilityReasonEntryId?: string | null\n unavailabilityReasonValue?: string | null\n note?: string | null\n}\n\nexport type LeaveRequestFormProps = {\n title: string\n submitLabel?: string\n backHref: string\n cancelHref: string\n initialValues: LeaveRequestFormValues\n onSubmit: (values: LeaveRequestFormValues) => Promise<void>\n onDelete?: () => Promise<void>\n isLoading?: boolean\n loadingMessage?: string\n allowMemberSelect?: boolean\n memberLabel?: string | null\n extraActions?: React.ReactNode\n}\n\nconst DEFAULT_TIMEZONE = 'UTC'\n\nimport { toDateInputValue } from '@open-mercato/shared/lib/date/format'\n\nexport function buildLeaveRequestPayload(\n values: LeaveRequestFormValues,\n options: { id?: string } = {},\n): Record<string, unknown> {\n const timezone = values.timezone ?? Intl.DateTimeFormat().resolvedOptions().timeZone ?? DEFAULT_TIMEZONE\n return {\n ...(options.id ? { id: options.id } : {}),\n memberId: values.memberId ?? null,\n startDate: values.startDate ?? null,\n endDate: values.endDate ?? null,\n timezone,\n unavailabilityReasonEntryId: values.unavailabilityReasonEntryId ?? null,\n unavailabilityReasonValue: values.unavailabilityReasonValue ?? null,\n note: values.note ?? null,\n }\n}\n\nexport function LeaveRequestForm(props: LeaveRequestFormProps) {\n const {\n title,\n submitLabel,\n backHref,\n cancelHref,\n initialValues,\n onSubmit,\n onDelete,\n isLoading,\n loadingMessage,\n allowMemberSelect = true,\n memberLabel,\n extraActions,\n } = props\n const t = useT()\n const scopeVersion = useOrganizationScopeVersion()\n const [memberOptions, setMemberOptions] = React.useState<LookupSelectItem[]>([])\n const [reasonEntriesById, setReasonEntriesById] = React.useState<Record<string, UnavailabilityReasonEntry>>({})\n const [canManageReasons, setCanManageReasons] = React.useState(false)\n const resolvedMemberLabel = memberLabel ?? initialValues.memberLabel ?? null\n const normalizedInitialValues = React.useMemo<LeaveRequestFormValues>(() => ({\n ...initialValues,\n startDate: toDateInputValue(initialValues.startDate),\n endDate: toDateInputValue(initialValues.endDate),\n }), [initialValues])\n\n const labels = React.useMemo(() => ({\n member: t('staff.leaveRequests.form.fields.member', 'Team member'),\n startDate: t('staff.leaveRequests.form.fields.startDate', 'Start date'),\n endDate: t('staff.leaveRequests.form.fields.endDate', 'End date'),\n reason: t('staff.leaveRequests.form.fields.reason', 'Reason'),\n note: t('staff.leaveRequests.form.fields.note', 'Note'),\n notePlaceholder: t('staff.leaveRequests.form.fields.notePlaceholder', 'Optional note'),\n }), [t])\n\n const reasonLabels = React.useMemo<DictionarySelectLabels>(() => ({\n placeholder: t('staff.leaveRequests.form.reason.placeholder', 'Select a reason'),\n addLabel: t('staff.leaveRequests.form.reason.add', 'Add reason'),\n addPrompt: t('staff.leaveRequests.form.reason.prompt', 'Name the reason'),\n dialogTitle: t('staff.leaveRequests.form.reason.dialogTitle', 'Add reason'),\n valueLabel: t('staff.leaveRequests.form.reason.valueLabel', 'Reason'),\n valuePlaceholder: t('staff.leaveRequests.form.reason.valuePlaceholder', 'Reason name'),\n labelLabel: t('staff.leaveRequests.form.reason.labelLabel', 'Label'),\n labelPlaceholder: t('staff.leaveRequests.form.reason.labelPlaceholder', 'Display label (optional)'),\n emptyError: t('staff.leaveRequests.form.reason.emptyError', 'Please enter a reason'),\n cancelLabel: t('staff.leaveRequests.form.reason.cancel', 'Cancel'),\n saveLabel: t('staff.leaveRequests.form.reason.save', 'Save'),\n saveShortcutHint: t('staff.leaveRequests.form.reason.saveShortcut', 'Cmd/Ctrl + Enter'),\n errorLoad: t('staff.leaveRequests.form.reason.errorLoad', 'Failed to load reasons'),\n errorSave: t('staff.leaveRequests.form.reason.errorSave', 'Failed to save reason'),\n loadingLabel: t('staff.leaveRequests.form.reason.loading', 'Loading\u2026'),\n manageTitle: t('staff.leaveRequests.form.reason.manage', 'Manage reasons'),\n }), [t])\n\n const fetchReasonOptions = React.useCallback(async () => {\n const entries = await loadUnavailabilityReasonEntries('member')\n const map: Record<string, UnavailabilityReasonEntry> = {}\n entries.forEach((entry) => {\n map[entry.id] = entry\n })\n setReasonEntriesById(map)\n return entries.map((entry) => ({\n value: entry.id,\n label: entry.label ?? entry.value,\n color: entry.color,\n icon: entry.icon,\n }))\n }, [])\n\n const createReasonOption = React.useCallback(\n async (input: { value: string; label?: string; color?: string | null; icon?: string | null }) => {\n const entry = await createUnavailabilityReasonEntry('member', input)\n if (!entry) return null\n setReasonEntriesById((prev) => ({ ...prev, [entry.id]: entry }))\n return {\n value: entry.id,\n label: entry.label ?? entry.value,\n color: entry.color,\n icon: entry.icon,\n }\n },\n [],\n )\n\n const fetchMemberOptions = React.useCallback(async (query?: string): Promise<LookupSelectItem[]> => {\n const params = new URLSearchParams({ page: '1', pageSize: '50' })\n if (query && query.trim().length) params.set('search', query.trim())\n const call = await apiCall<TeamMemberResponse>(`/api/staff/team-members?${params.toString()}`)\n const items = Array.isArray(call.result?.items) ? call.result.items : []\n const options = items\n .map((item) => {\n const id = typeof item?.id === 'string' ? item.id : null\n const displayName = typeof item?.displayName === 'string'\n ? item.displayName\n : typeof item?.display_name === 'string'\n ? item.display_name\n : null\n if (!id || !displayName) return null\n return { id, title: displayName }\n })\n .filter((option): option is LookupSelectItem => option !== null)\n setMemberOptions(options)\n return options\n }, [])\n\n React.useEffect(() => {\n if (!allowMemberSelect) return\n fetchMemberOptions().catch(() => {})\n }, [allowMemberSelect, fetchMemberOptions, scopeVersion])\n\n React.useEffect(() => {\n let cancelled = false\n async function loadPermissions() {\n try {\n const call = await apiCall<FeatureCheckResponse>('/api/auth/feature-check', {\n method: 'POST',\n headers: { 'content-type': 'application/json' },\n body: JSON.stringify({ features: ['dictionaries.manage'] }),\n })\n const granted = Array.isArray(call.result?.granted) ? call.result?.granted ?? [] : []\n if (!cancelled) {\n setCanManageReasons(granted.includes('dictionaries.manage'))\n }\n } catch {\n if (!cancelled) setCanManageReasons(false)\n }\n }\n void loadPermissions()\n return () => { cancelled = true }\n }, [scopeVersion])\n\n React.useEffect(() => {\n const selected = typeof initialValues.memberId === 'string' ? initialValues.memberId : null\n if (!selected || !allowMemberSelect) return\n if (memberOptions.some((option) => option.id === selected)) return\n const selectedId = selected\n let cancelled = false\n async function loadMember() {\n try {\n const params = new URLSearchParams({ page: '1', pageSize: '1' })\n params.set('ids', selectedId)\n const call = await apiCall<TeamMemberResponse>(`/api/staff/team-members?${params.toString()}`)\n const item = Array.isArray(call.result?.items) ? call.result.items[0] : null\n const id = typeof item?.id === 'string' ? item.id : null\n const displayName = typeof item?.displayName === 'string'\n ? item.displayName\n : typeof item?.display_name === 'string'\n ? item.display_name\n : null\n if (!id || !displayName) return\n if (!cancelled) {\n setMemberOptions((prev) => {\n if (prev.some((option) => option.id === id)) return prev\n return [{ id, title: displayName }, ...prev]\n })\n }\n } catch {\n if (!cancelled) setMemberOptions((prev) => prev)\n }\n }\n loadMember()\n return () => { cancelled = true }\n }, [allowMemberSelect, initialValues.memberId, memberOptions])\n\n const fields = React.useMemo<CrudField[]>(() => {\n const baseFields: CrudField[] = []\n baseFields.push({\n id: 'memberId',\n label: labels.member,\n type: 'custom',\n disabled: !allowMemberSelect,\n component: ({ value, setValue, disabled }) => {\n if (!allowMemberSelect) {\n return (\n <div className=\"text-sm text-muted-foreground\">\n {resolvedMemberLabel ?? t('staff.leaveRequests.form.fields.member.self', 'Your profile')}\n </div>\n )\n }\n return (\n <LookupSelect\n value={typeof value === 'string' ? value : null}\n onChange={(next) => setValue(next)}\n fetchOptions={fetchMemberOptions}\n options={memberOptions}\n placeholder={labels.member}\n disabled={disabled}\n />\n )\n },\n })\n baseFields.push(\n { id: 'startDate', label: labels.startDate, type: 'date', required: true, layout: 'half' },\n { id: 'endDate', label: labels.endDate, type: 'date', required: true, layout: 'half' },\n {\n id: 'unavailabilityReasonEntryId',\n label: labels.reason,\n type: 'custom',\n component: ({ value, setValue, setFormValue }) => (\n <DictionaryEntrySelect\n value={typeof value === 'string' ? value : undefined}\n onChange={(next) => {\n setValue(next ?? null)\n if (setFormValue) {\n const entry = next ? reasonEntriesById[next] : null\n setFormValue('unavailabilityReasonValue', entry?.value ?? null)\n }\n }}\n fetchOptions={fetchReasonOptions}\n createOption={canManageReasons ? createReasonOption : undefined}\n labels={reasonLabels}\n selectClassName=\"w-full\"\n sortOptions=\"none\"\n manageHref={canManageReasons ? '/backend/config/dictionaries' : undefined}\n allowInlineCreate={canManageReasons}\n showManage={canManageReasons}\n />\n ),\n },\n {\n id: 'note',\n label: labels.note,\n type: 'textarea',\n placeholder: labels.notePlaceholder,\n },\n )\n return baseFields\n }, [\n allowMemberSelect,\n fetchMemberOptions,\n fetchReasonOptions,\n createReasonOption,\n labels,\n memberOptions,\n reasonEntriesById,\n reasonLabels,\n resolvedMemberLabel,\n t,\n canManageReasons,\n ])\n\n return (\n <CrudForm\n title={title}\n fields={fields}\n initialValues={normalizedInitialValues}\n submitLabel={submitLabel}\n backHref={backHref}\n cancelHref={cancelHref}\n onSubmit={onSubmit}\n onDelete={onDelete}\n extraActions={extraActions}\n isLoading={isLoading}\n loadingMessage={loadingMessage}\n />\n )\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { CrudForm, type CrudField } from '@open-mercato/ui/backend/CrudForm'\nimport { LookupSelect, type LookupSelectItem } from '@open-mercato/ui/backend/inputs'\nimport { DictionaryEntrySelect, type DictionarySelectLabels } from '@open-mercato/core/modules/dictionaries/components/DictionaryEntrySelect'\nimport {\n createUnavailabilityReasonEntry,\n loadUnavailabilityReasonEntries,\n type UnavailabilityReasonEntry,\n} from '@open-mercato/core/modules/planner/components/unavailabilityReasons'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { apiCall } from '@open-mercato/ui/backend/utils/apiCall'\nimport { useOrganizationScopeVersion } from '@open-mercato/shared/lib/frontend/useOrganizationScope'\n\ntype TeamMemberResponse = {\n items?: Array<Record<string, unknown>>\n}\n\ntype FeatureCheckResponse = {\n ok?: boolean\n granted?: string[]\n}\n\nexport type LeaveRequestFormValues = {\n id?: string\n memberId?: string | null\n memberLabel?: string | null\n startDate?: string | Date | null\n endDate?: string | Date | null\n timezone?: string | null\n unavailabilityReasonEntryId?: string | null\n unavailabilityReasonValue?: string | null\n note?: string | null\n updatedAt?: string | null\n}\n\nexport type LeaveRequestFormProps = {\n title: string\n submitLabel?: string\n backHref: string\n cancelHref: string\n initialValues: LeaveRequestFormValues\n onSubmit: (values: LeaveRequestFormValues) => Promise<void>\n onDelete?: () => Promise<void>\n isLoading?: boolean\n loadingMessage?: string\n allowMemberSelect?: boolean\n memberLabel?: string | null\n extraActions?: React.ReactNode\n}\n\nconst DEFAULT_TIMEZONE = 'UTC'\n\nimport { toDateInputValue } from '@open-mercato/shared/lib/date/format'\n\nexport function buildLeaveRequestPayload(\n values: LeaveRequestFormValues,\n options: { id?: string } = {},\n): Record<string, unknown> {\n const timezone = values.timezone ?? Intl.DateTimeFormat().resolvedOptions().timeZone ?? DEFAULT_TIMEZONE\n return {\n ...(options.id ? { id: options.id } : {}),\n memberId: values.memberId ?? null,\n startDate: values.startDate ?? null,\n endDate: values.endDate ?? null,\n timezone,\n unavailabilityReasonEntryId: values.unavailabilityReasonEntryId ?? null,\n unavailabilityReasonValue: values.unavailabilityReasonValue ?? null,\n note: values.note ?? null,\n }\n}\n\nexport function LeaveRequestForm(props: LeaveRequestFormProps) {\n const {\n title,\n submitLabel,\n backHref,\n cancelHref,\n initialValues,\n onSubmit,\n onDelete,\n isLoading,\n loadingMessage,\n allowMemberSelect = true,\n memberLabel,\n extraActions,\n } = props\n const t = useT()\n const scopeVersion = useOrganizationScopeVersion()\n const [memberOptions, setMemberOptions] = React.useState<LookupSelectItem[]>([])\n const [reasonEntriesById, setReasonEntriesById] = React.useState<Record<string, UnavailabilityReasonEntry>>({})\n const [canManageReasons, setCanManageReasons] = React.useState(false)\n const resolvedMemberLabel = memberLabel ?? initialValues.memberLabel ?? null\n const normalizedInitialValues = React.useMemo<LeaveRequestFormValues>(() => ({\n ...initialValues,\n startDate: toDateInputValue(initialValues.startDate),\n endDate: toDateInputValue(initialValues.endDate),\n }), [initialValues])\n\n const labels = React.useMemo(() => ({\n member: t('staff.leaveRequests.form.fields.member', 'Team member'),\n startDate: t('staff.leaveRequests.form.fields.startDate', 'Start date'),\n endDate: t('staff.leaveRequests.form.fields.endDate', 'End date'),\n reason: t('staff.leaveRequests.form.fields.reason', 'Reason'),\n note: t('staff.leaveRequests.form.fields.note', 'Note'),\n notePlaceholder: t('staff.leaveRequests.form.fields.notePlaceholder', 'Optional note'),\n }), [t])\n\n const reasonLabels = React.useMemo<DictionarySelectLabels>(() => ({\n placeholder: t('staff.leaveRequests.form.reason.placeholder', 'Select a reason'),\n addLabel: t('staff.leaveRequests.form.reason.add', 'Add reason'),\n addPrompt: t('staff.leaveRequests.form.reason.prompt', 'Name the reason'),\n dialogTitle: t('staff.leaveRequests.form.reason.dialogTitle', 'Add reason'),\n valueLabel: t('staff.leaveRequests.form.reason.valueLabel', 'Reason'),\n valuePlaceholder: t('staff.leaveRequests.form.reason.valuePlaceholder', 'Reason name'),\n labelLabel: t('staff.leaveRequests.form.reason.labelLabel', 'Label'),\n labelPlaceholder: t('staff.leaveRequests.form.reason.labelPlaceholder', 'Display label (optional)'),\n emptyError: t('staff.leaveRequests.form.reason.emptyError', 'Please enter a reason'),\n cancelLabel: t('staff.leaveRequests.form.reason.cancel', 'Cancel'),\n saveLabel: t('staff.leaveRequests.form.reason.save', 'Save'),\n saveShortcutHint: t('staff.leaveRequests.form.reason.saveShortcut', 'Cmd/Ctrl + Enter'),\n errorLoad: t('staff.leaveRequests.form.reason.errorLoad', 'Failed to load reasons'),\n errorSave: t('staff.leaveRequests.form.reason.errorSave', 'Failed to save reason'),\n loadingLabel: t('staff.leaveRequests.form.reason.loading', 'Loading\u2026'),\n manageTitle: t('staff.leaveRequests.form.reason.manage', 'Manage reasons'),\n }), [t])\n\n const fetchReasonOptions = React.useCallback(async () => {\n const entries = await loadUnavailabilityReasonEntries('member')\n const map: Record<string, UnavailabilityReasonEntry> = {}\n entries.forEach((entry) => {\n map[entry.id] = entry\n })\n setReasonEntriesById(map)\n return entries.map((entry) => ({\n value: entry.id,\n label: entry.label ?? entry.value,\n color: entry.color,\n icon: entry.icon,\n }))\n }, [])\n\n const createReasonOption = React.useCallback(\n async (input: { value: string; label?: string; color?: string | null; icon?: string | null }) => {\n const entry = await createUnavailabilityReasonEntry('member', input)\n if (!entry) return null\n setReasonEntriesById((prev) => ({ ...prev, [entry.id]: entry }))\n return {\n value: entry.id,\n label: entry.label ?? entry.value,\n color: entry.color,\n icon: entry.icon,\n }\n },\n [],\n )\n\n const fetchMemberOptions = React.useCallback(async (query?: string): Promise<LookupSelectItem[]> => {\n const params = new URLSearchParams({ page: '1', pageSize: '50' })\n if (query && query.trim().length) params.set('search', query.trim())\n const call = await apiCall<TeamMemberResponse>(`/api/staff/team-members?${params.toString()}`)\n const items = Array.isArray(call.result?.items) ? call.result.items : []\n const options = items\n .map((item) => {\n const id = typeof item?.id === 'string' ? item.id : null\n const displayName = typeof item?.displayName === 'string'\n ? item.displayName\n : typeof item?.display_name === 'string'\n ? item.display_name\n : null\n if (!id || !displayName) return null\n return { id, title: displayName }\n })\n .filter((option): option is LookupSelectItem => option !== null)\n setMemberOptions(options)\n return options\n }, [])\n\n React.useEffect(() => {\n if (!allowMemberSelect) return\n fetchMemberOptions().catch(() => {})\n }, [allowMemberSelect, fetchMemberOptions, scopeVersion])\n\n React.useEffect(() => {\n let cancelled = false\n async function loadPermissions() {\n try {\n const call = await apiCall<FeatureCheckResponse>('/api/auth/feature-check', {\n method: 'POST',\n headers: { 'content-type': 'application/json' },\n body: JSON.stringify({ features: ['dictionaries.manage'] }),\n })\n const granted = Array.isArray(call.result?.granted) ? call.result?.granted ?? [] : []\n if (!cancelled) {\n setCanManageReasons(granted.includes('dictionaries.manage'))\n }\n } catch {\n if (!cancelled) setCanManageReasons(false)\n }\n }\n void loadPermissions()\n return () => { cancelled = true }\n }, [scopeVersion])\n\n React.useEffect(() => {\n const selected = typeof initialValues.memberId === 'string' ? initialValues.memberId : null\n if (!selected || !allowMemberSelect) return\n if (memberOptions.some((option) => option.id === selected)) return\n const selectedId = selected\n let cancelled = false\n async function loadMember() {\n try {\n const params = new URLSearchParams({ page: '1', pageSize: '1' })\n params.set('ids', selectedId)\n const call = await apiCall<TeamMemberResponse>(`/api/staff/team-members?${params.toString()}`)\n const item = Array.isArray(call.result?.items) ? call.result.items[0] : null\n const id = typeof item?.id === 'string' ? item.id : null\n const displayName = typeof item?.displayName === 'string'\n ? item.displayName\n : typeof item?.display_name === 'string'\n ? item.display_name\n : null\n if (!id || !displayName) return\n if (!cancelled) {\n setMemberOptions((prev) => {\n if (prev.some((option) => option.id === id)) return prev\n return [{ id, title: displayName }, ...prev]\n })\n }\n } catch {\n if (!cancelled) setMemberOptions((prev) => prev)\n }\n }\n loadMember()\n return () => { cancelled = true }\n }, [allowMemberSelect, initialValues.memberId, memberOptions])\n\n const fields = React.useMemo<CrudField[]>(() => {\n const baseFields: CrudField[] = []\n baseFields.push({\n id: 'memberId',\n label: labels.member,\n type: 'custom',\n disabled: !allowMemberSelect,\n component: ({ value, setValue, disabled }) => {\n if (!allowMemberSelect) {\n return (\n <div className=\"text-sm text-muted-foreground\">\n {resolvedMemberLabel ?? t('staff.leaveRequests.form.fields.member.self', 'Your profile')}\n </div>\n )\n }\n return (\n <LookupSelect\n value={typeof value === 'string' ? value : null}\n onChange={(next) => setValue(next)}\n fetchOptions={fetchMemberOptions}\n options={memberOptions}\n placeholder={labels.member}\n disabled={disabled}\n />\n )\n },\n })\n baseFields.push(\n { id: 'startDate', label: labels.startDate, type: 'date', required: true, layout: 'half' },\n { id: 'endDate', label: labels.endDate, type: 'date', required: true, layout: 'half' },\n {\n id: 'unavailabilityReasonEntryId',\n label: labels.reason,\n type: 'custom',\n component: ({ value, setValue, setFormValue }) => (\n <DictionaryEntrySelect\n value={typeof value === 'string' ? value : undefined}\n onChange={(next) => {\n setValue(next ?? null)\n if (setFormValue) {\n const entry = next ? reasonEntriesById[next] : null\n setFormValue('unavailabilityReasonValue', entry?.value ?? null)\n }\n }}\n fetchOptions={fetchReasonOptions}\n createOption={canManageReasons ? createReasonOption : undefined}\n labels={reasonLabels}\n selectClassName=\"w-full\"\n sortOptions=\"none\"\n manageHref={canManageReasons ? '/backend/config/dictionaries' : undefined}\n allowInlineCreate={canManageReasons}\n showManage={canManageReasons}\n />\n ),\n },\n {\n id: 'note',\n label: labels.note,\n type: 'textarea',\n placeholder: labels.notePlaceholder,\n },\n )\n return baseFields\n }, [\n allowMemberSelect,\n fetchMemberOptions,\n fetchReasonOptions,\n createReasonOption,\n labels,\n memberOptions,\n reasonEntriesById,\n reasonLabels,\n resolvedMemberLabel,\n t,\n canManageReasons,\n ])\n\n return (\n <CrudForm\n title={title}\n fields={fields}\n initialValues={normalizedInitialValues}\n optimisticLockUpdatedAt={normalizedInitialValues.updatedAt}\n submitLabel={submitLabel}\n backHref={backHref}\n cancelHref={cancelHref}\n onSubmit={onSubmit}\n onDelete={onDelete}\n extraActions={extraActions}\n isLoading={isLoading}\n loadingMessage={loadingMessage}\n />\n )\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -57,6 +57,7 @@ function TeamForm(props) {
57
57
  groups,
58
58
  entityId: E.staff.staff_team,
59
59
  initialValues,
60
+ optimisticLockUpdatedAt: initialValues.updatedAt,
60
61
  onSubmit,
61
62
  onDelete,
62
63
  isLoading,
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../src/modules/staff/components/TeamForm.tsx"],
4
- "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { CrudForm, type CrudField, type CrudFormGroup } from '@open-mercato/ui/backend/CrudForm'\nimport { collectCustomFieldValues } from '@open-mercato/ui/backend/utils/customFieldValues'\nimport { normalizeCustomFieldValues } from '@open-mercato/shared/lib/custom-fields/normalize'\nimport { E } from '#generated/entities.ids.generated'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\n\nexport type TeamFormValues = {\n id?: string\n name: string\n description?: string | null\n isActive?: boolean\n} & Record<string, unknown>\n\nexport type TeamFormProps = {\n title: string\n submitLabel?: string\n backHref: string\n cancelHref: string\n initialValues: TeamFormValues\n onSubmit: (values: TeamFormValues) => Promise<void>\n onDelete?: () => Promise<void>\n isLoading?: boolean\n loadingMessage?: string\n extraActions?: React.ReactNode\n}\n\nconst normalizeCustomFieldSubmitValue = (value: unknown): unknown => {\n const normalized = normalizeCustomFieldValues({ value })\n return normalized.value\n}\n\nexport const buildTeamPayload = (\n values: TeamFormValues,\n options: { id?: string } = {},\n): Record<string, unknown> => {\n const name = typeof values.name === 'string' ? values.name.trim() : ''\n const description = typeof values.description === 'string' && values.description.trim().length\n ? values.description.trim()\n : null\n const customFields = collectCustomFieldValues(values, { transform: normalizeCustomFieldSubmitValue })\n return {\n ...(options.id ? { id: options.id } : {}),\n name,\n description,\n isActive: values.isActive ?? true,\n ...(Object.keys(customFields).length ? { customFields } : {}),\n }\n}\n\nexport function TeamForm(props: TeamFormProps) {\n const {\n title,\n submitLabel,\n backHref,\n cancelHref,\n initialValues,\n onSubmit,\n onDelete,\n isLoading,\n loadingMessage,\n extraActions,\n } = props\n const t = useT()\n\n const fields = React.useMemo<CrudField[]>(() => [\n { id: 'name', label: t('staff.teams.form.fields.name', 'Name'), type: 'text', required: true },\n { id: 'description', label: t('staff.teams.form.fields.description', 'Description'), type: 'richtext', editor: 'html' },\n { id: 'isActive', label: t('staff.teams.form.fields.active', 'Active'), type: 'checkbox' },\n ], [t])\n\n const groups = React.useMemo<CrudFormGroup[]>(() => [\n { id: 'details', fields: ['name', 'description', 'isActive'] },\n { id: 'custom', title: t('entities.customFields.title', 'Custom Attributes'), column: 2, kind: 'customFields' },\n ], [t])\n\n return (\n <CrudForm<TeamFormValues>\n title={title}\n backHref={backHref}\n cancelHref={cancelHref}\n versionHistory={initialValues.id\n ? { resourceKind: 'staff.team', resourceId: String(initialValues.id) }\n : undefined}\n submitLabel={submitLabel}\n fields={fields}\n groups={groups}\n entityId={E.staff.staff_team}\n initialValues={initialValues}\n onSubmit={onSubmit}\n onDelete={onDelete}\n isLoading={isLoading}\n loadingMessage={loadingMessage}\n extraActions={extraActions}\n />\n )\n}\n"],
5
- "mappings": ";AA+EI;AA7EJ,YAAY,WAAW;AACvB,SAAS,gBAAoD;AAC7D,SAAS,gCAAgC;AACzC,SAAS,kCAAkC;AAC3C,SAAS,SAAS;AAClB,SAAS,YAAY;AAsBrB,MAAM,kCAAkC,CAAC,UAA4B;AACnE,QAAM,aAAa,2BAA2B,EAAE,MAAM,CAAC;AACvD,SAAO,WAAW;AACpB;AAEO,MAAM,mBAAmB,CAC9B,QACA,UAA2B,CAAC,MACA;AAC5B,QAAM,OAAO,OAAO,OAAO,SAAS,WAAW,OAAO,KAAK,KAAK,IAAI;AACpE,QAAM,cAAc,OAAO,OAAO,gBAAgB,YAAY,OAAO,YAAY,KAAK,EAAE,SACpF,OAAO,YAAY,KAAK,IACxB;AACJ,QAAM,eAAe,yBAAyB,QAAQ,EAAE,WAAW,gCAAgC,CAAC;AACpG,SAAO;AAAA,IACL,GAAI,QAAQ,KAAK,EAAE,IAAI,QAAQ,GAAG,IAAI,CAAC;AAAA,IACvC;AAAA,IACA;AAAA,IACA,UAAU,OAAO,YAAY;AAAA,IAC7B,GAAI,OAAO,KAAK,YAAY,EAAE,SAAS,EAAE,aAAa,IAAI,CAAC;AAAA,EAC7D;AACF;AAEO,SAAS,SAAS,OAAsB;AAC7C,QAAM;AAAA,IACJ;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF,IAAI;AACJ,QAAM,IAAI,KAAK;AAEf,QAAM,SAAS,MAAM,QAAqB,MAAM;AAAA,IAC9C,EAAE,IAAI,QAAQ,OAAO,EAAE,gCAAgC,MAAM,GAAG,MAAM,QAAQ,UAAU,KAAK;AAAA,IAC7F,EAAE,IAAI,eAAe,OAAO,EAAE,uCAAuC,aAAa,GAAG,MAAM,YAAY,QAAQ,OAAO;AAAA,IACtH,EAAE,IAAI,YAAY,OAAO,EAAE,kCAAkC,QAAQ,GAAG,MAAM,WAAW;AAAA,EAC3F,GAAG,CAAC,CAAC,CAAC;AAEN,QAAM,SAAS,MAAM,QAAyB,MAAM;AAAA,IAClD,EAAE,IAAI,WAAW,QAAQ,CAAC,QAAQ,eAAe,UAAU,EAAE;AAAA,IAC7D,EAAE,IAAI,UAAU,OAAO,EAAE,+BAA+B,mBAAmB,GAAG,QAAQ,GAAG,MAAM,eAAe;AAAA,EAChH,GAAG,CAAC,CAAC,CAAC;AAEN,SACE;AAAA,IAAC;AAAA;AAAA,MACC;AAAA,MACA;AAAA,MACA;AAAA,MACA,gBAAgB,cAAc,KAC1B,EAAE,cAAc,cAAc,YAAY,OAAO,cAAc,EAAE,EAAE,IACnE;AAAA,MACJ;AAAA,MACA;AAAA,MACA;AAAA,MACA,UAAU,EAAE,MAAM;AAAA,MAClB;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA;AAAA,EACF;AAEJ;",
4
+ "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { CrudForm, type CrudField, type CrudFormGroup } from '@open-mercato/ui/backend/CrudForm'\nimport { collectCustomFieldValues } from '@open-mercato/ui/backend/utils/customFieldValues'\nimport { normalizeCustomFieldValues } from '@open-mercato/shared/lib/custom-fields/normalize'\nimport { E } from '#generated/entities.ids.generated'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\n\nexport type TeamFormValues = {\n id?: string\n name: string\n description?: string | null\n isActive?: boolean\n updatedAt?: string | null\n} & Record<string, unknown>\n\nexport type TeamFormProps = {\n title: string\n submitLabel?: string\n backHref: string\n cancelHref: string\n initialValues: TeamFormValues\n onSubmit: (values: TeamFormValues) => Promise<void>\n onDelete?: () => Promise<void>\n isLoading?: boolean\n loadingMessage?: string\n extraActions?: React.ReactNode\n}\n\nconst normalizeCustomFieldSubmitValue = (value: unknown): unknown => {\n const normalized = normalizeCustomFieldValues({ value })\n return normalized.value\n}\n\nexport const buildTeamPayload = (\n values: TeamFormValues,\n options: { id?: string } = {},\n): Record<string, unknown> => {\n const name = typeof values.name === 'string' ? values.name.trim() : ''\n const description = typeof values.description === 'string' && values.description.trim().length\n ? values.description.trim()\n : null\n const customFields = collectCustomFieldValues(values, { transform: normalizeCustomFieldSubmitValue })\n return {\n ...(options.id ? { id: options.id } : {}),\n name,\n description,\n isActive: values.isActive ?? true,\n ...(Object.keys(customFields).length ? { customFields } : {}),\n }\n}\n\nexport function TeamForm(props: TeamFormProps) {\n const {\n title,\n submitLabel,\n backHref,\n cancelHref,\n initialValues,\n onSubmit,\n onDelete,\n isLoading,\n loadingMessage,\n extraActions,\n } = props\n const t = useT()\n\n const fields = React.useMemo<CrudField[]>(() => [\n { id: 'name', label: t('staff.teams.form.fields.name', 'Name'), type: 'text', required: true },\n { id: 'description', label: t('staff.teams.form.fields.description', 'Description'), type: 'richtext', editor: 'html' },\n { id: 'isActive', label: t('staff.teams.form.fields.active', 'Active'), type: 'checkbox' },\n ], [t])\n\n const groups = React.useMemo<CrudFormGroup[]>(() => [\n { id: 'details', fields: ['name', 'description', 'isActive'] },\n { id: 'custom', title: t('entities.customFields.title', 'Custom Attributes'), column: 2, kind: 'customFields' },\n ], [t])\n\n return (\n <CrudForm<TeamFormValues>\n title={title}\n backHref={backHref}\n cancelHref={cancelHref}\n versionHistory={initialValues.id\n ? { resourceKind: 'staff.team', resourceId: String(initialValues.id) }\n : undefined}\n submitLabel={submitLabel}\n fields={fields}\n groups={groups}\n entityId={E.staff.staff_team}\n initialValues={initialValues}\n optimisticLockUpdatedAt={initialValues.updatedAt}\n onSubmit={onSubmit}\n onDelete={onDelete}\n isLoading={isLoading}\n loadingMessage={loadingMessage}\n extraActions={extraActions}\n />\n )\n}\n"],
5
+ "mappings": ";AAgFI;AA9EJ,YAAY,WAAW;AACvB,SAAS,gBAAoD;AAC7D,SAAS,gCAAgC;AACzC,SAAS,kCAAkC;AAC3C,SAAS,SAAS;AAClB,SAAS,YAAY;AAuBrB,MAAM,kCAAkC,CAAC,UAA4B;AACnE,QAAM,aAAa,2BAA2B,EAAE,MAAM,CAAC;AACvD,SAAO,WAAW;AACpB;AAEO,MAAM,mBAAmB,CAC9B,QACA,UAA2B,CAAC,MACA;AAC5B,QAAM,OAAO,OAAO,OAAO,SAAS,WAAW,OAAO,KAAK,KAAK,IAAI;AACpE,QAAM,cAAc,OAAO,OAAO,gBAAgB,YAAY,OAAO,YAAY,KAAK,EAAE,SACpF,OAAO,YAAY,KAAK,IACxB;AACJ,QAAM,eAAe,yBAAyB,QAAQ,EAAE,WAAW,gCAAgC,CAAC;AACpG,SAAO;AAAA,IACL,GAAI,QAAQ,KAAK,EAAE,IAAI,QAAQ,GAAG,IAAI,CAAC;AAAA,IACvC;AAAA,IACA;AAAA,IACA,UAAU,OAAO,YAAY;AAAA,IAC7B,GAAI,OAAO,KAAK,YAAY,EAAE,SAAS,EAAE,aAAa,IAAI,CAAC;AAAA,EAC7D;AACF;AAEO,SAAS,SAAS,OAAsB;AAC7C,QAAM;AAAA,IACJ;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF,IAAI;AACJ,QAAM,IAAI,KAAK;AAEf,QAAM,SAAS,MAAM,QAAqB,MAAM;AAAA,IAC9C,EAAE,IAAI,QAAQ,OAAO,EAAE,gCAAgC,MAAM,GAAG,MAAM,QAAQ,UAAU,KAAK;AAAA,IAC7F,EAAE,IAAI,eAAe,OAAO,EAAE,uCAAuC,aAAa,GAAG,MAAM,YAAY,QAAQ,OAAO;AAAA,IACtH,EAAE,IAAI,YAAY,OAAO,EAAE,kCAAkC,QAAQ,GAAG,MAAM,WAAW;AAAA,EAC3F,GAAG,CAAC,CAAC,CAAC;AAEN,QAAM,SAAS,MAAM,QAAyB,MAAM;AAAA,IAClD,EAAE,IAAI,WAAW,QAAQ,CAAC,QAAQ,eAAe,UAAU,EAAE;AAAA,IAC7D,EAAE,IAAI,UAAU,OAAO,EAAE,+BAA+B,mBAAmB,GAAG,QAAQ,GAAG,MAAM,eAAe;AAAA,EAChH,GAAG,CAAC,CAAC,CAAC;AAEN,SACE;AAAA,IAAC;AAAA;AAAA,MACC;AAAA,MACA;AAAA,MACA;AAAA,MACA,gBAAgB,cAAc,KAC1B,EAAE,cAAc,cAAc,YAAY,OAAO,cAAc,EAAE,EAAE,IACnE;AAAA,MACJ;AAAA,MACA;AAAA,MACA;AAAA,MACA,UAAU,EAAE,MAAM;AAAA,MAClB;AAAA,MACA,yBAAyB,cAAc;AAAA,MACvC;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA;AAAA,EACF;AAEJ;",
6
6
  "names": []
7
7
  }
@@ -422,6 +422,7 @@ function TeamMemberForm(props) {
422
422
  groups,
423
423
  entityId: E.staff.staff_team_member,
424
424
  initialValues,
425
+ optimisticLockUpdatedAt: initialValues.updatedAt,
425
426
  onSubmit,
426
427
  onDelete,
427
428
  isLoading,