@hed-hog/operations 0.0.330 → 0.0.331
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +5 -5
- package/dist/controllers/operations-collaborators.controller.d.ts +7 -216
- package/dist/controllers/operations-collaborators.controller.d.ts.map +1 -1
- package/dist/controllers/operations-contracts.controller.d.ts +6 -6
- package/dist/controllers/operations-projects.controller.d.ts +25 -0
- package/dist/controllers/operations-projects.controller.d.ts.map +1 -1
- package/dist/controllers/operations-projects.controller.js +48 -0
- package/dist/controllers/operations-projects.controller.js.map +1 -1
- package/dist/controllers/operations-reports.controller.d.ts +1 -1
- package/dist/controllers/operations-tasks.controller.d.ts +34 -9
- package/dist/controllers/operations-tasks.controller.d.ts.map +1 -1
- package/dist/controllers/operations-tasks.controller.js +43 -32
- package/dist/controllers/operations-tasks.controller.js.map +1 -1
- package/dist/controllers/operations-timesheets.controller.d.ts +9 -9
- package/dist/dashboard/components/DashboardLayout.d.ts +30 -0
- package/dist/dashboard/components/DashboardLayout.d.ts.map +1 -0
- package/dist/dashboard/components/DashboardLayout.js +87 -0
- package/dist/dashboard/components/DashboardLayout.js.map +1 -0
- package/dist/dashboard/components/widget-registry.d.ts +23 -0
- package/dist/dashboard/components/widget-registry.d.ts.map +1 -0
- package/dist/dashboard/components/widget-registry.js +245 -0
- package/dist/dashboard/components/widget-registry.js.map +1 -0
- package/dist/dashboard/hooks/useDashboardData.d.ts +20 -0
- package/dist/dashboard/hooks/useDashboardData.d.ts.map +1 -0
- package/dist/dashboard/hooks/useDashboardData.js +24 -0
- package/dist/dashboard/hooks/useDashboardData.js.map +1 -0
- package/dist/dashboard/types/widgets.types.d.ts +233 -0
- package/dist/dashboard/types/widgets.types.d.ts.map +1 -0
- package/dist/dashboard/types/widgets.types.js +6 -0
- package/dist/dashboard/types/widgets.types.js.map +1 -0
- package/dist/dashboard/widgets/CapacityDistribution.d.ts +23 -0
- package/dist/dashboard/widgets/CapacityDistribution.d.ts.map +1 -0
- package/dist/dashboard/widgets/CapacityDistribution.js +11 -0
- package/dist/dashboard/widgets/CapacityDistribution.js.map +1 -0
- package/dist/dashboard/widgets/EffortByProject.d.ts +22 -0
- package/dist/dashboard/widgets/EffortByProject.d.ts.map +1 -0
- package/dist/dashboard/widgets/EffortByProject.js +11 -0
- package/dist/dashboard/widgets/EffortByProject.js.map +1 -0
- package/dist/dashboard/widgets/HeadcountByArea.d.ts +24 -0
- package/dist/dashboard/widgets/HeadcountByArea.d.ts.map +1 -0
- package/dist/dashboard/widgets/HeadcountByArea.js +11 -0
- package/dist/dashboard/widgets/HeadcountByArea.js.map +1 -0
- package/dist/dashboard/widgets/ManagedProjectsStatus.d.ts +18 -0
- package/dist/dashboard/widgets/ManagedProjectsStatus.d.ts.map +1 -0
- package/dist/dashboard/widgets/ManagedProjectsStatus.js +12 -0
- package/dist/dashboard/widgets/ManagedProjectsStatus.js.map +1 -0
- package/dist/dashboard/widgets/MyHoursPeriodKpi.d.ts +22 -0
- package/dist/dashboard/widgets/MyHoursPeriodKpi.d.ts.map +1 -0
- package/dist/dashboard/widgets/MyHoursPeriodKpi.js +12 -0
- package/dist/dashboard/widgets/MyHoursPeriodKpi.js.map +1 -0
- package/dist/dashboard/widgets/MyOpenRequestsKpi.d.ts +19 -0
- package/dist/dashboard/widgets/MyOpenRequestsKpi.d.ts.map +1 -0
- package/dist/dashboard/widgets/MyOpenRequestsKpi.js +17 -0
- package/dist/dashboard/widgets/MyOpenRequestsKpi.js.map +1 -0
- package/dist/dashboard/widgets/MyPendingRequestsList.d.ts +23 -0
- package/dist/dashboard/widgets/MyPendingRequestsList.d.ts.map +1 -0
- package/dist/dashboard/widgets/MyPendingRequestsList.js +14 -0
- package/dist/dashboard/widgets/MyPendingRequestsList.js.map +1 -0
- package/dist/dashboard/widgets/MyProjectAllocationsKpi.d.ts +22 -0
- package/dist/dashboard/widgets/MyProjectAllocationsKpi.d.ts.map +1 -0
- package/dist/dashboard/widgets/MyProjectAllocationsKpi.js +11 -0
- package/dist/dashboard/widgets/MyProjectAllocationsKpi.js.map +1 -0
- package/dist/dashboard/widgets/MyQuickActions.d.ts +23 -0
- package/dist/dashboard/widgets/MyQuickActions.d.ts.map +1 -0
- package/dist/dashboard/widgets/MyQuickActions.js +18 -0
- package/dist/dashboard/widgets/MyQuickActions.js.map +1 -0
- package/dist/dashboard/widgets/MyRelevantDeadlines.d.ts +23 -0
- package/dist/dashboard/widgets/MyRelevantDeadlines.d.ts.map +1 -0
- package/dist/dashboard/widgets/MyRelevantDeadlines.js +22 -0
- package/dist/dashboard/widgets/MyRelevantDeadlines.js.map +1 -0
- package/dist/dashboard/widgets/MyTimesheetStatusKpi.d.ts +17 -0
- package/dist/dashboard/widgets/MyTimesheetStatusKpi.d.ts.map +1 -0
- package/dist/dashboard/widgets/MyTimesheetStatusKpi.js +11 -0
- package/dist/dashboard/widgets/MyTimesheetStatusKpi.js.map +1 -0
- package/dist/dashboard/widgets/MyWeeklyJourney.d.ts +21 -0
- package/dist/dashboard/widgets/MyWeeklyJourney.d.ts.map +1 -0
- package/dist/dashboard/widgets/MyWeeklyJourney.js +19 -0
- package/dist/dashboard/widgets/MyWeeklyJourney.js.map +1 -0
- package/dist/dashboard/widgets/PortfolioCostsKpi.d.ts +19 -0
- package/dist/dashboard/widgets/PortfolioCostsKpi.d.ts.map +1 -0
- package/dist/dashboard/widgets/PortfolioCostsKpi.js +12 -0
- package/dist/dashboard/widgets/PortfolioCostsKpi.js.map +1 -0
- package/dist/dashboard/widgets/PortfolioEffortKpi.d.ts +18 -0
- package/dist/dashboard/widgets/PortfolioEffortKpi.d.ts.map +1 -0
- package/dist/dashboard/widgets/PortfolioEffortKpi.js +8 -0
- package/dist/dashboard/widgets/PortfolioEffortKpi.js.map +1 -0
- package/dist/dashboard/widgets/PortfolioProjectsKpi.d.ts +22 -0
- package/dist/dashboard/widgets/PortfolioProjectsKpi.d.ts.map +1 -0
- package/dist/dashboard/widgets/PortfolioProjectsKpi.js +56 -0
- package/dist/dashboard/widgets/PortfolioProjectsKpi.js.map +1 -0
- package/dist/dashboard/widgets/PortfolioRiskKpi.d.ts +19 -0
- package/dist/dashboard/widgets/PortfolioRiskKpi.d.ts.map +1 -0
- package/dist/dashboard/widgets/PortfolioRiskKpi.js +11 -0
- package/dist/dashboard/widgets/PortfolioRiskKpi.js.map +1 -0
- package/dist/dashboard/widgets/ProjectStatusOverview.d.ts +19 -0
- package/dist/dashboard/widgets/ProjectStatusOverview.d.ts.map +1 -0
- package/dist/dashboard/widgets/ProjectStatusOverview.js +18 -0
- package/dist/dashboard/widgets/ProjectStatusOverview.js.map +1 -0
- package/dist/dashboard/widgets/StrategicDeadlines.d.ts +24 -0
- package/dist/dashboard/widgets/StrategicDeadlines.d.ts.map +1 -0
- package/dist/dashboard/widgets/StrategicDeadlines.js +22 -0
- package/dist/dashboard/widgets/StrategicDeadlines.js.map +1 -0
- package/dist/dashboard/widgets/TeamApprovalQueue.d.ts +24 -0
- package/dist/dashboard/widgets/TeamApprovalQueue.d.ts.map +1 -0
- package/dist/dashboard/widgets/TeamApprovalQueue.js +12 -0
- package/dist/dashboard/widgets/TeamApprovalQueue.js.map +1 -0
- package/dist/dashboard/widgets/TeamCapacityKpi.d.ts +18 -0
- package/dist/dashboard/widgets/TeamCapacityKpi.d.ts.map +1 -0
- package/dist/dashboard/widgets/TeamCapacityKpi.js +19 -0
- package/dist/dashboard/widgets/TeamCapacityKpi.js.map +1 -0
- package/dist/dashboard/widgets/TeamHeadcountKpi.d.ts +22 -0
- package/dist/dashboard/widgets/TeamHeadcountKpi.d.ts.map +1 -0
- package/dist/dashboard/widgets/TeamHeadcountKpi.js +56 -0
- package/dist/dashboard/widgets/TeamHeadcountKpi.js.map +1 -0
- package/dist/dashboard/widgets/TeamHoursKpi.d.ts +19 -0
- package/dist/dashboard/widgets/TeamHoursKpi.d.ts.map +1 -0
- package/dist/dashboard/widgets/TeamHoursKpi.js +13 -0
- package/dist/dashboard/widgets/TeamHoursKpi.js.map +1 -0
- package/dist/dashboard/widgets/TeamPendingApprovalsKpi.d.ts +20 -0
- package/dist/dashboard/widgets/TeamPendingApprovalsKpi.d.ts.map +1 -0
- package/dist/dashboard/widgets/TeamPendingApprovalsKpi.js +11 -0
- package/dist/dashboard/widgets/TeamPendingApprovalsKpi.js.map +1 -0
- package/dist/dashboard/widgets/TeamUtilizationOverview.d.ts +18 -0
- package/dist/dashboard/widgets/TeamUtilizationOverview.d.ts.map +1 -0
- package/dist/dashboard/widgets/TeamUtilizationOverview.js +17 -0
- package/dist/dashboard/widgets/TeamUtilizationOverview.js.map +1 -0
- package/dist/dashboard/widgets/TeamWorkloadAlerts.d.ts +24 -0
- package/dist/dashboard/widgets/TeamWorkloadAlerts.d.ts.map +1 -0
- package/dist/dashboard/widgets/TeamWorkloadAlerts.js +19 -0
- package/dist/dashboard/widgets/TeamWorkloadAlerts.js.map +1 -0
- package/dist/dashboard/widgets/index.d.ts +24 -0
- package/dist/dashboard/widgets/index.d.ts.map +1 -0
- package/dist/dashboard/widgets/index.js +54 -0
- package/dist/dashboard/widgets/index.js.map +1 -0
- package/dist/dto/create-collaborator.dto.d.ts +0 -1
- package/dist/dto/create-collaborator.dto.d.ts.map +1 -1
- package/dist/dto/create-collaborator.dto.js +0 -6
- package/dist/dto/create-collaborator.dto.js.map +1 -1
- package/dist/index.d.ts +2 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +2 -0
- package/dist/index.js.map +1 -1
- package/dist/operations.controller.d.ts +42 -0
- package/dist/operations.controller.d.ts.map +1 -1
- package/dist/operations.service.d.ts +182 -268
- package/dist/operations.service.d.ts.map +1 -1
- package/dist/operations.service.js +2147 -1337
- package/dist/operations.service.js.map +1 -1
- package/dist/operations.service.spec.js +345 -174
- package/dist/operations.service.spec.js.map +1 -1
- package/hedhog/data/dashboard_component.yaml +66 -0
- package/hedhog/data/dashboard_item.yaml +25 -25
- package/hedhog/data/route.yaml +61 -0
- package/hedhog/frontend/app/_components/collaborator-form-screen.tsx.ejs +39 -99
- package/hedhog/frontend/app/_components/collaborator-picker.tsx.ejs +158 -0
- package/hedhog/frontend/app/_components/my-project-summary-screen.tsx.ejs +314 -116
- package/hedhog/frontend/app/_components/project-assignments-tab.tsx.ejs +434 -449
- package/hedhog/frontend/app/_components/project-details-screen.tsx.ejs +289 -412
- package/hedhog/frontend/app/_components/project-file-attachments.tsx.ejs +371 -0
- package/hedhog/frontend/app/_components/project-form-screen.tsx.ejs +426 -374
- package/hedhog/frontend/app/_components/task-detail-sheet.tsx.ejs +803 -581
- package/hedhog/frontend/app/_components/task-file-attachments.tsx.ejs +4 -1
- package/hedhog/frontend/app/_components/task-form-fields.tsx.ejs +406 -0
- package/hedhog/frontend/app/_components/task-form-sheet.tsx.ejs +629 -784
- package/hedhog/frontend/app/_components/task-info-display.tsx.ejs +137 -0
- package/hedhog/frontend/app/_components/timesheet-entry-create-sheet.tsx.ejs +306 -0
- package/hedhog/frontend/app/_lib/api.ts.ejs +480 -476
- package/hedhog/frontend/app/_lib/types.ts.ejs +66 -5
- package/hedhog/frontend/app/_lib/utils/format.ts.ejs +0 -2
- package/hedhog/frontend/app/_lib/utils/task-ui.ts.ejs +43 -0
- package/hedhog/frontend/app/approvals/page.tsx.ejs +6 -1
- package/hedhog/frontend/app/collaborator-types/page.tsx.ejs +6 -1
- package/hedhog/frontend/app/collaborators/page.tsx.ejs +59 -8
- package/hedhog/frontend/app/contracts/page.tsx.ejs +29 -8
- package/hedhog/frontend/app/dashboard/widgets/CapacityDistribution.tsx.ejs +84 -0
- package/hedhog/frontend/app/dashboard/widgets/EffortByProject.tsx.ejs +85 -0
- package/hedhog/frontend/app/dashboard/widgets/HeadcountByArea.tsx.ejs +101 -0
- package/hedhog/frontend/app/dashboard/widgets/ManagedProjectsStatus.tsx.ejs +113 -0
- package/hedhog/frontend/app/dashboard/widgets/MyHoursPeriodKpi.tsx.ejs +87 -0
- package/hedhog/frontend/app/dashboard/widgets/MyOpenRequestsKpi.tsx.ejs +97 -0
- package/hedhog/frontend/app/dashboard/widgets/MyPendingRequestsList.tsx.ejs +99 -0
- package/hedhog/frontend/app/dashboard/widgets/MyProjectAllocationsKpi.tsx.ejs +78 -0
- package/hedhog/frontend/app/dashboard/widgets/MyQuickActions.tsx.ejs +130 -0
- package/hedhog/frontend/app/dashboard/widgets/MyRelevantDeadlines.tsx.ejs +144 -0
- package/hedhog/frontend/app/dashboard/widgets/MyTimesheetStatusKpi.tsx.ejs +78 -0
- package/hedhog/frontend/app/dashboard/widgets/MyWeeklyJourney.tsx.ejs +99 -0
- package/hedhog/frontend/app/dashboard/widgets/PortfolioCostsKpi.tsx.ejs +112 -0
- package/hedhog/frontend/app/dashboard/widgets/PortfolioEffortKpi.tsx.ejs +93 -0
- package/hedhog/frontend/app/dashboard/widgets/PortfolioProjectsKpi.tsx.ejs +96 -0
- package/hedhog/frontend/app/dashboard/widgets/PortfolioRiskKpi.tsx.ejs +115 -0
- package/hedhog/frontend/app/dashboard/widgets/ProjectStatusOverview.tsx.ejs +120 -0
- package/hedhog/frontend/app/dashboard/widgets/StrategicDeadlines.tsx.ejs +146 -0
- package/hedhog/frontend/app/dashboard/widgets/TeamApprovalQueue.tsx.ejs +108 -0
- package/hedhog/frontend/app/dashboard/widgets/TeamCapacityKpi.tsx.ejs +97 -0
- package/hedhog/frontend/app/dashboard/widgets/TeamHeadcountKpi.tsx.ejs +100 -0
- package/hedhog/frontend/app/dashboard/widgets/TeamHoursKpi.tsx.ejs +104 -0
- package/hedhog/frontend/app/dashboard/widgets/TeamPendingApprovalsKpi.tsx.ejs +110 -0
- package/hedhog/frontend/app/dashboard/widgets/TeamUtilizationOverview.tsx.ejs +115 -0
- package/hedhog/frontend/app/dashboard/widgets/TeamWorkloadAlerts.tsx.ejs +117 -0
- package/hedhog/frontend/app/dashboard/widgets/index.ts.ejs +26 -0
- package/hedhog/frontend/app/departments/page.tsx.ejs +6 -1
- package/hedhog/frontend/app/my-projects/page.tsx.ejs +14 -10
- package/hedhog/frontend/app/my-tasks/page.tsx.ejs +328 -105
- package/hedhog/frontend/app/project-cost-categories/page.tsx.ejs +58 -52
- package/hedhog/frontend/app/project-cost-types/page.tsx.ejs +58 -51
- package/hedhog/frontend/app/projects/page.tsx.ejs +376 -30
- package/hedhog/frontend/app/schedule-adjustments/page.tsx.ejs +6 -1
- package/hedhog/frontend/app/time-off/page.tsx.ejs +6 -1
- package/hedhog/frontend/app/timesheets/page.tsx.ejs +10 -4
- package/hedhog/frontend/messages/en.json +238 -46
- package/hedhog/frontend/messages/operations/en.json +61 -52
- package/hedhog/frontend/messages/operations/pt.json +59 -43
- package/hedhog/frontend/messages/pt.json +238 -46
- package/hedhog/frontend/widgets/capacity-distribution.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/effort-by-project.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/headcount-by-area.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/index.ts.ejs +25 -0
- package/hedhog/frontend/widgets/managed-projects-status.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/my-hours-period-kpi.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/my-open-requests-kpi.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/my-pending-requests-list.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/my-project-allocations-kpi.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/my-quick-actions.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/my-relevant-deadlines.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/my-timesheet-status-kpi.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/my-weekly-journey.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/portfolio-costs-kpi.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/portfolio-effort-kpi.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/portfolio-projects-kpi.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/portfolio-risk-kpi.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/project-status-overview.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/shared-operations-widget.tsx.ejs +170 -0
- package/hedhog/frontend/widgets/strategic-deadlines.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/team-approval-queue.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/team-capacity-kpi.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/team-headcount-kpi.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/team-hours-kpi.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/team-pending-approvals-kpi.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/team-utilization-overview.tsx.ejs +17 -0
- package/hedhog/frontend/widgets/team-workload-alerts.tsx.ejs +17 -0
- package/hedhog/table/operations_collaborator.yaml +8 -13
- package/hedhog/table/operations_project.yaml +1 -1
- package/hedhog/table/operations_project_file.yaml +23 -0
- package/hedhog/table/operations_task.yaml +76 -69
- package/hedhog/table/operations_task_activity.yaml +51 -0
- package/package.json +6 -5
- package/src/controllers/operations-projects.controller.ts +41 -8
- package/src/controllers/operations-tasks.controller.ts +156 -166
- package/src/dashboard/README.md +214 -0
- package/src/dashboard/components/DashboardLayout.tsx +131 -0
- package/src/dashboard/components/widget-registry.ts +255 -0
- package/src/dashboard/hooks/useDashboardData.ts +29 -0
- package/src/dashboard/types/widgets.types.ts +237 -0
- package/src/dashboard/widgets/CapacityDistribution.tsx +56 -0
- package/src/dashboard/widgets/EffortByProject.tsx +51 -0
- package/src/dashboard/widgets/HeadcountByArea.tsx +57 -0
- package/src/dashboard/widgets/ManagedProjectsStatus.tsx +53 -0
- package/src/dashboard/widgets/MyHoursPeriodKpi.tsx +87 -0
- package/src/dashboard/widgets/MyOpenRequestsKpi.tsx +51 -0
- package/src/dashboard/widgets/MyPendingRequestsList.tsx +63 -0
- package/src/dashboard/widgets/MyProjectAllocationsKpi.tsx +57 -0
- package/src/dashboard/widgets/MyQuickActions.tsx +62 -0
- package/src/dashboard/widgets/MyRelevantDeadlines.tsx +84 -0
- package/src/dashboard/widgets/MyTimesheetStatusKpi.tsx +65 -0
- package/src/dashboard/widgets/MyWeeklyJourney.tsx +57 -0
- package/src/dashboard/widgets/PortfolioCostsKpi.tsx +48 -0
- package/src/dashboard/widgets/PortfolioEffortKpi.tsx +41 -0
- package/src/dashboard/widgets/PortfolioRiskKpi.tsx +50 -0
- package/src/dashboard/widgets/ProjectStatusOverview.tsx +52 -0
- package/src/dashboard/widgets/StrategicDeadlines.tsx +93 -0
- package/src/dashboard/widgets/TeamApprovalQueue.tsx +70 -0
- package/src/dashboard/widgets/TeamCapacityKpi.tsx +50 -0
- package/src/dashboard/widgets/TeamHoursKpi.tsx +51 -0
- package/src/dashboard/widgets/TeamPendingApprovalsKpi.tsx +53 -0
- package/src/dashboard/widgets/TeamUtilizationOverview.tsx +62 -0
- package/src/dashboard/widgets/TeamWorkloadAlerts.tsx +81 -0
- package/src/dashboard/widgets/index.ts +26 -0
- package/src/dto/create-collaborator.dto.ts +4 -11
- package/src/index.ts +3 -0
- package/src/operations.service.spec.ts +988 -764
- package/src/operations.service.ts +4277 -2535
|
@@ -0,0 +1,131 @@
|
|
|
1
|
+
'use client';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Dashboard Principal - Renderiza widgets conforme role do usuário
|
|
5
|
+
*
|
|
6
|
+
* Fluxo:
|
|
7
|
+
* 1. Detecta role do usuário (collaborator/supervisor/director)
|
|
8
|
+
* 2. Busca dados via GET /operations/dashboard
|
|
9
|
+
* 3. Renderiza widgets apropriados para a role
|
|
10
|
+
* 4. Cada widget é um componente reutilizável
|
|
11
|
+
*/
|
|
12
|
+
|
|
13
|
+
import React, { useMemo } from 'react';
|
|
14
|
+
|
|
15
|
+
// Tipos de widgets por role
|
|
16
|
+
interface WidgetDefinition {
|
|
17
|
+
slug: string;
|
|
18
|
+
title: string;
|
|
19
|
+
roleSlug: string;
|
|
20
|
+
width: number;
|
|
21
|
+
height: number;
|
|
22
|
+
xAxis: number;
|
|
23
|
+
yAxis: number;
|
|
24
|
+
componentPath: string;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
// Mapear slugs aos componentes - importados dinamicamente do admin app
|
|
28
|
+
// Os widgets estão em: apps/admin/src/app/(app)/(libraries)/operations/dashboard/widgets/
|
|
29
|
+
const WIDGET_REGISTRY: Record<string, React.ComponentType<any>> = {};
|
|
30
|
+
|
|
31
|
+
// Lazy loading de widgets - para evitar imports circulares
|
|
32
|
+
const getWidget = (slug: string): React.ComponentType<any> | null => {
|
|
33
|
+
// Este mapa será populado em runtime via importação dinâmica
|
|
34
|
+
// Por enquanto, retorna null se widget não estiver pronto
|
|
35
|
+
return WIDGET_REGISTRY[slug] || null;
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
interface DashboardLayoutProps {
|
|
39
|
+
roleSlugs: string[];
|
|
40
|
+
dashboardSlug: string;
|
|
41
|
+
widgets: WidgetDefinition[];
|
|
42
|
+
data: any;
|
|
43
|
+
isLoading?: boolean;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
export const DashboardLayout: React.FC<DashboardLayoutProps> = ({
|
|
47
|
+
roleSlugs,
|
|
48
|
+
dashboardSlug,
|
|
49
|
+
widgets,
|
|
50
|
+
data,
|
|
51
|
+
isLoading = false,
|
|
52
|
+
}) => {
|
|
53
|
+
// Filtrar widgets para a role atual
|
|
54
|
+
const visibleWidgets = useMemo(() => {
|
|
55
|
+
return widgets.filter(widget =>
|
|
56
|
+
roleSlugs.includes(widget.roleSlug)
|
|
57
|
+
);
|
|
58
|
+
}, [widgets, roleSlugs]);
|
|
59
|
+
|
|
60
|
+
if (isLoading) {
|
|
61
|
+
return (
|
|
62
|
+
<div className="grid gap-4 w-full h-full">
|
|
63
|
+
{Array(4).fill(0).map((_, i) => (
|
|
64
|
+
<div
|
|
65
|
+
key={i}
|
|
66
|
+
className="bg-gray-200 rounded-lg animate-pulse"
|
|
67
|
+
style={{ minHeight: '200px' }}
|
|
68
|
+
/>
|
|
69
|
+
))}
|
|
70
|
+
</div>
|
|
71
|
+
);
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
return (
|
|
75
|
+
<div className="grid gap-4 w-full auto-rows-max"
|
|
76
|
+
style={{
|
|
77
|
+
gridTemplateColumns: 'repeat(12, 1fr)',
|
|
78
|
+
}}>
|
|
79
|
+
{visibleWidgets.map(widget => {
|
|
80
|
+
const Component = WIDGET_REGISTRY[widget.slug];
|
|
81
|
+
|
|
82
|
+
if (!Component) {
|
|
83
|
+
return (
|
|
84
|
+
<div
|
|
85
|
+
key={widget.slug}
|
|
86
|
+
style={{
|
|
87
|
+
gridColumn: `span ${Math.min(widget.width, 12)}`,
|
|
88
|
+
gridRow: `auto / span ${widget.height}`,
|
|
89
|
+
}}
|
|
90
|
+
className="bg-red-50 border-2 border-red-200 rounded-lg p-4"
|
|
91
|
+
>
|
|
92
|
+
<p className="text-red-700 text-sm">
|
|
93
|
+
Component not found: {widget.slug}
|
|
94
|
+
</p>
|
|
95
|
+
</div>
|
|
96
|
+
);
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
return (
|
|
100
|
+
<React.Suspense
|
|
101
|
+
key={widget.slug}
|
|
102
|
+
fallback={
|
|
103
|
+
<div
|
|
104
|
+
style={{
|
|
105
|
+
gridColumn: `span ${Math.min(widget.width, 12)}`,
|
|
106
|
+
gridRow: `auto / span ${widget.height}`,
|
|
107
|
+
}}
|
|
108
|
+
className="bg-gray-200 rounded-lg animate-pulse"
|
|
109
|
+
/>
|
|
110
|
+
}
|
|
111
|
+
>
|
|
112
|
+
<Component
|
|
113
|
+
slug={widget.slug}
|
|
114
|
+
title={widget.title}
|
|
115
|
+
roleSlug={roleSlugs[0]}
|
|
116
|
+
width={widget.width}
|
|
117
|
+
height={widget.height}
|
|
118
|
+
data={data?.[widget.slug]}
|
|
119
|
+
style={{
|
|
120
|
+
gridColumn: `span ${Math.min(widget.width, 12)}`,
|
|
121
|
+
gridRow: `auto / span ${widget.height}`,
|
|
122
|
+
}}
|
|
123
|
+
/>
|
|
124
|
+
</React.Suspense>
|
|
125
|
+
);
|
|
126
|
+
})}
|
|
127
|
+
</div>
|
|
128
|
+
);
|
|
129
|
+
};
|
|
130
|
+
|
|
131
|
+
export default DashboardLayout;
|
|
@@ -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;
|