@hed-hog/operations 0.0.330 → 0.0.332

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 (320) hide show
  1. package/README.md +5 -5
  2. package/dist/controllers/operations-collaborators.controller.d.ts +58 -213
  3. package/dist/controllers/operations-collaborators.controller.d.ts.map +1 -1
  4. package/dist/controllers/operations-collaborators.controller.js +100 -0
  5. package/dist/controllers/operations-collaborators.controller.js.map +1 -1
  6. package/dist/controllers/operations-contracts.controller.d.ts +6 -6
  7. package/dist/controllers/operations-projects.controller.d.ts +25 -0
  8. package/dist/controllers/operations-projects.controller.d.ts.map +1 -1
  9. package/dist/controllers/operations-projects.controller.js +48 -0
  10. package/dist/controllers/operations-projects.controller.js.map +1 -1
  11. package/dist/controllers/operations-reports.controller.d.ts +1 -1
  12. package/dist/controllers/operations-tasks.controller.d.ts +34 -9
  13. package/dist/controllers/operations-tasks.controller.d.ts.map +1 -1
  14. package/dist/controllers/operations-tasks.controller.js +43 -32
  15. package/dist/controllers/operations-tasks.controller.js.map +1 -1
  16. package/dist/controllers/operations-timesheets.controller.d.ts +9 -9
  17. package/dist/dashboard/components/DashboardLayout.d.ts +30 -0
  18. package/dist/dashboard/components/DashboardLayout.d.ts.map +1 -0
  19. package/dist/dashboard/components/DashboardLayout.js +87 -0
  20. package/dist/dashboard/components/DashboardLayout.js.map +1 -0
  21. package/dist/dashboard/components/widget-registry.d.ts +23 -0
  22. package/dist/dashboard/components/widget-registry.d.ts.map +1 -0
  23. package/dist/dashboard/components/widget-registry.js +245 -0
  24. package/dist/dashboard/components/widget-registry.js.map +1 -0
  25. package/dist/dashboard/hooks/useDashboardData.d.ts +20 -0
  26. package/dist/dashboard/hooks/useDashboardData.d.ts.map +1 -0
  27. package/dist/dashboard/hooks/useDashboardData.js +24 -0
  28. package/dist/dashboard/hooks/useDashboardData.js.map +1 -0
  29. package/dist/dashboard/types/widgets.types.d.ts +233 -0
  30. package/dist/dashboard/types/widgets.types.d.ts.map +1 -0
  31. package/dist/dashboard/types/widgets.types.js +6 -0
  32. package/dist/dashboard/types/widgets.types.js.map +1 -0
  33. package/dist/dashboard/widgets/CapacityDistribution.d.ts +23 -0
  34. package/dist/dashboard/widgets/CapacityDistribution.d.ts.map +1 -0
  35. package/dist/dashboard/widgets/CapacityDistribution.js +11 -0
  36. package/dist/dashboard/widgets/CapacityDistribution.js.map +1 -0
  37. package/dist/dashboard/widgets/EffortByProject.d.ts +22 -0
  38. package/dist/dashboard/widgets/EffortByProject.d.ts.map +1 -0
  39. package/dist/dashboard/widgets/EffortByProject.js +11 -0
  40. package/dist/dashboard/widgets/EffortByProject.js.map +1 -0
  41. package/dist/dashboard/widgets/HeadcountByArea.d.ts +24 -0
  42. package/dist/dashboard/widgets/HeadcountByArea.d.ts.map +1 -0
  43. package/dist/dashboard/widgets/HeadcountByArea.js +11 -0
  44. package/dist/dashboard/widgets/HeadcountByArea.js.map +1 -0
  45. package/dist/dashboard/widgets/ManagedProjectsStatus.d.ts +18 -0
  46. package/dist/dashboard/widgets/ManagedProjectsStatus.d.ts.map +1 -0
  47. package/dist/dashboard/widgets/ManagedProjectsStatus.js +12 -0
  48. package/dist/dashboard/widgets/ManagedProjectsStatus.js.map +1 -0
  49. package/dist/dashboard/widgets/MyHoursPeriodKpi.d.ts +22 -0
  50. package/dist/dashboard/widgets/MyHoursPeriodKpi.d.ts.map +1 -0
  51. package/dist/dashboard/widgets/MyHoursPeriodKpi.js +12 -0
  52. package/dist/dashboard/widgets/MyHoursPeriodKpi.js.map +1 -0
  53. package/dist/dashboard/widgets/MyOpenRequestsKpi.d.ts +19 -0
  54. package/dist/dashboard/widgets/MyOpenRequestsKpi.d.ts.map +1 -0
  55. package/dist/dashboard/widgets/MyOpenRequestsKpi.js +17 -0
  56. package/dist/dashboard/widgets/MyOpenRequestsKpi.js.map +1 -0
  57. package/dist/dashboard/widgets/MyPendingRequestsList.d.ts +23 -0
  58. package/dist/dashboard/widgets/MyPendingRequestsList.d.ts.map +1 -0
  59. package/dist/dashboard/widgets/MyPendingRequestsList.js +14 -0
  60. package/dist/dashboard/widgets/MyPendingRequestsList.js.map +1 -0
  61. package/dist/dashboard/widgets/MyProjectAllocationsKpi.d.ts +22 -0
  62. package/dist/dashboard/widgets/MyProjectAllocationsKpi.d.ts.map +1 -0
  63. package/dist/dashboard/widgets/MyProjectAllocationsKpi.js +11 -0
  64. package/dist/dashboard/widgets/MyProjectAllocationsKpi.js.map +1 -0
  65. package/dist/dashboard/widgets/MyQuickActions.d.ts +23 -0
  66. package/dist/dashboard/widgets/MyQuickActions.d.ts.map +1 -0
  67. package/dist/dashboard/widgets/MyQuickActions.js +18 -0
  68. package/dist/dashboard/widgets/MyQuickActions.js.map +1 -0
  69. package/dist/dashboard/widgets/MyRelevantDeadlines.d.ts +23 -0
  70. package/dist/dashboard/widgets/MyRelevantDeadlines.d.ts.map +1 -0
  71. package/dist/dashboard/widgets/MyRelevantDeadlines.js +22 -0
  72. package/dist/dashboard/widgets/MyRelevantDeadlines.js.map +1 -0
  73. package/dist/dashboard/widgets/MyTimesheetStatusKpi.d.ts +17 -0
  74. package/dist/dashboard/widgets/MyTimesheetStatusKpi.d.ts.map +1 -0
  75. package/dist/dashboard/widgets/MyTimesheetStatusKpi.js +11 -0
  76. package/dist/dashboard/widgets/MyTimesheetStatusKpi.js.map +1 -0
  77. package/dist/dashboard/widgets/MyWeeklyJourney.d.ts +21 -0
  78. package/dist/dashboard/widgets/MyWeeklyJourney.d.ts.map +1 -0
  79. package/dist/dashboard/widgets/MyWeeklyJourney.js +19 -0
  80. package/dist/dashboard/widgets/MyWeeklyJourney.js.map +1 -0
  81. package/dist/dashboard/widgets/PortfolioCostsKpi.d.ts +19 -0
  82. package/dist/dashboard/widgets/PortfolioCostsKpi.d.ts.map +1 -0
  83. package/dist/dashboard/widgets/PortfolioCostsKpi.js +12 -0
  84. package/dist/dashboard/widgets/PortfolioCostsKpi.js.map +1 -0
  85. package/dist/dashboard/widgets/PortfolioEffortKpi.d.ts +18 -0
  86. package/dist/dashboard/widgets/PortfolioEffortKpi.d.ts.map +1 -0
  87. package/dist/dashboard/widgets/PortfolioEffortKpi.js +8 -0
  88. package/dist/dashboard/widgets/PortfolioEffortKpi.js.map +1 -0
  89. package/dist/dashboard/widgets/PortfolioProjectsKpi.d.ts +22 -0
  90. package/dist/dashboard/widgets/PortfolioProjectsKpi.d.ts.map +1 -0
  91. package/dist/dashboard/widgets/PortfolioProjectsKpi.js +56 -0
  92. package/dist/dashboard/widgets/PortfolioProjectsKpi.js.map +1 -0
  93. package/dist/dashboard/widgets/PortfolioRiskKpi.d.ts +19 -0
  94. package/dist/dashboard/widgets/PortfolioRiskKpi.d.ts.map +1 -0
  95. package/dist/dashboard/widgets/PortfolioRiskKpi.js +11 -0
  96. package/dist/dashboard/widgets/PortfolioRiskKpi.js.map +1 -0
  97. package/dist/dashboard/widgets/ProjectStatusOverview.d.ts +19 -0
  98. package/dist/dashboard/widgets/ProjectStatusOverview.d.ts.map +1 -0
  99. package/dist/dashboard/widgets/ProjectStatusOverview.js +18 -0
  100. package/dist/dashboard/widgets/ProjectStatusOverview.js.map +1 -0
  101. package/dist/dashboard/widgets/StrategicDeadlines.d.ts +24 -0
  102. package/dist/dashboard/widgets/StrategicDeadlines.d.ts.map +1 -0
  103. package/dist/dashboard/widgets/StrategicDeadlines.js +22 -0
  104. package/dist/dashboard/widgets/StrategicDeadlines.js.map +1 -0
  105. package/dist/dashboard/widgets/TeamApprovalQueue.d.ts +24 -0
  106. package/dist/dashboard/widgets/TeamApprovalQueue.d.ts.map +1 -0
  107. package/dist/dashboard/widgets/TeamApprovalQueue.js +12 -0
  108. package/dist/dashboard/widgets/TeamApprovalQueue.js.map +1 -0
  109. package/dist/dashboard/widgets/TeamCapacityKpi.d.ts +18 -0
  110. package/dist/dashboard/widgets/TeamCapacityKpi.d.ts.map +1 -0
  111. package/dist/dashboard/widgets/TeamCapacityKpi.js +19 -0
  112. package/dist/dashboard/widgets/TeamCapacityKpi.js.map +1 -0
  113. package/dist/dashboard/widgets/TeamHeadcountKpi.d.ts +22 -0
  114. package/dist/dashboard/widgets/TeamHeadcountKpi.d.ts.map +1 -0
  115. package/dist/dashboard/widgets/TeamHeadcountKpi.js +56 -0
  116. package/dist/dashboard/widgets/TeamHeadcountKpi.js.map +1 -0
  117. package/dist/dashboard/widgets/TeamHoursKpi.d.ts +19 -0
  118. package/dist/dashboard/widgets/TeamHoursKpi.d.ts.map +1 -0
  119. package/dist/dashboard/widgets/TeamHoursKpi.js +13 -0
  120. package/dist/dashboard/widgets/TeamHoursKpi.js.map +1 -0
  121. package/dist/dashboard/widgets/TeamPendingApprovalsKpi.d.ts +20 -0
  122. package/dist/dashboard/widgets/TeamPendingApprovalsKpi.d.ts.map +1 -0
  123. package/dist/dashboard/widgets/TeamPendingApprovalsKpi.js +11 -0
  124. package/dist/dashboard/widgets/TeamPendingApprovalsKpi.js.map +1 -0
  125. package/dist/dashboard/widgets/TeamUtilizationOverview.d.ts +18 -0
  126. package/dist/dashboard/widgets/TeamUtilizationOverview.d.ts.map +1 -0
  127. package/dist/dashboard/widgets/TeamUtilizationOverview.js +17 -0
  128. package/dist/dashboard/widgets/TeamUtilizationOverview.js.map +1 -0
  129. package/dist/dashboard/widgets/TeamWorkloadAlerts.d.ts +24 -0
  130. package/dist/dashboard/widgets/TeamWorkloadAlerts.d.ts.map +1 -0
  131. package/dist/dashboard/widgets/TeamWorkloadAlerts.js +19 -0
  132. package/dist/dashboard/widgets/TeamWorkloadAlerts.js.map +1 -0
  133. package/dist/dashboard/widgets/index.d.ts +24 -0
  134. package/dist/dashboard/widgets/index.d.ts.map +1 -0
  135. package/dist/dashboard/widgets/index.js +54 -0
  136. package/dist/dashboard/widgets/index.js.map +1 -0
  137. package/dist/dto/create-collaborator-invoice.dto.d.ts +11 -0
  138. package/dist/dto/create-collaborator-invoice.dto.d.ts.map +1 -0
  139. package/dist/dto/create-collaborator-invoice.dto.js +55 -0
  140. package/dist/dto/create-collaborator-invoice.dto.js.map +1 -0
  141. package/dist/dto/create-collaborator-payment.dto.d.ts +10 -0
  142. package/dist/dto/create-collaborator-payment.dto.d.ts.map +1 -0
  143. package/dist/dto/create-collaborator-payment.dto.js +50 -0
  144. package/dist/dto/create-collaborator-payment.dto.js.map +1 -0
  145. package/dist/dto/create-collaborator.dto.d.ts +0 -1
  146. package/dist/dto/create-collaborator.dto.d.ts.map +1 -1
  147. package/dist/dto/create-collaborator.dto.js +0 -6
  148. package/dist/dto/create-collaborator.dto.js.map +1 -1
  149. package/dist/dto/list-collaborator-invoice.dto.d.ts +4 -0
  150. package/dist/dto/list-collaborator-invoice.dto.d.ts.map +1 -0
  151. package/dist/dto/list-collaborator-invoice.dto.js +8 -0
  152. package/dist/dto/list-collaborator-invoice.dto.js.map +1 -0
  153. package/dist/dto/list-collaborator-payment.dto.d.ts +4 -0
  154. package/dist/dto/list-collaborator-payment.dto.d.ts.map +1 -0
  155. package/dist/dto/list-collaborator-payment.dto.js +8 -0
  156. package/dist/dto/list-collaborator-payment.dto.js.map +1 -0
  157. package/dist/dto/update-collaborator-invoice.dto.d.ts +6 -0
  158. package/dist/dto/update-collaborator-invoice.dto.d.ts.map +1 -0
  159. package/dist/dto/update-collaborator-invoice.dto.js +9 -0
  160. package/dist/dto/update-collaborator-invoice.dto.js.map +1 -0
  161. package/dist/dto/update-collaborator-payment.dto.d.ts +6 -0
  162. package/dist/dto/update-collaborator-payment.dto.d.ts.map +1 -0
  163. package/dist/dto/update-collaborator-payment.dto.js +9 -0
  164. package/dist/dto/update-collaborator-payment.dto.js.map +1 -0
  165. package/dist/index.d.ts +2 -0
  166. package/dist/index.d.ts.map +1 -1
  167. package/dist/index.js +2 -0
  168. package/dist/index.js.map +1 -1
  169. package/dist/operations.controller.d.ts +42 -0
  170. package/dist/operations.controller.d.ts.map +1 -1
  171. package/dist/operations.service.d.ts +258 -268
  172. package/dist/operations.service.d.ts.map +1 -1
  173. package/dist/operations.service.js +2381 -1341
  174. package/dist/operations.service.js.map +1 -1
  175. package/dist/operations.service.spec.js +345 -174
  176. package/dist/operations.service.spec.js.map +1 -1
  177. package/hedhog/data/dashboard_component.yaml +66 -0
  178. package/hedhog/data/dashboard_item.yaml +25 -25
  179. package/hedhog/data/menu.yaml +27 -8
  180. package/hedhog/data/route.yaml +133 -0
  181. package/hedhog/frontend/app/_components/collaborator-form-screen.tsx.ejs +78 -102
  182. package/hedhog/frontend/app/_components/collaborator-invoices-tab.tsx.ejs +443 -0
  183. package/hedhog/frontend/app/_components/collaborator-payment-history-tab.tsx.ejs +429 -0
  184. package/hedhog/frontend/app/_components/collaborator-picker.tsx.ejs +158 -0
  185. package/hedhog/frontend/app/_components/my-project-summary-screen.tsx.ejs +247 -50
  186. package/hedhog/frontend/app/_components/project-assignments-tab.tsx.ejs +643 -450
  187. package/hedhog/frontend/app/_components/project-details-screen.tsx.ejs +992 -431
  188. package/hedhog/frontend/app/_components/project-file-attachments.tsx.ejs +371 -0
  189. package/hedhog/frontend/app/_components/project-form-screen.tsx.ejs +558 -386
  190. package/hedhog/frontend/app/_components/task-detail-sheet.tsx.ejs +383 -157
  191. package/hedhog/frontend/app/_components/task-file-attachments.tsx.ejs +4 -1
  192. package/hedhog/frontend/app/_components/task-form-fields.tsx.ejs +406 -0
  193. package/hedhog/frontend/app/_components/task-form-sheet.tsx.ejs +629 -784
  194. package/hedhog/frontend/app/_components/task-info-display.tsx.ejs +137 -0
  195. package/hedhog/frontend/app/_components/timesheet-entry-create-sheet.tsx.ejs +306 -0
  196. package/hedhog/frontend/app/_lib/api.ts.ejs +155 -0
  197. package/hedhog/frontend/app/_lib/types.ts.ejs +62 -0
  198. package/hedhog/frontend/app/_lib/utils/format.ts.ejs +0 -2
  199. package/hedhog/frontend/app/_lib/utils/task-ui.ts.ejs +61 -0
  200. package/hedhog/frontend/app/approvals/page.tsx.ejs +6 -1
  201. package/hedhog/frontend/app/collaborator-types/page.tsx.ejs +6 -1
  202. package/hedhog/frontend/app/collaborators/page.tsx.ejs +59 -8
  203. package/hedhog/frontend/app/contracts/page.tsx.ejs +29 -8
  204. package/hedhog/frontend/app/dashboard/widgets/CapacityDistribution.tsx.ejs +84 -0
  205. package/hedhog/frontend/app/dashboard/widgets/EffortByProject.tsx.ejs +85 -0
  206. package/hedhog/frontend/app/dashboard/widgets/HeadcountByArea.tsx.ejs +101 -0
  207. package/hedhog/frontend/app/dashboard/widgets/ManagedProjectsStatus.tsx.ejs +113 -0
  208. package/hedhog/frontend/app/dashboard/widgets/MyHoursPeriodKpi.tsx.ejs +87 -0
  209. package/hedhog/frontend/app/dashboard/widgets/MyOpenRequestsKpi.tsx.ejs +97 -0
  210. package/hedhog/frontend/app/dashboard/widgets/MyPendingRequestsList.tsx.ejs +99 -0
  211. package/hedhog/frontend/app/dashboard/widgets/MyProjectAllocationsKpi.tsx.ejs +78 -0
  212. package/hedhog/frontend/app/dashboard/widgets/MyQuickActions.tsx.ejs +130 -0
  213. package/hedhog/frontend/app/dashboard/widgets/MyRelevantDeadlines.tsx.ejs +144 -0
  214. package/hedhog/frontend/app/dashboard/widgets/MyTimesheetStatusKpi.tsx.ejs +78 -0
  215. package/hedhog/frontend/app/dashboard/widgets/MyWeeklyJourney.tsx.ejs +99 -0
  216. package/hedhog/frontend/app/dashboard/widgets/PortfolioCostsKpi.tsx.ejs +112 -0
  217. package/hedhog/frontend/app/dashboard/widgets/PortfolioEffortKpi.tsx.ejs +93 -0
  218. package/hedhog/frontend/app/dashboard/widgets/PortfolioProjectsKpi.tsx.ejs +96 -0
  219. package/hedhog/frontend/app/dashboard/widgets/PortfolioRiskKpi.tsx.ejs +115 -0
  220. package/hedhog/frontend/app/dashboard/widgets/ProjectStatusOverview.tsx.ejs +120 -0
  221. package/hedhog/frontend/app/dashboard/widgets/StrategicDeadlines.tsx.ejs +146 -0
  222. package/hedhog/frontend/app/dashboard/widgets/TeamApprovalQueue.tsx.ejs +108 -0
  223. package/hedhog/frontend/app/dashboard/widgets/TeamCapacityKpi.tsx.ejs +97 -0
  224. package/hedhog/frontend/app/dashboard/widgets/TeamHeadcountKpi.tsx.ejs +100 -0
  225. package/hedhog/frontend/app/dashboard/widgets/TeamHoursKpi.tsx.ejs +104 -0
  226. package/hedhog/frontend/app/dashboard/widgets/TeamPendingApprovalsKpi.tsx.ejs +110 -0
  227. package/hedhog/frontend/app/dashboard/widgets/TeamUtilizationOverview.tsx.ejs +115 -0
  228. package/hedhog/frontend/app/dashboard/widgets/TeamWorkloadAlerts.tsx.ejs +117 -0
  229. package/hedhog/frontend/app/dashboard/widgets/index.ts.ejs +26 -0
  230. package/hedhog/frontend/app/departments/page.tsx.ejs +6 -1
  231. package/hedhog/frontend/app/my-projects/page.tsx.ejs +30 -12
  232. package/hedhog/frontend/app/my-tasks/page.tsx.ejs +286 -125
  233. package/hedhog/frontend/app/project-cost-categories/page.tsx.ejs +58 -52
  234. package/hedhog/frontend/app/project-cost-types/page.tsx.ejs +58 -51
  235. package/hedhog/frontend/app/projects/page.tsx.ejs +415 -33
  236. package/hedhog/frontend/app/schedule-adjustments/page.tsx.ejs +6 -1
  237. package/hedhog/frontend/app/tasks-gantt/page.tsx.ejs +953 -0
  238. package/hedhog/frontend/app/time-off/page.tsx.ejs +6 -1
  239. package/hedhog/frontend/app/timesheets/page.tsx.ejs +10 -4
  240. package/hedhog/frontend/messages/en.json +332 -46
  241. package/hedhog/frontend/messages/operations/en.json +61 -52
  242. package/hedhog/frontend/messages/operations/pt.json +59 -43
  243. package/hedhog/frontend/messages/pt.json +332 -46
  244. package/hedhog/frontend/widgets/capacity-distribution.tsx.ejs +17 -0
  245. package/hedhog/frontend/widgets/effort-by-project.tsx.ejs +17 -0
  246. package/hedhog/frontend/widgets/headcount-by-area.tsx.ejs +17 -0
  247. package/hedhog/frontend/widgets/index.ts.ejs +25 -0
  248. package/hedhog/frontend/widgets/managed-projects-status.tsx.ejs +17 -0
  249. package/hedhog/frontend/widgets/my-hours-period-kpi.tsx.ejs +17 -0
  250. package/hedhog/frontend/widgets/my-open-requests-kpi.tsx.ejs +17 -0
  251. package/hedhog/frontend/widgets/my-pending-requests-list.tsx.ejs +17 -0
  252. package/hedhog/frontend/widgets/my-project-allocations-kpi.tsx.ejs +17 -0
  253. package/hedhog/frontend/widgets/my-quick-actions.tsx.ejs +17 -0
  254. package/hedhog/frontend/widgets/my-relevant-deadlines.tsx.ejs +17 -0
  255. package/hedhog/frontend/widgets/my-timesheet-status-kpi.tsx.ejs +17 -0
  256. package/hedhog/frontend/widgets/my-weekly-journey.tsx.ejs +17 -0
  257. package/hedhog/frontend/widgets/portfolio-costs-kpi.tsx.ejs +17 -0
  258. package/hedhog/frontend/widgets/portfolio-effort-kpi.tsx.ejs +17 -0
  259. package/hedhog/frontend/widgets/portfolio-projects-kpi.tsx.ejs +17 -0
  260. package/hedhog/frontend/widgets/portfolio-risk-kpi.tsx.ejs +17 -0
  261. package/hedhog/frontend/widgets/project-status-overview.tsx.ejs +17 -0
  262. package/hedhog/frontend/widgets/shared-operations-widget.tsx.ejs +170 -0
  263. package/hedhog/frontend/widgets/strategic-deadlines.tsx.ejs +17 -0
  264. package/hedhog/frontend/widgets/team-approval-queue.tsx.ejs +17 -0
  265. package/hedhog/frontend/widgets/team-capacity-kpi.tsx.ejs +17 -0
  266. package/hedhog/frontend/widgets/team-headcount-kpi.tsx.ejs +17 -0
  267. package/hedhog/frontend/widgets/team-hours-kpi.tsx.ejs +17 -0
  268. package/hedhog/frontend/widgets/team-pending-approvals-kpi.tsx.ejs +17 -0
  269. package/hedhog/frontend/widgets/team-utilization-overview.tsx.ejs +17 -0
  270. package/hedhog/frontend/widgets/team-workload-alerts.tsx.ejs +17 -0
  271. package/hedhog/table/operations_collaborator.yaml +8 -13
  272. package/hedhog/table/operations_collaborator_invoice.yaml +35 -0
  273. package/hedhog/table/operations_collaborator_payment.yaml +32 -0
  274. package/hedhog/table/operations_project.yaml +1 -1
  275. package/hedhog/table/operations_project_file.yaml +23 -0
  276. package/hedhog/table/operations_task.yaml +76 -69
  277. package/hedhog/table/operations_task_activity.yaml +51 -0
  278. package/package.json +6 -5
  279. package/src/controllers/operations-collaborators.controller.ts +117 -8
  280. package/src/controllers/operations-projects.controller.ts +41 -8
  281. package/src/controllers/operations-tasks.controller.ts +156 -166
  282. package/src/dashboard/README.md +214 -0
  283. package/src/dashboard/components/DashboardLayout.tsx +131 -0
  284. package/src/dashboard/components/widget-registry.ts +255 -0
  285. package/src/dashboard/hooks/useDashboardData.ts +29 -0
  286. package/src/dashboard/types/widgets.types.ts +237 -0
  287. package/src/dashboard/widgets/CapacityDistribution.tsx +56 -0
  288. package/src/dashboard/widgets/EffortByProject.tsx +51 -0
  289. package/src/dashboard/widgets/HeadcountByArea.tsx +57 -0
  290. package/src/dashboard/widgets/ManagedProjectsStatus.tsx +53 -0
  291. package/src/dashboard/widgets/MyHoursPeriodKpi.tsx +87 -0
  292. package/src/dashboard/widgets/MyOpenRequestsKpi.tsx +51 -0
  293. package/src/dashboard/widgets/MyPendingRequestsList.tsx +63 -0
  294. package/src/dashboard/widgets/MyProjectAllocationsKpi.tsx +57 -0
  295. package/src/dashboard/widgets/MyQuickActions.tsx +62 -0
  296. package/src/dashboard/widgets/MyRelevantDeadlines.tsx +84 -0
  297. package/src/dashboard/widgets/MyTimesheetStatusKpi.tsx +65 -0
  298. package/src/dashboard/widgets/MyWeeklyJourney.tsx +57 -0
  299. package/src/dashboard/widgets/PortfolioCostsKpi.tsx +48 -0
  300. package/src/dashboard/widgets/PortfolioEffortKpi.tsx +41 -0
  301. package/src/dashboard/widgets/PortfolioRiskKpi.tsx +50 -0
  302. package/src/dashboard/widgets/ProjectStatusOverview.tsx +52 -0
  303. package/src/dashboard/widgets/StrategicDeadlines.tsx +93 -0
  304. package/src/dashboard/widgets/TeamApprovalQueue.tsx +70 -0
  305. package/src/dashboard/widgets/TeamCapacityKpi.tsx +50 -0
  306. package/src/dashboard/widgets/TeamHoursKpi.tsx +51 -0
  307. package/src/dashboard/widgets/TeamPendingApprovalsKpi.tsx +53 -0
  308. package/src/dashboard/widgets/TeamUtilizationOverview.tsx +62 -0
  309. package/src/dashboard/widgets/TeamWorkloadAlerts.tsx +81 -0
  310. package/src/dashboard/widgets/index.ts +26 -0
  311. package/src/dto/create-collaborator-invoice.dto.ts +39 -0
  312. package/src/dto/create-collaborator-payment.dto.ts +35 -0
  313. package/src/dto/create-collaborator.dto.ts +4 -11
  314. package/src/dto/list-collaborator-invoice.dto.ts +3 -0
  315. package/src/dto/list-collaborator-payment.dto.ts +3 -0
  316. package/src/dto/update-collaborator-invoice.dto.ts +6 -0
  317. package/src/dto/update-collaborator-payment.dto.ts +6 -0
  318. package/src/index.ts +3 -0
  319. package/src/operations.service.spec.ts +988 -764
  320. package/src/operations.service.ts +4689 -2624
