mirror of
https://github.com/Shadowfita/docmost.git
synced 2025-11-12 07:42:34 +10:00
* fix page import title bug * fix youtube embed in markdown export * add link to rendered file html * fix: markdown callout import * update local generateJSON * feat: switch spaces from sidebar * remove unused package * feat: editor date menu command * fix date description * update default locale code * feat: add more code highlight languages
237 lines
6.7 KiB
TypeScript
237 lines
6.7 KiB
TypeScript
import {
|
|
ActionIcon,
|
|
Group,
|
|
Menu,
|
|
Text,
|
|
Tooltip,
|
|
UnstyledButton,
|
|
} from '@mantine/core';
|
|
import { spotlight } from '@mantine/spotlight';
|
|
import {
|
|
IconArrowDown,
|
|
IconDots,
|
|
IconHome,
|
|
IconPlus,
|
|
IconSearch,
|
|
IconSettings,
|
|
} from '@tabler/icons-react';
|
|
|
|
import classes from './space-sidebar.module.css';
|
|
import React, { useMemo } from 'react';
|
|
import { useAtom } from 'jotai';
|
|
import { SearchSpotlight } from '@/features/search/search-spotlight.tsx';
|
|
import { treeApiAtom } from '@/features/page/tree/atoms/tree-api-atom.ts';
|
|
import { Link, useLocation, useParams } from 'react-router-dom';
|
|
import clsx from 'clsx';
|
|
import { useDisclosure } from '@mantine/hooks';
|
|
import SpaceSettingsModal from '@/features/space/components/settings-modal.tsx';
|
|
import { useGetSpaceBySlugQuery } from '@/features/space/queries/space-query.ts';
|
|
import { getSpaceUrl } from '@/lib/config.ts';
|
|
import SpaceTree from '@/features/page/tree/components/space-tree.tsx';
|
|
import { useSpaceAbility } from '@/features/space/permissions/use-space-ability.ts';
|
|
import {
|
|
SpaceCaslAction,
|
|
SpaceCaslSubject,
|
|
} from '@/features/space/permissions/permissions.type.ts';
|
|
import PageImportModal from '@/features/page/components/page-import-modal.tsx';
|
|
import { SwitchSpace } from './switch-space';
|
|
|
|
export function SpaceSidebar() {
|
|
const [tree] = useAtom(treeApiAtom);
|
|
const location = useLocation();
|
|
const [opened, { open: openSettings, close: closeSettings }] =
|
|
useDisclosure(false);
|
|
const { spaceSlug } = useParams();
|
|
const { data: space, isLoading, isError } = useGetSpaceBySlugQuery(spaceSlug);
|
|
|
|
const spaceRules = space?.membership?.permissions;
|
|
const spaceAbility = useMemo(() => useSpaceAbility(spaceRules), [spaceRules]);
|
|
|
|
if (!space) {
|
|
return <></>;
|
|
}
|
|
|
|
function handleCreatePage() {
|
|
tree?.create({ parentId: null, type: 'internal', index: 0 });
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<div className={classes.navbar}>
|
|
<div
|
|
className={classes.section}
|
|
style={{
|
|
border: 'none',
|
|
marginTop: 2,
|
|
marginBottom: 3,
|
|
}}
|
|
>
|
|
<SwitchSpace spaceName={space?.name} spaceSlug={space?.slug} />
|
|
</div>
|
|
|
|
<div className={classes.section}>
|
|
<div className={classes.menuItems}>
|
|
<UnstyledButton
|
|
component={Link}
|
|
to={getSpaceUrl(spaceSlug)}
|
|
className={clsx(
|
|
classes.menu,
|
|
location.pathname.toLowerCase() === getSpaceUrl(spaceSlug)
|
|
? classes.activeButton
|
|
: ''
|
|
)}
|
|
>
|
|
<div className={classes.menuItemInner}>
|
|
<IconHome
|
|
size={18}
|
|
className={classes.menuItemIcon}
|
|
stroke={2}
|
|
/>
|
|
<span>Overview</span>
|
|
</div>
|
|
</UnstyledButton>
|
|
|
|
<UnstyledButton className={classes.menu} onClick={spotlight.open}>
|
|
<div className={classes.menuItemInner}>
|
|
<IconSearch
|
|
size={18}
|
|
className={classes.menuItemIcon}
|
|
stroke={2}
|
|
/>
|
|
<span>Search</span>
|
|
</div>
|
|
</UnstyledButton>
|
|
|
|
<UnstyledButton className={classes.menu} onClick={openSettings}>
|
|
<div className={classes.menuItemInner}>
|
|
<IconSettings
|
|
size={18}
|
|
className={classes.menuItemIcon}
|
|
stroke={2}
|
|
/>
|
|
<span>Space settings</span>
|
|
</div>
|
|
</UnstyledButton>
|
|
|
|
{spaceAbility.can(
|
|
SpaceCaslAction.Manage,
|
|
SpaceCaslSubject.Page
|
|
) && (
|
|
<UnstyledButton
|
|
className={classes.menu}
|
|
onClick={handleCreatePage}
|
|
>
|
|
<div className={classes.menuItemInner}>
|
|
<IconPlus
|
|
size={18}
|
|
className={classes.menuItemIcon}
|
|
stroke={2}
|
|
/>
|
|
<span>New page</span>
|
|
</div>
|
|
</UnstyledButton>
|
|
)}
|
|
</div>
|
|
</div>
|
|
|
|
<div className={classes.section}>
|
|
<Group className={classes.pagesHeader} justify="space-between">
|
|
<Text size="xs" fw={500} c="dimmed">
|
|
Pages
|
|
</Text>
|
|
|
|
{spaceAbility.can(
|
|
SpaceCaslAction.Manage,
|
|
SpaceCaslSubject.Page
|
|
) && (
|
|
<Group gap="xs">
|
|
<SpaceMenu spaceId={space.id} onSpaceSettings={openSettings} />
|
|
|
|
<Tooltip label="Create page" withArrow position="right">
|
|
<ActionIcon
|
|
variant="default"
|
|
size={18}
|
|
onClick={handleCreatePage}
|
|
aria-label="Create page"
|
|
>
|
|
<IconPlus />
|
|
</ActionIcon>
|
|
</Tooltip>
|
|
</Group>
|
|
)}
|
|
</Group>
|
|
|
|
<div className={classes.pages}>
|
|
<SpaceTree
|
|
spaceId={space.id}
|
|
readOnly={spaceAbility.cannot(
|
|
SpaceCaslAction.Manage,
|
|
SpaceCaslSubject.Page
|
|
)}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<SpaceSettingsModal
|
|
opened={opened}
|
|
onClose={closeSettings}
|
|
spaceId={space?.slug}
|
|
/>
|
|
|
|
<SearchSpotlight spaceId={space.id} />
|
|
</>
|
|
);
|
|
}
|
|
|
|
interface SpaceMenuProps {
|
|
spaceId: string;
|
|
onSpaceSettings: () => void;
|
|
}
|
|
function SpaceMenu({ spaceId, onSpaceSettings }: SpaceMenuProps) {
|
|
const [importOpened, { open: openImportModal, close: closeImportModal }] =
|
|
useDisclosure(false);
|
|
|
|
return (
|
|
<>
|
|
<Menu width={200} shadow="md" withArrow>
|
|
<Menu.Target>
|
|
<Tooltip
|
|
label="Import pages & space settings"
|
|
withArrow
|
|
position="top"
|
|
>
|
|
<ActionIcon variant="default" size={18} aria-label="Space menu">
|
|
<IconDots />
|
|
</ActionIcon>
|
|
</Tooltip>
|
|
</Menu.Target>
|
|
|
|
<Menu.Dropdown>
|
|
<Menu.Item
|
|
onClick={openImportModal}
|
|
leftSection={<IconArrowDown size={16} />}
|
|
>
|
|
Import pages
|
|
</Menu.Item>
|
|
|
|
<Menu.Divider />
|
|
|
|
<Menu.Item
|
|
onClick={onSpaceSettings}
|
|
leftSection={<IconSettings size={16} />}
|
|
>
|
|
Space settings
|
|
</Menu.Item>
|
|
</Menu.Dropdown>
|
|
</Menu>
|
|
|
|
<PageImportModal
|
|
spaceId={spaceId}
|
|
open={importOpened}
|
|
onClose={closeImportModal}
|
|
/>
|
|
</>
|
|
);
|
|
}
|