631 lines
23 KiB
TypeScript
631 lines
23 KiB
TypeScript
import * as icon from '@mdi/js';
|
|
import Head from 'next/head';
|
|
import React from 'react';
|
|
import axios from 'axios';
|
|
import type { ReactElement } from 'react';
|
|
import LayoutAuthenticated from '../layouts/Authenticated';
|
|
import SectionMain from '../components/SectionMain';
|
|
import SectionTitleLineWithButton from '../components/SectionTitleLineWithButton';
|
|
import BaseIcon from '../components/BaseIcon';
|
|
import { getPageTitle } from '../config';
|
|
import Link from 'next/link';
|
|
import { useTranslation } from 'next-i18next';
|
|
|
|
import { hasPermission } from '../helpers/userPermissions';
|
|
import { fetchWidgets } from '../stores/roles/rolesSlice';
|
|
import { WidgetCreator } from '../components/WidgetCreator/WidgetCreator';
|
|
import { SmartWidget } from '../components/SmartWidget/SmartWidget';
|
|
|
|
import { useAppDispatch, useAppSelector } from '../stores/hooks';
|
|
const Dashboard = () => {
|
|
const { t } = useTranslation('common');
|
|
const dispatch = useAppDispatch();
|
|
const iconsColor = useAppSelector((state) => state.style.iconsColor);
|
|
const corners = useAppSelector((state) => state.style.corners);
|
|
const cardsStyle = useAppSelector((state) => state.style.cardsStyle);
|
|
|
|
const loadingMessage = t('pages.dashboard.loading', {
|
|
defaultValue: 'Loading...',
|
|
});
|
|
|
|
const [users, setUsers] = React.useState(loadingMessage);
|
|
const [activities, setActivities] = React.useState(loadingMessage);
|
|
const [appraisals, setAppraisals] = React.useState(loadingMessage);
|
|
const [clients, setClients] = React.useState(loadingMessage);
|
|
const [invoices, setInvoices] = React.useState(loadingMessage);
|
|
const [messages, setMessages] = React.useState(loadingMessage);
|
|
const [notifications, setNotifications] = React.useState(loadingMessage);
|
|
const [orders, setOrders] = React.useState(loadingMessage);
|
|
const [properties, setProperties] = React.useState(loadingMessage);
|
|
const [roles, setRoles] = React.useState(loadingMessage);
|
|
const [permissions, setPermissions] = React.useState(loadingMessage);
|
|
const [organizations, setOrganizations] = React.useState(loadingMessage);
|
|
|
|
const [widgetsRole, setWidgetsRole] = React.useState({
|
|
role: { value: '', label: '' },
|
|
});
|
|
const { currentUser } = useAppSelector((state) => state.auth);
|
|
const { isFetchingQuery } = useAppSelector((state) => state.openAi);
|
|
|
|
const { rolesWidgets, loading } = useAppSelector((state) => state.roles);
|
|
|
|
const organizationId = currentUser?.organizations?.id;
|
|
|
|
async function loadData() {
|
|
const entities = [
|
|
'users',
|
|
'activities',
|
|
'appraisals',
|
|
'clients',
|
|
'invoices',
|
|
'messages',
|
|
'notifications',
|
|
'orders',
|
|
'properties',
|
|
'roles',
|
|
'permissions',
|
|
'organizations',
|
|
];
|
|
const fns = [
|
|
setUsers,
|
|
setActivities,
|
|
setAppraisals,
|
|
setClients,
|
|
setInvoices,
|
|
setMessages,
|
|
setNotifications,
|
|
setOrders,
|
|
setProperties,
|
|
setRoles,
|
|
setPermissions,
|
|
setOrganizations,
|
|
];
|
|
|
|
const requests = entities.map((entity, index) => {
|
|
if (hasPermission(currentUser, `READ_${entity.toUpperCase()}`)) {
|
|
return axios.get(`/${entity.toLowerCase()}/count`);
|
|
} else {
|
|
fns[index](null);
|
|
return Promise.resolve({ data: { count: null } });
|
|
}
|
|
});
|
|
|
|
Promise.allSettled(requests).then((results) => {
|
|
results.forEach((result, i) => {
|
|
if (result.status === 'fulfilled') {
|
|
fns[i](result.value.data.count);
|
|
} else {
|
|
fns[i](result.reason.message);
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
async function getWidgets(roleId) {
|
|
await dispatch(fetchWidgets(roleId));
|
|
}
|
|
React.useEffect(() => {
|
|
if (!currentUser) return;
|
|
loadData().then();
|
|
setWidgetsRole({
|
|
role: {
|
|
value: currentUser?.app_role?.id,
|
|
label: currentUser?.app_role?.name,
|
|
},
|
|
});
|
|
}, [currentUser]);
|
|
|
|
React.useEffect(() => {
|
|
if (!currentUser || !widgetsRole?.role?.value) return;
|
|
getWidgets(widgetsRole?.role?.value || '').then();
|
|
}, [widgetsRole?.role?.value]);
|
|
|
|
return (
|
|
<>
|
|
<Head>
|
|
<title>
|
|
{getPageTitle(
|
|
t('pages.dashboard.pageTitle', { defaultValue: 'Overview' }),
|
|
)}
|
|
</title>
|
|
</Head>
|
|
<SectionMain>
|
|
<SectionTitleLineWithButton
|
|
icon={icon.mdiChartTimelineVariant}
|
|
title={t('pages.dashboard.overview', { defaultValue: 'Overview' })}
|
|
main
|
|
>
|
|
{''}
|
|
</SectionTitleLineWithButton>
|
|
{/* Welcome message under page title */}
|
|
{currentUser?.first_name && (
|
|
<p className="text-lg font-medium mt-2 mb-6">
|
|
Welcome to VRES, {currentUser.first_name}!
|
|
</p>
|
|
)}
|
|
|
|
|
|
{hasPermission(currentUser, 'CREATE_ROLES') && (
|
|
<WidgetCreator
|
|
currentUser={currentUser}
|
|
isFetchingQuery={isFetchingQuery}
|
|
setWidgetsRole={setWidgetsRole}
|
|
widgetsRole={widgetsRole}
|
|
/>
|
|
)}
|
|
{!!rolesWidgets.length &&
|
|
hasPermission(currentUser, 'CREATE_ROLES') && (
|
|
<p className=' text-gray-500 dark:text-gray-400 mb-4'>
|
|
{`${widgetsRole?.role?.label || 'Users'}'s widgets`}
|
|
</p>
|
|
)}
|
|
|
|
<div className='grid grid-cols-1 gap-6 lg:grid-cols-4 mb-6 grid-flow-dense'>
|
|
{(isFetchingQuery || loading) && (
|
|
<div
|
|
className={` ${
|
|
corners !== 'rounded-full' ? corners : 'rounded-3xl'
|
|
} dark:bg-dark-900 text-lg leading-tight text-gray-500 flex items-center ${cardsStyle} dark:border-dark-700 p-6`}
|
|
>
|
|
<BaseIcon
|
|
className={`${iconsColor} animate-spin mr-5`}
|
|
w='w-16'
|
|
h='h-16'
|
|
size={48}
|
|
path={icon.mdiLoading}
|
|
/>{' '}
|
|
{t('pages.dashboard.loadingWidgets', {
|
|
defaultValue: 'Loading widgets...',
|
|
})}
|
|
</div>
|
|
)}
|
|
|
|
{rolesWidgets &&
|
|
rolesWidgets.map((widget) => (
|
|
<SmartWidget
|
|
key={widget.id}
|
|
userId={currentUser?.id}
|
|
widget={widget}
|
|
roleId={widgetsRole?.role?.value || ''}
|
|
admin={hasPermission(currentUser, 'CREATE_ROLES')}
|
|
/>
|
|
))}
|
|
</div>
|
|
|
|
{!!rolesWidgets.length && (
|
|
<hr className='my-6 text-pastelEmeraldTheme-mainBG ' />
|
|
)}
|
|
|
|
<div
|
|
id='dashboard'
|
|
className='grid grid-cols-1 gap-6 lg:grid-cols-3 mb-6'
|
|
>
|
|
{hasPermission(currentUser, 'READ_USERS') && (
|
|
<Link href={'/users/users-list'}>
|
|
<div
|
|
className={`${
|
|
corners !== 'rounded-full' ? corners : 'rounded-3xl'
|
|
} dark:bg-dark-900 ${cardsStyle} dark:border-dark-700 p-6`}
|
|
>
|
|
<div className='flex justify-between align-center'>
|
|
<div>
|
|
<div className='text-lg leading-tight text-gray-500 dark:text-gray-400'>
|
|
Users
|
|
</div>
|
|
<div className='text-3xl leading-tight font-semibold'>
|
|
{users}
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<BaseIcon
|
|
className={`${iconsColor}`}
|
|
w='w-16'
|
|
h='h-16'
|
|
size={48}
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
// @ts-ignore
|
|
path={icon.mdiAccountGroup || icon.mdiTable}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</Link>
|
|
)}
|
|
|
|
{hasPermission(currentUser, 'READ_ACTIVITIES') && (
|
|
<Link href={'/activities/activities-list'}>
|
|
<div
|
|
className={`${
|
|
corners !== 'rounded-full' ? corners : 'rounded-3xl'
|
|
} dark:bg-dark-900 ${cardsStyle} dark:border-dark-700 p-6`}
|
|
>
|
|
<div className='flex justify-between align-center'>
|
|
<div>
|
|
<div className='text-lg leading-tight text-gray-500 dark:text-gray-400'>
|
|
Activities
|
|
</div>
|
|
<div className='text-3xl leading-tight font-semibold'>
|
|
{activities}
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<BaseIcon
|
|
className={`${iconsColor}`}
|
|
w='w-16'
|
|
h='h-16'
|
|
size={48}
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
// @ts-ignore
|
|
path={
|
|
'mdiHistory' in icon
|
|
? icon['mdiHistory' as keyof typeof icon]
|
|
: icon.mdiTable || icon.mdiTable
|
|
}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</Link>
|
|
)}
|
|
|
|
{hasPermission(currentUser, 'READ_APPRAISALS') && (
|
|
<Link href={'/appraisals/appraisals-list'}>
|
|
<div
|
|
className={`${
|
|
corners !== 'rounded-full' ? corners : 'rounded-3xl'
|
|
} dark:bg-dark-900 ${cardsStyle} dark:border-dark-700 p-6`}
|
|
>
|
|
<div className='flex justify-between align-center'>
|
|
<div>
|
|
<div className='text-lg leading-tight text-gray-500 dark:text-gray-400'>
|
|
Appraisals
|
|
</div>
|
|
<div className='text-3xl leading-tight font-semibold'>
|
|
{appraisals}
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<BaseIcon
|
|
className={`${iconsColor}`}
|
|
w='w-16'
|
|
h='h-16'
|
|
size={48}
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
// @ts-ignore
|
|
path={
|
|
'mdiFileDocument' in icon
|
|
? icon['mdiFileDocument' as keyof typeof icon]
|
|
: icon.mdiTable || icon.mdiTable
|
|
}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</Link>
|
|
)}
|
|
|
|
{hasPermission(currentUser, 'READ_CLIENTS') && (
|
|
<Link href={'/clients/clients-list'}>
|
|
<div
|
|
className={`${
|
|
corners !== 'rounded-full' ? corners : 'rounded-3xl'
|
|
} dark:bg-dark-900 ${cardsStyle} dark:border-dark-700 p-6`}
|
|
>
|
|
<div className='flex justify-between align-center'>
|
|
<div>
|
|
<div className='text-lg leading-tight text-gray-500 dark:text-gray-400'>
|
|
Clients
|
|
</div>
|
|
<div className='text-3xl leading-tight font-semibold'>
|
|
{clients}
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<BaseIcon
|
|
className={`${iconsColor}`}
|
|
w='w-16'
|
|
h='h-16'
|
|
size={48}
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
// @ts-ignore
|
|
path={
|
|
'mdiAccountGroup' in icon
|
|
? icon['mdiAccountGroup' as keyof typeof icon]
|
|
: icon.mdiTable || icon.mdiTable
|
|
}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</Link>
|
|
)}
|
|
|
|
{hasPermission(currentUser, 'READ_INVOICES') && (
|
|
<Link href={'/invoices/invoices-list'}>
|
|
<div
|
|
className={`${
|
|
corners !== 'rounded-full' ? corners : 'rounded-3xl'
|
|
} dark:bg-dark-900 ${cardsStyle} dark:border-dark-700 p-6`}
|
|
>
|
|
<div className='flex justify-between align-center'>
|
|
<div>
|
|
<div className='text-lg leading-tight text-gray-500 dark:text-gray-400'>
|
|
Invoices
|
|
</div>
|
|
<div className='text-3xl leading-tight font-semibold'>
|
|
{invoices}
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<BaseIcon
|
|
className={`${iconsColor}`}
|
|
w='w-16'
|
|
h='h-16'
|
|
size={48}
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
// @ts-ignore
|
|
path={
|
|
'mdiFileInvoice' in icon
|
|
? icon['mdiFileInvoice' as keyof typeof icon]
|
|
: icon.mdiTable || icon.mdiTable
|
|
}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</Link>
|
|
)}
|
|
|
|
{hasPermission(currentUser, 'READ_MESSAGES') && (
|
|
<Link href={'/messages/messages-list'}>
|
|
<div
|
|
className={`${
|
|
corners !== 'rounded-full' ? corners : 'rounded-3xl'
|
|
} dark:bg-dark-900 ${cardsStyle} dark:border-dark-700 p-6`}
|
|
>
|
|
<div className='flex justify-between align-center'>
|
|
<div>
|
|
<div className='text-lg leading-tight text-gray-500 dark:text-gray-400'>
|
|
Messages
|
|
</div>
|
|
<div className='text-3xl leading-tight font-semibold'>
|
|
{messages}
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<BaseIcon
|
|
className={`${iconsColor}`}
|
|
w='w-16'
|
|
h='h-16'
|
|
size={48}
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
// @ts-ignore
|
|
path={
|
|
'mdiMessage' in icon
|
|
? icon['mdiMessage' as keyof typeof icon]
|
|
: icon.mdiTable || icon.mdiTable
|
|
}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</Link>
|
|
)}
|
|
|
|
{hasPermission(currentUser, 'READ_NOTIFICATIONS') && (
|
|
<Link href={'/notifications/notifications-list'}>
|
|
<div
|
|
className={`${
|
|
corners !== 'rounded-full' ? corners : 'rounded-3xl'
|
|
} dark:bg-dark-900 ${cardsStyle} dark:border-dark-700 p-6`}
|
|
>
|
|
<div className='flex justify-between align-center'>
|
|
<div>
|
|
<div className='text-lg leading-tight text-gray-500 dark:text-gray-400'>
|
|
Notifications
|
|
</div>
|
|
<div className='text-3xl leading-tight font-semibold'>
|
|
{notifications}
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<BaseIcon
|
|
className={`${iconsColor}`}
|
|
w='w-16'
|
|
h='h-16'
|
|
size={48}
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
// @ts-ignore
|
|
path={
|
|
'mdiBell' in icon
|
|
? icon['mdiBell' as keyof typeof icon]
|
|
: icon.mdiTable || icon.mdiTable
|
|
}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</Link>
|
|
)}
|
|
|
|
{hasPermission(currentUser, 'READ_ORDERS') && (
|
|
<Link href={'/orders/orders-list'}>
|
|
<div
|
|
className={`${
|
|
corners !== 'rounded-full' ? corners : 'rounded-3xl'
|
|
} dark:bg-dark-900 ${cardsStyle} dark:border-dark-700 p-6`}
|
|
>
|
|
<div className='flex justify-between align-center'>
|
|
<div>
|
|
<div className='text-lg leading-tight text-gray-500 dark:text-gray-400'>
|
|
Orders
|
|
</div>
|
|
<div className='text-3xl leading-tight font-semibold'>
|
|
{orders}
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<BaseIcon
|
|
className={`${iconsColor}`}
|
|
w='w-16'
|
|
h='h-16'
|
|
size={48}
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
// @ts-ignore
|
|
path={
|
|
'mdiClipboardList' in icon
|
|
? icon['mdiClipboardList' as keyof typeof icon]
|
|
: icon.mdiTable || icon.mdiTable
|
|
}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</Link>
|
|
)}
|
|
|
|
{hasPermission(currentUser, 'READ_PROPERTIES') && (
|
|
<Link href={'/properties/properties-list'}>
|
|
<div
|
|
className={`${
|
|
corners !== 'rounded-full' ? corners : 'rounded-3xl'
|
|
} dark:bg-dark-900 ${cardsStyle} dark:border-dark-700 p-6`}
|
|
>
|
|
<div className='flex justify-between align-center'>
|
|
<div>
|
|
<div className='text-lg leading-tight text-gray-500 dark:text-gray-400'>
|
|
Properties
|
|
</div>
|
|
<div className='text-3xl leading-tight font-semibold'>
|
|
{properties}
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<BaseIcon
|
|
className={`${iconsColor}`}
|
|
w='w-16'
|
|
h='h-16'
|
|
size={48}
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
// @ts-ignore
|
|
path={
|
|
'mdiHome' in icon
|
|
? icon['mdiHome' as keyof typeof icon]
|
|
: icon.mdiTable || icon.mdiTable
|
|
}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</Link>
|
|
)}
|
|
|
|
{hasPermission(currentUser, 'READ_ROLES') && (
|
|
<Link href={'/roles/roles-list'}>
|
|
<div
|
|
className={`${
|
|
corners !== 'rounded-full' ? corners : 'rounded-3xl'
|
|
} dark:bg-dark-900 ${cardsStyle} dark:border-dark-700 p-6`}
|
|
>
|
|
<div className='flex justify-between align-center'>
|
|
<div>
|
|
<div className='text-lg leading-tight text-gray-500 dark:text-gray-400'>
|
|
Roles
|
|
</div>
|
|
<div className='text-3xl leading-tight font-semibold'>
|
|
{roles}
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<BaseIcon
|
|
className={`${iconsColor}`}
|
|
w='w-16'
|
|
h='h-16'
|
|
size={48}
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
// @ts-ignore
|
|
path={
|
|
icon.mdiShieldAccountVariantOutline || icon.mdiTable
|
|
}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</Link>
|
|
)}
|
|
|
|
{hasPermission(currentUser, 'READ_PERMISSIONS') && (
|
|
<Link href={'/permissions/permissions-list'}>
|
|
<div
|
|
className={`${
|
|
corners !== 'rounded-full' ? corners : 'rounded-3xl'
|
|
} dark:bg-dark-900 ${cardsStyle} dark:border-dark-700 p-6`}
|
|
>
|
|
<div className='flex justify-between align-center'>
|
|
<div>
|
|
<div className='text-lg leading-tight text-gray-500 dark:text-gray-400'>
|
|
Permissions
|
|
</div>
|
|
<div className='text-3xl leading-tight font-semibold'>
|
|
{permissions}
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<BaseIcon
|
|
className={`${iconsColor}`}
|
|
w='w-16'
|
|
h='h-16'
|
|
size={48}
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
// @ts-ignore
|
|
path={icon.mdiShieldAccountOutline || icon.mdiTable}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</Link>
|
|
)}
|
|
|
|
{hasPermission(currentUser, 'READ_ORGANIZATIONS') && (
|
|
<Link href={'/organizations/organizations-list'}>
|
|
<div
|
|
className={`${
|
|
corners !== 'rounded-full' ? corners : 'rounded-3xl'
|
|
} dark:bg-dark-900 ${cardsStyle} dark:border-dark-700 p-6`}
|
|
>
|
|
<div className='flex justify-between align-center'>
|
|
<div>
|
|
<div className='text-lg leading-tight text-gray-500 dark:text-gray-400'>
|
|
Organizations
|
|
</div>
|
|
<div className='text-3xl leading-tight font-semibold'>
|
|
{organizations}
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<BaseIcon
|
|
className={`${iconsColor}`}
|
|
w='w-16'
|
|
h='h-16'
|
|
size={48}
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
// @ts-ignore
|
|
path={icon.mdiTable || icon.mdiTable}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</Link>
|
|
)}
|
|
</div>
|
|
</SectionMain>
|
|
</>
|
|
);
|
|
};
|
|
|
|
Dashboard.getLayout = function getLayout(page: ReactElement) {
|
|
return <LayoutAuthenticated>{page}</LayoutAuthenticated>;
|
|
};
|
|
|
|
export default Dashboard;
|