@@ -0,0 +1,255 @@
1
+ /**
2
+ * Widget Registry - Mapa de widgets do dashboard
3
+ *
4
+ * Este arquivo centraliza todas as informações de widgets para o dashboard da operations.
5
+ * Os widgets são importados dinamicamente em runtime.
6
+ */
7
+
8
+ export interface WidgetInfo {
9
+ slug: string;
10
+ title: string;
11
+ component: string;
12
+ roleSlug: string;
13
+ width: number;
14
+ height: number;
15
+ category: 'collaborator' | 'supervisor' | 'director';
16
+ }
17
+
18
+ // Widgets para Colaboradores (7)
19
+ export const COLLABORATOR_WIDGETS: WidgetInfo[] = [
20
+ {
21
+ slug: 'my-hours-period-kpi',
22
+ title: 'Minhas Horas (Período)',
23
+ component: 'MyHoursPeriodKpi',
24
+ roleSlug: 'collaborator',
25
+ width: 6,
26
+ height: 2,
27
+ category: 'collaborator',
28
+ },
29
+ {
30
+ slug: 'my-timesheet-status-kpi',
31
+ title: 'Status do Timesheet',
32
+ component: 'MyTimesheetStatusKpi',
33
+ roleSlug: 'collaborator',
34
+ width: 6,
35
+ height: 2,
36
+ category: 'collaborator',
37
+ },
38
+ {
39
+ slug: 'my-project-allocations-kpi',
40
+ title: 'Minhas Alocações',
41
+ component: 'MyProjectAllocationsKpi',
42
+ roleSlug: 'collaborator',
43
+ width: 6,
44
+ height: 2,
45
+ category: 'collaborator',
46
+ },
47
+ {
48
+ slug: 'my-open-requests-kpi',
49
+ title: 'Requisições Abertas',
50
+ component: 'MyOpenRequestsKpi',
51
+ roleSlug: 'collaborator',
52
+ width: 6,
53
+ height: 2,
54
+ category: 'collaborator',
55
+ },
56
+ {
57
+ slug: 'my-weekly-journey',
58
+ title: 'Jornada Semanal',
59
+ component: 'MyWeeklyJourney',
60
+ roleSlug: 'collaborator',
61
+ width: 12,
62
+ height: 3,
63
+ category: 'collaborator',
64
+ },
65
+ {
66
+ slug: 'my-relevant-deadlines',
67
+ title: 'Prazos Relevantes',
68
+ component: 'MyRelevantDeadlines',
69
+ roleSlug: 'collaborator',
70
+ width: 6,
71
+ height: 3,
72
+ category: 'collaborator',
73
+ },
74
+ {
75
+ slug: 'my-pending-requests-list',
76
+ title: 'Requisições Pendentes',
77
+ component: 'MyPendingRequestsList',
78
+ roleSlug: 'collaborator',
79
+ width: 6,
80
+ height: 3,
81
+ category: 'collaborator',
82
+ },
83
+ {
84
+ slug: 'my-quick-actions',
85
+ title: 'Ações Rápidas',
86
+ component: 'MyQuickActions',
87
+ roleSlug: 'collaborator',
88
+ width: 12,
89
+ height: 2,
90
+ category: 'collaborator',
91
+ },
92
+ ];
93
+
94
+ // Widgets para Supervisores (7)
95
+ export const SUPERVISOR_WIDGETS: WidgetInfo[] = [
96
+ {
97
+ slug: 'team-pending-approvals-kpi',
98
+ title: 'Aprovações Pendentes',
99
+ component: 'TeamPendingApprovalsKpi',
100
+ roleSlug: 'supervisor',
101
+ width: 6,
102
+ height: 2,
103
+ category: 'supervisor',
104
+ },
105
+ {
106
+ slug: 'team-hours-kpi',
107
+ title: 'Horas do Time',
108
+ component: 'TeamHoursKpi',
109
+ roleSlug: 'supervisor',
110
+ width: 6,
111
+ height: 2,
112
+ category: 'supervisor',
113
+ },
114
+ {
115
+ slug: 'team-capacity-kpi',
116
+ title: 'Capacidade do Time',
117
+ component: 'TeamCapacityKpi',
118
+ roleSlug: 'supervisor',
119
+ width: 6,
120
+ height: 2,
121
+ category: 'supervisor',
122
+ },
123
+ {
124
+ slug: 'team-utilization-overview',
125
+ title: 'Utilização do Time',
126
+ component: 'TeamUtilizationOverview',
127
+ roleSlug: 'supervisor',
128
+ width: 6,
129
+ height: 2,
130
+ category: 'supervisor',
131
+ },
132
+ {
133
+ slug: 'team-workload-alerts',
134
+ title: 'Alertas de Carga',
135
+ component: 'TeamWorkloadAlerts',
136
+ roleSlug: 'supervisor',
137
+ width: 12,
138
+ height: 3,
139
+ category: 'supervisor',
140
+ },
141
+ {
142
+ slug: 'managed-projects-status',
143
+ title: 'Status dos Projetos',
144
+ component: 'ManagedProjectsStatus',
145
+ roleSlug: 'supervisor',
146
+ width: 6,
147
+ height: 3,
148
+ category: 'supervisor',
149
+ },
150
+ {
151
+ slug: 'team-approval-queue',
152
+ title: 'Fila de Aprovações',
153
+ component: 'TeamApprovalQueue',
154
+ roleSlug: 'supervisor',
155
+ width: 6,
156
+ height: 3,
157
+ category: 'supervisor',
158
+ },
159
+ ];
160
+
161
+ // Widgets para Diretores (8)
162
+ export const DIRECTOR_WIDGETS: WidgetInfo[] = [
163
+ {
164
+ slug: 'portfolio-costs-kpi',
165
+ title: 'Custos do Portfólio',
166
+ component: 'PortfolioCostsKpi',
167
+ roleSlug: 'director',
168
+ width: 6,
169
+ height: 2,
170
+ category: 'director',
171
+ },
172
+ {
173
+ slug: 'portfolio-effort-kpi',
174
+ title: 'Esforço do Portfólio',
175
+ component: 'PortfolioEffortKpi',
176
+ roleSlug: 'director',
177
+ width: 6,
178
+ height: 2,
179
+ category: 'director',
180
+ },
181
+ {
182
+ slug: 'portfolio-risk-kpi',
183
+ title: 'Risco do Portfólio',
184
+ component: 'PortfolioRiskKpi',
185
+ roleSlug: 'director',
186
+ width: 6,
187
+ height: 2,
188
+ category: 'director',
189
+ },
190
+ {
191
+ slug: 'project-status-overview',
192
+ title: 'Status dos Projetos',
193
+ component: 'ProjectStatusOverview',
194
+ roleSlug: 'director',
195
+ width: 6,
196
+ height: 2,
197
+ category: 'director',
198
+ },
199
+ {
200
+ slug: 'effort-by-project',
201
+ title: 'Esforço por Projeto',
202
+ component: 'EffortByProject',
203
+ roleSlug: 'director',
204
+ width: 12,
205
+ height: 3,
206
+ category: 'director',
207
+ },
208
+ {
209
+ slug: 'capacity-distribution',
210
+ title: 'Distribuição de Capacidade',
211
+ component: 'CapacityDistribution',
212
+ roleSlug: 'director',
213
+ width: 6,
214
+ height: 3,
215
+ category: 'director',
216
+ },
217
+ {
218
+ slug: 'headcount-by-area',
219
+ title: 'Headcount por Área',
220
+ component: 'HeadcountByArea',
221
+ roleSlug: 'director',
222
+ width: 6,
223
+ height: 3,
224
+ category: 'director',
225
+ },
226
+ {
227
+ slug: 'strategic-deadlines',
228
+ title: 'Prazos Estratégicos',
229
+ component: 'StrategicDeadlines',
230
+ roleSlug: 'director',
231
+ width: 12,
232
+ height: 3,
233
+ category: 'director',
234
+ },
235
+ ];
236
+
237
+ // Registro completo de todos os widgets
238
+ export const ALL_WIDGETS = [
239
+ ...COLLABORATOR_WIDGETS,
240
+ ...SUPERVISOR_WIDGETS,
241
+ ...DIRECTOR_WIDGETS,
242
+ ];
243
+
244
+ // Acessores úteis
245
+ export const getWidgetsByRole = (roleSlug: string): WidgetInfo[] => {
246
+ return ALL_WIDGETS.filter(w => w.roleSlug === roleSlug);
247
+ };
248
+
249
+ export const getWidgetBySlug = (slug: string): WidgetInfo | undefined => {
250
+ return ALL_WIDGETS.find(w => w.slug === slug);
251
+ };
252
+
253
+ export const getWidgetsByCategory = (category: 'collaborator' | 'supervisor' | 'director'): WidgetInfo[] => {
254
+ return ALL_WIDGETS.filter(w => w.category === category);
255
+ };
@@ -0,0 +1,29 @@
1
+ /**
2
+ * Hook para buscar dados do dashboard de operations
3
+ * Cacheia dados por 60s usando react-query
4
+ * Filtra automaticamente por role do usuário
5
+ */
6
+
7
+ import type { OperationsDashboardData } from '../types/widgets.types';
8
+
9
+ type DashboardRequestFn = <T extends object>(config: {
10
+ url: string;
11
+ method: 'GET';
12
+ }) => Promise<{ data: T }>;
13
+
14
+ /**
15
+ * Loader enxuto para o payload agregado do dashboard.
16
+ * A camada de UI pode embrulhar isso em react-query sem repetir o contrato.
17
+ */
18
+ export async function fetchDashboardData(request: DashboardRequestFn) {
19
+ const response = await request<OperationsDashboardData>({
20
+ url: '/operations/dashboard',
21
+ method: 'GET',
22
+ });
23
+
24
+ return response.data;
25
+ }
26
+
27
+ export async function useDashboardData(request: DashboardRequestFn) {
28
+ return fetchDashboardData(request);
29
+ }
@@ -0,0 +1,237 @@
1
+ /**
2
+ * Tipos para widgets do dashboard
3
+ */
4
+
5
+ export interface OperationsDashboardData {
6
+ actor: {
7
+ roleScope: 'self' | 'team' | 'full';
8
+ collaboratorId: number | null;
9
+ collaboratorName: string | null;
10
+ teamSize: number;
11
+ };
12
+ cards: {
13
+ projectsTotal: number;
14
+ activeProjects: number;
15
+ visibleTimesheets: number;
16
+ pendingTimesheets: number;
17
+ timeOffRequests: number;
18
+ scheduleAdjustmentRequests: number;
19
+ pendingApprovals: number;
20
+ };
21
+ windowDays: number;
22
+ taskSummary: {
23
+ total: number;
24
+ backlog: number;
25
+ inProgress: number;
26
+ completed: number;
27
+ };
28
+ avgTaskCompletionHours: number;
29
+ completedTasksInWindow: number;
30
+ hoursInExecution: number;
31
+ timesheetHeatmap: Array<{
32
+ workDate: string;
33
+ weekday: number;
34
+ hours: number;
35
+ }>;
36
+ activeProjects: Array<{
37
+ id: number;
38
+ name: string;
39
+ code: string | null;
40
+ taskBacklog: number;
41
+ taskInProgress: number;
42
+ taskCompleted: number;
43
+ }>;
44
+ recentTaskActivities: Array<{
45
+ taskId: number;
46
+ taskName: string;
47
+ taskStatus: string;
48
+ taskPriority: string | null;
49
+ taskDueDate: string | null;
50
+ taskEstimateHours: number | null;
51
+ taskTags: string | null;
52
+ taskAssigneeCollaboratorId: number | null;
53
+ taskAssigneeName: string | null;
54
+ projectId: number;
55
+ projectName: string;
56
+ projectCode: string | null;
57
+ projectAssignmentId: number | null;
58
+ action: string | null;
59
+ fromStatus: string;
60
+ toStatus: string;
61
+ createdAt: string;
62
+ }>;
63
+ recentTimesheets: Array<{
64
+ id: number;
65
+ collaboratorName: string;
66
+ weekStartDate: string;
67
+ weekEndDate: string;
68
+ totalHours: number | null;
69
+ status: string;
70
+ }>;
71
+ }
72
+
73
+ export interface DashboardWidgetProps {
74
+ slug: string;
75
+ title: string;
76
+ roleSlug: string; // operations-collaborator | admin-operations-supervisor | admin-operations-director
77
+ }
78
+
79
+ export interface CollaboratorLevelData {
80
+ myHoursPeriod: {
81
+ plannedHours: number;
82
+ approvedHours: number;
83
+ pendingHours: number;
84
+ };
85
+ timesheetStatus: {
86
+ submitted: number;
87
+ pending: number;
88
+ approved: number;
89
+ };
90
+ projectAllocations: {
91
+ active: number;
92
+ totalCapacity: number;
93
+ allocatedCapacity: number;
94
+ };
95
+ openRequests: {
96
+ totalOpen: number;
97
+ byType: Record<string, number>;
98
+ };
99
+ weeklyJourney: {
100
+ weekStartDate: string;
101
+ weekEndDate: string;
102
+ plannedHours: number;
103
+ workedHours: number;
104
+ dailyHours: Array<{
105
+ date: string;
106
+ planned: number;
107
+ worked: number;
108
+ }>;
109
+ };
110
+ relevantDeadlines: Array<{
111
+ id: number;
112
+ description: string;
113
+ dueDate: string;
114
+ type: 'delivery' | 'contract' | 'approval';
115
+ priority: 'high' | 'medium' | 'low';
116
+ }>;
117
+ pendingRequestsList: Array<{
118
+ id: number;
119
+ type: string;
120
+ title: string;
121
+ submittedAt: string;
122
+ status: string;
123
+ }>;
124
+ quickActions: {
125
+ timesheetLink: string;
126
+ requestsLink: string;
127
+ allocationsLink: string;
128
+ };
129
+ }
130
+
131
+ export interface SupervisorLevelData {
132
+ teamHeadcount: {
133
+ total: number;
134
+ active: number;
135
+ onLeave: number;
136
+ };
137
+ pendingApprovals: {
138
+ timesheets: number;
139
+ timeOff: number;
140
+ scheduleAdjustments: number;
141
+ };
142
+ teamHours: {
143
+ logged: number;
144
+ approved: number;
145
+ pending: number;
146
+ };
147
+ teamCapacity: {
148
+ total: number;
149
+ allocated: number;
150
+ available: number;
151
+ };
152
+ teamUtilizationOverview: {
153
+ byPeriod: Array<{
154
+ period: string;
155
+ allocation: number;
156
+ effort: number;
157
+ availability: number;
158
+ }>;
159
+ };
160
+ workloadAlerts: Array<{
161
+ collaboratorId: number;
162
+ collaboratorName: string;
163
+ alertType: 'overload' | 'underutilized' | 'conflict';
164
+ severity: 'high' | 'medium' | 'low';
165
+ message: string;
166
+ }>;
167
+ managedProjectsStatus: Array<{
168
+ projectId: number;
169
+ projectName: string;
170
+ status: string;
171
+ progress: number;
172
+ health: 'green' | 'yellow' | 'red';
173
+ }>;
174
+ approvalQueue: Array<{
175
+ id: number;
176
+ itemType: string;
177
+ itemTitle: string;
178
+ submittedBy: string;
179
+ submittedAt: string;
180
+ priority: 'high' | 'medium' | 'low';
181
+ }>;
182
+ }
183
+
184
+ export interface DirectorLevelData {
185
+ portfolioProjects: {
186
+ total: number;
187
+ active: number;
188
+ completed: number;
189
+ };
190
+ portfolioCosts: {
191
+ planned: number;
192
+ actual: number;
193
+ variance: number;
194
+ };
195
+ portfolioEffort: {
196
+ delivered: number;
197
+ inProgress: number;
198
+ planned: number;
199
+ };
200
+ portfolioRisk: Array<{
201
+ projectId: number;
202
+ projectName: string;
203
+ riskLevel: 'high' | 'medium' | 'low';
204
+ description: string;
205
+ }>;
206
+ projectStatusOverview: Array<{
207
+ projectId: number;
208
+ projectName: string;
209
+ status: string;
210
+ scheduleHealth: number; // 0-100
211
+ costHealth: number; // 0-100
212
+ deliveryHealth: number; // 0-100
213
+ }>;
214
+ effortByProject: Array<{
215
+ projectId: number;
216
+ projectName: string;
217
+ allocatedEffort: number;
218
+ percentOfTotal: number;
219
+ }>;
220
+ capacityDistribution: Array<{
221
+ teamName: string;
222
+ capacity: number;
223
+ utilized: number;
224
+ available: number;
225
+ }>;
226
+ headcountByArea: Array<{
227
+ areaName: string;
228
+ headcount: number;
229
+ percentOfTotal: number;
230
+ }>;
231
+ strategicDeadlines: Array<{
232
+ description: string;
233
+ dueDate: string;
234
+ impact: 'critical' | 'high' | 'medium' | 'low';
235
+ status: 'onTrack' | 'atRisk' | 'delayed';
236
+ }>;
237
+ }
@@ -0,0 +1,56 @@
1
+ 'use client';
2
+
3
+ import React from 'react';
4
+ import { PieChart } from 'lucide-react';
5
+
6
+ interface CapacityArea {
7
+ name: string;
8
+ capacity: number;
9
+ percentage: number;
10
+ color: string;
11
+ icon: string;
12
+ }
13
+
14
+ interface CapacityDistributionProps {
15
+ slug: string;
16
+ title: string;
17
+ roleSlug: string;
18
+ width?: number;
19
+ height?: number;
20
+ data?: { areas: CapacityArea[]; totalCapacity: number };
21
+ style?: React.CSSProperties;
22
+ }
23
+
24
+ const CapacityDistribution: React.FC<CapacityDistributionProps> = ({
25
+ title,
26
+ data = { areas: [], totalCapacity: 0 },
27
+ style,
28
+ }) => {
29
+ const colorMap = ['blue', 'purple', 'pink', 'green', 'yellow', 'red'];
30
+ return (
31
+ <div className="bg-linear-to-br from-white to-teal-50 rounded-lg shadow-sm p-4 border border-teal-100" style={style}>
32
+ <div className="flex items-center justify-between mb-4"><h3 className="text-sm font-medium text-gray-700">{title}</h3><PieChart className="w-4 h-4 text-teal-600" /></div>
33
+ <div className="space-y-3">
34
+ {data.areas.length === 0 ? (
35
+ <div className="text-center py-6 text-gray-500"><PieChart className="w-8 h-8 mx-auto mb-2 opacity-50" /><p className="text-sm">Nenhuma área com capacidade</p></div>
36
+ ) : (
37
+ <>
38
+ {data.areas.map((area, idx) => (
39
+ <div key={idx} className="space-y-1">
40
+ <div className="flex items-center justify-between">
41
+ <div className="flex items-center gap-2"><span className="text-lg">{area.icon}</span><span className="text-xs font-medium text-gray-900 truncate">{area.name}</span></div>
42
+ <span className="text-xs font-bold text-gray-700">{area.percentage}%</span>
43
+ </div>
44
+ <div className="relative w-full h-3 bg-gray-100 rounded-full overflow-hidden"><div className={`h-full bg-${colorMap[idx % colorMap.length]}-500 transition-all`} style={{ width: `${area.percentage}%` }} /></div>
45
+ <div className="text-xs text-gray-600">{area.capacity}h</div>
46
+ </div>
47
+ ))}
48
+ <div className="pt-3 border-t border-teal-100 bg-teal-50 rounded-lg p-3 text-center"><p className="text-xs text-teal-700 mb-1">Capacidade Total</p><p className="text-2xl font-bold text-teal-600">{data.totalCapacity}h</p></div>
49
+ </>
50
+ )}
51
+ </div>
52
+ </div>
53
+ );
54
+ };
55
+
56
+ export default CapacityDistribution;
@@ -0,0 +1,51 @@
1
+ 'use client';
2
+
3
+ import React from 'react';
4
+ import { BarChart3 } from 'lucide-react';
5
+
6
+ interface ProjectEffort {
7
+ name: string;
8
+ hours: number;
9
+ percent: number;
10
+ color: string;
11
+ }
12
+
13
+ interface EffortByProjectProps {
14
+ slug: string;
15
+ title: string;
16
+ roleSlug: string;
17
+ width?: number;
18
+ height?: number;
19
+ data?: { projects: ProjectEffort[]; totalHours: number };
20
+ style?: React.CSSProperties;
21
+ }
22
+
23
+ const EffortByProject: React.FC<EffortByProjectProps> = ({
24
+ title,
25
+ data = { projects: [], totalHours: 0 },
26
+ style,
27
+ }) => {
28
+ const colorMap = ['bg-blue-500', 'bg-purple-500', 'bg-pink-500', 'bg-green-500', 'bg-orange-500'];
29
+ return (
30
+ <div className="bg-linear-to-br from-white to-pink-50 rounded-lg shadow-sm p-4 border border-pink-100" style={style}>
31
+ <div className="flex items-center justify-between mb-4"><h3 className="text-sm font-medium text-gray-700">{title}</h3><BarChart3 className="w-4 h-4 text-pink-600" /></div>
32
+ <div className="space-y-3">
33
+ {data.projects.length === 0 ? (
34
+ <div className="text-center py-6 text-gray-500"><BarChart3 className="w-8 h-8 mx-auto mb-2 opacity-50" /><p className="text-sm">Nenhum projeto com esforço</p></div>
35
+ ) : (
36
+ <>
37
+ {data.projects.slice(0, 5).map((project, idx) => (
38
+ <div key={idx} className="space-y-1">
39
+ <div className="flex justify-between items-center"><span className="text-xs font-medium text-gray-900 truncate">{project.name}</span><span className="text-xs font-bold text-gray-700">{project.hours}h ({project.percent}%)</span></div>
40
+ <div className="relative w-full h-4 bg-gray-100 rounded-full overflow-hidden"><div className={`h-full ${colorMap[idx % colorMap.length]} transition-all`} style={{ width: `${project.percent}%` }} /></div>
41
+ </div>
42
+ ))}
43
+ <div className="pt-2 border-t border-pink-100 text-center"><p className="text-xs text-gray-600 mb-1">Esforço Total</p><p className="text-2xl font-bold text-pink-600">{data.totalHours}h</p></div>
44
+ </>
45
+ )}
46
+ </div>
47
+ </div>
48
+ );
49
+ };
50
+
51
+ export default EffortByProject;
@@ -0,0 +1,57 @@
1
+ 'use client';
2
+
3
+ import React from 'react';
4
+ import { Users } from 'lucide-react';
5
+
6
+ interface DepartmentHeadcount {
7
+ name: string;
8
+ count: number;
9
+ percentage: number;
10
+ maxCapacity: number;
11
+ utilization: number;
12
+ icon: string;
13
+ }
14
+
15
+ interface HeadcountByAreaProps {
16
+ slug: string;
17
+ title: string;
18
+ roleSlug: string;
19
+ width?: number;
20
+ height?: number;
21
+ data?: { departments: DepartmentHeadcount[]; totalHeadcount: number };
22
+ style?: React.CSSProperties;
23
+ }
24
+
25
+ const HeadcountByArea: React.FC<HeadcountByAreaProps> = ({
26
+ title,
27
+ data = { departments: [], totalHeadcount: 0 },
28
+ style,
29
+ }) => {
30
+ const colorMap = ['blue', 'indigo', 'purple', 'pink', 'red', 'orange', 'yellow', 'green'];
31
+ return (
32
+ <div className="bg-linear-to-br from-white to-blue-50 rounded-lg shadow-sm p-4 border border-blue-100" style={style}>
33
+ <div className="flex items-center justify-between mb-4"><h3 className="text-sm font-medium text-gray-700">{title}</h3><Users className="w-4 h-4 text-blue-600" /></div>
34
+ <div className="space-y-3">
35
+ {data.departments.length === 0 ? (
36
+ <div className="text-center py-6 text-gray-500"><Users className="w-8 h-8 mx-auto mb-2 opacity-50" /><p className="text-sm">Nenhuma área com colaboradores</p></div>
37
+ ) : (
38
+ <>
39
+ {data.departments.map((dept, idx) => (
40
+ <div key={idx} className="space-y-1">
41
+ <div className="flex items-center justify-between">
42
+ <div className="flex items-center gap-2 flex-1"><span className="text-lg">{dept.icon}</span><span className="text-xs font-medium text-gray-900 truncate">{dept.name}</span></div>
43
+ <span className="text-xs font-bold text-gray-700 ml-2">{dept.count}</span>
44
+ </div>
45
+ <div className="relative w-full h-4 bg-gray-100 rounded-full overflow-hidden border border-gray-200"><div className={`h-full bg-${colorMap[idx % colorMap.length]}-500 transition-all`} style={{ width: `${(dept.count / Math.max(...data.departments.map(d => d.count))) * 100}%` }} /></div>
46
+ <div className="flex justify-between text-xs text-gray-600"><span>{dept.percentage}% total</span><span>{dept.utilization}% capacidade</span></div>
47
+ </div>
48
+ ))}
49
+ <div className="pt-3 border-t border-blue-100 bg-blue-50 rounded-lg p-3 text-center"><p className="text-xs text-blue-700 mb-1">Total de Colaboradores</p><p className="text-3xl font-bold text-blue-600">{data.totalHeadcount}</p></div>
50
+ </>
51
+ )}
52
+ </div>
53
+ </div>
54
+ );
55
+ };
56
+
57
+ export default HeadcountByArea;