- Introduced a new document summarizing performance optimizations across the application. - Applied React.memo(), useMemo(), and useCallback() to key components to minimize unnecessary re-renders and optimize rendering performance. - Implemented a route preloading system to enhance navigation speed and user experience. - Added performance monitoring utilities for development to track component render times and function execution. - Enhanced lazy loading and suspense boundaries for better loading states. - Conducted production optimizations, including TypeScript error fixes and memory management improvements. - Memoized style and configuration objects to reduce garbage collection pressure and improve memory usage.
127 lines
4.1 KiB
TypeScript
127 lines
4.1 KiB
TypeScript
import { useEffect, memo, useMemo, useCallback } from 'react';
|
|
import { useMediaQuery } from 'react-responsive';
|
|
import Col from 'antd/es/col';
|
|
import Flex from 'antd/es/flex';
|
|
|
|
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 { useDocumentTitle } from '@/hooks/useDoumentTItle';
|
|
import { useAppDispatch } from '@/hooks/useAppDispatch';
|
|
import { useAuthService } from '@/hooks/useAuth';
|
|
|
|
import { fetchProjectStatuses } from '@/features/projects/lookups/projectStatuses/projectStatusesSlice';
|
|
import { fetchProjectCategories } from '@/features/projects/lookups/projectCategories/projectCategoriesSlice';
|
|
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';
|
|
|
|
const DESKTOP_MIN_WIDTH = 1024;
|
|
const TASK_LIST_MIN_WIDTH = 500;
|
|
const SIDEBAR_MAX_WIDTH = 400;
|
|
|
|
// Lazy load heavy components
|
|
const TaskDrawer = React.lazy(() => import('@components/task-drawer/task-drawer'));
|
|
|
|
const HomePage = memo(() => {
|
|
const dispatch = useAppDispatch();
|
|
const isDesktop = useMediaQuery({ query: `(min-width: ${DESKTOP_MIN_WIDTH}px)` });
|
|
const isOwnerOrAdmin = useAuthService().isOwnerOrAdmin();
|
|
|
|
useDocumentTitle('Home');
|
|
|
|
// Memoize fetch function to prevent recreation on every render
|
|
const fetchLookups = useCallback(async () => {
|
|
const fetchPromises = [
|
|
dispatch(fetchProjectHealth()),
|
|
dispatch(fetchProjectCategories()),
|
|
dispatch(fetchProjectStatuses()),
|
|
dispatch(fetchProjects()),
|
|
].filter(Boolean);
|
|
|
|
await Promise.all(fetchPromises);
|
|
}, [dispatch]);
|
|
|
|
useEffect(() => {
|
|
fetchLookups();
|
|
}, [fetchLookups]);
|
|
|
|
// Memoize project drawer close handler
|
|
const handleProjectDrawerClose = useCallback(() => {}, []);
|
|
|
|
// Memoize desktop flex styles to prevent object recreation
|
|
const desktopFlexStyle = useMemo(() => ({
|
|
minWidth: TASK_LIST_MIN_WIDTH,
|
|
width: '100%'
|
|
}), []);
|
|
|
|
const sidebarFlexStyle = useMemo(() => ({
|
|
width: '100%',
|
|
maxWidth: SIDEBAR_MAX_WIDTH
|
|
}), []);
|
|
|
|
// Memoize components to prevent unnecessary re-renders
|
|
const CreateProjectButtonComponent = useMemo(() => {
|
|
if (!isOwnerOrAdmin) return null;
|
|
|
|
return isDesktop ? (
|
|
<div className="absolute right-0 top-1/2 -translate-y-1/2">
|
|
<CreateProjectButton />
|
|
</div>
|
|
) : (
|
|
<CreateProjectButton />
|
|
);
|
|
}, [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">
|
|
<GreetingWithTime />
|
|
{CreateProjectButtonComponent}
|
|
</Col>
|
|
|
|
{MainContent}
|
|
|
|
{/* Use Suspense for lazy-loaded components */}
|
|
<Suspense fallback={null}>
|
|
{createPortal(<TaskDrawer />, document.body, 'home-task-drawer')}
|
|
</Suspense>
|
|
|
|
{createPortal(
|
|
<ProjectDrawer onClose={handleProjectDrawerClose} />,
|
|
document.body,
|
|
'project-drawer'
|
|
)}
|
|
</div>
|
|
);
|
|
});
|
|
|
|
HomePage.displayName = 'HomePage';
|
|
|
|
export default HomePage;
|