mirror of
https://github.com/docmost/docmost.git
synced 2026-05-07 06:23:06 +08:00
1ad53c2581
* feat(ee): public sharing controls * lint
280 lines
8.8 KiB
TypeScript
280 lines
8.8 KiB
TypeScript
import {
|
|
ActionIcon,
|
|
Anchor,
|
|
Button,
|
|
Group,
|
|
Indicator,
|
|
Popover,
|
|
Switch,
|
|
Text,
|
|
TextInput,
|
|
} from "@mantine/core";
|
|
import { IconExternalLink, IconWorld, IconLock } from "@tabler/icons-react";
|
|
import React, { useEffect, useMemo, useState } from "react";
|
|
import {
|
|
useCreateShareMutation,
|
|
useDeleteShareMutation,
|
|
useShareForPageQuery,
|
|
useUpdateShareMutation,
|
|
} from "@/features/share/queries/share-query.ts";
|
|
import { Link, useNavigate, useParams } from "react-router-dom";
|
|
import { extractPageSlugId, getPageIcon } from "@/lib";
|
|
import { useTranslation } from "react-i18next";
|
|
import { usePageQuery } from "@/features/page/queries/page-query.ts";
|
|
import CopyTextButton from "@/components/common/copy.tsx";
|
|
import { getAppUrl, isCloud } from "@/lib/config.ts";
|
|
import { buildPageUrl } from "@/features/page/page.utils.ts";
|
|
import classes from "@/features/share/components/share.module.css";
|
|
import useTrial from "@/ee/hooks/use-trial.tsx";
|
|
import { useAtom } from "jotai";
|
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
|
import { useSpaceQuery } from "@/features/space/queries/space-query.ts";
|
|
|
|
interface ShareModalProps {
|
|
readOnly: boolean;
|
|
}
|
|
export default function ShareModal({ readOnly }: ShareModalProps) {
|
|
const { t } = useTranslation();
|
|
const navigate = useNavigate();
|
|
const { pageSlug } = useParams();
|
|
const pageSlugId = extractPageSlugId(pageSlug);
|
|
const { data: page } = usePageQuery({ pageId: pageSlugId });
|
|
const pageId = page?.id;
|
|
const { data: share } = useShareForPageQuery(pageId);
|
|
const { spaceSlug } = useParams();
|
|
const { isTrial } = useTrial();
|
|
const [workspace] = useAtom(workspaceAtom);
|
|
const { data: space } = useSpaceQuery(spaceSlug);
|
|
const workspaceDisabled =
|
|
workspace?.settings?.sharing?.disabled === true;
|
|
const spaceDisabled = space?.settings?.sharing?.disabled === true;
|
|
const sharingDisabled = workspaceDisabled || spaceDisabled;
|
|
const createShareMutation = useCreateShareMutation();
|
|
const updateShareMutation = useUpdateShareMutation();
|
|
const deleteShareMutation = useDeleteShareMutation();
|
|
// pageIsShared means that the share exists and its level equals zero.
|
|
const pageIsShared = share && share.level === 0;
|
|
// if level is greater than zero, then it is a descendant page from a shared page
|
|
const isDescendantShared = share && share.level > 0;
|
|
|
|
const publicLink = `${getAppUrl()}/share/${share?.key}/p/${pageSlug}`;
|
|
|
|
const [isPagePublic, setIsPagePublic] = useState<boolean>(false);
|
|
useEffect(() => {
|
|
if (share) {
|
|
setIsPagePublic(true);
|
|
} else {
|
|
setIsPagePublic(false);
|
|
}
|
|
}, [share, pageId]);
|
|
|
|
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
|
const value = event.currentTarget.checked;
|
|
|
|
if (value) {
|
|
createShareMutation.mutateAsync({
|
|
pageId: pageId,
|
|
includeSubPages: true,
|
|
searchIndexing: false,
|
|
});
|
|
setIsPagePublic(value);
|
|
} else {
|
|
if (share && share.id) {
|
|
deleteShareMutation.mutateAsync(share.id);
|
|
setIsPagePublic(value);
|
|
}
|
|
}
|
|
};
|
|
|
|
const handleSubPagesChange = async (
|
|
event: React.ChangeEvent<HTMLInputElement>,
|
|
) => {
|
|
const value = event.currentTarget.checked;
|
|
updateShareMutation.mutateAsync({
|
|
shareId: share.id,
|
|
includeSubPages: value,
|
|
});
|
|
};
|
|
|
|
const handleIndexSearchChange = async (
|
|
event: React.ChangeEvent<HTMLInputElement>,
|
|
) => {
|
|
const value = event.currentTarget.checked;
|
|
updateShareMutation.mutateAsync({
|
|
shareId: share.id,
|
|
searchIndexing: value,
|
|
});
|
|
};
|
|
|
|
const shareLink = useMemo(
|
|
() => (
|
|
<Group my="sm" gap={4} wrap="nowrap">
|
|
<TextInput
|
|
variant="filled"
|
|
value={publicLink}
|
|
readOnly
|
|
rightSection={<CopyTextButton text={publicLink} />}
|
|
style={{ width: "100%" }}
|
|
/>
|
|
<ActionIcon
|
|
component="a"
|
|
variant="default"
|
|
target="_blank"
|
|
href={publicLink}
|
|
size="sm"
|
|
>
|
|
<IconExternalLink size={16} />
|
|
</ActionIcon>
|
|
</Group>
|
|
),
|
|
[publicLink],
|
|
);
|
|
|
|
return (
|
|
<Popover width={350} position="bottom" withArrow shadow="md">
|
|
<Popover.Target>
|
|
<Button
|
|
style={{ border: "none" }}
|
|
size="compact-sm"
|
|
leftSection={
|
|
<Indicator
|
|
color="green"
|
|
offset={5}
|
|
disabled={!isPagePublic}
|
|
withBorder
|
|
>
|
|
<IconWorld size={20} stroke={1.5} />
|
|
</Indicator>
|
|
}
|
|
variant="default"
|
|
>
|
|
{t("Share")}
|
|
</Button>
|
|
</Popover.Target>
|
|
<Popover.Dropdown style={{ userSelect: "none" }}>
|
|
{isCloud() && isTrial ? (
|
|
<>
|
|
<Group justify="center" mb="sm">
|
|
<IconLock size={20} stroke={1.5} />
|
|
</Group>
|
|
<Text size="sm" ta="center" fw={500} mb="xs">
|
|
{t("Upgrade to share pages")}
|
|
</Text>
|
|
<Text size="sm" c="dimmed" ta="center" mb="sm">
|
|
{t(
|
|
"Page sharing is available on paid plans. Upgrade to share your pages publicly.",
|
|
)}
|
|
</Text>
|
|
<Button
|
|
size="xs"
|
|
onClick={() => navigate("/settings/billing")}
|
|
fullWidth
|
|
>
|
|
{t("Upgrade Plan")}
|
|
</Button>
|
|
</>
|
|
) : sharingDisabled ? (
|
|
<>
|
|
<Group justify="center" mb="sm">
|
|
<IconLock size={20} stroke={1.5} />
|
|
</Group>
|
|
<Text size="sm" ta="center" fw={500} mb="xs">
|
|
{t("Public sharing is disabled")}
|
|
</Text>
|
|
<Text size="sm" c="dimmed" ta="center">
|
|
{workspaceDisabled
|
|
? t("Public sharing has been disabled at the workspace level.")
|
|
: t("Public sharing has been disabled for this space.")}
|
|
</Text>
|
|
</>
|
|
) : isDescendantShared ? (
|
|
<>
|
|
<Text size="sm">{t("Inherits public sharing from")}</Text>
|
|
<Anchor
|
|
size="sm"
|
|
underline="never"
|
|
style={{
|
|
cursor: "pointer",
|
|
color: "var(--mantine-color-text)",
|
|
}}
|
|
component={Link}
|
|
to={buildPageUrl(
|
|
spaceSlug,
|
|
share.sharedPage.slugId,
|
|
share.sharedPage.title,
|
|
)}
|
|
>
|
|
<Group gap="4" wrap="nowrap" my="sm">
|
|
{getPageIcon(share.sharedPage.icon)}
|
|
<div className={classes.shareLinkText}>
|
|
<Text fz="sm" fw={500} lineClamp={1}>
|
|
{share.sharedPage.title || t("untitled")}
|
|
</Text>
|
|
</div>
|
|
</Group>
|
|
</Anchor>
|
|
|
|
{shareLink}
|
|
</>
|
|
) : (
|
|
<>
|
|
<Group justify="space-between" wrap="nowrap" gap="xl">
|
|
<div>
|
|
<Text size="sm">
|
|
{isPagePublic ? t("Shared to web") : t("Share to web")}
|
|
</Text>
|
|
<Text size="xs" c="dimmed">
|
|
{isPagePublic
|
|
? t("Anyone with the link can view this page")
|
|
: t("Make this page publicly accessible")}
|
|
</Text>
|
|
</div>
|
|
<Switch
|
|
onChange={handleChange}
|
|
defaultChecked={isPagePublic}
|
|
disabled={readOnly}
|
|
size="xs"
|
|
/>
|
|
</Group>
|
|
|
|
{pageIsShared && (
|
|
<>
|
|
{shareLink}
|
|
<Group justify="space-between" wrap="nowrap" gap="xl">
|
|
<div>
|
|
<Text size="sm">{t("Include sub-pages")}</Text>
|
|
<Text size="xs" c="dimmed">
|
|
{t("Make sub-pages public too")}
|
|
</Text>
|
|
</div>
|
|
|
|
<Switch
|
|
onChange={handleSubPagesChange}
|
|
checked={share.includeSubPages}
|
|
size="xs"
|
|
disabled={readOnly}
|
|
/>
|
|
</Group>
|
|
<Group justify="space-between" wrap="nowrap" gap="xl" mt="sm">
|
|
<div>
|
|
<Text size="sm">{t("Search engine indexing")}</Text>
|
|
<Text size="xs" c="dimmed">
|
|
{t("Allow search engines to index page")}
|
|
</Text>
|
|
</div>
|
|
<Switch
|
|
onChange={handleIndexSearchChange}
|
|
checked={share.searchIndexing}
|
|
size="xs"
|
|
disabled={readOnly}
|
|
/>
|
|
</Group>
|
|
</>
|
|
)}
|
|
</>
|
|
)}
|
|
</Popover.Dropdown>
|
|
</Popover>
|
|
);
|
|
}
|