Merge branch 'release-v2.1.4' of https://github.com/Worklenz/worklenz into feature/team-utilization
This commit is contained in:
@@ -0,0 +1,46 @@
|
||||
import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react';
|
||||
import { API_BASE_URL } from '@/shared/constants';
|
||||
import { getCsrfToken } from '../api-client';
|
||||
import { IUserRecentTask, IUserTimeLoggedTask } from '@/types/home/user-activity.types';
|
||||
import config from '@/config/env';
|
||||
|
||||
const rootUrl = '/logs';
|
||||
|
||||
export const userActivityApiService = createApi({
|
||||
reducerPath: 'userActivityApi',
|
||||
baseQuery: fetchBaseQuery({
|
||||
baseUrl: `${config.apiUrl}${API_BASE_URL}`,
|
||||
prepareHeaders: (headers) => {
|
||||
headers.set('X-CSRF-Token', getCsrfToken() || '');
|
||||
headers.set('Content-Type', 'application/json');
|
||||
return headers;
|
||||
},
|
||||
credentials: 'include',
|
||||
}),
|
||||
tagTypes: ['UserRecentTasks', 'UserTimeLoggedTasks'],
|
||||
endpoints: (builder) => ({
|
||||
getUserRecentTasks: builder.query<IUserRecentTask[], { limit?: number; offset?: number }>({
|
||||
query: ({ limit = 10, offset = 0 }) => ({
|
||||
url: `${rootUrl}/user-recent-tasks`,
|
||||
params: { limit, offset },
|
||||
method: 'GET',
|
||||
}),
|
||||
providesTags: ['UserRecentTasks'],
|
||||
}),
|
||||
getUserTimeLoggedTasks: builder.query<IUserTimeLoggedTask[], { limit?: number; offset?: number }>({
|
||||
query: ({ limit = 10, offset = 0 }) => ({
|
||||
url: `${rootUrl}/user-time-logged-tasks`,
|
||||
params: { limit, offset },
|
||||
method: 'GET',
|
||||
}),
|
||||
providesTags: ['UserTimeLoggedTasks'],
|
||||
}),
|
||||
}),
|
||||
});
|
||||
|
||||
export const {
|
||||
useGetUserRecentTasksQuery,
|
||||
useGetUserTimeLoggedTasksQuery,
|
||||
} = userActivityApiService;
|
||||
|
||||
|
||||
@@ -0,0 +1,92 @@
|
||||
import { API_BASE_URL } from '@/shared/constants';
|
||||
import { toQueryString } from '@/utils/toQueryString';
|
||||
import apiClient from '../api-client';
|
||||
import { IServerResponse } from '@/types/common.types';
|
||||
import { IAllocationViewModel } from '@/types/reporting/reporting-allocation.types';
|
||||
import {
|
||||
IProjectLogsBreakdown,
|
||||
IRPTTimeMember,
|
||||
IRPTTimeProject,
|
||||
ITimeLogBreakdownReq,
|
||||
} from '@/types/reporting/reporting.types';
|
||||
|
||||
const rootUrl = `${API_BASE_URL}/reporting`;
|
||||
|
||||
// Helper function to get user's timezone
|
||||
const getUserTimezone = () => {
|
||||
return Intl.DateTimeFormat().resolvedOptions().timeZone;
|
||||
};
|
||||
|
||||
export const reportingTimesheetApiService = {
|
||||
getTimeSheetData: async (
|
||||
body = {},
|
||||
archived = false
|
||||
): Promise<IServerResponse<IAllocationViewModel>> => {
|
||||
const q = toQueryString({ archived });
|
||||
const bodyWithTimezone = {
|
||||
...body,
|
||||
timezone: getUserTimezone()
|
||||
};
|
||||
const response = await apiClient.post(`${rootUrl}/allocation/${q}`, bodyWithTimezone);
|
||||
return response.data;
|
||||
},
|
||||
|
||||
getAllocationProjects: async (body = {}) => {
|
||||
const bodyWithTimezone = {
|
||||
...body,
|
||||
timezone: getUserTimezone()
|
||||
};
|
||||
const response = await apiClient.post(`${rootUrl}/allocation/allocation-projects`, bodyWithTimezone);
|
||||
return response.data;
|
||||
},
|
||||
|
||||
getProjectTimeSheets: async (
|
||||
body = {},
|
||||
archived = false
|
||||
): Promise<IServerResponse<IRPTTimeProject[]>> => {
|
||||
const q = toQueryString({ archived });
|
||||
const bodyWithTimezone = {
|
||||
...body,
|
||||
timezone: getUserTimezone()
|
||||
};
|
||||
const response = await apiClient.post(`${rootUrl}/time-reports/projects/${q}`, bodyWithTimezone);
|
||||
return response.data;
|
||||
},
|
||||
|
||||
getMemberTimeSheets: async (
|
||||
body = {},
|
||||
archived = false
|
||||
): Promise<IServerResponse<IRPTTimeMember[]>> => {
|
||||
const q = toQueryString({ archived });
|
||||
const bodyWithTimezone = {
|
||||
...body,
|
||||
timezone: getUserTimezone()
|
||||
};
|
||||
const response = await apiClient.post(`${rootUrl}/time-reports/members/${q}`, bodyWithTimezone);
|
||||
return response.data;
|
||||
},
|
||||
|
||||
getProjectTimeLogs: async (
|
||||
body: ITimeLogBreakdownReq
|
||||
): Promise<IServerResponse<IProjectLogsBreakdown[]>> => {
|
||||
const bodyWithTimezone = {
|
||||
...body,
|
||||
timezone: getUserTimezone()
|
||||
};
|
||||
const response = await apiClient.post(`${rootUrl}/project-timelogs`, bodyWithTimezone);
|
||||
return response.data;
|
||||
},
|
||||
|
||||
getProjectEstimatedVsActual: async (
|
||||
body = {},
|
||||
archived = false
|
||||
): Promise<IServerResponse<IRPTTimeProject[]>> => {
|
||||
const q = toQueryString({ archived });
|
||||
const bodyWithTimezone = {
|
||||
...body,
|
||||
timezone: getUserTimezone()
|
||||
};
|
||||
const response = await apiClient.post(`${rootUrl}/time-reports/estimated-vs-actual${q}`, bodyWithTimezone);
|
||||
return response.data;
|
||||
},
|
||||
};
|
||||
27
worklenz-frontend/src/api/survey/survey.api.service.ts
Normal file
27
worklenz-frontend/src/api/survey/survey.api.service.ts
Normal file
@@ -0,0 +1,27 @@
|
||||
import { IServerResponse } from '@/types/common.types';
|
||||
import { ISurvey, ISurveySubmissionRequest, ISurveyResponse } from '@/types/account-setup/survey.types';
|
||||
import apiClient from '../api-client';
|
||||
|
||||
const API_BASE_URL = '/api/v1';
|
||||
|
||||
export const surveyApiService = {
|
||||
async getAccountSetupSurvey(): Promise<IServerResponse<ISurvey>> {
|
||||
const response = await apiClient.get<IServerResponse<ISurvey>>(`${API_BASE_URL}/surveys/account-setup`);
|
||||
return response.data;
|
||||
},
|
||||
|
||||
async submitSurveyResponse(data: ISurveySubmissionRequest): Promise<IServerResponse<{ response_id: string }>> {
|
||||
const response = await apiClient.post<IServerResponse<{ response_id: string }>>(`${API_BASE_URL}/surveys/responses`, data);
|
||||
return response.data;
|
||||
},
|
||||
|
||||
async getUserSurveyResponse(surveyId: string): Promise<IServerResponse<ISurveyResponse>> {
|
||||
const response = await apiClient.get<IServerResponse<ISurveyResponse>>(`${API_BASE_URL}/surveys/responses/${surveyId}`);
|
||||
return response.data;
|
||||
},
|
||||
|
||||
async checkAccountSetupSurveyStatus(): Promise<IServerResponse<{ is_completed: boolean; completed_at?: string }>> {
|
||||
const response = await apiClient.get<IServerResponse<{ is_completed: boolean; completed_at?: string }>>(`${API_BASE_URL}/surveys/account-setup/status`);
|
||||
return response.data;
|
||||
}
|
||||
};
|
||||
@@ -2,6 +2,7 @@ import { API_BASE_URL } from '@/shared/constants';
|
||||
import apiClient from '../api-client';
|
||||
import { IServerResponse } from '@/types/common.types';
|
||||
import { ITaskLogViewModel } from '@/types/tasks/task-log-view.types';
|
||||
import { getUserSession } from '@/utils/session-helper';
|
||||
|
||||
const rootUrl = `${API_BASE_URL}/task-time-log`;
|
||||
|
||||
@@ -17,7 +18,11 @@ export interface IRunningTimer {
|
||||
|
||||
export const taskTimeLogsApiService = {
|
||||
getByTask: async (id: string): Promise<IServerResponse<ITaskLogViewModel[]>> => {
|
||||
const response = await apiClient.get(`${rootUrl}/task/${id}`);
|
||||
const session = getUserSession();
|
||||
const timezone = session?.timezone_name || 'UTC';
|
||||
const response = await apiClient.get(`${rootUrl}/task/${id}`, {
|
||||
params: { time_zone_name: timezone }
|
||||
});
|
||||
return response.data;
|
||||
},
|
||||
|
||||
|
||||
@@ -17,6 +17,7 @@ const ProjectTemplateEditView = lazy(
|
||||
const LicenseExpired = lazy(() => import('@/pages/license-expired/license-expired'));
|
||||
const ProjectView = lazy(() => import('@/pages/projects/projectView/project-view'));
|
||||
const Unauthorized = lazy(() => import('@/pages/unauthorized/unauthorized'));
|
||||
const GanttDemoPage = lazy(() => import('@/pages/GanttDemoPage'));
|
||||
|
||||
// Define AdminGuard component with defensive programming
|
||||
const AdminGuard = ({ children }: { children: React.ReactNode }) => {
|
||||
@@ -106,6 +107,14 @@ const mainRoutes: RouteObject[] = [
|
||||
</Suspense>
|
||||
),
|
||||
},
|
||||
{
|
||||
path: 'gantt-demo',
|
||||
element: (
|
||||
<Suspense fallback={<SuspenseFallback />}>
|
||||
<GanttDemoPage />
|
||||
</Suspense>
|
||||
),
|
||||
},
|
||||
...settingsRoutes,
|
||||
...adminCenterRoutes,
|
||||
],
|
||||
|
||||
@@ -7,6 +7,7 @@ import userReducer from '@features/user/userSlice';
|
||||
|
||||
// Home Page
|
||||
import homePageReducer from '@features/home-page/home-page.slice';
|
||||
import userActivityReducer from '@features/home-page/user-activity.slice';
|
||||
|
||||
// Account Setup
|
||||
import accountSetupReducer from '@features/account-setup/account-setup.slice';
|
||||
@@ -82,6 +83,7 @@ import groupingReducer from '@/features/task-management/grouping.slice';
|
||||
import selectionReducer from '@/features/task-management/selection.slice';
|
||||
import homePageApiService from '@/api/home-page/home-page.api.service';
|
||||
import { projectsApi } from '@/api/projects/projects.v1.api.service';
|
||||
import { userActivityApiService } from '@/api/home-page/user-activity.api.service';
|
||||
|
||||
import projectViewReducer from '@features/project/project-view-slice';
|
||||
import taskManagementFieldsReducer from '@features/task-management/taskListFields.slice';
|
||||
@@ -94,7 +96,7 @@ export const store = configureStore({
|
||||
middleware: getDefaultMiddleware =>
|
||||
getDefaultMiddleware({
|
||||
serializableCheck: false,
|
||||
}).concat(homePageApiService.middleware, projectsApi.middleware),
|
||||
}).concat(homePageApiService.middleware, projectsApi.middleware, userActivityApiService.middleware),
|
||||
reducer: {
|
||||
// Auth & User
|
||||
auth: authReducer,
|
||||
@@ -107,6 +109,9 @@ export const store = configureStore({
|
||||
homePageReducer: homePageReducer,
|
||||
[homePageApiService.reducerPath]: homePageApiService.reducer,
|
||||
[projectsApi.reducerPath]: projectsApi.reducer,
|
||||
userActivityReducer: userActivityReducer,
|
||||
[userActivityApiService.reducerPath]: userActivityApiService.reducer,
|
||||
|
||||
// Core UI
|
||||
themeReducer: themeReducer,
|
||||
localesReducer: localesReducer,
|
||||
|
||||
@@ -11,8 +11,10 @@ import { useAuthService } from '@/hooks/useAuth';
|
||||
import { Avatar, Button, Checkbox } from '@/components';
|
||||
import { sortTeamMembers } from '@/utils/sort-team-members';
|
||||
import { useAppDispatch } from '@/hooks/useAppDispatch';
|
||||
import { toggleProjectMemberDrawer } from '@/features/projects/singleProject/members/projectMembersSlice';
|
||||
import { setIsFromAssigner, toggleProjectMemberDrawer } from '@/features/projects/singleProject/members/projectMembersSlice';
|
||||
import { updateEnhancedKanbanTaskAssignees } from '@/features/enhanced-kanban/enhanced-kanban.slice';
|
||||
import useIsProjectManager from '@/hooks/useIsProjectManager';
|
||||
import { useAuthStatus } from '@/hooks/useAuthStatus';
|
||||
|
||||
interface AssigneeSelectorProps {
|
||||
task: IProjectTask;
|
||||
@@ -42,6 +44,8 @@ const AssigneeSelector: React.FC<AssigneeSelectorProps> = ({
|
||||
const currentSession = useAuthService().getCurrentSession();
|
||||
const { socket } = useSocket();
|
||||
const dispatch = useAppDispatch();
|
||||
const { isAdmin } = useAuthStatus();
|
||||
const isProjectManager = useIsProjectManager();
|
||||
|
||||
const filteredMembers = useMemo(() => {
|
||||
return teamMembers?.data?.filter(member =>
|
||||
@@ -63,12 +67,8 @@ const AssigneeSelector: React.FC<AssigneeSelectorProps> = ({
|
||||
// Close dropdown when clicking outside and handle scroll
|
||||
useEffect(() => {
|
||||
const handleClickOutside = (event: MouseEvent) => {
|
||||
if (
|
||||
dropdownRef.current &&
|
||||
!dropdownRef.current.contains(event.target as Node) &&
|
||||
buttonRef.current &&
|
||||
!buttonRef.current.contains(event.target as Node)
|
||||
) {
|
||||
if (dropdownRef.current && !dropdownRef.current.contains(event.target as Node) &&
|
||||
buttonRef.current && !buttonRef.current.contains(event.target as Node)) {
|
||||
setIsOpen(false);
|
||||
}
|
||||
};
|
||||
@@ -78,9 +78,7 @@ const AssigneeSelector: React.FC<AssigneeSelectorProps> = ({
|
||||
// Check if the button is still visible in the viewport
|
||||
if (buttonRef.current) {
|
||||
const rect = buttonRef.current.getBoundingClientRect();
|
||||
const isVisible =
|
||||
rect.top >= 0 &&
|
||||
rect.left >= 0 &&
|
||||
const isVisible = rect.top >= 0 && rect.left >= 0 &&
|
||||
rect.bottom <= window.innerHeight &&
|
||||
rect.right <= window.innerWidth;
|
||||
|
||||
@@ -167,8 +165,10 @@ const AssigneeSelector: React.FC<AssigneeSelectorProps> = ({
|
||||
setTeamMembers(prev => ({
|
||||
...prev,
|
||||
data: (prev.data || []).map(member =>
|
||||
member.id === memberId ? { ...member, selected: checked } : member
|
||||
),
|
||||
member.id === memberId
|
||||
? { ...member, selected: checked }
|
||||
: member
|
||||
)
|
||||
}));
|
||||
|
||||
const body = {
|
||||
@@ -199,15 +199,15 @@ const AssigneeSelector: React.FC<AssigneeSelectorProps> = ({
|
||||
const checkMemberSelected = (memberId: string) => {
|
||||
if (!memberId) return false;
|
||||
// Use optimistic assignees if available, otherwise fall back to task assignees
|
||||
const assignees =
|
||||
optimisticAssignees.length > 0
|
||||
? optimisticAssignees
|
||||
: task?.assignees?.map(assignee => assignee.team_member_id) || [];
|
||||
const assignees = optimisticAssignees.length > 0
|
||||
? optimisticAssignees
|
||||
: task?.assignees?.map(assignee => assignee.team_member_id) || [];
|
||||
return assignees.includes(memberId);
|
||||
};
|
||||
|
||||
const handleInviteProjectMemberDrawer = () => {
|
||||
setIsOpen(false); // Close the assignee dropdown first
|
||||
dispatch(setIsFromAssigner(true));
|
||||
dispatch(toggleProjectMemberDrawer()); // Then open the invite drawer
|
||||
};
|
||||
|
||||
@@ -219,14 +219,13 @@ const AssigneeSelector: React.FC<AssigneeSelectorProps> = ({
|
||||
className={`
|
||||
w-5 h-5 rounded-full border border-dashed flex items-center justify-center
|
||||
transition-colors duration-200
|
||||
${
|
||||
isOpen
|
||||
? isDarkMode
|
||||
? 'border-blue-500 bg-blue-900/20 text-blue-400'
|
||||
: 'border-blue-500 bg-blue-50 text-blue-600'
|
||||
: isDarkMode
|
||||
? 'border-gray-600 hover:border-gray-500 hover:bg-gray-800 text-gray-400'
|
||||
: 'border-gray-300 hover:border-gray-400 hover:bg-gray-100 text-gray-600'
|
||||
${isOpen
|
||||
? isDarkMode
|
||||
? 'border-blue-500 bg-blue-900/20 text-blue-400'
|
||||
: 'border-blue-500 bg-blue-50 text-blue-600'
|
||||
: isDarkMode
|
||||
? 'border-gray-600 hover:border-gray-500 hover:bg-gray-800 text-gray-400'
|
||||
: 'border-gray-300 hover:border-gray-400 hover:bg-gray-100 text-gray-600'
|
||||
}
|
||||
`}
|
||||
>
|
||||
@@ -240,7 +239,10 @@ const AssigneeSelector: React.FC<AssigneeSelectorProps> = ({
|
||||
onClick={e => e.stopPropagation()}
|
||||
className={`
|
||||
fixed z-[99999] w-72 rounded-md shadow-lg border
|
||||
${isDarkMode ? 'bg-gray-800 border-gray-600' : 'bg-white border-gray-200'}
|
||||
${isDarkMode
|
||||
? 'bg-gray-800 border-gray-600'
|
||||
: 'bg-white border-gray-200'
|
||||
}
|
||||
`}
|
||||
style={{
|
||||
top: dropdownPosition.top,
|
||||
@@ -275,101 +277,84 @@ const AssigneeSelector: React.FC<AssigneeSelectorProps> = ({
|
||||
key={member.id}
|
||||
className={`
|
||||
flex items-center gap-2 p-2 cursor-pointer transition-colors
|
||||
${
|
||||
member.pending_invitation
|
||||
? 'opacity-50 cursor-not-allowed'
|
||||
: isDarkMode
|
||||
? 'hover:bg-gray-700'
|
||||
: 'hover:bg-gray-50'
|
||||
${member.pending_invitation
|
||||
? 'opacity-50 cursor-not-allowed'
|
||||
: isDarkMode
|
||||
? 'hover:bg-gray-700'
|
||||
: 'hover:bg-gray-50'
|
||||
}
|
||||
`}
|
||||
onClick={() => {
|
||||
if (!member.pending_invitation) {
|
||||
const isSelected = checkMemberSelected(member.id || '');
|
||||
handleMemberToggle(member.id || '', !isSelected);
|
||||
}
|
||||
}}
|
||||
style={{
|
||||
// Add visual feedback for immediate response
|
||||
transition: 'all 0.15s ease-in-out',
|
||||
}}
|
||||
>
|
||||
<div className="relative">
|
||||
<span onClick={e => e.stopPropagation()}>
|
||||
<Checkbox
|
||||
checked={checkMemberSelected(member.id || '')}
|
||||
onChange={checked => handleMemberToggle(member.id || '', checked)}
|
||||
disabled={
|
||||
member.pending_invitation || pendingChanges.has(member.id || '')
|
||||
}
|
||||
isDarkMode={isDarkMode}
|
||||
/>
|
||||
</span>
|
||||
{pendingChanges.has(member.id || '') && (
|
||||
<div
|
||||
className={`absolute inset-0 flex items-center justify-center ${
|
||||
isDarkMode ? 'bg-gray-800/50' : 'bg-white/50'
|
||||
}`}
|
||||
>
|
||||
<div
|
||||
className={`w-3 h-3 border border-t-transparent rounded-full animate-spin ${
|
||||
isDarkMode ? 'border-blue-400' : 'border-blue-600'
|
||||
}`}
|
||||
/>
|
||||
</div>
|
||||
onClick={() => {
|
||||
if (!member.pending_invitation) {
|
||||
const isSelected = checkMemberSelected(member.id || '');
|
||||
handleMemberToggle(member.id || '', !isSelected);
|
||||
}
|
||||
}}
|
||||
style={{
|
||||
// Add visual feedback for immediate response
|
||||
transition: 'all 0.15s ease-in-out',
|
||||
}}
|
||||
>
|
||||
<div className="relative">
|
||||
<span onClick={e => e.stopPropagation()}>
|
||||
<Checkbox
|
||||
checked={checkMemberSelected(member.id || '')}
|
||||
onChange={(checked) => handleMemberToggle(member.id || '', checked)}
|
||||
disabled={member.pending_invitation || pendingChanges.has(member.id || '')}
|
||||
isDarkMode={isDarkMode}
|
||||
/>
|
||||
</span>
|
||||
{pendingChanges.has(member.id || '') && (
|
||||
<div className={`absolute inset-0 flex items-center justify-center ${isDarkMode ? 'bg-gray-800/50' : 'bg-white/50'
|
||||
}`}>
|
||||
<div className={`w-3 h-3 border border-t-transparent rounded-full animate-spin ${isDarkMode ? 'border-blue-400' : 'border-blue-600'
|
||||
}`} />
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
<Avatar
|
||||
src={member.avatar_url}
|
||||
name={member.name || ''}
|
||||
size={24}
|
||||
isDarkMode={isDarkMode}
|
||||
/>
|
||||
|
||||
<div className="flex-1 min-w-0">
|
||||
<div className={`text-xs font-medium truncate ${isDarkMode ? 'text-gray-100' : 'text-gray-900'}`}>
|
||||
{member.name}
|
||||
</div>
|
||||
<div className={`text-xs truncate ${isDarkMode ? 'text-gray-400' : 'text-gray-500'}`}>
|
||||
{member.email}
|
||||
{member.pending_invitation && (
|
||||
<span className="text-red-400 ml-1">(Pending)</span>
|
||||
)}
|
||||
</div>
|
||||
|
||||
<Avatar
|
||||
src={member.avatar_url}
|
||||
name={member.name || ''}
|
||||
size={24}
|
||||
isDarkMode={isDarkMode}
|
||||
/>
|
||||
{/* Footer */}
|
||||
|
||||
<div className="flex-1 min-w-0">
|
||||
<div
|
||||
className={`text-xs font-medium truncate ${isDarkMode ? 'text-gray-100' : 'text-gray-900'}`}
|
||||
>
|
||||
{member.name}
|
||||
</div>
|
||||
<div
|
||||
className={`text-xs truncate ${isDarkMode ? 'text-gray-400' : 'text-gray-500'}`}
|
||||
>
|
||||
{member.email}
|
||||
{member.pending_invitation && (
|
||||
<span className="text-red-400 ml-1">(Pending)</span>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
))
|
||||
) : (
|
||||
<div
|
||||
className={`p-4 text-center ${isDarkMode ? 'text-gray-400' : 'text-gray-500'}`}
|
||||
>
|
||||
<div className="text-xs">No members found</div>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Footer */}
|
||||
{(isAdmin || isProjectManager) && (
|
||||
<div className={`p-2 border-t ${isDarkMode ? 'border-gray-600' : 'border-gray-200'}`}>
|
||||
<button
|
||||
className={`
|
||||
w-full flex items-center justify-center gap-1 px-2 py-1 text-xs rounded
|
||||
transition-colors
|
||||
${isDarkMode ? 'text-blue-400 hover:bg-gray-700' : 'text-blue-600 hover:bg-blue-50'}
|
||||
`}
|
||||
w-full flex items-center justify-center gap-1 px-2 py-1 text-xs rounded
|
||||
transition-colors
|
||||
${isDarkMode
|
||||
? 'text-blue-400 hover:bg-gray-700'
|
||||
: 'text-blue-600 hover:bg-blue-50'
|
||||
}
|
||||
`}
|
||||
onClick={handleInviteProjectMemberDrawer}
|
||||
>
|
||||
<UserAddOutlined />
|
||||
Invite member
|
||||
</button>
|
||||
</div>
|
||||
</div>,
|
||||
document.body
|
||||
)}
|
||||
)}
|
||||
|
||||
</div>,
|
||||
document.body
|
||||
)}
|
||||
</>
|
||||
);
|
||||
};
|
||||
|
||||
@@ -1,19 +0,0 @@
|
||||
@media (max-width: 1000px) {
|
||||
.step-content,
|
||||
.step-form,
|
||||
.create-first-task-form,
|
||||
.setup-action-buttons,
|
||||
.invite-members-form {
|
||||
width: 400px !important;
|
||||
}
|
||||
}
|
||||
|
||||
@media (max-width: 500px) {
|
||||
.step-content,
|
||||
.step-form,
|
||||
.create-first-task-form,
|
||||
.setup-action-buttons,
|
||||
.invite-members-form {
|
||||
width: 200px !important;
|
||||
}
|
||||
}
|
||||
@@ -1,16 +1,15 @@
|
||||
import React, { useEffect, useRef } from 'react';
|
||||
import { Form, Input, Button, List, Alert, message, InputRef } from '@/shared/antd-imports';
|
||||
import { CloseCircleOutlined, MailOutlined, PlusOutlined } from '@/shared/antd-imports';
|
||||
import React, { useEffect, useRef, useState } from 'react';
|
||||
import { Form, Input, Button, Typography, Card, Avatar, Tag, Alert, Space, Dropdown, MenuProps } from '@/shared/antd-imports';
|
||||
import { CloseCircleOutlined, MailOutlined, PlusOutlined, UserOutlined, CheckCircleOutlined, ExclamationCircleOutlined, GlobalOutlined } from '@/shared/antd-imports';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { Typography } from '@/shared/antd-imports';
|
||||
import { setTeamMembers, setTasks } from '@/features/account-setup/account-setup.slice';
|
||||
import { setTeamMembers } from '@/features/account-setup/account-setup.slice';
|
||||
import { useDispatch, useSelector } from 'react-redux';
|
||||
import { RootState } from '@/app/store';
|
||||
import { validateEmail } from '@/utils/validateEmail';
|
||||
import { sanitizeInput } from '@/utils/sanitizeInput';
|
||||
import { Rule } from 'antd/es/form';
|
||||
import { setLanguage } from '@/features/i18n/localesSlice';
|
||||
|
||||
const { Title } = Typography;
|
||||
const { Title, Paragraph, Text } = Typography;
|
||||
|
||||
interface Email {
|
||||
id: number;
|
||||
@@ -20,163 +19,233 @@ interface Email {
|
||||
interface MembersStepProps {
|
||||
isDarkMode: boolean;
|
||||
styles: any;
|
||||
token?: any;
|
||||
}
|
||||
|
||||
const MembersStep: React.FC<MembersStepProps> = ({ isDarkMode, styles }) => {
|
||||
const { t } = useTranslation('account-setup');
|
||||
const getEmailSuggestions = (orgName?: string) => {
|
||||
if (!orgName) return [];
|
||||
const cleanOrgName = orgName.toLowerCase().replace(/[^a-z0-9]/g, '');
|
||||
return [`info@${cleanOrgName}.com`, `team@${cleanOrgName}.com`, `hello@${cleanOrgName}.com`, `contact@${cleanOrgName}.com`];
|
||||
};
|
||||
|
||||
const getRoleSuggestions = (t: any) => [
|
||||
{ role: 'Designer', icon: '🎨', description: t('roleSuggestions.designer') },
|
||||
{ role: 'Developer', icon: '💻', description: t('roleSuggestions.developer') },
|
||||
{ role: 'Project Manager', icon: '📊', description: t('roleSuggestions.projectManager') },
|
||||
{ role: 'Marketing', icon: '📢', description: t('roleSuggestions.marketing') },
|
||||
{ role: 'Sales', icon: '💼', description: t('roleSuggestions.sales') },
|
||||
{ role: 'Operations', icon: '⚙️', description: t('roleSuggestions.operations') }
|
||||
];
|
||||
|
||||
const MembersStep: React.FC<MembersStepProps> = ({ isDarkMode, styles, token }) => {
|
||||
const { t, i18n } = useTranslation('account-setup');
|
||||
const { teamMembers, organizationName } = useSelector(
|
||||
(state: RootState) => state.accountSetupReducer
|
||||
);
|
||||
const inputRefs = useRef<(InputRef | null)[]>([]);
|
||||
const { language } = useSelector((state: RootState) => state.localesReducer);
|
||||
const inputRefs = useRef<(HTMLInputElement | null)[]>([]);
|
||||
const dispatch = useDispatch();
|
||||
const [form] = Form.useForm();
|
||||
const [focusedIndex, setFocusedIndex] = useState<number | null>(null);
|
||||
const [showSuggestions, setShowSuggestions] = useState(false);
|
||||
const [validatedEmails, setValidatedEmails] = useState<Set<number>>(new Set());
|
||||
|
||||
const emailSuggestions = getEmailSuggestions(organizationName);
|
||||
|
||||
const addEmail = () => {
|
||||
if (teamMembers.length == 5) return;
|
||||
|
||||
if (teamMembers.length >= 5) return;
|
||||
const newId = teamMembers.length > 0 ? Math.max(...teamMembers.map(t => t.id)) + 1 : 0;
|
||||
dispatch(setTeamMembers([...teamMembers, { id: newId, value: '' }]));
|
||||
setTimeout(() => {
|
||||
inputRefs.current[newId]?.focus();
|
||||
}, 0);
|
||||
setTimeout(() => inputRefs.current[teamMembers.length]?.focus(), 100);
|
||||
};
|
||||
|
||||
const removeEmail = (id: number) => {
|
||||
if (teamMembers.length > 1) {
|
||||
dispatch(setTeamMembers(teamMembers.filter(teamMember => teamMember.id !== id)));
|
||||
}
|
||||
if (teamMembers.length > 1) dispatch(setTeamMembers(teamMembers.filter(teamMember => teamMember.id !== id)));
|
||||
};
|
||||
|
||||
const updateEmail = (id: number, value: string) => {
|
||||
const sanitizedValue = sanitizeInput(value);
|
||||
dispatch(
|
||||
setTeamMembers(
|
||||
teamMembers.map(teamMember =>
|
||||
teamMember.id === id ? { ...teamMember, value: sanitizedValue } : teamMember
|
||||
)
|
||||
)
|
||||
);
|
||||
dispatch(setTeamMembers(teamMembers.map(teamMember => teamMember.id === id ? { ...teamMember, value: sanitizedValue } : teamMember)));
|
||||
};
|
||||
|
||||
const handleKeyPress = (e: React.KeyboardEvent<HTMLInputElement>) => {
|
||||
const input = e.currentTarget as HTMLInputElement;
|
||||
if (!input.value.trim()) return;
|
||||
e.preventDefault();
|
||||
addEmail();
|
||||
const handleKeyPress = (e: React.KeyboardEvent<HTMLInputElement>, index: number) => {
|
||||
if (e.key === 'Enter') {
|
||||
const input = e.currentTarget as HTMLInputElement;
|
||||
if (input.value.trim() && validateEmail(input.value.trim())) {
|
||||
e.preventDefault();
|
||||
if (index === teamMembers.length - 1 && teamMembers.length < 5) addEmail();
|
||||
else if (index < teamMembers.length - 1) inputRefs.current[index + 1]?.focus();
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
// Function to set ref that doesn't return anything (void)
|
||||
const setInputRef = (index: number) => (el: InputRef | null) => {
|
||||
inputRefs.current[index] = el;
|
||||
const handleSuggestionClick = (suggestion: string) => {
|
||||
const emptyEmailIndex = teamMembers.findIndex(member => !member.value.trim());
|
||||
if (emptyEmailIndex !== -1) {
|
||||
updateEmail(teamMembers[emptyEmailIndex].id, suggestion);
|
||||
} else if (teamMembers.length < 5) {
|
||||
const newId = teamMembers.length > 0 ? Math.max(...teamMembers.map(t => t.id)) + 1 : 0;
|
||||
dispatch(setTeamMembers([...teamMembers, { id: newId, value: suggestion }]));
|
||||
}
|
||||
setShowSuggestions(false);
|
||||
};
|
||||
|
||||
useEffect(() => {
|
||||
setTimeout(() => {
|
||||
inputRefs.current[teamMembers.length - 1]?.focus();
|
||||
// Set initial form values
|
||||
const initialValues: Record<string, string> = {};
|
||||
teamMembers.forEach(teamMember => {
|
||||
initialValues[`email-${teamMember.id}`] = teamMember.value;
|
||||
});
|
||||
form.setFieldsValue(initialValues);
|
||||
}, 200);
|
||||
setTimeout(() => inputRefs.current[0]?.focus(), 200);
|
||||
}, []);
|
||||
|
||||
const formRules = {
|
||||
email: [
|
||||
{
|
||||
validator: async (_: any, value: string) => {
|
||||
if (!value) return;
|
||||
if (!validateEmail(value)) {
|
||||
throw new Error(t('invalidEmail'));
|
||||
}
|
||||
},
|
||||
},
|
||||
],
|
||||
const getEmailStatus = (email: string, memberId: number) => {
|
||||
if (!email.trim()) return 'empty';
|
||||
if (!validatedEmails.has(memberId)) return 'empty';
|
||||
return validateEmail(email) ? 'valid' : 'invalid';
|
||||
};
|
||||
|
||||
const handleBlur = (memberId: number, email: string) => {
|
||||
setFocusedIndex(null);
|
||||
if (email.trim()) setValidatedEmails(prev => new Set(prev).add(memberId));
|
||||
};
|
||||
|
||||
const languages = [
|
||||
{ key: 'en', label: t('languages.en'), flag: '🇺🇸' },
|
||||
{ key: 'es', label: t('languages.es'), flag: '🇪🇸' },
|
||||
{ key: 'pt', label: t('languages.pt'), flag: '🇵🇹' },
|
||||
{ key: 'de', label: t('languages.de'), flag: '🇩🇪' },
|
||||
{ key: 'alb', label: t('languages.alb'), flag: '🇦🇱' },
|
||||
{ key: 'zh', label: t('languages.zh'), flag: '🇨🇳' }
|
||||
];
|
||||
|
||||
const handleLanguageChange = (languageKey: string) => {
|
||||
dispatch(setLanguage(languageKey));
|
||||
i18n.changeLanguage(languageKey);
|
||||
};
|
||||
|
||||
const currentLanguage = languages.find(lang => lang.key === language) || languages[0];
|
||||
const languageMenuItems: MenuProps['items'] = languages.map(lang => ({ key: lang.key, label: <div className="flex items-center space-x-2"><span>{lang.flag}</span><span>{lang.label}</span></div>, onClick: () => handleLanguageChange(lang.key) }));
|
||||
|
||||
return (
|
||||
<Form
|
||||
form={form}
|
||||
className="invite-members-form"
|
||||
style={{
|
||||
minHeight: '300px',
|
||||
width: '600px',
|
||||
paddingBottom: '1rem',
|
||||
marginBottom: '3rem',
|
||||
marginTop: '3rem',
|
||||
display: 'flex',
|
||||
flexDirection: 'column',
|
||||
}}
|
||||
>
|
||||
<Form.Item>
|
||||
<Title level={2} style={{ marginBottom: '1rem' }}>
|
||||
{t('step3Title')} "<mark>{organizationName}</mark>".
|
||||
<div className="w-full members-step">
|
||||
{/* Header */}
|
||||
<div className="text-center mb-8">
|
||||
<Title level={3} className="mb-2" style={{ color: token?.colorText }}>
|
||||
{t('membersStepTitle')}
|
||||
</Title>
|
||||
</Form.Item>
|
||||
<Form.Item
|
||||
layout="vertical"
|
||||
rules={[{ required: true }]}
|
||||
label={
|
||||
<span className="font-medium">
|
||||
{t('step3InputLabel')} <MailOutlined /> {t('maxMembers')}
|
||||
</span>
|
||||
}
|
||||
>
|
||||
<List
|
||||
dataSource={teamMembers}
|
||||
bordered={false}
|
||||
itemLayout="vertical"
|
||||
renderItem={(teamMember, index) => (
|
||||
<List.Item key={teamMember.id}>
|
||||
<div className="invite-members-form" style={{ display: 'flex', width: '600px' }}>
|
||||
<Form.Item
|
||||
rules={formRules.email as Rule[]}
|
||||
className="w-full"
|
||||
validateTrigger={['onChange', 'onBlur']}
|
||||
name={`email-${teamMember.id}`}
|
||||
>
|
||||
<Paragraph className="text-base" style={{ color: token?.colorTextSecondary }}>
|
||||
{t('membersStepDescription', { organizationName })}
|
||||
</Paragraph>
|
||||
</div>
|
||||
|
||||
{/* Team Members List */}
|
||||
<div className="mb-6">
|
||||
<div className="space-y-3">
|
||||
{teamMembers.map((teamMember, index) => {
|
||||
const emailStatus = getEmailStatus(teamMember.value, teamMember.id);
|
||||
return (
|
||||
<div
|
||||
key={teamMember.id}
|
||||
className={`flex items-center space-x-3 p-3 rounded-lg border transition-all duration-200 ${
|
||||
focusedIndex === index ? 'border-2' : ''
|
||||
}`}
|
||||
style={{
|
||||
borderColor: focusedIndex === index ? token?.colorPrimary :
|
||||
emailStatus === 'invalid' ? token?.colorError : token?.colorBorder,
|
||||
backgroundColor: token?.colorBgContainer
|
||||
}}
|
||||
>
|
||||
<Avatar
|
||||
size={32}
|
||||
style={{
|
||||
backgroundColor: emailStatus === 'valid' ? token?.colorSuccess :
|
||||
emailStatus === 'invalid' ? token?.colorError : token?.colorBorderSecondary,
|
||||
color: '#fff'
|
||||
}}
|
||||
icon={
|
||||
emailStatus === 'valid' ? <CheckCircleOutlined /> :
|
||||
emailStatus === 'invalid' ? <ExclamationCircleOutlined /> :
|
||||
<UserOutlined />
|
||||
}
|
||||
/>
|
||||
|
||||
<div className="flex-1">
|
||||
<Input
|
||||
placeholder={t('emailPlaceholder')}
|
||||
placeholder={t('memberPlaceholder', { index: index + 1 })}
|
||||
value={teamMember.value}
|
||||
onChange={e => updateEmail(teamMember.id, e.target.value)}
|
||||
onPressEnter={handleKeyPress}
|
||||
ref={setInputRef(index)}
|
||||
status={teamMember.value && !validateEmail(teamMember.value) ? 'error' : ''}
|
||||
id={`member-${index}`}
|
||||
onKeyPress={e => handleKeyPress(e, index)}
|
||||
onFocus={() => setFocusedIndex(index)}
|
||||
onBlur={() => handleBlur(teamMember.id, teamMember.value)}
|
||||
ref={el => inputRefs.current[index] = el}
|
||||
className="border-0 shadow-none"
|
||||
style={{
|
||||
backgroundColor: 'transparent',
|
||||
color: token?.colorText
|
||||
}}
|
||||
prefix={<MailOutlined style={{ color: token?.colorTextTertiary }} />}
|
||||
status={emailStatus === 'invalid' ? 'error' : undefined}
|
||||
suffix={
|
||||
emailStatus === 'valid' ? (
|
||||
<CheckCircleOutlined style={{ color: token?.colorSuccess }} />
|
||||
) : emailStatus === 'invalid' ? (
|
||||
<ExclamationCircleOutlined style={{ color: token?.colorError }} />
|
||||
) : null
|
||||
}
|
||||
/>
|
||||
</Form.Item>
|
||||
<Button
|
||||
className="custom-close-button"
|
||||
style={{ marginLeft: '48px' }}
|
||||
type="text"
|
||||
icon={<CloseCircleOutlined />}
|
||||
disabled={teamMembers.length === 1}
|
||||
onClick={() => removeEmail(teamMember.id)}
|
||||
/>
|
||||
{emailStatus === 'invalid' && (
|
||||
<Text type="danger" className="text-xs mt-1 block">
|
||||
{t('invalidEmail')}
|
||||
</Text>
|
||||
)}
|
||||
{emailStatus === 'valid' && (
|
||||
<Text type="success" className="text-xs mt-1 block">
|
||||
{t('validEmailAddress')}
|
||||
</Text>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{teamMembers.length > 1 && (
|
||||
<Button
|
||||
type="text"
|
||||
size="small"
|
||||
icon={<CloseCircleOutlined />}
|
||||
onClick={() => removeEmail(teamMember.id)}
|
||||
style={{ color: token?.colorTextTertiary }}
|
||||
/>
|
||||
)}
|
||||
</div>
|
||||
</List.Item>
|
||||
)}
|
||||
/>
|
||||
<Button
|
||||
type="dashed"
|
||||
icon={<PlusOutlined />}
|
||||
onClick={addEmail}
|
||||
style={{ marginTop: '16px' }}
|
||||
disabled={teamMembers.length == 5}
|
||||
>
|
||||
{t('tasksStepAddAnother')}
|
||||
</Button>
|
||||
<div
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
|
||||
{/* Add Member Button */}
|
||||
{teamMembers.length < 5 && (
|
||||
<Button
|
||||
type="dashed"
|
||||
icon={<PlusOutlined />}
|
||||
onClick={addEmail}
|
||||
className="w-full mt-4 h-12 text-base"
|
||||
style={{
|
||||
borderColor: token?.colorBorder,
|
||||
color: token?.colorTextSecondary
|
||||
}}
|
||||
>
|
||||
{t('addAnotherTeamMember', { current: teamMembers.length, max: 5 })}
|
||||
</Button>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Skip Option */}
|
||||
<div className="mb-6">
|
||||
<Alert
|
||||
message={t('canInviteLater')}
|
||||
description={t('skipStepDescription')}
|
||||
type="info"
|
||||
showIcon
|
||||
style={{
|
||||
marginTop: '24px',
|
||||
display: 'flex',
|
||||
justifyContent: 'space-between',
|
||||
backgroundColor: token?.colorInfoBg,
|
||||
borderColor: token?.colorInfoBorder
|
||||
}}
|
||||
></div>
|
||||
</Form.Item>
|
||||
</Form>
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
export default MembersStep;
|
||||
export default MembersStep;
|
||||
@@ -1,31 +1,40 @@
|
||||
import React, { useEffect, useRef } from 'react';
|
||||
import { Form, Input, InputRef, Typography } from '@/shared/antd-imports';
|
||||
import React, { useEffect, useRef, useState } from 'react';
|
||||
import { Form, Input, InputRef, Typography, Card, Tooltip } from '@/shared/antd-imports';
|
||||
import { useDispatch, useSelector } from 'react-redux';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { setOrganizationName } from '@/features/account-setup/account-setup.slice';
|
||||
import { RootState } from '@/app/store';
|
||||
import { sanitizeInput } from '@/utils/sanitizeInput';
|
||||
import './admin-center-common.css';
|
||||
|
||||
const { Title } = Typography;
|
||||
const { Title, Paragraph, Text } = Typography;
|
||||
|
||||
interface Props {
|
||||
onEnter: () => void;
|
||||
styles: any;
|
||||
organizationNamePlaceholder: string;
|
||||
organizationNameInitialValue?: string;
|
||||
isDarkMode: boolean;
|
||||
token?: any;
|
||||
}
|
||||
|
||||
export const OrganizationStep: React.FC<Props> = ({
|
||||
onEnter,
|
||||
styles,
|
||||
organizationNamePlaceholder,
|
||||
organizationNameInitialValue,
|
||||
isDarkMode,
|
||||
token,
|
||||
}) => {
|
||||
const { t } = useTranslation('account-setup');
|
||||
const dispatch = useDispatch();
|
||||
const { organizationName } = useSelector((state: RootState) => state.accountSetupReducer);
|
||||
const inputRef = useRef<InputRef>(null);
|
||||
|
||||
// Autofill organization name if not already set
|
||||
useEffect(() => {
|
||||
if (!organizationName && organizationNameInitialValue) {
|
||||
dispatch(setOrganizationName(organizationNameInitialValue));
|
||||
}
|
||||
setTimeout(() => inputRef.current?.focus(), 300);
|
||||
}, []);
|
||||
|
||||
@@ -40,25 +49,85 @@ export const OrganizationStep: React.FC<Props> = ({
|
||||
};
|
||||
|
||||
return (
|
||||
<Form className="step-form" style={styles.form}>
|
||||
<Form.Item>
|
||||
<Title level={2} style={{ marginBottom: '1rem' }}>
|
||||
{t('organizationStepTitle')}
|
||||
<div className="w-full organization-step">
|
||||
{/* Header */}
|
||||
<div className="text-center mb-8">
|
||||
<Title level={3} className="mb-2" style={{ color: token?.colorText }}>
|
||||
{t('organizationStepWelcome')}
|
||||
</Title>
|
||||
</Form.Item>
|
||||
<Form.Item
|
||||
layout="vertical"
|
||||
rules={[{ required: true }]}
|
||||
label={<span style={styles.label}>{t('organizationStepLabel')}</span>}
|
||||
<Paragraph className="text-base" style={{ color: token?.colorTextSecondary }}>
|
||||
{t('organizationStepDescription')}
|
||||
</Paragraph>
|
||||
</div>
|
||||
|
||||
{/* Main Form Card */}
|
||||
<div className="mb-6">
|
||||
<Card
|
||||
className="border-2 hover:shadow-md transition-all duration-200"
|
||||
style={{
|
||||
borderColor: token?.colorPrimary,
|
||||
backgroundColor: token?.colorBgContainer
|
||||
}}
|
||||
>
|
||||
<Form.Item
|
||||
className="mb-4"
|
||||
label={
|
||||
<div className="flex items-center space-x-2">
|
||||
<span className="font-medium text-base" style={{ color: token?.colorText }}>
|
||||
{t('organizationStepLabel')}
|
||||
</span>
|
||||
<Tooltip title={t('organizationStepTooltip')}>
|
||||
<span
|
||||
className="text-sm cursor-help"
|
||||
style={{ color: token?.colorTextTertiary }}
|
||||
>
|
||||
ⓘ
|
||||
</span>
|
||||
</Tooltip>
|
||||
</div>
|
||||
}
|
||||
>
|
||||
<Input
|
||||
placeholder={organizationNamePlaceholder || t('organizationStepPlaceholder')}
|
||||
value={organizationName}
|
||||
onChange={handleOrgNameChange}
|
||||
onPressEnter={onPressEnter}
|
||||
ref={inputRef}
|
||||
className="text-base"
|
||||
/>
|
||||
</Form.Item>
|
||||
|
||||
{/* Character Count and Validation */}
|
||||
<div className="flex justify-between items-center text-sm">
|
||||
<Text type="secondary">
|
||||
{organizationName.length}/50 {t('organizationStepCharacters')}
|
||||
</Text>
|
||||
{organizationName.length > 0 && (
|
||||
<div className="flex items-center space-x-1">
|
||||
{organizationName.length >= 2 ? (
|
||||
<span style={{ color: token?.colorSuccess }}>✓ {t('organizationStepGoodLength')}</span>
|
||||
) : (
|
||||
<span style={{ color: token?.colorWarning }}>⚠ {t('organizationStepTooShort')}</span>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</Card>
|
||||
</div>
|
||||
|
||||
{/* Footer Note */}
|
||||
<div
|
||||
className="text-center p-4 rounded-lg"
|
||||
style={{
|
||||
backgroundColor: token?.colorInfoBg,
|
||||
borderColor: token?.colorInfoBorder,
|
||||
border: '1px solid'
|
||||
}}
|
||||
>
|
||||
<Input
|
||||
placeholder={organizationNamePlaceholder}
|
||||
value={organizationName}
|
||||
onChange={handleOrgNameChange}
|
||||
onPressEnter={onPressEnter}
|
||||
ref={inputRef}
|
||||
/>
|
||||
</Form.Item>
|
||||
</Form>
|
||||
<Text type="secondary" className="text-sm">
|
||||
🔒 {t('organizationStepPrivacyNote')}
|
||||
</Text>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
};
|
||||
@@ -3,7 +3,7 @@ import { useSelector } from 'react-redux';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { useNavigate } from 'react-router-dom';
|
||||
|
||||
import { Button, Drawer, Form, Input, InputRef, Select, Typography } from '@/shared/antd-imports';
|
||||
import { Button, Drawer, Form, Input, InputRef, Typography, Card, Row, Col, Tag, Tooltip, Spin, Alert } from '@/shared/antd-imports';
|
||||
import TemplateDrawer from '../common/template-drawer/template-drawer';
|
||||
|
||||
import { RootState } from '@/app/store';
|
||||
@@ -13,7 +13,7 @@ import { sanitizeInput } from '@/utils/sanitizeInput';
|
||||
import { projectTemplatesApiService } from '@/api/project-templates/project-templates.api.service';
|
||||
import logger from '@/utils/errorLogger';
|
||||
|
||||
import { IAccountSetupRequest } from '@/types/project-templates/project-templates.types';
|
||||
import { IAccountSetupRequest, IWorklenzTemplate, IProjectTemplate } from '@/types/project-templates/project-templates.types';
|
||||
|
||||
import { evt_account_setup_template_complete } from '@/shared/worklenz-analytics-events';
|
||||
import { useMixpanelTracking } from '@/hooks/useMixpanelTracking';
|
||||
@@ -24,15 +24,43 @@ import { setUser } from '@/features/user/userSlice';
|
||||
import { setSession } from '@/utils/session-helper';
|
||||
import { IAuthorizeResponse } from '@/types/auth/login.types';
|
||||
|
||||
const { Title } = Typography;
|
||||
const { Title, Paragraph, Text } = Typography;
|
||||
|
||||
interface Props {
|
||||
onEnter: () => void;
|
||||
styles: any;
|
||||
isDarkMode: boolean;
|
||||
token?: any;
|
||||
}
|
||||
|
||||
export const ProjectStep: React.FC<Props> = ({ onEnter, styles, isDarkMode = false }) => {
|
||||
// Default icon mapping for templates (fallback if no image_url)
|
||||
const getTemplateIcon = (name?: string) => {
|
||||
if (!name) return '📁';
|
||||
const lowercaseName = name.toLowerCase();
|
||||
if (lowercaseName.includes('software') || lowercaseName.includes('development')) return '💻';
|
||||
if (lowercaseName.includes('marketing') || lowercaseName.includes('campaign')) return '📢';
|
||||
if (lowercaseName.includes('construction') || lowercaseName.includes('building')) return '🏗️';
|
||||
if (lowercaseName.includes('startup') || lowercaseName.includes('launch')) return '🚀';
|
||||
if (lowercaseName.includes('design') || lowercaseName.includes('creative')) return '🎨';
|
||||
if (lowercaseName.includes('education') || lowercaseName.includes('learning')) return '📚';
|
||||
if (lowercaseName.includes('event') || lowercaseName.includes('planning')) return '📅';
|
||||
if (lowercaseName.includes('retail') || lowercaseName.includes('sales')) return '🛍️';
|
||||
return '📁';
|
||||
};
|
||||
|
||||
const getProjectSuggestions = (orgType?: string) => {
|
||||
const suggestions: Record<string, string[]> = {
|
||||
'freelancer': ['Client Website', 'Logo Design', 'Content Writing', 'App Development'],
|
||||
'startup': ['MVP Development', 'Product Launch', 'Marketing Campaign', 'Investor Pitch'],
|
||||
'small_medium_business': ['Q1 Sales Initiative', 'Website Redesign', 'Process Improvement', 'Team Training'],
|
||||
'agency': ['Client Campaign', 'Brand Strategy', 'Website Project', 'Creative Brief'],
|
||||
'enterprise': ['Digital Transformation', 'System Migration', 'Annual Planning', 'Department Initiative'],
|
||||
'other': ['New Project', 'Team Initiative', 'Q1 Goals', 'Special Project']
|
||||
};
|
||||
return suggestions[orgType || 'other'] || suggestions['other'];
|
||||
};
|
||||
|
||||
export const ProjectStep: React.FC<Props> = ({ onEnter, styles, isDarkMode = false, token }) => {
|
||||
const { t } = useTranslation('account-setup');
|
||||
const dispatch = useAppDispatch();
|
||||
const navigate = useNavigate();
|
||||
@@ -42,13 +70,58 @@ export const ProjectStep: React.FC<Props> = ({ onEnter, styles, isDarkMode = fal
|
||||
|
||||
useEffect(() => {
|
||||
setTimeout(() => inputRef.current?.focus(), 200);
|
||||
fetchTemplates();
|
||||
}, []);
|
||||
|
||||
const { projectName, templateId, organizationName } = useSelector(
|
||||
const fetchTemplates = async () => {
|
||||
try {
|
||||
setLoadingTemplates(true);
|
||||
setTemplateError(null);
|
||||
|
||||
// Fetch list of available templates
|
||||
const templatesResponse = await projectTemplatesApiService.getWorklenzTemplates();
|
||||
|
||||
if (templatesResponse.done && templatesResponse.body) {
|
||||
// Fetch detailed information for first 4 templates for preview
|
||||
const templateDetails = await Promise.all(
|
||||
templatesResponse.body.slice(0, 4).map(async (template) => {
|
||||
if (template.id) {
|
||||
try {
|
||||
const detailResponse = await projectTemplatesApiService.getByTemplateId(template.id);
|
||||
return detailResponse.done ? detailResponse.body : null;
|
||||
} catch (error) {
|
||||
logger.error(`Failed to fetch template details for ${template.id}`, error);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
return null;
|
||||
})
|
||||
);
|
||||
|
||||
// Filter out null results and set templates
|
||||
const validTemplates = templateDetails.filter((template): template is IProjectTemplate => template !== null);
|
||||
setTemplates(validTemplates);
|
||||
}
|
||||
} catch (error) {
|
||||
logger.error('Failed to fetch templates', error);
|
||||
setTemplateError('Failed to load templates');
|
||||
} finally {
|
||||
setLoadingTemplates(false);
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
const { projectName, templateId, organizationName, surveyData } = useSelector(
|
||||
(state: RootState) => state.accountSetupReducer
|
||||
);
|
||||
const [open, setOpen] = useState(false);
|
||||
const [creatingFromTemplate, setCreatingFromTemplate] = useState(false);
|
||||
const [selectedTemplate, setSelectedTemplate] = useState<string | null>(templateId || null);
|
||||
const [templates, setTemplates] = useState<IProjectTemplate[]>([]);
|
||||
const [loadingTemplates, setLoadingTemplates] = useState(true);
|
||||
const [templateError, setTemplateError] = useState<string | null>(null);
|
||||
|
||||
const projectSuggestions = getProjectSuggestions(surveyData.organization_type);
|
||||
|
||||
const handleTemplateSelected = (templateId: string) => {
|
||||
if (!templateId) return;
|
||||
@@ -74,8 +147,6 @@ export const ProjectStep: React.FC<Props> = ({ onEnter, styles, isDarkMode = fal
|
||||
if (res.done && res.body.id) {
|
||||
toggleTemplateSelector(false);
|
||||
trackMixpanelEvent(evt_account_setup_template_complete);
|
||||
|
||||
// Refresh user session to update setup_completed status
|
||||
try {
|
||||
const authResponse = (await dispatch(
|
||||
verifyAuthentication()
|
||||
@@ -87,7 +158,6 @@ export const ProjectStep: React.FC<Props> = ({ onEnter, styles, isDarkMode = fal
|
||||
} catch (error) {
|
||||
logger.error('Failed to refresh user session after template setup completion', error);
|
||||
}
|
||||
|
||||
navigate(`/worklenz/projects/${res.body.id}?tab=tasks-list&pinned_tab=tasks-list`);
|
||||
}
|
||||
} catch (error) {
|
||||
@@ -96,8 +166,7 @@ export const ProjectStep: React.FC<Props> = ({ onEnter, styles, isDarkMode = fal
|
||||
};
|
||||
|
||||
const onPressEnter = () => {
|
||||
if (!projectName.trim()) return;
|
||||
onEnter();
|
||||
if (projectName.trim()) onEnter();
|
||||
};
|
||||
|
||||
const handleProjectNameChange = (e: React.ChangeEvent<HTMLInputElement>) => {
|
||||
@@ -105,43 +174,205 @@ export const ProjectStep: React.FC<Props> = ({ onEnter, styles, isDarkMode = fal
|
||||
dispatch(setProjectName(sanitizedValue));
|
||||
};
|
||||
|
||||
const handleProjectNameFocus = () => {
|
||||
if (templateId) {
|
||||
dispatch(setTemplateId(null));
|
||||
setSelectedTemplate(null);
|
||||
}
|
||||
};
|
||||
|
||||
const handleSuggestionClick = (suggestion: string) => {
|
||||
dispatch(setProjectName(suggestion));
|
||||
inputRef.current?.focus();
|
||||
};
|
||||
|
||||
return (
|
||||
<div>
|
||||
<Form className="step-form" style={styles.form}>
|
||||
<Form.Item>
|
||||
<Title level={2} style={{ marginBottom: '1rem' }}>
|
||||
{t('projectStepTitle')}
|
||||
</Title>
|
||||
</Form.Item>
|
||||
<Form.Item
|
||||
layout="vertical"
|
||||
rules={[{ required: true }]}
|
||||
label={<span style={styles.label}>{t('projectStepLabel')}</span>}
|
||||
>
|
||||
<Input
|
||||
placeholder={t('projectStepPlaceholder')}
|
||||
value={projectName}
|
||||
onChange={handleProjectNameChange}
|
||||
onPressEnter={onPressEnter}
|
||||
ref={inputRef}
|
||||
/>
|
||||
</Form.Item>
|
||||
</Form>
|
||||
<div style={{ position: 'relative' }}>
|
||||
<Title level={4} className={isDarkMode ? 'vert-text-dark' : 'vert-text'}>
|
||||
{t('or')}
|
||||
<div className="w-full project-step">
|
||||
{/* Header */}
|
||||
<div className="text-center mb-8">
|
||||
<Title level={3} className="mb-2" style={{ color: token?.colorText }}>
|
||||
{t('projectStepHeader')}
|
||||
</Title>
|
||||
<div className={isDarkMode ? 'vert-line-dark' : 'vert-line'} />
|
||||
<Paragraph className="text-base" style={{ color: token?.colorTextSecondary }}>
|
||||
{t('projectStepSubheader')}
|
||||
</Paragraph>
|
||||
</div>
|
||||
|
||||
<div className="flex justify-center">
|
||||
<Button onClick={() => toggleTemplateSelector(true)} type="primary">
|
||||
{t('templateButton')}
|
||||
</Button>
|
||||
{/* Project Name Section */}
|
||||
<div className="mb-8">
|
||||
<Card
|
||||
className={`border-2 hover:shadow-md transition-all duration-200 ${
|
||||
templateId ? 'opacity-50' : ''
|
||||
}`}
|
||||
style={{
|
||||
borderColor: templateId ? token?.colorBorder : token?.colorPrimary,
|
||||
backgroundColor: token?.colorBgContainer
|
||||
}}
|
||||
>
|
||||
<div className="mb-4">
|
||||
<div className="flex items-center justify-between">
|
||||
<Text strong className="text-lg" style={{ color: token?.colorText }}>
|
||||
{t('startFromScratch')}
|
||||
</Text>
|
||||
{templateId && (
|
||||
<Text type="secondary" className="text-sm">
|
||||
{t('templateSelected')}
|
||||
</Text>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<Form.Item
|
||||
className="mb-4"
|
||||
label={<span className="font-medium" style={{ color: token?.colorText }}>{t('projectStepLabel')}</span>}
|
||||
>
|
||||
<Input
|
||||
size="large"
|
||||
placeholder={projectSuggestions[0] || t('projectStepPlaceholder')}
|
||||
value={projectName}
|
||||
onChange={handleProjectNameChange}
|
||||
onPressEnter={onPressEnter}
|
||||
onFocus={handleProjectNameFocus}
|
||||
ref={inputRef}
|
||||
className="text-base"
|
||||
style={{ backgroundColor: token?.colorBgContainer, borderColor: token?.colorBorder, color: token?.colorText }}
|
||||
/>
|
||||
</Form.Item>
|
||||
|
||||
<div>
|
||||
<Text type="secondary" className="text-sm">{t('quickSuggestions')}</Text>
|
||||
<div className="mt-2 flex flex-wrap gap-2">
|
||||
{projectSuggestions.map((suggestion, index) => (
|
||||
<button key={index} onClick={() => handleSuggestionClick(suggestion)} className="px-3 py-1 rounded-full text-sm border project-suggestion-button" style={{ backgroundColor: token?.colorBgContainer, borderColor: token?.colorBorder, color: token?.colorTextSecondary }}>
|
||||
{suggestion}
|
||||
</button>
|
||||
))}
|
||||
</div>
|
||||
</div>
|
||||
</Card>
|
||||
</div>
|
||||
|
||||
<div className="relative my-8">
|
||||
<div className="absolute inset-0 flex items-center" style={{ color: token?.colorTextQuaternary }}>
|
||||
<div className="w-full border-t" style={{ borderColor: token?.colorBorder }}></div>
|
||||
</div>
|
||||
<div className="relative flex justify-center">
|
||||
<span className="px-4 text-sm font-medium" style={{ backgroundColor: token?.colorBgLayout, color: token?.colorTextSecondary }}>{t('orText')}</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div>
|
||||
<div className="text-center mb-6">
|
||||
<Title level={4} className="mb-2" style={{ color: token?.colorText }}>{t('startWithTemplate')}</Title>
|
||||
<Text type="secondary">
|
||||
{t('templateHeadStart')}
|
||||
</Text>
|
||||
</div>
|
||||
|
||||
{/* Template Preview Cards */}
|
||||
<div className="mb-6">
|
||||
{loadingTemplates ? (
|
||||
<div className="text-center py-12">
|
||||
<Spin size="large" />
|
||||
<div className="mt-4">
|
||||
<Text type="secondary">Loading templates...</Text>
|
||||
</div>
|
||||
</div>
|
||||
) : templateError ? (
|
||||
<Alert
|
||||
message="Failed to load templates"
|
||||
description={templateError}
|
||||
type="error"
|
||||
showIcon
|
||||
action={
|
||||
<Button size="small" onClick={fetchTemplates}>
|
||||
Retry
|
||||
</Button>
|
||||
}
|
||||
/>
|
||||
) : (
|
||||
<Row gutter={[16, 16]}>
|
||||
{templates.map((template) => (
|
||||
<Col xs={24} sm={12} key={template.id}>
|
||||
<Card
|
||||
hoverable
|
||||
className={`h-full template-preview-card ${
|
||||
selectedTemplate === template.id ? 'selected border-2' : ''
|
||||
}`}
|
||||
style={{
|
||||
borderColor: selectedTemplate === template.id ? token?.colorPrimary : token?.colorBorder,
|
||||
backgroundColor: token?.colorBgContainer
|
||||
}}
|
||||
onClick={() => {
|
||||
setSelectedTemplate(template.id || null);
|
||||
dispatch(setTemplateId(template.id || ''));
|
||||
}}
|
||||
>
|
||||
<div className="flex items-start space-x-3">
|
||||
{template.image_url ? (
|
||||
<img
|
||||
src={template.image_url}
|
||||
alt={template.name}
|
||||
className="w-12 h-12 object-cover rounded"
|
||||
onError={(e) => {
|
||||
// Fallback to icon if image fails to load
|
||||
e.currentTarget.style.display = 'none';
|
||||
if (e.currentTarget.nextSibling) {
|
||||
(e.currentTarget.nextSibling as HTMLElement).style.display = 'block';
|
||||
}
|
||||
}}
|
||||
/>
|
||||
) : null}
|
||||
<span
|
||||
className="text-3xl"
|
||||
style={{ display: template.image_url ? 'none' : 'block' }}
|
||||
>
|
||||
{getTemplateIcon(template.name)}
|
||||
</span>
|
||||
<div className="flex-1">
|
||||
<Text strong className="block mb-2" style={{ color: token?.colorText }}>
|
||||
{template.name || 'Untitled Template'}
|
||||
</Text>
|
||||
<div className="flex flex-wrap gap-1">
|
||||
{template.phases?.slice(0, 3).map((phase, index) => (
|
||||
<Tag key={index} color={phase.color_code || 'blue'} className="text-xs">
|
||||
{phase.name}
|
||||
</Tag>
|
||||
))}
|
||||
{(template.phases?.length || 0) > 3 && (
|
||||
<Tag className="text-xs">+{(template.phases?.length || 0) - 3} more</Tag>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</Card>
|
||||
</Col>
|
||||
))}
|
||||
</Row>
|
||||
)}
|
||||
</div>
|
||||
|
||||
<div className="text-center">
|
||||
<Button type="primary" size="large" icon={<span className="mr-2">🎨</span>} onClick={() => toggleTemplateSelector(true)} className="min-w-[200px]">{t('browseAllTemplates')}</Button>
|
||||
<div className="mt-2">
|
||||
<Text type="secondary" className="text-sm">{t('templatesAvailable')}</Text>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Template Drawer */}
|
||||
{createPortal(
|
||||
<Drawer
|
||||
title={t('templateDrawerTitle')}
|
||||
title={
|
||||
<div>
|
||||
<Title level={4} style={{ marginBottom: 0 }}>
|
||||
{t('templateDrawerTitle')}
|
||||
</Title>
|
||||
<Text type="secondary">
|
||||
{t('chooseTemplate')}
|
||||
</Text>
|
||||
</div>
|
||||
}
|
||||
width={1000}
|
||||
onClose={() => toggleTemplateSelector(false)}
|
||||
open={open}
|
||||
@@ -154,11 +385,13 @@ export const ProjectStep: React.FC<Props> = ({ onEnter, styles, isDarkMode = fal
|
||||
type="primary"
|
||||
onClick={() => createFromTemplate()}
|
||||
loading={creatingFromTemplate}
|
||||
disabled={!templateId}
|
||||
>
|
||||
{t('create')}
|
||||
{t('createProject')}
|
||||
</Button>
|
||||
</div>
|
||||
}
|
||||
style={{ backgroundColor: token?.colorBgLayout }}
|
||||
>
|
||||
<TemplateDrawer
|
||||
showBothTabs={false}
|
||||
@@ -171,4 +404,4 @@ export const ProjectStep: React.FC<Props> = ({ onEnter, styles, isDarkMode = fal
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
};
|
||||
374
worklenz-frontend/src/components/account-setup/survey-step.tsx
Normal file
374
worklenz-frontend/src/components/account-setup/survey-step.tsx
Normal file
@@ -0,0 +1,374 @@
|
||||
import React from 'react';
|
||||
import { Form, Input, Typography, Button, Progress, Space } from '@/shared/antd-imports';
|
||||
import { useDispatch, useSelector } from 'react-redux';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { setSurveyData, setSurveySubStep } from '@/features/account-setup/account-setup.slice';
|
||||
import { RootState } from '@/app/store';
|
||||
import {
|
||||
OrganizationType,
|
||||
UserRole,
|
||||
UseCase,
|
||||
HowHeardAbout,
|
||||
IAccountSetupSurveyData
|
||||
} from '@/types/account-setup/survey.types';
|
||||
|
||||
const { Title, Paragraph } = Typography;
|
||||
const { TextArea } = Input;
|
||||
|
||||
interface Props {
|
||||
onEnter: () => void;
|
||||
styles: any;
|
||||
isDarkMode: boolean;
|
||||
token?: any;
|
||||
isModal?: boolean; // New prop to indicate if used in modal context
|
||||
}
|
||||
|
||||
interface SurveyPageProps {
|
||||
styles: any;
|
||||
isDarkMode: boolean;
|
||||
token?: any;
|
||||
surveyData: IAccountSetupSurveyData;
|
||||
handleSurveyDataChange: (field: keyof IAccountSetupSurveyData, value: any) => void;
|
||||
handleUseCaseToggle?: (value: UseCase) => void;
|
||||
isModal?: boolean;
|
||||
}
|
||||
|
||||
// Page 1: About You
|
||||
const AboutYouPage: React.FC<SurveyPageProps> = ({ styles, token, surveyData, handleSurveyDataChange }) => {
|
||||
const { t } = useTranslation('account-setup');
|
||||
|
||||
const organizationTypeOptions: { value: OrganizationType; label: string; icon?: string }[] = [
|
||||
{ value: 'freelancer', label: t('organizationTypeFreelancer'), icon: '👤' },
|
||||
{ value: 'startup', label: t('organizationTypeStartup'), icon: '🚀' },
|
||||
{ value: 'small_medium_business', label: t('organizationTypeSmallMediumBusiness'), icon: '🏢' },
|
||||
{ value: 'agency', label: t('organizationTypeAgency'), icon: '🎯' },
|
||||
{ value: 'enterprise', label: t('organizationTypeEnterprise'), icon: '🏛️' },
|
||||
{ value: 'other', label: t('organizationTypeOther'), icon: '📋' },
|
||||
];
|
||||
|
||||
const userRoleOptions: { value: UserRole; label: string; icon?: string }[] = [
|
||||
{ value: 'founder_ceo', label: t('userRoleFounderCeo'), icon: '👔' },
|
||||
{ value: 'project_manager', label: t('userRoleProjectManager'), icon: '📊' },
|
||||
{ value: 'software_developer', label: t('userRoleSoftwareDeveloper'), icon: '💻' },
|
||||
{ value: 'designer', label: t('userRoleDesigner'), icon: '🎨' },
|
||||
{ value: 'operations', label: t('userRoleOperations'), icon: '⚙️' },
|
||||
{ value: 'other', label: t('userRoleOther'), icon: '✋' },
|
||||
];
|
||||
|
||||
return (
|
||||
<div className="w-full">
|
||||
<div className="text-center mb-8">
|
||||
<Title level={3} className="mb-2" style={{ color: token?.colorText }}>
|
||||
{t('aboutYouStepTitle')}
|
||||
</Title>
|
||||
<Paragraph className="text-base" style={{ color: token?.colorTextSecondary }}>
|
||||
{t('aboutYouStepDescription')}
|
||||
</Paragraph>
|
||||
</div>
|
||||
|
||||
{/* Organization Type */}
|
||||
<Form.Item className="mb-8">
|
||||
<label className="block font-medium text-base mb-4" style={{ color: token?.colorText }}>
|
||||
{t('orgTypeQuestion')}
|
||||
</label>
|
||||
<div className="grid grid-cols-1 md:grid-cols-2 gap-1">
|
||||
{organizationTypeOptions.map((option) => {
|
||||
const isSelected = surveyData.organization_type === option.value;
|
||||
return (
|
||||
<button
|
||||
key={option.value}
|
||||
onClick={() => handleSurveyDataChange('organization_type', option.value)}
|
||||
className={`p-2 rounded border transition-all duration-200 text-left hover:shadow-sm ${isSelected ? 'border-blue-500 bg-blue-50 dark:bg-blue-900/20' : 'border-gray-200 dark:border-gray-700 hover:border-gray-300 dark:hover:border-gray-600'}`}
|
||||
style={{
|
||||
backgroundColor: isSelected ? undefined : token?.colorBgContainer,
|
||||
borderColor: isSelected ? undefined : token?.colorBorder,
|
||||
}}
|
||||
>
|
||||
<div className="flex items-center space-x-2">
|
||||
<div className={`w-3 h-3 rounded-full border flex items-center justify-center ${isSelected ? 'border-blue-500 bg-blue-500' : 'border-gray-300 dark:border-gray-600'}`}>
|
||||
{isSelected && <div className="w-1.5 h-1.5 bg-white rounded-full"></div>}
|
||||
</div>
|
||||
<span className="text-base">{option.icon}</span>
|
||||
<span
|
||||
className={`font-medium text-xs ${isSelected ? 'text-blue-600 dark:text-blue-400' : ''}`}
|
||||
style={{ color: isSelected ? undefined : token?.colorText }}
|
||||
>
|
||||
{option.label}
|
||||
</span>
|
||||
</div>
|
||||
</button>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
</Form.Item>
|
||||
|
||||
{/* User Role */}
|
||||
<Form.Item className="mb-4">
|
||||
<label className="block font-medium text-base mb-4" style={{ color: token?.colorText }}>
|
||||
{t('userRoleQuestion')}
|
||||
</label>
|
||||
<div className="grid grid-cols-1 md:grid-cols-2 gap-1">
|
||||
{userRoleOptions.map((option) => {
|
||||
const isSelected = surveyData.user_role === option.value;
|
||||
return (
|
||||
<button
|
||||
key={option.value}
|
||||
onClick={() => handleSurveyDataChange('user_role', option.value)}
|
||||
className={`p-2 rounded border transition-all duration-200 text-left hover:shadow-sm ${isSelected ? 'border-blue-500 bg-blue-50 dark:bg-blue-900/20' : 'border-gray-200 dark:border-gray-700 hover:border-gray-300 dark:hover:border-gray-600'}`}
|
||||
style={{
|
||||
backgroundColor: isSelected ? undefined : token?.colorBgContainer,
|
||||
borderColor: isSelected ? undefined : token?.colorBorder,
|
||||
}}
|
||||
>
|
||||
<div className="flex items-center space-x-2">
|
||||
<div className={`w-3 h-3 rounded-full border flex items-center justify-center ${isSelected ? 'border-blue-500 bg-blue-500' : 'border-gray-300 dark:border-gray-600'}`}>
|
||||
{isSelected && <div className="w-1.5 h-1.5 bg-white rounded-full"></div>}
|
||||
</div>
|
||||
<span className="text-base">{option.icon}</span>
|
||||
<span
|
||||
className={`font-medium text-xs ${isSelected ? 'text-blue-600 dark:text-blue-400' : ''}`}
|
||||
style={{ color: isSelected ? undefined : token?.colorText }}
|
||||
>
|
||||
{option.label}
|
||||
</span>
|
||||
</div>
|
||||
</button>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
</Form.Item>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
// Page 2: Your Needs
|
||||
const YourNeedsPage: React.FC<SurveyPageProps> = ({ styles, token, surveyData, handleSurveyDataChange, handleUseCaseToggle }) => {
|
||||
const { t } = useTranslation('account-setup');
|
||||
|
||||
const useCaseOptions: { value: UseCase; label: string; description: string }[] = [
|
||||
{ value: 'task_management', label: t('mainUseCasesTaskManagement'), description: 'Organize and track tasks' },
|
||||
{ value: 'team_collaboration', label: t('mainUseCasesTeamCollaboration'), description: 'Work together seamlessly' },
|
||||
{ value: 'resource_planning', label: t('mainUseCasesResourcePlanning'), description: 'Manage time and resources' },
|
||||
{ value: 'client_communication', label: t('mainUseCasesClientCommunication'), description: 'Stay connected with clients' },
|
||||
{ value: 'time_tracking', label: t('mainUseCasesTimeTracking'), description: 'Monitor project hours' },
|
||||
{ value: 'other', label: t('mainUseCasesOther'), description: 'Something else' },
|
||||
];
|
||||
|
||||
const onUseCaseClick = (value: UseCase) => {
|
||||
if (handleUseCaseToggle) {
|
||||
handleUseCaseToggle(value);
|
||||
} else {
|
||||
const currentUseCases = surveyData.main_use_cases || [];
|
||||
const isSelected = currentUseCases.includes(value);
|
||||
const newUseCases = isSelected ? currentUseCases.filter(useCase => useCase !== value) : [...currentUseCases, value];
|
||||
handleSurveyDataChange('main_use_cases', newUseCases);
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<div className="w-full">
|
||||
<div className="text-center mb-8">
|
||||
<Title level={3} className="mb-2" style={{ color: token?.colorText }}>
|
||||
{t('yourNeedsStepTitle')}
|
||||
</Title>
|
||||
<Paragraph className="text-base" style={{ color: token?.colorTextSecondary }}>
|
||||
{t('yourNeedsStepDescription')}
|
||||
</Paragraph>
|
||||
</div>
|
||||
|
||||
{/* Main Use Cases */}
|
||||
<Form.Item className="mb-8">
|
||||
<label className="block font-medium text-base mb-4" style={{ color: token?.colorText }}>
|
||||
{t('yourNeedsQuestion')}
|
||||
</label>
|
||||
<div className="grid grid-cols-1 gap-1">
|
||||
{useCaseOptions.map((option) => {
|
||||
const isSelected = (surveyData.main_use_cases || []).includes(option.value);
|
||||
return (
|
||||
<button
|
||||
key={option.value}
|
||||
onClick={() => onUseCaseClick(option.value)}
|
||||
className={`p-2 rounded border transition-all duration-200 text-left hover:shadow-sm ${isSelected ? 'border-blue-500 bg-blue-50 dark:bg-blue-900/20' : 'border-gray-200 dark:border-gray-700 hover:border-gray-300 dark:hover:border-gray-600'}`}
|
||||
style={{
|
||||
backgroundColor: isSelected ? undefined : token?.colorBgContainer,
|
||||
borderColor: isSelected ? undefined : token?.colorBorder,
|
||||
}}
|
||||
>
|
||||
<div className="flex items-center space-x-2">
|
||||
<div className={`w-3 h-3 rounded border flex items-center justify-center ${isSelected ? 'border-blue-500 bg-blue-500' : 'border-gray-300 dark:border-gray-600'}`}>
|
||||
{isSelected && (
|
||||
<svg width="10" height="10" fill="white" viewBox="0 0 20 20">
|
||||
<path fillRule="evenodd" d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z" clipRule="evenodd" />
|
||||
</svg>
|
||||
)}
|
||||
</div>
|
||||
<div className="flex-1">
|
||||
<span className={`font-medium text-xs ${isSelected ? 'text-blue-600 dark:text-blue-400' : ''}`} style={{ color: isSelected ? undefined : token?.colorText }}>{option.label}</span>
|
||||
<span className="text-xs ml-2" style={{ color: token?.colorTextSecondary }}>- {option.description}</span>
|
||||
</div>
|
||||
</div>
|
||||
</button>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
{surveyData.main_use_cases && surveyData.main_use_cases.length > 0 && (
|
||||
<p className="mt-3 text-sm" style={{ color: token?.colorTextSecondary }}>
|
||||
{surveyData.main_use_cases.length} {t('selected')}
|
||||
</p>
|
||||
)}
|
||||
</Form.Item>
|
||||
|
||||
{/* Previous Tools */}
|
||||
<Form.Item className="mb-4">
|
||||
<label className="block font-medium text-base mb-2" style={{ color: token?.colorText }}>
|
||||
{t('previousToolsLabel')}
|
||||
</label>
|
||||
<TextArea
|
||||
placeholder="e.g., Asana, Trello, Jira, Monday.com, etc."
|
||||
value={surveyData.previous_tools || ''}
|
||||
onChange={(e) => handleSurveyDataChange('previous_tools', e.target.value)}
|
||||
autoSize={{ minRows: 3, maxRows: 5 }}
|
||||
className="text-base"
|
||||
style={{ backgroundColor: token?.colorBgContainer, borderColor: token?.colorBorder, color: token?.colorText }}
|
||||
/>
|
||||
</Form.Item>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
// Page 3: Discovery
|
||||
const DiscoveryPage: React.FC<SurveyPageProps> = ({ styles, token, surveyData, handleSurveyDataChange, isModal }) => {
|
||||
const { t } = useTranslation('account-setup');
|
||||
|
||||
const howHeardAboutOptions: { value: HowHeardAbout; label: string; icon: string }[] = [
|
||||
{ value: 'google_search', label: t('howHeardAboutGoogleSearch'), icon: '🔍' },
|
||||
{ value: 'twitter', label: t('howHeardAboutTwitter'), icon: '🐦' },
|
||||
{ value: 'linkedin', label: t('howHeardAboutLinkedin'), icon: '💼' },
|
||||
{ value: 'friend_colleague', label: t('howHeardAboutFriendColleague'), icon: '👥' },
|
||||
{ value: 'blog_article', label: t('howHeardAboutBlogArticle'), icon: '📰' },
|
||||
{ value: 'other', label: t('howHeardAboutOther'), icon: '💡' },
|
||||
];
|
||||
|
||||
return (
|
||||
<div className="w-full">
|
||||
<div className="text-center mb-8">
|
||||
<Title level={3} className="mb-2" style={{ color: token?.colorText }}>
|
||||
{t('discoveryTitle')}
|
||||
</Title>
|
||||
<Paragraph className="text-base" style={{ color: token?.colorTextSecondary }}>
|
||||
{t('discoveryDescription')}
|
||||
</Paragraph>
|
||||
</div>
|
||||
|
||||
{/* How Heard About */}
|
||||
<Form.Item className="mb-8">
|
||||
<label className="block font-medium text-base mb-4" style={{ color: token?.colorText }}>
|
||||
{t('discoveryQuestion')}
|
||||
</label>
|
||||
<div className="grid grid-cols-1 md:grid-cols-2 gap-1">
|
||||
{howHeardAboutOptions.map((option) => {
|
||||
const isSelected = surveyData.how_heard_about === option.value;
|
||||
return (
|
||||
<button
|
||||
key={option.value}
|
||||
onClick={() => handleSurveyDataChange('how_heard_about', option.value)}
|
||||
className={`p-2 rounded border transition-all duration-200 hover:shadow-sm ${isSelected ? 'border-blue-500 bg-blue-50 dark:bg-blue-900/20' : 'border-gray-200 dark:border-gray-700 hover:border-gray-300 dark:hover:border-gray-600'}`}
|
||||
style={{
|
||||
backgroundColor: isSelected ? undefined : token?.colorBgContainer,
|
||||
borderColor: isSelected ? undefined : token?.colorBorder,
|
||||
}}
|
||||
>
|
||||
<div className="flex items-center space-x-2">
|
||||
<div className={`w-3 h-3 rounded-full border flex items-center justify-center ${isSelected ? 'border-blue-500 bg-blue-500' : 'border-gray-300 dark:border-gray-600'}`}>
|
||||
{isSelected && <div className="w-1.5 h-1.5 bg-white rounded-full"></div>}
|
||||
</div>
|
||||
<span className="text-base">{option.icon}</span>
|
||||
<span className={`font-medium text-xs ${isSelected ? 'text-blue-600 dark:text-blue-400' : ''}`} style={{ color: isSelected ? undefined : token?.colorText }}>{option.label}</span>
|
||||
</div>
|
||||
</button>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
</Form.Item>
|
||||
|
||||
<div className="mt-12 p-1.5 rounded-lg text-center" style={{ backgroundColor: token?.colorSuccessBg, borderColor: token?.colorSuccessBorder, border: '1px solid' }}>
|
||||
<div className="text-4xl mb-3">🎉</div>
|
||||
<Title level={4} style={{ color: token?.colorText, marginBottom: 8 }}>
|
||||
{isModal ? t('surveyCompleteTitle') : t('allSetTitle')}
|
||||
</Title>
|
||||
<Paragraph style={{ color: token?.colorTextSecondary, marginBottom: 0 }}>
|
||||
{isModal ? t('surveyCompleteDescription') : t('allSetDescription')}
|
||||
</Paragraph>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
export const SurveyStep: React.FC<Props> = ({ onEnter, styles, isDarkMode, token, isModal = false }) => {
|
||||
const { t } = useTranslation('account-setup');
|
||||
const dispatch = useDispatch();
|
||||
const { surveyData, surveySubStep } = useSelector((state: RootState) => state.accountSetupReducer);
|
||||
|
||||
const handleSurveyDataChange = (field: keyof IAccountSetupSurveyData, value: any) => {
|
||||
dispatch(setSurveyData({ [field]: value }));
|
||||
};
|
||||
|
||||
React.useEffect(() => {
|
||||
const handleKeyPress = (e: KeyboardEvent) => {
|
||||
if (e.key === 'Enter') {
|
||||
const isValid = (surveySubStep === 0 && surveyData.organization_type && surveyData.user_role) || (surveySubStep === 1 && surveyData.main_use_cases && surveyData.main_use_cases.length > 0) || (surveySubStep === 2 && surveyData.how_heard_about);
|
||||
if (isValid && surveySubStep < 2) {
|
||||
dispatch(setSurveySubStep(surveySubStep + 1));
|
||||
} else if (isValid && surveySubStep === 2) {
|
||||
onEnter();
|
||||
}
|
||||
}
|
||||
};
|
||||
window.addEventListener('keypress', handleKeyPress);
|
||||
return () => window.removeEventListener('keypress', handleKeyPress);
|
||||
}, [surveySubStep, surveyData, dispatch, onEnter]);
|
||||
|
||||
const handleUseCaseToggle = (value: UseCase) => {
|
||||
const currentUseCases = surveyData.main_use_cases || [];
|
||||
const isSelected = currentUseCases.includes(value);
|
||||
const newUseCases = isSelected ? currentUseCases.filter(useCase => useCase !== value) : [...currentUseCases, value];
|
||||
handleSurveyDataChange('main_use_cases', newUseCases);
|
||||
};
|
||||
|
||||
const getSubStepTitle = () => {
|
||||
switch (surveySubStep) {
|
||||
case 0: return t('aboutYouStepName');
|
||||
case 1: return t('yourNeedsStepName');
|
||||
case 2: return t('discoveryStepName');
|
||||
default: return '';
|
||||
}
|
||||
};
|
||||
|
||||
const surveyPages = [
|
||||
<AboutYouPage key="about-you" styles={styles} isDarkMode={isDarkMode} token={token} surveyData={surveyData} handleSurveyDataChange={handleSurveyDataChange} isModal={isModal} />,
|
||||
<YourNeedsPage key="your-needs" styles={styles} isDarkMode={isDarkMode} token={token} surveyData={surveyData} handleSurveyDataChange={handleSurveyDataChange} handleUseCaseToggle={handleUseCaseToggle} isModal={isModal} />,
|
||||
<DiscoveryPage key="discovery" styles={styles} isDarkMode={isDarkMode} token={token} surveyData={surveyData} handleSurveyDataChange={handleSurveyDataChange} isModal={isModal} />
|
||||
];
|
||||
|
||||
React.useEffect(() => {
|
||||
dispatch(setSurveySubStep(0));
|
||||
}, []);
|
||||
|
||||
return (
|
||||
<div className="w-full">
|
||||
{/* Progress Indicator */}
|
||||
<div className="mb-8">
|
||||
<div className="flex items-center justify-between mb-2">
|
||||
<span className="text-sm font-medium" style={{ color: token?.colorTextSecondary }}>Step {surveySubStep + 1} of 3: {getSubStepTitle()}</span>
|
||||
<span className="text-sm" style={{ color: token?.colorTextSecondary }}>{Math.round(((surveySubStep + 1) / 3) * 100)}%</span>
|
||||
</div>
|
||||
<Progress percent={Math.round(((surveySubStep + 1) / 3) * 100)} showInfo={false} strokeColor={token?.colorPrimary} className="mb-0" />
|
||||
</div>
|
||||
|
||||
{/* Current Page Content */}
|
||||
<div className="min-h-[400px] flex flex-col survey-page-transition" key={surveySubStep}>
|
||||
{surveyPages[surveySubStep]}
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
@@ -1,134 +1,130 @@
|
||||
import React, { useEffect, useRef } from 'react';
|
||||
import { Form, Input, Button, Typography, List, InputRef } from '@/shared/antd-imports';
|
||||
import { PlusOutlined, DeleteOutlined, CloseCircleOutlined } from '@/shared/antd-imports';
|
||||
import React, { useEffect, useRef, useState } from 'react';
|
||||
import { Input, Button, Typography, Card } from '@/shared/antd-imports';
|
||||
import { PlusOutlined, DeleteOutlined, CloseCircleOutlined, CheckCircleOutlined } from '@/shared/antd-imports';
|
||||
import { useDispatch, useSelector } from 'react-redux';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { RootState } from '@/app/store';
|
||||
import { setTasks } from '@/features/account-setup/account-setup.slice';
|
||||
import { sanitizeInput } from '@/utils/sanitizeInput';
|
||||
|
||||
const { Title } = Typography;
|
||||
const { Title, Paragraph, Text } = Typography;
|
||||
|
||||
interface Props {
|
||||
onEnter: () => void;
|
||||
styles: any;
|
||||
isDarkMode: boolean;
|
||||
token?: any;
|
||||
}
|
||||
|
||||
export const TasksStep: React.FC<Props> = ({ onEnter, styles, isDarkMode }) => {
|
||||
|
||||
export const TasksStep: React.FC<Props> = ({ onEnter, styles, isDarkMode, token }) => {
|
||||
const { t } = useTranslation('account-setup');
|
||||
const dispatch = useDispatch();
|
||||
const { tasks, projectName } = useSelector((state: RootState) => state.accountSetupReducer);
|
||||
const inputRefs = useRef<(InputRef | null)[]>([]);
|
||||
const { tasks, projectName, surveyData } = useSelector((state: RootState) => state.accountSetupReducer);
|
||||
const inputRefs = useRef<(HTMLInputElement | null)[]>([]);
|
||||
const [showSuggestions, setShowSuggestions] = useState(false);
|
||||
const [focusedIndex, setFocusedIndex] = useState<number | null>(null);
|
||||
|
||||
const addTask = () => {
|
||||
if (tasks.length == 5) return;
|
||||
|
||||
if (tasks.length >= 5) return;
|
||||
const newId = tasks.length > 0 ? Math.max(...tasks.map(t => t.id)) + 1 : 0;
|
||||
dispatch(setTasks([...tasks, { id: newId, value: '' }]));
|
||||
setTimeout(() => {
|
||||
inputRefs.current[newId]?.focus();
|
||||
}, 0);
|
||||
setTimeout(() => inputRefs.current[tasks.length]?.focus(), 100);
|
||||
};
|
||||
|
||||
const removeTask = (id: number) => {
|
||||
if (tasks.length > 1) {
|
||||
dispatch(setTasks(tasks.filter(task => task.id !== id)));
|
||||
}
|
||||
if (tasks.length > 1) dispatch(setTasks(tasks.filter(task => task.id !== id)));
|
||||
};
|
||||
|
||||
const updateTask = (id: number, value: string) => {
|
||||
const sanitizedValue = sanitizeInput(value);
|
||||
dispatch(
|
||||
setTasks(tasks.map(task => (task.id === id ? { ...task, value: sanitizedValue } : task)))
|
||||
);
|
||||
dispatch(setTasks(tasks.map(task => (task.id === id ? { ...task, value: sanitizedValue } : task))));
|
||||
};
|
||||
|
||||
const handleKeyPress = (e: React.KeyboardEvent<HTMLInputElement>) => {
|
||||
const input = e.currentTarget as HTMLInputElement;
|
||||
if (!input.value.trim()) return;
|
||||
e.preventDefault();
|
||||
addTask();
|
||||
const handleKeyPress = (e: React.KeyboardEvent<HTMLInputElement>, index: number) => {
|
||||
if (e.key === 'Enter') {
|
||||
const input = e.currentTarget as HTMLInputElement;
|
||||
if (input.value.trim()) {
|
||||
e.preventDefault();
|
||||
if (index === tasks.length - 1 && tasks.length < 5) addTask();
|
||||
else if (index < tasks.length - 1) inputRefs.current[index + 1]?.focus();
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
const handleSuggestionClick = (suggestion: string) => {
|
||||
const emptyTaskIndex = tasks.findIndex(task => !task.value.trim());
|
||||
if (emptyTaskIndex !== -1) {
|
||||
updateTask(tasks[emptyTaskIndex].id, suggestion);
|
||||
} else if (tasks.length < 5) {
|
||||
const newId = tasks.length > 0 ? Math.max(...tasks.map(t => t.id)) + 1 : 0;
|
||||
dispatch(setTasks([...tasks, { id: newId, value: suggestion }]));
|
||||
}
|
||||
setShowSuggestions(false);
|
||||
};
|
||||
|
||||
useEffect(() => {
|
||||
setTimeout(() => inputRefs.current[0]?.focus(), 200);
|
||||
}, []);
|
||||
|
||||
// Function to set ref that doesn't return anything (void)
|
||||
const setInputRef = (index: number) => (el: InputRef | null) => {
|
||||
inputRefs.current[index] = el;
|
||||
};
|
||||
|
||||
return (
|
||||
<Form
|
||||
className="create-first-task-form"
|
||||
style={{
|
||||
minHeight: '300px',
|
||||
width: '600px',
|
||||
paddingBottom: '1rem',
|
||||
marginBottom: '3rem',
|
||||
marginTop: '3rem',
|
||||
display: 'flex',
|
||||
flexDirection: 'column',
|
||||
}}
|
||||
>
|
||||
<Form.Item>
|
||||
<Title level={2} style={{ marginBottom: '1rem' }}>
|
||||
<div className="w-full tasks-step">
|
||||
{/* Header */}
|
||||
<div className="text-center mb-8">
|
||||
<Title level={3} className="mb-2" style={{ color: token?.colorText }}>
|
||||
{t('tasksStepTitle')}
|
||||
</Title>
|
||||
</Form.Item>
|
||||
<Form.Item
|
||||
layout="vertical"
|
||||
rules={[{ required: true }]}
|
||||
label={
|
||||
<span className="font-medium">
|
||||
{t('tasksStepLabel')} "<mark>{projectName}</mark>". {t('maxTasks')}
|
||||
</span>
|
||||
}
|
||||
>
|
||||
<List
|
||||
dataSource={tasks}
|
||||
bordered={false}
|
||||
renderItem={(task, index) => (
|
||||
<List.Item key={task.id}>
|
||||
<div style={{ display: 'flex', width: '600px' }}>
|
||||
<Input
|
||||
placeholder="Your Task"
|
||||
value={task.value}
|
||||
onChange={e => updateTask(task.id, e.target.value)}
|
||||
onPressEnter={handleKeyPress}
|
||||
ref={setInputRef(index)}
|
||||
/>
|
||||
<Button
|
||||
className="custom-close-button"
|
||||
style={{ marginLeft: '48px' }}
|
||||
type="text"
|
||||
icon={<CloseCircleOutlined />}
|
||||
disabled={tasks.length === 1}
|
||||
onClick={() => removeTask(task.id)}
|
||||
/>
|
||||
<Paragraph className="text-base" style={{ color: token?.colorTextSecondary }}>
|
||||
{t('tasksStepDescription', { projectName })}
|
||||
</Paragraph>
|
||||
</div>
|
||||
|
||||
|
||||
{/* Tasks List */}
|
||||
<div className="mb-6">
|
||||
<div className="space-y-4">
|
||||
{tasks.map((task, index) => (
|
||||
<Card
|
||||
key={task.id}
|
||||
className={`task-item-card transition-all duration-200 ${
|
||||
focusedIndex === index ? 'border-2' : ''
|
||||
}`}
|
||||
style={{
|
||||
borderColor: focusedIndex === index ? token?.colorPrimary : token?.colorBorder,
|
||||
backgroundColor: token?.colorBgContainer
|
||||
}}
|
||||
>
|
||||
<div className="flex items-center space-x-3">
|
||||
<div className="flex items-center justify-center w-8 h-8 rounded-full text-sm font-medium" style={{ backgroundColor: task.value.trim() ? token?.colorSuccess : token?.colorBorderSecondary, color: task.value.trim() ? '#fff' : token?.colorTextSecondary }}>
|
||||
{task.value.trim() ? <CheckCircleOutlined /> : index + 1}
|
||||
</div>
|
||||
|
||||
<div className="flex-1">
|
||||
<Input
|
||||
placeholder={t('taskPlaceholder', { index: index + 1 })}
|
||||
value={task.value}
|
||||
onChange={e => updateTask(task.id, e.target.value)}
|
||||
onKeyPress={e => handleKeyPress(e, index)}
|
||||
onFocus={() => setFocusedIndex(index)}
|
||||
onBlur={() => setFocusedIndex(null)}
|
||||
ref={(el) => { inputRefs.current[index] = el as any; }}
|
||||
className="text-base border-0 shadow-none task-input"
|
||||
style={{ backgroundColor: 'transparent', color: token?.colorText }}
|
||||
/>
|
||||
</div>
|
||||
|
||||
{tasks.length > 1 && <Button type="text" icon={<CloseCircleOutlined />} onClick={() => removeTask(task.id)} className="text-gray-400 hover:text-red-500" style={{ color: token?.colorTextTertiary }} />}
|
||||
</div>
|
||||
</List.Item>
|
||||
)}
|
||||
/>
|
||||
<Button
|
||||
type="dashed"
|
||||
icon={<PlusOutlined />}
|
||||
onClick={addTask}
|
||||
disabled={tasks.length == 5}
|
||||
style={{ marginTop: '16px' }}
|
||||
>
|
||||
{t('tasksStepAddAnother')}
|
||||
</Button>
|
||||
<div
|
||||
style={{
|
||||
marginTop: '24px',
|
||||
display: 'flex',
|
||||
justifyContent: 'space-between',
|
||||
}}
|
||||
></div>
|
||||
</Form.Item>
|
||||
</Form>
|
||||
</Card>
|
||||
))}
|
||||
</div>
|
||||
|
||||
{tasks.length < 5 && (
|
||||
<Button type="dashed" icon={<PlusOutlined />} onClick={addTask} className="w-full mt-4 h-12 text-base" style={{ borderColor: token?.colorBorder, color: token?.colorTextSecondary }}>{t('addAnotherTask', { current: tasks.length, max: 5 })}</Button>
|
||||
)}
|
||||
</div>
|
||||
|
||||
</div>
|
||||
);
|
||||
};
|
||||
};
|
||||
@@ -0,0 +1,612 @@
|
||||
import React, { useReducer, useMemo, useCallback, useRef, useEffect, useState } from 'react';
|
||||
import {
|
||||
GanttTask,
|
||||
ColumnConfig,
|
||||
TimelineConfig,
|
||||
VirtualScrollConfig,
|
||||
ZoomLevel,
|
||||
GanttState,
|
||||
GanttAction,
|
||||
AdvancedGanttProps,
|
||||
SelectionState,
|
||||
GanttViewState,
|
||||
DragState
|
||||
} from '../../types/advanced-gantt.types';
|
||||
import GanttGrid from './GanttGrid';
|
||||
import DraggableTaskBar from './DraggableTaskBar';
|
||||
import TimelineMarkers, { holidayPresets, workingDayPresets } from './TimelineMarkers';
|
||||
import VirtualScrollContainer, { VirtualTimeline } from './VirtualScrollContainer';
|
||||
import {
|
||||
usePerformanceMonitoring,
|
||||
useTaskCalculations,
|
||||
useDateCalculations,
|
||||
useDebounce,
|
||||
useThrottle
|
||||
} from '../../utils/gantt-performance';
|
||||
import { useAppSelector } from '../../hooks/useAppSelector';
|
||||
import { themeWiseColor } from '../../utils/themeWiseColor';
|
||||
|
||||
// Default configurations
|
||||
const defaultColumns: ColumnConfig[] = [
|
||||
{
|
||||
field: 'name',
|
||||
title: 'Task Name',
|
||||
width: 250,
|
||||
minWidth: 150,
|
||||
resizable: true,
|
||||
sortable: true,
|
||||
fixed: true,
|
||||
editor: 'text'
|
||||
},
|
||||
{
|
||||
field: 'startDate',
|
||||
title: 'Start Date',
|
||||
width: 120,
|
||||
minWidth: 100,
|
||||
resizable: true,
|
||||
sortable: true,
|
||||
fixed: true,
|
||||
editor: 'date'
|
||||
},
|
||||
{
|
||||
field: 'endDate',
|
||||
title: 'End Date',
|
||||
width: 120,
|
||||
minWidth: 100,
|
||||
resizable: true,
|
||||
sortable: true,
|
||||
fixed: true,
|
||||
editor: 'date'
|
||||
},
|
||||
{
|
||||
field: 'duration',
|
||||
title: 'Duration',
|
||||
width: 80,
|
||||
minWidth: 60,
|
||||
resizable: true,
|
||||
sortable: false,
|
||||
fixed: true
|
||||
},
|
||||
{
|
||||
field: 'progress',
|
||||
title: 'Progress',
|
||||
width: 100,
|
||||
minWidth: 80,
|
||||
resizable: true,
|
||||
sortable: true,
|
||||
fixed: true,
|
||||
editor: 'number'
|
||||
},
|
||||
];
|
||||
|
||||
const defaultTimelineConfig: TimelineConfig = {
|
||||
topTier: { unit: 'month', format: 'MMM yyyy', height: 30 },
|
||||
bottomTier: { unit: 'day', format: 'dd', height: 25 },
|
||||
showWeekends: true,
|
||||
showNonWorkingDays: true,
|
||||
holidays: holidayPresets.US,
|
||||
workingDays: workingDayPresets.standard,
|
||||
workingHours: { start: 9, end: 17 },
|
||||
dayWidth: 30,
|
||||
};
|
||||
|
||||
const defaultVirtualScrollConfig: VirtualScrollConfig = {
|
||||
enableRowVirtualization: true,
|
||||
enableTimelineVirtualization: true,
|
||||
bufferSize: 10,
|
||||
itemHeight: 40,
|
||||
overscan: 5,
|
||||
};
|
||||
|
||||
const defaultZoomLevels: ZoomLevel[] = [
|
||||
{
|
||||
name: 'Year',
|
||||
dayWidth: 2,
|
||||
scale: 0.1,
|
||||
topTier: { unit: 'year', format: 'yyyy' },
|
||||
bottomTier: { unit: 'month', format: 'MMM' }
|
||||
},
|
||||
{
|
||||
name: 'Month',
|
||||
dayWidth: 8,
|
||||
scale: 0.5,
|
||||
topTier: { unit: 'month', format: 'MMM yyyy' },
|
||||
bottomTier: { unit: 'week', format: 'w' }
|
||||
},
|
||||
{
|
||||
name: 'Week',
|
||||
dayWidth: 25,
|
||||
scale: 1,
|
||||
topTier: { unit: 'week', format: 'MMM dd' },
|
||||
bottomTier: { unit: 'day', format: 'dd' }
|
||||
},
|
||||
{
|
||||
name: 'Day',
|
||||
dayWidth: 50,
|
||||
scale: 2,
|
||||
topTier: { unit: 'day', format: 'MMM dd' },
|
||||
bottomTier: { unit: 'hour', format: 'HH' }
|
||||
},
|
||||
];
|
||||
|
||||
// Gantt state reducer
|
||||
function ganttReducer(state: GanttState, action: GanttAction): GanttState {
|
||||
switch (action.type) {
|
||||
case 'SET_TASKS':
|
||||
return { ...state, tasks: action.payload };
|
||||
|
||||
case 'UPDATE_TASK':
|
||||
return {
|
||||
...state,
|
||||
tasks: state.tasks.map(task =>
|
||||
task.id === action.payload.id
|
||||
? { ...task, ...action.payload.updates }
|
||||
: task
|
||||
),
|
||||
};
|
||||
|
||||
case 'ADD_TASK':
|
||||
return { ...state, tasks: [...state.tasks, action.payload] };
|
||||
|
||||
case 'DELETE_TASK':
|
||||
return {
|
||||
...state,
|
||||
tasks: state.tasks.filter(task => task.id !== action.payload),
|
||||
};
|
||||
|
||||
case 'SET_SELECTION':
|
||||
return {
|
||||
...state,
|
||||
selectionState: { ...state.selectionState, selectedTasks: action.payload },
|
||||
};
|
||||
|
||||
case 'SET_DRAG_STATE':
|
||||
return { ...state, dragState: action.payload };
|
||||
|
||||
case 'SET_ZOOM_LEVEL':
|
||||
const newZoomLevel = Math.max(0, Math.min(state.zoomLevels.length - 1, action.payload));
|
||||
return {
|
||||
...state,
|
||||
viewState: { ...state.viewState, zoomLevel: newZoomLevel },
|
||||
timelineConfig: {
|
||||
...state.timelineConfig,
|
||||
dayWidth: state.zoomLevels[newZoomLevel].dayWidth,
|
||||
topTier: state.zoomLevels[newZoomLevel].topTier,
|
||||
bottomTier: state.zoomLevels[newZoomLevel].bottomTier,
|
||||
},
|
||||
};
|
||||
|
||||
case 'SET_SCROLL_POSITION':
|
||||
return {
|
||||
...state,
|
||||
viewState: { ...state.viewState, scrollPosition: action.payload },
|
||||
};
|
||||
|
||||
case 'SET_SPLITTER_POSITION':
|
||||
return {
|
||||
...state,
|
||||
viewState: { ...state.viewState, splitterPosition: action.payload },
|
||||
};
|
||||
|
||||
case 'TOGGLE_TASK_EXPANSION':
|
||||
return {
|
||||
...state,
|
||||
tasks: state.tasks.map(task =>
|
||||
task.id === action.payload
|
||||
? { ...task, isExpanded: !task.isExpanded }
|
||||
: task
|
||||
),
|
||||
};
|
||||
|
||||
case 'SET_VIEW_STATE':
|
||||
return {
|
||||
...state,
|
||||
viewState: { ...state.viewState, ...action.payload },
|
||||
};
|
||||
|
||||
case 'UPDATE_COLUMN_WIDTH':
|
||||
return {
|
||||
...state,
|
||||
columns: state.columns.map(col =>
|
||||
col.field === action.payload.field
|
||||
? { ...col, width: action.payload.width }
|
||||
: col
|
||||
),
|
||||
};
|
||||
|
||||
default:
|
||||
return state;
|
||||
}
|
||||
}
|
||||
|
||||
const AdvancedGanttChart: React.FC<AdvancedGanttProps> = ({
|
||||
tasks: initialTasks,
|
||||
columns = defaultColumns,
|
||||
timelineConfig = {},
|
||||
virtualScrollConfig = {},
|
||||
zoomLevels = defaultZoomLevels,
|
||||
initialViewState = {},
|
||||
initialSelection = [],
|
||||
onTaskUpdate,
|
||||
onTaskCreate,
|
||||
onTaskDelete,
|
||||
onTaskMove,
|
||||
onTaskResize,
|
||||
onProgressChange,
|
||||
onSelectionChange,
|
||||
onColumnResize,
|
||||
onDependencyCreate,
|
||||
onDependencyDelete,
|
||||
className = '',
|
||||
style = {},
|
||||
theme = 'auto',
|
||||
enableDragDrop = true,
|
||||
enableResize = true,
|
||||
enableProgressEdit = true,
|
||||
enableInlineEdit = true,
|
||||
enableVirtualScrolling = true,
|
||||
enableDebouncing = true,
|
||||
debounceDelay = 300,
|
||||
maxVisibleTasks = 1000,
|
||||
}) => {
|
||||
const containerRef = useRef<HTMLDivElement>(null);
|
||||
const [containerSize, setContainerSize] = useState({ width: 0, height: 0 });
|
||||
const themeMode = useAppSelector(state => state.themeReducer.mode);
|
||||
const { startMeasure, endMeasure, metrics } = usePerformanceMonitoring();
|
||||
const { getDaysBetween } = useDateCalculations();
|
||||
|
||||
// Initialize state
|
||||
const initialState: GanttState = {
|
||||
tasks: initialTasks,
|
||||
columns,
|
||||
timelineConfig: { ...defaultTimelineConfig, ...timelineConfig },
|
||||
virtualScrollConfig: { ...defaultVirtualScrollConfig, ...virtualScrollConfig },
|
||||
dragState: null,
|
||||
selectionState: {
|
||||
selectedTasks: initialSelection,
|
||||
selectedRows: [],
|
||||
focusedTask: undefined,
|
||||
},
|
||||
viewState: {
|
||||
zoomLevel: 2, // Week view by default
|
||||
scrollPosition: { x: 0, y: 0 },
|
||||
viewportSize: { width: 0, height: 0 },
|
||||
splitterPosition: 40, // 40% for grid, 60% for timeline
|
||||
showCriticalPath: false,
|
||||
showBaseline: false,
|
||||
showProgress: true,
|
||||
showDependencies: true,
|
||||
autoSchedule: false,
|
||||
readOnly: false,
|
||||
...initialViewState,
|
||||
},
|
||||
zoomLevels,
|
||||
performanceMetrics: {
|
||||
renderTime: 0,
|
||||
taskCount: initialTasks.length,
|
||||
visibleTaskCount: 0,
|
||||
},
|
||||
};
|
||||
|
||||
const [state, dispatch] = useReducer(ganttReducer, initialState);
|
||||
const { taskMap, parentChildMap, totalTasks } = useTaskCalculations(state.tasks);
|
||||
|
||||
// Calculate project timeline bounds
|
||||
const projectBounds = useMemo(() => {
|
||||
if (state.tasks.length === 0) {
|
||||
const today = new Date();
|
||||
return {
|
||||
start: new Date(today.getFullYear(), today.getMonth(), 1),
|
||||
end: new Date(today.getFullYear(), today.getMonth() + 3, 0),
|
||||
};
|
||||
}
|
||||
|
||||
const startDates = state.tasks.map(task => task.startDate);
|
||||
const endDates = state.tasks.map(task => task.endDate);
|
||||
const minStart = new Date(Math.min(...startDates.map(d => d.getTime())));
|
||||
const maxEnd = new Date(Math.max(...endDates.map(d => d.getTime())));
|
||||
|
||||
// Add some padding
|
||||
minStart.setDate(minStart.getDate() - 7);
|
||||
maxEnd.setDate(maxEnd.getDate() + 7);
|
||||
|
||||
return { start: minStart, end: maxEnd };
|
||||
}, [state.tasks]);
|
||||
|
||||
// Debounced event handlers
|
||||
const debouncedTaskUpdate = useDebounce(
|
||||
useCallback((taskId: string, updates: Partial<GanttTask>) => {
|
||||
dispatch({ type: 'UPDATE_TASK', payload: { id: taskId, updates } });
|
||||
onTaskUpdate?.(taskId, updates);
|
||||
}, [onTaskUpdate]),
|
||||
enableDebouncing ? debounceDelay : 0
|
||||
);
|
||||
|
||||
const debouncedTaskMove = useDebounce(
|
||||
useCallback((taskId: string, newDates: { start: Date; end: Date }) => {
|
||||
dispatch({ type: 'UPDATE_TASK', payload: {
|
||||
id: taskId,
|
||||
updates: { startDate: newDates.start, endDate: newDates.end }
|
||||
}});
|
||||
onTaskMove?.(taskId, newDates);
|
||||
}, [onTaskMove]),
|
||||
enableDebouncing ? debounceDelay : 0
|
||||
);
|
||||
|
||||
const debouncedProgressChange = useDebounce(
|
||||
useCallback((taskId: string, progress: number) => {
|
||||
dispatch({ type: 'UPDATE_TASK', payload: { id: taskId, updates: { progress } }});
|
||||
onProgressChange?.(taskId, progress);
|
||||
}, [onProgressChange]),
|
||||
enableDebouncing ? debounceDelay : 0
|
||||
);
|
||||
|
||||
// Throttled scroll handler
|
||||
const throttledScrollHandler = useThrottle(
|
||||
useCallback((scrollLeft: number, scrollTop: number) => {
|
||||
dispatch({ type: 'SET_SCROLL_POSITION', payload: { x: scrollLeft, y: scrollTop } });
|
||||
}, []),
|
||||
16 // 60fps
|
||||
);
|
||||
|
||||
// Container size observer
|
||||
useEffect(() => {
|
||||
const observer = new ResizeObserver((entries) => {
|
||||
for (const entry of entries) {
|
||||
const { width, height } = entry.contentRect;
|
||||
setContainerSize({ width, height });
|
||||
dispatch({
|
||||
type: 'SET_VIEW_STATE',
|
||||
payload: { viewportSize: { width, height } }
|
||||
});
|
||||
}
|
||||
});
|
||||
|
||||
if (containerRef.current) {
|
||||
observer.observe(containerRef.current);
|
||||
}
|
||||
|
||||
return () => observer.disconnect();
|
||||
}, []);
|
||||
|
||||
// Calculate grid and timeline dimensions
|
||||
const gridWidth = useMemo(() => {
|
||||
return Math.floor(containerSize.width * (state.viewState.splitterPosition / 100));
|
||||
}, [containerSize.width, state.viewState.splitterPosition]);
|
||||
|
||||
const timelineWidth = useMemo(() => {
|
||||
return containerSize.width - gridWidth;
|
||||
}, [containerSize.width, gridWidth]);
|
||||
|
||||
// Handle zoom changes
|
||||
const handleZoomChange = useCallback((direction: 'in' | 'out') => {
|
||||
const currentZoom = state.viewState.zoomLevel;
|
||||
const newZoom = direction === 'in'
|
||||
? Math.min(state.zoomLevels.length - 1, currentZoom + 1)
|
||||
: Math.max(0, currentZoom - 1);
|
||||
|
||||
dispatch({ type: 'SET_ZOOM_LEVEL', payload: newZoom });
|
||||
}, [state.viewState.zoomLevel, state.zoomLevels.length]);
|
||||
|
||||
// Theme-aware colors
|
||||
const colors = useMemo(() => ({
|
||||
background: themeWiseColor('#ffffff', '#1f2937', themeMode),
|
||||
border: themeWiseColor('#e5e7eb', '#4b5563', themeMode),
|
||||
timelineBackground: themeWiseColor('#f8f9fa', '#374151', themeMode),
|
||||
}), [themeMode]);
|
||||
|
||||
// Render timeline header
|
||||
const renderTimelineHeader = () => {
|
||||
const currentZoom = state.zoomLevels[state.viewState.zoomLevel];
|
||||
const totalDays = getDaysBetween(projectBounds.start, projectBounds.end);
|
||||
const totalWidth = totalDays * state.timelineConfig.dayWidth;
|
||||
|
||||
return (
|
||||
<div className="timeline-header border-b" style={{
|
||||
height: (currentZoom.topTier.height || 30) + (currentZoom.bottomTier.height || 25),
|
||||
backgroundColor: colors.timelineBackground,
|
||||
borderColor: colors.border,
|
||||
}}>
|
||||
<VirtualTimeline
|
||||
startDate={projectBounds.start}
|
||||
endDate={projectBounds.end}
|
||||
dayWidth={state.timelineConfig.dayWidth}
|
||||
containerWidth={timelineWidth}
|
||||
containerHeight={(currentZoom.topTier.height || 30) + (currentZoom.bottomTier.height || 25)}
|
||||
onScroll={throttledScrollHandler}
|
||||
>
|
||||
{(date, index, style) => (
|
||||
<div className="timeline-cell flex flex-col border-r text-xs text-center" style={{
|
||||
...style,
|
||||
borderColor: colors.border,
|
||||
}}>
|
||||
<div className="top-tier border-b px-1 py-1" style={{
|
||||
height: currentZoom.topTier.height || 30,
|
||||
borderColor: colors.border,
|
||||
}}>
|
||||
{formatDateForUnit(date, currentZoom.topTier.unit)}
|
||||
</div>
|
||||
<div className="bottom-tier px-1 py-1" style={{
|
||||
height: currentZoom.bottomTier.height || 25,
|
||||
}}>
|
||||
{formatDateForUnit(date, currentZoom.bottomTier.unit)}
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
</VirtualTimeline>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
// Render timeline content
|
||||
const renderTimelineContent = () => {
|
||||
const headerHeight = (state.zoomLevels[state.viewState.zoomLevel].topTier.height || 30) +
|
||||
(state.zoomLevels[state.viewState.zoomLevel].bottomTier.height || 25);
|
||||
const contentHeight = containerSize.height - headerHeight;
|
||||
|
||||
return (
|
||||
<div className="timeline-content relative" style={{ height: contentHeight }}>
|
||||
{/* Timeline markers (weekends, holidays, etc.) */}
|
||||
<TimelineMarkers
|
||||
startDate={projectBounds.start}
|
||||
endDate={projectBounds.end}
|
||||
dayWidth={state.timelineConfig.dayWidth}
|
||||
containerHeight={contentHeight}
|
||||
timelineConfig={state.timelineConfig}
|
||||
holidays={state.timelineConfig.holidays}
|
||||
showWeekends={state.timelineConfig.showWeekends}
|
||||
showHolidays={true}
|
||||
showToday={true}
|
||||
/>
|
||||
|
||||
{/* Task bars */}
|
||||
<VirtualScrollContainer
|
||||
items={state.tasks}
|
||||
itemHeight={state.virtualScrollConfig.itemHeight}
|
||||
containerHeight={contentHeight}
|
||||
containerWidth={timelineWidth}
|
||||
overscan={state.virtualScrollConfig.overscan}
|
||||
onScroll={throttledScrollHandler}
|
||||
>
|
||||
{(task, index, style) => (
|
||||
<DraggableTaskBar
|
||||
key={task.id}
|
||||
task={task}
|
||||
timelineStart={projectBounds.start}
|
||||
dayWidth={state.timelineConfig.dayWidth}
|
||||
rowHeight={state.virtualScrollConfig.itemHeight}
|
||||
index={index}
|
||||
onTaskMove={debouncedTaskMove}
|
||||
onTaskResize={debouncedTaskMove}
|
||||
onProgressChange={debouncedProgressChange}
|
||||
enableDragDrop={enableDragDrop}
|
||||
enableResize={enableResize}
|
||||
enableProgressEdit={enableProgressEdit}
|
||||
readOnly={state.viewState.readOnly}
|
||||
/>
|
||||
)}
|
||||
</VirtualScrollContainer>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
// Render toolbar
|
||||
const renderToolbar = () => (
|
||||
<div className="gantt-toolbar flex items-center justify-between p-2 border-b bg-gray-50 dark:bg-gray-800" style={{
|
||||
borderColor: colors.border,
|
||||
}}>
|
||||
<div className="toolbar-left flex items-center space-x-2">
|
||||
<button
|
||||
onClick={() => handleZoomChange('out')}
|
||||
disabled={state.viewState.zoomLevel === 0}
|
||||
className="px-2 py-1 text-sm border rounded hover:bg-gray-100 dark:hover:bg-gray-700 disabled:opacity-50"
|
||||
>
|
||||
Zoom Out
|
||||
</button>
|
||||
<span className="text-sm text-gray-600 dark:text-gray-400">
|
||||
{state.zoomLevels[state.viewState.zoomLevel].name}
|
||||
</span>
|
||||
<button
|
||||
onClick={() => handleZoomChange('in')}
|
||||
disabled={state.viewState.zoomLevel === state.zoomLevels.length - 1}
|
||||
className="px-2 py-1 text-sm border rounded hover:bg-gray-100 dark:hover:bg-gray-700 disabled:opacity-50"
|
||||
>
|
||||
Zoom In
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<div className="toolbar-right flex items-center space-x-2 text-xs text-gray-500">
|
||||
<span>Tasks: {state.tasks.length}</span>
|
||||
<span>•</span>
|
||||
<span>Render: {Math.round(metrics.renderTime)}ms</span>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
|
||||
// Performance monitoring
|
||||
useEffect(() => {
|
||||
startMeasure('render');
|
||||
return () => endMeasure('render');
|
||||
});
|
||||
|
||||
return (
|
||||
<div
|
||||
ref={containerRef}
|
||||
className={`advanced-gantt-chart flex flex-col ${className}`}
|
||||
style={{
|
||||
height: '100%',
|
||||
backgroundColor: colors.background,
|
||||
...style,
|
||||
}}
|
||||
>
|
||||
{/* Toolbar */}
|
||||
{renderToolbar()}
|
||||
|
||||
{/* Main content */}
|
||||
<div className="gantt-content flex flex-1 overflow-hidden">
|
||||
{/* Grid */}
|
||||
<div className="gantt-grid-container" style={{ width: gridWidth }}>
|
||||
<GanttGrid
|
||||
tasks={state.tasks}
|
||||
columns={state.columns}
|
||||
rowHeight={state.virtualScrollConfig.itemHeight}
|
||||
containerHeight={containerSize.height - 50} // Subtract toolbar height
|
||||
selection={state.selectionState}
|
||||
enableInlineEdit={enableInlineEdit}
|
||||
onTaskClick={(task) => {
|
||||
// Handle task selection
|
||||
const newSelection = { ...state.selectionState, selectedTasks: [task.id] };
|
||||
dispatch({ type: 'SET_SELECTION', payload: [task.id] });
|
||||
onSelectionChange?.(newSelection);
|
||||
}}
|
||||
onTaskExpand={(taskId) => {
|
||||
dispatch({ type: 'TOGGLE_TASK_EXPANSION', payload: taskId });
|
||||
}}
|
||||
onColumnResize={(field, width) => {
|
||||
dispatch({ type: 'UPDATE_COLUMN_WIDTH', payload: { field, width } });
|
||||
onColumnResize?.(field, width);
|
||||
}}
|
||||
onTaskUpdate={debouncedTaskUpdate}
|
||||
/>
|
||||
</div>
|
||||
|
||||
{/* Timeline */}
|
||||
<div className="gantt-timeline-container border-l" style={{
|
||||
width: timelineWidth,
|
||||
borderColor: colors.border,
|
||||
}}>
|
||||
{renderTimelineHeader()}
|
||||
{renderTimelineContent()}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
// Helper function to format dates based on unit
|
||||
function formatDateForUnit(date: Date, unit: string): string {
|
||||
switch (unit) {
|
||||
case 'year':
|
||||
return date.getFullYear().toString();
|
||||
case 'month':
|
||||
return date.toLocaleDateString('en-US', { month: 'short', year: 'numeric' });
|
||||
case 'week':
|
||||
return `W${getWeekNumber(date)}`;
|
||||
case 'day':
|
||||
return date.getDate().toString();
|
||||
case 'hour':
|
||||
return date.getHours().toString().padStart(2, '0');
|
||||
default:
|
||||
return '';
|
||||
}
|
||||
}
|
||||
|
||||
function getWeekNumber(date: Date): number {
|
||||
const d = new Date(Date.UTC(date.getFullYear(), date.getMonth(), date.getDate()));
|
||||
const dayNum = d.getUTCDay() || 7;
|
||||
d.setUTCDate(d.getUTCDate() + 4 - dayNum);
|
||||
const yearStart = new Date(Date.UTC(d.getUTCFullYear(), 0, 1));
|
||||
return Math.ceil((((d.getTime() - yearStart.getTime()) / 86400000) + 1) / 7);
|
||||
}
|
||||
|
||||
export default AdvancedGanttChart;
|
||||
@@ -0,0 +1,668 @@
|
||||
import React, { useState, useMemo } from 'react';
|
||||
import { Button, Space, message, Card } from 'antd';
|
||||
import AdvancedGanttChart from './AdvancedGanttChart';
|
||||
import { GanttTask, ColumnConfig } from '../../types/advanced-gantt.types';
|
||||
import { useAppSelector } from '../../hooks/useAppSelector';
|
||||
import { holidayPresets, workingDayPresets } from './TimelineMarkers';
|
||||
|
||||
// Enhanced sample data with more realistic project structure
|
||||
const generateSampleTasks = (): GanttTask[] => {
|
||||
const baseDate = new Date(2024, 11, 1); // December 1, 2024
|
||||
|
||||
return [
|
||||
// Project Phase 1: Planning & Design
|
||||
{
|
||||
id: 'project-1',
|
||||
name: '🚀 Web Platform Development',
|
||||
startDate: new Date(2024, 11, 1),
|
||||
endDate: new Date(2025, 2, 31),
|
||||
progress: 45,
|
||||
type: 'project',
|
||||
status: 'in-progress',
|
||||
priority: 'high',
|
||||
color: '#1890ff',
|
||||
hasChildren: true,
|
||||
isExpanded: true,
|
||||
level: 0,
|
||||
},
|
||||
{
|
||||
id: 'planning-phase',
|
||||
name: '📋 Planning & Analysis Phase',
|
||||
startDate: new Date(2024, 11, 1),
|
||||
endDate: new Date(2024, 11, 20),
|
||||
progress: 85,
|
||||
type: 'project',
|
||||
status: 'in-progress',
|
||||
priority: 'high',
|
||||
parent: 'project-1',
|
||||
color: '#52c41a',
|
||||
hasChildren: true,
|
||||
isExpanded: true,
|
||||
level: 1,
|
||||
},
|
||||
{
|
||||
id: 'requirements-analysis',
|
||||
name: 'Requirements Gathering & Analysis',
|
||||
startDate: new Date(2024, 11, 1),
|
||||
endDate: new Date(2024, 11, 8),
|
||||
progress: 100,
|
||||
type: 'task',
|
||||
status: 'completed',
|
||||
priority: 'high',
|
||||
parent: 'planning-phase',
|
||||
assignee: {
|
||||
id: 'user-1',
|
||||
name: 'Alice Johnson',
|
||||
avatar: 'https://ui-avatars.com/api/?name=Alice+Johnson&background=1890ff&color=fff',
|
||||
},
|
||||
tags: ['research', 'documentation'],
|
||||
level: 2,
|
||||
},
|
||||
{
|
||||
id: 'technical-architecture',
|
||||
name: 'Technical Architecture Design',
|
||||
startDate: new Date(2024, 11, 8),
|
||||
endDate: new Date(2024, 11, 18),
|
||||
progress: 75,
|
||||
type: 'task',
|
||||
status: 'in-progress',
|
||||
priority: 'high',
|
||||
parent: 'planning-phase',
|
||||
assignee: {
|
||||
id: 'user-2',
|
||||
name: 'Bob Smith',
|
||||
avatar: 'https://ui-avatars.com/api/?name=Bob+Smith&background=52c41a&color=fff',
|
||||
},
|
||||
dependencies: ['requirements-analysis'],
|
||||
tags: ['architecture', 'design'],
|
||||
level: 2,
|
||||
},
|
||||
{
|
||||
id: 'ui-ux-design',
|
||||
name: 'UI/UX Design & Prototyping',
|
||||
startDate: new Date(2024, 11, 10),
|
||||
endDate: new Date(2024, 11, 20),
|
||||
progress: 60,
|
||||
type: 'task',
|
||||
status: 'in-progress',
|
||||
priority: 'medium',
|
||||
parent: 'planning-phase',
|
||||
assignee: {
|
||||
id: 'user-3',
|
||||
name: 'Carol Davis',
|
||||
avatar: 'https://ui-avatars.com/api/?name=Carol+Davis&background=faad14&color=fff',
|
||||
},
|
||||
dependencies: ['requirements-analysis'],
|
||||
tags: ['design', 'prototype'],
|
||||
level: 2,
|
||||
},
|
||||
{
|
||||
id: 'milestone-planning-complete',
|
||||
name: '🎯 Planning Phase Complete',
|
||||
startDate: new Date(2024, 11, 20),
|
||||
endDate: new Date(2024, 11, 20),
|
||||
progress: 0,
|
||||
type: 'milestone',
|
||||
status: 'not-started',
|
||||
priority: 'critical',
|
||||
parent: 'planning-phase',
|
||||
dependencies: ['technical-architecture', 'ui-ux-design'],
|
||||
level: 2,
|
||||
},
|
||||
|
||||
// Development Phase
|
||||
{
|
||||
id: 'development-phase',
|
||||
name: '⚡ Development Phase',
|
||||
startDate: new Date(2024, 11, 21),
|
||||
endDate: new Date(2025, 1, 28),
|
||||
progress: 35,
|
||||
type: 'project',
|
||||
status: 'in-progress',
|
||||
priority: 'high',
|
||||
parent: 'project-1',
|
||||
color: '#722ed1',
|
||||
hasChildren: true,
|
||||
isExpanded: true,
|
||||
level: 1,
|
||||
},
|
||||
{
|
||||
id: 'backend-development',
|
||||
name: 'Backend API Development',
|
||||
startDate: new Date(2024, 11, 21),
|
||||
endDate: new Date(2025, 1, 15),
|
||||
progress: 45,
|
||||
type: 'task',
|
||||
status: 'in-progress',
|
||||
priority: 'high',
|
||||
parent: 'development-phase',
|
||||
assignee: {
|
||||
id: 'user-4',
|
||||
name: 'David Wilson',
|
||||
avatar: 'https://ui-avatars.com/api/?name=David+Wilson&background=722ed1&color=fff',
|
||||
},
|
||||
dependencies: ['milestone-planning-complete'],
|
||||
tags: ['backend', 'api'],
|
||||
level: 2,
|
||||
},
|
||||
{
|
||||
id: 'frontend-development',
|
||||
name: 'Frontend React Application',
|
||||
startDate: new Date(2025, 0, 5),
|
||||
endDate: new Date(2025, 1, 25),
|
||||
progress: 25,
|
||||
type: 'task',
|
||||
status: 'in-progress',
|
||||
priority: 'high',
|
||||
parent: 'development-phase',
|
||||
assignee: {
|
||||
id: 'user-5',
|
||||
name: 'Eva Brown',
|
||||
avatar: 'https://ui-avatars.com/api/?name=Eva+Brown&background=ff7a45&color=fff',
|
||||
},
|
||||
dependencies: ['backend-development'],
|
||||
tags: ['frontend', 'react'],
|
||||
level: 2,
|
||||
},
|
||||
{
|
||||
id: 'database-setup',
|
||||
name: 'Database Schema & Migration',
|
||||
startDate: new Date(2024, 11, 21),
|
||||
endDate: new Date(2025, 0, 10),
|
||||
progress: 80,
|
||||
type: 'task',
|
||||
status: 'in-progress',
|
||||
priority: 'medium',
|
||||
parent: 'development-phase',
|
||||
assignee: {
|
||||
id: 'user-6',
|
||||
name: 'Frank Miller',
|
||||
avatar: 'https://ui-avatars.com/api/?name=Frank+Miller&background=13c2c2&color=fff',
|
||||
},
|
||||
dependencies: ['milestone-planning-complete'],
|
||||
tags: ['database', 'migration'],
|
||||
level: 2,
|
||||
},
|
||||
|
||||
// Testing Phase
|
||||
{
|
||||
id: 'testing-phase',
|
||||
name: '🧪 Testing & QA Phase',
|
||||
startDate: new Date(2025, 2, 1),
|
||||
endDate: new Date(2025, 2, 20),
|
||||
progress: 0,
|
||||
type: 'project',
|
||||
status: 'not-started',
|
||||
priority: 'high',
|
||||
parent: 'project-1',
|
||||
color: '#fa8c16',
|
||||
hasChildren: true,
|
||||
isExpanded: false,
|
||||
level: 1,
|
||||
},
|
||||
{
|
||||
id: 'unit-testing',
|
||||
name: 'Unit Testing Implementation',
|
||||
startDate: new Date(2025, 2, 1),
|
||||
endDate: new Date(2025, 2, 10),
|
||||
progress: 0,
|
||||
type: 'task',
|
||||
status: 'not-started',
|
||||
priority: 'high',
|
||||
parent: 'testing-phase',
|
||||
assignee: {
|
||||
id: 'user-7',
|
||||
name: 'Grace Lee',
|
||||
avatar: 'https://ui-avatars.com/api/?name=Grace+Lee&background=fa8c16&color=fff',
|
||||
},
|
||||
dependencies: ['frontend-development'],
|
||||
tags: ['testing', 'unit'],
|
||||
level: 2,
|
||||
},
|
||||
{
|
||||
id: 'integration-testing',
|
||||
name: 'Integration & E2E Testing',
|
||||
startDate: new Date(2025, 2, 8),
|
||||
endDate: new Date(2025, 2, 18),
|
||||
progress: 0,
|
||||
type: 'task',
|
||||
status: 'not-started',
|
||||
priority: 'high',
|
||||
parent: 'testing-phase',
|
||||
assignee: {
|
||||
id: 'user-8',
|
||||
name: 'Henry Clark',
|
||||
avatar: 'https://ui-avatars.com/api/?name=Henry+Clark&background=eb2f96&color=fff',
|
||||
},
|
||||
dependencies: ['unit-testing'],
|
||||
tags: ['testing', 'integration'],
|
||||
level: 2,
|
||||
},
|
||||
{
|
||||
id: 'milestone-beta-ready',
|
||||
name: '🎯 Beta Release Ready',
|
||||
startDate: new Date(2025, 2, 20),
|
||||
endDate: new Date(2025, 2, 20),
|
||||
progress: 0,
|
||||
type: 'milestone',
|
||||
status: 'not-started',
|
||||
priority: 'critical',
|
||||
parent: 'testing-phase',
|
||||
dependencies: ['integration-testing'],
|
||||
level: 2,
|
||||
},
|
||||
|
||||
// Deployment Phase
|
||||
{
|
||||
id: 'deployment-phase',
|
||||
name: '🚀 Deployment & Launch',
|
||||
startDate: new Date(2025, 2, 21),
|
||||
endDate: new Date(2025, 2, 31),
|
||||
progress: 0,
|
||||
type: 'project',
|
||||
status: 'not-started',
|
||||
priority: 'critical',
|
||||
parent: 'project-1',
|
||||
color: '#f5222d',
|
||||
hasChildren: true,
|
||||
isExpanded: false,
|
||||
level: 1,
|
||||
},
|
||||
{
|
||||
id: 'production-deployment',
|
||||
name: 'Production Environment Setup',
|
||||
startDate: new Date(2025, 2, 21),
|
||||
endDate: new Date(2025, 2, 25),
|
||||
progress: 0,
|
||||
type: 'task',
|
||||
status: 'not-started',
|
||||
priority: 'critical',
|
||||
parent: 'deployment-phase',
|
||||
assignee: {
|
||||
id: 'user-9',
|
||||
name: 'Ivy Taylor',
|
||||
avatar: 'https://ui-avatars.com/api/?name=Ivy+Taylor&background=f5222d&color=fff',
|
||||
},
|
||||
dependencies: ['milestone-beta-ready'],
|
||||
tags: ['deployment', 'production'],
|
||||
level: 2,
|
||||
},
|
||||
{
|
||||
id: 'go-live',
|
||||
name: 'Go Live & Monitoring',
|
||||
startDate: new Date(2025, 2, 26),
|
||||
endDate: new Date(2025, 2, 31),
|
||||
progress: 0,
|
||||
type: 'task',
|
||||
status: 'not-started',
|
||||
priority: 'critical',
|
||||
parent: 'deployment-phase',
|
||||
assignee: {
|
||||
id: 'user-10',
|
||||
name: 'Jack Anderson',
|
||||
avatar: 'https://ui-avatars.com/api/?name=Jack+Anderson&background=2f54eb&color=fff',
|
||||
},
|
||||
dependencies: ['production-deployment'],
|
||||
tags: ['launch', 'monitoring'],
|
||||
level: 2,
|
||||
},
|
||||
{
|
||||
id: 'milestone-project-complete',
|
||||
name: '🎉 Project Launch Complete',
|
||||
startDate: new Date(2025, 2, 31),
|
||||
endDate: new Date(2025, 2, 31),
|
||||
progress: 0,
|
||||
type: 'milestone',
|
||||
status: 'not-started',
|
||||
priority: 'critical',
|
||||
parent: 'deployment-phase',
|
||||
dependencies: ['go-live'],
|
||||
level: 2,
|
||||
},
|
||||
];
|
||||
};
|
||||
|
||||
// Enhanced column configuration
|
||||
const sampleColumns: ColumnConfig[] = [
|
||||
{
|
||||
field: 'name',
|
||||
title: 'Task / Phase Name',
|
||||
width: 300,
|
||||
minWidth: 200,
|
||||
resizable: true,
|
||||
sortable: true,
|
||||
fixed: true,
|
||||
editor: 'text'
|
||||
},
|
||||
{
|
||||
field: 'assignee',
|
||||
title: 'Assignee',
|
||||
width: 150,
|
||||
minWidth: 120,
|
||||
resizable: true,
|
||||
sortable: true,
|
||||
fixed: true
|
||||
},
|
||||
{
|
||||
field: 'startDate',
|
||||
title: 'Start Date',
|
||||
width: 120,
|
||||
minWidth: 100,
|
||||
resizable: true,
|
||||
sortable: true,
|
||||
fixed: true,
|
||||
editor: 'date'
|
||||
},
|
||||
{
|
||||
field: 'endDate',
|
||||
title: 'End Date',
|
||||
width: 120,
|
||||
minWidth: 100,
|
||||
resizable: true,
|
||||
sortable: true,
|
||||
fixed: true,
|
||||
editor: 'date'
|
||||
},
|
||||
{
|
||||
field: 'duration',
|
||||
title: 'Duration',
|
||||
width: 80,
|
||||
minWidth: 60,
|
||||
resizable: true,
|
||||
sortable: false,
|
||||
fixed: true,
|
||||
align: 'center'
|
||||
},
|
||||
{
|
||||
field: 'progress',
|
||||
title: 'Progress',
|
||||
width: 120,
|
||||
minWidth: 100,
|
||||
resizable: true,
|
||||
sortable: true,
|
||||
fixed: true,
|
||||
editor: 'number'
|
||||
},
|
||||
{
|
||||
field: 'status',
|
||||
title: 'Status',
|
||||
width: 100,
|
||||
minWidth: 80,
|
||||
resizable: true,
|
||||
sortable: true,
|
||||
fixed: true,
|
||||
editor: 'select',
|
||||
editorOptions: [
|
||||
{ value: 'not-started', label: 'Not Started' },
|
||||
{ value: 'in-progress', label: 'In Progress' },
|
||||
{ value: 'completed', label: 'Completed' },
|
||||
{ value: 'on-hold', label: 'On Hold' },
|
||||
{ value: 'overdue', label: 'Overdue' },
|
||||
]
|
||||
},
|
||||
{
|
||||
field: 'priority',
|
||||
title: 'Priority',
|
||||
width: 100,
|
||||
minWidth: 80,
|
||||
resizable: true,
|
||||
sortable: true,
|
||||
fixed: true,
|
||||
editor: 'select',
|
||||
editorOptions: [
|
||||
{ value: 'low', label: 'Low' },
|
||||
{ value: 'medium', label: 'Medium' },
|
||||
{ value: 'high', label: 'High' },
|
||||
{ value: 'critical', label: 'Critical' },
|
||||
]
|
||||
},
|
||||
];
|
||||
|
||||
const AdvancedGanttDemo: React.FC = () => {
|
||||
const [tasks, setTasks] = useState<GanttTask[]>(generateSampleTasks());
|
||||
const [selectedTasks, setSelectedTasks] = useState<string[]>([]);
|
||||
const themeMode = useAppSelector(state => state.themeReducer.mode);
|
||||
|
||||
const handleTaskUpdate = (taskId: string, updates: Partial<GanttTask>) => {
|
||||
setTasks(prevTasks =>
|
||||
prevTasks.map(task =>
|
||||
task.id === taskId ? { ...task, ...updates } : task
|
||||
)
|
||||
);
|
||||
message.success(`Task "${tasks.find(t => t.id === taskId)?.name}" updated`);
|
||||
};
|
||||
|
||||
const handleTaskMove = (taskId: string, newDates: { start: Date; end: Date }) => {
|
||||
setTasks(prevTasks =>
|
||||
prevTasks.map(task =>
|
||||
task.id === taskId
|
||||
? { ...task, startDate: newDates.start, endDate: newDates.end }
|
||||
: task
|
||||
)
|
||||
);
|
||||
message.info(`Task moved: ${newDates.start.toLocaleDateString()} - ${newDates.end.toLocaleDateString()}`);
|
||||
};
|
||||
|
||||
const handleProgressChange = (taskId: string, progress: number) => {
|
||||
setTasks(prevTasks =>
|
||||
prevTasks.map(task =>
|
||||
task.id === taskId ? { ...task, progress } : task
|
||||
)
|
||||
);
|
||||
message.info(`Progress updated: ${Math.round(progress)}%`);
|
||||
};
|
||||
|
||||
const handleSelectionChange = (selection: any) => {
|
||||
setSelectedTasks(selection.selectedTasks);
|
||||
};
|
||||
|
||||
const resetToSampleData = () => {
|
||||
setTasks(generateSampleTasks());
|
||||
setSelectedTasks([]);
|
||||
message.info('Gantt chart reset to sample data');
|
||||
};
|
||||
|
||||
const addSampleTask = () => {
|
||||
const newTask: GanttTask = {
|
||||
id: `task-${Date.now()}`,
|
||||
name: `New Task ${tasks.length + 1}`,
|
||||
startDate: new Date(),
|
||||
endDate: new Date(Date.now() + 7 * 24 * 60 * 60 * 1000), // +7 days
|
||||
progress: 0,
|
||||
type: 'task',
|
||||
status: 'not-started',
|
||||
priority: 'medium',
|
||||
level: 0,
|
||||
};
|
||||
setTasks(prev => [...prev, newTask]);
|
||||
message.success('New task added');
|
||||
};
|
||||
|
||||
const deleteSelectedTasks = () => {
|
||||
if (selectedTasks.length === 0) {
|
||||
message.warning('No tasks selected');
|
||||
return;
|
||||
}
|
||||
|
||||
setTasks(prev => prev.filter(task => !selectedTasks.includes(task.id)));
|
||||
setSelectedTasks([]);
|
||||
message.success(`${selectedTasks.length} task(s) deleted`);
|
||||
};
|
||||
|
||||
const taskStats = useMemo(() => {
|
||||
const total = tasks.length;
|
||||
const completed = tasks.filter(t => t.status === 'completed').length;
|
||||
const inProgress = tasks.filter(t => t.status === 'in-progress').length;
|
||||
const overdue = tasks.filter(t => t.status === 'overdue').length;
|
||||
const avgProgress = tasks.reduce((sum, t) => sum + t.progress, 0) / total;
|
||||
|
||||
return { total, completed, inProgress, overdue, avgProgress };
|
||||
}, [tasks]);
|
||||
|
||||
return (
|
||||
<div className="advanced-gantt-demo p-6 bg-gray-50 dark:bg-gray-900 min-h-screen">
|
||||
{/* Header */}
|
||||
<div className="bg-white dark:bg-gray-800 border border-gray-200 dark:border-gray-700 rounded-lg shadow-sm mb-4">
|
||||
<div className="p-6">
|
||||
<div className="flex justify-between items-start mb-4">
|
||||
<div>
|
||||
<h1 className="text-3xl font-bold text-gray-900 dark:text-gray-100 mb-2">
|
||||
🚀 Advanced Gantt Chart Demo
|
||||
</h1>
|
||||
<p className="text-gray-600 dark:text-gray-400 mb-4">
|
||||
Professional Gantt chart with draggable tasks, virtual scrolling, holiday markers,
|
||||
and performance optimizations for modern project management.
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<div className="flex flex-col items-end space-y-2">
|
||||
<Space>
|
||||
<Button
|
||||
onClick={addSampleTask}
|
||||
type="primary"
|
||||
className="dark:border-gray-600"
|
||||
>
|
||||
Add Task
|
||||
</Button>
|
||||
<Button
|
||||
onClick={deleteSelectedTasks}
|
||||
danger
|
||||
disabled={selectedTasks.length === 0}
|
||||
className="dark:border-gray-600"
|
||||
>
|
||||
Delete Selected ({selectedTasks.length})
|
||||
</Button>
|
||||
<Button
|
||||
onClick={resetToSampleData}
|
||||
className="dark:border-gray-600 dark:text-gray-300"
|
||||
>
|
||||
Reset Data
|
||||
</Button>
|
||||
</Space>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Project Statistics */}
|
||||
<div className="grid grid-cols-2 md:grid-cols-4 gap-4">
|
||||
<div className="bg-blue-50 dark:bg-blue-900 dark:bg-opacity-20 rounded-lg p-3">
|
||||
<div className="text-blue-600 dark:text-blue-400 text-sm font-medium">Total Tasks</div>
|
||||
<div className="text-2xl font-bold text-blue-700 dark:text-blue-300">{taskStats.total}</div>
|
||||
</div>
|
||||
<div className="bg-green-50 dark:bg-green-900 dark:bg-opacity-20 rounded-lg p-3">
|
||||
<div className="text-green-600 dark:text-green-400 text-sm font-medium">Completed</div>
|
||||
<div className="text-2xl font-bold text-green-700 dark:text-green-300">{taskStats.completed}</div>
|
||||
</div>
|
||||
<div className="bg-yellow-50 dark:bg-yellow-900 dark:bg-opacity-20 rounded-lg p-3">
|
||||
<div className="text-yellow-600 dark:text-yellow-400 text-sm font-medium">In Progress</div>
|
||||
<div className="text-2xl font-bold text-yellow-700 dark:text-yellow-300">{taskStats.inProgress}</div>
|
||||
</div>
|
||||
<div className="bg-purple-50 dark:bg-purple-900 dark:bg-opacity-20 rounded-lg p-3">
|
||||
<div className="text-purple-600 dark:text-purple-400 text-sm font-medium">Avg Progress</div>
|
||||
<div className="text-2xl font-bold text-purple-700 dark:text-purple-300">
|
||||
{Math.round(taskStats.avgProgress)}%
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Gantt Chart */}
|
||||
<div className="bg-white dark:bg-gray-800 border border-gray-200 dark:border-gray-700 rounded-lg shadow-sm" style={{ height: '70vh' }}>
|
||||
<AdvancedGanttChart
|
||||
tasks={tasks}
|
||||
columns={sampleColumns}
|
||||
timelineConfig={{
|
||||
showWeekends: true,
|
||||
showNonWorkingDays: true,
|
||||
holidays: holidayPresets.US,
|
||||
workingDays: workingDayPresets.standard,
|
||||
dayWidth: 30,
|
||||
}}
|
||||
onTaskUpdate={handleTaskUpdate}
|
||||
onTaskMove={handleTaskMove}
|
||||
onProgressChange={handleProgressChange}
|
||||
onSelectionChange={handleSelectionChange}
|
||||
enableDragDrop={true}
|
||||
enableResize={true}
|
||||
enableProgressEdit={true}
|
||||
enableInlineEdit={true}
|
||||
enableVirtualScrolling={true}
|
||||
className="h-full"
|
||||
/>
|
||||
</div>
|
||||
|
||||
{/* Feature List */}
|
||||
<div className="bg-white dark:bg-gray-800 border border-gray-200 dark:border-gray-700 rounded-lg shadow-sm mt-4">
|
||||
<div className="p-6">
|
||||
<h3 className="text-lg font-semibold text-gray-900 dark:text-gray-100 mb-3">
|
||||
✨ Advanced Features Demonstrated
|
||||
</h3>
|
||||
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4 text-sm">
|
||||
<div className="space-y-2">
|
||||
<h4 className="font-medium text-gray-900 dark:text-gray-100">Performance & UX</h4>
|
||||
<ul className="space-y-1 text-gray-600 dark:text-gray-400">
|
||||
<li>• Virtual scrolling for 1000+ tasks</li>
|
||||
<li>• Smooth 60fps drag & drop</li>
|
||||
<li>• Debounced updates</li>
|
||||
<li>• Memory-optimized rendering</li>
|
||||
<li>• Responsive design</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div className="space-y-2">
|
||||
<h4 className="font-medium text-gray-900 dark:text-gray-100">Gantt Features</h4>
|
||||
<ul className="space-y-1 text-gray-600 dark:text-gray-400">
|
||||
<li>• Draggable task bars</li>
|
||||
<li>• Resizable task duration</li>
|
||||
<li>• Progress editing</li>
|
||||
<li>• Multi-level hierarchy</li>
|
||||
<li>• Task dependencies</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div className="space-y-2">
|
||||
<h4 className="font-medium text-gray-900 dark:text-gray-100">Timeline & Markers</h4>
|
||||
<ul className="space-y-1 text-gray-600 dark:text-gray-400">
|
||||
<li>• Weekend & holiday markers</li>
|
||||
<li>• Working day indicators</li>
|
||||
<li>• Today line</li>
|
||||
<li>• Multi-tier timeline</li>
|
||||
<li>• Zoom levels (Year/Month/Week/Day)</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div className="space-y-2">
|
||||
<h4 className="font-medium text-gray-900 dark:text-gray-100">Grid Features</h4>
|
||||
<ul className="space-y-1 text-gray-600 dark:text-gray-400">
|
||||
<li>• Fixed columns layout</li>
|
||||
<li>• Inline editing</li>
|
||||
<li>• Column resizing</li>
|
||||
<li>• Multi-select</li>
|
||||
<li>• Hierarchical tree view</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div className="space-y-2">
|
||||
<h4 className="font-medium text-gray-900 dark:text-gray-100">UI/UX</h4>
|
||||
<ul className="space-y-1 text-gray-600 dark:text-gray-400">
|
||||
<li>• Dark/Light theme support</li>
|
||||
<li>• Tailwind CSS styling</li>
|
||||
<li>• Consistent with Worklenz</li>
|
||||
<li>• Accessibility features</li>
|
||||
<li>• Mobile responsive</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div className="space-y-2">
|
||||
<h4 className="font-medium text-gray-900 dark:text-gray-100">Architecture</h4>
|
||||
<ul className="space-y-1 text-gray-600 dark:text-gray-400">
|
||||
<li>• Modern React patterns</li>
|
||||
<li>• TypeScript safety</li>
|
||||
<li>• Optimized performance</li>
|
||||
<li>• Enterprise features</li>
|
||||
<li>• Best practices 2025</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
export default AdvancedGanttDemo;
|
||||
@@ -0,0 +1,304 @@
|
||||
import React, { useState, useRef, useCallback, useMemo } from 'react';
|
||||
import { GanttTask, DragState } from '../../types/advanced-gantt.types';
|
||||
import { useAppSelector } from '../../hooks/useAppSelector';
|
||||
import { themeWiseColor } from '../../utils/themeWiseColor';
|
||||
import { useDateCalculations } from '../../utils/gantt-performance';
|
||||
|
||||
interface DraggableTaskBarProps {
|
||||
task: GanttTask;
|
||||
timelineStart: Date;
|
||||
dayWidth: number;
|
||||
rowHeight: number;
|
||||
index: number;
|
||||
onTaskMove?: (taskId: string, newDates: { start: Date; end: Date }) => void;
|
||||
onTaskResize?: (taskId: string, newDates: { start: Date; end: Date }) => void;
|
||||
onProgressChange?: (taskId: string, progress: number) => void;
|
||||
onTaskClick?: (task: GanttTask) => void;
|
||||
onTaskDoubleClick?: (task: GanttTask) => void;
|
||||
enableDragDrop?: boolean;
|
||||
enableResize?: boolean;
|
||||
enableProgressEdit?: boolean;
|
||||
readOnly?: boolean;
|
||||
}
|
||||
|
||||
const DraggableTaskBar: React.FC<DraggableTaskBarProps> = ({
|
||||
task,
|
||||
timelineStart,
|
||||
dayWidth,
|
||||
rowHeight,
|
||||
index,
|
||||
onTaskMove,
|
||||
onTaskResize,
|
||||
onProgressChange,
|
||||
onTaskClick,
|
||||
onTaskDoubleClick,
|
||||
enableDragDrop = true,
|
||||
enableResize = true,
|
||||
enableProgressEdit = true,
|
||||
readOnly = false,
|
||||
}) => {
|
||||
const [dragState, setDragState] = useState<DragState | null>(null);
|
||||
const [hoverState, setHoverState] = useState<string | null>(null);
|
||||
const taskBarRef = useRef<HTMLDivElement>(null);
|
||||
const themeMode = useAppSelector(state => state.themeReducer.mode);
|
||||
const { getDaysBetween, addDays } = useDateCalculations();
|
||||
|
||||
// Calculate task position and dimensions
|
||||
const taskPosition = useMemo(() => {
|
||||
const startDays = getDaysBetween(timelineStart, task.startDate);
|
||||
const duration = getDaysBetween(task.startDate, task.endDate);
|
||||
|
||||
return {
|
||||
x: startDays * dayWidth,
|
||||
width: Math.max(dayWidth * 0.5, duration * dayWidth),
|
||||
y: index * rowHeight + 8, // 8px padding
|
||||
height: rowHeight - 16, // 16px total padding
|
||||
};
|
||||
}, [task.startDate, task.endDate, timelineStart, dayWidth, rowHeight, index, getDaysBetween]);
|
||||
|
||||
// Theme-aware colors
|
||||
const colors = useMemo(() => {
|
||||
const baseColor = task.color || getDefaultTaskColor(task.status);
|
||||
return {
|
||||
background: themeWiseColor(baseColor, adjustColorForDarkMode(baseColor), themeMode),
|
||||
border: themeWiseColor(darkenColor(baseColor, 0.2), lightenColor(baseColor, 0.2), themeMode),
|
||||
progress: themeWiseColor('#52c41a', '#34d399', themeMode),
|
||||
text: themeWiseColor('#ffffff', '#f9fafb', themeMode),
|
||||
hover: themeWiseColor(lightenColor(baseColor, 0.1), darkenColor(baseColor, 0.1), themeMode),
|
||||
};
|
||||
}, [task.color, task.status, themeMode]);
|
||||
|
||||
// Mouse event handlers
|
||||
const handleMouseDown = useCallback((e: React.MouseEvent, dragType: DragState['dragType']) => {
|
||||
if (readOnly || !enableDragDrop) return;
|
||||
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
|
||||
const rect = taskBarRef.current?.getBoundingClientRect();
|
||||
if (!rect) return;
|
||||
|
||||
setDragState({
|
||||
isDragging: true,
|
||||
dragType,
|
||||
taskId: task.id,
|
||||
initialPosition: { x: e.clientX, y: e.clientY },
|
||||
currentPosition: { x: e.clientX, y: e.clientY },
|
||||
initialDates: { start: task.startDate, end: task.endDate },
|
||||
initialProgress: task.progress,
|
||||
snapToGrid: true,
|
||||
});
|
||||
|
||||
// Add global mouse event listeners
|
||||
const handleMouseMove = (moveEvent: MouseEvent) => {
|
||||
handleMouseMove_Internal(moveEvent, dragType);
|
||||
};
|
||||
|
||||
const handleMouseUp = () => {
|
||||
handleMouseUp_Internal();
|
||||
document.removeEventListener('mousemove', handleMouseMove);
|
||||
document.removeEventListener('mouseup', handleMouseUp);
|
||||
};
|
||||
|
||||
document.addEventListener('mousemove', handleMouseMove);
|
||||
document.addEventListener('mouseup', handleMouseUp);
|
||||
}, [readOnly, enableDragDrop, task]);
|
||||
|
||||
const handleMouseMove_Internal = useCallback((e: MouseEvent, dragType: DragState['dragType']) => {
|
||||
if (!dragState) return;
|
||||
|
||||
const deltaX = e.clientX - dragState.initialPosition.x;
|
||||
const deltaDays = Math.round(deltaX / dayWidth);
|
||||
|
||||
let newStartDate = task.startDate;
|
||||
let newEndDate = task.endDate;
|
||||
|
||||
switch (dragType) {
|
||||
case 'move':
|
||||
newStartDate = addDays(dragState.initialDates.start, deltaDays);
|
||||
newEndDate = addDays(dragState.initialDates.end, deltaDays);
|
||||
break;
|
||||
|
||||
case 'resize-start':
|
||||
newStartDate = addDays(dragState.initialDates.start, deltaDays);
|
||||
// Ensure minimum duration
|
||||
if (newStartDate >= newEndDate) {
|
||||
newStartDate = addDays(newEndDate, -1);
|
||||
}
|
||||
break;
|
||||
|
||||
case 'resize-end':
|
||||
newEndDate = addDays(dragState.initialDates.end, deltaDays);
|
||||
// Ensure minimum duration
|
||||
if (newEndDate <= newStartDate) {
|
||||
newEndDate = addDays(newStartDate, 1);
|
||||
}
|
||||
break;
|
||||
|
||||
case 'progress':
|
||||
if (enableProgressEdit) {
|
||||
const progressDelta = deltaX / taskPosition.width;
|
||||
const newProgress = Math.max(0, Math.min(100, (dragState.initialProgress || 0) + progressDelta * 100));
|
||||
onProgressChange?.(task.id, newProgress);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
// Update drag state
|
||||
setDragState(prev => prev ? {
|
||||
...prev,
|
||||
currentPosition: { x: e.clientX, y: e.clientY },
|
||||
} : null);
|
||||
|
||||
// Call appropriate handler
|
||||
if (dragType === 'move') {
|
||||
onTaskMove?.(task.id, { start: newStartDate, end: newEndDate });
|
||||
} else if (dragType.startsWith('resize')) {
|
||||
onTaskResize?.(task.id, { start: newStartDate, end: newEndDate });
|
||||
}
|
||||
}, [dragState, dayWidth, task, taskPosition.width, enableProgressEdit, onTaskMove, onTaskResize, onProgressChange, addDays]);
|
||||
|
||||
const handleMouseUp_Internal = useCallback(() => {
|
||||
setDragState(null);
|
||||
}, []);
|
||||
|
||||
const handleClick = useCallback((e: React.MouseEvent) => {
|
||||
e.stopPropagation();
|
||||
onTaskClick?.(task);
|
||||
}, [task, onTaskClick]);
|
||||
|
||||
const handleDoubleClick = useCallback((e: React.MouseEvent) => {
|
||||
e.stopPropagation();
|
||||
onTaskDoubleClick?.(task);
|
||||
}, [task, onTaskDoubleClick]);
|
||||
|
||||
// Render task bar with handles
|
||||
const renderTaskBar = () => {
|
||||
const isSelected = false; // TODO: Get from selection state
|
||||
const isDragging = dragState?.isDragging || false;
|
||||
|
||||
return (
|
||||
<div
|
||||
ref={taskBarRef}
|
||||
className={`task-bar relative cursor-pointer select-none transition-all duration-200 ${
|
||||
isDragging ? 'z-10 shadow-lg' : ''
|
||||
} ${isSelected ? 'ring-2 ring-blue-500 ring-opacity-50' : ''}`}
|
||||
style={{
|
||||
position: 'absolute',
|
||||
left: taskPosition.x,
|
||||
top: taskPosition.y,
|
||||
width: taskPosition.width,
|
||||
height: taskPosition.height,
|
||||
backgroundColor: dragState?.isDragging ? colors.hover : colors.background,
|
||||
border: `1px solid ${colors.border}`,
|
||||
borderRadius: '4px',
|
||||
transform: isDragging ? 'translateY(-2px)' : 'none',
|
||||
boxShadow: isDragging ? '0 4px 12px rgba(0,0,0,0.15)' : '0 1px 3px rgba(0,0,0,0.1)',
|
||||
}}
|
||||
onClick={handleClick}
|
||||
onDoubleClick={handleDoubleClick}
|
||||
onMouseEnter={() => setHoverState('task')}
|
||||
onMouseLeave={() => setHoverState(null)}
|
||||
onMouseDown={(e) => handleMouseDown(e, 'move')}
|
||||
>
|
||||
{/* Progress bar */}
|
||||
<div
|
||||
className="progress-bar absolute inset-0 rounded-l"
|
||||
style={{
|
||||
width: `${task.progress}%`,
|
||||
backgroundColor: colors.progress,
|
||||
opacity: 0.7,
|
||||
borderRadius: '3px 0 0 3px',
|
||||
}}
|
||||
/>
|
||||
|
||||
{/* Task content */}
|
||||
<div className="task-content relative z-10 h-full flex items-center px-2">
|
||||
<span
|
||||
className="task-name text-xs font-medium truncate"
|
||||
style={{ color: colors.text }}
|
||||
>
|
||||
{task.name}
|
||||
</span>
|
||||
|
||||
{/* Duration display for smaller tasks */}
|
||||
{taskPosition.width < 100 && (
|
||||
<span
|
||||
className="task-duration text-xs ml-auto"
|
||||
style={{ color: colors.text, opacity: 0.8 }}
|
||||
>
|
||||
{getDaysBetween(task.startDate, task.endDate)}d
|
||||
</span>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Resize handles */}
|
||||
{enableResize && !readOnly && hoverState === 'task' && (
|
||||
<>
|
||||
{/* Left resize handle */}
|
||||
<div
|
||||
className="resize-handle-left absolute left-0 top-0 w-1 h-full cursor-ew-resize bg-white bg-opacity-50 hover:bg-opacity-80"
|
||||
onMouseDown={(e) => handleMouseDown(e, 'resize-start')}
|
||||
onMouseEnter={() => setHoverState('resize-start')}
|
||||
/>
|
||||
|
||||
{/* Right resize handle */}
|
||||
<div
|
||||
className="resize-handle-right absolute right-0 top-0 w-1 h-full cursor-ew-resize bg-white bg-opacity-50 hover:bg-opacity-80"
|
||||
onMouseDown={(e) => handleMouseDown(e, 'resize-end')}
|
||||
onMouseEnter={() => setHoverState('resize-end')}
|
||||
/>
|
||||
</>
|
||||
)}
|
||||
|
||||
{/* Progress handle */}
|
||||
{enableProgressEdit && !readOnly && hoverState === 'task' && (
|
||||
<div
|
||||
className="progress-handle absolute top-0 h-full w-1 cursor-ew-resize bg-blue-500 opacity-75"
|
||||
style={{ left: `${task.progress}%` }}
|
||||
onMouseDown={(e) => handleMouseDown(e, 'progress')}
|
||||
onMouseEnter={() => setHoverState('progress')}
|
||||
/>
|
||||
)}
|
||||
|
||||
{/* Task type indicator */}
|
||||
{task.type === 'milestone' && (
|
||||
<div
|
||||
className="milestone-indicator absolute -top-1 -right-1 w-3 h-3 transform rotate-45"
|
||||
style={{ backgroundColor: colors.border }}
|
||||
/>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
return renderTaskBar();
|
||||
};
|
||||
|
||||
// Helper functions
|
||||
function getDefaultTaskColor(status: GanttTask['status']): string {
|
||||
switch (status) {
|
||||
case 'completed': return '#52c41a';
|
||||
case 'in-progress': return '#1890ff';
|
||||
case 'overdue': return '#ff4d4f';
|
||||
case 'on-hold': return '#faad14';
|
||||
default: return '#d9d9d9';
|
||||
}
|
||||
}
|
||||
|
||||
function darkenColor(color: string, amount: number): string {
|
||||
// Simple color darkening - in a real app, use a proper color manipulation library
|
||||
return color;
|
||||
}
|
||||
|
||||
function lightenColor(color: string, amount: number): string {
|
||||
// Simple color lightening - in a real app, use a proper color manipulation library
|
||||
return color;
|
||||
}
|
||||
|
||||
function adjustColorForDarkMode(color: string): string {
|
||||
// Adjust color for dark mode - in a real app, use a proper color manipulation library
|
||||
return color;
|
||||
}
|
||||
|
||||
export default DraggableTaskBar;
|
||||
492
worklenz-frontend/src/components/advanced-gantt/GanttGrid.tsx
Normal file
492
worklenz-frontend/src/components/advanced-gantt/GanttGrid.tsx
Normal file
@@ -0,0 +1,492 @@
|
||||
import React, { useMemo, useRef, useState, useCallback } from 'react';
|
||||
import { GanttTask, ColumnConfig, SelectionState } from '../../types/advanced-gantt.types';
|
||||
import { useAppSelector } from '../../hooks/useAppSelector';
|
||||
import { themeWiseColor } from '../../utils/themeWiseColor';
|
||||
import { ChevronRightIcon, ChevronDownIcon } from '@heroicons/react/24/outline';
|
||||
import { CalendarIcon, UserIcon, FlagIcon } from '@heroicons/react/24/solid';
|
||||
|
||||
interface GanttGridProps {
|
||||
tasks: GanttTask[];
|
||||
columns: ColumnConfig[];
|
||||
rowHeight: number;
|
||||
containerHeight: number;
|
||||
selection: SelectionState;
|
||||
enableInlineEdit?: boolean;
|
||||
enableMultiSelect?: boolean;
|
||||
onTaskClick?: (task: GanttTask, event: React.MouseEvent) => void;
|
||||
onTaskDoubleClick?: (task: GanttTask) => void;
|
||||
onTaskExpand?: (taskId: string) => void;
|
||||
onSelectionChange?: (selection: SelectionState) => void;
|
||||
onColumnResize?: (columnField: string, newWidth: number) => void;
|
||||
onTaskUpdate?: (taskId: string, field: string, value: any) => void;
|
||||
className?: string;
|
||||
}
|
||||
|
||||
const GanttGrid: React.FC<GanttGridProps> = ({
|
||||
tasks,
|
||||
columns,
|
||||
rowHeight,
|
||||
containerHeight,
|
||||
selection,
|
||||
enableInlineEdit = true,
|
||||
enableMultiSelect = true,
|
||||
onTaskClick,
|
||||
onTaskDoubleClick,
|
||||
onTaskExpand,
|
||||
onSelectionChange,
|
||||
onColumnResize,
|
||||
onTaskUpdate,
|
||||
className = '',
|
||||
}) => {
|
||||
const [editingCell, setEditingCell] = useState<{ taskId: string; field: string } | null>(null);
|
||||
const [columnWidths, setColumnWidths] = useState<Record<string, number>>(
|
||||
columns.reduce((acc, col) => ({ ...acc, [col.field]: col.width }), {})
|
||||
);
|
||||
const gridRef = useRef<HTMLDivElement>(null);
|
||||
const themeMode = useAppSelector(state => state.themeReducer.mode);
|
||||
|
||||
// Theme-aware colors
|
||||
const colors = useMemo(() => ({
|
||||
background: themeWiseColor('#ffffff', '#1f2937', themeMode),
|
||||
alternateRow: themeWiseColor('#f9fafb', '#374151', themeMode),
|
||||
border: themeWiseColor('#e5e7eb', '#4b5563', themeMode),
|
||||
text: themeWiseColor('#111827', '#f9fafb', themeMode),
|
||||
textSecondary: themeWiseColor('#6b7280', '#d1d5db', themeMode),
|
||||
selected: themeWiseColor('#eff6ff', '#1e3a8a', themeMode),
|
||||
hover: themeWiseColor('#f3f4f6', '#4b5563', themeMode),
|
||||
headerBg: themeWiseColor('#f8f9fa', '#374151', themeMode),
|
||||
}), [themeMode]);
|
||||
|
||||
// Calculate total grid width
|
||||
const totalWidth = useMemo(() => {
|
||||
return columns.reduce((sum, col) => sum + columnWidths[col.field], 0);
|
||||
}, [columns, columnWidths]);
|
||||
|
||||
// Handle column resize
|
||||
const handleColumnResize = useCallback((columnField: string, deltaX: number) => {
|
||||
const column = columns.find(col => col.field === columnField);
|
||||
if (!column) return;
|
||||
|
||||
const currentWidth = columnWidths[columnField];
|
||||
const newWidth = Math.max(column.minWidth || 60, Math.min(column.maxWidth || 400, currentWidth + deltaX));
|
||||
|
||||
setColumnWidths(prev => ({ ...prev, [columnField]: newWidth }));
|
||||
onColumnResize?.(columnField, newWidth);
|
||||
}, [columns, columnWidths, onColumnResize]);
|
||||
|
||||
// Handle task selection
|
||||
const handleTaskSelection = useCallback((task: GanttTask, event: React.MouseEvent) => {
|
||||
const { ctrlKey, shiftKey } = event;
|
||||
let newSelectedTasks = [...selection.selectedTasks];
|
||||
|
||||
if (shiftKey && enableMultiSelect && selection.selectedTasks.length > 0) {
|
||||
// Range selection
|
||||
const lastSelectedIndex = tasks.findIndex(t => t.id === selection.selectedTasks[selection.selectedTasks.length - 1]);
|
||||
const currentIndex = tasks.findIndex(t => t.id === task.id);
|
||||
const [start, end] = [Math.min(lastSelectedIndex, currentIndex), Math.max(lastSelectedIndex, currentIndex)];
|
||||
|
||||
newSelectedTasks = tasks.slice(start, end + 1).map(t => t.id);
|
||||
} else if (ctrlKey && enableMultiSelect) {
|
||||
// Multi selection
|
||||
if (newSelectedTasks.includes(task.id)) {
|
||||
newSelectedTasks = newSelectedTasks.filter(id => id !== task.id);
|
||||
} else {
|
||||
newSelectedTasks.push(task.id);
|
||||
}
|
||||
} else {
|
||||
// Single selection
|
||||
newSelectedTasks = [task.id];
|
||||
}
|
||||
|
||||
onSelectionChange?.({
|
||||
...selection,
|
||||
selectedTasks: newSelectedTasks,
|
||||
focusedTask: task.id,
|
||||
});
|
||||
|
||||
onTaskClick?.(task, event);
|
||||
}, [tasks, selection, enableMultiSelect, onSelectionChange, onTaskClick]);
|
||||
|
||||
// Handle cell editing
|
||||
const handleCellDoubleClick = useCallback((task: GanttTask, column: ColumnConfig) => {
|
||||
if (!enableInlineEdit || !column.editor) return;
|
||||
|
||||
setEditingCell({ taskId: task.id, field: column.field });
|
||||
}, [enableInlineEdit]);
|
||||
|
||||
const handleCellEditComplete = useCallback((value: any) => {
|
||||
if (!editingCell) return;
|
||||
|
||||
onTaskUpdate?.(editingCell.taskId, editingCell.field, value);
|
||||
setEditingCell(null);
|
||||
}, [editingCell, onTaskUpdate]);
|
||||
|
||||
// Render cell content
|
||||
const renderCellContent = useCallback((task: GanttTask, column: ColumnConfig) => {
|
||||
const value = task[column.field as keyof GanttTask];
|
||||
const isEditing = editingCell?.taskId === task.id && editingCell?.field === column.field;
|
||||
|
||||
if (isEditing) {
|
||||
return renderCellEditor(value, column, handleCellEditComplete);
|
||||
}
|
||||
|
||||
if (column.renderer) {
|
||||
return column.renderer(value, task);
|
||||
}
|
||||
|
||||
// Default renderers
|
||||
switch (column.field) {
|
||||
case 'name':
|
||||
return (
|
||||
<div className="flex items-center space-x-2">
|
||||
{task.hasChildren && (
|
||||
<button
|
||||
onClick={(e) => {
|
||||
e.stopPropagation();
|
||||
onTaskExpand?.(task.id);
|
||||
}}
|
||||
className="p-0.5 hover:bg-gray-200 dark:hover:bg-gray-600 rounded"
|
||||
>
|
||||
{task.isExpanded ? (
|
||||
<ChevronDownIcon className="w-3 h-3" />
|
||||
) : (
|
||||
<ChevronRightIcon className="w-3 h-3" />
|
||||
)}
|
||||
</button>
|
||||
)}
|
||||
<div
|
||||
className="flex items-center space-x-2"
|
||||
style={{ paddingLeft: `${(task.level || 0) * 16}px` }}
|
||||
>
|
||||
{getTaskTypeIcon(task.type)}
|
||||
<span className="truncate font-medium">{task.name}</span>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
|
||||
case 'startDate':
|
||||
case 'endDate':
|
||||
return (
|
||||
<div className="flex items-center space-x-1">
|
||||
<CalendarIcon className="w-3 h-3 text-gray-400" />
|
||||
<span>{(value as Date)?.toLocaleDateString() || '-'}</span>
|
||||
</div>
|
||||
);
|
||||
|
||||
case 'assignee':
|
||||
return task.assignee ? (
|
||||
<div className="flex items-center space-x-2">
|
||||
{task.assignee.avatar ? (
|
||||
<img
|
||||
src={task.assignee.avatar}
|
||||
alt={task.assignee.name}
|
||||
className="w-6 h-6 rounded-full"
|
||||
/>
|
||||
) : (
|
||||
<UserIcon className="w-6 h-6 text-gray-400" />
|
||||
)}
|
||||
<span className="truncate">{task.assignee.name}</span>
|
||||
</div>
|
||||
) : (
|
||||
<span className="text-gray-400">Unassigned</span>
|
||||
);
|
||||
|
||||
case 'progress':
|
||||
return (
|
||||
<div className="flex items-center space-x-2">
|
||||
<div className="flex-1 bg-gray-200 dark:bg-gray-600 rounded-full h-2">
|
||||
<div
|
||||
className="bg-blue-500 h-2 rounded-full transition-all duration-300"
|
||||
style={{ width: `${task.progress}%` }}
|
||||
/>
|
||||
</div>
|
||||
<span className="text-xs w-8 text-right">{task.progress}%</span>
|
||||
</div>
|
||||
);
|
||||
|
||||
case 'status':
|
||||
return (
|
||||
<span className={`px-2 py-1 rounded-full text-xs font-medium ${getStatusColor(task.status)}`}>
|
||||
{task.status.replace('-', ' ')}
|
||||
</span>
|
||||
);
|
||||
|
||||
case 'priority':
|
||||
return (
|
||||
<div className="flex items-center space-x-1">
|
||||
<FlagIcon className={`w-3 h-3 ${getPriorityColor(task.priority)}`} />
|
||||
<span className="capitalize">{task.priority}</span>
|
||||
</div>
|
||||
);
|
||||
|
||||
case 'duration':
|
||||
const duration = task.duration || Math.ceil((task.endDate.getTime() - task.startDate.getTime()) / (1000 * 60 * 60 * 24));
|
||||
return <span>{duration}d</span>;
|
||||
|
||||
default:
|
||||
return <span>{String(value || '')}</span>;
|
||||
}
|
||||
}, [editingCell, onTaskExpand, handleCellEditComplete]);
|
||||
|
||||
// Render header
|
||||
const renderHeader = () => (
|
||||
<div
|
||||
className="grid-header flex border-b sticky top-0 z-10"
|
||||
style={{
|
||||
backgroundColor: colors.headerBg,
|
||||
borderColor: colors.border,
|
||||
height: rowHeight,
|
||||
}}
|
||||
>
|
||||
{columns.map((column, index) => (
|
||||
<div
|
||||
key={column.field}
|
||||
className="column-header flex items-center px-3 py-2 font-medium text-sm border-r relative group"
|
||||
style={{
|
||||
width: columnWidths[column.field],
|
||||
borderColor: colors.border,
|
||||
color: colors.text,
|
||||
}}
|
||||
>
|
||||
<span className="truncate" title={column.title}>
|
||||
{column.title}
|
||||
</span>
|
||||
|
||||
{/* Resize handle */}
|
||||
{column.resizable && (
|
||||
<ResizeHandle
|
||||
onResize={(deltaX) => handleColumnResize(column.field, deltaX)}
|
||||
className="absolute right-0 top-0 h-full w-1 cursor-col-resize hover:bg-blue-500 opacity-0 group-hover:opacity-100"
|
||||
/>
|
||||
)}
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
);
|
||||
|
||||
// Render task rows
|
||||
const renderRows = () => (
|
||||
<div className="grid-body">
|
||||
{tasks.map((task, rowIndex) => {
|
||||
const isSelected = selection.selectedTasks.includes(task.id);
|
||||
const isFocused = selection.focusedTask === task.id;
|
||||
|
||||
return (
|
||||
<div
|
||||
key={task.id}
|
||||
className={`grid-row flex border-b cursor-pointer hover:bg-opacity-75 ${
|
||||
isSelected ? 'bg-blue-50 dark:bg-blue-900 bg-opacity-50' :
|
||||
rowIndex % 2 === 0 ? '' : 'bg-gray-50 dark:bg-gray-800 bg-opacity-30'
|
||||
}`}
|
||||
style={{
|
||||
height: rowHeight,
|
||||
borderColor: colors.border,
|
||||
backgroundColor: isSelected ? colors.selected :
|
||||
rowIndex % 2 === 0 ? 'transparent' : colors.alternateRow,
|
||||
}}
|
||||
onClick={(e) => handleTaskSelection(task, e)}
|
||||
onDoubleClick={() => onTaskDoubleClick?.(task)}
|
||||
>
|
||||
{columns.map((column) => (
|
||||
<div
|
||||
key={`${task.id}-${column.field}`}
|
||||
className="grid-cell flex items-center px-3 py-1 border-r overflow-hidden"
|
||||
style={{
|
||||
width: columnWidths[column.field],
|
||||
borderColor: colors.border,
|
||||
textAlign: column.align || 'left',
|
||||
}}
|
||||
onDoubleClick={() => handleCellDoubleClick(task, column)}
|
||||
>
|
||||
{renderCellContent(task, column)}
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
);
|
||||
|
||||
return (
|
||||
<div
|
||||
ref={gridRef}
|
||||
className={`gantt-grid border-r ${className}`}
|
||||
style={{
|
||||
width: totalWidth,
|
||||
height: containerHeight,
|
||||
backgroundColor: colors.background,
|
||||
borderColor: colors.border,
|
||||
}}
|
||||
>
|
||||
{renderHeader()}
|
||||
<div
|
||||
className="grid-content overflow-auto"
|
||||
style={{ height: containerHeight - rowHeight }}
|
||||
>
|
||||
{renderRows()}
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
// Resize handle component
|
||||
interface ResizeHandleProps {
|
||||
onResize: (deltaX: number) => void;
|
||||
className?: string;
|
||||
}
|
||||
|
||||
const ResizeHandle: React.FC<ResizeHandleProps> = ({ onResize, className }) => {
|
||||
const [isDragging, setIsDragging] = useState(false);
|
||||
const startXRef = useRef<number>(0);
|
||||
|
||||
const handleMouseDown = useCallback((e: React.MouseEvent) => {
|
||||
e.preventDefault();
|
||||
setIsDragging(true);
|
||||
startXRef.current = e.clientX;
|
||||
|
||||
const handleMouseMove = (moveEvent: MouseEvent) => {
|
||||
const deltaX = moveEvent.clientX - startXRef.current;
|
||||
onResize(deltaX);
|
||||
startXRef.current = moveEvent.clientX;
|
||||
};
|
||||
|
||||
const handleMouseUp = () => {
|
||||
setIsDragging(false);
|
||||
document.removeEventListener('mousemove', handleMouseMove);
|
||||
document.removeEventListener('mouseup', handleMouseUp);
|
||||
};
|
||||
|
||||
document.addEventListener('mousemove', handleMouseMove);
|
||||
document.addEventListener('mouseup', handleMouseUp);
|
||||
}, [onResize]);
|
||||
|
||||
return (
|
||||
<div
|
||||
className={`resize-handle ${className} ${isDragging ? 'bg-blue-500' : ''}`}
|
||||
onMouseDown={handleMouseDown}
|
||||
/>
|
||||
);
|
||||
};
|
||||
|
||||
// Cell editor component
|
||||
const renderCellEditor = (value: any, column: ColumnConfig, onComplete: (value: any) => void) => {
|
||||
const [editValue, setEditValue] = useState(value);
|
||||
|
||||
const handleKeyDown = (e: React.KeyboardEvent) => {
|
||||
if (e.key === 'Enter') {
|
||||
onComplete(editValue);
|
||||
} else if (e.key === 'Escape') {
|
||||
onComplete(value); // Cancel editing
|
||||
}
|
||||
};
|
||||
|
||||
const handleBlur = () => {
|
||||
onComplete(editValue);
|
||||
};
|
||||
|
||||
switch (column.editor) {
|
||||
case 'text':
|
||||
return (
|
||||
<input
|
||||
type="text"
|
||||
value={editValue}
|
||||
onChange={(e) => setEditValue(e.target.value)}
|
||||
onKeyDown={handleKeyDown}
|
||||
onBlur={handleBlur}
|
||||
className="w-full px-1 py-0.5 border rounded text-sm"
|
||||
autoFocus
|
||||
/>
|
||||
);
|
||||
|
||||
case 'date':
|
||||
return (
|
||||
<input
|
||||
type="date"
|
||||
value={editValue instanceof Date ? editValue.toISOString().split('T')[0] : editValue}
|
||||
onChange={(e) => setEditValue(new Date(e.target.value))}
|
||||
onKeyDown={handleKeyDown}
|
||||
onBlur={handleBlur}
|
||||
className="w-full px-1 py-0.5 border rounded text-sm"
|
||||
autoFocus
|
||||
/>
|
||||
);
|
||||
|
||||
case 'number':
|
||||
return (
|
||||
<input
|
||||
type="number"
|
||||
value={editValue}
|
||||
onChange={(e) => setEditValue(parseFloat(e.target.value))}
|
||||
onKeyDown={handleKeyDown}
|
||||
onBlur={handleBlur}
|
||||
className="w-full px-1 py-0.5 border rounded text-sm"
|
||||
autoFocus
|
||||
/>
|
||||
);
|
||||
|
||||
case 'select':
|
||||
return (
|
||||
<select
|
||||
value={editValue}
|
||||
onChange={(e) => setEditValue(e.target.value)}
|
||||
onKeyDown={handleKeyDown}
|
||||
onBlur={handleBlur}
|
||||
className="w-full px-1 py-0.5 border rounded text-sm"
|
||||
autoFocus
|
||||
>
|
||||
{column.editorOptions?.map((option: any) => (
|
||||
<option key={option.value} value={option.value}>
|
||||
{option.label}
|
||||
</option>
|
||||
))}
|
||||
</select>
|
||||
);
|
||||
|
||||
default:
|
||||
return <span>{String(value)}</span>;
|
||||
}
|
||||
};
|
||||
|
||||
// Helper functions
|
||||
const getTaskTypeIcon = (type: GanttTask['type']) => {
|
||||
switch (type) {
|
||||
case 'project':
|
||||
return <div className="w-3 h-3 bg-blue-500 rounded-sm" />;
|
||||
case 'milestone':
|
||||
return <div className="w-3 h-3 bg-yellow-500 rotate-45" />;
|
||||
default:
|
||||
return <div className="w-3 h-3 bg-gray-400 rounded-full" />;
|
||||
}
|
||||
};
|
||||
|
||||
const getStatusColor = (status: GanttTask['status']) => {
|
||||
switch (status) {
|
||||
case 'completed':
|
||||
return 'bg-green-100 text-green-800 dark:bg-green-900 dark:text-green-200';
|
||||
case 'in-progress':
|
||||
return 'bg-blue-100 text-blue-800 dark:bg-blue-900 dark:text-blue-200';
|
||||
case 'overdue':
|
||||
return 'bg-red-100 text-red-800 dark:bg-red-900 dark:text-red-200';
|
||||
case 'on-hold':
|
||||
return 'bg-yellow-100 text-yellow-800 dark:bg-yellow-900 dark:text-yellow-200';
|
||||
default:
|
||||
return 'bg-gray-100 text-gray-800 dark:bg-gray-700 dark:text-gray-200';
|
||||
}
|
||||
};
|
||||
|
||||
const getPriorityColor = (priority: GanttTask['priority']) => {
|
||||
switch (priority) {
|
||||
case 'critical':
|
||||
return 'text-red-600';
|
||||
case 'high':
|
||||
return 'text-orange-500';
|
||||
case 'medium':
|
||||
return 'text-yellow-500';
|
||||
case 'low':
|
||||
return 'text-green-500';
|
||||
default:
|
||||
return 'text-gray-400';
|
||||
}
|
||||
};
|
||||
|
||||
export default GanttGrid;
|
||||
@@ -0,0 +1,295 @@
|
||||
import React, { useMemo } from 'react';
|
||||
import { Holiday, TimelineConfig } from '../../types/advanced-gantt.types';
|
||||
import { useAppSelector } from '../../hooks/useAppSelector';
|
||||
import { themeWiseColor } from '../../utils/themeWiseColor';
|
||||
import { useDateCalculations } from '../../utils/gantt-performance';
|
||||
|
||||
interface TimelineMarkersProps {
|
||||
startDate: Date;
|
||||
endDate: Date;
|
||||
dayWidth: number;
|
||||
containerHeight: number;
|
||||
timelineConfig: TimelineConfig;
|
||||
holidays?: Holiday[];
|
||||
showWeekends?: boolean;
|
||||
showHolidays?: boolean;
|
||||
showToday?: boolean;
|
||||
className?: string;
|
||||
}
|
||||
|
||||
const TimelineMarkers: React.FC<TimelineMarkersProps> = ({
|
||||
startDate,
|
||||
endDate,
|
||||
dayWidth,
|
||||
containerHeight,
|
||||
timelineConfig,
|
||||
holidays = [],
|
||||
showWeekends = true,
|
||||
showHolidays = true,
|
||||
showToday = true,
|
||||
className = '',
|
||||
}) => {
|
||||
const themeMode = useAppSelector(state => state.themeReducer.mode);
|
||||
const { getDaysBetween, isWeekend, isWorkingDay } = useDateCalculations();
|
||||
|
||||
// Generate all dates in the timeline
|
||||
const timelineDates = useMemo(() => {
|
||||
const dates: Date[] = [];
|
||||
const totalDays = getDaysBetween(startDate, endDate);
|
||||
|
||||
for (let i = 0; i <= totalDays; i++) {
|
||||
const date = new Date(startDate);
|
||||
date.setDate(date.getDate() + i);
|
||||
dates.push(date);
|
||||
}
|
||||
|
||||
return dates;
|
||||
}, [startDate, endDate, getDaysBetween]);
|
||||
|
||||
// Theme-aware colors
|
||||
const colors = useMemo(() => ({
|
||||
weekend: themeWiseColor('rgba(0, 0, 0, 0.05)', 'rgba(255, 255, 255, 0.05)', themeMode),
|
||||
nonWorkingDay: themeWiseColor('rgba(0, 0, 0, 0.03)', 'rgba(255, 255, 255, 0.03)', themeMode),
|
||||
holiday: themeWiseColor('rgba(255, 107, 107, 0.1)', 'rgba(255, 107, 107, 0.15)', themeMode),
|
||||
today: themeWiseColor('rgba(24, 144, 255, 0.15)', 'rgba(64, 169, 255, 0.2)', themeMode),
|
||||
todayLine: themeWiseColor('#1890ff', '#40a9ff', themeMode),
|
||||
holidayBorder: themeWiseColor('#ff6b6b', '#ff8787', themeMode),
|
||||
}), [themeMode]);
|
||||
|
||||
// Check if a date is a holiday
|
||||
const isHoliday = (date: Date): Holiday | undefined => {
|
||||
return holidays.find(holiday => {
|
||||
if (holiday.recurring) {
|
||||
return holiday.date.getMonth() === date.getMonth() &&
|
||||
holiday.date.getDate() === date.getDate();
|
||||
}
|
||||
return holiday.date.toDateString() === date.toDateString();
|
||||
});
|
||||
};
|
||||
|
||||
// Check if date is today
|
||||
const isToday = (date: Date): boolean => {
|
||||
const today = new Date();
|
||||
return date.toDateString() === today.toDateString();
|
||||
};
|
||||
|
||||
// Render weekend markers
|
||||
const renderWeekendMarkers = () => {
|
||||
if (!showWeekends) return null;
|
||||
|
||||
return timelineDates.map((date, index) => {
|
||||
if (!isWeekend(date)) return null;
|
||||
|
||||
return (
|
||||
<div
|
||||
key={`weekend-${index}`}
|
||||
className="weekend-marker absolute top-0 pointer-events-none"
|
||||
style={{
|
||||
left: index * dayWidth,
|
||||
width: dayWidth,
|
||||
height: containerHeight,
|
||||
backgroundColor: colors.weekend,
|
||||
zIndex: 1,
|
||||
}}
|
||||
/>
|
||||
);
|
||||
});
|
||||
};
|
||||
|
||||
// Render non-working day markers
|
||||
const renderNonWorkingDayMarkers = () => {
|
||||
return timelineDates.map((date, index) => {
|
||||
if (isWorkingDay(date, timelineConfig.workingDays)) return null;
|
||||
|
||||
return (
|
||||
<div
|
||||
key={`non-working-${index}`}
|
||||
className="non-working-day-marker absolute top-0 pointer-events-none"
|
||||
style={{
|
||||
left: index * dayWidth,
|
||||
width: dayWidth,
|
||||
height: containerHeight,
|
||||
backgroundColor: colors.nonWorkingDay,
|
||||
zIndex: 1,
|
||||
}}
|
||||
/>
|
||||
);
|
||||
});
|
||||
};
|
||||
|
||||
// Render holiday markers
|
||||
const renderHolidayMarkers = () => {
|
||||
if (!showHolidays) return null;
|
||||
|
||||
return timelineDates.map((date, index) => {
|
||||
const holiday = isHoliday(date);
|
||||
if (!holiday) return null;
|
||||
|
||||
const holidayColor = holiday.color || colors.holiday;
|
||||
|
||||
return (
|
||||
<div
|
||||
key={`holiday-${index}`}
|
||||
className="holiday-marker absolute top-0 pointer-events-none group"
|
||||
style={{
|
||||
left: index * dayWidth,
|
||||
width: dayWidth,
|
||||
height: containerHeight,
|
||||
backgroundColor: holidayColor,
|
||||
borderLeft: `2px solid ${colors.holidayBorder}`,
|
||||
zIndex: 2,
|
||||
}}
|
||||
>
|
||||
{/* Holiday tooltip */}
|
||||
<div className="holiday-tooltip absolute top-0 left-1/2 transform -translate-x-1/2 -translate-y-full bg-gray-900 dark:bg-gray-100 text-white dark:text-gray-900 text-xs px-2 py-1 rounded opacity-0 group-hover:opacity-100 transition-opacity duration-200 whitespace-nowrap z-50">
|
||||
<div className="font-medium">{holiday.name}</div>
|
||||
<div className="text-xs opacity-75">{date.toLocaleDateString()}</div>
|
||||
<div className="absolute top-full left-1/2 transform -translate-x-1/2 w-0 h-0 border-l-2 border-r-2 border-t-4 border-transparent border-t-gray-900 dark:border-t-gray-100"></div>
|
||||
</div>
|
||||
|
||||
{/* Holiday icon */}
|
||||
<div className="holiday-icon absolute top-1 left-1 w-3 h-3 rounded-full bg-red-500 opacity-75">
|
||||
<div className="w-full h-full rounded-full animate-pulse"></div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
});
|
||||
};
|
||||
|
||||
// Render today marker
|
||||
const renderTodayMarker = () => {
|
||||
if (!showToday) return null;
|
||||
|
||||
const todayIndex = timelineDates.findIndex(date => isToday(date));
|
||||
if (todayIndex === -1) return null;
|
||||
|
||||
return (
|
||||
<div
|
||||
className="today-marker absolute top-0 pointer-events-none"
|
||||
style={{
|
||||
left: todayIndex * dayWidth,
|
||||
width: dayWidth,
|
||||
height: containerHeight,
|
||||
backgroundColor: colors.today,
|
||||
zIndex: 3,
|
||||
}}
|
||||
>
|
||||
{/* Today line */}
|
||||
<div
|
||||
className="today-line absolute top-0 left-1/2 transform -translate-x-1/2"
|
||||
style={{
|
||||
width: '2px',
|
||||
height: containerHeight,
|
||||
backgroundColor: colors.todayLine,
|
||||
zIndex: 4,
|
||||
}}
|
||||
/>
|
||||
|
||||
{/* Today label */}
|
||||
<div className="today-label absolute top-2 left-1/2 transform -translate-x-1/2 bg-blue-500 text-white text-xs px-2 py-1 rounded shadow-sm">
|
||||
Today
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
// Render time period markers (quarters, months, etc.)
|
||||
const renderTimePeriodMarkers = () => {
|
||||
const markers: React.ReactNode[] = [];
|
||||
const currentDate = new Date(startDate);
|
||||
currentDate.setDate(1); // Start of month
|
||||
|
||||
while (currentDate <= endDate) {
|
||||
const daysSinceStart = getDaysBetween(startDate, currentDate);
|
||||
const isQuarterStart = currentDate.getMonth() % 3 === 0 && currentDate.getDate() === 1;
|
||||
const isYearStart = currentDate.getMonth() === 0 && currentDate.getDate() === 1;
|
||||
|
||||
if (isYearStart) {
|
||||
markers.push(
|
||||
<div
|
||||
key={`year-${currentDate.getTime()}`}
|
||||
className="year-marker absolute top-0 border-l-2 border-blue-600 dark:border-blue-400"
|
||||
style={{
|
||||
left: daysSinceStart * dayWidth,
|
||||
height: containerHeight,
|
||||
zIndex: 5,
|
||||
}}
|
||||
>
|
||||
<div className="year-label absolute top-2 left-1 bg-blue-600 dark:bg-blue-400 text-white text-xs px-1 py-0.5 rounded">
|
||||
{currentDate.getFullYear()}
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
} else if (isQuarterStart) {
|
||||
markers.push(
|
||||
<div
|
||||
key={`quarter-${currentDate.getTime()}`}
|
||||
className="quarter-marker absolute top-0 border-l border-green-500 dark:border-green-400 opacity-60"
|
||||
style={{
|
||||
left: daysSinceStart * dayWidth,
|
||||
height: containerHeight,
|
||||
zIndex: 4,
|
||||
}}
|
||||
>
|
||||
<div className="quarter-label absolute top-2 left-1 bg-green-500 dark:bg-green-400 text-white text-xs px-1 py-0.5 rounded">
|
||||
Q{Math.floor(currentDate.getMonth() / 3) + 1}
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
// Move to next month
|
||||
currentDate.setMonth(currentDate.getMonth() + 1);
|
||||
}
|
||||
|
||||
return markers;
|
||||
};
|
||||
|
||||
return (
|
||||
<div className={`timeline-markers absolute inset-0 ${className}`}>
|
||||
{renderNonWorkingDayMarkers()}
|
||||
{renderWeekendMarkers()}
|
||||
{renderHolidayMarkers()}
|
||||
{renderTodayMarker()}
|
||||
{renderTimePeriodMarkers()}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
// Holiday presets for common countries
|
||||
export const holidayPresets = {
|
||||
US: [
|
||||
{ date: new Date(2024, 0, 1), name: "New Year's Day", type: 'national' as const, recurring: true },
|
||||
{ date: new Date(2024, 0, 15), name: "Martin Luther King Jr. Day", type: 'national' as const, recurring: true },
|
||||
{ date: new Date(2024, 1, 19), name: "Presidents' Day", type: 'national' as const, recurring: true },
|
||||
{ date: new Date(2024, 4, 27), name: "Memorial Day", type: 'national' as const, recurring: true },
|
||||
{ date: new Date(2024, 5, 19), name: "Juneteenth", type: 'national' as const, recurring: true },
|
||||
{ date: new Date(2024, 6, 4), name: "Independence Day", type: 'national' as const, recurring: true },
|
||||
{ date: new Date(2024, 8, 2), name: "Labor Day", type: 'national' as const, recurring: true },
|
||||
{ date: new Date(2024, 9, 14), name: "Columbus Day", type: 'national' as const, recurring: true },
|
||||
{ date: new Date(2024, 10, 11), name: "Veterans Day", type: 'national' as const, recurring: true },
|
||||
{ date: new Date(2024, 10, 28), name: "Thanksgiving", type: 'national' as const, recurring: true },
|
||||
{ date: new Date(2024, 11, 25), name: "Christmas Day", type: 'national' as const, recurring: true },
|
||||
],
|
||||
|
||||
UK: [
|
||||
{ date: new Date(2024, 0, 1), name: "New Year's Day", type: 'national' as const, recurring: true },
|
||||
{ date: new Date(2024, 2, 29), name: "Good Friday", type: 'religious' as const, recurring: false },
|
||||
{ date: new Date(2024, 3, 1), name: "Easter Monday", type: 'religious' as const, recurring: false },
|
||||
{ date: new Date(2024, 4, 6), name: "Early May Bank Holiday", type: 'national' as const, recurring: true },
|
||||
{ date: new Date(2024, 4, 27), name: "Spring Bank Holiday", type: 'national' as const, recurring: true },
|
||||
{ date: new Date(2024, 7, 26), name: "Summer Bank Holiday", type: 'national' as const, recurring: true },
|
||||
{ date: new Date(2024, 11, 25), name: "Christmas Day", type: 'religious' as const, recurring: true },
|
||||
{ date: new Date(2024, 11, 26), name: "Boxing Day", type: 'national' as const, recurring: true },
|
||||
],
|
||||
};
|
||||
|
||||
// Working day presets
|
||||
export const workingDayPresets = {
|
||||
standard: [1, 2, 3, 4, 5], // Monday to Friday
|
||||
middle_east: [0, 1, 2, 3, 4], // Sunday to Thursday
|
||||
six_day: [1, 2, 3, 4, 5, 6], // Monday to Saturday
|
||||
four_day: [1, 2, 3, 4], // Monday to Thursday
|
||||
};
|
||||
|
||||
export default TimelineMarkers;
|
||||
@@ -0,0 +1,372 @@
|
||||
import React, { useRef, useEffect, useState, useCallback, ReactNode } from 'react';
|
||||
import { useThrottle, usePerformanceMonitoring } from '../../utils/gantt-performance';
|
||||
import { useAppSelector } from '../../hooks/useAppSelector';
|
||||
|
||||
interface VirtualScrollContainerProps {
|
||||
items: any[];
|
||||
itemHeight: number;
|
||||
containerHeight: number;
|
||||
containerWidth?: number;
|
||||
overscan?: number;
|
||||
horizontal?: boolean;
|
||||
children: (item: any, index: number, style: React.CSSProperties) => ReactNode;
|
||||
onScroll?: (scrollLeft: number, scrollTop: number) => void;
|
||||
className?: string;
|
||||
style?: React.CSSProperties;
|
||||
}
|
||||
|
||||
const VirtualScrollContainer: React.FC<VirtualScrollContainerProps> = ({
|
||||
items,
|
||||
itemHeight,
|
||||
containerHeight,
|
||||
containerWidth = 0,
|
||||
overscan = 5,
|
||||
horizontal = false,
|
||||
children,
|
||||
onScroll,
|
||||
className = '',
|
||||
style = {},
|
||||
}) => {
|
||||
const containerRef = useRef<HTMLDivElement>(null);
|
||||
const [scrollTop, setScrollTop] = useState(0);
|
||||
const [scrollLeft, setScrollLeft] = useState(0);
|
||||
const { startMeasure, endMeasure, recordMetric } = usePerformanceMonitoring();
|
||||
const themeMode = useAppSelector(state => state.themeReducer.mode);
|
||||
|
||||
// Calculate visible range
|
||||
const totalHeight = items.length * itemHeight;
|
||||
const startIndex = Math.max(0, Math.floor(scrollTop / itemHeight) - overscan);
|
||||
const endIndex = Math.min(
|
||||
items.length - 1,
|
||||
Math.ceil((scrollTop + containerHeight) / itemHeight) + overscan
|
||||
);
|
||||
const visibleItems = items.slice(startIndex, endIndex + 1);
|
||||
const offsetY = startIndex * itemHeight;
|
||||
|
||||
// Throttled scroll handler
|
||||
const throttledScrollHandler = useThrottle(
|
||||
useCallback((event: Event) => {
|
||||
const target = event.target as HTMLDivElement;
|
||||
const newScrollTop = target.scrollTop;
|
||||
const newScrollLeft = target.scrollLeft;
|
||||
|
||||
setScrollTop(newScrollTop);
|
||||
setScrollLeft(newScrollLeft);
|
||||
onScroll?.(newScrollLeft, newScrollTop);
|
||||
}, [onScroll]),
|
||||
16 // ~60fps
|
||||
);
|
||||
|
||||
useEffect(() => {
|
||||
const container = containerRef.current;
|
||||
if (!container) return;
|
||||
|
||||
container.addEventListener('scroll', throttledScrollHandler, { passive: true });
|
||||
|
||||
return () => {
|
||||
container.removeEventListener('scroll', throttledScrollHandler);
|
||||
};
|
||||
}, [throttledScrollHandler]);
|
||||
|
||||
// Performance monitoring
|
||||
useEffect(() => {
|
||||
startMeasure('virtualScroll');
|
||||
recordMetric('visibleTaskCount', visibleItems.length);
|
||||
recordMetric('taskCount', items.length);
|
||||
endMeasure('virtualScroll');
|
||||
}, [visibleItems.length, items.length, startMeasure, endMeasure, recordMetric]);
|
||||
|
||||
const renderVisibleItems = () => {
|
||||
return visibleItems.map((item, virtualIndex) => {
|
||||
const actualIndex = startIndex + virtualIndex;
|
||||
const itemStyle: React.CSSProperties = {
|
||||
position: 'absolute',
|
||||
top: horizontal ? 0 : actualIndex * itemHeight,
|
||||
left: horizontal ? actualIndex * itemHeight : 0,
|
||||
height: horizontal ? '100%' : itemHeight,
|
||||
width: horizontal ? itemHeight : '100%',
|
||||
transform: horizontal ? 'none' : `translateY(${offsetY}px)`,
|
||||
};
|
||||
|
||||
return (
|
||||
<div key={item.id || actualIndex} style={itemStyle}>
|
||||
{children(item, actualIndex, itemStyle)}
|
||||
</div>
|
||||
);
|
||||
});
|
||||
};
|
||||
|
||||
return (
|
||||
<div
|
||||
ref={containerRef}
|
||||
className={`virtual-scroll-container overflow-auto ${className}`}
|
||||
style={{
|
||||
height: containerHeight,
|
||||
width: containerWidth || '100%',
|
||||
position: 'relative',
|
||||
...style,
|
||||
}}
|
||||
>
|
||||
{/* Spacer to maintain scroll height */}
|
||||
<div
|
||||
className="virtual-scroll-spacer"
|
||||
style={{
|
||||
height: horizontal ? '100%' : totalHeight,
|
||||
width: horizontal ? totalHeight : '100%',
|
||||
position: 'relative',
|
||||
pointerEvents: 'none',
|
||||
}}
|
||||
>
|
||||
{/* Visible items container */}
|
||||
<div
|
||||
className="virtual-scroll-content"
|
||||
style={{
|
||||
position: 'absolute',
|
||||
top: 0,
|
||||
left: 0,
|
||||
height: '100%',
|
||||
width: '100%',
|
||||
pointerEvents: 'auto',
|
||||
}}
|
||||
>
|
||||
{renderVisibleItems()}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
// Grid virtual scrolling component for both rows and columns
|
||||
interface VirtualGridProps {
|
||||
data: any[][];
|
||||
rowHeight: number;
|
||||
columnWidth: number | number[];
|
||||
containerHeight: number;
|
||||
containerWidth: number;
|
||||
overscan?: number;
|
||||
children: (item: any, rowIndex: number, colIndex: number, style: React.CSSProperties) => ReactNode;
|
||||
onScroll?: (scrollLeft: number, scrollTop: number) => void;
|
||||
className?: string;
|
||||
}
|
||||
|
||||
export const VirtualGrid: React.FC<VirtualGridProps> = ({
|
||||
data,
|
||||
rowHeight,
|
||||
columnWidth,
|
||||
containerHeight,
|
||||
containerWidth,
|
||||
overscan = 3,
|
||||
children,
|
||||
onScroll,
|
||||
className = '',
|
||||
}) => {
|
||||
const containerRef = useRef<HTMLDivElement>(null);
|
||||
const [scrollTop, setScrollTop] = useState(0);
|
||||
const [scrollLeft, setScrollLeft] = useState(0);
|
||||
|
||||
const rowCount = data.length;
|
||||
const colCount = data[0]?.length || 0;
|
||||
|
||||
// Calculate column positions for variable width columns
|
||||
const columnWidths = Array.isArray(columnWidth) ? columnWidth : new Array(colCount).fill(columnWidth);
|
||||
const columnPositions = columnWidths.reduce((acc, width, index) => {
|
||||
acc[index] = index === 0 ? 0 : acc[index - 1] + columnWidths[index - 1];
|
||||
return acc;
|
||||
}, {} as Record<number, number>);
|
||||
|
||||
const totalWidth = columnWidths.reduce((sum, width) => sum + width, 0);
|
||||
const totalHeight = rowCount * rowHeight;
|
||||
|
||||
// Calculate visible ranges
|
||||
const startRowIndex = Math.max(0, Math.floor(scrollTop / rowHeight) - overscan);
|
||||
const endRowIndex = Math.min(rowCount - 1, Math.ceil((scrollTop + containerHeight) / rowHeight) + overscan);
|
||||
|
||||
const startColIndex = Math.max(0, findColumnIndex(scrollLeft) - overscan);
|
||||
const endColIndex = Math.min(colCount - 1, findColumnIndex(scrollLeft + containerWidth) + overscan);
|
||||
|
||||
function findColumnIndex(position: number): number {
|
||||
for (let i = 0; i < colCount; i++) {
|
||||
if (columnPositions[i] <= position && position < columnPositions[i] + columnWidths[i]) {
|
||||
return i;
|
||||
}
|
||||
}
|
||||
return colCount - 1;
|
||||
}
|
||||
|
||||
const throttledScrollHandler = useThrottle(
|
||||
useCallback((event: Event) => {
|
||||
const target = event.target as HTMLDivElement;
|
||||
const newScrollTop = target.scrollTop;
|
||||
const newScrollLeft = target.scrollLeft;
|
||||
|
||||
setScrollTop(newScrollTop);
|
||||
setScrollLeft(newScrollLeft);
|
||||
onScroll?.(newScrollLeft, newScrollTop);
|
||||
}, [onScroll]),
|
||||
16
|
||||
);
|
||||
|
||||
useEffect(() => {
|
||||
const container = containerRef.current;
|
||||
if (!container) return;
|
||||
|
||||
container.addEventListener('scroll', throttledScrollHandler, { passive: true });
|
||||
|
||||
return () => {
|
||||
container.removeEventListener('scroll', throttledScrollHandler);
|
||||
};
|
||||
}, [throttledScrollHandler]);
|
||||
|
||||
const renderVisibleCells = () => {
|
||||
const cells: ReactNode[] = [];
|
||||
|
||||
for (let rowIndex = startRowIndex; rowIndex <= endRowIndex; rowIndex++) {
|
||||
for (let colIndex = startColIndex; colIndex <= endColIndex; colIndex++) {
|
||||
const item = data[rowIndex]?.[colIndex];
|
||||
if (!item) continue;
|
||||
|
||||
const cellStyle: React.CSSProperties = {
|
||||
position: 'absolute',
|
||||
top: rowIndex * rowHeight,
|
||||
left: columnPositions[colIndex],
|
||||
height: rowHeight,
|
||||
width: columnWidths[colIndex],
|
||||
};
|
||||
|
||||
cells.push(
|
||||
<div key={`${rowIndex}-${colIndex}`} style={cellStyle}>
|
||||
{children(item, rowIndex, colIndex, cellStyle)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
return cells;
|
||||
};
|
||||
|
||||
return (
|
||||
<div
|
||||
ref={containerRef}
|
||||
className={`virtual-grid overflow-auto ${className}`}
|
||||
style={{
|
||||
height: containerHeight,
|
||||
width: containerWidth,
|
||||
position: 'relative',
|
||||
}}
|
||||
>
|
||||
<div
|
||||
style={{
|
||||
height: totalHeight,
|
||||
width: totalWidth,
|
||||
position: 'relative',
|
||||
}}
|
||||
>
|
||||
{renderVisibleCells()}
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
// Timeline virtual scrolling component
|
||||
interface VirtualTimelineProps {
|
||||
startDate: Date;
|
||||
endDate: Date;
|
||||
dayWidth: number;
|
||||
containerWidth: number;
|
||||
containerHeight: number;
|
||||
overscan?: number;
|
||||
children: (date: Date, index: number, style: React.CSSProperties) => ReactNode;
|
||||
onScroll?: (scrollLeft: number) => void;
|
||||
className?: string;
|
||||
}
|
||||
|
||||
export const VirtualTimeline: React.FC<VirtualTimelineProps> = ({
|
||||
startDate,
|
||||
endDate,
|
||||
dayWidth,
|
||||
containerWidth,
|
||||
containerHeight,
|
||||
overscan = 10,
|
||||
children,
|
||||
onScroll,
|
||||
className = '',
|
||||
}) => {
|
||||
const containerRef = useRef<HTMLDivElement>(null);
|
||||
const [scrollLeft, setScrollLeft] = useState(0);
|
||||
|
||||
const totalDays = Math.ceil((endDate.getTime() - startDate.getTime()) / (1000 * 60 * 60 * 24));
|
||||
const totalWidth = totalDays * dayWidth;
|
||||
|
||||
const startDayIndex = Math.max(0, Math.floor(scrollLeft / dayWidth) - overscan);
|
||||
const endDayIndex = Math.min(totalDays - 1, Math.ceil((scrollLeft + containerWidth) / dayWidth) + overscan);
|
||||
|
||||
const throttledScrollHandler = useThrottle(
|
||||
useCallback((event: Event) => {
|
||||
const target = event.target as HTMLDivElement;
|
||||
const newScrollLeft = target.scrollLeft;
|
||||
setScrollLeft(newScrollLeft);
|
||||
onScroll?.(newScrollLeft);
|
||||
}, [onScroll]),
|
||||
16
|
||||
);
|
||||
|
||||
useEffect(() => {
|
||||
const container = containerRef.current;
|
||||
if (!container) return;
|
||||
|
||||
container.addEventListener('scroll', throttledScrollHandler, { passive: true });
|
||||
|
||||
return () => {
|
||||
container.removeEventListener('scroll', throttledScrollHandler);
|
||||
};
|
||||
}, [throttledScrollHandler]);
|
||||
|
||||
const renderVisibleDays = () => {
|
||||
const days: ReactNode[] = [];
|
||||
|
||||
for (let dayIndex = startDayIndex; dayIndex <= endDayIndex; dayIndex++) {
|
||||
const date = new Date(startDate);
|
||||
date.setDate(date.getDate() + dayIndex);
|
||||
|
||||
const dayStyle: React.CSSProperties = {
|
||||
position: 'absolute',
|
||||
left: dayIndex * dayWidth,
|
||||
top: 0,
|
||||
width: dayWidth,
|
||||
height: '100%',
|
||||
};
|
||||
|
||||
days.push(
|
||||
<div key={dayIndex} style={dayStyle}>
|
||||
{children(date, dayIndex, dayStyle)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
return days;
|
||||
};
|
||||
|
||||
return (
|
||||
<div
|
||||
ref={containerRef}
|
||||
className={`virtual-timeline overflow-x-auto ${className}`}
|
||||
style={{
|
||||
height: containerHeight,
|
||||
width: containerWidth,
|
||||
position: 'relative',
|
||||
}}
|
||||
>
|
||||
<div
|
||||
style={{
|
||||
height: '100%',
|
||||
width: totalWidth,
|
||||
position: 'relative',
|
||||
}}
|
||||
>
|
||||
{renderVisibleDays()}
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
export default VirtualScrollContainer;
|
||||
17
worklenz-frontend/src/components/advanced-gantt/index.ts
Normal file
17
worklenz-frontend/src/components/advanced-gantt/index.ts
Normal file
@@ -0,0 +1,17 @@
|
||||
// Main Components
|
||||
export { default as AdvancedGanttChart } from './AdvancedGanttChart';
|
||||
export { default as AdvancedGanttDemo } from './AdvancedGanttDemo';
|
||||
|
||||
// Core Components
|
||||
export { default as GanttGrid } from './GanttGrid';
|
||||
export { default as DraggableTaskBar } from './DraggableTaskBar';
|
||||
export { default as TimelineMarkers, holidayPresets, workingDayPresets } from './TimelineMarkers';
|
||||
|
||||
// Utility Components
|
||||
export { default as VirtualScrollContainer, VirtualGrid, VirtualTimeline } from './VirtualScrollContainer';
|
||||
|
||||
// Types
|
||||
export * from '../../types/advanced-gantt.types';
|
||||
|
||||
// Performance Utilities
|
||||
export * from '../../utils/gantt-performance';
|
||||
@@ -1,14 +1,4 @@
|
||||
import {
|
||||
AutoComplete,
|
||||
Button,
|
||||
Drawer,
|
||||
Flex,
|
||||
Form,
|
||||
message,
|
||||
Select,
|
||||
Spin,
|
||||
Typography,
|
||||
} from '@/shared/antd-imports';
|
||||
import { AutoComplete, Button, Drawer, Flex, Form, message, Modal, Select, Spin, Typography } from '@/shared/antd-imports';
|
||||
import { useAppSelector } from '@/hooks/useAppSelector';
|
||||
import { useAppDispatch } from '@/hooks/useAppDispatch';
|
||||
import {
|
||||
@@ -21,6 +11,7 @@ import { jobTitlesApiService } from '@/api/settings/job-titles/job-titles.api.se
|
||||
import { IJobTitle } from '@/types/job.types';
|
||||
import { teamMembersApiService } from '@/api/team-members/teamMembers.api.service';
|
||||
import { ITeamMemberCreateRequest } from '@/types/teamMembers/team-member-create-request';
|
||||
import { LinkOutlined } from '@ant-design/icons';
|
||||
|
||||
interface FormValues {
|
||||
email: string[];
|
||||
@@ -97,23 +88,33 @@ const InviteTeamMembers = () => {
|
||||
};
|
||||
|
||||
return (
|
||||
<Drawer
|
||||
<Modal
|
||||
title={
|
||||
<Typography.Text strong style={{ fontSize: 16 }}>
|
||||
{t('addMemberDrawerTitle')}
|
||||
</Typography.Text>
|
||||
}
|
||||
open={isDrawerOpen}
|
||||
onClose={handleClose}
|
||||
onCancel={handleClose}
|
||||
destroyOnClose
|
||||
afterOpenChange={visible => visible && handleSearch('')}
|
||||
width={400}
|
||||
loading={loading}
|
||||
footer={
|
||||
<Flex justify="end">
|
||||
<Button type="primary" onClick={form.submit}>
|
||||
{t('addToTeamButton')}
|
||||
</Button>
|
||||
<Flex justify="space-between">
|
||||
{/* <Button
|
||||
style={{ width: 140, fontSize: 12 }}
|
||||
block
|
||||
icon={<LinkOutlined />}
|
||||
disabled
|
||||
>
|
||||
{t('copyTeamLink')}
|
||||
</Button> */}
|
||||
<Flex justify="end">
|
||||
<Button onClick={form.submit} style={{ fontSize: 12 }}>
|
||||
{t('addToTeamButton')}
|
||||
</Button>
|
||||
</Flex>
|
||||
</Flex>
|
||||
}
|
||||
>
|
||||
@@ -186,7 +187,7 @@ const InviteTeamMembers = () => {
|
||||
/>
|
||||
</Form.Item>
|
||||
</Form>
|
||||
</Drawer>
|
||||
</Modal>
|
||||
);
|
||||
};
|
||||
|
||||
|
||||
@@ -10,6 +10,7 @@ import {
|
||||
Image,
|
||||
Input,
|
||||
Flex,
|
||||
theme,
|
||||
} from '@/shared/antd-imports';
|
||||
import React, { useEffect, useState } from 'react';
|
||||
import { useSelector } from 'react-redux';
|
||||
@@ -37,13 +38,12 @@ const TemplateDrawer: React.FC<TemplateDrawerProps> = ({
|
||||
showBothTabs = false,
|
||||
templateSelected = (templateId: string) => {
|
||||
if (!templateId) return;
|
||||
templateId;
|
||||
},
|
||||
selectedTemplateType = (type: 'worklenz' | 'custom') => {
|
||||
type;
|
||||
},
|
||||
}) => {
|
||||
const themeMode = useSelector((state: RootState) => state.themeReducer.mode);
|
||||
const { token } = theme.useToken();
|
||||
const { t } = useTranslation('template-drawer');
|
||||
|
||||
const [searchQuery, setSearchQuery] = useState('');
|
||||
@@ -149,7 +149,12 @@ const TemplateDrawer: React.FC<TemplateDrawerProps> = ({
|
||||
<Tag
|
||||
key={phase.name}
|
||||
color={phase.color_code}
|
||||
style={{ color: 'black', marginBottom: '8px' }}
|
||||
style={{
|
||||
color: token.colorText,
|
||||
marginBottom: '8px',
|
||||
backgroundColor: phase.color_code ? undefined : token.colorBgContainer,
|
||||
borderColor: phase.color_code ? undefined : token.colorBorder
|
||||
}}
|
||||
>
|
||||
{phase.name}
|
||||
</Tag>
|
||||
@@ -171,7 +176,12 @@ const TemplateDrawer: React.FC<TemplateDrawerProps> = ({
|
||||
<Tag
|
||||
key={status.name}
|
||||
color={status.color_code}
|
||||
style={{ color: 'black', marginBottom: '8px' }}
|
||||
style={{
|
||||
color: token.colorText,
|
||||
marginBottom: '8px',
|
||||
backgroundColor: status.color_code ? undefined : token.colorBgContainer,
|
||||
borderColor: status.color_code ? undefined : token.colorBorder
|
||||
}}
|
||||
>
|
||||
{status.name}
|
||||
</Tag>
|
||||
@@ -193,7 +203,12 @@ const TemplateDrawer: React.FC<TemplateDrawerProps> = ({
|
||||
<Tag
|
||||
key={priority.name}
|
||||
color={priority.color_code}
|
||||
style={{ color: 'black', marginBottom: '8px' }}
|
||||
style={{
|
||||
color: token.colorText,
|
||||
marginBottom: '8px',
|
||||
backgroundColor: priority.color_code ? undefined : token.colorBgContainer,
|
||||
borderColor: priority.color_code ? undefined : token.colorBorder
|
||||
}}
|
||||
>
|
||||
{priority.name}
|
||||
</Tag>
|
||||
@@ -215,7 +230,12 @@ const TemplateDrawer: React.FC<TemplateDrawerProps> = ({
|
||||
<Tag
|
||||
key={label.name}
|
||||
color={label.color_code}
|
||||
style={{ color: 'black', marginBottom: '8px' }}
|
||||
style={{
|
||||
color: token.colorText,
|
||||
marginBottom: '8px',
|
||||
backgroundColor: label.color_code ? undefined : token.colorBgContainer,
|
||||
borderColor: label.color_code ? undefined : token.colorBorder
|
||||
}}
|
||||
>
|
||||
{label.name}
|
||||
</Tag>
|
||||
@@ -251,14 +271,24 @@ const TemplateDrawer: React.FC<TemplateDrawerProps> = ({
|
||||
};
|
||||
|
||||
const menuContent = (
|
||||
<div style={{ display: 'flex' }}>
|
||||
<div style={{ display: 'flex', backgroundColor: token.colorBgContainer }}>
|
||||
{/* Menu Area */}
|
||||
<div style={{ minWidth: '250px', overflowY: 'auto', height: '100%' }}>
|
||||
<div style={{
|
||||
minWidth: '250px',
|
||||
overflowY: 'auto',
|
||||
height: '100%',
|
||||
backgroundColor: token.colorBgContainer,
|
||||
borderRight: `1px solid ${token.colorBorder}`
|
||||
}}>
|
||||
<Skeleton loading={loadingTemplates} active>
|
||||
<Menu
|
||||
className="template-menu"
|
||||
onClick={({ key }) => handleMenuClick(key)}
|
||||
style={{ width: 256 }}
|
||||
style={{
|
||||
width: 256,
|
||||
backgroundColor: token.colorBgContainer,
|
||||
borderColor: token.colorBorder
|
||||
}}
|
||||
defaultSelectedKeys={[templates[0]?.id || '']}
|
||||
mode="inline"
|
||||
items={menuItems}
|
||||
@@ -272,9 +302,11 @@ const TemplateDrawer: React.FC<TemplateDrawerProps> = ({
|
||||
flex: 1,
|
||||
maxHeight: 'calc(100vh - 200px)',
|
||||
padding: '16px',
|
||||
backgroundColor: token.colorBgContainer,
|
||||
color: token.colorText
|
||||
}}
|
||||
>
|
||||
<Title level={4}>Details</Title>
|
||||
<Title level={4} style={{ color: token.colorText }}>Details</Title>
|
||||
<Skeleton loading={loadingSelectedTemplate} active>
|
||||
{selectedTemplate?.image_url && (
|
||||
<Image preview={false} src={selectedTemplate.image_url} alt={selectedTemplate.name} />
|
||||
@@ -297,12 +329,17 @@ const TemplateDrawer: React.FC<TemplateDrawerProps> = ({
|
||||
};
|
||||
|
||||
const customTemplatesContent = (
|
||||
<div>
|
||||
<div style={{ backgroundColor: token.colorBgContainer, padding: '16px' }}>
|
||||
<Flex justify="space-between" align="center">
|
||||
<Input
|
||||
placeholder={t('searchTemplates')}
|
||||
suffix={<SearchOutlined />}
|
||||
style={{ maxWidth: '300px' }}
|
||||
suffix={<SearchOutlined style={{ color: token.colorTextTertiary }} />}
|
||||
style={{
|
||||
maxWidth: '300px',
|
||||
backgroundColor: token.colorBgContainer,
|
||||
borderColor: token.colorBorder,
|
||||
color: token.colorText
|
||||
}}
|
||||
onChange={e => setSearchQuery(e.target.value)}
|
||||
/>
|
||||
</Flex>
|
||||
@@ -312,10 +349,20 @@ const TemplateDrawer: React.FC<TemplateDrawerProps> = ({
|
||||
bordered
|
||||
dataSource={filteredCustomTemplates}
|
||||
loading={loadingCustomTemplates}
|
||||
style={{
|
||||
backgroundColor: token.colorBgContainer,
|
||||
borderColor: token.colorBorder
|
||||
}}
|
||||
renderItem={item => (
|
||||
<List.Item
|
||||
key={item.id}
|
||||
onClick={() => handleCustomTemplateClick(item.id || '')}
|
||||
style={{
|
||||
backgroundColor: item.selected ? token.colorPrimaryBg : token.colorBgContainer,
|
||||
borderColor: item.selected ? token.colorPrimary : token.colorBorder,
|
||||
color: token.colorText,
|
||||
cursor: 'pointer'
|
||||
}}
|
||||
className={
|
||||
item.selected && themeMode === 'dark'
|
||||
? 'selected-custom-template-dark'
|
||||
@@ -324,7 +371,7 @@ const TemplateDrawer: React.FC<TemplateDrawerProps> = ({
|
||||
: ''
|
||||
}
|
||||
>
|
||||
{item.name}
|
||||
<span style={{ color: token.colorText }}>{item.name}</span>
|
||||
</List.Item>
|
||||
)}
|
||||
/>
|
||||
@@ -355,18 +402,31 @@ const TemplateDrawer: React.FC<TemplateDrawerProps> = ({
|
||||
};
|
||||
|
||||
return (
|
||||
<div style={{ height: '100vh', overflow: 'hidden' }}>
|
||||
<div style={{
|
||||
height: '100vh',
|
||||
overflow: 'hidden',
|
||||
backgroundColor: token.colorBgLayout
|
||||
}}>
|
||||
<div
|
||||
style={{
|
||||
position: 'sticky',
|
||||
top: 0,
|
||||
zIndex: 100,
|
||||
backgroundColor: themeMode === 'dark' ? '' : '#fff',
|
||||
backgroundColor: token.colorBgContainer,
|
||||
overflow: 'hidden',
|
||||
borderBottom: `1px solid ${token.colorBorder}`
|
||||
}}
|
||||
>
|
||||
{showBothTabs ? (
|
||||
<Tabs type="card" items={tabs} onChange={handleTabChange} destroyInactiveTabPane />
|
||||
<Tabs
|
||||
type="card"
|
||||
items={tabs}
|
||||
onChange={handleTabChange}
|
||||
destroyInactiveTabPane
|
||||
style={{
|
||||
backgroundColor: token.colorBgContainer
|
||||
}}
|
||||
/>
|
||||
) : (
|
||||
menuContent
|
||||
)}
|
||||
|
||||
@@ -8,15 +8,7 @@ import ImprovedTaskFilters from '../../task-management/improved-task-filters';
|
||||
import Card from 'antd/es/card';
|
||||
import Spin from 'antd/es/spin';
|
||||
import Empty from 'antd/es/empty';
|
||||
import {
|
||||
reorderGroups,
|
||||
reorderEnhancedKanbanGroups,
|
||||
reorderTasks,
|
||||
reorderEnhancedKanbanTasks,
|
||||
fetchEnhancedKanbanLabels,
|
||||
fetchEnhancedKanbanGroups,
|
||||
fetchEnhancedKanbanTaskAssignees,
|
||||
} from '@/features/enhanced-kanban/enhanced-kanban.slice';
|
||||
import { reorderGroups, reorderEnhancedKanbanGroups, reorderTasks, reorderEnhancedKanbanTasks, fetchEnhancedKanbanLabels, fetchEnhancedKanbanGroups, fetchEnhancedKanbanTaskAssignees, updateEnhancedKanbanTaskPriority } from '@/features/enhanced-kanban/enhanced-kanban.slice';
|
||||
import { fetchStatusesCategories } from '@/features/taskAttributes/taskStatusSlice';
|
||||
import { useAppSelector } from '@/hooks/useAppSelector';
|
||||
import KanbanGroup from './KanbanGroup';
|
||||
@@ -29,17 +21,26 @@ import alertService from '@/services/alerts/alertService';
|
||||
import logger from '@/utils/errorLogger';
|
||||
import { checkTaskDependencyStatus } from '@/utils/check-task-dependency-status';
|
||||
import { useTaskSocketHandlers } from '@/hooks/useTaskSocketHandlers';
|
||||
import { phasesApiService } from '@/api/taskAttributes/phases/phases.api.service';
|
||||
import { ITaskListGroup } from '@/types/tasks/taskList.types';
|
||||
import { fetchPhasesByProjectId, updatePhaseListOrder } from '@/features/projects/singleProject/phase/phases.slice';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { ITaskListPriorityChangeResponse } from '@/types/tasks/task-list-priority.types';
|
||||
|
||||
const EnhancedKanbanBoardNativeDnD: React.FC<{ projectId: string }> = ({ projectId }) => {
|
||||
const { t } = useTranslation('kanban-board');
|
||||
const dispatch = useDispatch();
|
||||
const authService = useAuthService();
|
||||
const { socket } = useSocket();
|
||||
const project = useAppSelector((state: RootState) => state.projectReducer.project);
|
||||
const groupBy = useSelector((state: RootState) => state.enhancedKanbanReducer.groupBy);
|
||||
const teamId = authService.getCurrentSession()?.team_id;
|
||||
const { taskGroups, loadingGroups, error } = useSelector(
|
||||
(state: RootState) => state.enhancedKanbanReducer
|
||||
);
|
||||
const {
|
||||
taskGroups,
|
||||
loadingGroups,
|
||||
error,
|
||||
} = useSelector((state: RootState) => state.enhancedKanbanReducer);
|
||||
const { phaseList, loadingPhases } = useAppSelector(state => state.phaseReducer);
|
||||
const [draggedGroupId, setDraggedGroupId] = useState<string | null>(null);
|
||||
const [draggedTaskId, setDraggedTaskId] = useState<string | null>(null);
|
||||
const [draggedTaskGroupId, setDraggedTaskGroupId] = useState<string | null>(null);
|
||||
@@ -64,6 +65,9 @@ const EnhancedKanbanBoardNativeDnD: React.FC<{ projectId: string }> = ({ project
|
||||
if (!statusCategories.length) {
|
||||
dispatch(fetchStatusesCategories() as any);
|
||||
}
|
||||
if (groupBy === 'phase' && !phaseList.length) {
|
||||
dispatch(fetchPhasesByProjectId(projectId) as any);
|
||||
}
|
||||
}, [dispatch, projectId]);
|
||||
// Reset drag state if taskGroups changes (e.g., real-time update)
|
||||
useEffect(() => {
|
||||
@@ -80,6 +84,9 @@ const EnhancedKanbanBoardNativeDnD: React.FC<{ projectId: string }> = ({ project
|
||||
setDraggedGroupId(groupId);
|
||||
setDragType('group');
|
||||
e.dataTransfer.effectAllowed = 'move';
|
||||
try {
|
||||
e.dataTransfer.setData('text/plain', groupId);
|
||||
} catch {}
|
||||
};
|
||||
const handleGroupDragOver = (e: React.DragEvent) => {
|
||||
if (dragType !== 'group') return;
|
||||
@@ -97,34 +104,44 @@ const EnhancedKanbanBoardNativeDnD: React.FC<{ projectId: string }> = ({ project
|
||||
const [moved] = reorderedGroups.splice(fromIdx, 1);
|
||||
reorderedGroups.splice(toIdx, 0, moved);
|
||||
dispatch(reorderGroups({ fromIndex: fromIdx, toIndex: toIdx, reorderedGroups }));
|
||||
dispatch(
|
||||
reorderEnhancedKanbanGroups({ fromIndex: fromIdx, toIndex: toIdx, reorderedGroups }) as any
|
||||
);
|
||||
|
||||
dispatch(reorderEnhancedKanbanGroups({ fromIndex: fromIdx, toIndex: toIdx, reorderedGroups }) as any);
|
||||
// API call for group order
|
||||
try {
|
||||
const columnOrder = reorderedGroups.map(group => group.id);
|
||||
const requestBody = { status_order: columnOrder };
|
||||
const response = await statusApiService.updateStatusOrder(requestBody, projectId);
|
||||
if (!response.done) {
|
||||
// Revert the change if API call fails
|
||||
const revertedGroups = [...reorderedGroups];
|
||||
const [movedBackGroup] = revertedGroups.splice(toIdx, 1);
|
||||
revertedGroups.splice(fromIdx, 0, movedBackGroup);
|
||||
dispatch(
|
||||
reorderGroups({ fromIndex: toIdx, toIndex: fromIdx, reorderedGroups: revertedGroups })
|
||||
);
|
||||
alertService.error('Failed to update column order', 'Please try again');
|
||||
if (groupBy === 'status') {
|
||||
const columnOrder = reorderedGroups.map(group => group.id);
|
||||
const requestBody = { status_order: columnOrder };
|
||||
const response = await statusApiService.updateStatusOrder(requestBody, projectId);
|
||||
if (!response.done) {
|
||||
// Revert the change if API call fails
|
||||
const revertedGroups = [...reorderedGroups];
|
||||
const [movedBackGroup] = revertedGroups.splice(toIdx, 1);
|
||||
revertedGroups.splice(fromIdx, 0, movedBackGroup);
|
||||
dispatch(reorderGroups({ fromIndex: toIdx, toIndex: fromIdx, reorderedGroups: revertedGroups }));
|
||||
alertService.error(t('failedToUpdateColumnOrder'), t('pleaseTryAgain'));
|
||||
}
|
||||
} else if (groupBy === 'phase') {
|
||||
const newPhaseList = [...phaseList];
|
||||
const [movedItem] = newPhaseList.splice(fromIdx, 1);
|
||||
newPhaseList.splice(toIdx, 0, movedItem);
|
||||
dispatch(updatePhaseListOrder(newPhaseList));
|
||||
const requestBody = {
|
||||
from_index: fromIdx,
|
||||
to_index: toIdx,
|
||||
phases: newPhaseList,
|
||||
project_id: projectId,
|
||||
};
|
||||
const response = await phasesApiService.updatePhaseOrder(projectId, requestBody);
|
||||
if (!response.done) {
|
||||
alertService.error(t('failedToUpdatePhaseOrder'), t('pleaseTryAgain'));
|
||||
}
|
||||
}
|
||||
} catch (error) {
|
||||
// Revert the change if API call fails
|
||||
const revertedGroups = [...reorderedGroups];
|
||||
const [movedBackGroup] = revertedGroups.splice(toIdx, 1);
|
||||
revertedGroups.splice(fromIdx, 0, movedBackGroup);
|
||||
dispatch(
|
||||
reorderGroups({ fromIndex: toIdx, toIndex: fromIdx, reorderedGroups: revertedGroups })
|
||||
);
|
||||
alertService.error('Failed to update column order', 'Please try again');
|
||||
dispatch(reorderGroups({ fromIndex: toIdx, toIndex: fromIdx, reorderedGroups: revertedGroups }));
|
||||
alertService.error(t('failedToUpdateColumnOrder'), t('pleaseTryAgain'));
|
||||
logger.error('Failed to update column order', error);
|
||||
}
|
||||
|
||||
@@ -132,12 +149,47 @@ const EnhancedKanbanBoardNativeDnD: React.FC<{ projectId: string }> = ({ project
|
||||
setDragType(null);
|
||||
};
|
||||
|
||||
// Utility to recalculate all task orders for all groups
|
||||
function getAllTaskUpdates(allGroups: ITaskListGroup[], groupBy: string) {
|
||||
const taskUpdates: Array<{
|
||||
task_id: string | undefined;
|
||||
sort_order: number;
|
||||
status_id?: string;
|
||||
priority_id?: string;
|
||||
phase_id?: string;
|
||||
}> = [];
|
||||
let currentSortOrder = 0;
|
||||
for (const group of allGroups) {
|
||||
for (const task of group.tasks) {
|
||||
const update: {
|
||||
task_id: string | undefined;
|
||||
sort_order: number;
|
||||
status_id?: string;
|
||||
priority_id?: string;
|
||||
phase_id?: string;
|
||||
} = {
|
||||
task_id: task.id,
|
||||
sort_order: currentSortOrder,
|
||||
};
|
||||
if (groupBy === 'status') update.status_id = group.id;
|
||||
else if (groupBy === 'priority') update.priority_id = group.id;
|
||||
else if (groupBy === 'phase' && group.name !== 'Unmapped') update.phase_id = group.id;
|
||||
taskUpdates.push(update);
|
||||
currentSortOrder++;
|
||||
}
|
||||
}
|
||||
return taskUpdates;
|
||||
}
|
||||
|
||||
// Task drag handlers
|
||||
const handleTaskDragStart = (e: React.DragEvent, taskId: string, groupId: string) => {
|
||||
setDraggedTaskId(taskId);
|
||||
setDraggedTaskGroupId(groupId);
|
||||
setDragType('task');
|
||||
e.dataTransfer.effectAllowed = 'move';
|
||||
try {
|
||||
e.dataTransfer.setData('text/plain', taskId);
|
||||
} catch {}
|
||||
};
|
||||
const handleTaskDragOver = (e: React.DragEvent, groupId: string, taskIdx: number | null) => {
|
||||
if (dragType !== 'task') return;
|
||||
@@ -176,8 +228,8 @@ const EnhancedKanbanBoardNativeDnD: React.FC<{ projectId: string }> = ({ project
|
||||
const canContinue = await checkTaskDependencyStatus(movedTask.id, targetGroupId);
|
||||
if (!canContinue) {
|
||||
alertService.error(
|
||||
'Task is not completed',
|
||||
'Please complete the task dependencies before proceeding'
|
||||
t('taskNotCompleted'),
|
||||
t('completeTaskDependencies')
|
||||
);
|
||||
return;
|
||||
}
|
||||
@@ -187,6 +239,7 @@ const EnhancedKanbanBoardNativeDnD: React.FC<{ projectId: string }> = ({ project
|
||||
let insertIdx = hoveredTaskIdx;
|
||||
|
||||
// Handle same group reordering
|
||||
let newTaskGroups = [...taskGroups];
|
||||
if (sourceGroup.id === targetGroup.id) {
|
||||
// Create a single updated array for the same group
|
||||
const updatedTasks = [...sourceGroup.tasks];
|
||||
@@ -201,29 +254,26 @@ const EnhancedKanbanBoardNativeDnD: React.FC<{ projectId: string }> = ({ project
|
||||
if (insertIdx > updatedTasks.length) insertIdx = updatedTasks.length;
|
||||
|
||||
updatedTasks.splice(insertIdx, 0, movedTask); // Insert at new position
|
||||
|
||||
dispatch(
|
||||
reorderTasks({
|
||||
activeGroupId: sourceGroup.id,
|
||||
overGroupId: targetGroup.id,
|
||||
fromIndex: taskIdx,
|
||||
toIndex: insertIdx,
|
||||
task: movedTask,
|
||||
updatedSourceTasks: updatedTasks,
|
||||
updatedTargetTasks: updatedTasks,
|
||||
})
|
||||
);
|
||||
dispatch(
|
||||
reorderEnhancedKanbanTasks({
|
||||
activeGroupId: sourceGroup.id,
|
||||
overGroupId: targetGroup.id,
|
||||
fromIndex: taskIdx,
|
||||
toIndex: insertIdx,
|
||||
task: movedTask,
|
||||
updatedSourceTasks: updatedTasks,
|
||||
updatedTargetTasks: updatedTasks,
|
||||
}) as any
|
||||
);
|
||||
dispatch(reorderTasks({
|
||||
activeGroupId: sourceGroup.id,
|
||||
overGroupId: targetGroup.id,
|
||||
fromIndex: taskIdx,
|
||||
toIndex: insertIdx,
|
||||
task: movedTask,
|
||||
updatedSourceTasks: updatedTasks,
|
||||
updatedTargetTasks: updatedTasks,
|
||||
}));
|
||||
dispatch(reorderEnhancedKanbanTasks({
|
||||
activeGroupId: sourceGroup.id,
|
||||
overGroupId: targetGroup.id,
|
||||
fromIndex: taskIdx,
|
||||
toIndex: insertIdx,
|
||||
task: movedTask,
|
||||
updatedSourceTasks: updatedTasks,
|
||||
updatedTargetTasks: updatedTasks,
|
||||
}) as any);
|
||||
// Update newTaskGroups for socket emit
|
||||
newTaskGroups = newTaskGroups.map(g => g.id === sourceGroup.id ? { ...g, tasks: updatedTasks } : g);
|
||||
} else {
|
||||
// Handle cross-group reordering
|
||||
const updatedSourceTasks = [...sourceGroup.tasks];
|
||||
@@ -234,56 +284,51 @@ const EnhancedKanbanBoardNativeDnD: React.FC<{ projectId: string }> = ({ project
|
||||
if (insertIdx > updatedTargetTasks.length) insertIdx = updatedTargetTasks.length;
|
||||
updatedTargetTasks.splice(insertIdx, 0, movedTask);
|
||||
|
||||
dispatch(
|
||||
reorderTasks({
|
||||
activeGroupId: sourceGroup.id,
|
||||
overGroupId: targetGroup.id,
|
||||
fromIndex: taskIdx,
|
||||
toIndex: insertIdx,
|
||||
task: movedTask,
|
||||
updatedSourceTasks,
|
||||
updatedTargetTasks,
|
||||
})
|
||||
);
|
||||
dispatch(
|
||||
reorderEnhancedKanbanTasks({
|
||||
activeGroupId: sourceGroup.id,
|
||||
overGroupId: targetGroup.id,
|
||||
fromIndex: taskIdx,
|
||||
toIndex: insertIdx,
|
||||
task: movedTask,
|
||||
updatedSourceTasks,
|
||||
updatedTargetTasks,
|
||||
}) as any
|
||||
);
|
||||
dispatch(reorderTasks({
|
||||
activeGroupId: sourceGroup.id,
|
||||
overGroupId: targetGroup.id,
|
||||
fromIndex: taskIdx,
|
||||
toIndex: insertIdx,
|
||||
task: movedTask,
|
||||
updatedSourceTasks,
|
||||
updatedTargetTasks,
|
||||
}));
|
||||
dispatch(reorderEnhancedKanbanTasks({
|
||||
activeGroupId: sourceGroup.id,
|
||||
overGroupId: targetGroup.id,
|
||||
fromIndex: taskIdx,
|
||||
toIndex: insertIdx,
|
||||
task: movedTask,
|
||||
updatedSourceTasks,
|
||||
updatedTargetTasks,
|
||||
}) as any);
|
||||
// Update newTaskGroups for socket emit
|
||||
newTaskGroups = newTaskGroups.map(g => {
|
||||
if (g.id === sourceGroup.id) return { ...g, tasks: updatedSourceTasks };
|
||||
if (g.id === targetGroup.id) return { ...g, tasks: updatedTargetTasks };
|
||||
return g;
|
||||
});
|
||||
}
|
||||
|
||||
// Socket emit for task order
|
||||
// Socket emit for full task order
|
||||
if (socket && projectId && teamId && movedTask) {
|
||||
let toSortOrder = -1;
|
||||
let toLastIndex = false;
|
||||
if (insertIdx === targetGroup.tasks.length) {
|
||||
toSortOrder = -1;
|
||||
toLastIndex = true;
|
||||
} else if (targetGroup.tasks[insertIdx]) {
|
||||
const sortOrder = targetGroup.tasks[insertIdx].sort_order;
|
||||
toSortOrder = typeof sortOrder === 'number' ? sortOrder : 0;
|
||||
toLastIndex = false;
|
||||
} else if (targetGroup.tasks.length > 0) {
|
||||
const lastSortOrder = targetGroup.tasks[targetGroup.tasks.length - 1].sort_order;
|
||||
toSortOrder = typeof lastSortOrder === 'number' ? lastSortOrder : 0;
|
||||
toLastIndex = false;
|
||||
}
|
||||
const taskUpdates = getAllTaskUpdates(newTaskGroups, groupBy);
|
||||
socket.emit(SocketEvents.TASK_SORT_ORDER_CHANGE.toString(), {
|
||||
project_id: projectId,
|
||||
from_index: movedTask.sort_order ?? 0,
|
||||
to_index: toSortOrder,
|
||||
to_last_index: toLastIndex,
|
||||
group_by: groupBy || 'status',
|
||||
task_updates: taskUpdates,
|
||||
from_group: sourceGroup.id,
|
||||
to_group: targetGroup.id,
|
||||
group_by: groupBy || 'status',
|
||||
task: movedTask,
|
||||
team_id: teamId,
|
||||
from_index: taskIdx,
|
||||
to_index: insertIdx,
|
||||
to_last_index: insertIdx === (targetGroup.id === sourceGroup.id ? (newTaskGroups.find(g => g.id === targetGroup.id)?.tasks.length || 0) - 1 : targetGroup.tasks.length),
|
||||
task: {
|
||||
id: movedTask.id,
|
||||
project_id: movedTask.project_id || projectId,
|
||||
status: movedTask.status || '',
|
||||
priority: movedTask.priority || '',
|
||||
}
|
||||
});
|
||||
|
||||
// Emit progress update if status changed
|
||||
@@ -298,6 +343,22 @@ const EnhancedKanbanBoardNativeDnD: React.FC<{ projectId: string }> = ({ project
|
||||
})
|
||||
);
|
||||
}
|
||||
if (groupBy === 'priority' && movedTask.id) {
|
||||
socket?.emit(
|
||||
SocketEvents.TASK_PRIORITY_CHANGE.toString(),
|
||||
JSON.stringify({
|
||||
task_id: movedTask.id,
|
||||
priority_id: targetGroupId,
|
||||
team_id: teamId,
|
||||
})
|
||||
);
|
||||
socket?.once(
|
||||
SocketEvents.TASK_PRIORITY_CHANGE.toString(),
|
||||
(data: ITaskListPriorityChangeResponse) => {
|
||||
dispatch(updateEnhancedKanbanTaskPriority(data));
|
||||
}
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
setDraggedTaskId(null);
|
||||
@@ -318,7 +379,7 @@ const EnhancedKanbanBoardNativeDnD: React.FC<{ projectId: string }> = ({ project
|
||||
if (error) {
|
||||
return (
|
||||
<Card>
|
||||
<Empty description={`Error loading tasks: ${error}`} image={Empty.PRESENTED_IMAGE_SIMPLE} />
|
||||
<Empty description={`${t('errorLoadingTasks')}: ${error}`} image={Empty.PRESENTED_IMAGE_SIMPLE} />
|
||||
</Card>
|
||||
);
|
||||
}
|
||||
@@ -326,33 +387,21 @@ const EnhancedKanbanBoardNativeDnD: React.FC<{ projectId: string }> = ({ project
|
||||
return (
|
||||
<>
|
||||
<div className="mb-4">
|
||||
<React.Suspense fallback={<div>Loading filters...</div>}>
|
||||
<React.Suspense fallback={<div>{t('loadingFilters')}</div>}>
|
||||
<ImprovedTaskFilters position="board" />
|
||||
</React.Suspense>
|
||||
</div>
|
||||
<div className="enhanced-kanban-board">
|
||||
{loadingGroups ? (
|
||||
<div className="flex flex-row gap-2 h-[600px]">
|
||||
<div
|
||||
className="rounded bg-gray-200 dark:bg-gray-700 animate-pulse w-1/4"
|
||||
style={{ height: '60%' }}
|
||||
/>
|
||||
<div
|
||||
className="rounded bg-gray-200 dark:bg-gray-700 animate-pulse w-1/4"
|
||||
style={{ height: '100%' }}
|
||||
/>
|
||||
<div
|
||||
className="rounded bg-gray-200 dark:bg-gray-700 animate-pulse w-1/4"
|
||||
style={{ height: '80%' }}
|
||||
/>
|
||||
<div
|
||||
className="rounded bg-gray-200 dark:bg-gray-700 animate-pulse w-1/4"
|
||||
style={{ height: '40%' }}
|
||||
/>
|
||||
<div className="rounded bg-gray-200 dark:bg-gray-700 animate-pulse w-1/4" style={{ height: '60%' }} />
|
||||
<div className="rounded bg-gray-200 dark:bg-gray-700 animate-pulse w-1/4" style={{ height: '100%' }} />
|
||||
<div className="rounded bg-gray-200 dark:bg-gray-700 animate-pulse w-1/4" style={{ height: '80%' }} />
|
||||
<div className="rounded bg-gray-200 dark:bg-gray-700 animate-pulse w-1/4" style={{ height: '40%' }} />
|
||||
</div>
|
||||
) : taskGroups.length === 0 ? (
|
||||
<Card>
|
||||
<Empty description="No tasks found" image={Empty.PRESENTED_IMAGE_SIMPLE} />
|
||||
<Empty description={t('noTasksFound')} image={Empty.PRESENTED_IMAGE_SIMPLE} />
|
||||
</Card>
|
||||
) : (
|
||||
<div className="kanban-groups-container">
|
||||
|
||||
@@ -25,6 +25,7 @@ import {
|
||||
IGroupBy,
|
||||
} from '@/features/enhanced-kanban/enhanced-kanban.slice';
|
||||
import { createPortal } from 'react-dom';
|
||||
import { Modal } from 'antd';
|
||||
|
||||
// Simple Portal component
|
||||
const Portal: React.FC<{ children: React.ReactNode }> = ({ children }) => {
|
||||
@@ -219,8 +220,43 @@ const KanbanGroup: React.FC<KanbanGroupProps> = memo(
|
||||
};
|
||||
|
||||
const handleDelete = () => {
|
||||
setShowDeleteConfirm(true);
|
||||
setShowDropdown(false);
|
||||
if (groupBy === IGroupBy.STATUS) {
|
||||
Modal.confirm({
|
||||
title: t('deleteStatusTitle'),
|
||||
content: t('deleteStatusContent'),
|
||||
okText: t('deleteTaskConfirm'),
|
||||
okType: 'danger',
|
||||
cancelText: t('deleteTaskCancel'),
|
||||
centered: true,
|
||||
onOk: async () => {
|
||||
await handleDeleteSection();
|
||||
},
|
||||
});
|
||||
} else if (groupBy === IGroupBy.PHASE) {
|
||||
Modal.confirm({
|
||||
title: t('deletePhaseTitle'),
|
||||
content: t('deletePhaseContent'),
|
||||
okText: t('deleteTaskConfirm'),
|
||||
okType: 'danger',
|
||||
cancelText: t('deleteTaskCancel'),
|
||||
centered: true,
|
||||
onOk: async () => {
|
||||
await handleDeleteSection();
|
||||
},
|
||||
});
|
||||
} else {
|
||||
Modal.confirm({
|
||||
title: t('deleteConfirmationTitle'),
|
||||
okText: t('deleteTaskConfirm'),
|
||||
okType: 'danger',
|
||||
cancelText: t('deleteTaskCancel'),
|
||||
centered: true,
|
||||
onOk: async () => {
|
||||
await handleDeleteSection();
|
||||
},
|
||||
});
|
||||
}
|
||||
setShowDropdown(false);
|
||||
};
|
||||
|
||||
// Close dropdown when clicking outside
|
||||
@@ -290,51 +326,180 @@ const KanbanGroup: React.FC<KanbanGroupProps> = memo(
|
||||
if ((isProjectManager || isOwnerOrAdmin) && group.name !== t('unmapped'))
|
||||
setIsEditable(true);
|
||||
}}
|
||||
onMouseDown={e => {
|
||||
e.stopPropagation();
|
||||
}}
|
||||
>
|
||||
{isLoading && (
|
||||
<div className="w-4 h-4 border-2 border-gray-300 border-t-blue-600 rounded-full animate-spin"></div>
|
||||
)}
|
||||
{isEditable ? (
|
||||
<input
|
||||
ref={inputRef}
|
||||
value={name}
|
||||
className={`bg-transparent border-none outline-none text-sm font-semibold capitalize min-w-[185px] ${
|
||||
themeMode === 'dark' ? 'text-gray-800' : 'text-gray-900'
|
||||
}`}
|
||||
onChange={handleChange}
|
||||
onBlur={handleBlur}
|
||||
onKeyDown={handlePressEnter}
|
||||
onMouseDown={e => {
|
||||
e.stopPropagation();
|
||||
onDragOver={e => { e.preventDefault(); onTaskDragOver(e, group.id, null); }}
|
||||
onDrop={e => { e.preventDefault(); onTaskDrop(e, group.id, null); }}
|
||||
/>
|
||||
|
||||
{/* Content layer - z-index 1 */}
|
||||
<div style={{ position: 'relative', zIndex: 1 }}>
|
||||
<div
|
||||
className="enhanced-kanban-group-header"
|
||||
style={{
|
||||
backgroundColor: headerBackgroundColor,
|
||||
}}
|
||||
onClick={e => {
|
||||
e.stopPropagation();
|
||||
}}
|
||||
/>
|
||||
) : (
|
||||
<div
|
||||
className={`min-w-[185px] text-sm font-semibold capitalize truncate ${
|
||||
themeMode === 'dark' ? 'text-gray-800' : 'text-gray-900'
|
||||
}`}
|
||||
title={isEllipsisActive ? name : undefined}
|
||||
onMouseDown={e => {
|
||||
e.stopPropagation();
|
||||
e.preventDefault();
|
||||
}}
|
||||
onMouseUp={e => {
|
||||
e.stopPropagation();
|
||||
}}
|
||||
onClick={e => {
|
||||
e.stopPropagation();
|
||||
}}
|
||||
>
|
||||
{name} ({group.tasks.length})
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
draggable
|
||||
onDragStart={e => onGroupDragStart(e, group.id)}
|
||||
onDragOver={onGroupDragOver}
|
||||
onDrop={e => onGroupDrop(e, group.id)}
|
||||
onDragEnd={onDragEnd}
|
||||
>
|
||||
<div
|
||||
className="flex items-center justify-between w-full font-semibold rounded-md"
|
||||
onMouseEnter={() => setIsHover(true)}
|
||||
onMouseLeave={() => setIsHover(false)}
|
||||
>
|
||||
<div
|
||||
className="flex items-center gap-2 cursor-pointer"
|
||||
onClick={e => {
|
||||
e.stopPropagation();
|
||||
if ((isProjectManager || isOwnerOrAdmin) && group.name !== t('unmapped'))
|
||||
setIsEditable(true);
|
||||
}}
|
||||
onMouseDown={e => {
|
||||
e.stopPropagation();
|
||||
}}
|
||||
>
|
||||
{isLoading && (
|
||||
<div className="w-4 h-4 border-2 border-gray-300 border-t-blue-600 rounded-full animate-spin"></div>
|
||||
)}
|
||||
{isEditable ? (
|
||||
<input
|
||||
ref={inputRef}
|
||||
value={name}
|
||||
className={`bg-transparent border-none outline-none text-sm font-semibold capitalize min-w-[185px] ${themeMode === 'dark' ? 'text-gray-800' : 'text-gray-900'
|
||||
}`}
|
||||
onChange={handleChange}
|
||||
onBlur={handleBlur}
|
||||
onKeyDown={handlePressEnter}
|
||||
onMouseDown={e => {
|
||||
e.stopPropagation();
|
||||
}}
|
||||
onClick={e => {
|
||||
e.stopPropagation();
|
||||
}}
|
||||
/>
|
||||
) : (
|
||||
<div
|
||||
className={`min-w-[185px] text-sm font-semibold capitalize truncate ${themeMode === 'dark' ? 'text-gray-800' : 'text-gray-900'
|
||||
}`}
|
||||
title={isEllipsisActive ? name : undefined}
|
||||
onMouseDown={e => {
|
||||
e.stopPropagation();
|
||||
e.preventDefault();
|
||||
}}
|
||||
onMouseUp={e => {
|
||||
e.stopPropagation();
|
||||
}}
|
||||
onClick={e => {
|
||||
e.stopPropagation();
|
||||
}}
|
||||
>
|
||||
{name} ({group.tasks.length})
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
<div className="flex items-center gap-1">
|
||||
<button
|
||||
type="button"
|
||||
className="w-7 h-7 flex items-center justify-center rounded-full hover:bg-black/10 transition-colors"
|
||||
onClick={() => {
|
||||
setShowNewCardTop(true);
|
||||
setShowNewCardBottom(false);
|
||||
}}
|
||||
>
|
||||
<svg className="w-4 h-4 text-gray-800" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M12 4v16m8-8H4" />
|
||||
</svg>
|
||||
</button>
|
||||
|
||||
{(isOwnerOrAdmin || isProjectManager) && name !== t('unmapped') && (
|
||||
<div className="relative" ref={dropdownRef}>
|
||||
<button
|
||||
type="button"
|
||||
className="w-7 h-7 flex items-center justify-center rounded-full hover:bg-black/10 transition-colors"
|
||||
onClick={() => setShowDropdown(!showDropdown)}
|
||||
>
|
||||
<svg className="w-4 h-4 text-gray-800 rotate-90" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M12 5v.01M12 12v.01M12 19v.01M12 6a1 1 0 110-2 1 1 0 010 2zm0 7a1 1 0 110-2 1 1 0 010 2zm0 7a1 1 0 110-2 1 1 0 010 2z" />
|
||||
</svg>
|
||||
</button>
|
||||
|
||||
{showDropdown && (
|
||||
<div className="absolute right-0 top-full mt-1 w-48 bg-white dark:bg-gray-800 rounded-md shadow-lg border border-gray-200 dark:border-gray-700 z-50">
|
||||
<div className="py-1">
|
||||
<button
|
||||
type="button"
|
||||
className="w-full px-4 py-2 text-left text-sm hover:bg-gray-100 dark:hover:bg-gray-700 flex items-center gap-2"
|
||||
onClick={handleRename}
|
||||
>
|
||||
<svg className="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M11 5H6a2 2 0 00-2 2v11a2 2 0 002 2h11a2 2 0 002-2v-5m-1.414-9.414a2 2 0 112.828 2.828L11.828 15H9v-2.828l8.586-8.586z" />
|
||||
</svg>
|
||||
{t('rename')}
|
||||
</button>
|
||||
|
||||
{groupBy === IGroupBy.STATUS && statusCategories && (
|
||||
<div className="border-t border-gray-200 dark:border-gray-700">
|
||||
<div className="px-4 py-2 text-xs font-medium text-gray-500 uppercase tracking-wide">
|
||||
{t('changeCategory')}
|
||||
</div>
|
||||
{statusCategories.map(status => (
|
||||
<button
|
||||
key={status.id}
|
||||
type="button"
|
||||
className="w-full px-4 py-2 text-left text-sm hover:bg-gray-100 dark:hover:bg-gray-700 flex items-center gap-2"
|
||||
onClick={() => status.id && handleCategoryChange(status.id)}
|
||||
>
|
||||
<div
|
||||
className="w-3 h-3 rounded-full"
|
||||
style={{ backgroundColor: status.color_code }}
|
||||
></div>
|
||||
<span className={group.category_id === status.id ? 'font-bold' : ''}>
|
||||
{status.name}
|
||||
</span>
|
||||
</button>
|
||||
))}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{groupBy !== IGroupBy.PRIORITY && (
|
||||
<div className="border-t border-gray-200 dark:border-gray-700">
|
||||
<button
|
||||
type="button"
|
||||
className="w-full px-4 py-2 text-left text-sm hover:bg-gray-100 dark:hover:bg-gray-700 flex items-center gap-2 text-red-600 dark:text-red-400"
|
||||
onClick={e => {
|
||||
e.stopPropagation();
|
||||
handleDelete();
|
||||
}}
|
||||
>
|
||||
<svg className="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M19 7l-.867 12.142A2 2 0 0116.138 21H7.862a2 2 0 01-1.995-1.858L5 7m5 4v6m4-6v6m1-10V4a1 1 0 00-1-1h-4a1 1 0 00-1 1v3M4 7h16" />
|
||||
</svg>
|
||||
{t('delete')}
|
||||
</button>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Simple Delete Confirmation */}
|
||||
{/* Portal-based confirmation removed, now handled by Modal.confirm */}
|
||||
<div className="enhanced-kanban-group-tasks">
|
||||
{/* Create card at top */}
|
||||
{showNewCardTop && (
|
||||
<EnhancedKanbanCreateTaskCard
|
||||
sectionId={group.id}
|
||||
setShowNewCard={setShowNewCardTop}
|
||||
position="top"
|
||||
/>
|
||||
)}
|
||||
|
||||
<div className="flex items-center gap-1">
|
||||
<button
|
||||
|
||||
@@ -17,8 +17,12 @@ import { themeWiseColor } from '@/utils/themeWiseColor';
|
||||
import {
|
||||
toggleTaskExpansion,
|
||||
fetchBoardSubTasks,
|
||||
deleteTask as deleteKanbanTask,
|
||||
updateEnhancedKanbanSubtask,
|
||||
} from '@/features/enhanced-kanban/enhanced-kanban.slice';
|
||||
import TaskProgressCircle from './TaskProgressCircle';
|
||||
import { Button, Modal, DeleteOutlined } from '@/shared/antd-imports';
|
||||
import { tasksApiService } from '@/api/tasks/tasks.api.service';
|
||||
|
||||
// Simple Portal component
|
||||
const Portal: React.FC<{ children: React.ReactNode }> = ({ children }) => {
|
||||
@@ -76,6 +80,13 @@ const TaskCard: React.FC<TaskCardProps> = memo(
|
||||
const d = selectedDate || new Date();
|
||||
return new Date(d.getFullYear(), d.getMonth(), 1);
|
||||
});
|
||||
const [contextMenu, setContextMenu] = useState<{ visible: boolean; x: number; y: number }>({
|
||||
visible: false,
|
||||
x: 0,
|
||||
y: 0,
|
||||
});
|
||||
const contextMenuRef = useRef<HTMLDivElement>(null);
|
||||
const [selectedTask, setSelectedTask] = useState<IProjectTask | null>(null);
|
||||
|
||||
useEffect(() => {
|
||||
setSelectedDate(task.end_date ? new Date(task.end_date) : null);
|
||||
@@ -108,6 +119,21 @@ const TaskCard: React.FC<TaskCardProps> = memo(
|
||||
}
|
||||
}, [showDatePicker]);
|
||||
|
||||
// Hide context menu on click elsewhere
|
||||
useEffect(() => {
|
||||
const handleClick = (e: MouseEvent) => {
|
||||
if (contextMenuRef.current && !contextMenuRef.current.contains(e.target as Node)) {
|
||||
setContextMenu({ ...contextMenu, visible: false });
|
||||
}
|
||||
};
|
||||
if (contextMenu.visible) {
|
||||
document.addEventListener('mousedown', handleClick);
|
||||
}
|
||||
return () => {
|
||||
document.removeEventListener('mousedown', handleClick);
|
||||
};
|
||||
}, [contextMenu]);
|
||||
|
||||
const handleCardClick = useCallback(
|
||||
(e: React.MouseEvent, id: string) => {
|
||||
e.stopPropagation();
|
||||
@@ -195,6 +221,51 @@ const TaskCard: React.FC<TaskCardProps> = memo(
|
||||
[handleSubTaskExpand]
|
||||
);
|
||||
|
||||
// Delete logic (similar to task-drawer-header)
|
||||
const handleDeleteTask = async (task: IProjectTask | null) => {
|
||||
if (!task || !task.id) return;
|
||||
Modal.confirm({
|
||||
title: t('deleteTaskTitle'),
|
||||
content: t('deleteTaskContent'),
|
||||
okText: t('deleteTaskConfirm'),
|
||||
okType: 'danger',
|
||||
cancelText: t('deleteTaskCancel'),
|
||||
centered: true,
|
||||
onOk: async () => {
|
||||
if (!task.id) return;
|
||||
const res = await tasksApiService.deleteTask(task.id);
|
||||
if (res.done) {
|
||||
dispatch(setSelectedTaskId(null));
|
||||
if (task.is_sub_task) {
|
||||
dispatch(
|
||||
updateEnhancedKanbanSubtask({
|
||||
sectionId: '',
|
||||
subtask: {
|
||||
id: task.id,
|
||||
parent_task_id: task.parent_task_id || '',
|
||||
manual_progress: false,
|
||||
},
|
||||
mode: 'delete',
|
||||
})
|
||||
);
|
||||
} else {
|
||||
dispatch(deleteKanbanTask(task.id));
|
||||
}
|
||||
dispatch(setShowTaskDrawer(false));
|
||||
if (task.parent_task_id) {
|
||||
socket?.emit(SocketEvents.GET_TASK_PROGRESS.toString(), task.parent_task_id);
|
||||
}
|
||||
}
|
||||
setContextMenu({ visible: false, x: 0, y: 0 });
|
||||
setSelectedTask(null);
|
||||
},
|
||||
onCancel: () => {
|
||||
setContextMenu({ visible: false, x: 0, y: 0 });
|
||||
setSelectedTask(null);
|
||||
},
|
||||
});
|
||||
};
|
||||
|
||||
// Calendar rendering helpers
|
||||
const year = calendarMonth.getFullYear();
|
||||
const month = calendarMonth.getMonth();
|
||||
@@ -219,6 +290,39 @@ const TaskCard: React.FC<TaskCardProps> = memo(
|
||||
|
||||
return (
|
||||
<>
|
||||
{/* Context menu for delete */}
|
||||
{contextMenu.visible && (
|
||||
<div
|
||||
ref={contextMenuRef}
|
||||
style={{
|
||||
position: 'fixed',
|
||||
top: contextMenu.y,
|
||||
left: contextMenu.x,
|
||||
zIndex: 9999,
|
||||
background: themeMode === 'dark' ? '#23272f' : '#fff',
|
||||
borderRadius: 8,
|
||||
boxShadow: '0 2px 8px rgba(0,0,0,0.12)',
|
||||
padding: 0,
|
||||
minWidth: 120,
|
||||
transition: 'translateY(0)',
|
||||
}}
|
||||
>
|
||||
<Button
|
||||
type="text"
|
||||
icon={<DeleteOutlined style={{ color: '#ef4444', fontSize: 16 }} />}
|
||||
style={{
|
||||
color: '#ef4444',
|
||||
width: '100%',
|
||||
textAlign: 'left',
|
||||
padding: '8px 16px',
|
||||
fontWeight: 500,
|
||||
}}
|
||||
onClick={() => handleDeleteTask(selectedTask || null)}
|
||||
>
|
||||
{t('delete')}
|
||||
</Button>
|
||||
</div>
|
||||
)}
|
||||
<div
|
||||
className="enhanced-kanban-task-card"
|
||||
style={{ background, color, display: 'block', position: 'relative' }}
|
||||
@@ -241,6 +345,11 @@ const TaskCard: React.FC<TaskCardProps> = memo(
|
||||
onDrop={e => onTaskDrop(e, groupId, idx)}
|
||||
onDragEnd={onDragEnd} // <-- add this
|
||||
onClick={e => handleCardClick(e, task.id!)}
|
||||
onContextMenu={e => {
|
||||
e.preventDefault();
|
||||
setContextMenu({ visible: true, x: e.clientX, y: e.clientY });
|
||||
setSelectedTask(task);
|
||||
}}
|
||||
>
|
||||
<div className="task-content">
|
||||
<div className="task_labels" style={{ display: 'flex', gap: 4, marginBottom: 4 }}>
|
||||
@@ -528,6 +637,11 @@ const TaskCard: React.FC<TaskCardProps> = memo(
|
||||
key={sub.id}
|
||||
onClick={e => handleCardClick(e, sub.id!)}
|
||||
className="flex items-center gap-2 px-2 py-1 rounded hover:bg-gray-50 dark:hover:bg-gray-800"
|
||||
onContextMenu={e => {
|
||||
e.preventDefault();
|
||||
setContextMenu({ visible: true, x: e.clientX, y: e.clientY });
|
||||
setSelectedTask(sub);
|
||||
}}
|
||||
>
|
||||
{sub.priority_color || sub.priority_color_dark ? (
|
||||
<span
|
||||
|
||||
@@ -0,0 +1,406 @@
|
||||
import React, { useState } from 'react';
|
||||
import {
|
||||
Modal,
|
||||
Tabs,
|
||||
Progress,
|
||||
Tag,
|
||||
List,
|
||||
Avatar,
|
||||
Badge,
|
||||
Space,
|
||||
Button,
|
||||
Statistic,
|
||||
Row,
|
||||
Col,
|
||||
Timeline,
|
||||
Input,
|
||||
Form,
|
||||
DatePicker,
|
||||
Select
|
||||
} from 'antd';
|
||||
import {
|
||||
CalendarOutlined,
|
||||
TeamOutlined,
|
||||
CheckCircleOutlined,
|
||||
ClockCircleOutlined,
|
||||
FlagOutlined,
|
||||
ExclamationCircleOutlined,
|
||||
EditOutlined,
|
||||
SaveOutlined,
|
||||
CloseOutlined
|
||||
} from '@ant-design/icons';
|
||||
import { PhaseModalData, ProjectPhase, PhaseTask, PhaseMilestone } from '../../types/project-roadmap.types';
|
||||
import { useAppSelector } from '../../hooks/useAppSelector';
|
||||
import { themeWiseColor } from '../../utils/themeWiseColor';
|
||||
import dayjs from 'dayjs';
|
||||
|
||||
const { TabPane } = Tabs;
|
||||
const { TextArea } = Input;
|
||||
|
||||
interface PhaseModalProps {
|
||||
visible: boolean;
|
||||
phase: PhaseModalData | null;
|
||||
onClose: () => void;
|
||||
onUpdate?: (updates: Partial<ProjectPhase>) => void;
|
||||
}
|
||||
|
||||
const PhaseModal: React.FC<PhaseModalProps> = ({
|
||||
visible,
|
||||
phase,
|
||||
onClose,
|
||||
onUpdate,
|
||||
}) => {
|
||||
const [isEditing, setIsEditing] = useState(false);
|
||||
const [form] = Form.useForm();
|
||||
|
||||
// Theme support
|
||||
const themeMode = useAppSelector(state => state.themeReducer.mode);
|
||||
const isDarkMode = themeMode === 'dark';
|
||||
|
||||
if (!phase) return null;
|
||||
|
||||
const handleEdit = () => {
|
||||
setIsEditing(true);
|
||||
form.setFieldsValue({
|
||||
name: phase.name,
|
||||
description: phase.description,
|
||||
startDate: dayjs(phase.startDate),
|
||||
endDate: dayjs(phase.endDate),
|
||||
status: phase.status,
|
||||
});
|
||||
};
|
||||
|
||||
const handleSave = async () => {
|
||||
try {
|
||||
const values = await form.validateFields();
|
||||
const updates: Partial<ProjectPhase> = {
|
||||
name: values.name,
|
||||
description: values.description,
|
||||
startDate: values.startDate.toDate(),
|
||||
endDate: values.endDate.toDate(),
|
||||
status: values.status,
|
||||
};
|
||||
|
||||
onUpdate?.(updates);
|
||||
setIsEditing(false);
|
||||
} catch (error) {
|
||||
console.error('Validation failed:', error);
|
||||
}
|
||||
};
|
||||
|
||||
const handleCancel = () => {
|
||||
setIsEditing(false);
|
||||
form.resetFields();
|
||||
};
|
||||
|
||||
const getStatusColor = (status: string) => {
|
||||
switch (status) {
|
||||
case 'completed': return 'success';
|
||||
case 'in-progress': return 'processing';
|
||||
case 'on-hold': return 'warning';
|
||||
default: return 'default';
|
||||
}
|
||||
};
|
||||
|
||||
const getPriorityColor = (priority: string) => {
|
||||
switch (priority) {
|
||||
case 'high': return 'red';
|
||||
case 'medium': return 'orange';
|
||||
case 'low': return 'green';
|
||||
default: return 'default';
|
||||
}
|
||||
};
|
||||
|
||||
const getTaskStatusIcon = (status: string) => {
|
||||
switch (status) {
|
||||
case 'done': return <CheckCircleOutlined style={{ color: '#52c41a' }} />;
|
||||
case 'in-progress': return <ClockCircleOutlined style={{ color: '#1890ff' }} />;
|
||||
default: return <ExclamationCircleOutlined style={{ color: '#d9d9d9' }} />;
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<Modal
|
||||
title={
|
||||
<div className="flex justify-between items-center">
|
||||
<Space>
|
||||
<Badge status={getStatusColor(phase.status)} />
|
||||
{isEditing ? (
|
||||
<Form.Item name="name" className="mb-0">
|
||||
<Input className="dark:bg-gray-700 dark:border-gray-600 dark:text-gray-100" />
|
||||
</Form.Item>
|
||||
) : (
|
||||
<h4 className="text-lg font-semibold text-gray-900 dark:text-gray-100 mb-0">
|
||||
{phase.name}
|
||||
</h4>
|
||||
)}
|
||||
</Space>
|
||||
<Space>
|
||||
{isEditing ? (
|
||||
<>
|
||||
<Button
|
||||
type="primary"
|
||||
icon={<SaveOutlined />}
|
||||
onClick={handleSave}
|
||||
size="small"
|
||||
>
|
||||
Save
|
||||
</Button>
|
||||
<Button
|
||||
icon={<CloseOutlined />}
|
||||
onClick={handleCancel}
|
||||
size="small"
|
||||
className="dark:border-gray-600 dark:text-gray-300"
|
||||
>
|
||||
Cancel
|
||||
</Button>
|
||||
</>
|
||||
) : (
|
||||
<Button
|
||||
type="text"
|
||||
icon={<EditOutlined />}
|
||||
onClick={handleEdit}
|
||||
size="small"
|
||||
className="dark:text-gray-300 dark:hover:bg-gray-700"
|
||||
>
|
||||
Edit
|
||||
</Button>
|
||||
)}
|
||||
</Space>
|
||||
</div>
|
||||
}
|
||||
open={visible}
|
||||
onCancel={onClose}
|
||||
width={800}
|
||||
footer={null}
|
||||
className="dark:bg-gray-800"
|
||||
>
|
||||
<Form form={form} layout="vertical">
|
||||
<div className="mb-4">
|
||||
{isEditing ? (
|
||||
<Form.Item name="description" label={<span className="text-gray-700 dark:text-gray-300">Description</span>}>
|
||||
<TextArea
|
||||
rows={2}
|
||||
className="dark:bg-gray-700 dark:border-gray-600 dark:text-gray-100"
|
||||
/>
|
||||
</Form.Item>
|
||||
) : (
|
||||
<p className="text-gray-600 dark:text-gray-400">{phase.description}</p>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Phase Statistics */}
|
||||
<Row gutter={16} className="mb-6">
|
||||
<Col span={6}>
|
||||
<div className="bg-gray-50 dark:bg-gray-700 border border-gray-200 dark:border-gray-600 rounded-lg p-4">
|
||||
<Statistic
|
||||
title={<span className="text-gray-600 dark:text-gray-400">Progress</span>}
|
||||
value={phase.progress}
|
||||
suffix="%"
|
||||
valueStyle={{ color: themeWiseColor('#1890ff', '#40a9ff', themeMode) }}
|
||||
/>
|
||||
<Progress
|
||||
percent={phase.progress}
|
||||
showInfo={false}
|
||||
size="small"
|
||||
strokeColor={themeWiseColor('#1890ff', '#40a9ff', themeMode)}
|
||||
trailColor={themeWiseColor('#f0f0f0', '#4b5563', themeMode)}
|
||||
/>
|
||||
</div>
|
||||
</Col>
|
||||
<Col span={6}>
|
||||
<div className="bg-gray-50 dark:bg-gray-700 border border-gray-200 dark:border-gray-600 rounded-lg p-4">
|
||||
<Statistic
|
||||
title={<span className="text-gray-600 dark:text-gray-400">Tasks</span>}
|
||||
value={phase.completedTaskCount}
|
||||
suffix={`/ ${phase.taskCount}`}
|
||||
valueStyle={{ color: themeWiseColor('#52c41a', '#34d399', themeMode) }}
|
||||
/>
|
||||
</div>
|
||||
</Col>
|
||||
<Col span={6}>
|
||||
<div className="bg-gray-50 dark:bg-gray-700 border border-gray-200 dark:border-gray-600 rounded-lg p-4">
|
||||
<Statistic
|
||||
title={<span className="text-gray-600 dark:text-gray-400">Milestones</span>}
|
||||
value={phase.completedMilestoneCount}
|
||||
suffix={`/ ${phase.milestoneCount}`}
|
||||
valueStyle={{ color: themeWiseColor('#722ed1', '#9f7aea', themeMode) }}
|
||||
/>
|
||||
</div>
|
||||
</Col>
|
||||
<Col span={6}>
|
||||
<div className="bg-gray-50 dark:bg-gray-700 border border-gray-200 dark:border-gray-600 rounded-lg p-4">
|
||||
<Statistic
|
||||
title={<span className="text-gray-600 dark:text-gray-400">Team</span>}
|
||||
value={phase.teamMembers.length}
|
||||
suffix="members"
|
||||
valueStyle={{ color: themeWiseColor('#fa8c16', '#fbbf24', themeMode) }}
|
||||
/>
|
||||
</div>
|
||||
</Col>
|
||||
</Row>
|
||||
|
||||
{/* Timeline */}
|
||||
<Row gutter={16} className="mb-6">
|
||||
<Col span={12}>
|
||||
{isEditing ? (
|
||||
<Form.Item name="startDate" label={<span className="text-gray-700 dark:text-gray-300">Start Date</span>}>
|
||||
<DatePicker className="w-full dark:bg-gray-700 dark:border-gray-600" />
|
||||
</Form.Item>
|
||||
) : (
|
||||
<div className="bg-gray-50 dark:bg-gray-700 border border-gray-200 dark:border-gray-600 rounded-lg p-4">
|
||||
<div className="flex items-center gap-2">
|
||||
<CalendarOutlined className="text-gray-600 dark:text-gray-400" />
|
||||
<span className="font-medium text-gray-900 dark:text-gray-100">Start:</span>
|
||||
<span className="text-gray-600 dark:text-gray-400">{phase.startDate.toLocaleDateString()}</span>
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
</Col>
|
||||
<Col span={12}>
|
||||
{isEditing ? (
|
||||
<Form.Item name="endDate" label={<span className="text-gray-700 dark:text-gray-300">End Date</span>}>
|
||||
<DatePicker className="w-full dark:bg-gray-700 dark:border-gray-600" />
|
||||
</Form.Item>
|
||||
) : (
|
||||
<div className="bg-gray-50 dark:bg-gray-700 border border-gray-200 dark:border-gray-600 rounded-lg p-4">
|
||||
<div className="flex items-center gap-2">
|
||||
<CalendarOutlined className="text-gray-600 dark:text-gray-400" />
|
||||
<span className="font-medium text-gray-900 dark:text-gray-100">End:</span>
|
||||
<span className="text-gray-600 dark:text-gray-400">{phase.endDate.toLocaleDateString()}</span>
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
</Col>
|
||||
</Row>
|
||||
|
||||
{isEditing && (
|
||||
<Row gutter={16} className="mb-6">
|
||||
<Col span={12}>
|
||||
<Form.Item name="status" label={<span className="text-gray-700 dark:text-gray-300">Status</span>}>
|
||||
<Select className="dark:bg-gray-700 dark:border-gray-600">
|
||||
<Select.Option value="not-started">Not Started</Select.Option>
|
||||
<Select.Option value="in-progress">In Progress</Select.Option>
|
||||
<Select.Option value="completed">Completed</Select.Option>
|
||||
<Select.Option value="on-hold">On Hold</Select.Option>
|
||||
</Select>
|
||||
</Form.Item>
|
||||
</Col>
|
||||
</Row>
|
||||
)}
|
||||
|
||||
<Tabs
|
||||
defaultActiveKey="tasks"
|
||||
className="dark:bg-gray-800"
|
||||
tabBarStyle={{
|
||||
borderBottom: `1px solid ${themeWiseColor('#f0f0f0', '#4b5563', themeMode)}`
|
||||
}}
|
||||
>
|
||||
<TabPane tab={`Tasks (${phase.taskCount})`} key="tasks">
|
||||
<List
|
||||
dataSource={phase.tasks}
|
||||
renderItem={(task: PhaseTask) => (
|
||||
<List.Item>
|
||||
<List.Item.Meta
|
||||
avatar={getTaskStatusIcon(task.status)}
|
||||
title={
|
||||
<div className="flex justify-between items-center">
|
||||
<Text strong>{task.name}</Text>
|
||||
<Space>
|
||||
<Tag color={getPriorityColor(task.priority)}>
|
||||
{task.priority}
|
||||
</Tag>
|
||||
<Progress
|
||||
percent={task.progress}
|
||||
size="small"
|
||||
style={{ width: 100 }}
|
||||
/>
|
||||
</Space>
|
||||
</div>
|
||||
}
|
||||
description={
|
||||
<div>
|
||||
<Text type="secondary">{task.description}</Text>
|
||||
<div className="mt-2 flex justify-between items-center">
|
||||
<Space>
|
||||
<CalendarOutlined />
|
||||
<Text type="secondary">
|
||||
{task.startDate.toLocaleDateString()} - {task.endDate.toLocaleDateString()}
|
||||
</Text>
|
||||
</Space>
|
||||
{task.assigneeName && (
|
||||
<Space>
|
||||
<TeamOutlined />
|
||||
<Text type="secondary">{task.assigneeName}</Text>
|
||||
</Space>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
}
|
||||
/>
|
||||
</List.Item>
|
||||
)}
|
||||
/>
|
||||
</TabPane>
|
||||
|
||||
<TabPane tab={`Milestones (${phase.milestoneCount})`} key="milestones">
|
||||
<Timeline>
|
||||
{phase.milestones.map((milestone: PhaseMilestone) => (
|
||||
<Timeline.Item
|
||||
key={milestone.id}
|
||||
color={milestone.isCompleted ? 'green' : milestone.criticalPath ? 'red' : 'blue'}
|
||||
dot={milestone.isCompleted ? <CheckCircleOutlined /> : <FlagOutlined />}
|
||||
>
|
||||
<div className="flex justify-between items-start">
|
||||
<div>
|
||||
<Text strong>{milestone.name}</Text>
|
||||
{milestone.criticalPath && (
|
||||
<Tag color="red" className="ml-2">Critical Path</Tag>
|
||||
)}
|
||||
{milestone.description && (
|
||||
<div className="mt-1">
|
||||
<Text type="secondary">{milestone.description}</Text>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
<div className="text-right">
|
||||
<div>
|
||||
<CalendarOutlined />
|
||||
<span className="ml-1">{milestone.dueDate.toLocaleDateString()}</span>
|
||||
</div>
|
||||
<Badge
|
||||
status={milestone.isCompleted ? 'success' : 'processing'}
|
||||
text={milestone.isCompleted ? 'Completed' : 'Pending'}
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</Timeline.Item>
|
||||
))}
|
||||
</Timeline>
|
||||
</TabPane>
|
||||
|
||||
<TabPane tab={`Team (${phase.teamMembers.length})`} key="team">
|
||||
<List
|
||||
dataSource={phase.teamMembers}
|
||||
renderItem={(member: string) => (
|
||||
<List.Item>
|
||||
<List.Item.Meta
|
||||
avatar={<Avatar>{member.charAt(0).toUpperCase()}</Avatar>}
|
||||
title={member}
|
||||
description={
|
||||
<Text type="secondary">
|
||||
{phase.tasks.filter(task => task.assigneeName === member).length} tasks assigned
|
||||
</Text>
|
||||
}
|
||||
/>
|
||||
</List.Item>
|
||||
)}
|
||||
/>
|
||||
</TabPane>
|
||||
</Tabs>
|
||||
</Form>
|
||||
</Modal>
|
||||
);
|
||||
};
|
||||
|
||||
export default PhaseModal;
|
||||
@@ -0,0 +1,333 @@
|
||||
import React, { useState, useMemo } from 'react';
|
||||
import { Gantt, Task, ViewMode } from 'gantt-task-react';
|
||||
import { Button, Space, Badge } from 'antd';
|
||||
import { CalendarOutlined, TeamOutlined, CheckCircleOutlined } from '@ant-design/icons';
|
||||
import { ProjectPhase, ProjectRoadmap, GanttViewOptions, PhaseModalData } from '../../types/project-roadmap.types';
|
||||
import PhaseModal from './PhaseModal';
|
||||
import { useAppSelector } from '../../hooks/useAppSelector';
|
||||
import { themeWiseColor } from '../../utils/themeWiseColor';
|
||||
import 'gantt-task-react/dist/index.css';
|
||||
import './gantt-theme.css';
|
||||
|
||||
interface ProjectRoadmapGanttProps {
|
||||
roadmap: ProjectRoadmap;
|
||||
viewOptions?: Partial<GanttViewOptions>;
|
||||
onPhaseUpdate?: (phaseId: string, updates: Partial<ProjectPhase>) => void;
|
||||
onTaskUpdate?: (phaseId: string, taskId: string, updates: any) => void;
|
||||
}
|
||||
|
||||
const ProjectRoadmapGantt: React.FC<ProjectRoadmapGanttProps> = ({
|
||||
roadmap,
|
||||
viewOptions = {},
|
||||
onPhaseUpdate,
|
||||
onTaskUpdate,
|
||||
}) => {
|
||||
const [selectedPhase, setSelectedPhase] = useState<PhaseModalData | null>(null);
|
||||
const [isModalVisible, setIsModalVisible] = useState(false);
|
||||
const [viewMode, setViewMode] = useState<ViewMode>(ViewMode.Month);
|
||||
|
||||
// Theme support
|
||||
const themeMode = useAppSelector(state => state.themeReducer.mode);
|
||||
const isDarkMode = themeMode === 'dark';
|
||||
|
||||
const defaultViewOptions: GanttViewOptions = {
|
||||
viewMode: 'month',
|
||||
showTasks: true,
|
||||
showMilestones: true,
|
||||
groupByPhase: true,
|
||||
...viewOptions,
|
||||
};
|
||||
|
||||
// Theme-aware colors
|
||||
const ganttColors = useMemo(() => {
|
||||
return {
|
||||
background: themeWiseColor('#ffffff', '#1f2937', themeMode),
|
||||
surface: themeWiseColor('#f8f9fa', '#374151', themeMode),
|
||||
border: themeWiseColor('#e5e7eb', '#4b5563', themeMode),
|
||||
taskBar: themeWiseColor('#3b82f6', '#60a5fa', themeMode),
|
||||
taskBarHover: themeWiseColor('#2563eb', '#93c5fd', themeMode),
|
||||
progressBar: themeWiseColor('#10b981', '#34d399', themeMode),
|
||||
milestone: themeWiseColor('#f59e0b', '#fbbf24', themeMode),
|
||||
criticalPath: themeWiseColor('#ef4444', '#f87171', themeMode),
|
||||
text: {
|
||||
primary: themeWiseColor('#111827', '#f9fafb', themeMode),
|
||||
secondary: themeWiseColor('#6b7280', '#d1d5db', themeMode),
|
||||
},
|
||||
grid: themeWiseColor('#f3f4f6', '#4b5563', themeMode),
|
||||
today: themeWiseColor('rgba(59, 130, 246, 0.1)', 'rgba(96, 165, 250, 0.2)', themeMode),
|
||||
};
|
||||
}, [themeMode]);
|
||||
|
||||
// Convert phases to Gantt tasks
|
||||
const ganttTasks = useMemo(() => {
|
||||
const tasks: Task[] = [];
|
||||
|
||||
roadmap.phases.forEach((phase, phaseIndex) => {
|
||||
// Add phase as main task with theme-aware colors
|
||||
const phaseTask: Task = {
|
||||
id: phase.id,
|
||||
name: phase.name,
|
||||
start: phase.startDate,
|
||||
end: phase.endDate,
|
||||
progress: phase.progress,
|
||||
type: 'project',
|
||||
styles: {
|
||||
progressColor: themeWiseColor(phase.color, phase.color, themeMode),
|
||||
progressSelectedColor: themeWiseColor(phase.color, phase.color, themeMode),
|
||||
backgroundColor: themeWiseColor(`${phase.color}20`, `${phase.color}30`, themeMode),
|
||||
},
|
||||
};
|
||||
tasks.push(phaseTask);
|
||||
|
||||
// Add phase tasks if enabled
|
||||
if (defaultViewOptions.showTasks) {
|
||||
phase.tasks.forEach((task) => {
|
||||
const ganttTask: Task = {
|
||||
id: task.id,
|
||||
name: task.name,
|
||||
start: task.startDate,
|
||||
end: task.endDate,
|
||||
progress: task.progress,
|
||||
type: 'task',
|
||||
project: phase.id,
|
||||
dependencies: task.dependencies,
|
||||
styles: {
|
||||
progressColor: ganttColors.taskBar,
|
||||
progressSelectedColor: ganttColors.taskBarHover,
|
||||
backgroundColor: themeWiseColor('rgba(59, 130, 246, 0.1)', 'rgba(96, 165, 250, 0.2)', themeMode),
|
||||
},
|
||||
};
|
||||
tasks.push(ganttTask);
|
||||
});
|
||||
}
|
||||
|
||||
// Add milestones if enabled
|
||||
if (defaultViewOptions.showMilestones) {
|
||||
phase.milestones.forEach((milestone) => {
|
||||
const milestoneTask: Task = {
|
||||
id: milestone.id,
|
||||
name: milestone.name,
|
||||
start: milestone.dueDate,
|
||||
end: milestone.dueDate,
|
||||
progress: milestone.isCompleted ? 100 : 0,
|
||||
type: 'milestone',
|
||||
project: phase.id,
|
||||
styles: {
|
||||
progressColor: milestone.criticalPath ? ganttColors.criticalPath : ganttColors.progressBar,
|
||||
progressSelectedColor: milestone.criticalPath ? ganttColors.criticalPath : ganttColors.progressBar,
|
||||
backgroundColor: milestone.criticalPath ?
|
||||
themeWiseColor('rgba(239, 68, 68, 0.1)', 'rgba(248, 113, 113, 0.2)', themeMode) :
|
||||
themeWiseColor('rgba(16, 185, 129, 0.1)', 'rgba(52, 211, 153, 0.2)', themeMode),
|
||||
},
|
||||
};
|
||||
tasks.push(milestoneTask);
|
||||
});
|
||||
}
|
||||
});
|
||||
|
||||
return tasks;
|
||||
}, [roadmap.phases, defaultViewOptions, ganttColors, themeMode]);
|
||||
|
||||
const handlePhaseClick = (phase: ProjectPhase) => {
|
||||
const taskCount = phase.tasks.length;
|
||||
const completedTaskCount = phase.tasks.filter(task => task.status === 'done').length;
|
||||
const milestoneCount = phase.milestones.length;
|
||||
const completedMilestoneCount = phase.milestones.filter(m => m.isCompleted).length;
|
||||
const teamMembers = [...new Set(phase.tasks.map(task => task.assigneeName).filter(Boolean))];
|
||||
|
||||
const phaseModalData: PhaseModalData = {
|
||||
...phase,
|
||||
taskCount,
|
||||
completedTaskCount,
|
||||
milestoneCount,
|
||||
completedMilestoneCount,
|
||||
teamMembers,
|
||||
};
|
||||
|
||||
setSelectedPhase(phaseModalData);
|
||||
setIsModalVisible(true);
|
||||
};
|
||||
|
||||
const handleTaskClick = (task: Task) => {
|
||||
// Find the phase this task belongs to
|
||||
const phase = roadmap.phases.find(p =>
|
||||
p.tasks.some(t => t.id === task.id) || p.milestones.some(m => m.id === task.id)
|
||||
);
|
||||
|
||||
if (phase) {
|
||||
handlePhaseClick(phase);
|
||||
}
|
||||
};
|
||||
|
||||
const handleDateChange = (task: Task) => {
|
||||
const phase = roadmap.phases.find(p => p.id === task.id);
|
||||
if (phase && onPhaseUpdate) {
|
||||
onPhaseUpdate(phase.id, {
|
||||
startDate: task.start,
|
||||
endDate: task.end,
|
||||
});
|
||||
} else if (onTaskUpdate) {
|
||||
const parentPhase = roadmap.phases.find(p =>
|
||||
p.tasks.some(t => t.id === task.id)
|
||||
);
|
||||
if (parentPhase) {
|
||||
onTaskUpdate(parentPhase.id, task.id, {
|
||||
startDate: task.start,
|
||||
endDate: task.end,
|
||||
});
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
const handleProgressChange = (task: Task) => {
|
||||
const phase = roadmap.phases.find(p => p.id === task.id);
|
||||
if (phase && onPhaseUpdate) {
|
||||
onPhaseUpdate(phase.id, { progress: task.progress });
|
||||
} else if (onTaskUpdate) {
|
||||
const parentPhase = roadmap.phases.find(p =>
|
||||
p.tasks.some(t => t.id === task.id)
|
||||
);
|
||||
if (parentPhase) {
|
||||
onTaskUpdate(parentPhase.id, task.id, { progress: task.progress });
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
const getStatusColor = (status: string) => {
|
||||
switch (status) {
|
||||
case 'completed': return '#52c41a';
|
||||
case 'in-progress': return '#1890ff';
|
||||
case 'on-hold': return '#faad14';
|
||||
default: return '#d9d9d9';
|
||||
}
|
||||
};
|
||||
|
||||
const columnWidth = viewMode === ViewMode.Year ? 350 :
|
||||
viewMode === ViewMode.Month ? 300 :
|
||||
viewMode === ViewMode.Week ? 250 : 60;
|
||||
|
||||
return (
|
||||
<div className="project-roadmap-gantt w-full">
|
||||
{/* Header */}
|
||||
<div className="bg-white dark:bg-gray-800 border border-gray-200 dark:border-gray-700 rounded-lg shadow-sm mb-4">
|
||||
<div className="p-6">
|
||||
<div className="flex justify-between items-center mb-4">
|
||||
<div>
|
||||
<h3 className="text-xl font-semibold text-gray-900 dark:text-gray-100 mb-2">
|
||||
{roadmap.name}
|
||||
</h3>
|
||||
{roadmap.description && (
|
||||
<p className="text-gray-600 dark:text-gray-400 mb-0">
|
||||
{roadmap.description}
|
||||
</p>
|
||||
)}
|
||||
</div>
|
||||
<Space>
|
||||
<Button
|
||||
type={viewMode === ViewMode.Week ? 'primary' : 'default'}
|
||||
onClick={() => setViewMode(ViewMode.Week)}
|
||||
className="dark:border-gray-600 dark:text-gray-300"
|
||||
>
|
||||
Week
|
||||
</Button>
|
||||
<Button
|
||||
type={viewMode === ViewMode.Month ? 'primary' : 'default'}
|
||||
onClick={() => setViewMode(ViewMode.Month)}
|
||||
className="dark:border-gray-600 dark:text-gray-300"
|
||||
>
|
||||
Month
|
||||
</Button>
|
||||
<Button
|
||||
type={viewMode === ViewMode.Year ? 'primary' : 'default'}
|
||||
onClick={() => setViewMode(ViewMode.Year)}
|
||||
className="dark:border-gray-600 dark:text-gray-300"
|
||||
>
|
||||
Year
|
||||
</Button>
|
||||
</Space>
|
||||
</div>
|
||||
|
||||
{/* Phase Overview */}
|
||||
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-4 mb-4">
|
||||
{roadmap.phases.map((phase) => (
|
||||
<div
|
||||
key={phase.id}
|
||||
className="bg-gray-50 dark:bg-gray-700 border border-gray-200 dark:border-gray-600 rounded-lg p-4 cursor-pointer hover:shadow-md hover:bg-gray-100 dark:hover:bg-gray-600 transition-all duration-200"
|
||||
onClick={() => handlePhaseClick(phase)}
|
||||
>
|
||||
<div className="flex items-center justify-between mb-3">
|
||||
<Badge
|
||||
color={getStatusColor(phase.status)}
|
||||
text={
|
||||
<span className="font-medium text-gray-900 dark:text-gray-100">
|
||||
{phase.name}
|
||||
</span>
|
||||
}
|
||||
/>
|
||||
</div>
|
||||
<div className="space-y-2 text-sm">
|
||||
<div className="flex items-center gap-2 text-gray-600 dark:text-gray-400">
|
||||
<CalendarOutlined className="w-4 h-4" />
|
||||
<span>{phase.startDate.toLocaleDateString()} - {phase.endDate.toLocaleDateString()}</span>
|
||||
</div>
|
||||
<div className="flex items-center gap-2 text-gray-600 dark:text-gray-400">
|
||||
<TeamOutlined className="w-4 h-4" />
|
||||
<span>{phase.tasks.length} tasks</span>
|
||||
</div>
|
||||
<div className="flex items-center gap-2 text-gray-600 dark:text-gray-400">
|
||||
<CheckCircleOutlined className="w-4 h-4" />
|
||||
<span>{phase.progress}% complete</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Gantt Chart */}
|
||||
<div className="bg-white dark:bg-gray-800 border border-gray-200 dark:border-gray-700 rounded-lg shadow-sm">
|
||||
<div className="p-4">
|
||||
<div className="w-full overflow-x-auto">
|
||||
<div
|
||||
className="gantt-container"
|
||||
style={{
|
||||
'--gantt-background': ganttColors.background,
|
||||
'--gantt-grid': ganttColors.grid,
|
||||
'--gantt-text': ganttColors.text.primary,
|
||||
'--gantt-border': ganttColors.border,
|
||||
} as React.CSSProperties}
|
||||
>
|
||||
<Gantt
|
||||
tasks={ganttTasks}
|
||||
viewMode={viewMode}
|
||||
onDateChange={handleDateChange}
|
||||
onProgressChange={handleProgressChange}
|
||||
onDoubleClick={handleTaskClick}
|
||||
listCellWidth=""
|
||||
columnWidth={columnWidth}
|
||||
todayColor={ganttColors.today}
|
||||
projectProgressColor={ganttColors.progressBar}
|
||||
projectBackgroundColor={themeWiseColor('rgba(82, 196, 26, 0.1)', 'rgba(52, 211, 153, 0.2)', themeMode)}
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Phase Modal */}
|
||||
<PhaseModal
|
||||
visible={isModalVisible}
|
||||
phase={selectedPhase}
|
||||
onClose={() => setIsModalVisible(false)}
|
||||
onUpdate={(updates) => {
|
||||
if (selectedPhase && onPhaseUpdate) {
|
||||
onPhaseUpdate(selectedPhase.id, updates);
|
||||
}
|
||||
}}
|
||||
/>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
export default ProjectRoadmapGantt;
|
||||
@@ -0,0 +1,112 @@
|
||||
import React, { useState } from 'react';
|
||||
import { Button, Space, message } from 'antd';
|
||||
import ProjectRoadmapGantt from './ProjectRoadmapGantt';
|
||||
import { sampleProjectRoadmap } from './sample-data';
|
||||
import { ProjectPhase, ProjectRoadmap } from '../../types/project-roadmap.types';
|
||||
import { useAppSelector } from '../../hooks/useAppSelector';
|
||||
|
||||
const RoadmapDemo: React.FC = () => {
|
||||
const [roadmap, setRoadmap] = useState<ProjectRoadmap>(sampleProjectRoadmap);
|
||||
const themeMode = useAppSelector(state => state.themeReducer.mode);
|
||||
|
||||
const handlePhaseUpdate = (phaseId: string, updates: Partial<ProjectPhase>) => {
|
||||
setRoadmap(prevRoadmap => ({
|
||||
...prevRoadmap,
|
||||
phases: prevRoadmap.phases.map(phase =>
|
||||
phase.id === phaseId
|
||||
? { ...phase, ...updates }
|
||||
: phase
|
||||
)
|
||||
}));
|
||||
|
||||
message.success('Phase updated successfully!');
|
||||
};
|
||||
|
||||
const handleTaskUpdate = (phaseId: string, taskId: string, updates: any) => {
|
||||
setRoadmap(prevRoadmap => ({
|
||||
...prevRoadmap,
|
||||
phases: prevRoadmap.phases.map(phase =>
|
||||
phase.id === phaseId
|
||||
? {
|
||||
...phase,
|
||||
tasks: phase.tasks.map(task =>
|
||||
task.id === taskId
|
||||
? { ...task, ...updates }
|
||||
: task
|
||||
)
|
||||
}
|
||||
: phase
|
||||
)
|
||||
}));
|
||||
|
||||
message.success('Task updated successfully!');
|
||||
};
|
||||
|
||||
const resetToSampleData = () => {
|
||||
setRoadmap(sampleProjectRoadmap);
|
||||
message.info('Roadmap reset to sample data');
|
||||
};
|
||||
|
||||
return (
|
||||
<div className="roadmap-demo p-6 bg-gray-50 dark:bg-gray-900 min-h-screen">
|
||||
<div className="bg-white dark:bg-gray-800 border border-gray-200 dark:border-gray-700 rounded-lg shadow-sm mb-4">
|
||||
<div className="p-6">
|
||||
<div className="flex justify-between items-center">
|
||||
<div>
|
||||
<h2 className="text-2xl font-bold text-gray-900 dark:text-gray-100 mb-2">
|
||||
Project Roadmap Gantt Chart Demo
|
||||
</h2>
|
||||
<p className="text-gray-600 dark:text-gray-400 mb-0">
|
||||
Interactive Gantt chart showing project phases as milestones/epics.
|
||||
Click on any phase card or Gantt bar to view detailed information in a modal.
|
||||
</p>
|
||||
</div>
|
||||
<Space>
|
||||
<Button
|
||||
onClick={resetToSampleData}
|
||||
className="dark:border-gray-600 dark:text-gray-300 dark:hover:bg-gray-700"
|
||||
>
|
||||
Reset to Sample Data
|
||||
</Button>
|
||||
</Space>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<ProjectRoadmapGantt
|
||||
roadmap={roadmap}
|
||||
viewOptions={{
|
||||
viewMode: 'month',
|
||||
showTasks: true,
|
||||
showMilestones: true,
|
||||
groupByPhase: true,
|
||||
}}
|
||||
onPhaseUpdate={handlePhaseUpdate}
|
||||
onTaskUpdate={handleTaskUpdate}
|
||||
/>
|
||||
|
||||
<div className="bg-white dark:bg-gray-800 border border-gray-200 dark:border-gray-700 rounded-lg shadow-sm mt-4">
|
||||
<div className="p-6">
|
||||
<h3 className="text-lg font-semibold text-gray-900 dark:text-gray-100 mb-3">
|
||||
Features Demonstrated:
|
||||
</h3>
|
||||
<ul className="space-y-2 text-gray-700 dark:text-gray-300">
|
||||
<li>• <strong className="text-gray-900 dark:text-gray-100">Phase-based Grouping:</strong> Projects organized by phases (Planning, Development, Testing, Deployment)</li>
|
||||
<li>• <strong className="text-gray-900 dark:text-gray-100">Interactive Phase Cards:</strong> Click on phase cards for detailed view</li>
|
||||
<li>• <strong className="text-gray-900 dark:text-gray-100">Gantt Chart Visualization:</strong> Timeline view with tasks, milestones, and dependencies</li>
|
||||
<li>• <strong className="text-gray-900 dark:text-gray-100">Modal Details:</strong> Comprehensive phase information with tasks, milestones, and team members</li>
|
||||
<li>• <strong className="text-gray-900 dark:text-gray-100">Progress Tracking:</strong> Visual progress indicators and completion statistics</li>
|
||||
<li>• <strong className="text-gray-900 dark:text-gray-100">Multiple View Modes:</strong> Week, Month, and Year timeline views</li>
|
||||
<li>• <strong className="text-gray-900 dark:text-gray-100">Task Management:</strong> Task assignments, priorities, and status tracking</li>
|
||||
<li>• <strong className="text-gray-900 dark:text-gray-100">Milestone Tracking:</strong> Critical path milestones and completion status</li>
|
||||
<li>• <strong className="text-gray-900 dark:text-gray-100">Team Overview:</strong> Team member assignments and workload distribution</li>
|
||||
<li>• <strong className="text-gray-900 dark:text-gray-100">Editable Fields:</strong> In-modal editing for phase attributes (name, description, dates, status)</li>
|
||||
<li>• <strong className="text-gray-900 dark:text-gray-100">Theme Support:</strong> Automatic light/dark theme adaptation with consistent styling</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
export default RoadmapDemo;
|
||||
@@ -0,0 +1,221 @@
|
||||
/* Custom Gantt Chart Theme Overrides */
|
||||
|
||||
/* Light theme styles */
|
||||
.gantt-container {
|
||||
font-family: inherit;
|
||||
}
|
||||
|
||||
.gantt-container .gantt-header {
|
||||
background-color: var(--gantt-background, #ffffff);
|
||||
border-bottom: 1px solid var(--gantt-border, #e5e7eb);
|
||||
color: var(--gantt-text, #111827);
|
||||
}
|
||||
|
||||
.gantt-container .gantt-task-list-wrapper {
|
||||
background-color: var(--gantt-background, #ffffff);
|
||||
border-right: 1px solid var(--gantt-border, #e5e7eb);
|
||||
}
|
||||
|
||||
.gantt-container .gantt-task-list-table {
|
||||
background-color: var(--gantt-background, #ffffff);
|
||||
}
|
||||
|
||||
.gantt-container .gantt-task-list-table-row {
|
||||
color: var(--gantt-text, #111827);
|
||||
border-bottom: 1px solid var(--gantt-grid, #f3f4f6);
|
||||
}
|
||||
|
||||
.gantt-container .gantt-task-list-table-row:hover {
|
||||
background-color: var(--gantt-grid, #f3f4f6);
|
||||
}
|
||||
|
||||
.gantt-container .gantt-gantt {
|
||||
background-color: var(--gantt-background, #ffffff);
|
||||
}
|
||||
|
||||
.gantt-container .gantt-vertical-container {
|
||||
background-color: var(--gantt-background, #ffffff);
|
||||
}
|
||||
|
||||
.gantt-container .gantt-horizontal-container {
|
||||
background-color: var(--gantt-background, #ffffff);
|
||||
}
|
||||
|
||||
.gantt-container .gantt-calendar {
|
||||
background-color: var(--gantt-background, #ffffff);
|
||||
border-bottom: 1px solid var(--gantt-border, #e5e7eb);
|
||||
}
|
||||
|
||||
.gantt-container .gantt-calendar-row {
|
||||
color: var(--gantt-text, #111827);
|
||||
background-color: var(--gantt-background, #ffffff);
|
||||
border-bottom: 1px solid var(--gantt-grid, #f3f4f6);
|
||||
}
|
||||
|
||||
.gantt-container .gantt-grid-body {
|
||||
background-color: var(--gantt-background, #ffffff);
|
||||
}
|
||||
|
||||
.gantt-container .gantt-grid-row {
|
||||
border-bottom: 1px solid var(--gantt-grid, #f3f4f6);
|
||||
}
|
||||
|
||||
.gantt-container .gantt-grid-row-line {
|
||||
border-left: 1px solid var(--gantt-grid, #f3f4f6);
|
||||
}
|
||||
|
||||
/* Dark theme overrides */
|
||||
html.dark .gantt-container .gantt-header {
|
||||
background-color: #1f2937;
|
||||
border-bottom-color: #4b5563;
|
||||
color: #f9fafb;
|
||||
}
|
||||
|
||||
html.dark .gantt-container .gantt-task-list-wrapper {
|
||||
background-color: #1f2937;
|
||||
border-right-color: #4b5563;
|
||||
}
|
||||
|
||||
html.dark .gantt-container .gantt-task-list-table {
|
||||
background-color: #1f2937;
|
||||
}
|
||||
|
||||
html.dark .gantt-container .gantt-task-list-table-row {
|
||||
color: #f9fafb;
|
||||
border-bottom-color: #4b5563;
|
||||
}
|
||||
|
||||
html.dark .gantt-container .gantt-task-list-table-row:hover {
|
||||
background-color: #374151;
|
||||
}
|
||||
|
||||
html.dark .gantt-container .gantt-gantt {
|
||||
background-color: #1f2937;
|
||||
}
|
||||
|
||||
html.dark .gantt-container .gantt-vertical-container {
|
||||
background-color: #1f2937;
|
||||
}
|
||||
|
||||
html.dark .gantt-container .gantt-horizontal-container {
|
||||
background-color: #1f2937;
|
||||
}
|
||||
|
||||
html.dark .gantt-container .gantt-calendar {
|
||||
background-color: #1f2937;
|
||||
border-bottom-color: #4b5563;
|
||||
}
|
||||
|
||||
html.dark .gantt-container .gantt-calendar-row {
|
||||
color: #f9fafb;
|
||||
background-color: #1f2937;
|
||||
border-bottom-color: #4b5563;
|
||||
}
|
||||
|
||||
html.dark .gantt-container .gantt-grid-body {
|
||||
background-color: #1f2937;
|
||||
}
|
||||
|
||||
html.dark .gantt-container .gantt-grid-row {
|
||||
border-bottom-color: #4b5563;
|
||||
}
|
||||
|
||||
html.dark .gantt-container .gantt-grid-row-line {
|
||||
border-left-color: #4b5563;
|
||||
}
|
||||
|
||||
/* Tooltip theming */
|
||||
.gantt-container .gantt-tooltip {
|
||||
background-color: var(--gantt-background, #ffffff);
|
||||
border: 1px solid var(--gantt-border, #e5e7eb);
|
||||
color: var(--gantt-text, #111827);
|
||||
box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
html.dark .gantt-container .gantt-tooltip {
|
||||
background-color: #374151;
|
||||
border-color: #4b5563;
|
||||
color: #f9fafb;
|
||||
box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.3);
|
||||
}
|
||||
|
||||
/* Task bar styling improvements */
|
||||
.gantt-container .gantt-task-progress {
|
||||
border-radius: 2px;
|
||||
}
|
||||
|
||||
.gantt-container .gantt-task-background {
|
||||
border-radius: 4px;
|
||||
border: 1px solid var(--gantt-border, #e5e7eb);
|
||||
}
|
||||
|
||||
html.dark .gantt-container .gantt-task-background {
|
||||
border-color: #4b5563;
|
||||
}
|
||||
|
||||
/* Milestone styling */
|
||||
.gantt-container .gantt-milestone {
|
||||
filter: drop-shadow(0 1px 2px rgba(0, 0, 0, 0.1));
|
||||
}
|
||||
|
||||
html.dark .gantt-container .gantt-milestone {
|
||||
filter: drop-shadow(0 1px 2px rgba(0, 0, 0, 0.3));
|
||||
}
|
||||
|
||||
/* Selection and hover states */
|
||||
.gantt-container .gantt-task:hover .gantt-task-background {
|
||||
opacity: 0.9;
|
||||
transform: translateY(-1px);
|
||||
transition: all 0.2s ease;
|
||||
}
|
||||
|
||||
.gantt-container .gantt-task-selected .gantt-task-background {
|
||||
box-shadow: 0 0 0 2px rgba(59, 130, 246, 0.3);
|
||||
}
|
||||
|
||||
html.dark .gantt-container .gantt-task-selected .gantt-task-background {
|
||||
box-shadow: 0 0 0 2px rgba(96, 165, 250, 0.3);
|
||||
}
|
||||
|
||||
/* Responsive improvements */
|
||||
@media (max-width: 768px) {
|
||||
.gantt-container {
|
||||
font-size: 14px;
|
||||
}
|
||||
|
||||
.gantt-container .gantt-task-list-wrapper {
|
||||
min-width: 200px;
|
||||
}
|
||||
}
|
||||
|
||||
/* Scrollbar theming */
|
||||
.gantt-container ::-webkit-scrollbar {
|
||||
width: 8px;
|
||||
height: 8px;
|
||||
}
|
||||
|
||||
.gantt-container ::-webkit-scrollbar-track {
|
||||
background: var(--gantt-grid, #f3f4f6);
|
||||
border-radius: 4px;
|
||||
}
|
||||
|
||||
.gantt-container ::-webkit-scrollbar-thumb {
|
||||
background: var(--gantt-border, #e5e7eb);
|
||||
border-radius: 4px;
|
||||
}
|
||||
|
||||
.gantt-container ::-webkit-scrollbar-thumb:hover {
|
||||
background: #9ca3af;
|
||||
}
|
||||
|
||||
html.dark .gantt-container ::-webkit-scrollbar-track {
|
||||
background: #4b5563;
|
||||
}
|
||||
|
||||
html.dark .gantt-container ::-webkit-scrollbar-thumb {
|
||||
background: #6b7280;
|
||||
}
|
||||
|
||||
html.dark .gantt-container ::-webkit-scrollbar-thumb:hover {
|
||||
background: #9ca3af;
|
||||
}
|
||||
@@ -0,0 +1,5 @@
|
||||
export { default as ProjectRoadmapGantt } from './ProjectRoadmapGantt';
|
||||
export { default as PhaseModal } from './PhaseModal';
|
||||
export { default as RoadmapDemo } from './RoadmapDemo';
|
||||
export { sampleProjectRoadmap } from './sample-data';
|
||||
export * from '../../types/project-roadmap.types';
|
||||
@@ -0,0 +1,317 @@
|
||||
import { ProjectRoadmap, ProjectPhase, PhaseTask, PhaseMilestone } from '../../types/project-roadmap.types';
|
||||
|
||||
// Sample tasks for Planning Phase
|
||||
const planningTasks: PhaseTask[] = [
|
||||
{
|
||||
id: 'task-planning-1',
|
||||
name: 'Project Requirements Analysis',
|
||||
description: 'Gather and analyze project requirements from stakeholders',
|
||||
startDate: new Date(2024, 11, 1), // December 1, 2024
|
||||
endDate: new Date(2024, 11, 5),
|
||||
progress: 100,
|
||||
assigneeId: 'user-1',
|
||||
assigneeName: 'Alice Johnson',
|
||||
priority: 'high',
|
||||
status: 'done',
|
||||
},
|
||||
{
|
||||
id: 'task-planning-2',
|
||||
name: 'Technical Architecture Design',
|
||||
description: 'Design the technical architecture and system components',
|
||||
startDate: new Date(2024, 11, 6),
|
||||
endDate: new Date(2024, 11, 12),
|
||||
progress: 75,
|
||||
assigneeId: 'user-2',
|
||||
assigneeName: 'Bob Smith',
|
||||
priority: 'high',
|
||||
status: 'in-progress',
|
||||
},
|
||||
{
|
||||
id: 'task-planning-3',
|
||||
name: 'Resource Allocation Planning',
|
||||
description: 'Plan and allocate team resources for the project',
|
||||
startDate: new Date(2024, 11, 8),
|
||||
endDate: new Date(2024, 11, 15),
|
||||
progress: 50,
|
||||
assigneeId: 'user-3',
|
||||
assigneeName: 'Carol Davis',
|
||||
priority: 'medium',
|
||||
status: 'in-progress',
|
||||
dependencies: ['task-planning-1'],
|
||||
}
|
||||
];
|
||||
|
||||
// Sample milestones for Planning Phase
|
||||
const planningMilestones: PhaseMilestone[] = [
|
||||
{
|
||||
id: 'milestone-planning-1',
|
||||
name: 'Requirements Approved',
|
||||
description: 'All project requirements have been reviewed and approved by stakeholders',
|
||||
dueDate: new Date(2024, 11, 5),
|
||||
isCompleted: true,
|
||||
criticalPath: true,
|
||||
},
|
||||
{
|
||||
id: 'milestone-planning-2',
|
||||
name: 'Architecture Review Complete',
|
||||
description: 'Technical architecture has been reviewed and approved',
|
||||
dueDate: new Date(2024, 11, 15),
|
||||
isCompleted: false,
|
||||
criticalPath: true,
|
||||
}
|
||||
];
|
||||
|
||||
// Sample tasks for Development Phase
|
||||
const developmentTasks: PhaseTask[] = [
|
||||
{
|
||||
id: 'task-dev-1',
|
||||
name: 'Frontend Component Development',
|
||||
description: 'Develop core frontend components using React',
|
||||
startDate: new Date(2024, 11, 16),
|
||||
endDate: new Date(2025, 0, 31), // January 31, 2025
|
||||
progress: 30,
|
||||
assigneeId: 'user-4',
|
||||
assigneeName: 'David Wilson',
|
||||
priority: 'high',
|
||||
status: 'in-progress',
|
||||
dependencies: ['task-planning-2'],
|
||||
},
|
||||
{
|
||||
id: 'task-dev-2',
|
||||
name: 'Backend API Development',
|
||||
description: 'Develop REST APIs and database models',
|
||||
startDate: new Date(2024, 11, 16),
|
||||
endDate: new Date(2025, 0, 25),
|
||||
progress: 45,
|
||||
assigneeId: 'user-5',
|
||||
assigneeName: 'Eva Brown',
|
||||
priority: 'high',
|
||||
status: 'in-progress',
|
||||
dependencies: ['task-planning-2'],
|
||||
},
|
||||
{
|
||||
id: 'task-dev-3',
|
||||
name: 'Database Setup and Migration',
|
||||
description: 'Set up production database and create migration scripts',
|
||||
startDate: new Date(2024, 11, 20),
|
||||
endDate: new Date(2025, 0, 15),
|
||||
progress: 80,
|
||||
assigneeId: 'user-6',
|
||||
assigneeName: 'Frank Miller',
|
||||
priority: 'medium',
|
||||
status: 'in-progress',
|
||||
}
|
||||
];
|
||||
|
||||
// Sample milestones for Development Phase
|
||||
const developmentMilestones: PhaseMilestone[] = [
|
||||
{
|
||||
id: 'milestone-dev-1',
|
||||
name: 'Core Components Complete',
|
||||
description: 'All core frontend components have been developed and tested',
|
||||
dueDate: new Date(2025, 0, 20),
|
||||
isCompleted: false,
|
||||
criticalPath: false,
|
||||
},
|
||||
{
|
||||
id: 'milestone-dev-2',
|
||||
name: 'API Development Complete',
|
||||
description: 'All backend APIs are developed and documented',
|
||||
dueDate: new Date(2025, 0, 25),
|
||||
isCompleted: false,
|
||||
criticalPath: true,
|
||||
}
|
||||
];
|
||||
|
||||
// Sample tasks for Testing Phase
|
||||
const testingTasks: PhaseTask[] = [
|
||||
{
|
||||
id: 'task-test-1',
|
||||
name: 'Unit Testing Implementation',
|
||||
description: 'Write and execute comprehensive unit tests',
|
||||
startDate: new Date(2025, 1, 1), // February 1, 2025
|
||||
endDate: new Date(2025, 1, 15),
|
||||
progress: 0,
|
||||
assigneeId: 'user-7',
|
||||
assigneeName: 'Grace Lee',
|
||||
priority: 'high',
|
||||
status: 'todo',
|
||||
dependencies: ['task-dev-1', 'task-dev-2'],
|
||||
},
|
||||
{
|
||||
id: 'task-test-2',
|
||||
name: 'Integration Testing',
|
||||
description: 'Perform integration testing between frontend and backend',
|
||||
startDate: new Date(2025, 1, 10),
|
||||
endDate: new Date(2025, 1, 25),
|
||||
progress: 0,
|
||||
assigneeId: 'user-8',
|
||||
assigneeName: 'Henry Clark',
|
||||
priority: 'high',
|
||||
status: 'todo',
|
||||
dependencies: ['task-test-1'],
|
||||
},
|
||||
{
|
||||
id: 'task-test-3',
|
||||
name: 'User Acceptance Testing',
|
||||
description: 'Conduct user acceptance testing with stakeholders',
|
||||
startDate: new Date(2025, 1, 20),
|
||||
endDate: new Date(2025, 2, 5), // March 5, 2025
|
||||
progress: 0,
|
||||
assigneeId: 'user-9',
|
||||
assigneeName: 'Ivy Taylor',
|
||||
priority: 'medium',
|
||||
status: 'todo',
|
||||
dependencies: ['task-test-2'],
|
||||
}
|
||||
];
|
||||
|
||||
// Sample milestones for Testing Phase
|
||||
const testingMilestones: PhaseMilestone[] = [
|
||||
{
|
||||
id: 'milestone-test-1',
|
||||
name: 'All Tests Passing',
|
||||
description: 'All unit and integration tests are passing',
|
||||
dueDate: new Date(2025, 1, 25),
|
||||
isCompleted: false,
|
||||
criticalPath: true,
|
||||
},
|
||||
{
|
||||
id: 'milestone-test-2',
|
||||
name: 'UAT Sign-off',
|
||||
description: 'User acceptance testing completed and signed off',
|
||||
dueDate: new Date(2025, 2, 5),
|
||||
isCompleted: false,
|
||||
criticalPath: true,
|
||||
}
|
||||
];
|
||||
|
||||
// Sample tasks for Deployment Phase
|
||||
const deploymentTasks: PhaseTask[] = [
|
||||
{
|
||||
id: 'task-deploy-1',
|
||||
name: 'Production Environment Setup',
|
||||
description: 'Configure and set up production environment',
|
||||
startDate: new Date(2025, 2, 6), // March 6, 2025
|
||||
endDate: new Date(2025, 2, 12),
|
||||
progress: 0,
|
||||
assigneeId: 'user-10',
|
||||
assigneeName: 'Jack Anderson',
|
||||
priority: 'high',
|
||||
status: 'todo',
|
||||
dependencies: ['task-test-3'],
|
||||
},
|
||||
{
|
||||
id: 'task-deploy-2',
|
||||
name: 'Application Deployment',
|
||||
description: 'Deploy application to production environment',
|
||||
startDate: new Date(2025, 2, 13),
|
||||
endDate: new Date(2025, 2, 15),
|
||||
progress: 0,
|
||||
assigneeId: 'user-11',
|
||||
assigneeName: 'Kelly White',
|
||||
priority: 'high',
|
||||
status: 'todo',
|
||||
dependencies: ['task-deploy-1'],
|
||||
},
|
||||
{
|
||||
id: 'task-deploy-3',
|
||||
name: 'Post-Deployment Monitoring',
|
||||
description: 'Monitor application performance post-deployment',
|
||||
startDate: new Date(2025, 2, 16),
|
||||
endDate: new Date(2025, 2, 20),
|
||||
progress: 0,
|
||||
assigneeId: 'user-12',
|
||||
assigneeName: 'Liam Garcia',
|
||||
priority: 'medium',
|
||||
status: 'todo',
|
||||
dependencies: ['task-deploy-2'],
|
||||
}
|
||||
];
|
||||
|
||||
// Sample milestones for Deployment Phase
|
||||
const deploymentMilestones: PhaseMilestone[] = [
|
||||
{
|
||||
id: 'milestone-deploy-1',
|
||||
name: 'Production Go-Live',
|
||||
description: 'Application is live in production environment',
|
||||
dueDate: new Date(2025, 2, 15),
|
||||
isCompleted: false,
|
||||
criticalPath: true,
|
||||
},
|
||||
{
|
||||
id: 'milestone-deploy-2',
|
||||
name: 'Project Handover',
|
||||
description: 'Project handed over to maintenance team',
|
||||
dueDate: new Date(2025, 2, 20),
|
||||
isCompleted: false,
|
||||
criticalPath: false,
|
||||
}
|
||||
];
|
||||
|
||||
// Sample project phases
|
||||
const samplePhases: ProjectPhase[] = [
|
||||
{
|
||||
id: 'phase-planning',
|
||||
name: 'Planning & Analysis',
|
||||
description: 'Initial project planning, requirements gathering, and technical analysis',
|
||||
startDate: new Date(2024, 11, 1),
|
||||
endDate: new Date(2024, 11, 15),
|
||||
progress: 75,
|
||||
color: '#1890ff',
|
||||
status: 'in-progress',
|
||||
tasks: planningTasks,
|
||||
milestones: planningMilestones,
|
||||
},
|
||||
{
|
||||
id: 'phase-development',
|
||||
name: 'Development',
|
||||
description: 'Core development of frontend, backend, and database components',
|
||||
startDate: new Date(2024, 11, 16),
|
||||
endDate: new Date(2025, 0, 31),
|
||||
progress: 40,
|
||||
color: '#52c41a',
|
||||
status: 'in-progress',
|
||||
tasks: developmentTasks,
|
||||
milestones: developmentMilestones,
|
||||
},
|
||||
{
|
||||
id: 'phase-testing',
|
||||
name: 'Testing & QA',
|
||||
description: 'Comprehensive testing including unit, integration, and user acceptance testing',
|
||||
startDate: new Date(2025, 1, 1),
|
||||
endDate: new Date(2025, 2, 5),
|
||||
progress: 0,
|
||||
color: '#faad14',
|
||||
status: 'not-started',
|
||||
tasks: testingTasks,
|
||||
milestones: testingMilestones,
|
||||
},
|
||||
{
|
||||
id: 'phase-deployment',
|
||||
name: 'Deployment & Launch',
|
||||
description: 'Production deployment and project launch activities',
|
||||
startDate: new Date(2025, 2, 6),
|
||||
endDate: new Date(2025, 2, 20),
|
||||
progress: 0,
|
||||
color: '#722ed1',
|
||||
status: 'not-started',
|
||||
tasks: deploymentTasks,
|
||||
milestones: deploymentMilestones,
|
||||
}
|
||||
];
|
||||
|
||||
// Sample project roadmap
|
||||
export const sampleProjectRoadmap: ProjectRoadmap = {
|
||||
id: 'roadmap-sample-project',
|
||||
projectId: 'project-web-platform',
|
||||
name: 'Web Platform Development Roadmap',
|
||||
description: 'Comprehensive roadmap for developing a new web-based platform with modern technologies and agile methodologies',
|
||||
startDate: new Date(2024, 11, 1),
|
||||
endDate: new Date(2025, 2, 20),
|
||||
phases: samplePhases,
|
||||
createdAt: new Date(2024, 10, 15),
|
||||
updatedAt: new Date(2024, 11, 1),
|
||||
};
|
||||
|
||||
export default sampleProjectRoadmap;
|
||||
@@ -1,4 +1,4 @@
|
||||
import { Drawer, Flex, Form, Select, Typography, List, Button } from 'antd/es';
|
||||
import { Drawer, Flex, Form, Select, Typography, List, Button, Modal, Divider } from 'antd/es';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { useEffect, useState } from 'react';
|
||||
|
||||
@@ -21,7 +21,7 @@ import { teamMembersApiService } from '@/api/team-members/teamMembers.api.servic
|
||||
|
||||
const ProjectMemberDrawer = () => {
|
||||
const { t } = useTranslation('project-view/project-member-drawer');
|
||||
const { isDrawerOpen, currentMembersList, isLoading } = useAppSelector(
|
||||
const { isDrawerOpen, currentMembersList, isLoading, isFromAssigner } = useAppSelector(
|
||||
state => state.projectMemberReducer
|
||||
);
|
||||
const { projectId } = useAppSelector(state => state.projectReducer);
|
||||
@@ -33,6 +33,12 @@ const ProjectMemberDrawer = () => {
|
||||
const [members, setMembers] = useState<ITeamMembersViewModel>({ data: [], total: 0 });
|
||||
const [teamMembersLoading, setTeamMembersLoading] = useState(false);
|
||||
|
||||
// Filter out members already in the project
|
||||
const currentProjectMemberIds = (currentMembersList || []).map(m => m.team_member_id).filter(Boolean);
|
||||
const availableMembers = (members?.data || []).filter(
|
||||
member => member.id && !currentProjectMemberIds.includes(member.id)
|
||||
);
|
||||
|
||||
const fetchProjectMembers = async () => {
|
||||
if (!projectId) return;
|
||||
dispatch(getAllProjectMembers(projectId));
|
||||
@@ -175,8 +181,9 @@ const ProjectMemberDrawer = () => {
|
||||
);
|
||||
|
||||
const renderNotFoundContent = () => (
|
||||
<Flex>
|
||||
<Flex style={{ display: 'block'}}>
|
||||
<Button
|
||||
className='mb-2'
|
||||
block
|
||||
type="primary"
|
||||
onClick={sendInviteToProject}
|
||||
@@ -188,60 +195,74 @@ const ProjectMemberDrawer = () => {
|
||||
{validateEmail(searchTerm) ? t('inviteAsAMember') : t('inviteNewMemberByEmail')}
|
||||
</span>
|
||||
</Button>
|
||||
{/* {isFromAssigner && <Flex>
|
||||
<input className='mr-2' type="checkbox" checked={true} name={t('alsoInviteToProject')} id="AlsoInviteToProject" />
|
||||
<label htmlFor={t('alsoInviteToProject')}>{t('alsoInviteToProject')}</label>
|
||||
</Flex>} */}
|
||||
</Flex>
|
||||
);
|
||||
|
||||
return (
|
||||
<Drawer
|
||||
title={
|
||||
<Typography.Text style={{ fontWeight: 500, fontSize: 16 }}>{t('title')}</Typography.Text>
|
||||
}
|
||||
open={isDrawerOpen}
|
||||
onClose={() => dispatch(toggleProjectMemberDrawer())}
|
||||
afterOpenChange={handleOpenChange}
|
||||
>
|
||||
<Form form={form} layout="vertical" onFinish={handleSelectChange}>
|
||||
<Form.Item name="memberName" label={t('searchLabel')}>
|
||||
<Select
|
||||
loading={teamMembersLoading}
|
||||
placeholder={t('searchPlaceholder')}
|
||||
showSearch
|
||||
onSearch={handleSearch}
|
||||
onChange={handleSelectChange}
|
||||
onKeyDown={handleKeyDown}
|
||||
options={members?.data?.map(member => ({
|
||||
key: member.id,
|
||||
value: member.id,
|
||||
name: member.name,
|
||||
label: renderMemberOption(member),
|
||||
}))}
|
||||
filterOption={false}
|
||||
notFoundContent={renderNotFoundContent()}
|
||||
optionLabelProp="name"
|
||||
<Modal
|
||||
title={
|
||||
<Typography.Text style={{ fontWeight: 500, fontSize: 16 }}>{isFromAssigner ? t('inviteMember') : t('title')}</Typography.Text>
|
||||
}
|
||||
open={isDrawerOpen}
|
||||
onCancel={() => dispatch(toggleProjectMemberDrawer())}
|
||||
afterOpenChange={handleOpenChange}
|
||||
footer={
|
||||
<>
|
||||
{/* {!isFromAssigner && <Button
|
||||
style={{ width: 140, fontSize: 12 }}
|
||||
block
|
||||
icon={<LinkOutlined />}
|
||||
disabled
|
||||
>
|
||||
{t('copyProjectLink')}
|
||||
</Button>} */}
|
||||
</>
|
||||
}
|
||||
>
|
||||
<Form form={form} layout="vertical" onFinish={handleSelectChange}>
|
||||
<Form.Item name="memberName" label={t('searchLabel')}>
|
||||
<Select
|
||||
loading={teamMembersLoading}
|
||||
placeholder={t('searchPlaceholder')}
|
||||
showSearch
|
||||
onSearch={handleSearch}
|
||||
onChange={handleSelectChange}
|
||||
onKeyDown={handleKeyDown}
|
||||
options={availableMembers.map(member => ({
|
||||
key: member.id,
|
||||
value: member.id,
|
||||
name: member.name,
|
||||
label: renderMemberOption(member),
|
||||
}))}
|
||||
filterOption={false}
|
||||
notFoundContent={renderNotFoundContent()}
|
||||
optionLabelProp="name"
|
||||
/>
|
||||
</Form.Item>
|
||||
</Form>
|
||||
{!isFromAssigner && <><div style={{ fontSize: 14, fontWeight: 500, marginBottom: 8 }}>{t('members')}</div>
|
||||
<div style={{ maxHeight: 360, minHeight: 120, overflowY: 'auto', marginBottom: 16 }}>
|
||||
<List
|
||||
loading={isLoading}
|
||||
bordered
|
||||
size="small"
|
||||
itemLayout="horizontal"
|
||||
dataSource={currentMembersList}
|
||||
renderItem={member => (
|
||||
<List.Item key={member.id} >
|
||||
<Flex gap={4} align="center" justify="space-between" style={{ width: '100%' }}>
|
||||
{renderMemberOption(member)}
|
||||
</Flex>
|
||||
</List.Item>
|
||||
)}
|
||||
/>
|
||||
</Form.Item>
|
||||
</Form>
|
||||
|
||||
<List
|
||||
loading={isLoading}
|
||||
bordered
|
||||
size="small"
|
||||
itemLayout="horizontal"
|
||||
dataSource={currentMembersList}
|
||||
renderItem={member => (
|
||||
<List.Item key={member.id}>
|
||||
<Flex gap={4} align="center" justify="space-between" style={{ width: '100%' }}>
|
||||
{renderMemberOption(member)}
|
||||
<Button
|
||||
onClick={() => handleDeleteMember(member.id)}
|
||||
size="small"
|
||||
icon={<DeleteOutlined />}
|
||||
/>
|
||||
</Flex>
|
||||
</List.Item>
|
||||
)}
|
||||
/>
|
||||
</Drawer>
|
||||
</div></>
|
||||
}
|
||||
</Modal>
|
||||
);
|
||||
};
|
||||
|
||||
|
||||
@@ -94,7 +94,7 @@ const UpdateMemberDrawer = ({ selectedMemberId, onRoleUpdate }: UpdateMemberDraw
|
||||
|
||||
try {
|
||||
const body: ITeamMemberCreateRequest = {
|
||||
job_title: selectedJobTitle,
|
||||
job_title: form.getFieldValue('jobTitle'),
|
||||
emails: [teamMember.email],
|
||||
is_admin: values.access === 'admin',
|
||||
};
|
||||
|
||||
269
worklenz-frontend/src/components/survey/SurveyPromptModal.tsx
Normal file
269
worklenz-frontend/src/components/survey/SurveyPromptModal.tsx
Normal file
@@ -0,0 +1,269 @@
|
||||
import React, { useState, useEffect, useRef } from 'react';
|
||||
import { Modal, Button, Result, Spin, Flex } from '@/shared/antd-imports';
|
||||
import { SurveyStep } from '@/components/account-setup/survey-step';
|
||||
import { useSurveyStatus } from '@/hooks/useSurveyStatus';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { useAppSelector } from '@/hooks/useAppSelector';
|
||||
import { useAppDispatch } from '@/hooks/useAppDispatch';
|
||||
import { surveyApiService } from '@/api/survey/survey.api.service';
|
||||
import { appMessage } from '@/shared/antd-imports';
|
||||
import { ISurveySubmissionRequest } from '@/types/account-setup/survey.types';
|
||||
import logger from '@/utils/errorLogger';
|
||||
import { resetSurveyData, setSurveySubStep } from '@/features/account-setup/account-setup.slice';
|
||||
|
||||
interface SurveyPromptModalProps {
|
||||
forceShow?: boolean;
|
||||
onClose?: () => void;
|
||||
}
|
||||
|
||||
export const SurveyPromptModal: React.FC<SurveyPromptModalProps> = ({ forceShow = false, onClose }) => {
|
||||
const { t } = useTranslation('survey');
|
||||
const dispatch = useAppDispatch();
|
||||
const [visible, setVisible] = useState(false);
|
||||
const [submitting, setSubmitting] = useState(false);
|
||||
const [surveyCompleted, setSurveyCompleted] = useState(false);
|
||||
const [surveyInfo, setSurveyInfo] = useState<{ id: string; questions: any[] } | null>(null);
|
||||
const { hasCompletedSurvey, loading, refetch } = useSurveyStatus();
|
||||
const themeMode = useAppSelector(state => state.themeReducer.mode);
|
||||
const surveyData = useAppSelector(state => state.accountSetupReducer.surveyData);
|
||||
const surveySubStep = useAppSelector(state => state.accountSetupReducer.surveySubStep);
|
||||
const isDarkMode = themeMode === 'dark';
|
||||
|
||||
useEffect(() => {
|
||||
// Check if survey modal is disabled via environment variable
|
||||
if (import.meta.env.VITE_ENABLE_SURVEY_MODAL !== 'true' && !forceShow) {
|
||||
return; // Don't show modal if disabled in environment
|
||||
}
|
||||
|
||||
// Check if survey was skipped recently (within 7 days)
|
||||
const skippedAt = localStorage.getItem('survey_skipped_at');
|
||||
if (!forceShow && skippedAt) {
|
||||
const skippedDate = new Date(skippedAt);
|
||||
const now = new Date();
|
||||
const diffDays = (now.getTime() - skippedDate.getTime()) / (1000 * 60 * 60 * 24);
|
||||
if (diffDays < 3) {
|
||||
return; // Don't show modal if skipped within 7 days
|
||||
}
|
||||
}
|
||||
|
||||
if (forceShow) {
|
||||
setVisible(true);
|
||||
dispatch(resetSurveyData());
|
||||
dispatch(setSurveySubStep(0));
|
||||
// Fetch survey info
|
||||
const fetchSurvey = async () => {
|
||||
try {
|
||||
const response = await surveyApiService.getAccountSetupSurvey();
|
||||
if (response.done && response.body) {
|
||||
setSurveyInfo({
|
||||
id: response.body.id,
|
||||
questions: response.body.questions || []
|
||||
});
|
||||
}
|
||||
} catch (error) {
|
||||
logger.error(t('survey:fetchErrorLog'), error);
|
||||
}
|
||||
};
|
||||
fetchSurvey();
|
||||
} else if (!loading && hasCompletedSurvey === false) {
|
||||
dispatch(resetSurveyData());
|
||||
dispatch(setSurveySubStep(0));
|
||||
// Fetch survey info
|
||||
const fetchSurvey = async () => {
|
||||
try {
|
||||
const response = await surveyApiService.getAccountSetupSurvey();
|
||||
if (response.done && response.body) {
|
||||
setSurveyInfo({
|
||||
id: response.body.id,
|
||||
questions: response.body.questions || []
|
||||
});
|
||||
}
|
||||
} catch (error) {
|
||||
logger.error(t('survey:fetchErrorLog'), error);
|
||||
}
|
||||
};
|
||||
fetchSurvey();
|
||||
// Show modal after a 5 second delay to not interrupt user immediately
|
||||
const timer = setTimeout(() => {
|
||||
setVisible(true);
|
||||
}, 5000);
|
||||
return () => clearTimeout(timer);
|
||||
}
|
||||
}, [loading, hasCompletedSurvey, dispatch, forceShow, t]);
|
||||
|
||||
const handleComplete = async () => {
|
||||
try {
|
||||
setSubmitting(true);
|
||||
|
||||
if (!surveyData || !surveyInfo) {
|
||||
throw new Error('Survey data not found');
|
||||
}
|
||||
|
||||
// Create a map of question keys to IDs
|
||||
const questionMap = surveyInfo.questions.reduce((acc, q) => {
|
||||
acc[q.question_key] = q.id;
|
||||
return acc;
|
||||
}, {} as Record<string, string>);
|
||||
|
||||
// Prepare submission data with actual question IDs - only include answered questions
|
||||
const answers: any[] = [];
|
||||
|
||||
if (surveyData.organization_type && questionMap['organization_type']) {
|
||||
answers.push({
|
||||
question_id: questionMap['organization_type'],
|
||||
answer_text: surveyData.organization_type
|
||||
});
|
||||
}
|
||||
|
||||
if (surveyData.user_role && questionMap['user_role']) {
|
||||
answers.push({
|
||||
question_id: questionMap['user_role'],
|
||||
answer_text: surveyData.user_role
|
||||
});
|
||||
}
|
||||
|
||||
if (surveyData.main_use_cases && surveyData.main_use_cases.length > 0 && questionMap['main_use_cases']) {
|
||||
answers.push({
|
||||
question_id: questionMap['main_use_cases'],
|
||||
answer_json: surveyData.main_use_cases
|
||||
});
|
||||
}
|
||||
|
||||
if (surveyData.previous_tools && questionMap['previous_tools']) {
|
||||
answers.push({
|
||||
question_id: questionMap['previous_tools'],
|
||||
answer_text: surveyData.previous_tools
|
||||
});
|
||||
}
|
||||
|
||||
if (surveyData.how_heard_about && questionMap['how_heard_about']) {
|
||||
answers.push({
|
||||
question_id: questionMap['how_heard_about'],
|
||||
answer_text: surveyData.how_heard_about
|
||||
});
|
||||
}
|
||||
|
||||
const submissionData: ISurveySubmissionRequest = {
|
||||
survey_id: surveyInfo.id,
|
||||
answers
|
||||
};
|
||||
|
||||
const response = await surveyApiService.submitSurveyResponse(submissionData);
|
||||
|
||||
if (response.done) {
|
||||
setSurveyCompleted(true);
|
||||
appMessage.success(t('survey:submitSuccessMessage'));
|
||||
|
||||
// Wait a moment before closing
|
||||
setTimeout(() => {
|
||||
setVisible(false);
|
||||
refetch(); // Update the survey status
|
||||
}, 2000);
|
||||
} else {
|
||||
throw new Error(response.message || t('survey:submitErrorMessage'));
|
||||
}
|
||||
} catch (error) {
|
||||
logger.error(t('survey:submitErrorLog'), error);
|
||||
appMessage.error(t('survey:submitErrorMessage'));
|
||||
} finally {
|
||||
setSubmitting(false);
|
||||
}
|
||||
};
|
||||
|
||||
const handleSkip = () => {
|
||||
setVisible(false);
|
||||
// Optionally, you can set a flag in localStorage to not show again for some time
|
||||
localStorage.setItem('survey_skipped_at', new Date().toISOString());
|
||||
onClose?.();
|
||||
};
|
||||
|
||||
const isCurrentStepValid = () => {
|
||||
switch (surveySubStep) {
|
||||
case 0:
|
||||
return surveyData.organization_type && surveyData.user_role;
|
||||
case 1:
|
||||
return surveyData.main_use_cases && surveyData.main_use_cases.length > 0;
|
||||
case 2:
|
||||
return surveyData.how_heard_about;
|
||||
default:
|
||||
return false;
|
||||
}
|
||||
};
|
||||
|
||||
const handleNext = () => {
|
||||
if (surveySubStep < 2) {
|
||||
dispatch(setSurveySubStep(surveySubStep + 1));
|
||||
} else {
|
||||
handleComplete();
|
||||
}
|
||||
};
|
||||
|
||||
const handlePrevious = () => {
|
||||
if (surveySubStep > 0) {
|
||||
dispatch(setSurveySubStep(surveySubStep - 1));
|
||||
}
|
||||
};
|
||||
|
||||
if (loading) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return (
|
||||
<Modal
|
||||
open={visible}
|
||||
title={surveyCompleted ? null : t('survey:modalTitle')}
|
||||
onCancel={handleSkip}
|
||||
footer={
|
||||
surveyCompleted ? null : (
|
||||
<Flex justify="space-between" align="center">
|
||||
<div>
|
||||
<Button onClick={handleSkip}>
|
||||
{t('survey:skip')}
|
||||
</Button>
|
||||
</div>
|
||||
<Flex gap={8}>
|
||||
{surveySubStep > 0 && (
|
||||
<Button onClick={handlePrevious}>
|
||||
{t('survey:previous')}
|
||||
</Button>
|
||||
)}
|
||||
<Button
|
||||
type="primary"
|
||||
onClick={handleNext}
|
||||
disabled={!isCurrentStepValid()}
|
||||
loading={submitting && surveySubStep === 2}
|
||||
>
|
||||
{surveySubStep === 2 ? t('survey:completeSurvey') : t('survey:next')}
|
||||
</Button>
|
||||
</Flex>
|
||||
</Flex>
|
||||
)
|
||||
}
|
||||
width={800}
|
||||
maskClosable={false}
|
||||
centered
|
||||
>
|
||||
{submitting ? (
|
||||
<div style={{ textAlign: 'center', padding: '40px' }}>
|
||||
<Spin size="large" />
|
||||
<p style={{ marginTop: 16 }}>{t('survey:submitting')}</p>
|
||||
</div>
|
||||
) : surveyCompleted ? (
|
||||
<Result
|
||||
status="success"
|
||||
title={t('survey:submitSuccessTitle')}
|
||||
subTitle={t('survey:submitSuccessSubtitle')}
|
||||
/>
|
||||
) : (
|
||||
<div style={{ maxHeight: '70vh', overflowY: 'auto' }}>
|
||||
<SurveyStep
|
||||
onEnter={() => {}} // Empty function since we handle navigation via buttons
|
||||
styles={{}}
|
||||
isDarkMode={isDarkMode}
|
||||
isModal={true} // Pass true to indicate modal context
|
||||
/>
|
||||
</div>
|
||||
)}
|
||||
</Modal>
|
||||
);
|
||||
};
|
||||
@@ -0,0 +1,73 @@
|
||||
import React, { useState } from 'react';
|
||||
import { Card, Button, Result, Alert } from '@/shared/antd-imports';
|
||||
import { CheckCircleOutlined, FormOutlined } from '@/shared/antd-imports';
|
||||
import { useSurveyStatus } from '@/hooks/useSurveyStatus';
|
||||
import { SurveyPromptModal } from './SurveyPromptModal';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
|
||||
export const SurveySettingsCard: React.FC = () => {
|
||||
const { t } = useTranslation('settings');
|
||||
const [showModal, setShowModal] = useState(false);
|
||||
const { hasCompletedSurvey, loading } = useSurveyStatus();
|
||||
|
||||
if (loading) {
|
||||
return (
|
||||
<Card loading={true} />
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<>
|
||||
<Card
|
||||
title={
|
||||
<span>
|
||||
<FormOutlined style={{ marginRight: 8 }} />
|
||||
Personalization Survey
|
||||
</span>
|
||||
}
|
||||
extra={
|
||||
hasCompletedSurvey && (
|
||||
<Button type="link" onClick={() => setShowModal(true)}>
|
||||
Update Responses
|
||||
</Button>
|
||||
)
|
||||
}
|
||||
>
|
||||
{hasCompletedSurvey ? (
|
||||
<Result
|
||||
icon={<CheckCircleOutlined style={{ color: '#52c41a' }} />}
|
||||
title="Survey Completed"
|
||||
subTitle="Thank you for completing the personalization survey. Your responses help us improve Worklenz."
|
||||
extra={
|
||||
<Button onClick={() => setShowModal(true)}>
|
||||
Update Your Responses
|
||||
</Button>
|
||||
}
|
||||
/>
|
||||
) : (
|
||||
<>
|
||||
<Alert
|
||||
message="Help us personalize your experience"
|
||||
description="Take a quick survey to tell us about your organization and how you use Worklenz."
|
||||
type="info"
|
||||
showIcon
|
||||
style={{ marginBottom: 16 }}
|
||||
/>
|
||||
<div style={{ textAlign: 'center' }}>
|
||||
<Button type="primary" size="large" onClick={() => setShowModal(true)}>
|
||||
Take Survey Now
|
||||
</Button>
|
||||
</div>
|
||||
</>
|
||||
)}
|
||||
</Card>
|
||||
|
||||
{showModal && (
|
||||
<SurveyPromptModal
|
||||
forceShow={true}
|
||||
onClose={() => setShowModal(false)}
|
||||
/>
|
||||
)}
|
||||
</>
|
||||
);
|
||||
};
|
||||
2
worklenz-frontend/src/components/survey/index.ts
Normal file
2
worklenz-frontend/src/components/survey/index.ts
Normal file
@@ -0,0 +1,2 @@
|
||||
export { SurveyPromptModal } from './SurveyPromptModal';
|
||||
export { SurveySettingsCard } from './SurveySettingsCard';
|
||||
@@ -12,7 +12,7 @@ import TimeLogList from './time-log-list';
|
||||
import { taskTimeLogsApiService } from '@/api/tasks/task-time-logs.api.service';
|
||||
import { ITaskLogViewModel } from '@/types/tasks/task-log-view.types';
|
||||
import TaskTimer from '@/components/taskListCommon/task-timer/task-timer';
|
||||
import { useTaskTimer } from '@/hooks/useTaskTimer';
|
||||
import { useTaskTimerWithConflictCheck } from '@/hooks/useTaskTimerWithConflictCheck';
|
||||
import logger from '@/utils/errorLogger';
|
||||
|
||||
interface TaskDrawerTimeLogProps {
|
||||
@@ -31,7 +31,7 @@ const TaskDrawerTimeLog = ({ t, refreshTrigger = 0 }: TaskDrawerTimeLogProps) =>
|
||||
state => state.taskDrawerReducer
|
||||
);
|
||||
|
||||
const { started, timeString, handleStartTimer, handleStopTimer } = useTaskTimer(
|
||||
const { started, timeString, handleStartTimer, handleStopTimer } = useTaskTimerWithConflictCheck(
|
||||
selectedTaskId || '',
|
||||
taskFormViewModel?.task?.timer_start_time || null
|
||||
);
|
||||
|
||||
@@ -3,7 +3,7 @@ import { Button, Divider, Flex, Popconfirm, Typography, Space } from '@/shared/a
|
||||
import { colors } from '@/styles/colors';
|
||||
import { ITaskLogViewModel } from '@/types/tasks/task-log-view.types';
|
||||
import SingleAvatar from '@/components/common/single-avatar/single-avatar';
|
||||
import { formatDateTimeWithLocale } from '@/utils/format-date-time-with-locale';
|
||||
import { formatDateTimeWithUserTimezone } from '@/utils/format-date-time-with-user-timezone';
|
||||
import { calculateTimeGap } from '@/utils/calculate-time-gap';
|
||||
import './time-log-item.css';
|
||||
import { taskTimeLogsApiService } from '@/api/tasks/task-time-logs.api.service';
|
||||
@@ -101,7 +101,7 @@ const TimeLogItem = ({ log, onDelete }: TimeLogItemProps) => {
|
||||
{renderLoggedByTimer()} {calculateTimeGap(created_at || '')}
|
||||
</Typography.Text>
|
||||
<Typography.Text type="secondary" style={{ fontSize: 12 }}>
|
||||
{formatDateTimeWithLocale(created_at || '')}
|
||||
{formatDateTimeWithUserTimezone(created_at || '', currentSession?.timezone_name)}
|
||||
</Typography.Text>
|
||||
</Flex>
|
||||
{renderActionButtons()}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
import React from 'react';
|
||||
import TaskTimer from '@/components/taskListCommon/task-timer/task-timer';
|
||||
import { useTaskTimer } from '@/hooks/useTaskTimer';
|
||||
import { useTaskTimerWithConflictCheck } from '@/hooks/useTaskTimerWithConflictCheck';
|
||||
|
||||
interface TaskTimeTrackingProps {
|
||||
taskId: string;
|
||||
@@ -8,7 +8,7 @@ interface TaskTimeTrackingProps {
|
||||
}
|
||||
|
||||
const TaskTimeTracking: React.FC<TaskTimeTrackingProps> = React.memo(({ taskId, isDarkMode }) => {
|
||||
const { started, timeString, handleStartTimer, handleStopTimer } = useTaskTimer(
|
||||
const { started, timeString, handleStartTimer, handleStopTimer } = useTaskTimerWithConflictCheck(
|
||||
taskId,
|
||||
null // The hook will get the timer start time from Redux
|
||||
);
|
||||
|
||||
@@ -32,6 +32,8 @@ import {
|
||||
RetweetOutlined,
|
||||
UserAddOutlined,
|
||||
LoadingOutlined,
|
||||
CopyOutlined,
|
||||
message,
|
||||
} from '@/shared/antd-imports';
|
||||
|
||||
interface TaskContextMenuProps {
|
||||
@@ -340,6 +342,21 @@ const TaskContextMenu: React.FC<TaskContextMenuProps> = ({
|
||||
}
|
||||
}, [task?.id, projectId, dispatch, onClose]);
|
||||
|
||||
const handleCopyLink = useCallback(async () => {
|
||||
if (!projectId || !task.id) return;
|
||||
|
||||
try {
|
||||
const taskLink = `${window.location.origin}/worklenz/projects/${projectId}?tab=tasks-list&pinned_tab=tasks-list&task=${task.id}`;
|
||||
await navigator.clipboard.writeText(taskLink);
|
||||
message.success(t('contextMenu.linkCopied'));
|
||||
} catch (error) {
|
||||
logger.error('Error copying link:', error);
|
||||
message.error(t('contextMenu.linkCopyFailed'));
|
||||
} finally {
|
||||
onClose();
|
||||
}
|
||||
}, [projectId, task.id, onClose, t]);
|
||||
|
||||
const menuItems = useMemo(() => {
|
||||
const items = [
|
||||
{
|
||||
@@ -359,6 +376,18 @@ const TaskContextMenu: React.FC<TaskContextMenuProps> = ({
|
||||
</button>
|
||||
),
|
||||
},
|
||||
{
|
||||
key: 'copyLink',
|
||||
label: (
|
||||
<button
|
||||
onClick={handleCopyLink}
|
||||
className="flex items-center gap-2 px-4 py-2 text-sm text-gray-700 dark:text-gray-300 hover:bg-gray-100 dark:hover:bg-gray-700 w-full text-left"
|
||||
>
|
||||
<CopyOutlined className="text-gray-500 dark:text-gray-400" />
|
||||
<span>{t('contextMenu.copyLink')}</span>
|
||||
</button>
|
||||
),
|
||||
},
|
||||
];
|
||||
|
||||
// Add Move To submenu if there are options
|
||||
@@ -516,6 +545,7 @@ const TaskContextMenu: React.FC<TaskContextMenuProps> = ({
|
||||
handleArchive,
|
||||
handleDelete,
|
||||
handleConvertToTask,
|
||||
handleCopyLink,
|
||||
getMoveToOptions,
|
||||
dispatch,
|
||||
t,
|
||||
|
||||
@@ -395,7 +395,7 @@ export const ReporterColumn: React.FC<ReporterColumnProps> = memo(({ width, repo
|
||||
style={{ width }}
|
||||
>
|
||||
{reporter ? (
|
||||
<span className="text-sm text-gray-500 dark:text-gray-400 whitespace-nowrap">{reporter}</span>
|
||||
<span className="text-sm text-gray-500 dark:text-gray-400 truncate" title={reporter}>{reporter}</span>
|
||||
) : (
|
||||
<span className="text-sm text-gray-400 dark:text-gray-500 whitespace-nowrap">-</span>
|
||||
)}
|
||||
|
||||
@@ -14,6 +14,8 @@ import {
|
||||
EyeOutlined,
|
||||
InboxOutlined,
|
||||
CheckOutlined,
|
||||
SortAscendingOutlined,
|
||||
SortDescendingOutlined,
|
||||
} from '@/shared/antd-imports';
|
||||
import { RootState } from '@/app/store';
|
||||
import { useAppSelector } from '@/hooks/useAppSelector';
|
||||
@@ -33,6 +35,12 @@ import {
|
||||
setArchived as setTaskManagementArchived,
|
||||
toggleArchived as toggleTaskManagementArchived,
|
||||
selectArchived,
|
||||
setSort,
|
||||
setSortField,
|
||||
setSortOrder,
|
||||
selectSort,
|
||||
selectSortField,
|
||||
selectSortOrder,
|
||||
} from '@/features/task-management/task-management.slice';
|
||||
import {
|
||||
setCurrentGrouping,
|
||||
@@ -47,11 +55,13 @@ import {
|
||||
setLabels,
|
||||
setSearch,
|
||||
setPriorities,
|
||||
setFields,
|
||||
} from '@/features/tasks/tasks.slice';
|
||||
import { getTeamMembers } from '@/features/team-members/team-members.slice';
|
||||
import { ITaskPriority } from '@/types/tasks/taskPriority.types';
|
||||
import { ITaskListColumn } from '@/types/tasks/taskList.types';
|
||||
import { IGroupBy } from '@/features/tasks/tasks.slice';
|
||||
import { ITaskListSortableColumn } from '@/types/tasks/taskListFilters.types';
|
||||
// --- Enhanced Kanban imports ---
|
||||
import {
|
||||
setGroupBy as setKanbanGroupBy,
|
||||
@@ -87,7 +97,11 @@ const FILTER_DEBOUNCE_DELAY = 300; // ms
|
||||
const SEARCH_DEBOUNCE_DELAY = 500; // ms
|
||||
const MAX_FILTER_OPTIONS = 100;
|
||||
|
||||
// Limit options to prevent UI lag
|
||||
// Sort order enum
|
||||
enum SORT_ORDER {
|
||||
ASCEND = 'ascend',
|
||||
DESCEND = 'descend',
|
||||
}
|
||||
|
||||
// Optimized selectors with proper transformation logic
|
||||
const selectFilterData = createSelector(
|
||||
@@ -741,6 +755,192 @@ const SearchFilter: React.FC<{
|
||||
);
|
||||
};
|
||||
|
||||
// Sort Dropdown Component - Simplified version using task-management slice
|
||||
const SortDropdown: React.FC<{ themeClasses: any; isDarkMode: boolean }> = ({
|
||||
themeClasses,
|
||||
isDarkMode,
|
||||
}) => {
|
||||
const { t } = useTranslation('task-list-filters');
|
||||
const dispatch = useAppDispatch();
|
||||
const { projectId } = useAppSelector(state => state.projectReducer);
|
||||
|
||||
// Get current sort state from task-management slice
|
||||
const currentSortField = useAppSelector(selectSortField);
|
||||
const currentSortOrder = useAppSelector(selectSortOrder);
|
||||
|
||||
const [open, setOpen] = React.useState(false);
|
||||
const dropdownRef = useRef<HTMLDivElement>(null);
|
||||
|
||||
// Close dropdown on outside click
|
||||
React.useEffect(() => {
|
||||
if (!open) return;
|
||||
const handleClick = (e: MouseEvent) => {
|
||||
if (dropdownRef.current && !dropdownRef.current.contains(e.target as Node)) {
|
||||
setOpen(false);
|
||||
}
|
||||
};
|
||||
document.addEventListener('mousedown', handleClick);
|
||||
return () => document.removeEventListener('mousedown', handleClick);
|
||||
}, [open]);
|
||||
|
||||
const sortFieldsList = [
|
||||
{ label: t('taskText'), key: 'name' },
|
||||
{ label: t('statusText'), key: 'status' },
|
||||
{ label: t('priorityText'), key: 'priority' },
|
||||
{ label: t('startDateText'), key: 'start_date' },
|
||||
{ label: t('endDateText'), key: 'end_date' },
|
||||
{ label: t('completedDateText'), key: 'completed_at' },
|
||||
{ label: t('createdDateText'), key: 'created_at' },
|
||||
{ label: t('lastUpdatedText'), key: 'updated_at' },
|
||||
];
|
||||
|
||||
const handleSortFieldChange = (fieldKey: string) => {
|
||||
// If clicking the same field, toggle order, otherwise set new field with ASC
|
||||
if (currentSortField === fieldKey) {
|
||||
const newOrder = currentSortOrder === 'ASC' ? 'DESC' : 'ASC';
|
||||
dispatch(setSort({ field: fieldKey, order: newOrder }));
|
||||
} else {
|
||||
dispatch(setSort({ field: fieldKey, order: 'ASC' }));
|
||||
}
|
||||
|
||||
// Fetch updated tasks
|
||||
if (projectId) {
|
||||
dispatch(fetchTasksV3(projectId));
|
||||
}
|
||||
|
||||
setOpen(false);
|
||||
};
|
||||
|
||||
const clearSort = () => {
|
||||
dispatch(setSort({ field: '', order: 'ASC' }));
|
||||
if (projectId) {
|
||||
dispatch(fetchTasksV3(projectId));
|
||||
}
|
||||
};
|
||||
|
||||
const isActive = currentSortField !== '';
|
||||
const currentFieldLabel = sortFieldsList.find(f => f.key === currentSortField)?.label;
|
||||
const orderText = currentSortOrder === 'ASC' ? t('ascendingOrder') : t('descendingOrder');
|
||||
|
||||
return (
|
||||
<div className="relative" ref={dropdownRef}>
|
||||
{/* Trigger Button - matching FilterDropdown style */}
|
||||
<button
|
||||
onClick={() => setOpen(!open)}
|
||||
title={
|
||||
isActive
|
||||
? t('currentSort', { field: currentFieldLabel, order: orderText })
|
||||
: t('sortText')
|
||||
}
|
||||
className={`
|
||||
inline-flex items-center gap-1.5 px-2.5 py-1.5 text-xs font-medium rounded-md
|
||||
border transition-all duration-200 ease-in-out
|
||||
${
|
||||
isActive
|
||||
? isDarkMode
|
||||
? 'bg-gray-600 text-white border-gray-500'
|
||||
: 'bg-gray-200 text-gray-800 border-gray-300 font-semibold'
|
||||
: `${themeClasses.buttonBg} ${themeClasses.buttonBorder} ${themeClasses.buttonText}`
|
||||
}
|
||||
hover:shadow-sm focus:outline-none focus:ring-2 focus:ring-gray-500 focus:ring-offset-2
|
||||
${isDarkMode ? 'focus:ring-offset-gray-900' : 'focus:ring-offset-white'}
|
||||
`}
|
||||
aria-expanded={open}
|
||||
aria-haspopup="true"
|
||||
>
|
||||
{currentSortOrder === 'ASC' ? (
|
||||
<SortAscendingOutlined className="w-3.5 h-3.5" />
|
||||
) : (
|
||||
<SortDescendingOutlined className="w-3.5 h-3.5" />
|
||||
)}
|
||||
<span className="hidden sm:inline">{t('sortText')}</span>
|
||||
{isActive && currentFieldLabel && (
|
||||
<span className={`text-xs ${isDarkMode ? 'text-gray-300' : 'text-gray-600'} max-w-16 truncate hidden md:inline`}>
|
||||
{currentFieldLabel}
|
||||
</span>
|
||||
)}
|
||||
<DownOutlined
|
||||
className={`w-3.5 h-3.5 transition-transform duration-200 ${open ? 'rotate-180' : ''}`}
|
||||
/>
|
||||
</button>
|
||||
|
||||
{/* Dropdown Panel - matching FilterDropdown style */}
|
||||
{open && (
|
||||
<div
|
||||
className={`absolute top-full left-0 z-50 mt-1 w-64 ${themeClasses.dropdownBg} rounded-md shadow-sm border ${themeClasses.dropdownBorder}`}
|
||||
>
|
||||
{/* Clear Sort Option */}
|
||||
{isActive && (
|
||||
<div className={`p-2 border-b ${themeClasses.dividerBorder}`}>
|
||||
<button
|
||||
onClick={clearSort}
|
||||
className={`w-full text-left px-2 py-1.5 text-xs rounded transition-colors duration-150 ${themeClasses.optionText} ${themeClasses.optionHover}`}
|
||||
>
|
||||
{t('clearSort')}
|
||||
</button>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{/* Options List */}
|
||||
<div className="max-h-48 overflow-y-auto">
|
||||
<div className="p-0.5">
|
||||
{sortFieldsList.map(sortField => {
|
||||
const isSelected = currentSortField === sortField.key;
|
||||
|
||||
return (
|
||||
<button
|
||||
key={sortField.key}
|
||||
onClick={() => handleSortFieldChange(sortField.key)}
|
||||
className={`
|
||||
w-full flex items-center justify-between gap-2 px-2 py-1.5 text-xs rounded
|
||||
transition-colors duration-150 text-left
|
||||
${
|
||||
isSelected
|
||||
? isDarkMode
|
||||
? 'bg-gray-600 text-white'
|
||||
: 'bg-gray-200 text-gray-800 font-semibold'
|
||||
: `${themeClasses.optionText} ${themeClasses.optionHover}`
|
||||
}
|
||||
`}
|
||||
title={
|
||||
isSelected
|
||||
? t('currentSort', {
|
||||
field: sortField.label,
|
||||
order: orderText
|
||||
}) + ` - ${t('sortDescending')}`
|
||||
: t('sortByField', { field: sortField.label }) + ` - ${t('sortAscending')}`
|
||||
}
|
||||
>
|
||||
<div className="flex items-center gap-2">
|
||||
<span className="truncate">{sortField.label}</span>
|
||||
{isSelected && (
|
||||
<span className={`text-xs ${isDarkMode ? 'text-gray-300' : 'text-gray-600'}`}>
|
||||
({orderText})
|
||||
</span>
|
||||
)}
|
||||
</div>
|
||||
<div className="flex items-center gap-1">
|
||||
{isSelected ? (
|
||||
currentSortOrder === 'ASC' ? (
|
||||
<SortAscendingOutlined className="w-3.5 h-3.5" />
|
||||
) : (
|
||||
<SortDescendingOutlined className="w-3.5 h-3.5" />
|
||||
)
|
||||
) : (
|
||||
<SortAscendingOutlined className="w-3.5 h-3.5 opacity-50" />
|
||||
)}
|
||||
</div>
|
||||
</button>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
const LOCAL_STORAGE_KEY = 'worklenz.taskManagement.fields';
|
||||
|
||||
const FieldsDropdown: React.FC<{ themeClasses: any; isDarkMode: boolean }> = ({
|
||||
@@ -1056,14 +1256,20 @@ const ImprovedTaskFilters: React.FC<ImprovedTaskFiltersProps> = ({ position, cla
|
||||
};
|
||||
}, [dispatch, projectView]);
|
||||
|
||||
// Get sort fields for active count calculation
|
||||
const sortFields = useAppSelector(state => state.taskReducer.fields);
|
||||
const taskManagementSortField = useAppSelector(selectSortField);
|
||||
|
||||
// Calculate active filters count - memoized to prevent unnecessary recalculations
|
||||
const calculatedActiveFiltersCount = useMemo(() => {
|
||||
const count = filterSections.reduce(
|
||||
(acc, section) => (section.id === 'groupBy' ? acc : acc + section.selectedValues.length),
|
||||
0
|
||||
);
|
||||
return count + (searchValue ? 1 : 0);
|
||||
}, [filterSections, searchValue]);
|
||||
const sortFieldsCount = position === 'list' ? sortFields.length : 0;
|
||||
const taskManagementSortCount = position === 'list' && taskManagementSortField ? 1 : 0;
|
||||
return count + (searchValue ? 1 : 0) + sortFieldsCount + taskManagementSortCount;
|
||||
}, [filterSections, searchValue, sortFields, taskManagementSortField, position]);
|
||||
|
||||
useEffect(() => {
|
||||
if (activeFiltersCount !== calculatedActiveFiltersCount) {
|
||||
@@ -1237,6 +1443,12 @@ const ImprovedTaskFilters: React.FC<ImprovedTaskFiltersProps> = ({ position, cla
|
||||
// Clear priority filters
|
||||
dispatch(setPriorities([]));
|
||||
|
||||
// Clear sort fields
|
||||
dispatch(setFields([]));
|
||||
|
||||
// Clear sort from task-management slice
|
||||
dispatch(setSort({ field: '', order: 'ASC' }));
|
||||
|
||||
// Clear archived state based on position
|
||||
if (position === 'list') {
|
||||
dispatch(setTaskManagementArchived(false));
|
||||
@@ -1282,9 +1494,9 @@ const ImprovedTaskFilters: React.FC<ImprovedTaskFiltersProps> = ({ position, cla
|
||||
<div
|
||||
className={`${themeClasses.containerBg} border ${themeClasses.containerBorder} rounded-md p-1.5 shadow-sm ${className}`}
|
||||
>
|
||||
<div className="flex flex-wrap items-center gap-2">
|
||||
<div className="flex flex-wrap items-center justify-between gap-2 min-h-[36px]">
|
||||
{/* Left Section - Main Filters */}
|
||||
<div className="flex flex-wrap items-center gap-2">
|
||||
<div className="flex flex-wrap items-center gap-2 flex-1 min-w-0">
|
||||
{/* Search */}
|
||||
<SearchFilter
|
||||
value={searchValue}
|
||||
@@ -1293,6 +1505,11 @@ const ImprovedTaskFilters: React.FC<ImprovedTaskFiltersProps> = ({ position, cla
|
||||
themeClasses={themeClasses}
|
||||
/>
|
||||
|
||||
{/* Sort Filter Button (for list view) - appears after search */}
|
||||
{position === 'list' && (
|
||||
<SortDropdown themeClasses={themeClasses} isDarkMode={isDarkMode} />
|
||||
)}
|
||||
|
||||
{/* Filter Dropdowns - Only render when data is loaded */}
|
||||
{isDataLoaded ? (
|
||||
filterSectionsData.map(section => (
|
||||
@@ -1322,7 +1539,7 @@ const ImprovedTaskFilters: React.FC<ImprovedTaskFiltersProps> = ({ position, cla
|
||||
</div>
|
||||
|
||||
{/* Right Section - Additional Controls */}
|
||||
<div className="flex items-center gap-2 ml-auto">
|
||||
<div className="flex flex-wrap items-center gap-2 ml-auto min-w-0 shrink-0">
|
||||
{/* Active Filters Indicator */}
|
||||
{activeFiltersCount > 0 && (
|
||||
<div className="flex items-center gap-1.5">
|
||||
|
||||
@@ -1,4 +1,5 @@
|
||||
import { createSlice, PayloadAction } from '@reduxjs/toolkit';
|
||||
import { IAccountSetupSurveyData } from '@/types/account-setup/survey.types';
|
||||
|
||||
interface Task {
|
||||
id: number;
|
||||
@@ -17,6 +18,8 @@ interface AccountSetupState {
|
||||
tasks: Task[];
|
||||
teamMembers: Email[];
|
||||
currentStep: number;
|
||||
surveyData: IAccountSetupSurveyData;
|
||||
surveySubStep: number;
|
||||
}
|
||||
|
||||
const initialState: AccountSetupState = {
|
||||
@@ -26,6 +29,8 @@ const initialState: AccountSetupState = {
|
||||
tasks: [{ id: 0, value: '' }],
|
||||
teamMembers: [{ id: 0, value: '' }],
|
||||
currentStep: 0,
|
||||
surveyData: {},
|
||||
surveySubStep: 0,
|
||||
};
|
||||
|
||||
const accountSetupSlice = createSlice({
|
||||
@@ -50,6 +55,16 @@ const accountSetupSlice = createSlice({
|
||||
setCurrentStep: (state, action: PayloadAction<number>) => {
|
||||
state.currentStep = action.payload;
|
||||
},
|
||||
setSurveyData: (state, action: PayloadAction<Partial<IAccountSetupSurveyData>>) => {
|
||||
state.surveyData = { ...state.surveyData, ...action.payload };
|
||||
},
|
||||
setSurveySubStep: (state, action: PayloadAction<number>) => {
|
||||
state.surveySubStep = action.payload;
|
||||
},
|
||||
resetSurveyData: (state) => {
|
||||
state.surveyData = {};
|
||||
state.surveySubStep = 0;
|
||||
},
|
||||
resetAccountSetup: () => initialState,
|
||||
},
|
||||
});
|
||||
@@ -61,6 +76,9 @@ export const {
|
||||
setTasks,
|
||||
setTeamMembers,
|
||||
setCurrentStep,
|
||||
setSurveyData,
|
||||
setSurveySubStep,
|
||||
resetSurveyData,
|
||||
resetAccountSetup,
|
||||
} = accountSetupSlice.actions;
|
||||
|
||||
|
||||
@@ -575,7 +575,6 @@ const enhancedKanbanSlice = createSlice({
|
||||
action: PayloadAction<ITaskListPriorityChangeResponse>
|
||||
) => {
|
||||
const { id, priority_id, color_code, color_code_dark } = action.payload;
|
||||
|
||||
// Find the task in any group
|
||||
const taskInfo = findTaskInAllGroups(state.taskGroups, id);
|
||||
if (!taskInfo || !priority_id) return;
|
||||
@@ -603,7 +602,6 @@ const enhancedKanbanSlice = createSlice({
|
||||
// Update cache
|
||||
state.taskCache[id] = task;
|
||||
},
|
||||
|
||||
// Enhanced Kanban assignee update (for use in task drawer dropdown)
|
||||
updateEnhancedKanbanTaskAssignees: (
|
||||
state,
|
||||
|
||||
@@ -0,0 +1,59 @@
|
||||
import { createSlice, PayloadAction } from '@reduxjs/toolkit';
|
||||
import { ActivityFeedType } from '@/types/home/user-activity.types';
|
||||
|
||||
interface ActivityItem {
|
||||
id: string;
|
||||
type: string;
|
||||
description: string;
|
||||
timestamp: string;
|
||||
}
|
||||
|
||||
interface UserActivityState {
|
||||
activeTab: ActivityFeedType;
|
||||
activities: ActivityItem[];
|
||||
loading: boolean;
|
||||
error: string | null;
|
||||
}
|
||||
|
||||
const initialState: UserActivityState = {
|
||||
activeTab: ActivityFeedType.TIME_LOGGED_TASKS,
|
||||
activities: [],
|
||||
loading: false,
|
||||
error: null,
|
||||
};
|
||||
|
||||
const userActivitySlice = createSlice({
|
||||
name: 'userActivity',
|
||||
initialState,
|
||||
reducers: {
|
||||
setActiveTab(state, action: PayloadAction<ActivityFeedType>) {
|
||||
state.activeTab = action.payload;
|
||||
},
|
||||
fetchActivitiesStart(state) {
|
||||
state.loading = true;
|
||||
state.error = null;
|
||||
},
|
||||
fetchActivitiesSuccess(state, action: PayloadAction<ActivityItem[]>) {
|
||||
state.activities = action.payload;
|
||||
state.loading = false;
|
||||
state.error = null;
|
||||
},
|
||||
fetchActivitiesFailure(state, action: PayloadAction<string>) {
|
||||
state.loading = false;
|
||||
state.error = action.payload;
|
||||
},
|
||||
clearActivities(state) {
|
||||
state.activities = [];
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
export const {
|
||||
setActiveTab,
|
||||
fetchActivitiesStart,
|
||||
fetchActivitiesSuccess,
|
||||
fetchActivitiesFailure,
|
||||
clearActivities,
|
||||
} = userActivitySlice.actions;
|
||||
|
||||
export default userActivitySlice.reducer;
|
||||
@@ -153,8 +153,8 @@ const Navbar = () => {
|
||||
<Flex align="center">
|
||||
<SwitchTeamButton />
|
||||
<NotificationButton />
|
||||
{/* <TimerButton /> */}
|
||||
<HelpButton />
|
||||
<TimerButton />
|
||||
{/* <HelpButton /> */}
|
||||
<ProfileButton isOwnerOrAdmin={isOwnerOrAdmin} />
|
||||
</Flex>
|
||||
</Flex>
|
||||
|
||||
@@ -17,7 +17,7 @@ import { useAppDispatch } from '@/hooks/useAppDispatch';
|
||||
import { useSocket } from '@/socket/socketContext';
|
||||
import { SocketEvents } from '@/shared/socket-events';
|
||||
import { updateTaskTimeTracking } from '@/features/tasks/tasks.slice';
|
||||
import moment from 'moment';
|
||||
import { format, differenceInSeconds, isValid, parseISO } from 'date-fns';
|
||||
|
||||
const { Text } = Typography;
|
||||
const { useToken } = theme;
|
||||
@@ -70,17 +70,17 @@ const TimerButton = () => {
|
||||
try {
|
||||
if (!timer || !timer.task_id || !timer.start_time) return;
|
||||
|
||||
const startTime = moment(timer.start_time);
|
||||
if (!startTime.isValid()) {
|
||||
const startTime = parseISO(timer.start_time);
|
||||
if (!isValid(startTime)) {
|
||||
logError(`Invalid start time for timer ${timer.task_id}: ${timer.start_time}`);
|
||||
return;
|
||||
}
|
||||
|
||||
const now = moment();
|
||||
const duration = moment.duration(now.diff(startTime));
|
||||
const hours = Math.floor(duration.asHours());
|
||||
const minutes = duration.minutes();
|
||||
const seconds = duration.seconds();
|
||||
const now = new Date();
|
||||
const totalSeconds = differenceInSeconds(now, startTime);
|
||||
const hours = Math.floor(totalSeconds / 3600);
|
||||
const minutes = Math.floor((totalSeconds % 3600) / 60);
|
||||
const seconds = totalSeconds % 60;
|
||||
newTimes[timer.task_id] =
|
||||
`${hours.toString().padStart(2, '0')}:${minutes.toString().padStart(2, '0')}:${seconds.toString().padStart(2, '0')}`;
|
||||
} catch (error) {
|
||||
@@ -96,12 +96,7 @@ const TimerButton = () => {
|
||||
useEffect(() => {
|
||||
fetchRunningTimers();
|
||||
|
||||
// Set up polling to refresh timers every 30 seconds
|
||||
const pollInterval = setInterval(() => {
|
||||
fetchRunningTimers();
|
||||
}, 30000);
|
||||
|
||||
return () => clearInterval(pollInterval);
|
||||
// Removed periodic polling - rely on socket events for real-time updates
|
||||
}, [fetchRunningTimers]);
|
||||
|
||||
useEffect(() => {
|
||||
@@ -283,7 +278,7 @@ const TimerButton = () => {
|
||||
<Text type="secondary" style={{ fontSize: 11 }}>
|
||||
Started:{' '}
|
||||
{timer.start_time
|
||||
? moment(timer.start_time).format('HH:mm')
|
||||
? format(parseISO(timer.start_time), 'HH:mm')
|
||||
: '--:--'}
|
||||
</Text>
|
||||
<Text
|
||||
|
||||
@@ -8,6 +8,7 @@ interface ProjectMembersState {
|
||||
currentMembersList: IMentionMemberViewModel[];
|
||||
isDrawerOpen: boolean;
|
||||
isLoading: boolean;
|
||||
isFromAssigner: boolean;
|
||||
error: string | null;
|
||||
}
|
||||
|
||||
@@ -16,6 +17,7 @@ const initialState: ProjectMembersState = {
|
||||
currentMembersList: [],
|
||||
isDrawerOpen: false,
|
||||
isLoading: false,
|
||||
isFromAssigner: false,
|
||||
error: null,
|
||||
};
|
||||
|
||||
@@ -89,6 +91,12 @@ const projectMembersSlice = createSlice({
|
||||
reducers: {
|
||||
toggleProjectMemberDrawer: state => {
|
||||
state.isDrawerOpen = !state.isDrawerOpen;
|
||||
if (state.isDrawerOpen === false) {
|
||||
state.isFromAssigner = false;
|
||||
}
|
||||
},
|
||||
setIsFromAssigner: (state, action: PayloadAction<boolean>) => {
|
||||
state.isFromAssigner = action.payload;
|
||||
},
|
||||
},
|
||||
extraReducers: builder => {
|
||||
@@ -139,7 +147,7 @@ const projectMembersSlice = createSlice({
|
||||
},
|
||||
});
|
||||
|
||||
export const { toggleProjectMemberDrawer } = projectMembersSlice.actions;
|
||||
export const { toggleProjectMemberDrawer, setIsFromAssigner } = projectMembersSlice.actions;
|
||||
export {
|
||||
getProjectMembers,
|
||||
getAllProjectMembers,
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
import { Drawer, Typography, Flex, Button, Dropdown } from '@/shared/antd-imports';
|
||||
import React, { useState } from 'react';
|
||||
import React, { useState, useCallback } from 'react';
|
||||
import { useAppSelector } from '../../../../hooks/useAppSelector';
|
||||
import { useAppDispatch } from '../../../../hooks/useAppDispatch';
|
||||
import { setSelectedProject, toggleProjectReportsDrawer } from '../project-reports-slice';
|
||||
@@ -8,6 +8,8 @@ import ProjectReportsDrawerTabs from './ProjectReportsDrawerTabs';
|
||||
import { colors } from '../../../../styles/colors';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { IRPTProject } from '@/types/reporting/reporting.types';
|
||||
import { useAuthService } from '../../../../hooks/useAuth';
|
||||
import { reportingExportApiService } from '@/api/reporting/reporting-export.api.service';
|
||||
|
||||
type ProjectReportsDrawerProps = {
|
||||
selectedProject: IRPTProject | null;
|
||||
@@ -17,6 +19,8 @@ const ProjectReportsDrawer = ({ selectedProject }: ProjectReportsDrawerProps) =>
|
||||
const { t } = useTranslation('reporting-projects-drawer');
|
||||
|
||||
const dispatch = useAppDispatch();
|
||||
const currentSession = useAuthService().getCurrentSession();
|
||||
const [exporting, setExporting] = useState<boolean>(false);
|
||||
|
||||
// get drawer open state and project list from the reducer
|
||||
const isDrawerOpen = useAppSelector(
|
||||
@@ -35,6 +39,54 @@ const ProjectReportsDrawer = ({ selectedProject }: ProjectReportsDrawerProps) =>
|
||||
}
|
||||
};
|
||||
|
||||
// Export handlers
|
||||
const handleExportMembers = useCallback(() => {
|
||||
if (!selectedProject?.id) return;
|
||||
try {
|
||||
setExporting(true);
|
||||
const teamName = currentSession?.team_name || 'Team';
|
||||
reportingExportApiService.exportProjectMembers(
|
||||
selectedProject.id,
|
||||
selectedProject.name,
|
||||
teamName
|
||||
);
|
||||
} catch (error) {
|
||||
console.error('Error exporting project members:', error);
|
||||
} finally {
|
||||
setExporting(false);
|
||||
}
|
||||
}, [selectedProject, currentSession?.team_name]);
|
||||
|
||||
const handleExportTasks = useCallback(() => {
|
||||
if (!selectedProject?.id) return;
|
||||
try {
|
||||
setExporting(true);
|
||||
const teamName = currentSession?.team_name || 'Team';
|
||||
reportingExportApiService.exportProjectTasks(
|
||||
selectedProject.id,
|
||||
selectedProject.name,
|
||||
teamName
|
||||
);
|
||||
} catch (error) {
|
||||
console.error('Error exporting project tasks:', error);
|
||||
} finally {
|
||||
setExporting(false);
|
||||
}
|
||||
}, [selectedProject, currentSession?.team_name]);
|
||||
|
||||
const handleExportClick = useCallback((key: string) => {
|
||||
switch (key) {
|
||||
case '1':
|
||||
handleExportMembers();
|
||||
break;
|
||||
case '2':
|
||||
handleExportTasks();
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}, [handleExportMembers, handleExportTasks]);
|
||||
|
||||
return (
|
||||
<Drawer
|
||||
open={isDrawerOpen}
|
||||
@@ -56,9 +108,15 @@ const ProjectReportsDrawer = ({ selectedProject }: ProjectReportsDrawerProps) =>
|
||||
{ key: '1', label: t('membersButton') },
|
||||
{ key: '2', label: t('tasksButton') },
|
||||
],
|
||||
onClick: ({ key }) => handleExportClick(key),
|
||||
}}
|
||||
>
|
||||
<Button type="primary" icon={<DownOutlined />} iconPosition="end">
|
||||
<Button
|
||||
type="primary"
|
||||
loading={exporting}
|
||||
icon={<DownOutlined />}
|
||||
iconPosition="end"
|
||||
>
|
||||
{t('exportButton')}
|
||||
</Button>
|
||||
</Dropdown>
|
||||
|
||||
@@ -70,6 +70,9 @@ const initialState: TaskManagementState = {
|
||||
loadingColumns: false,
|
||||
columns: [],
|
||||
customColumns: [],
|
||||
// Add sort-related state
|
||||
sortField: '',
|
||||
sortOrder: 'ASC',
|
||||
};
|
||||
|
||||
// Async thunk to fetch tasks from API
|
||||
@@ -239,12 +242,16 @@ export const fetchTasksV3 = createAsyncThunk(
|
||||
// Get archived state from task management slice
|
||||
const archivedState = state.taskManagement.archived;
|
||||
|
||||
// Get sort state from task management slice
|
||||
const sortField = state.taskManagement.sortField;
|
||||
const sortOrder = state.taskManagement.sortOrder;
|
||||
|
||||
const config: ITaskListConfigV2 = {
|
||||
id: projectId,
|
||||
archived: archivedState,
|
||||
group: currentGrouping || '',
|
||||
field: '',
|
||||
order: '',
|
||||
field: sortField,
|
||||
order: sortOrder,
|
||||
search: searchValue,
|
||||
statuses: '',
|
||||
members: selectedAssignees,
|
||||
@@ -763,6 +770,16 @@ const taskManagementSlice = createSlice({
|
||||
toggleArchived: state => {
|
||||
state.archived = !state.archived;
|
||||
},
|
||||
setSortField: (state, action: PayloadAction<string>) => {
|
||||
state.sortField = action.payload;
|
||||
},
|
||||
setSortOrder: (state, action: PayloadAction<'ASC' | 'DESC'>) => {
|
||||
state.sortOrder = action.payload;
|
||||
},
|
||||
setSort: (state, action: PayloadAction<{ field: string; order: 'ASC' | 'DESC' }>) => {
|
||||
state.sortField = action.payload.field;
|
||||
state.sortOrder = action.payload.order;
|
||||
},
|
||||
resetTaskManagement: state => {
|
||||
state.loading = false;
|
||||
state.error = null;
|
||||
@@ -771,6 +788,8 @@ const taskManagementSlice = createSlice({
|
||||
state.selectedPriorities = [];
|
||||
state.search = '';
|
||||
state.archived = false;
|
||||
state.sortField = '';
|
||||
state.sortOrder = 'ASC';
|
||||
state.ids = [];
|
||||
state.entities = {};
|
||||
},
|
||||
@@ -1160,6 +1179,9 @@ export const {
|
||||
setSearch,
|
||||
setArchived,
|
||||
toggleArchived,
|
||||
setSortField,
|
||||
setSortOrder,
|
||||
setSort,
|
||||
resetTaskManagement,
|
||||
toggleTaskExpansion,
|
||||
addSubtaskToParent,
|
||||
@@ -1192,8 +1214,10 @@ export const selectError = (state: RootState) => state.taskManagement.error;
|
||||
export const selectSelectedPriorities = (state: RootState) =>
|
||||
state.taskManagement.selectedPriorities;
|
||||
export const selectSearch = (state: RootState) => state.taskManagement.search;
|
||||
export const selectSubtaskLoading = (state: RootState, taskId: string) =>
|
||||
state.taskManagement.loadingSubtasks[taskId] || false;
|
||||
export const selectSortField = (state: RootState) => state.taskManagement.sortField;
|
||||
export const selectSortOrder = (state: RootState) => state.taskManagement.sortOrder;
|
||||
export const selectSort = (state: RootState) => ({ field: state.taskManagement.sortField, order: state.taskManagement.sortOrder });
|
||||
export const selectSubtaskLoading = (state: RootState, taskId: string) => state.taskManagement.loadingSubtasks[taskId] || false;
|
||||
|
||||
// Memoized selectors to prevent unnecessary re-renders
|
||||
export const selectTasksByStatus = createSelector(
|
||||
|
||||
49
worklenz-frontend/src/hooks/useSurveyStatus.ts
Normal file
49
worklenz-frontend/src/hooks/useSurveyStatus.ts
Normal file
@@ -0,0 +1,49 @@
|
||||
import { useState, useEffect } from 'react';
|
||||
import { surveyApiService } from '@/api/survey/survey.api.service';
|
||||
import logger from '@/utils/errorLogger';
|
||||
|
||||
export interface UseSurveyStatusResult {
|
||||
hasCompletedSurvey: boolean | null;
|
||||
loading: boolean;
|
||||
error: Error | null;
|
||||
refetch: () => Promise<void>;
|
||||
}
|
||||
|
||||
export const useSurveyStatus = (): UseSurveyStatusResult => {
|
||||
const [hasCompletedSurvey, setHasCompletedSurvey] = useState<boolean | null>(null);
|
||||
const [loading, setLoading] = useState(true);
|
||||
const [error, setError] = useState<Error | null>(null);
|
||||
|
||||
const checkSurveyStatus = async () => {
|
||||
try {
|
||||
setLoading(true);
|
||||
setError(null);
|
||||
|
||||
const response = await surveyApiService.checkAccountSetupSurveyStatus();
|
||||
|
||||
if (response.done) {
|
||||
setHasCompletedSurvey(response.body.is_completed);
|
||||
} else {
|
||||
setHasCompletedSurvey(false);
|
||||
}
|
||||
} catch (err) {
|
||||
logger.error('Failed to check survey status', err);
|
||||
setError(err as Error);
|
||||
// Assume not completed if there's an error
|
||||
setHasCompletedSurvey(false);
|
||||
} finally {
|
||||
setLoading(false);
|
||||
}
|
||||
};
|
||||
|
||||
useEffect(() => {
|
||||
checkSurveyStatus();
|
||||
}, []);
|
||||
|
||||
return {
|
||||
hasCompletedSurvey,
|
||||
loading,
|
||||
error,
|
||||
refetch: checkSurveyStatus
|
||||
};
|
||||
};
|
||||
@@ -288,23 +288,7 @@ export const useTaskSocketHandlers = () => {
|
||||
);
|
||||
}
|
||||
|
||||
console.log('🔄 Status change group movement debug:', {
|
||||
taskId: response.id,
|
||||
newStatusValue,
|
||||
currentGroupId: currentGroup?.id,
|
||||
currentGroupValue: currentGroup?.groupValue,
|
||||
currentGroupTitle: currentGroup?.title,
|
||||
targetGroupId: targetGroup?.id,
|
||||
targetGroupValue: targetGroup?.groupValue,
|
||||
targetGroupTitle: targetGroup?.title,
|
||||
allGroups: groups.map(g => ({ id: g.id, title: g.title, groupValue: g.groupValue })),
|
||||
});
|
||||
|
||||
if (currentGroup && targetGroup && currentGroup.id !== targetGroup.id) {
|
||||
console.log('✅ Moving task between groups:', {
|
||||
from: currentGroup.title,
|
||||
to: targetGroup.title,
|
||||
});
|
||||
// Use the action to move task between groups
|
||||
dispatch(
|
||||
moveTaskBetweenGroups({
|
||||
@@ -452,12 +436,6 @@ export const useTaskSocketHandlers = () => {
|
||||
}
|
||||
|
||||
if (currentGroup && targetGroup && currentGroup.id !== targetGroup.id) {
|
||||
console.log('🔄 Moving task between priority groups:', {
|
||||
taskId: response.id,
|
||||
from: currentGroup.title,
|
||||
to: targetGroup.title,
|
||||
newPriorityValue,
|
||||
});
|
||||
dispatch(
|
||||
moveTaskBetweenGroups({
|
||||
taskId: response.id,
|
||||
@@ -610,12 +588,6 @@ export const useTaskSocketHandlers = () => {
|
||||
}
|
||||
|
||||
if (currentGroup && targetGroup && currentGroup.id !== targetGroup.id) {
|
||||
console.log('🔄 Moving task between phase groups:', {
|
||||
taskId,
|
||||
from: currentGroup.title,
|
||||
to: targetGroup.title,
|
||||
newPhaseValue,
|
||||
});
|
||||
dispatch(
|
||||
moveTaskBetweenGroups({
|
||||
taskId: taskId,
|
||||
@@ -937,10 +909,6 @@ export const useTaskSocketHandlers = () => {
|
||||
// Handler for TASK_ASSIGNEES_CHANGE (fallback event with limited data)
|
||||
const handleTaskAssigneesChange = useCallback((data: { assigneeIds: string[] }) => {
|
||||
if (!data || !data.assigneeIds) return;
|
||||
|
||||
// This event only provides assignee IDs, so we update what we can
|
||||
// The full assignee data will come from QUICK_ASSIGNEES_UPDATE
|
||||
// console.log('🔄 Task assignees change (limited data):', data);
|
||||
}, []);
|
||||
|
||||
// Handler for timer start events
|
||||
@@ -956,60 +924,55 @@ export const useTaskSocketHandlers = () => {
|
||||
: parseInt(start_time)
|
||||
: Date.now();
|
||||
|
||||
// Update the task-management slice to include timer state
|
||||
const currentTask = store.getState().taskManagement.entities[task_id];
|
||||
if (currentTask) {
|
||||
const updatedTask: Task = {
|
||||
...currentTask,
|
||||
timeTracking: {
|
||||
...currentTask.timeTracking,
|
||||
activeTimer: timerTimestamp,
|
||||
},
|
||||
updatedAt: new Date().toISOString(),
|
||||
updated_at: new Date().toISOString(),
|
||||
};
|
||||
dispatch(updateTask(updatedTask));
|
||||
}
|
||||
|
||||
// Also update the tasks slice activeTimers to keep both slices in sync
|
||||
dispatch(updateTaskTimeTracking({ taskId: task_id, timeTracking: timerTimestamp }));
|
||||
} catch (error) {
|
||||
logger.error('Error handling timer start event:', error);
|
||||
// Update the task-management slice to include timer state
|
||||
const currentTask = store.getState().taskManagement.entities[task_id];
|
||||
if (currentTask) {
|
||||
const updatedTask: Task = {
|
||||
...currentTask,
|
||||
timeTracking: {
|
||||
...currentTask.timeTracking,
|
||||
activeTimer: timerTimestamp,
|
||||
},
|
||||
updatedAt: new Date().toISOString(),
|
||||
updated_at: new Date().toISOString(),
|
||||
};
|
||||
dispatch(updateTask(updatedTask));
|
||||
}
|
||||
},
|
||||
[dispatch]
|
||||
);
|
||||
|
||||
// Handler for timer stop events
|
||||
const handleTimerStop = useCallback(
|
||||
(data: string) => {
|
||||
try {
|
||||
const { task_id } = typeof data === 'string' ? JSON.parse(data) : data;
|
||||
if (!task_id) return;
|
||||
// Also update the tasks slice activeTimers to keep both slices in sync
|
||||
dispatch(updateTaskTimeTracking({ taskId: task_id, timeTracking: timerTimestamp }));
|
||||
} catch (error) {
|
||||
logger.error('Error handling timer start event:', error);
|
||||
}
|
||||
}, [dispatch]);
|
||||
|
||||
// Update the task-management slice to remove timer state
|
||||
const currentTask = store.getState().taskManagement.entities[task_id];
|
||||
if (currentTask) {
|
||||
const updatedTask: Task = {
|
||||
...currentTask,
|
||||
timeTracking: {
|
||||
...currentTask.timeTracking,
|
||||
activeTimer: undefined,
|
||||
},
|
||||
updatedAt: new Date().toISOString(),
|
||||
updated_at: new Date().toISOString(),
|
||||
};
|
||||
dispatch(updateTask(updatedTask));
|
||||
}
|
||||
// Handler for timer stop events
|
||||
const handleTimerStop = useCallback((data: string) => {
|
||||
try {
|
||||
const { task_id } = typeof data === 'string' ? JSON.parse(data) : data;
|
||||
if (!task_id) return;
|
||||
|
||||
// Also update the tasks slice activeTimers to keep both slices in sync
|
||||
dispatch(updateTaskTimeTracking({ taskId: task_id, timeTracking: null }));
|
||||
} catch (error) {
|
||||
logger.error('Error handling timer stop event:', error);
|
||||
// Update the task-management slice to remove timer state
|
||||
const currentTask = store.getState().taskManagement.entities[task_id];
|
||||
if (currentTask) {
|
||||
const updatedTask: Task = {
|
||||
...currentTask,
|
||||
timeTracking: {
|
||||
...currentTask.timeTracking,
|
||||
activeTimer: undefined,
|
||||
},
|
||||
updatedAt: new Date().toISOString(),
|
||||
updated_at: new Date().toISOString(),
|
||||
};
|
||||
dispatch(updateTask(updatedTask));
|
||||
}
|
||||
},
|
||||
[dispatch]
|
||||
);
|
||||
|
||||
// Also update the tasks slice activeTimers to keep both slices in sync
|
||||
dispatch(updateTaskTimeTracking({ taskId: task_id, timeTracking: null }));
|
||||
} catch (error) {
|
||||
logger.error('Error handling timer stop event:', error);
|
||||
}
|
||||
}, [dispatch]);
|
||||
|
||||
// Handler for task sort order change events
|
||||
const handleTaskSortOrderChange = useCallback(
|
||||
|
||||
94
worklenz-frontend/src/hooks/useTaskTimerWithConflictCheck.ts
Normal file
94
worklenz-frontend/src/hooks/useTaskTimerWithConflictCheck.ts
Normal file
@@ -0,0 +1,94 @@
|
||||
import { useState, useEffect, useCallback } from 'react';
|
||||
import { Modal } from 'antd';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { useSocket } from '@/socket/socketContext';
|
||||
import { SocketEvents } from '@/shared/socket-events';
|
||||
import { taskTimeLogsApiService } from '@/api/tasks/task-time-logs.api.service';
|
||||
import { useTaskTimer } from './useTaskTimer';
|
||||
|
||||
interface ConflictingTimer {
|
||||
task_id: string;
|
||||
task_name: string;
|
||||
project_name: string;
|
||||
}
|
||||
|
||||
export const useTaskTimerWithConflictCheck = (taskId: string, timerStartTime: string | null) => {
|
||||
const { socket } = useSocket();
|
||||
const { t: tTable } = useTranslation('task-list-table');
|
||||
const { t: tCommon } = useTranslation('common');
|
||||
|
||||
// Ensure timerStartTime is a number or null, as required by useTaskTimer
|
||||
const parsedTimerStartTime = typeof timerStartTime === 'string' && timerStartTime !== null
|
||||
? Number(timerStartTime)
|
||||
: timerStartTime;
|
||||
|
||||
const originalHook = useTaskTimer(taskId, parsedTimerStartTime as number | null);
|
||||
const [isCheckingConflict, setIsCheckingConflict] = useState(false);
|
||||
|
||||
const checkForConflictingTimers = useCallback(async () => {
|
||||
try {
|
||||
const response = await taskTimeLogsApiService.getRunningTimers();
|
||||
const runningTimers = response.body || [];
|
||||
|
||||
// Find conflicting timer (running timer for a different task)
|
||||
const conflictingTimer = runningTimers.find((timer: ConflictingTimer) =>
|
||||
timer.task_id !== taskId
|
||||
);
|
||||
|
||||
return conflictingTimer;
|
||||
} catch (error) {
|
||||
console.error('Error checking for conflicting timers:', error);
|
||||
return null;
|
||||
}
|
||||
}, [taskId]);
|
||||
|
||||
const handleStartTimerWithConflictCheck = useCallback(async () => {
|
||||
if (isCheckingConflict) return;
|
||||
|
||||
setIsCheckingConflict(true);
|
||||
|
||||
try {
|
||||
const conflictingTimer = await checkForConflictingTimers();
|
||||
|
||||
if (conflictingTimer) {
|
||||
Modal.confirm({
|
||||
title: tTable('timer.conflictTitle'),
|
||||
content: tTable('timer.conflictMessage', {
|
||||
taskName: conflictingTimer.task_name,
|
||||
projectName: conflictingTimer.project_name
|
||||
}),
|
||||
okText: tTable('timer.stopAndStart'),
|
||||
cancelText: tCommon('cancel'),
|
||||
onOk: () => {
|
||||
// Stop the conflicting timer
|
||||
if (socket) {
|
||||
socket.emit(SocketEvents.TASK_TIMER_STOP.toString(), JSON.stringify({
|
||||
task_id: conflictingTimer.task_id
|
||||
}));
|
||||
}
|
||||
|
||||
// Start the new timer after a short delay
|
||||
setTimeout(() => {
|
||||
originalHook.handleStartTimer();
|
||||
}, 100);
|
||||
},
|
||||
});
|
||||
} else {
|
||||
// No conflict, start timer directly
|
||||
originalHook.handleStartTimer();
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error handling timer start with conflict check:', error);
|
||||
// Fallback to original behavior
|
||||
originalHook.handleStartTimer();
|
||||
} finally {
|
||||
setIsCheckingConflict(false);
|
||||
}
|
||||
}, [isCheckingConflict, checkForConflictingTimers, tTable, tCommon, socket, originalHook]);
|
||||
|
||||
return {
|
||||
...originalHook,
|
||||
handleStartTimer: handleStartTimerWithConflictCheck,
|
||||
isCheckingConflict,
|
||||
};
|
||||
};
|
||||
70
worklenz-frontend/src/hooks/useUserTimezone.ts
Normal file
70
worklenz-frontend/src/hooks/useUserTimezone.ts
Normal file
@@ -0,0 +1,70 @@
|
||||
import { useState, useEffect } from 'react';
|
||||
|
||||
/**
|
||||
* Custom hook to get and manage user's timezone
|
||||
* @returns {Object} Object containing timezone and related utilities
|
||||
*/
|
||||
export const useUserTimezone = () => {
|
||||
const [timezone, setTimezone] = useState<string>('UTC');
|
||||
const [timezoneOffset, setTimezoneOffset] = useState<string>('+00:00');
|
||||
|
||||
useEffect(() => {
|
||||
// Get browser's timezone
|
||||
const browserTimezone = Intl.DateTimeFormat().resolvedOptions().timeZone;
|
||||
setTimezone(browserTimezone);
|
||||
|
||||
// Calculate timezone offset
|
||||
const date = new Date();
|
||||
const offset = -date.getTimezoneOffset();
|
||||
const hours = Math.floor(Math.abs(offset) / 60);
|
||||
const minutes = Math.abs(offset) % 60;
|
||||
const sign = offset >= 0 ? '+' : '-';
|
||||
const formattedOffset = `${sign}${hours.toString().padStart(2, '0')}:${minutes.toString().padStart(2, '0')}`;
|
||||
setTimezoneOffset(formattedOffset);
|
||||
}, []);
|
||||
|
||||
/**
|
||||
* Format a date in the user's timezone
|
||||
* @param date - Date to format
|
||||
* @param format - Format options
|
||||
* @returns Formatted date string
|
||||
*/
|
||||
const formatInUserTimezone = (date: Date | string, format?: Intl.DateTimeFormatOptions) => {
|
||||
const dateObj = typeof date === 'string' ? new Date(date) : date;
|
||||
return dateObj.toLocaleString('en-US', {
|
||||
timeZone: timezone,
|
||||
...format
|
||||
});
|
||||
};
|
||||
|
||||
/**
|
||||
* Get the start of day in user's timezone
|
||||
* @param date - Date to get start of day for
|
||||
* @returns Date object representing start of day
|
||||
*/
|
||||
const getStartOfDayInTimezone = (date: Date = new Date()) => {
|
||||
const localDate = new Date(date.toLocaleString('en-US', { timeZone: timezone }));
|
||||
localDate.setHours(0, 0, 0, 0);
|
||||
return localDate;
|
||||
};
|
||||
|
||||
/**
|
||||
* Get the end of day in user's timezone
|
||||
* @param date - Date to get end of day for
|
||||
* @returns Date object representing end of day
|
||||
*/
|
||||
const getEndOfDayInTimezone = (date: Date = new Date()) => {
|
||||
const localDate = new Date(date.toLocaleString('en-US', { timeZone: timezone }));
|
||||
localDate.setHours(23, 59, 59, 999);
|
||||
return localDate;
|
||||
};
|
||||
|
||||
return {
|
||||
timezone,
|
||||
timezoneOffset,
|
||||
formatInUserTimezone,
|
||||
getStartOfDayInTimezone,
|
||||
getEndOfDayInTimezone,
|
||||
setTimezone
|
||||
};
|
||||
};
|
||||
8
worklenz-frontend/src/pages/GanttDemoPage.tsx
Normal file
8
worklenz-frontend/src/pages/GanttDemoPage.tsx
Normal file
@@ -0,0 +1,8 @@
|
||||
import React from 'react';
|
||||
import { AdvancedGanttDemo } from '../components/advanced-gantt';
|
||||
|
||||
const GanttDemoPage: React.FC = () => {
|
||||
return <AdvancedGanttDemo />;
|
||||
};
|
||||
|
||||
export default GanttDemoPage;
|
||||
@@ -1,5 +1,9 @@
|
||||
/* Account Setup Page Styles */
|
||||
|
||||
/* Steps styling - using Ant Design theme tokens */
|
||||
.ant-steps-item-finish .ant-steps-item-icon {
|
||||
border-color: #1890ff !important;
|
||||
border-color: var(--ant-color-primary) !important;
|
||||
background-color: var(--ant-color-primary) !important;
|
||||
}
|
||||
|
||||
.ant-steps-item-icon {
|
||||
@@ -9,37 +13,53 @@
|
||||
font-size: 16px !important;
|
||||
line-height: 32px !important;
|
||||
text-align: center !important;
|
||||
border: 1px solid rgba(0, 0, 0, 0.25) !important;
|
||||
border: 1px solid var(--ant-color-border) !important;
|
||||
border-radius: 32px !important;
|
||||
transition:
|
||||
background-color 0.3s,
|
||||
border-color 0.3s !important;
|
||||
transition: all 0.3s !important;
|
||||
background-color: var(--ant-color-bg-container) !important;
|
||||
color: var(--ant-color-text) !important;
|
||||
}
|
||||
|
||||
.ant-steps-item-wait .ant-steps-item-icon {
|
||||
cursor: not-allowed;
|
||||
opacity: 0.6;
|
||||
}
|
||||
|
||||
.dark-mode .ant-steps-item-wait .ant-steps-item-icon {
|
||||
cursor: not-allowed;
|
||||
.ant-steps-item-process .ant-steps-item-icon {
|
||||
border-color: var(--ant-color-primary) !important;
|
||||
background-color: var(--ant-color-primary) !important;
|
||||
color: var(--ant-color-white) !important;
|
||||
}
|
||||
|
||||
.progress-steps .ant-steps-item.ant-steps-item-process .ant-steps-item-title::after {
|
||||
background-color: #1677ff !important;
|
||||
background-color: var(--ant-color-primary) !important;
|
||||
width: 60px !important;
|
||||
}
|
||||
|
||||
.ant-steps-item-title {
|
||||
color: var(--ant-color-text) !important;
|
||||
}
|
||||
|
||||
.ant-steps-item-description {
|
||||
color: var(--ant-color-text-secondary) !important;
|
||||
}
|
||||
|
||||
/* Responsive layout */
|
||||
@media (max-width: 1000px) {
|
||||
.progress-steps {
|
||||
width: 400px !important;
|
||||
}
|
||||
|
||||
.step {
|
||||
width: 400px !important;
|
||||
}
|
||||
|
||||
.progress-steps,
|
||||
.step,
|
||||
.step-content {
|
||||
width: 400px !important;
|
||||
width: 90% !important;
|
||||
max-width: 500px !important;
|
||||
}
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.progress-steps,
|
||||
.step,
|
||||
.step-content {
|
||||
width: 95% !important;
|
||||
max-width: 400px !important;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -53,98 +73,93 @@
|
||||
}
|
||||
|
||||
.step-content {
|
||||
width: 200px !important;
|
||||
width: 100% !important;
|
||||
max-width: 300px !important;
|
||||
padding: 0 1rem !important;
|
||||
}
|
||||
}
|
||||
|
||||
@media (max-width: 1000px) {
|
||||
.organization-name-form {
|
||||
width: 400px !important;
|
||||
/* Tasks step specific styles */
|
||||
.tasks-step .task-item-card {
|
||||
transition: all 0.3s ease;
|
||||
}
|
||||
|
||||
.tasks-step .task-item-card:hover {
|
||||
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.06);
|
||||
}
|
||||
|
||||
.tasks-step .task-input {
|
||||
font-size: 16px;
|
||||
}
|
||||
|
||||
.tasks-step .task-input:focus {
|
||||
box-shadow: none !important;
|
||||
}
|
||||
|
||||
/* Project step specific styles */
|
||||
.project-step .ant-input-affix-wrapper {
|
||||
transition: all 0.3s ease;
|
||||
}
|
||||
|
||||
.project-step .ant-input-affix-wrapper:focus-within {
|
||||
box-shadow: 0 0 0 2px rgba(24, 144, 255, 0.1);
|
||||
}
|
||||
|
||||
.project-suggestion-button {
|
||||
transition: all 0.2s ease;
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.project-suggestion-button:hover {
|
||||
transform: translateY(-1px);
|
||||
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.project-suggestion-button:active {
|
||||
transform: translateY(0);
|
||||
}
|
||||
|
||||
/* Organization step specific styles */
|
||||
.organization-step .ant-input-affix-wrapper {
|
||||
transition: all 0.3s ease;
|
||||
}
|
||||
|
||||
.organization-step .ant-input-affix-wrapper:focus-within {
|
||||
box-shadow: 0 0 0 2px rgba(24, 144, 255, 0.1);
|
||||
border-color: #1890ff;
|
||||
}
|
||||
|
||||
.organization-suggestion-button {
|
||||
transition: all 0.2s ease;
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.organization-suggestion-button:hover {
|
||||
transform: translateY(-1px);
|
||||
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.organization-suggestion-button:active {
|
||||
transform: translateY(0);
|
||||
}
|
||||
|
||||
/* Survey step animations */
|
||||
.survey-page-transition {
|
||||
animation: fadeInUp 0.4s ease-out;
|
||||
}
|
||||
|
||||
@keyframes fadeInUp {
|
||||
from {
|
||||
opacity: 0;
|
||||
transform: translateY(20px);
|
||||
}
|
||||
to {
|
||||
opacity: 1;
|
||||
transform: translateY(0);
|
||||
}
|
||||
}
|
||||
|
||||
@media (max-width: 500px) {
|
||||
.organization-name-form {
|
||||
width: 200px !important;
|
||||
}
|
||||
}
|
||||
|
||||
.vert-text {
|
||||
max-width: 40px;
|
||||
background-color: #fff;
|
||||
position: relative;
|
||||
z-index: 99;
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
text-align: center;
|
||||
justify-content: center;
|
||||
margin-top: 2rem;
|
||||
}
|
||||
|
||||
.vert-text-dark {
|
||||
max-width: 40px;
|
||||
background-color: #141414;
|
||||
position: relative;
|
||||
z-index: 99;
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
text-align: center;
|
||||
justify-content: center;
|
||||
margin-top: 2rem;
|
||||
}
|
||||
|
||||
.vert-line {
|
||||
position: absolute;
|
||||
left: 0;
|
||||
right: 0;
|
||||
width: 100%;
|
||||
content: "";
|
||||
height: 2px;
|
||||
background-color: #00000047;
|
||||
bottom: 0;
|
||||
top: 0;
|
||||
margin-bottom: auto;
|
||||
margin-top: auto;
|
||||
}
|
||||
|
||||
.vert-line-dark {
|
||||
position: absolute;
|
||||
left: 0;
|
||||
right: 0;
|
||||
width: 100%;
|
||||
content: "";
|
||||
height: 2px;
|
||||
background-color: white;
|
||||
bottom: 0;
|
||||
top: 0;
|
||||
margin-bottom: auto;
|
||||
margin-top: auto;
|
||||
}
|
||||
|
||||
@media (max-width: 1000px) {
|
||||
.first-project-form {
|
||||
width: 400px !important;
|
||||
}
|
||||
}
|
||||
|
||||
@media (max-width: 500px) {
|
||||
.first-project-form {
|
||||
width: 200px !important;
|
||||
}
|
||||
}
|
||||
|
||||
.custom-close-button:hover {
|
||||
background-color: transparent !important;
|
||||
}
|
||||
|
||||
@media (max-width: 1000px) {
|
||||
.create-first-task-form {
|
||||
width: 400px !important;
|
||||
}
|
||||
}
|
||||
|
||||
@media (max-width: 500px) {
|
||||
.create-first-task-form {
|
||||
width: 200px !important;
|
||||
}
|
||||
}
|
||||
/* Theme transitions */
|
||||
* {
|
||||
transition: background-color 0.3s ease, color 0.3s ease, border-color 0.3s ease;
|
||||
}
|
||||
@@ -2,13 +2,15 @@ import React, { useEffect } from 'react';
|
||||
import { useSelector } from 'react-redux';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { useNavigate } from 'react-router-dom';
|
||||
import { Space, Steps, Button, Typography } from 'antd/es';
|
||||
import { Space, Steps, Button, Typography, theme, Dropdown, MenuProps } from '@/shared/antd-imports';
|
||||
import { GlobalOutlined, MoonOutlined, SunOutlined } from '@/shared/antd-imports';
|
||||
|
||||
import logger from '@/utils/errorLogger';
|
||||
import { setCurrentStep } from '@/features/account-setup/account-setup.slice';
|
||||
import { setCurrentStep, setSurveySubStep } from '@/features/account-setup/account-setup.slice';
|
||||
import { OrganizationStep } from '@/components/account-setup/organization-step';
|
||||
import { ProjectStep } from '@/components/account-setup/project-step';
|
||||
import { TasksStep } from '@/components/account-setup/tasks-step';
|
||||
import { SurveyStep } from '@/components/account-setup/survey-step';
|
||||
import MembersStep from '@/components/account-setup/members-step';
|
||||
import {
|
||||
evt_account_setup_complete,
|
||||
@@ -31,23 +33,110 @@ import { useAppDispatch } from '@/hooks/useAppDispatch';
|
||||
import './account-setup.css';
|
||||
import { IAccountSetupRequest } from '@/types/project-templates/project-templates.types';
|
||||
import { profileSettingsApiService } from '@/api/settings/profile/profile-settings.api.service';
|
||||
import { projectTemplatesApiService } from '@/api/project-templates/project-templates.api.service';
|
||||
import { surveyApiService } from '@/api/survey/survey.api.service';
|
||||
import { ISurveySubmissionRequest, ISurveyAnswer } from '@/types/account-setup/survey.types';
|
||||
import { setLanguage } from '@/features/i18n/localesSlice';
|
||||
import { ILanguageType, Language } from '@/features/i18n/localesSlice';
|
||||
import { toggleTheme } from '@/features/theme/themeSlice';
|
||||
|
||||
const { Title } = Typography;
|
||||
|
||||
// Simplified styles for form components using Ant Design theme tokens
|
||||
const getAccountSetupStyles = (token: any) => ({
|
||||
form: {
|
||||
width: '100%',
|
||||
maxWidth: '600px',
|
||||
},
|
||||
label: {
|
||||
color: token.colorText,
|
||||
fontWeight: 500,
|
||||
},
|
||||
buttonContainer: {
|
||||
display: 'flex',
|
||||
justifyContent: 'center',
|
||||
alignItems: 'center',
|
||||
marginTop: '1rem',
|
||||
},
|
||||
drawerFooter: {
|
||||
display: 'flex',
|
||||
justifyContent: 'right',
|
||||
padding: '10px 16px',
|
||||
},
|
||||
});
|
||||
|
||||
const AccountSetup: React.FC = () => {
|
||||
const dispatch = useAppDispatch();
|
||||
const { t } = useTranslation('account-setup');
|
||||
const { t, i18n } = useTranslation('account-setup');
|
||||
useDocumentTitle(t('setupYourAccount', 'Account Setup'));
|
||||
const navigate = useNavigate();
|
||||
const { trackMixpanelEvent } = useMixpanelTracking();
|
||||
const { token } = theme.useToken();
|
||||
|
||||
const { currentStep, organizationName, projectName, templateId, tasks, teamMembers } =
|
||||
const { currentStep, organizationName, projectName, templateId, tasks, teamMembers, surveyData, surveySubStep } =
|
||||
useSelector((state: RootState) => state.accountSetupReducer);
|
||||
const lng = useSelector((state: RootState) => state.localesReducer.lng);
|
||||
const userDetails = getUserSession();
|
||||
const themeMode = useSelector((state: RootState) => state.themeReducer.mode);
|
||||
|
||||
const [surveyId, setSurveyId] = React.useState<string | null>(null);
|
||||
const [isSkipping, setIsSkipping] = React.useState(false);
|
||||
|
||||
const isDarkMode = themeMode === 'dark';
|
||||
const organizationNamePlaceholder = userDetails?.name ? `e.g. ${userDetails?.name}'s Team` : '';
|
||||
// Helper to extract organization name from email or fallback to user name
|
||||
function getOrganizationNamePlaceholder(userDetails: { email?: string; name?: string } | null): string {
|
||||
if (!userDetails) return '';
|
||||
const email = userDetails.email || '';
|
||||
const name = userDetails.name || '';
|
||||
if (email) {
|
||||
const match = email.match(/^([^@]+)@([^@]+)$/);
|
||||
if (match) {
|
||||
const domain = match[2].toLowerCase();
|
||||
// List of common public email providers
|
||||
const publicProviders = [
|
||||
'gmail.com', 'yahoo.com', 'outlook.com', 'hotmail.com', 'icloud.com', 'aol.com', 'protonmail.com', 'zoho.com', 'gmx.com', 'mail.com', 'yandex.com', 'msn.com', 'live.com', 'me.com', 'comcast.net', 'rediffmail.com', 'ymail.com', 'rocketmail.com', 'inbox.com', 'mail.ru', 'qq.com', 'naver.com', '163.com', '126.com', 'sina.com', 'yeah.net', 'googlemail.com', 'fastmail.com', 'hushmail.com', 'tutanota.com', 'pm.me', 'mailbox.org', 'proton.me'
|
||||
];
|
||||
if (!publicProviders.includes(domain)) {
|
||||
// Use the first part of the domain (before the first dot)
|
||||
const org = domain.split('.')[0];
|
||||
if (org && org.length > 1) {
|
||||
return `e.g. ${org.charAt(0).toUpperCase() + org.slice(1)} Team`;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
// Fallback to user name
|
||||
return name ? `e.g. ${name}'s Team` : '';
|
||||
}
|
||||
|
||||
const organizationNamePlaceholder = getOrganizationNamePlaceholder(userDetails);
|
||||
|
||||
// Helper to extract organization name from email or fallback to user name
|
||||
function getOrganizationNameInitialValue(userDetails: { email?: string; name?: string } | null): string {
|
||||
if (!userDetails) return '';
|
||||
const email = userDetails.email || '';
|
||||
const name = userDetails.name || '';
|
||||
if (email) {
|
||||
const match = email.match(/^([^@]+)@([^@]+)$/);
|
||||
if (match) {
|
||||
const domain = match[2].toLowerCase();
|
||||
const publicProviders = [
|
||||
'gmail.com', 'yahoo.com', 'outlook.com', 'hotmail.com', 'icloud.com', 'aol.com', 'protonmail.com', 'zoho.com', 'gmx.com', 'mail.com', 'yandex.com', 'msn.com', 'live.com', 'me.com', 'comcast.net', 'rediffmail.com', 'ymail.com', 'rocketmail.com', 'inbox.com', 'mail.ru', 'qq.com', 'naver.com', '163.com', '126.com', 'sina.com', 'yeah.net', 'googlemail.com', 'fastmail.com', 'hushmail.com', 'tutanota.com', 'pm.me', 'mailbox.org', 'proton.me'
|
||||
];
|
||||
if (!publicProviders.includes(domain)) {
|
||||
const org = domain.split('.')[0];
|
||||
if (org && org.length > 1) {
|
||||
return org.charAt(0).toUpperCase() + org.slice(1);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
return name || '';
|
||||
}
|
||||
|
||||
const organizationNameInitialValue = getOrganizationNameInitialValue(userDetails);
|
||||
|
||||
const styles = getAccountSetupStyles(token);
|
||||
|
||||
useEffect(() => {
|
||||
trackMixpanelEvent(evt_account_setup_visit);
|
||||
@@ -65,88 +154,25 @@ const AccountSetup: React.FC = () => {
|
||||
logger.error('Failed to verify authentication status', error);
|
||||
}
|
||||
};
|
||||
|
||||
const loadSurvey = async () => {
|
||||
try {
|
||||
const response = await surveyApiService.getAccountSetupSurvey();
|
||||
if (response.done && response.body) {
|
||||
setSurveyId(response.body.id);
|
||||
} else {
|
||||
logger.error('Survey not found or inactive (warn replaced with error)');
|
||||
}
|
||||
} catch (error) {
|
||||
logger.error('Failed to load survey', error);
|
||||
// Continue without survey - don't block account setup
|
||||
}
|
||||
};
|
||||
|
||||
void verifyAuthStatus();
|
||||
void loadSurvey();
|
||||
}, [dispatch, navigate, trackMixpanelEvent]);
|
||||
|
||||
const calculateHeight = () => {
|
||||
if (currentStep === 2) {
|
||||
return tasks.length * 105;
|
||||
}
|
||||
|
||||
if (currentStep === 3) {
|
||||
return teamMembers.length * 105;
|
||||
}
|
||||
return 'min-content';
|
||||
};
|
||||
|
||||
const styles = {
|
||||
form: {
|
||||
width: '600px',
|
||||
paddingBottom: '1rem',
|
||||
marginTop: '3rem',
|
||||
height: '100%',
|
||||
overflow: 'hidden',
|
||||
},
|
||||
label: {
|
||||
color: isDarkMode ? '' : '#00000073',
|
||||
fontWeight: 500,
|
||||
},
|
||||
buttonContainer: {
|
||||
display: 'flex',
|
||||
justifyContent: 'center',
|
||||
alignItems: 'center',
|
||||
marginTop: '1rem',
|
||||
},
|
||||
drawerFooter: {
|
||||
display: 'flex',
|
||||
justifyContent: 'right',
|
||||
padding: '10px 16px',
|
||||
},
|
||||
container: {
|
||||
height: '100vh',
|
||||
width: '100vw',
|
||||
display: 'flex',
|
||||
flexDirection: 'column' as const,
|
||||
alignItems: 'center',
|
||||
padding: '3rem 0',
|
||||
backgroundColor: isDarkMode ? 'black' : '#FAFAFA',
|
||||
},
|
||||
contentContainer: {
|
||||
backgroundColor: isDarkMode ? '#141414' : 'white',
|
||||
marginTop: '1.5rem',
|
||||
paddingTop: '3rem',
|
||||
margin: '1.5rem auto 0',
|
||||
width: '100%',
|
||||
maxWidth: '66.66667%',
|
||||
minHeight: 'fit-content',
|
||||
display: 'flex',
|
||||
flexDirection: 'column' as const,
|
||||
},
|
||||
space: {
|
||||
display: 'flex',
|
||||
flexDirection: 'column' as const,
|
||||
alignItems: 'center',
|
||||
gap: '0',
|
||||
flexGrow: 1,
|
||||
width: '100%',
|
||||
minHeight: 'fit-content',
|
||||
},
|
||||
steps: {
|
||||
margin: '1rem 0',
|
||||
width: '600px',
|
||||
},
|
||||
stepContent: {
|
||||
flexGrow: 1,
|
||||
width: '600px',
|
||||
minHeight: calculateHeight(),
|
||||
overflow: 'visible',
|
||||
},
|
||||
actionButtons: {
|
||||
flexGrow: 1,
|
||||
width: '600px',
|
||||
marginBottom: '1rem',
|
||||
},
|
||||
};
|
||||
|
||||
const completeAccountSetup = async (skip = false) => {
|
||||
try {
|
||||
@@ -159,6 +185,13 @@ const AccountSetup: React.FC = () => {
|
||||
: teamMembers
|
||||
.map(teamMember => sanitizeInput(teamMember.value.trim()))
|
||||
.filter(email => validateEmail(email)),
|
||||
survey_data: {
|
||||
organization_type: surveyData.organization_type,
|
||||
user_role: surveyData.user_role,
|
||||
main_use_cases: surveyData.main_use_cases,
|
||||
previous_tools: surveyData.previous_tools,
|
||||
how_heard_about: surveyData.how_heard_about,
|
||||
},
|
||||
};
|
||||
const res = await profileSettingsApiService.setupAccount(model);
|
||||
if (res.done && res.body.id) {
|
||||
@@ -184,6 +217,59 @@ const AccountSetup: React.FC = () => {
|
||||
}
|
||||
};
|
||||
|
||||
const handleSkipMembers = async () => {
|
||||
try {
|
||||
setIsSkipping(true);
|
||||
// Bypass all validation and complete setup without team members
|
||||
await completeAccountSetup(true);
|
||||
} catch (error) {
|
||||
logger.error('Failed to skip members and complete setup', error);
|
||||
} finally {
|
||||
setIsSkipping(false);
|
||||
}
|
||||
};
|
||||
|
||||
const completeAccountSetupWithTemplate = async () => {
|
||||
try {
|
||||
await saveSurveyData(); // Save survey data first
|
||||
|
||||
const model: IAccountSetupRequest = {
|
||||
team_name: sanitizeInput(organizationName),
|
||||
project_name: null, // No project name when using template
|
||||
template_id: templateId,
|
||||
tasks: [],
|
||||
team_members: [],
|
||||
survey_data: {
|
||||
organization_type: surveyData.organization_type,
|
||||
user_role: surveyData.user_role,
|
||||
main_use_cases: surveyData.main_use_cases,
|
||||
previous_tools: surveyData.previous_tools,
|
||||
how_heard_about: surveyData.how_heard_about,
|
||||
},
|
||||
};
|
||||
|
||||
const res = await projectTemplatesApiService.setupAccount(model);
|
||||
if (res.done && res.body.id) {
|
||||
trackMixpanelEvent(evt_account_setup_complete);
|
||||
|
||||
// Refresh user session to update setup_completed status
|
||||
try {
|
||||
const authResponse = await dispatch(verifyAuthentication()).unwrap() as IAuthorizeResponse;
|
||||
if (authResponse?.authenticated && authResponse?.user) {
|
||||
setSession(authResponse.user);
|
||||
dispatch(setUser(authResponse.user));
|
||||
}
|
||||
} catch (error) {
|
||||
logger.error('Failed to refresh user session after template setup completion', error);
|
||||
}
|
||||
|
||||
navigate(`/worklenz/projects/${res.body.id}?tab=tasks-list&pinned_tab=tasks-list`);
|
||||
}
|
||||
} catch (error) {
|
||||
logger.error('completeAccountSetupWithTemplate', error);
|
||||
}
|
||||
};
|
||||
|
||||
const steps = [
|
||||
{
|
||||
title: '',
|
||||
@@ -192,6 +278,20 @@ const AccountSetup: React.FC = () => {
|
||||
onEnter={() => dispatch(setCurrentStep(currentStep + 1))}
|
||||
styles={styles}
|
||||
organizationNamePlaceholder={organizationNamePlaceholder}
|
||||
organizationNameInitialValue={organizationNameInitialValue}
|
||||
isDarkMode={isDarkMode}
|
||||
token={token}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
title: '',
|
||||
content: (
|
||||
<SurveyStep
|
||||
onEnter={() => dispatch(setCurrentStep(currentStep + 1))}
|
||||
styles={styles}
|
||||
isDarkMode={isDarkMode}
|
||||
token={token}
|
||||
/>
|
||||
),
|
||||
},
|
||||
@@ -202,6 +302,7 @@ const AccountSetup: React.FC = () => {
|
||||
onEnter={() => dispatch(setCurrentStep(currentStep + 1))}
|
||||
styles={styles}
|
||||
isDarkMode={isDarkMode}
|
||||
token={token}
|
||||
/>
|
||||
),
|
||||
},
|
||||
@@ -212,12 +313,13 @@ const AccountSetup: React.FC = () => {
|
||||
onEnter={() => dispatch(setCurrentStep(currentStep + 1))}
|
||||
styles={styles}
|
||||
isDarkMode={isDarkMode}
|
||||
token={token}
|
||||
/>
|
||||
),
|
||||
},
|
||||
{
|
||||
title: '',
|
||||
content: <MembersStep isDarkMode={isDarkMode} styles={styles} />,
|
||||
content: <MembersStep isDarkMode={isDarkMode} styles={styles} token={token} />,
|
||||
},
|
||||
];
|
||||
|
||||
@@ -226,10 +328,21 @@ const AccountSetup: React.FC = () => {
|
||||
case 0:
|
||||
return !organizationName?.trim();
|
||||
case 1:
|
||||
return !projectName?.trim() && !templateId;
|
||||
// Survey step - check current sub-step requirements
|
||||
if (surveySubStep === 0) {
|
||||
return !(surveyData.organization_type && surveyData.user_role);
|
||||
} else if (surveySubStep === 1) {
|
||||
return !(surveyData.main_use_cases && surveyData.main_use_cases.length > 0);
|
||||
} else if (surveySubStep === 2) {
|
||||
return !surveyData.how_heard_about;
|
||||
}
|
||||
return false;
|
||||
case 2:
|
||||
return tasks.length === 0 || tasks.every(task => !task.value?.trim());
|
||||
// Project step - either project name OR template must be provided
|
||||
return !projectName?.trim() && !templateId;
|
||||
case 3:
|
||||
return tasks.length === 0 || tasks.every(task => !task.value?.trim());
|
||||
case 4:
|
||||
return (
|
||||
teamMembers.length > 0 && !teamMembers.some(member => validateEmail(member.value?.trim()))
|
||||
);
|
||||
@@ -238,58 +351,267 @@ const AccountSetup: React.FC = () => {
|
||||
}
|
||||
};
|
||||
|
||||
const nextStep = () => {
|
||||
if (currentStep === 3) {
|
||||
const saveSurveyData = async () => {
|
||||
if (!surveyId || !surveyData) {
|
||||
logger.error('Skipping survey save - no survey ID or data (info replaced with error)');
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
const answers: ISurveyAnswer[] = [];
|
||||
|
||||
// Get the survey questions to map data properly
|
||||
const surveyResponse = await surveyApiService.getAccountSetupSurvey();
|
||||
if (!surveyResponse.done || !surveyResponse.body?.questions) {
|
||||
logger.error('Could not retrieve survey questions for data mapping (warn replaced with error)');
|
||||
return;
|
||||
}
|
||||
|
||||
const questions = surveyResponse.body.questions;
|
||||
|
||||
// Map survey data to answers based on question keys
|
||||
questions.forEach(question => {
|
||||
switch (question.question_key) {
|
||||
case 'organization_type':
|
||||
if (surveyData.organization_type) {
|
||||
answers.push({
|
||||
question_id: question.id,
|
||||
answer_text: surveyData.organization_type
|
||||
});
|
||||
}
|
||||
break;
|
||||
case 'user_role':
|
||||
if (surveyData.user_role) {
|
||||
answers.push({
|
||||
question_id: question.id,
|
||||
answer_text: surveyData.user_role
|
||||
});
|
||||
}
|
||||
break;
|
||||
case 'main_use_cases':
|
||||
if (surveyData.main_use_cases && surveyData.main_use_cases.length > 0) {
|
||||
answers.push({
|
||||
question_id: question.id,
|
||||
answer_json: surveyData.main_use_cases
|
||||
});
|
||||
}
|
||||
break;
|
||||
case 'previous_tools':
|
||||
if (surveyData.previous_tools) {
|
||||
answers.push({
|
||||
question_id: question.id,
|
||||
answer_text: surveyData.previous_tools
|
||||
});
|
||||
}
|
||||
break;
|
||||
case 'how_heard_about':
|
||||
if (surveyData.how_heard_about) {
|
||||
answers.push({
|
||||
question_id: question.id,
|
||||
answer_text: surveyData.how_heard_about
|
||||
});
|
||||
}
|
||||
break;
|
||||
}
|
||||
});
|
||||
|
||||
if (answers.length > 0) {
|
||||
const submissionData: ISurveySubmissionRequest = {
|
||||
survey_id: surveyId,
|
||||
answers
|
||||
};
|
||||
|
||||
const result = await surveyApiService.submitSurveyResponse(submissionData);
|
||||
if (result.done) {
|
||||
logger.error('Survey data saved successfully (info replaced with error)');
|
||||
} else {
|
||||
logger.error('Survey submission returned unsuccessful response (warn replaced with error)');
|
||||
}
|
||||
} else {
|
||||
logger.error('No survey answers to save (info replaced with error)');
|
||||
}
|
||||
} catch (error) {
|
||||
logger.error('Failed to save survey data', error);
|
||||
// Don't block account setup flow if survey fails
|
||||
}
|
||||
};
|
||||
|
||||
const nextStep = async () => {
|
||||
if (currentStep === 1) {
|
||||
// Handle survey sub-step navigation
|
||||
if (surveySubStep < 2) {
|
||||
// Move to next survey sub-step
|
||||
dispatch(setSurveySubStep(surveySubStep + 1));
|
||||
} else {
|
||||
// Survey completed, save data and move to next main step
|
||||
await saveSurveyData();
|
||||
dispatch(setCurrentStep(currentStep + 1));
|
||||
dispatch(setSurveySubStep(0)); // Reset for next time
|
||||
}
|
||||
} else if (currentStep === 2) {
|
||||
// Project step - check if template is selected
|
||||
if (templateId) {
|
||||
// Template selected, complete account setup with template
|
||||
await completeAccountSetupWithTemplate();
|
||||
} else {
|
||||
// No template, proceed to tasks step
|
||||
dispatch(setCurrentStep(currentStep + 1));
|
||||
}
|
||||
} else if (currentStep === 4) {
|
||||
// Complete setup after members step
|
||||
completeAccountSetup();
|
||||
} else {
|
||||
dispatch(setCurrentStep(currentStep + 1));
|
||||
}
|
||||
};
|
||||
|
||||
// Language switcher functionality
|
||||
const languages = [
|
||||
{ key: Language.EN, label: 'English', flag: '🇺🇸' },
|
||||
{ key: Language.ES, label: 'Español', flag: '🇪🇸' },
|
||||
{ key: Language.PT, label: 'Português', flag: '🇵🇹' },
|
||||
{ key: Language.DE, label: 'Deutsch', flag: '🇩🇪' },
|
||||
{ key: Language.ALB, label: 'Shqip', flag: '🇦🇱' },
|
||||
{ key: Language.ZH_CN, label: '简体中文', flag: '🇨🇳' }
|
||||
];
|
||||
|
||||
const handleLanguageChange = (languageKey: ILanguageType) => {
|
||||
dispatch(setLanguage(languageKey));
|
||||
i18n.changeLanguage(languageKey);
|
||||
};
|
||||
|
||||
const handleThemeToggle = () => {
|
||||
dispatch(toggleTheme());
|
||||
};
|
||||
|
||||
const languageMenuItems: MenuProps['items'] = languages.map(lang => ({
|
||||
key: lang.key,
|
||||
label: (
|
||||
<div className="flex items-center space-x-2">
|
||||
<span>{lang.flag}</span>
|
||||
<span>{lang.label}</span>
|
||||
</div>
|
||||
),
|
||||
onClick: () => handleLanguageChange(lang.key as ILanguageType)
|
||||
}));
|
||||
|
||||
const currentLanguage = languages.find(lang => lang.key === lng) || languages[0];
|
||||
|
||||
return (
|
||||
<div style={styles.container}>
|
||||
<div>
|
||||
<div
|
||||
className="account-setup-container min-h-screen w-full flex flex-col items-center py-8 px-4 relative"
|
||||
style={{ backgroundColor: token.colorBgLayout }}
|
||||
>
|
||||
{/* Controls - Top Right */}
|
||||
<div className="absolute top-6 right-6 flex items-center space-x-3">
|
||||
{/* Theme Switcher */}
|
||||
<Button
|
||||
type="text"
|
||||
size="small"
|
||||
icon={isDarkMode ? <SunOutlined /> : <MoonOutlined />}
|
||||
onClick={handleThemeToggle}
|
||||
className="flex items-center"
|
||||
style={{ color: token?.colorTextTertiary }}
|
||||
title={isDarkMode ? 'Switch to light mode' : 'Switch to dark mode'}
|
||||
/>
|
||||
|
||||
{/* Language Switcher */}
|
||||
<Dropdown
|
||||
menu={{ items: languageMenuItems }}
|
||||
placement="bottomRight"
|
||||
trigger={['click']}
|
||||
>
|
||||
<Button
|
||||
type="text"
|
||||
size="small"
|
||||
icon={<GlobalOutlined />}
|
||||
className="flex items-center space-x-2"
|
||||
style={{ color: token?.colorTextTertiary }}
|
||||
>
|
||||
<span>{currentLanguage.flag}</span>
|
||||
<span>{currentLanguage.label}</span>
|
||||
</Button>
|
||||
</Dropdown>
|
||||
</div>
|
||||
|
||||
{/* Logo */}
|
||||
<div className="mb-4">
|
||||
<img src={isDarkMode ? logoDark : logo} alt="Logo" width={235} height={50} />
|
||||
</div>
|
||||
<Title level={5} style={{ textAlign: 'center', margin: '4px 0 24px' }}>
|
||||
|
||||
{/* Title */}
|
||||
<Title
|
||||
level={3}
|
||||
className="text-center mb-6 font-semibold"
|
||||
style={{ color: token.colorText }}
|
||||
>
|
||||
{t('setupYourAccount')}
|
||||
</Title>
|
||||
<div style={styles.contentContainer}>
|
||||
<Space className={isDarkMode ? 'dark-mode' : ''} style={styles.space} direction="vertical">
|
||||
<Steps
|
||||
className={isContinueDisabled() ? 'step' : 'progress-steps'}
|
||||
current={currentStep}
|
||||
items={steps}
|
||||
style={styles.steps}
|
||||
/>
|
||||
<div className="step-content" style={styles.stepContent}>
|
||||
{steps[currentStep].content}
|
||||
|
||||
{/* Content Container */}
|
||||
<div
|
||||
className="w-full max-w-4xl rounded-lg shadow-lg mt-6 p-8"
|
||||
style={{
|
||||
backgroundColor: token.colorBgContainer,
|
||||
borderColor: token.colorBorder,
|
||||
border: `1px solid ${token.colorBorder}`,
|
||||
boxShadow: token.boxShadowTertiary
|
||||
}}
|
||||
>
|
||||
<div className="flex flex-col items-center space-y-6 w-full">
|
||||
{/* Steps */}
|
||||
<div className="w-full max-w-2xl">
|
||||
<Steps
|
||||
className={`${isContinueDisabled() ? 'step' : 'progress-steps'} ${isDarkMode ? 'dark-mode' : 'light-mode'}`}
|
||||
current={currentStep}
|
||||
items={steps}
|
||||
/>
|
||||
</div>
|
||||
<div style={styles.actionButtons} className="setup-action-buttons">
|
||||
<div
|
||||
style={{
|
||||
display: 'flex',
|
||||
justifyContent: currentStep !== 0 ? 'space-between' : 'flex-end',
|
||||
}}
|
||||
>
|
||||
|
||||
{/* Step Content */}
|
||||
<div className="w-full max-w-2xl flex flex-col items-center min-h-fit">
|
||||
<div className="step-content w-full">
|
||||
{steps[currentStep].content}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Action Buttons */}
|
||||
<div className="w-full max-w-2xl mt-8">
|
||||
<div className={`flex ${
|
||||
currentStep !== 0 ? 'justify-between' : 'justify-end'
|
||||
} items-center`}>
|
||||
{currentStep !== 0 && (
|
||||
<div>
|
||||
<div className="flex flex-col space-y-2">
|
||||
<Button
|
||||
style={{ padding: 0 }}
|
||||
type="link"
|
||||
className="my-7"
|
||||
onClick={() => dispatch(setCurrentStep(currentStep - 1))}
|
||||
className="p-0 font-medium"
|
||||
style={{ color: token.colorTextSecondary }}
|
||||
onClick={() => {
|
||||
if (currentStep === 1 && surveySubStep > 0) {
|
||||
// Go back within survey sub-steps
|
||||
dispatch(setSurveySubStep(surveySubStep - 1));
|
||||
} else {
|
||||
// Go back to previous main step
|
||||
dispatch(setCurrentStep(currentStep - 1));
|
||||
if (currentStep === 2) {
|
||||
// When going back to survey from next step, go to last sub-step
|
||||
dispatch(setSurveySubStep(2));
|
||||
}
|
||||
}
|
||||
}}
|
||||
>
|
||||
{t('goBack')}
|
||||
</Button>
|
||||
{currentStep === 3 && (
|
||||
{currentStep === 4 && (
|
||||
<Button
|
||||
style={{ color: isDarkMode ? '' : '#00000073', fontWeight: 500 }}
|
||||
type="link"
|
||||
className="my-7"
|
||||
onClick={() => completeAccountSetup(true)}
|
||||
className="p-0 font-medium"
|
||||
style={{ color: token.colorTextTertiary }}
|
||||
onClick={handleSkipMembers}
|
||||
loading={isSkipping}
|
||||
disabled={isSkipping}
|
||||
>
|
||||
{t('skipForNow')}
|
||||
{isSkipping ? t('skipping') : t('skipForNow')}
|
||||
</Button>
|
||||
)}
|
||||
</div>
|
||||
@@ -298,14 +620,13 @@ const AccountSetup: React.FC = () => {
|
||||
type="primary"
|
||||
htmlType="submit"
|
||||
disabled={isContinueDisabled()}
|
||||
className="mt-7 mb-7"
|
||||
onClick={nextStep}
|
||||
>
|
||||
{t('continue')}
|
||||
</Button>
|
||||
</div>
|
||||
</div>
|
||||
</Space>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
|
||||
@@ -1,14 +1,16 @@
|
||||
import { useEffect, memo, useMemo, useCallback } from 'react';
|
||||
import React, { useEffect, memo, useMemo, useCallback } from 'react';
|
||||
import { useMediaQuery } from 'react-responsive';
|
||||
import Col from 'antd/es/col';
|
||||
import Flex from 'antd/es/flex';
|
||||
import Row from 'antd/es/row';
|
||||
import Card from 'antd/es/card';
|
||||
|
||||
import GreetingWithTime from './greeting-with-time';
|
||||
import TasksList from '@/pages/home/task-list/tasks-list';
|
||||
import TodoList from '@/pages/home/todo-list/todo-list';
|
||||
import ProjectDrawer from '@/components/projects/project-drawer/project-drawer';
|
||||
import CreateProjectButton from '@/components/projects/project-create-button/project-create-button';
|
||||
import RecentAndFavouriteProjectList from '@/pages/home/recent-and-favourite-project-list/recent-and-favourite-project-list';
|
||||
import TodoList from './todo-list/todo-list';
|
||||
|
||||
import { useDocumentTitle } from '@/hooks/useDoumentTItle';
|
||||
import { useAppDispatch } from '@/hooks/useAppDispatch';
|
||||
@@ -19,7 +21,7 @@ import { fetchProjectCategories } from '@/features/projects/lookups/projectCateg
|
||||
import { fetchProjectHealth } from '@/features/projects/lookups/projectHealth/projectHealthSlice';
|
||||
import { fetchProjects } from '@/features/home-page/home-page.slice';
|
||||
import { createPortal } from 'react-dom';
|
||||
import React, { Suspense } from 'react';
|
||||
import UserActivityFeed from './user-activity-feed/user-activity-feed';
|
||||
|
||||
const DESKTOP_MIN_WIDTH = 1024;
|
||||
const TASK_LIST_MIN_WIDTH = 500;
|
||||
@@ -27,6 +29,9 @@ const SIDEBAR_MAX_WIDTH = 400;
|
||||
|
||||
// Lazy load heavy components
|
||||
const TaskDrawer = React.lazy(() => import('@/components/task-drawer/task-drawer'));
|
||||
const SurveyPromptModal = React.lazy(() =>
|
||||
import('@/components/survey/SurveyPromptModal').then(m => ({ default: m.SurveyPromptModal }))
|
||||
);
|
||||
|
||||
const HomePage = memo(() => {
|
||||
const dispatch = useAppDispatch();
|
||||
@@ -97,26 +102,6 @@ const HomePage = memo(() => {
|
||||
);
|
||||
}, [isDesktop, isOwnerOrAdmin]);
|
||||
|
||||
const MainContent = useMemo(() => {
|
||||
return isDesktop ? (
|
||||
<Flex gap={24} align="flex-start" className="w-full mt-12">
|
||||
<Flex style={desktopFlexStyle}>
|
||||
<TasksList />
|
||||
</Flex>
|
||||
<Flex vertical gap={24} style={sidebarFlexStyle}>
|
||||
<TodoList />
|
||||
<RecentAndFavouriteProjectList />
|
||||
</Flex>
|
||||
</Flex>
|
||||
) : (
|
||||
<Flex vertical gap={24} className="mt-6">
|
||||
<TasksList />
|
||||
<TodoList />
|
||||
<RecentAndFavouriteProjectList />
|
||||
</Flex>
|
||||
);
|
||||
}, [isDesktop, desktopFlexStyle, sidebarFlexStyle]);
|
||||
|
||||
return (
|
||||
<div className="my-24 min-h-[90vh]">
|
||||
<Col className="flex flex-col gap-6">
|
||||
@@ -124,24 +109,26 @@ const HomePage = memo(() => {
|
||||
{CreateProjectButtonComponent}
|
||||
</Col>
|
||||
|
||||
{MainContent}
|
||||
<Row gutter={[24, 24]} className="mt-12">
|
||||
<Col xs={24} lg={16}>
|
||||
<Flex vertical gap={24}>
|
||||
<TasksList />
|
||||
|
||||
{/* Use Suspense for lazy-loaded components with error boundary */}
|
||||
<Suspense fallback={<div>Loading...</div>}>
|
||||
{createPortal(
|
||||
<React.Suspense fallback={null}>
|
||||
<TaskDrawer />
|
||||
</React.Suspense>,
|
||||
document.body,
|
||||
'home-task-drawer'
|
||||
)}
|
||||
</Suspense>
|
||||
<TodoList />
|
||||
</Flex>
|
||||
</Col>
|
||||
|
||||
{createPortal(
|
||||
<ProjectDrawer onClose={handleProjectDrawerClose} />,
|
||||
document.body,
|
||||
'project-drawer'
|
||||
)}
|
||||
<Col xs={24} lg={8}>
|
||||
<Flex vertical gap={24}>
|
||||
<UserActivityFeed />
|
||||
|
||||
<RecentAndFavouriteProjectList />
|
||||
</Flex>
|
||||
</Col>
|
||||
</Row>
|
||||
|
||||
{createPortal(<TaskDrawer />, document.body, 'home-task-drawer')}
|
||||
{createPortal(<ProjectDrawer onClose={() => {}} />, document.body, 'project-drawer')}
|
||||
</div>
|
||||
);
|
||||
});
|
||||
|
||||
@@ -144,7 +144,7 @@ const TodoList = () => {
|
||||
</Form.Item>
|
||||
</Form>
|
||||
|
||||
<div style={{ maxHeight: 420, overflow: 'auto' }}>
|
||||
<div style={{ maxHeight: 300, overflow: 'auto' }}>
|
||||
{data?.body.length === 0 ? (
|
||||
<EmptyListPlaceholder
|
||||
imageSrc="https://s3.us-west-2.amazonaws.com/worklenz.com/assets/empty-box.webp"
|
||||
|
||||
@@ -0,0 +1,95 @@
|
||||
import React, { useCallback } from 'react';
|
||||
import { Table, Typography, Tooltip, theme } from 'antd';
|
||||
import { FileTextOutlined } from '@ant-design/icons';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { useAppDispatch } from '@/hooks/useAppDispatch';
|
||||
import { fromNow, formatDate } from '@/utils/dateUtils';
|
||||
import {
|
||||
setSelectedTaskId,
|
||||
setShowTaskDrawer,
|
||||
fetchTask,
|
||||
} from '@/features/task-drawer/task-drawer.slice';
|
||||
import { IUserRecentTask } from '@/types/home/user-activity.types';
|
||||
|
||||
const { Text } = Typography;
|
||||
|
||||
interface TaskActivityListProps {
|
||||
tasks: IUserRecentTask[];
|
||||
}
|
||||
|
||||
const TaskActivityList: React.FC<TaskActivityListProps> = React.memo(({ tasks }) => {
|
||||
const { t } = useTranslation('home');
|
||||
const dispatch = useAppDispatch();
|
||||
const { token } = theme.useToken();
|
||||
|
||||
const handleTaskClick = useCallback(
|
||||
(taskId: string, projectId: string) => {
|
||||
dispatch(setSelectedTaskId(taskId));
|
||||
dispatch(setShowTaskDrawer(true));
|
||||
dispatch(fetchTask({ taskId, projectId }));
|
||||
},
|
||||
[dispatch]
|
||||
);
|
||||
|
||||
const columns = [
|
||||
{
|
||||
key: 'task',
|
||||
render: (record: IUserRecentTask) => (
|
||||
<div
|
||||
style={{
|
||||
display: 'flex',
|
||||
alignItems: 'flex-start',
|
||||
gap: 12,
|
||||
width: '100%',
|
||||
cursor: 'pointer',
|
||||
padding: '8px 0'
|
||||
}}
|
||||
onClick={() => handleTaskClick(record.task_id, record.project_id)}
|
||||
aria-label={`${t('tasks.recentTaskAriaLabel')} ${record.task_name}`}
|
||||
>
|
||||
<div style={{
|
||||
marginTop: 2,
|
||||
color: token.colorPrimary,
|
||||
fontSize: 16
|
||||
}}>
|
||||
<FileTextOutlined />
|
||||
</div>
|
||||
<div style={{ flex: 1, minWidth: 0 }}>
|
||||
<div style={{ marginBottom: 4 }}>
|
||||
<Text strong style={{ fontSize: 14 }}>
|
||||
{record.task_name}
|
||||
</Text>
|
||||
</div>
|
||||
<div style={{ display: 'flex', justifyContent: 'space-between', alignItems: 'center' }}>
|
||||
<Text type="secondary" style={{ fontSize: 12 }}>
|
||||
{record.project_name}
|
||||
</Text>
|
||||
<Tooltip
|
||||
title={formatDate(record.last_activity_at, 'MMMM Do YYYY, h:mm:ss a')}
|
||||
placement="topRight"
|
||||
>
|
||||
<Text type="secondary" style={{ fontSize: 12 }}>
|
||||
{fromNow(record.last_activity_at)}
|
||||
</Text>
|
||||
</Tooltip>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
),
|
||||
},
|
||||
];
|
||||
|
||||
return (
|
||||
<Table
|
||||
className="custom-two-colors-row-table"
|
||||
dataSource={tasks}
|
||||
columns={columns}
|
||||
rowKey="task_id"
|
||||
showHeader={false}
|
||||
pagination={false}
|
||||
size="small"
|
||||
/>
|
||||
);
|
||||
});
|
||||
|
||||
export default TaskActivityList;
|
||||
@@ -0,0 +1,166 @@
|
||||
import React, { useCallback } from 'react';
|
||||
import { Table, Typography, Tag, Tooltip, Space, theme } from '@/shared/antd-imports';
|
||||
import { ClockCircleOutlined } from '@/shared/antd-imports';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { useAppDispatch } from '@/hooks/useAppDispatch';
|
||||
import { fromNow, formatDate } from '@/utils/dateUtils';
|
||||
import {
|
||||
setSelectedTaskId,
|
||||
setShowTaskDrawer,
|
||||
fetchTask,
|
||||
} from '@/features/task-drawer/task-drawer.slice';
|
||||
import { IUserTimeLoggedTask } from '@/types/home/user-activity.types';
|
||||
|
||||
const { Text } = Typography;
|
||||
|
||||
interface TimeLoggedTaskListProps {
|
||||
tasks: IUserTimeLoggedTask[];
|
||||
}
|
||||
|
||||
const TimeLoggedTaskList: React.FC<TimeLoggedTaskListProps> = React.memo(({ tasks }) => {
|
||||
const { t } = useTranslation('home');
|
||||
const dispatch = useAppDispatch();
|
||||
const { token } = theme.useToken();
|
||||
|
||||
const handleTaskClick = useCallback(
|
||||
(taskId: string, projectId: string) => {
|
||||
dispatch(setSelectedTaskId(taskId));
|
||||
dispatch(setShowTaskDrawer(true));
|
||||
dispatch(fetchTask({ taskId, projectId }));
|
||||
},
|
||||
[dispatch]
|
||||
);
|
||||
|
||||
const columns = [
|
||||
{
|
||||
key: 'task',
|
||||
render: (record: IUserTimeLoggedTask) => (
|
||||
<div
|
||||
style={{
|
||||
display: 'flex',
|
||||
alignItems: 'center',
|
||||
gap: 10,
|
||||
width: '100%',
|
||||
cursor: 'pointer',
|
||||
padding: '8px 0'
|
||||
}}
|
||||
onClick={() => handleTaskClick(record.task_id, record.project_id)}
|
||||
aria-label={`${t('tasks.timeLoggedTaskAriaLabel')} ${record.task_name}`}
|
||||
>
|
||||
{/* Clock Icon */}
|
||||
<div style={{
|
||||
color: token.colorSuccess,
|
||||
fontSize: 14,
|
||||
flexShrink: 0
|
||||
}}>
|
||||
<ClockCircleOutlined />
|
||||
</div>
|
||||
|
||||
{/* Main Content */}
|
||||
<div style={{ flex: 1, minWidth: 0 }}>
|
||||
{/* Task Name */}
|
||||
<div style={{ marginBottom: 2 }}>
|
||||
<Text
|
||||
strong
|
||||
style={{
|
||||
fontSize: 13,
|
||||
lineHeight: 1.4,
|
||||
color: token.colorText
|
||||
}}
|
||||
ellipsis={{ tooltip: record.task_name }}
|
||||
>
|
||||
{record.task_name}
|
||||
</Text>
|
||||
</div>
|
||||
|
||||
{/* Project Name */}
|
||||
<Text
|
||||
type="secondary"
|
||||
style={{
|
||||
fontSize: 11,
|
||||
lineHeight: 1.2,
|
||||
display: 'block',
|
||||
marginBottom: 4
|
||||
}}
|
||||
ellipsis={{ tooltip: record.project_name }}
|
||||
>
|
||||
{record.project_name}
|
||||
</Text>
|
||||
</div>
|
||||
|
||||
{/* Right Side - Time and Status */}
|
||||
<div style={{
|
||||
display: 'flex',
|
||||
flexDirection: 'column',
|
||||
alignItems: 'flex-end',
|
||||
gap: 3,
|
||||
flexShrink: 0
|
||||
}}>
|
||||
{/* Time Logged */}
|
||||
<div style={{ display: 'flex', alignItems: 'center', gap: 4 }}>
|
||||
<Tag
|
||||
color="success"
|
||||
style={{
|
||||
margin: 0,
|
||||
fontSize: 11,
|
||||
padding: '0 6px',
|
||||
height: 18,
|
||||
lineHeight: '16px',
|
||||
borderRadius: 3
|
||||
}}
|
||||
>
|
||||
{record.total_time_logged_string}
|
||||
</Tag>
|
||||
{record.logged_by_timer && (
|
||||
<Tag
|
||||
color="processing"
|
||||
style={{
|
||||
margin: 0,
|
||||
fontSize: 10,
|
||||
padding: '0 4px',
|
||||
height: 16,
|
||||
lineHeight: '14px',
|
||||
borderRadius: 2
|
||||
}}
|
||||
>
|
||||
{t('tasks.timerTag')}
|
||||
</Tag>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Time Ago */}
|
||||
<Tooltip
|
||||
title={formatDate(record.last_logged_at, 'MMMM Do YYYY, h:mm:ss a')}
|
||||
placement="topRight"
|
||||
>
|
||||
<Text
|
||||
type="secondary"
|
||||
style={{
|
||||
fontSize: 10,
|
||||
lineHeight: 1,
|
||||
color: token.colorTextTertiary
|
||||
}}
|
||||
>
|
||||
{fromNow(record.last_logged_at)}
|
||||
</Text>
|
||||
</Tooltip>
|
||||
</div>
|
||||
</div>
|
||||
),
|
||||
},
|
||||
];
|
||||
|
||||
return (
|
||||
<Table
|
||||
className="custom-two-colors-row-table"
|
||||
dataSource={tasks}
|
||||
columns={columns}
|
||||
rowKey="task_id"
|
||||
showHeader={false}
|
||||
pagination={false}
|
||||
size="small"
|
||||
/>
|
||||
);
|
||||
});
|
||||
|
||||
export default TimeLoggedTaskList;
|
||||
@@ -0,0 +1,19 @@
|
||||
.activity-feed-item:hover {
|
||||
background-color: var(--activity-hover, #fafafa);
|
||||
transform: translateY(-1px);
|
||||
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.activity-feed-item:active {
|
||||
transform: translateY(0);
|
||||
background-color: var(--activity-active, #f0f0f0);
|
||||
}
|
||||
|
||||
/* Dark theme support */
|
||||
[data-theme="dark"] .activity-feed-item:hover {
|
||||
background-color: var(--activity-hover, #262626);
|
||||
}
|
||||
|
||||
[data-theme="dark"] .activity-feed-item:active {
|
||||
background-color: var(--activity-active, #1f1f1f);
|
||||
}
|
||||
@@ -0,0 +1,203 @@
|
||||
import React, { useMemo, useCallback, useEffect } from 'react';
|
||||
import { Card, Segmented, Skeleton, Empty, Typography, Alert, Button, Tooltip } from '@/shared/antd-imports';
|
||||
import { ClockCircleOutlined, UnorderedListOutlined, SyncOutlined } from '@/shared/antd-imports';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { useAppSelector } from '@/hooks/useAppSelector';
|
||||
import { useAppDispatch } from '@/hooks/useAppDispatch';
|
||||
import { ActivityFeedType } from '@/types/home/user-activity.types';
|
||||
import { setActiveTab } from '@/features/home-page/user-activity.slice';
|
||||
import {
|
||||
useGetUserRecentTasksQuery,
|
||||
useGetUserTimeLoggedTasksQuery,
|
||||
} from '@/api/home-page/user-activity.api.service';
|
||||
import TaskActivityList from './task-activity-list';
|
||||
import TimeLoggedTaskList from './time-logged-task-list';
|
||||
|
||||
const { Title } = Typography;
|
||||
|
||||
const UserActivityFeed: React.FC = () => {
|
||||
const { t } = useTranslation('home');
|
||||
const dispatch = useAppDispatch();
|
||||
const { activeTab } = useAppSelector(state => state.userActivityReducer);
|
||||
|
||||
const {
|
||||
data: recentTasksData,
|
||||
isLoading: loadingRecentTasks,
|
||||
error: recentTasksError,
|
||||
refetch: refetchRecentTasks,
|
||||
} = useGetUserRecentTasksQuery(
|
||||
{ limit: 10 },
|
||||
{
|
||||
skip: false,
|
||||
refetchOnMountOrArgChange: true
|
||||
}
|
||||
);
|
||||
|
||||
const {
|
||||
data: timeLoggedTasksData,
|
||||
isLoading: loadingTimeLoggedTasks,
|
||||
error: timeLoggedTasksError,
|
||||
refetch: refetchTimeLoggedTasks,
|
||||
} = useGetUserTimeLoggedTasksQuery(
|
||||
{ limit: 10 },
|
||||
{
|
||||
skip: false,
|
||||
refetchOnMountOrArgChange: true
|
||||
}
|
||||
);
|
||||
|
||||
const recentTasks = useMemo(() => {
|
||||
if (!recentTasksData) return [];
|
||||
// Handle both array and object responses from the API
|
||||
if (Array.isArray(recentTasksData)) {
|
||||
return recentTasksData;
|
||||
}
|
||||
// If it's an object with a data property (common API pattern)
|
||||
if (recentTasksData && typeof recentTasksData === 'object' && 'data' in recentTasksData) {
|
||||
const data = (recentTasksData as any).data;
|
||||
return Array.isArray(data) ? data : [];
|
||||
}
|
||||
// If it's a different object structure, try to extract tasks
|
||||
if (recentTasksData && typeof recentTasksData === 'object') {
|
||||
const possibleArrays = Object.values(recentTasksData as any).filter(Array.isArray);
|
||||
return possibleArrays.length > 0 ? possibleArrays[0] : [];
|
||||
}
|
||||
return [];
|
||||
}, [recentTasksData]);
|
||||
|
||||
const timeLoggedTasks = useMemo(() => {
|
||||
if (!timeLoggedTasksData) return [];
|
||||
// Handle both array and object responses from the API
|
||||
if (Array.isArray(timeLoggedTasksData)) {
|
||||
return timeLoggedTasksData;
|
||||
}
|
||||
// If it's an object with a data property (common API pattern)
|
||||
if (timeLoggedTasksData && typeof timeLoggedTasksData === 'object' && 'data' in timeLoggedTasksData) {
|
||||
const data = (timeLoggedTasksData as any).data;
|
||||
return Array.isArray(data) ? data : [];
|
||||
}
|
||||
// If it's a different object structure, try to extract tasks
|
||||
if (timeLoggedTasksData && typeof timeLoggedTasksData === 'object') {
|
||||
const possibleArrays = Object.values(timeLoggedTasksData as any).filter(Array.isArray);
|
||||
return possibleArrays.length > 0 ? possibleArrays[0] : [];
|
||||
}
|
||||
return [];
|
||||
}, [timeLoggedTasksData]);
|
||||
|
||||
const segmentOptions = useMemo(
|
||||
() => [
|
||||
{
|
||||
value: ActivityFeedType.TIME_LOGGED_TASKS,
|
||||
label: (
|
||||
<span style={{ display: 'flex', alignItems: 'center', gap: 4 }}>
|
||||
<ClockCircleOutlined style={{ fontSize: 14 }} />
|
||||
{t('tasks.timeLoggedSegment')}
|
||||
</span>
|
||||
),
|
||||
},
|
||||
{
|
||||
value: ActivityFeedType.RECENT_TASKS,
|
||||
label: (
|
||||
<span style={{ display: 'flex', alignItems: 'center', gap: 4 }}>
|
||||
<UnorderedListOutlined style={{ fontSize: 14 }} />
|
||||
{t('tasks.recentTasksSegment')}
|
||||
</span>
|
||||
),
|
||||
},
|
||||
],
|
||||
[t]
|
||||
);
|
||||
|
||||
const handleTabChange = useCallback(
|
||||
(value: ActivityFeedType) => {
|
||||
dispatch(setActiveTab(value));
|
||||
},
|
||||
[dispatch]
|
||||
);
|
||||
|
||||
// Refetch data when the active tab changes
|
||||
useEffect(() => {
|
||||
if (activeTab === ActivityFeedType.RECENT_TASKS) {
|
||||
refetchRecentTasks();
|
||||
} else if (activeTab === ActivityFeedType.TIME_LOGGED_TASKS) {
|
||||
refetchTimeLoggedTasks();
|
||||
}
|
||||
}, [activeTab, refetchRecentTasks, refetchTimeLoggedTasks]);
|
||||
|
||||
const handleRefresh = useCallback(() => {
|
||||
if (activeTab === ActivityFeedType.TIME_LOGGED_TASKS) {
|
||||
refetchTimeLoggedTasks();
|
||||
} else {
|
||||
refetchRecentTasks();
|
||||
}
|
||||
}, [activeTab, refetchRecentTasks, refetchTimeLoggedTasks]);
|
||||
|
||||
const isLoading = activeTab === ActivityFeedType.TIME_LOGGED_TASKS ? loadingTimeLoggedTasks : loadingRecentTasks;
|
||||
const currentCount = activeTab === ActivityFeedType.TIME_LOGGED_TASKS ? timeLoggedTasks.length : recentTasks.length;
|
||||
|
||||
const renderContent = () => {
|
||||
if (activeTab === ActivityFeedType.TIME_LOGGED_TASKS) {
|
||||
if (loadingTimeLoggedTasks) {
|
||||
return <Skeleton active />;
|
||||
}
|
||||
if (timeLoggedTasksError) {
|
||||
return <Alert message={t('tasks.errorLoadingTimeLoggedTasks')} type="error" showIcon />;
|
||||
}
|
||||
if (timeLoggedTasks.length === 0) {
|
||||
return <Empty description={t('tasks.noTimeLoggedTasks')} />;
|
||||
}
|
||||
return (
|
||||
<div style={{ maxHeight: 450, overflow: 'auto' }}>
|
||||
<TimeLoggedTaskList tasks={timeLoggedTasks} />
|
||||
</div>
|
||||
);
|
||||
} else if (activeTab === ActivityFeedType.RECENT_TASKS) {
|
||||
if (loadingRecentTasks) {
|
||||
return <Skeleton active />;
|
||||
}
|
||||
if (recentTasksError) {
|
||||
return <Alert message={t('tasks.errorLoadingRecentTasks')} type="error" showIcon />;
|
||||
}
|
||||
if (recentTasks.length === 0) {
|
||||
return <Empty description={t('tasks.noRecentTasks')} />;
|
||||
}
|
||||
return (
|
||||
<div style={{ maxHeight: 450, overflow: 'auto' }}>
|
||||
<TaskActivityList tasks={recentTasks} />
|
||||
</div>
|
||||
);
|
||||
}
|
||||
return null;
|
||||
};
|
||||
|
||||
return (
|
||||
<Card
|
||||
title={
|
||||
<Typography.Title level={5} style={{ marginBlockEnd: 0 }}>
|
||||
{t('tasks.recentActivity')} ({currentCount})
|
||||
</Typography.Title>
|
||||
}
|
||||
extra={
|
||||
<Tooltip title={t('tasks.refresh')}>
|
||||
<Button
|
||||
shape="circle"
|
||||
icon={<SyncOutlined spin={isLoading} />}
|
||||
onClick={handleRefresh}
|
||||
/>
|
||||
</Tooltip>
|
||||
}
|
||||
style={{ width: '100%' }}
|
||||
>
|
||||
<Segmented
|
||||
options={segmentOptions}
|
||||
value={activeTab}
|
||||
onChange={handleTabChange}
|
||||
style={{ marginBottom: 16, width: '100%' }}
|
||||
block
|
||||
/>
|
||||
{renderContent()}
|
||||
</Card>
|
||||
);
|
||||
};
|
||||
|
||||
export default React.memo(UserActivityFeed);
|
||||
63
worklenz-frontend/src/pages/projects/ProjectGanttView.tsx
Normal file
63
worklenz-frontend/src/pages/projects/ProjectGanttView.tsx
Normal file
@@ -0,0 +1,63 @@
|
||||
import React, { useMemo } from 'react';
|
||||
import { useParams } from 'react-router-dom';
|
||||
import { AdvancedGanttChart } from '../../components/advanced-gantt';
|
||||
import { useAppSelector } from '../../hooks/useAppSelector';
|
||||
import { GanttTask } from '../../types/advanced-gantt.types';
|
||||
|
||||
const ProjectGanttView: React.FC = () => {
|
||||
const { projectId } = useParams<{ projectId: string }>();
|
||||
|
||||
// Get tasks from your Redux store (adjust based on your actual state structure)
|
||||
const tasks = useAppSelector(state => state.tasksReducer?.tasks || []);
|
||||
|
||||
// Transform your tasks to GanttTask format
|
||||
const ganttTasks = useMemo((): GanttTask[] => {
|
||||
return tasks.map(task => ({
|
||||
id: task.id,
|
||||
name: task.name,
|
||||
startDate: task.start_date ? new Date(task.start_date) : new Date(),
|
||||
endDate: task.end_date ? new Date(task.end_date) : new Date(),
|
||||
progress: task.progress || 0,
|
||||
type: 'task',
|
||||
status: task.status || 'not-started',
|
||||
priority: task.priority || 'medium',
|
||||
assignee: task.assignee ? {
|
||||
id: task.assignee.id,
|
||||
name: task.assignee.name,
|
||||
avatar: task.assignee.avatar,
|
||||
} : undefined,
|
||||
parent: task.parent_task_id,
|
||||
level: task.level || 0,
|
||||
// Map other fields as needed
|
||||
}));
|
||||
}, [tasks]);
|
||||
|
||||
const handleTaskUpdate = (taskId: string, updates: Partial<GanttTask>) => {
|
||||
// Implement your task update logic here
|
||||
console.log('Update task:', taskId, updates);
|
||||
// Dispatch Redux action to update task
|
||||
};
|
||||
|
||||
const handleTaskMove = (taskId: string, newDates: { start: Date; end: Date }) => {
|
||||
// Implement your task move logic here
|
||||
console.log('Move task:', taskId, newDates);
|
||||
// Dispatch Redux action to update task dates
|
||||
};
|
||||
|
||||
return (
|
||||
<div className="project-gantt-view h-full">
|
||||
<AdvancedGanttChart
|
||||
tasks={ganttTasks}
|
||||
onTaskUpdate={handleTaskUpdate}
|
||||
onTaskMove={handleTaskMove}
|
||||
enableDragDrop={true}
|
||||
enableResize={true}
|
||||
enableProgressEdit={true}
|
||||
enableInlineEdit={true}
|
||||
className="h-full"
|
||||
/>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
export default ProjectGanttView;
|
||||
@@ -0,0 +1,583 @@
|
||||
import { useEffect, useState, useRef, useMemo, useCallback } from 'react';
|
||||
import { useAppSelector } from '@/hooks/useAppSelector';
|
||||
|
||||
import TaskListFilters from '../taskList/task-list-filters/task-list-filters';
|
||||
import { Flex, Skeleton } from 'antd';
|
||||
import BoardSectionCardContainer from './board-section/board-section-container';
|
||||
import {
|
||||
fetchBoardTaskGroups,
|
||||
reorderTaskGroups,
|
||||
moveTaskBetweenGroups,
|
||||
IGroupBy,
|
||||
updateTaskProgress,
|
||||
} from '@features/board/board-slice';
|
||||
import { useAppDispatch } from '@/hooks/useAppDispatch';
|
||||
import {
|
||||
DndContext,
|
||||
DragEndEvent,
|
||||
DragOverEvent,
|
||||
DragStartEvent,
|
||||
closestCenter,
|
||||
DragOverlay,
|
||||
MouseSensor,
|
||||
TouchSensor,
|
||||
useSensor,
|
||||
useSensors,
|
||||
getFirstCollision,
|
||||
pointerWithin,
|
||||
rectIntersection,
|
||||
UniqueIdentifier,
|
||||
} from '@dnd-kit/core';
|
||||
import BoardViewTaskCard from './board-section/board-task-card/board-view-task-card';
|
||||
import { fetchStatusesCategories } from '@/features/taskAttributes/taskStatusSlice';
|
||||
import useTabSearchParam from '@/hooks/useTabSearchParam';
|
||||
import { useSocket } from '@/socket/socketContext';
|
||||
import { useAuthService } from '@/hooks/useAuth';
|
||||
import { SocketEvents } from '@/shared/socket-events';
|
||||
import alertService from '@/services/alerts/alertService';
|
||||
import { useMixpanelTracking } from '@/hooks/useMixpanelTracking';
|
||||
import { evt_project_board_visit, evt_project_task_list_drag_and_move } from '@/shared/worklenz-analytics-events';
|
||||
import { ITaskStatusCreateRequest } from '@/types/tasks/task-status-create-request';
|
||||
import { statusApiService } from '@/api/taskAttributes/status/status.api.service';
|
||||
import logger from '@/utils/errorLogger';
|
||||
import { checkTaskDependencyStatus } from '@/utils/check-task-dependency-status';
|
||||
import { debounce } from 'lodash';
|
||||
import { ITaskListPriorityChangeResponse } from '@/types/tasks/task-list-priority.types';
|
||||
import { updateTaskPriority as updateBoardTaskPriority } from '@/features/board/board-slice';
|
||||
|
||||
interface DroppableContainer {
|
||||
id: UniqueIdentifier;
|
||||
data: {
|
||||
current?: {
|
||||
type?: string;
|
||||
};
|
||||
};
|
||||
}
|
||||
|
||||
const ProjectViewBoard = () => {
|
||||
const dispatch = useAppDispatch();
|
||||
const { projectView } = useTabSearchParam();
|
||||
const { socket } = useSocket();
|
||||
const authService = useAuthService();
|
||||
const currentSession = authService.getCurrentSession();
|
||||
const { trackMixpanelEvent } = useMixpanelTracking();
|
||||
const [currentTaskIndex, setCurrentTaskIndex] = useState(-1);
|
||||
// Add local loading state to immediately show skeleton
|
||||
const [isLoading, setIsLoading] = useState(true);
|
||||
|
||||
const { projectId } = useAppSelector(state => state.projectReducer);
|
||||
const { taskGroups, groupBy, loadingGroups, search, archived } = useAppSelector(state => state.boardReducer);
|
||||
const { statusCategories, loading: loadingStatusCategories } = useAppSelector(
|
||||
state => state.taskStatusReducer
|
||||
);
|
||||
const [activeItem, setActiveItem] = useState<any>(null);
|
||||
|
||||
// Store the original source group ID when drag starts
|
||||
const originalSourceGroupIdRef = useRef<string | null>(null);
|
||||
const lastOverId = useRef<UniqueIdentifier | null>(null);
|
||||
const recentlyMovedToNewContainer = useRef(false);
|
||||
const [clonedItems, setClonedItems] = useState<any>(null);
|
||||
const isDraggingRef = useRef(false);
|
||||
|
||||
// Update loading state based on all loading conditions
|
||||
useEffect(() => {
|
||||
setIsLoading(loadingGroups || loadingStatusCategories);
|
||||
}, [loadingGroups, loadingStatusCategories]);
|
||||
|
||||
// Load data efficiently with async/await and Promise.all
|
||||
useEffect(() => {
|
||||
const loadData = async () => {
|
||||
if (projectId && groupBy && projectView === 'kanban') {
|
||||
const promises = [];
|
||||
|
||||
if (!loadingGroups) {
|
||||
promises.push(dispatch(fetchBoardTaskGroups(projectId)));
|
||||
}
|
||||
|
||||
if (!statusCategories.length) {
|
||||
promises.push(dispatch(fetchStatusesCategories()));
|
||||
}
|
||||
|
||||
// Wait for all data to load
|
||||
await Promise.all(promises);
|
||||
}
|
||||
};
|
||||
|
||||
loadData();
|
||||
}, [dispatch, projectId, groupBy, projectView, search, archived]);
|
||||
|
||||
// Create sensors with memoization to prevent unnecessary re-renders
|
||||
const sensors = useSensors(
|
||||
useSensor(MouseSensor, {
|
||||
activationConstraint: {
|
||||
distance: 10,
|
||||
delay: 100,
|
||||
tolerance: 5,
|
||||
},
|
||||
}),
|
||||
useSensor(TouchSensor, {
|
||||
activationConstraint: {
|
||||
delay: 250,
|
||||
tolerance: 5,
|
||||
},
|
||||
})
|
||||
);
|
||||
|
||||
const collisionDetectionStrategy = useCallback(
|
||||
(args: {
|
||||
active: { id: UniqueIdentifier; data: { current?: { type?: string } } };
|
||||
droppableContainers: DroppableContainer[];
|
||||
}) => {
|
||||
if (activeItem?.type === 'section') {
|
||||
return closestCenter({
|
||||
...args,
|
||||
droppableContainers: args.droppableContainers.filter(
|
||||
(container: DroppableContainer) => container.data.current?.type === 'section'
|
||||
),
|
||||
});
|
||||
}
|
||||
|
||||
// Start by finding any intersecting droppable
|
||||
const pointerIntersections = pointerWithin(args);
|
||||
const intersections =
|
||||
pointerIntersections.length > 0
|
||||
? pointerIntersections
|
||||
: rectIntersection(args);
|
||||
let overId = getFirstCollision(intersections, 'id');
|
||||
|
||||
if (overId !== null) {
|
||||
const overContainer = args.droppableContainers.find(
|
||||
(container: DroppableContainer) => container.id === overId
|
||||
);
|
||||
|
||||
if (overContainer?.data.current?.type === 'section') {
|
||||
const containerItems = taskGroups.find(
|
||||
(group) => group.id === overId
|
||||
)?.tasks || [];
|
||||
|
||||
if (containerItems.length > 0) {
|
||||
overId = closestCenter({
|
||||
...args,
|
||||
droppableContainers: args.droppableContainers.filter(
|
||||
(container: DroppableContainer) =>
|
||||
container.id !== overId &&
|
||||
container.data.current?.type === 'task'
|
||||
),
|
||||
})[0]?.id;
|
||||
}
|
||||
}
|
||||
|
||||
lastOverId.current = overId;
|
||||
return [{ id: overId }];
|
||||
}
|
||||
|
||||
if (recentlyMovedToNewContainer.current) {
|
||||
lastOverId.current = activeItem?.id;
|
||||
}
|
||||
|
||||
return lastOverId.current ? [{ id: lastOverId.current }] : [];
|
||||
},
|
||||
[activeItem, taskGroups]
|
||||
);
|
||||
|
||||
const handleTaskProgress = (data: {
|
||||
id: string;
|
||||
status: string;
|
||||
complete_ratio: number;
|
||||
completed_count: number;
|
||||
total_tasks_count: number;
|
||||
parent_task: string;
|
||||
}) => {
|
||||
dispatch(updateTaskProgress(data));
|
||||
};
|
||||
|
||||
// Debounced move task function to prevent rapid updates
|
||||
const debouncedMoveTask = useCallback(
|
||||
debounce((taskId: string, sourceGroupId: string, targetGroupId: string, targetIndex: number) => {
|
||||
dispatch(
|
||||
moveTaskBetweenGroups({
|
||||
taskId,
|
||||
sourceGroupId,
|
||||
targetGroupId,
|
||||
targetIndex,
|
||||
})
|
||||
);
|
||||
}, 100),
|
||||
[dispatch]
|
||||
);
|
||||
|
||||
const handleDragStart = (event: DragStartEvent) => {
|
||||
const { active } = event;
|
||||
isDraggingRef.current = true;
|
||||
setActiveItem(active.data.current);
|
||||
setCurrentTaskIndex(active.data.current?.sortable.index);
|
||||
if (active.data.current?.type === 'task') {
|
||||
originalSourceGroupIdRef.current = active.data.current.sectionId;
|
||||
}
|
||||
setClonedItems(taskGroups);
|
||||
};
|
||||
|
||||
const findGroupForId = (id: string) => {
|
||||
// If id is a sectionId
|
||||
if (taskGroups.some(group => group.id === id)) return id;
|
||||
// If id is a taskId, find the group containing it
|
||||
const group = taskGroups.find(g => g.tasks.some(t => t.id === id));
|
||||
return group?.id;
|
||||
};
|
||||
|
||||
const handleDragOver = (event: DragOverEvent) => {
|
||||
try {
|
||||
if (!isDraggingRef.current) return;
|
||||
|
||||
const { active, over } = event;
|
||||
if (!over) return;
|
||||
|
||||
// Get the ids
|
||||
const activeId = active.id;
|
||||
const overId = over.id;
|
||||
|
||||
// Find the group (section) for each
|
||||
const activeGroupId = findGroupForId(activeId as string);
|
||||
const overGroupId = findGroupForId(overId as string);
|
||||
|
||||
// Only move if both groups exist and are different, and the active is a task
|
||||
if (
|
||||
activeGroupId &&
|
||||
overGroupId &&
|
||||
active.data.current?.type === 'task'
|
||||
) {
|
||||
// Find the target index in the over group
|
||||
const targetGroup = taskGroups.find(g => g.id === overGroupId);
|
||||
let targetIndex = 0;
|
||||
if (targetGroup) {
|
||||
// If over is a task, insert before it; if over is a section, append to end
|
||||
if (over.data.current?.type === 'task') {
|
||||
targetIndex = targetGroup.tasks.findIndex(t => t.id === overId);
|
||||
if (targetIndex === -1) targetIndex = targetGroup.tasks.length;
|
||||
} else {
|
||||
targetIndex = targetGroup.tasks.length;
|
||||
}
|
||||
}
|
||||
// Use debounced move task to prevent rapid updates
|
||||
debouncedMoveTask(
|
||||
activeId as string,
|
||||
activeGroupId,
|
||||
overGroupId,
|
||||
targetIndex
|
||||
);
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('handleDragOver error:', error);
|
||||
}
|
||||
};
|
||||
|
||||
const handlePriorityChange = (taskId: string, priorityId: string) => {
|
||||
if (!taskId || !priorityId || !socket) return;
|
||||
|
||||
const payload = {
|
||||
task_id: taskId,
|
||||
priority_id: priorityId,
|
||||
team_id: currentSession?.team_id,
|
||||
};
|
||||
|
||||
socket.emit(SocketEvents.TASK_PRIORITY_CHANGE.toString(), JSON.stringify(payload));
|
||||
socket.once(SocketEvents.TASK_PRIORITY_CHANGE.toString(), (data: ITaskListPriorityChangeResponse) => {
|
||||
dispatch(updateBoardTaskPriority(data));
|
||||
});
|
||||
};
|
||||
|
||||
const handleDragEnd = async (event: DragEndEvent) => {
|
||||
isDraggingRef.current = false;
|
||||
const { active, over } = event;
|
||||
|
||||
if (!over || !projectId) {
|
||||
setActiveItem(null);
|
||||
originalSourceGroupIdRef.current = null;
|
||||
setClonedItems(null);
|
||||
return;
|
||||
}
|
||||
|
||||
const isActiveTask = active.data.current?.type === 'task';
|
||||
const isActiveSection = active.data.current?.type === 'section';
|
||||
|
||||
// Handle task dragging between columns
|
||||
if (isActiveTask) {
|
||||
const task = active.data.current?.task;
|
||||
|
||||
// Use the original source group ID from ref instead of the potentially modified one
|
||||
const sourceGroupId = originalSourceGroupIdRef.current || active.data.current?.sectionId;
|
||||
|
||||
// Fix: Ensure we correctly identify the target group ID
|
||||
let targetGroupId;
|
||||
if (over.data.current?.type === 'task') {
|
||||
// If dropping on a task, get its section ID
|
||||
targetGroupId = over.data.current?.sectionId;
|
||||
} else if (over.data.current?.type === 'section') {
|
||||
// If dropping directly on a section
|
||||
targetGroupId = over.id;
|
||||
} else {
|
||||
// Fallback to the over ID if type is not specified
|
||||
targetGroupId = over.id;
|
||||
}
|
||||
|
||||
// Find source and target groups
|
||||
const sourceGroup = taskGroups.find(group => group.id === sourceGroupId);
|
||||
const targetGroup = taskGroups.find(group => group.id === targetGroupId);
|
||||
|
||||
if (!sourceGroup || !targetGroup || !task) {
|
||||
logger.error('Could not find source or target group, or task is undefined');
|
||||
setActiveItem(null);
|
||||
originalSourceGroupIdRef.current = null; // Reset the ref
|
||||
return;
|
||||
}
|
||||
|
||||
if (targetGroupId !== sourceGroupId) {
|
||||
const canContinue = await checkTaskDependencyStatus(task.id, targetGroupId);
|
||||
if (!canContinue) {
|
||||
alertService.error(
|
||||
'Task is not completed',
|
||||
'Please complete the task dependencies before proceeding'
|
||||
);
|
||||
dispatch(
|
||||
moveTaskBetweenGroups({
|
||||
taskId: task.id,
|
||||
sourceGroupId: targetGroupId, // Current group (where it was moved optimistically)
|
||||
targetGroupId: sourceGroupId, // Move it back to the original source group
|
||||
targetIndex: currentTaskIndex !== -1 ? currentTaskIndex : 0, // Original position or append to end
|
||||
})
|
||||
);
|
||||
|
||||
setActiveItem(null);
|
||||
originalSourceGroupIdRef.current = null;
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
// Find indices
|
||||
let fromIndex = sourceGroup.tasks.findIndex(t => t.id === task.id);
|
||||
// Handle case where task is not found in source group (might have been moved already in UI)
|
||||
if (fromIndex === -1) {
|
||||
logger.info('Task not found in source group. Using task sort_order from task object.');
|
||||
|
||||
// Use the sort_order from the task object itself
|
||||
const fromSortOrder = task.sort_order;
|
||||
|
||||
// Calculate target index and position
|
||||
let toIndex = -1;
|
||||
if (over.data.current?.type === 'task') {
|
||||
const overTaskId = over.data.current?.task.id;
|
||||
toIndex = targetGroup.tasks.findIndex(t => t.id === overTaskId);
|
||||
} else {
|
||||
// If dropping on a section, append to the end
|
||||
toIndex = targetGroup.tasks.length;
|
||||
}
|
||||
|
||||
// Calculate toPos similar to Angular implementation
|
||||
const toPos = targetGroup.tasks[toIndex]?.sort_order ||
|
||||
targetGroup.tasks[targetGroup.tasks.length - 1]?.sort_order ||
|
||||
-1;
|
||||
|
||||
// Prepare socket event payload
|
||||
const body = {
|
||||
project_id: projectId,
|
||||
from_index: fromSortOrder,
|
||||
to_index: toPos,
|
||||
to_last_index: !toPos,
|
||||
from_group: sourceGroupId,
|
||||
to_group: targetGroupId,
|
||||
group_by: groupBy || 'status',
|
||||
task,
|
||||
team_id: currentSession?.team_id
|
||||
};
|
||||
|
||||
// Emit socket event
|
||||
if (socket) {
|
||||
socket.emit(SocketEvents.TASK_SORT_ORDER_CHANGE.toString(), body);
|
||||
|
||||
// Set up listener for task progress update
|
||||
socket.once(SocketEvents.TASK_SORT_ORDER_CHANGE.toString(), () => {
|
||||
if (task.is_sub_task) {
|
||||
socket.emit(SocketEvents.GET_TASK_PROGRESS.toString(), task.parent_task_id);
|
||||
} else {
|
||||
socket.emit(SocketEvents.GET_TASK_PROGRESS.toString(), task.id);
|
||||
}
|
||||
});
|
||||
|
||||
// Handle priority change if groupBy is priority
|
||||
if (groupBy === IGroupBy.PRIORITY) {
|
||||
handlePriorityChange(task.id, targetGroupId);
|
||||
}
|
||||
}
|
||||
|
||||
// Track analytics event
|
||||
trackMixpanelEvent(evt_project_task_list_drag_and_move);
|
||||
|
||||
setActiveItem(null);
|
||||
originalSourceGroupIdRef.current = null; // Reset the ref
|
||||
return;
|
||||
}
|
||||
|
||||
// Calculate target index and position
|
||||
let toIndex = -1;
|
||||
if (over.data.current?.type === 'task') {
|
||||
const overTaskId = over.data.current?.task.id;
|
||||
toIndex = targetGroup.tasks.findIndex(t => t.id === overTaskId);
|
||||
} else {
|
||||
// If dropping on a section, append to the end
|
||||
toIndex = targetGroup.tasks.length;
|
||||
}
|
||||
|
||||
// Calculate toPos similar to Angular implementation
|
||||
const toPos = targetGroup.tasks[toIndex]?.sort_order ||
|
||||
targetGroup.tasks[targetGroup.tasks.length - 1]?.sort_order ||
|
||||
-1;
|
||||
// Prepare socket event payload
|
||||
const body = {
|
||||
project_id: projectId,
|
||||
from_index: sourceGroup.tasks[fromIndex].sort_order,
|
||||
to_index: toPos,
|
||||
to_last_index: !toPos,
|
||||
from_group: sourceGroupId, // Use the direct IDs instead of group objects
|
||||
to_group: targetGroupId, // Use the direct IDs instead of group objects
|
||||
group_by: groupBy || 'status', // Use the current groupBy value
|
||||
task,
|
||||
team_id: currentSession?.team_id
|
||||
};
|
||||
// Emit socket event
|
||||
if (socket) {
|
||||
socket.emit(SocketEvents.TASK_SORT_ORDER_CHANGE.toString(), body);
|
||||
|
||||
// Set up listener for task progress update
|
||||
socket.once(SocketEvents.TASK_SORT_ORDER_CHANGE.toString(), () => {
|
||||
if (task.is_sub_task) {
|
||||
socket.emit(SocketEvents.GET_TASK_PROGRESS.toString(), task.parent_task_id);
|
||||
} else {
|
||||
socket.emit(SocketEvents.GET_TASK_PROGRESS.toString(), task.id);
|
||||
}
|
||||
});
|
||||
}
|
||||
// Track analytics event
|
||||
trackMixpanelEvent(evt_project_task_list_drag_and_move);
|
||||
}
|
||||
// Handle column reordering
|
||||
else if (isActiveSection) {
|
||||
// Don't allow reordering if groupBy is phases
|
||||
if (groupBy === IGroupBy.PHASE) {
|
||||
setActiveItem(null);
|
||||
originalSourceGroupIdRef.current = null;
|
||||
return;
|
||||
}
|
||||
|
||||
const sectionId = active.id;
|
||||
const fromIndex = taskGroups.findIndex(group => group.id === sectionId);
|
||||
const toIndex = taskGroups.findIndex(group => group.id === over.id);
|
||||
|
||||
if (fromIndex !== -1 && toIndex !== -1) {
|
||||
// Create a new array with the reordered groups
|
||||
const reorderedGroups = [...taskGroups];
|
||||
const [movedGroup] = reorderedGroups.splice(fromIndex, 1);
|
||||
reorderedGroups.splice(toIndex, 0, movedGroup);
|
||||
|
||||
// Dispatch action to reorder columns with the new array
|
||||
dispatch(reorderTaskGroups(reorderedGroups));
|
||||
|
||||
// Prepare column order for API
|
||||
const columnOrder = reorderedGroups.map(group => group.id);
|
||||
|
||||
// Call API to update status order
|
||||
try {
|
||||
// Use the correct API endpoint based on the Angular code
|
||||
const requestBody: ITaskStatusCreateRequest = {
|
||||
status_order: columnOrder
|
||||
};
|
||||
|
||||
const response = await statusApiService.updateStatusOrder(requestBody, projectId);
|
||||
if (!response.done) {
|
||||
const revertedGroups = [...reorderedGroups];
|
||||
const [movedBackGroup] = revertedGroups.splice(toIndex, 1);
|
||||
revertedGroups.splice(fromIndex, 0, movedBackGroup);
|
||||
dispatch(reorderTaskGroups(revertedGroups));
|
||||
alertService.error('Failed to update column order', 'Please try again');
|
||||
}
|
||||
} catch (error) {
|
||||
// Revert the change if API call fails
|
||||
const revertedGroups = [...reorderedGroups];
|
||||
const [movedBackGroup] = revertedGroups.splice(toIndex, 1);
|
||||
revertedGroups.splice(fromIndex, 0, movedBackGroup);
|
||||
dispatch(reorderTaskGroups(revertedGroups));
|
||||
alertService.error('Failed to update column order', 'Please try again');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
setActiveItem(null);
|
||||
originalSourceGroupIdRef.current = null; // Reset the ref
|
||||
};
|
||||
|
||||
const handleDragCancel = () => {
|
||||
isDraggingRef.current = false;
|
||||
if (clonedItems) {
|
||||
dispatch(reorderTaskGroups(clonedItems));
|
||||
}
|
||||
setActiveItem(null);
|
||||
setClonedItems(null);
|
||||
originalSourceGroupIdRef.current = null;
|
||||
};
|
||||
|
||||
// Reset the recently moved flag after animation frame
|
||||
useEffect(() => {
|
||||
requestAnimationFrame(() => {
|
||||
recentlyMovedToNewContainer.current = false;
|
||||
});
|
||||
}, [taskGroups]);
|
||||
|
||||
useEffect(() => {
|
||||
if (socket) {
|
||||
socket.on(SocketEvents.GET_TASK_PROGRESS.toString(), handleTaskProgress);
|
||||
}
|
||||
|
||||
return () => {
|
||||
socket?.off(SocketEvents.GET_TASK_PROGRESS.toString(), handleTaskProgress);
|
||||
};
|
||||
}, [socket]);
|
||||
|
||||
// Track analytics event on component mount
|
||||
useEffect(() => {
|
||||
trackMixpanelEvent(evt_project_board_visit);
|
||||
}, []);
|
||||
|
||||
// Cleanup debounced function on unmount
|
||||
useEffect(() => {
|
||||
return () => {
|
||||
debouncedMoveTask.cancel();
|
||||
};
|
||||
}, [debouncedMoveTask]);
|
||||
|
||||
return (
|
||||
<Flex vertical gap={16}>
|
||||
<TaskListFilters position={'board'} />
|
||||
<Skeleton active loading={isLoading} className='mt-4 p-4'>
|
||||
<DndContext
|
||||
sensors={sensors}
|
||||
collisionDetection={collisionDetectionStrategy}
|
||||
onDragStart={handleDragStart}
|
||||
onDragOver={handleDragOver}
|
||||
onDragEnd={handleDragEnd}
|
||||
onDragCancel={handleDragCancel}
|
||||
>
|
||||
<BoardSectionCardContainer
|
||||
datasource={taskGroups}
|
||||
group={groupBy as 'status' | 'priority' | 'phases'}
|
||||
/>
|
||||
<DragOverlay>
|
||||
{activeItem?.type === 'task' && (
|
||||
<BoardViewTaskCard task={activeItem.task} sectionId={activeItem.sectionId} />
|
||||
)}
|
||||
</DragOverlay>
|
||||
</DndContext>
|
||||
</Skeleton>
|
||||
</Flex>
|
||||
);
|
||||
};
|
||||
|
||||
export default ProjectViewBoard;
|
||||
@@ -11,6 +11,7 @@ import {
|
||||
TableProps,
|
||||
Tooltip,
|
||||
Typography,
|
||||
Input
|
||||
} from '@/shared/antd-imports';
|
||||
|
||||
// Icons
|
||||
@@ -70,26 +71,29 @@ const ProjectViewMembers = () => {
|
||||
field: 'name',
|
||||
order: 'ascend',
|
||||
total: 0,
|
||||
pageSizeOptions: ['5', '10', '15', '20', '50', '100'],
|
||||
pageSizeOptions: ['10', '20', '50', '100'],
|
||||
size: 'small',
|
||||
});
|
||||
const [searchQuery, setSearchQuery] = useState(''); // <-- Add search state
|
||||
|
||||
// API Functions
|
||||
const getProjectMembers = async () => {
|
||||
const getProjectMembers = async (search: string = searchQuery) => {
|
||||
if (!projectId) return;
|
||||
|
||||
setIsLoading(true);
|
||||
try {
|
||||
const offset = (pagination.current - 1) * pagination.pageSize;
|
||||
const res = await projectsApiService.getMembers(
|
||||
projectId,
|
||||
pagination.current,
|
||||
pagination.pageSize,
|
||||
pagination.current, // index
|
||||
pagination.pageSize, // size // offset
|
||||
pagination.field,
|
||||
pagination.order,
|
||||
null
|
||||
search
|
||||
);
|
||||
if (res.done) {
|
||||
setMembers(res.body);
|
||||
setPagination(p => ({ ...p, total: res.body.total ?? 0 })); // update total from backend, default to 0
|
||||
}
|
||||
} catch (error) {
|
||||
logger.error('Error fetching members:', error);
|
||||
@@ -123,16 +127,14 @@ const ProjectViewMembers = () => {
|
||||
return Math.floor((completed / total) * 100);
|
||||
};
|
||||
|
||||
const handleTableChange = (pagination: any, filters: any, sorter: any) => {
|
||||
setPagination({
|
||||
current: pagination.current,
|
||||
pageSize: pagination.pageSize,
|
||||
field: sorter.field || pagination.field,
|
||||
order: sorter.order || pagination.order,
|
||||
total: pagination.total,
|
||||
pageSizeOptions: pagination.pageSizeOptions,
|
||||
size: pagination.size,
|
||||
});
|
||||
const handleTableChange = (tablePagination: any, filters: any, sorter: any) => {
|
||||
setPagination(prev => ({
|
||||
...prev,
|
||||
current: tablePagination.current,
|
||||
pageSize: tablePagination.pageSize,
|
||||
field: sorter.field || prev.field,
|
||||
order: sorter.order || prev.order,
|
||||
}));
|
||||
};
|
||||
|
||||
// Effects
|
||||
@@ -145,6 +147,7 @@ const ProjectViewMembers = () => {
|
||||
pagination.pageSize,
|
||||
pagination.field,
|
||||
pagination.order,
|
||||
// searchQuery, // <-- Do NOT include here, search is triggered manually
|
||||
]);
|
||||
|
||||
useEffect(() => {
|
||||
@@ -269,18 +272,33 @@ const ProjectViewMembers = () => {
|
||||
<Card
|
||||
style={{ width: '100%' }}
|
||||
title={
|
||||
<Flex justify="space-between">
|
||||
<Flex justify="space-between" align="center">
|
||||
<Typography.Text style={{ fontSize: 16, fontWeight: 500 }}>
|
||||
{members?.total} {members?.total !== 1 ? t('membersCountPlural') : t('memberCount')}
|
||||
</Typography.Text>
|
||||
|
||||
<Tooltip title={t('refreshButtonTooltip')}>
|
||||
<Button
|
||||
shape="circle"
|
||||
icon={<SyncOutlined />}
|
||||
onClick={() => void getProjectMembers()}
|
||||
<Flex gap={8} align="center">
|
||||
<Input.Search
|
||||
allowClear
|
||||
placeholder={t('searchPlaceholder')}
|
||||
value={searchQuery}
|
||||
onChange={e => setSearchQuery(e.target.value)}
|
||||
onSearch={value => {
|
||||
setPagination(p => ({ ...p, current: 1 })); // Reset to first page
|
||||
void getProjectMembers(value);
|
||||
}}
|
||||
style={{ width: 220 }}
|
||||
enterButton
|
||||
size="middle"
|
||||
/>
|
||||
</Tooltip>
|
||||
<Tooltip title={t('refreshButtonTooltip')}>
|
||||
<Button
|
||||
shape="circle"
|
||||
icon={<SyncOutlined />}
|
||||
onClick={() => void getProjectMembers()}
|
||||
/>
|
||||
</Tooltip>
|
||||
</Flex>
|
||||
</Flex>
|
||||
}
|
||||
>
|
||||
@@ -299,8 +317,12 @@ const ProjectViewMembers = () => {
|
||||
columns={columns}
|
||||
rowKey={record => record.id}
|
||||
pagination={{
|
||||
current: pagination.current,
|
||||
pageSize: pagination.pageSize,
|
||||
total: pagination.total,
|
||||
showSizeChanger: true,
|
||||
defaultPageSize: 20,
|
||||
pageSizeOptions: pagination.pageSizeOptions,
|
||||
size: pagination.size,
|
||||
}}
|
||||
onChange={handleTableChange}
|
||||
onRow={record => ({
|
||||
|
||||
@@ -28,9 +28,8 @@ const ProjectsReports = () => {
|
||||
|
||||
// Memoize the Excel export handler to prevent recreation on every render
|
||||
const handleExcelExport = useCallback(() => {
|
||||
if (currentSession?.team_name) {
|
||||
reportingExportApiService.exportProjects(currentSession.team_name);
|
||||
}
|
||||
const teamName = currentSession?.team_name || 'Team';
|
||||
reportingExportApiService.exportProjects(teamName);
|
||||
}, [currentSession?.team_name]);
|
||||
|
||||
// Memoize the archived checkbox handler to prevent recreation on every render
|
||||
|
||||
@@ -0,0 +1,67 @@
|
||||
import { Card, Flex, Typography, Tooltip } from '@/shared/antd-imports';
|
||||
import TimeReportPageHeader from '@/pages/reporting/timeReports/page-header/time-report-page-header';
|
||||
import MembersTimeSheet, {
|
||||
MembersTimeSheetRef,
|
||||
} from '@/pages/reporting/time-reports/members-time-sheet/members-time-sheet';
|
||||
import TimeReportingRightHeader from './timeReportingRightHeader/TimeReportingRightHeader';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { useDocumentTitle } from '@/hooks/useDoumentTItle';
|
||||
import { useRef } from 'react';
|
||||
import { useUserTimezone } from '@/hooks/useUserTimezone';
|
||||
import { InfoCircleOutlined } from '@ant-design/icons';
|
||||
|
||||
const { Text } = Typography;
|
||||
|
||||
const MembersTimeReports = () => {
|
||||
const { t } = useTranslation('time-report');
|
||||
const chartRef = useRef<MembersTimeSheetRef>(null);
|
||||
const { timezone, timezoneOffset } = useUserTimezone();
|
||||
|
||||
useDocumentTitle('Reporting - Allocation');
|
||||
|
||||
const handleExport = (type: string) => {
|
||||
if (type === 'png') {
|
||||
chartRef.current?.exportChart();
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<Flex vertical>
|
||||
<TimeReportingRightHeader
|
||||
title={t('Members Time Sheet')}
|
||||
exportType={[{ key: 'png', label: 'PNG' }]}
|
||||
export={handleExport}
|
||||
/>
|
||||
|
||||
<Card
|
||||
style={{ borderRadius: '4px' }}
|
||||
title={
|
||||
<div style={{ padding: '16px 0' }}>
|
||||
<Flex justify="space-between" align="center">
|
||||
<TimeReportPageHeader />
|
||||
<Tooltip
|
||||
title={`All time logs are displayed in your local timezone. Times were logged by users in their respective timezones and converted for your viewing.`}
|
||||
>
|
||||
<Text type="secondary" style={{ fontSize: 12 }}>
|
||||
<InfoCircleOutlined style={{ marginRight: 4 }} />
|
||||
Timezone: {timezone} ({timezoneOffset})
|
||||
</Text>
|
||||
</Tooltip>
|
||||
</Flex>
|
||||
</div>
|
||||
}
|
||||
styles={{
|
||||
body: {
|
||||
maxHeight: 'calc(100vh - 300px)',
|
||||
overflowY: 'auto',
|
||||
padding: '16px',
|
||||
},
|
||||
}}
|
||||
>
|
||||
<MembersTimeSheet ref={chartRef} />
|
||||
</Card>
|
||||
</Flex>
|
||||
);
|
||||
};
|
||||
|
||||
export default MembersTimeReports;
|
||||
@@ -65,6 +65,7 @@ import {
|
||||
Timeline,
|
||||
Mentions,
|
||||
Radio,
|
||||
Steps
|
||||
} from 'antd/es';
|
||||
|
||||
// Icons - Import commonly used ones
|
||||
@@ -188,6 +189,7 @@ export {
|
||||
DollarOutlined,
|
||||
DollarCircleOutlined,
|
||||
DatabaseOutlined,
|
||||
CopyOutlined
|
||||
} from '@ant-design/icons';
|
||||
|
||||
// Re-export all components with React
|
||||
@@ -246,6 +248,7 @@ export {
|
||||
Timeline,
|
||||
Mentions,
|
||||
Radio,
|
||||
Steps
|
||||
};
|
||||
|
||||
// TypeScript Types - Import commonly used ones
|
||||
|
||||
@@ -179,43 +179,4 @@ body.dark .overview-stat-card .ant-card-body,
|
||||
|
||||
.ql-snow .ql-stroke {
|
||||
stroke: #6c757d !important;
|
||||
}
|
||||
|
||||
/* HubSpot Widget Dark Mode Fixes */
|
||||
html.dark #hubspot-messages-iframe-container {
|
||||
background-color: transparent !important;
|
||||
}
|
||||
|
||||
html.dark #hubspot-messages-iframe-container iframe {
|
||||
background-color: transparent !important;
|
||||
}
|
||||
|
||||
/* Target the HubSpot widget container */
|
||||
html.dark .hs-messages-widget-open {
|
||||
background-color: transparent !important;
|
||||
}
|
||||
|
||||
/* Remove white background from HubSpot elements */
|
||||
html.dark #hs-messages-iframe-container,
|
||||
html.dark .hs-messages-iframe-container {
|
||||
background: transparent !important;
|
||||
background-color: transparent !important;
|
||||
}
|
||||
|
||||
/* Ensure the iframe wrapper has no background */
|
||||
html.dark iframe#hs-messages-iframe {
|
||||
background: transparent !important;
|
||||
}
|
||||
|
||||
/* Target the widget launcher button area */
|
||||
html.dark .hs-messages-widget,
|
||||
html.dark .hs-messages-widget-container {
|
||||
background: transparent !important;
|
||||
background-color: transparent !important;
|
||||
}
|
||||
|
||||
/* Remove any box shadow that might appear white */
|
||||
html.dark #hubspot-messages-iframe-container,
|
||||
html.dark .hs-messages-widget {
|
||||
box-shadow: none !important;
|
||||
}
|
||||
}
|
||||
51
worklenz-frontend/src/types/account-setup/survey.types.ts
Normal file
51
worklenz-frontend/src/types/account-setup/survey.types.ts
Normal file
@@ -0,0 +1,51 @@
|
||||
export interface ISurveyQuestion {
|
||||
id: string;
|
||||
survey_id: string;
|
||||
question_key: string;
|
||||
question_type: 'single_choice' | 'multiple_choice' | 'text';
|
||||
is_required: boolean;
|
||||
sort_order: number;
|
||||
options?: string[];
|
||||
}
|
||||
|
||||
export interface ISurvey {
|
||||
id: string;
|
||||
name: string;
|
||||
description?: string;
|
||||
survey_type: 'account_setup' | 'onboarding' | 'feedback';
|
||||
is_active: boolean;
|
||||
questions?: ISurveyQuestion[];
|
||||
}
|
||||
|
||||
export interface ISurveyAnswer {
|
||||
question_id: string;
|
||||
answer_text?: string;
|
||||
answer_json?: string[];
|
||||
}
|
||||
|
||||
export interface ISurveyResponse {
|
||||
id?: string;
|
||||
survey_id: string;
|
||||
user_id?: string;
|
||||
is_completed: boolean;
|
||||
answers: ISurveyAnswer[];
|
||||
}
|
||||
|
||||
export interface ISurveySubmissionRequest {
|
||||
survey_id: string;
|
||||
answers: ISurveyAnswer[];
|
||||
}
|
||||
|
||||
// Account setup survey specific types
|
||||
export type OrganizationType = 'freelancer' | 'startup' | 'small_medium_business' | 'agency' | 'enterprise' | 'other';
|
||||
export type UserRole = 'founder_ceo' | 'project_manager' | 'software_developer' | 'designer' | 'operations' | 'other';
|
||||
export type UseCase = 'task_management' | 'team_collaboration' | 'resource_planning' | 'client_communication' | 'time_tracking' | 'other';
|
||||
export type HowHeardAbout = 'google_search' | 'twitter' | 'linkedin' | 'friend_colleague' | 'blog_article' | 'other';
|
||||
|
||||
export interface IAccountSetupSurveyData {
|
||||
organization_type?: OrganizationType;
|
||||
user_role?: UserRole;
|
||||
main_use_cases?: UseCase[];
|
||||
previous_tools?: string;
|
||||
how_heard_about?: HowHeardAbout;
|
||||
}
|
||||
307
worklenz-frontend/src/types/advanced-gantt.types.ts
Normal file
307
worklenz-frontend/src/types/advanced-gantt.types.ts
Normal file
@@ -0,0 +1,307 @@
|
||||
import { ReactNode } from 'react';
|
||||
|
||||
// Core Task Interface
|
||||
export interface GanttTask {
|
||||
id: string;
|
||||
name: string;
|
||||
startDate: Date;
|
||||
endDate: Date;
|
||||
progress: number;
|
||||
duration?: number; // in days
|
||||
parent?: string;
|
||||
type: 'task' | 'milestone' | 'project';
|
||||
status: 'not-started' | 'in-progress' | 'completed' | 'on-hold' | 'overdue';
|
||||
priority: 'low' | 'medium' | 'high' | 'critical';
|
||||
assignee?: {
|
||||
id: string;
|
||||
name: string;
|
||||
avatar?: string;
|
||||
};
|
||||
dependencies?: string[];
|
||||
description?: string;
|
||||
tags?: string[];
|
||||
color?: string;
|
||||
isCollapsed?: boolean;
|
||||
level?: number; // for hierarchical display
|
||||
hasChildren?: boolean;
|
||||
isExpanded?: boolean;
|
||||
}
|
||||
|
||||
// Column Configuration
|
||||
export interface ColumnConfig {
|
||||
field: keyof GanttTask | string;
|
||||
title: string;
|
||||
width: number;
|
||||
minWidth?: number;
|
||||
maxWidth?: number;
|
||||
resizable: boolean;
|
||||
sortable: boolean;
|
||||
fixed: boolean;
|
||||
align?: 'left' | 'center' | 'right';
|
||||
renderer?: (value: any, task: GanttTask) => ReactNode;
|
||||
editor?: 'text' | 'date' | 'select' | 'number' | 'progress';
|
||||
editorOptions?: any;
|
||||
}
|
||||
|
||||
// Timeline Configuration
|
||||
export interface TimelineConfig {
|
||||
topTier: {
|
||||
unit: 'year' | 'month' | 'week' | 'day';
|
||||
format: string;
|
||||
height?: number;
|
||||
};
|
||||
bottomTier: {
|
||||
unit: 'month' | 'week' | 'day' | 'hour';
|
||||
format: string;
|
||||
height?: number;
|
||||
};
|
||||
showWeekends: boolean;
|
||||
showNonWorkingDays: boolean;
|
||||
holidays: Holiday[];
|
||||
workingDays: number[]; // 0-6, Sunday-Saturday
|
||||
workingHours: {
|
||||
start: number; // 0-23
|
||||
end: number; // 0-23
|
||||
};
|
||||
minDate?: Date;
|
||||
maxDate?: Date;
|
||||
dayWidth: number; // pixels per day
|
||||
}
|
||||
|
||||
// Holiday Interface
|
||||
export interface Holiday {
|
||||
date: Date;
|
||||
name: string;
|
||||
type: 'national' | 'company' | 'religious' | 'custom';
|
||||
recurring?: boolean;
|
||||
color?: string;
|
||||
}
|
||||
|
||||
// Virtual Scrolling Configuration
|
||||
export interface VirtualScrollConfig {
|
||||
enableRowVirtualization: boolean;
|
||||
enableTimelineVirtualization: boolean;
|
||||
bufferSize: number;
|
||||
itemHeight: number;
|
||||
overscan?: number;
|
||||
}
|
||||
|
||||
// Drag and Drop State
|
||||
export interface DragState {
|
||||
isDragging: boolean;
|
||||
dragType: 'move' | 'resize-start' | 'resize-end' | 'progress' | 'link';
|
||||
taskId: string;
|
||||
initialPosition: { x: number; y: number };
|
||||
currentPosition?: { x: number; y: number };
|
||||
initialDates: { start: Date; end: Date };
|
||||
initialProgress?: number;
|
||||
snapToGrid?: boolean;
|
||||
constraints?: {
|
||||
minDate?: Date;
|
||||
maxDate?: Date;
|
||||
minDuration?: number;
|
||||
maxDuration?: number;
|
||||
};
|
||||
}
|
||||
|
||||
// Zoom Levels
|
||||
export interface ZoomLevel {
|
||||
name: string;
|
||||
dayWidth: number;
|
||||
topTier: TimelineConfig['topTier'];
|
||||
bottomTier: TimelineConfig['bottomTier'];
|
||||
scale: number;
|
||||
}
|
||||
|
||||
// Selection State
|
||||
export interface SelectionState {
|
||||
selectedTasks: string[];
|
||||
selectedRows: number[];
|
||||
selectionRange?: {
|
||||
start: { row: number; col: number };
|
||||
end: { row: number; col: number };
|
||||
};
|
||||
focusedTask?: string;
|
||||
}
|
||||
|
||||
// Gantt View State
|
||||
export interface GanttViewState {
|
||||
zoomLevel: number;
|
||||
scrollPosition: { x: number; y: number };
|
||||
viewportSize: { width: number; height: number };
|
||||
splitterPosition: number; // percentage for grid/timeline split
|
||||
showCriticalPath: boolean;
|
||||
showBaseline: boolean;
|
||||
showProgress: boolean;
|
||||
showDependencies: boolean;
|
||||
autoSchedule: boolean;
|
||||
readOnly: boolean;
|
||||
}
|
||||
|
||||
// Performance Metrics
|
||||
export interface PerformanceMetrics {
|
||||
renderTime: number;
|
||||
taskCount: number;
|
||||
visibleTaskCount: number;
|
||||
memoryUsage?: number;
|
||||
fps?: number;
|
||||
}
|
||||
|
||||
// Event Handlers
|
||||
export type TaskEventHandler<T = void> = (task: GanttTask, event: MouseEvent | TouchEvent) => T;
|
||||
export type DragEventHandler = (taskId: string, newDates: { start: Date; end: Date }) => void;
|
||||
export type ResizeEventHandler = (taskId: string, newDates: { start: Date; end: Date }) => void;
|
||||
export type ProgressEventHandler = (taskId: string, progress: number) => void;
|
||||
export type SelectionEventHandler = (selectedTasks: string[]) => void;
|
||||
export type ColumnResizeHandler = (columnField: string, newWidth: number) => void;
|
||||
|
||||
// Gantt Actions (for useReducer)
|
||||
export type GanttAction =
|
||||
| { type: 'SET_TASKS'; payload: GanttTask[] }
|
||||
| { type: 'UPDATE_TASK'; payload: { id: string; updates: Partial<GanttTask> } }
|
||||
| { type: 'ADD_TASK'; payload: GanttTask }
|
||||
| { type: 'DELETE_TASK'; payload: string }
|
||||
| { type: 'SET_SELECTION'; payload: string[] }
|
||||
| { type: 'SET_DRAG_STATE'; payload: DragState | null }
|
||||
| { type: 'SET_ZOOM_LEVEL'; payload: number }
|
||||
| { type: 'SET_SCROLL_POSITION'; payload: { x: number; y: number } }
|
||||
| { type: 'SET_SPLITTER_POSITION'; payload: number }
|
||||
| { type: 'TOGGLE_TASK_EXPANSION'; payload: string }
|
||||
| { type: 'SET_VIEW_STATE'; payload: Partial<GanttViewState> }
|
||||
| { type: 'UPDATE_COLUMN_WIDTH'; payload: { field: string; width: number } };
|
||||
|
||||
// Main Gantt State
|
||||
export interface GanttState {
|
||||
tasks: GanttTask[];
|
||||
columns: ColumnConfig[];
|
||||
timelineConfig: TimelineConfig;
|
||||
virtualScrollConfig: VirtualScrollConfig;
|
||||
dragState: DragState | null;
|
||||
selectionState: SelectionState;
|
||||
viewState: GanttViewState;
|
||||
zoomLevels: ZoomLevel[];
|
||||
performanceMetrics: PerformanceMetrics;
|
||||
}
|
||||
|
||||
// Gantt Chart Props
|
||||
export interface AdvancedGanttProps {
|
||||
// Data
|
||||
tasks: GanttTask[];
|
||||
columns?: ColumnConfig[];
|
||||
|
||||
// Configuration
|
||||
timelineConfig?: Partial<TimelineConfig>;
|
||||
virtualScrollConfig?: Partial<VirtualScrollConfig>;
|
||||
zoomLevels?: ZoomLevel[];
|
||||
|
||||
// Initial State
|
||||
initialViewState?: Partial<GanttViewState>;
|
||||
initialSelection?: string[];
|
||||
|
||||
// Event Handlers
|
||||
onTaskUpdate?: (taskId: string, updates: Partial<GanttTask>) => void;
|
||||
onTaskCreate?: (task: Omit<GanttTask, 'id'>) => void;
|
||||
onTaskDelete?: (taskId: string) => void;
|
||||
onTaskMove?: DragEventHandler;
|
||||
onTaskResize?: ResizeEventHandler;
|
||||
onProgressChange?: ProgressEventHandler;
|
||||
onSelectionChange?: SelectionEventHandler;
|
||||
onColumnResize?: ColumnResizeHandler;
|
||||
onDependencyCreate?: (fromTaskId: string, toTaskId: string) => void;
|
||||
onDependencyDelete?: (fromTaskId: string, toTaskId: string) => void;
|
||||
|
||||
// UI Customization
|
||||
className?: string;
|
||||
style?: React.CSSProperties;
|
||||
theme?: 'light' | 'dark' | 'auto';
|
||||
locale?: string;
|
||||
|
||||
// Feature Flags
|
||||
enableDragDrop?: boolean;
|
||||
enableResize?: boolean;
|
||||
enableProgressEdit?: boolean;
|
||||
enableInlineEdit?: boolean;
|
||||
enableContextMenu?: boolean;
|
||||
enableTooltips?: boolean;
|
||||
enableExport?: boolean;
|
||||
enablePrint?: boolean;
|
||||
|
||||
// Performance Options
|
||||
enableVirtualScrolling?: boolean;
|
||||
enableDebouncing?: boolean;
|
||||
debounceDelay?: number;
|
||||
maxVisibleTasks?: number;
|
||||
}
|
||||
|
||||
// Context Menu Options
|
||||
export interface ContextMenuOption {
|
||||
id: string;
|
||||
label: string;
|
||||
icon?: ReactNode;
|
||||
disabled?: boolean;
|
||||
separator?: boolean;
|
||||
children?: ContextMenuOption[];
|
||||
onClick?: (task?: GanttTask) => void;
|
||||
}
|
||||
|
||||
// Export Options
|
||||
export interface ExportOptions {
|
||||
format: 'png' | 'pdf' | 'svg' | 'json' | 'csv' | 'xlsx';
|
||||
includeColumns?: string[];
|
||||
dateRange?: { start: Date; end: Date };
|
||||
filename?: string;
|
||||
paperSize?: 'A4' | 'A3' | 'Letter' | 'Legal';
|
||||
orientation?: 'portrait' | 'landscape';
|
||||
scale?: number;
|
||||
}
|
||||
|
||||
// Filter and Search
|
||||
export interface FilterConfig {
|
||||
field: string;
|
||||
operator: 'equals' | 'contains' | 'startsWith' | 'endsWith' | 'greaterThan' | 'lessThan' | 'between';
|
||||
value: any;
|
||||
logic?: 'and' | 'or';
|
||||
}
|
||||
|
||||
export interface SearchConfig {
|
||||
query: string;
|
||||
fields: string[];
|
||||
caseSensitive?: boolean;
|
||||
wholeWord?: boolean;
|
||||
regex?: boolean;
|
||||
}
|
||||
|
||||
// Baseline and Critical Path
|
||||
export interface TaskBaseline {
|
||||
taskId: string;
|
||||
baselineStart: Date;
|
||||
baselineEnd: Date;
|
||||
baselineDuration: number;
|
||||
baselineProgress: number;
|
||||
variance?: number; // days
|
||||
}
|
||||
|
||||
export interface CriticalPath {
|
||||
taskIds: string[];
|
||||
totalDuration: number;
|
||||
slack: number; // days of buffer
|
||||
}
|
||||
|
||||
// Undo/Redo
|
||||
export interface HistoryState {
|
||||
past: GanttState[];
|
||||
present: GanttState;
|
||||
future: GanttState[];
|
||||
maxHistorySize: number;
|
||||
}
|
||||
|
||||
// Keyboard Shortcuts
|
||||
export interface KeyboardShortcut {
|
||||
key: string;
|
||||
ctrl?: boolean;
|
||||
shift?: boolean;
|
||||
alt?: boolean;
|
||||
action: string;
|
||||
description: string;
|
||||
handler: (event: KeyboardEvent) => void;
|
||||
}
|
||||
32
worklenz-frontend/src/types/home/user-activity.types.ts
Normal file
32
worklenz-frontend/src/types/home/user-activity.types.ts
Normal file
@@ -0,0 +1,32 @@
|
||||
export interface IUserRecentTask {
|
||||
task_id: string;
|
||||
task_name: string;
|
||||
project_id: string;
|
||||
project_name: string;
|
||||
last_activity_at: string;
|
||||
activity_count: number;
|
||||
project_color?: string;
|
||||
task_status?: string;
|
||||
status_color?: string;
|
||||
}
|
||||
|
||||
export interface IUserTimeLoggedTask {
|
||||
task_id: string;
|
||||
task_name: string;
|
||||
project_id: string;
|
||||
project_name: string;
|
||||
total_time_logged: number;
|
||||
total_time_logged_string: string;
|
||||
last_logged_at: string;
|
||||
logged_by_timer: boolean;
|
||||
project_color?: string;
|
||||
task_status?: string;
|
||||
status_color?: string;
|
||||
log_entries_count?: number;
|
||||
estimated_time?: number;
|
||||
}
|
||||
|
||||
export enum ActivityFeedType {
|
||||
RECENT_TASKS = 'recent-tasks',
|
||||
TIME_LOGGED_TASKS = 'time-logged-tasks'
|
||||
}
|
||||
62
worklenz-frontend/src/types/project-roadmap.types.ts
Normal file
62
worklenz-frontend/src/types/project-roadmap.types.ts
Normal file
@@ -0,0 +1,62 @@
|
||||
export interface ProjectPhase {
|
||||
id: string;
|
||||
name: string;
|
||||
description: string;
|
||||
startDate: Date;
|
||||
endDate: Date;
|
||||
progress: number;
|
||||
color: string;
|
||||
status: 'not-started' | 'in-progress' | 'completed' | 'on-hold';
|
||||
tasks: PhaseTask[];
|
||||
milestones: PhaseMilestone[];
|
||||
}
|
||||
|
||||
export interface PhaseTask {
|
||||
id: string;
|
||||
name: string;
|
||||
description?: string;
|
||||
startDate: Date;
|
||||
endDate: Date;
|
||||
progress: number;
|
||||
assigneeId?: string;
|
||||
assigneeName?: string;
|
||||
priority: 'low' | 'medium' | 'high';
|
||||
status: 'todo' | 'in-progress' | 'done';
|
||||
dependencies?: string[];
|
||||
}
|
||||
|
||||
export interface PhaseMilestone {
|
||||
id: string;
|
||||
name: string;
|
||||
description?: string;
|
||||
dueDate: Date;
|
||||
isCompleted: boolean;
|
||||
criticalPath: boolean;
|
||||
}
|
||||
|
||||
export interface ProjectRoadmap {
|
||||
id: string;
|
||||
projectId: string;
|
||||
name: string;
|
||||
description?: string;
|
||||
startDate: Date;
|
||||
endDate: Date;
|
||||
phases: ProjectPhase[];
|
||||
createdAt: Date;
|
||||
updatedAt: Date;
|
||||
}
|
||||
|
||||
export interface GanttViewOptions {
|
||||
viewMode: 'day' | 'week' | 'month' | 'year';
|
||||
showTasks: boolean;
|
||||
showMilestones: boolean;
|
||||
groupByPhase: boolean;
|
||||
}
|
||||
|
||||
export interface PhaseModalData extends ProjectPhase {
|
||||
taskCount: number;
|
||||
completedTaskCount: number;
|
||||
milestoneCount: number;
|
||||
completedMilestoneCount: number;
|
||||
teamMembers: string[];
|
||||
}
|
||||
@@ -52,6 +52,13 @@ export interface IAccountSetupRequest {
|
||||
tasks: string[];
|
||||
team_members: string[];
|
||||
template_id?: string | null;
|
||||
survey_data?: {
|
||||
organization_type?: string;
|
||||
user_role?: string;
|
||||
main_use_cases?: string[];
|
||||
previous_tools?: string;
|
||||
how_heard_about?: string;
|
||||
};
|
||||
}
|
||||
|
||||
export interface IAccountSetupResponse {
|
||||
|
||||
@@ -115,6 +115,9 @@ export interface TaskManagementState {
|
||||
loadingColumns: boolean;
|
||||
columns: ITaskListColumn[];
|
||||
customColumns: ITaskListColumn[];
|
||||
// Add sort-related state
|
||||
sortField: string;
|
||||
sortOrder: 'ASC' | 'DESC';
|
||||
}
|
||||
|
||||
export interface TaskGroupsState {
|
||||
|
||||
@@ -1,32 +1,59 @@
|
||||
import dayjs from 'dayjs';
|
||||
import relativeTime from 'dayjs/plugin/relativeTime';
|
||||
import localizedFormat from 'dayjs/plugin/localizedFormat';
|
||||
import 'dayjs/locale/de';
|
||||
import 'dayjs/locale/es';
|
||||
import 'dayjs/locale/pt';
|
||||
import 'dayjs/locale/zh-cn';
|
||||
import { getLanguageFromLocalStorage } from './language-utils';
|
||||
|
||||
// Initialize the relativeTime plugin
|
||||
// Initialize plugins
|
||||
dayjs.extend(relativeTime);
|
||||
dayjs.extend(localizedFormat);
|
||||
|
||||
/**
|
||||
* Formats a date to a relative time string (e.g., "2 hours ago", "a day ago")
|
||||
* This mimics the Angular fromNow pipe functionality
|
||||
*
|
||||
* @param date - The date to format (string, Date, or dayjs object)
|
||||
* @returns A string representing the relative time
|
||||
*/
|
||||
export const fromNow = (date: string | Date | dayjs.Dayjs): string => {
|
||||
if (!date) return '';
|
||||
return dayjs(date).fromNow();
|
||||
// Map application languages to dayjs locales
|
||||
const getLocaleFromLanguage = (language: string): string => {
|
||||
const localeMap: Record<string, string> = {
|
||||
'en': 'en',
|
||||
'de': 'de',
|
||||
'es': 'es',
|
||||
'pt': 'pt',
|
||||
'alb': 'en', // Albanian not supported by dayjs, fallback to English
|
||||
'zh': 'zh-cn'
|
||||
};
|
||||
return localeMap[language] || 'en';
|
||||
};
|
||||
|
||||
/**
|
||||
* Formats a date to a specific format
|
||||
* Formats a date to a relative time string (e.g., "2 hours ago", "a day ago")
|
||||
* This mimics the Angular fromNow pipe functionality with locale support
|
||||
*
|
||||
* @param date - The date to format (string, Date, or dayjs object)
|
||||
* @param language - Optional language override (defaults to stored language)
|
||||
* @returns A string representing the relative time
|
||||
*/
|
||||
export const fromNow = (date: string | Date | dayjs.Dayjs, language?: string): string => {
|
||||
if (!date) return '';
|
||||
const currentLanguage = language || getLanguageFromLocalStorage();
|
||||
const locale = getLocaleFromLanguage(currentLanguage);
|
||||
return dayjs(date).locale(locale).fromNow();
|
||||
};
|
||||
|
||||
/**
|
||||
* Formats a date to a specific format with locale support
|
||||
*
|
||||
* @param date - The date to format (string, Date, or dayjs object)
|
||||
* @param format - The format string (default: 'YYYY-MM-DD')
|
||||
* @param language - Optional language override (defaults to stored language)
|
||||
* @returns A formatted date string
|
||||
*/
|
||||
export const formatDate = (
|
||||
date: string | Date | dayjs.Dayjs,
|
||||
format: string = 'YYYY-MM-DD'
|
||||
format: string = 'YYYY-MM-DD',
|
||||
language?: string
|
||||
): string => {
|
||||
if (!date) return '';
|
||||
return dayjs(date).format(format);
|
||||
const currentLanguage = language || getLanguageFromLocalStorage();
|
||||
const locale = getLocaleFromLanguage(currentLanguage);
|
||||
return dayjs(date).locale(locale).format(format);
|
||||
};
|
||||
|
||||
@@ -85,8 +85,8 @@ export class EnhancedPerformanceMonitor {
|
||||
this.setupObservers();
|
||||
this.collectInitialMetrics();
|
||||
this.startPeriodicCollection();
|
||||
|
||||
console.log('🚀 Enhanced performance monitoring started');
|
||||
|
||||
// console.log('🚀 Enhanced performance monitoring started');
|
||||
}
|
||||
|
||||
// Stop monitoring and cleanup
|
||||
@@ -96,8 +96,8 @@ export class EnhancedPerformanceMonitor {
|
||||
this.isMonitoring = false;
|
||||
this.cleanupObservers();
|
||||
this.clearIntervals();
|
||||
|
||||
console.log('🛑 Enhanced performance monitoring stopped');
|
||||
|
||||
// console.log('🛑 Enhanced performance monitoring stopped');
|
||||
}
|
||||
|
||||
// Setup performance observers
|
||||
@@ -358,8 +358,8 @@ export class EnhancedPerformanceMonitor {
|
||||
|
||||
const recent = this.metrics.slice(-10); // Last 10 metrics
|
||||
const report = this.analyzeMetrics(recent);
|
||||
|
||||
console.log('📊 Performance Report:', report);
|
||||
|
||||
// console.log('📊 Performance Report:', report);
|
||||
|
||||
// Check for performance issues
|
||||
this.checkPerformanceIssues(report);
|
||||
|
||||
@@ -0,0 +1,94 @@
|
||||
import { format } from 'date-fns';
|
||||
import { enUS, es, pt } from 'date-fns/locale';
|
||||
import { getLanguageFromLocalStorage } from './language-utils';
|
||||
|
||||
/**
|
||||
* Formats a date/time string using the user's profile timezone
|
||||
* This ensures consistency between time logs display and reporting filters
|
||||
*
|
||||
* @param dateString - The date string to format (typically in UTC)
|
||||
* @param userTimezone - The user's timezone from their profile (e.g., 'America/New_York')
|
||||
* @returns Formatted date string in user's timezone
|
||||
*/
|
||||
export const formatDateTimeWithUserTimezone = (
|
||||
dateString: string,
|
||||
userTimezone?: string | null
|
||||
): string => {
|
||||
if (!dateString) return '';
|
||||
|
||||
try {
|
||||
const date = new Date(dateString);
|
||||
|
||||
// If timezone is provided, use it for formatting
|
||||
if (userTimezone && userTimezone !== 'UTC') {
|
||||
// Use the browser's toLocaleString with timezone option
|
||||
const options: Intl.DateTimeFormatOptions = {
|
||||
timeZone: userTimezone,
|
||||
year: 'numeric',
|
||||
month: 'short',
|
||||
day: 'numeric',
|
||||
hour: 'numeric',
|
||||
minute: '2-digit',
|
||||
second: '2-digit',
|
||||
hour12: true
|
||||
};
|
||||
|
||||
// Get the appropriate locale
|
||||
const localeString = getLanguageFromLocalStorage();
|
||||
const localeMap = {
|
||||
'en': 'en-US',
|
||||
'es': 'es-ES',
|
||||
'pt': 'pt-PT'
|
||||
};
|
||||
const locale = localeMap[localeString as keyof typeof localeMap] || 'en-US';
|
||||
|
||||
return date.toLocaleString(locale, options);
|
||||
}
|
||||
|
||||
// Fallback to date-fns formatting for UTC or when no timezone
|
||||
const localeString = getLanguageFromLocalStorage();
|
||||
const locale = localeString === 'en' ? enUS : localeString === 'es' ? es : pt;
|
||||
return format(date, 'MMM d, yyyy, h:mm:ss a', { locale });
|
||||
} catch (error) {
|
||||
console.error('Error formatting date with user timezone:', error);
|
||||
// Fallback to original date string if formatting fails
|
||||
return dateString;
|
||||
}
|
||||
};
|
||||
|
||||
/**
|
||||
* Checks if a date is "yesterday" in the user's timezone
|
||||
* This is used to ensure consistency with reporting filters
|
||||
*
|
||||
* @param dateString - The date string to check
|
||||
* @param userTimezone - The user's timezone from their profile
|
||||
* @returns true if the date is yesterday in user's timezone
|
||||
*/
|
||||
export const isYesterdayInUserTimezone = (dateString: string, userTimezone?: string | null): boolean => {
|
||||
if (!dateString || !userTimezone) return false;
|
||||
|
||||
try {
|
||||
const date = new Date(dateString);
|
||||
|
||||
// Get current date in user's timezone
|
||||
const nowInTimezone = new Date().toLocaleString('en-US', { timeZone: userTimezone });
|
||||
const now = new Date(nowInTimezone);
|
||||
|
||||
// Get yesterday in user's timezone
|
||||
const yesterday = new Date(now);
|
||||
yesterday.setDate(yesterday.getDate() - 1);
|
||||
|
||||
// Convert the input date to user's timezone
|
||||
const dateInTimezone = new Date(date.toLocaleString('en-US', { timeZone: userTimezone }));
|
||||
|
||||
// Compare dates (ignoring time)
|
||||
return (
|
||||
dateInTimezone.getFullYear() === yesterday.getFullYear() &&
|
||||
dateInTimezone.getMonth() === yesterday.getMonth() &&
|
||||
dateInTimezone.getDate() === yesterday.getDate()
|
||||
);
|
||||
} catch (error) {
|
||||
console.error('Error checking if date is yesterday:', error);
|
||||
return false;
|
||||
}
|
||||
};
|
||||
408
worklenz-frontend/src/utils/gantt-performance.ts
Normal file
408
worklenz-frontend/src/utils/gantt-performance.ts
Normal file
@@ -0,0 +1,408 @@
|
||||
import { useMemo, useCallback, useRef, useEffect } from 'react';
|
||||
import { GanttTask, PerformanceMetrics } from '../types/advanced-gantt.types';
|
||||
|
||||
// Debounce utility for drag operations
|
||||
export function useDebounce<T extends (...args: any[]) => any>(
|
||||
callback: T,
|
||||
delay: number
|
||||
): T {
|
||||
const timeoutRef = useRef<NodeJS.Timeout>();
|
||||
|
||||
return useCallback((...args: Parameters<T>) => {
|
||||
if (timeoutRef.current) {
|
||||
clearTimeout(timeoutRef.current);
|
||||
}
|
||||
|
||||
timeoutRef.current = setTimeout(() => {
|
||||
callback(...args);
|
||||
}, delay);
|
||||
}, [callback, delay]) as T;
|
||||
}
|
||||
|
||||
// Throttle utility for scroll events
|
||||
export function useThrottle<T extends (...args: any[]) => any>(
|
||||
callback: T,
|
||||
delay: number
|
||||
): T {
|
||||
const lastCall = useRef<number>(0);
|
||||
|
||||
return useCallback((...args: Parameters<T>) => {
|
||||
const now = Date.now();
|
||||
if (now - lastCall.current >= delay) {
|
||||
lastCall.current = now;
|
||||
callback(...args);
|
||||
}
|
||||
}, [callback, delay]) as T;
|
||||
}
|
||||
|
||||
// Memoized task calculations
|
||||
export const useTaskCalculations = (tasks: GanttTask[]) => {
|
||||
return useMemo(() => {
|
||||
const taskMap = new Map<string, GanttTask>();
|
||||
const parentChildMap = new Map<string, string[]>();
|
||||
const dependencyMap = new Map<string, string[]>();
|
||||
|
||||
// Build maps for efficient lookups
|
||||
tasks.forEach(task => {
|
||||
taskMap.set(task.id, task);
|
||||
|
||||
if (task.parent) {
|
||||
if (!parentChildMap.has(task.parent)) {
|
||||
parentChildMap.set(task.parent, []);
|
||||
}
|
||||
parentChildMap.get(task.parent)!.push(task.id);
|
||||
}
|
||||
|
||||
if (task.dependencies) {
|
||||
dependencyMap.set(task.id, task.dependencies);
|
||||
}
|
||||
});
|
||||
|
||||
return {
|
||||
taskMap,
|
||||
parentChildMap,
|
||||
dependencyMap,
|
||||
totalTasks: tasks.length,
|
||||
projectTasks: tasks.filter(t => t.type === 'project'),
|
||||
milestones: tasks.filter(t => t.type === 'milestone'),
|
||||
regularTasks: tasks.filter(t => t.type === 'task'),
|
||||
};
|
||||
}, [tasks]);
|
||||
};
|
||||
|
||||
// Virtual scrolling calculations
|
||||
export interface VirtualScrollData {
|
||||
startIndex: number;
|
||||
endIndex: number;
|
||||
visibleItems: GanttTask[];
|
||||
totalHeight: number;
|
||||
offsetY: number;
|
||||
}
|
||||
|
||||
export const useVirtualScrolling = (
|
||||
tasks: GanttTask[],
|
||||
containerHeight: number,
|
||||
itemHeight: number,
|
||||
scrollTop: number,
|
||||
overscan: number = 5
|
||||
): VirtualScrollData => {
|
||||
return useMemo(() => {
|
||||
const totalHeight = tasks.length * itemHeight;
|
||||
const startIndex = Math.max(0, Math.floor(scrollTop / itemHeight) - overscan);
|
||||
const endIndex = Math.min(
|
||||
tasks.length - 1,
|
||||
Math.ceil((scrollTop + containerHeight) / itemHeight) + overscan
|
||||
);
|
||||
const visibleItems = tasks.slice(startIndex, endIndex + 1);
|
||||
const offsetY = startIndex * itemHeight;
|
||||
|
||||
return {
|
||||
startIndex,
|
||||
endIndex,
|
||||
visibleItems,
|
||||
totalHeight,
|
||||
offsetY,
|
||||
};
|
||||
}, [tasks, containerHeight, itemHeight, scrollTop, overscan]);
|
||||
};
|
||||
|
||||
// Timeline virtual scrolling
|
||||
export interface TimelineVirtualData {
|
||||
startDate: Date;
|
||||
endDate: Date;
|
||||
visibleDays: Date[];
|
||||
totalWidth: number;
|
||||
offsetX: number;
|
||||
}
|
||||
|
||||
export const useTimelineVirtualScrolling = (
|
||||
projectStartDate: Date,
|
||||
projectEndDate: Date,
|
||||
dayWidth: number,
|
||||
containerWidth: number,
|
||||
scrollLeft: number,
|
||||
overscan: number = 10
|
||||
): TimelineVirtualData => {
|
||||
return useMemo(() => {
|
||||
const totalDays = Math.ceil((projectEndDate.getTime() - projectStartDate.getTime()) / (1000 * 60 * 60 * 24));
|
||||
const totalWidth = totalDays * dayWidth;
|
||||
|
||||
const startDayIndex = Math.max(0, Math.floor(scrollLeft / dayWidth) - overscan);
|
||||
const endDayIndex = Math.min(
|
||||
totalDays - 1,
|
||||
Math.ceil((scrollLeft + containerWidth) / dayWidth) + overscan
|
||||
);
|
||||
|
||||
const visibleDays: Date[] = [];
|
||||
for (let i = startDayIndex; i <= endDayIndex; i++) {
|
||||
const date = new Date(projectStartDate);
|
||||
date.setDate(date.getDate() + i);
|
||||
visibleDays.push(date);
|
||||
}
|
||||
|
||||
const offsetX = startDayIndex * dayWidth;
|
||||
const startDate = new Date(projectStartDate);
|
||||
startDate.setDate(startDate.getDate() + startDayIndex);
|
||||
|
||||
const endDate = new Date(projectStartDate);
|
||||
endDate.setDate(endDate.getDate() + endDayIndex);
|
||||
|
||||
return {
|
||||
startDate,
|
||||
endDate,
|
||||
visibleDays,
|
||||
totalWidth,
|
||||
offsetX,
|
||||
};
|
||||
}, [projectStartDate, projectEndDate, dayWidth, containerWidth, scrollLeft, overscan]);
|
||||
};
|
||||
|
||||
// Performance monitoring hook
|
||||
export const usePerformanceMonitoring = (): {
|
||||
metrics: PerformanceMetrics;
|
||||
startMeasure: (name: string) => void;
|
||||
endMeasure: (name: string) => void;
|
||||
recordMetric: (name: string, value: number) => void;
|
||||
} => {
|
||||
const metricsRef = useRef<PerformanceMetrics>({
|
||||
renderTime: 0,
|
||||
taskCount: 0,
|
||||
visibleTaskCount: 0,
|
||||
});
|
||||
|
||||
const measurementsRef = useRef<Map<string, number>>(new Map());
|
||||
|
||||
const startMeasure = useCallback((name: string) => {
|
||||
measurementsRef.current.set(name, performance.now());
|
||||
}, []);
|
||||
|
||||
const endMeasure = useCallback((name: string) => {
|
||||
const startTime = measurementsRef.current.get(name);
|
||||
if (startTime) {
|
||||
const duration = performance.now() - startTime;
|
||||
measurementsRef.current.delete(name);
|
||||
|
||||
if (name === 'render') {
|
||||
metricsRef.current.renderTime = duration;
|
||||
}
|
||||
}
|
||||
}, []);
|
||||
|
||||
const recordMetric = useCallback((name: string, value: number) => {
|
||||
switch (name) {
|
||||
case 'taskCount':
|
||||
metricsRef.current.taskCount = value;
|
||||
break;
|
||||
case 'visibleTaskCount':
|
||||
metricsRef.current.visibleTaskCount = value;
|
||||
break;
|
||||
case 'memoryUsage':
|
||||
metricsRef.current.memoryUsage = value;
|
||||
break;
|
||||
case 'fps':
|
||||
metricsRef.current.fps = value;
|
||||
break;
|
||||
}
|
||||
}, []);
|
||||
|
||||
return {
|
||||
metrics: metricsRef.current,
|
||||
startMeasure,
|
||||
endMeasure,
|
||||
recordMetric,
|
||||
};
|
||||
};
|
||||
|
||||
// Intersection Observer for lazy loading
|
||||
export const useIntersectionObserver = (
|
||||
callback: (entries: IntersectionObserverEntry[]) => void,
|
||||
options?: IntersectionObserverInit
|
||||
) => {
|
||||
const targetRef = useRef<HTMLElement>(null);
|
||||
const observerRef = useRef<IntersectionObserver>();
|
||||
|
||||
useEffect(() => {
|
||||
if (!targetRef.current) return;
|
||||
|
||||
observerRef.current = new IntersectionObserver(callback, {
|
||||
root: null,
|
||||
rootMargin: '100px',
|
||||
threshold: 0.1,
|
||||
...options,
|
||||
});
|
||||
|
||||
observerRef.current.observe(targetRef.current);
|
||||
|
||||
return () => {
|
||||
if (observerRef.current) {
|
||||
observerRef.current.disconnect();
|
||||
}
|
||||
};
|
||||
}, [callback, options]);
|
||||
|
||||
return targetRef;
|
||||
};
|
||||
|
||||
// Date calculations optimization
|
||||
export const useDateCalculations = () => {
|
||||
return useMemo(() => {
|
||||
const cache = new Map<string, number>();
|
||||
|
||||
const getDaysBetween = (start: Date, end: Date): number => {
|
||||
const key = `${start.getTime()}-${end.getTime()}`;
|
||||
if (cache.has(key)) {
|
||||
return cache.get(key)!;
|
||||
}
|
||||
|
||||
const days = Math.ceil((end.getTime() - start.getTime()) / (1000 * 60 * 60 * 24));
|
||||
cache.set(key, days);
|
||||
return days;
|
||||
};
|
||||
|
||||
const addDays = (date: Date, days: number): Date => {
|
||||
const result = new Date(date);
|
||||
result.setDate(result.getDate() + days);
|
||||
return result;
|
||||
};
|
||||
|
||||
const isWeekend = (date: Date): boolean => {
|
||||
const day = date.getDay();
|
||||
return day === 0 || day === 6; // Sunday or Saturday
|
||||
};
|
||||
|
||||
const isWorkingDay = (date: Date, workingDays: number[]): boolean => {
|
||||
return workingDays.includes(date.getDay());
|
||||
};
|
||||
|
||||
const getWorkingDaysBetween = (start: Date, end: Date, workingDays: number[]): number => {
|
||||
let count = 0;
|
||||
const current = new Date(start);
|
||||
|
||||
while (current <= end) {
|
||||
if (isWorkingDay(current, workingDays)) {
|
||||
count++;
|
||||
}
|
||||
current.setDate(current.getDate() + 1);
|
||||
}
|
||||
|
||||
return count;
|
||||
};
|
||||
|
||||
return {
|
||||
getDaysBetween,
|
||||
addDays,
|
||||
isWeekend,
|
||||
isWorkingDay,
|
||||
getWorkingDaysBetween,
|
||||
clearCache: () => cache.clear(),
|
||||
};
|
||||
}, []);
|
||||
};
|
||||
|
||||
// Task position calculations
|
||||
export const useTaskPositions = (
|
||||
tasks: GanttTask[],
|
||||
timelineStart: Date,
|
||||
dayWidth: number
|
||||
) => {
|
||||
return useMemo(() => {
|
||||
const positions = new Map<string, { x: number; width: number; y: number }>();
|
||||
|
||||
tasks.forEach((task, index) => {
|
||||
const startDays = Math.floor((task.startDate.getTime() - timelineStart.getTime()) / (1000 * 60 * 60 * 24));
|
||||
const endDays = Math.floor((task.endDate.getTime() - timelineStart.getTime()) / (1000 * 60 * 60 * 24));
|
||||
|
||||
positions.set(task.id, {
|
||||
x: startDays * dayWidth,
|
||||
width: Math.max(1, (endDays - startDays) * dayWidth),
|
||||
y: index * 40, // Assuming 40px row height
|
||||
});
|
||||
});
|
||||
|
||||
return positions;
|
||||
}, [tasks, timelineStart, dayWidth]);
|
||||
};
|
||||
|
||||
// Memory management utilities
|
||||
export const useMemoryManagement = () => {
|
||||
const cleanupFunctions = useRef<Array<() => void>>([]);
|
||||
|
||||
const addCleanup = useCallback((cleanup: () => void) => {
|
||||
cleanupFunctions.current.push(cleanup);
|
||||
}, []);
|
||||
|
||||
const runCleanup = useCallback(() => {
|
||||
cleanupFunctions.current.forEach(cleanup => cleanup());
|
||||
cleanupFunctions.current = [];
|
||||
}, []);
|
||||
|
||||
useEffect(() => {
|
||||
return runCleanup;
|
||||
}, [runCleanup]);
|
||||
|
||||
return { addCleanup, runCleanup };
|
||||
};
|
||||
|
||||
// Batch update utility for multiple task changes
|
||||
export const useBatchUpdates = <T>(
|
||||
updateFunction: (updates: T[]) => void,
|
||||
delay: number = 100
|
||||
) => {
|
||||
const batchRef = useRef<T[]>([]);
|
||||
const timeoutRef = useRef<NodeJS.Timeout>();
|
||||
|
||||
const addUpdate = useCallback((update: T) => {
|
||||
batchRef.current.push(update);
|
||||
|
||||
if (timeoutRef.current) {
|
||||
clearTimeout(timeoutRef.current);
|
||||
}
|
||||
|
||||
timeoutRef.current = setTimeout(() => {
|
||||
if (batchRef.current.length > 0) {
|
||||
updateFunction([...batchRef.current]);
|
||||
batchRef.current = [];
|
||||
}
|
||||
}, delay);
|
||||
}, [updateFunction, delay]);
|
||||
|
||||
const flushUpdates = useCallback(() => {
|
||||
if (timeoutRef.current) {
|
||||
clearTimeout(timeoutRef.current);
|
||||
}
|
||||
|
||||
if (batchRef.current.length > 0) {
|
||||
updateFunction([...batchRef.current]);
|
||||
batchRef.current = [];
|
||||
}
|
||||
}, [updateFunction]);
|
||||
|
||||
return { addUpdate, flushUpdates };
|
||||
};
|
||||
|
||||
// FPS monitoring
|
||||
export const useFPSMonitoring = () => {
|
||||
const fpsRef = useRef<number>(0);
|
||||
const frameCountRef = useRef<number>(0);
|
||||
const lastTimeRef = useRef<number>(performance.now());
|
||||
|
||||
const measureFPS = useCallback(() => {
|
||||
frameCountRef.current++;
|
||||
const now = performance.now();
|
||||
|
||||
if (now - lastTimeRef.current >= 1000) {
|
||||
fpsRef.current = Math.round((frameCountRef.current * 1000) / (now - lastTimeRef.current));
|
||||
frameCountRef.current = 0;
|
||||
lastTimeRef.current = now;
|
||||
}
|
||||
|
||||
requestAnimationFrame(measureFPS);
|
||||
}, []);
|
||||
|
||||
useEffect(() => {
|
||||
const rafId = requestAnimationFrame(measureFPS);
|
||||
return () => cancelAnimationFrame(rafId);
|
||||
}, [measureFPS]);
|
||||
|
||||
return fpsRef.current;
|
||||
};
|
||||
Reference in New Issue
Block a user