Reactive-Resume/apps/client/components/build/LeftSidebar/LeftSidebar.tsx

125 lines
3.8 KiB
TypeScript

import { Add, Star } from '@mui/icons-material';
import { Button, Divider, IconButton, SwipeableDrawer, Tooltip, useMediaQuery, useTheme } from '@mui/material';
import { Section as SectionRecord } from '@reactive-resume/schema';
import get from 'lodash/get';
import Link from 'next/link';
import { useTranslation } from 'next-i18next';
import { useMemo } from 'react';
import { validate } from 'uuid';
import Logo from '@/components/shared/Logo';
import { getCustomSections, left } from '@/config/sections';
import { setSidebarState } from '@/store/build/buildSlice';
import { useAppDispatch, useAppSelector } from '@/store/hooks';
import { addSection } from '@/store/resume/resumeSlice';
import styles from './LeftSidebar.module.scss';
import Section from './sections/Section';
const LeftSidebar = () => {
const theme = useTheme();
const { t } = useTranslation();
const dispatch = useAppDispatch();
const isDesktop = useMediaQuery(theme.breakpoints.up('lg'));
const sections = useAppSelector((state) => state.resume.sections);
const { open } = useAppSelector((state) => state.build.sidebar.left);
const customSections = useMemo(() => getCustomSections(sections), [sections]);
const handleOpen = () => dispatch(setSidebarState({ sidebar: 'left', state: { open: true } }));
const handleClose = () => dispatch(setSidebarState({ sidebar: 'left', state: { open: false } }));
const handleClick = (id: string) => {
const elementId = validate(id) ? `#section-${id}` : `#${id}`;
const section = document.querySelector(elementId);
section.scrollIntoView({ behavior: 'smooth', block: 'start' });
};
const handleAddSection = () => {
const newSection: SectionRecord = {
name: 'Custom Section',
type: 'custom',
visible: true,
columns: 2,
items: [],
};
dispatch(addSection({ value: newSection }));
};
return (
<SwipeableDrawer
open={open}
anchor="left"
onOpen={handleOpen}
onClose={handleClose}
PaperProps={{ className: '!shadow-lg' }}
variant={isDesktop ? 'persistent' : 'temporary'}
>
<div className={styles.container}>
<nav>
<div>
<Link href="/dashboard">
<a className="inline-flex">
<Logo size={40} />
</a>
</Link>
<Divider />
</div>
<div className={styles.sections}>
{left.map(({ id, icon }) => (
<Tooltip
arrow
key={id}
placement="right"
title={get(sections, `${id}.name`, t(`builder.leftSidebar.sections.${id}.heading`))}
>
<IconButton onClick={() => handleClick(id)}>{icon}</IconButton>
</Tooltip>
))}
{customSections.map(({ id }) => (
<Tooltip key={id} title={get(sections, `${id}.name`, '')} placement="right" arrow>
<IconButton onClick={() => handleClick(id)}>
<Star />
</IconButton>
</Tooltip>
))}
</div>
<div />
</nav>
<main>
{left.map(({ id, component }) => (
<section key={id} id={id}>
{component}
</section>
))}
{customSections.map(({ id }) => (
<section key={id} id={`section-${id}`}>
<Section path={`sections.${id}`} isEditable isHideable isDeletable />
</section>
))}
<div className="py-6 text-right">
<Button fullWidth variant="outlined" startIcon={<Add />} onClick={handleAddSection}>
{t('builder.common.actions.add', { token: t('builder.leftSidebar.sections.section.heading') })}
</Button>
</div>
</main>
</div>
</SwipeableDrawer>
);
};
export default LeftSidebar;