mirror of
https://github.com/usememos/memos.git
synced 2025-04-13 00:52:07 +02:00
189 lines
6.9 KiB
TypeScript
189 lines
6.9 KiB
TypeScript
import { Button, IconButton, Select, Option } from "@mui/joy";
|
|
import copy from "copy-to-clipboard";
|
|
import React, { useEffect, useRef } from "react";
|
|
import { toast } from "react-hot-toast";
|
|
import { getDateTimeString } from "@/helpers/datetime";
|
|
import { downloadFileFromUrl } from "@/helpers/utils";
|
|
import useCurrentUser from "@/hooks/useCurrentUser";
|
|
import useLoading from "@/hooks/useLoading";
|
|
import toImage from "@/labs/html2image";
|
|
import { useUserStore, useMemoStore, MemoNamePrefix } from "@/store/v1";
|
|
import { Visibility } from "@/types/proto/api/v1/memo_service";
|
|
import { useTranslate } from "@/utils/i18n";
|
|
import { convertVisibilityToString } from "@/utils/memo";
|
|
import { generateDialog } from "./Dialog";
|
|
import Icon from "./Icon";
|
|
import MemoContent from "./MemoContent";
|
|
import MemoResourceListView from "./MemoResourceListView";
|
|
import UserAvatar from "./UserAvatar";
|
|
import VisibilityIcon from "./VisibilityIcon";
|
|
import "@/less/share-memo-dialog.less";
|
|
|
|
interface Props extends DialogProps {
|
|
memoId: number;
|
|
}
|
|
|
|
const ShareMemoDialog: React.FC<Props> = (props: Props) => {
|
|
const { memoId, destroy } = props;
|
|
const t = useTranslate();
|
|
const currentUser = useCurrentUser();
|
|
const userStore = useUserStore();
|
|
const memoStore = useMemoStore();
|
|
const downloadingImageState = useLoading(false);
|
|
const loadingState = useLoading();
|
|
const memoContainerRef = useRef<HTMLDivElement>(null);
|
|
const memo = memoStore.getMemoByName(`${MemoNamePrefix}${memoId}`);
|
|
const user = userStore.getUserByName(memo.creator);
|
|
const readonly = memo?.creator !== currentUser?.name;
|
|
|
|
useEffect(() => {
|
|
(async () => {
|
|
await userStore.getOrFetchUserByName(memo.creator);
|
|
loadingState.setFinish();
|
|
})();
|
|
}, []);
|
|
|
|
const handleCloseBtnClick = () => {
|
|
destroy();
|
|
};
|
|
|
|
const handleDownloadImageBtnClick = () => {
|
|
if (!memoContainerRef.current) {
|
|
return;
|
|
}
|
|
|
|
downloadingImageState.setLoading();
|
|
toImage(memoContainerRef.current, {
|
|
pixelRatio: window.devicePixelRatio * 2,
|
|
})
|
|
.then((url) => {
|
|
downloadFileFromUrl(url, `memos-${getDateTimeString(Date.now())}.png`);
|
|
downloadingImageState.setFinish();
|
|
URL.revokeObjectURL(url);
|
|
})
|
|
.catch((err) => {
|
|
console.error(err);
|
|
});
|
|
};
|
|
|
|
const handleDownloadTextFileBtnClick = () => {
|
|
const blob = new Blob([memo.content], { type: "text/plain;charset=utf-8" });
|
|
const url = URL.createObjectURL(blob);
|
|
downloadFileFromUrl(url, `memos-${getDateTimeString(Date.now())}.md`);
|
|
URL.revokeObjectURL(url);
|
|
};
|
|
|
|
const handleCopyLinkBtnClick = () => {
|
|
copy(`${window.location.origin}/m/${memo.uid}`);
|
|
if (memo.visibility !== Visibility.PUBLIC) {
|
|
toast.success(t("message.succeed-copy-link-not-public"));
|
|
} else {
|
|
toast.success(t("message.succeed-copy-link"));
|
|
}
|
|
};
|
|
|
|
const handleMemoVisibilityOptionChanged = async (visibility: Visibility) => {
|
|
const updatedMemo = await memoStore.updateMemo(
|
|
{
|
|
name: memo.name,
|
|
visibility: visibility,
|
|
},
|
|
["visibility"],
|
|
);
|
|
|
|
if (updatedMemo.visibility == visibility) {
|
|
toast.success(t("message.update-succeed"));
|
|
}
|
|
};
|
|
|
|
if (loadingState.isLoading) {
|
|
return null;
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<div className="dialog-header-container py-3 px-4 !mb-0 rounded-t-lg">
|
|
<p className="">{t("common.share")} Memo</p>
|
|
<IconButton size="sm" onClick={handleCloseBtnClick}>
|
|
<Icon.X className="w-5 h-auto" />
|
|
</IconButton>
|
|
</div>
|
|
<div className="dialog-content-container w-full flex flex-col justify-start items-start relative">
|
|
<div className="px-4 pb-3 w-full flex flex-row justify-between items-center space-x-2">
|
|
<div className="flex flex-row justify-start items-center space-x-2">
|
|
<Button color="neutral" variant="outlined" disabled={downloadingImageState.isLoading} onClick={handleDownloadImageBtnClick}>
|
|
{downloadingImageState.isLoading ? (
|
|
<Icon.Loader className="w-4 h-auto mr-1 animate-spin" />
|
|
) : (
|
|
<Icon.Download className="w-4 h-auto mr-1" />
|
|
)}
|
|
{t("common.image")}
|
|
</Button>
|
|
<Button color="neutral" variant="outlined" onClick={handleDownloadTextFileBtnClick}>
|
|
<Icon.File className="w-4 h-auto mr-1" />
|
|
{t("common.file")}
|
|
</Button>
|
|
<Button color="neutral" variant="outlined" onClick={handleCopyLinkBtnClick}>
|
|
<Icon.Link className="w-4 h-auto mr-1" />
|
|
{t("common.link")}
|
|
</Button>
|
|
</div>
|
|
{!readonly && (
|
|
<Select
|
|
className="w-auto text-sm"
|
|
variant="plain"
|
|
value={memo.visibility}
|
|
startDecorator={<VisibilityIcon visibility={memo.visibility} />}
|
|
onChange={(_, visibility) => {
|
|
if (visibility) {
|
|
handleMemoVisibilityOptionChanged(visibility);
|
|
}
|
|
}}
|
|
>
|
|
{[Visibility.PRIVATE, Visibility.PROTECTED, Visibility.PUBLIC].map((item) => (
|
|
<Option key={item} value={item} className="whitespace-nowrap">
|
|
{t(`memo.visibility.${convertVisibilityToString(item).toLowerCase()}` as any)}
|
|
</Option>
|
|
))}
|
|
</Select>
|
|
)}
|
|
</div>
|
|
<div className="w-full border-t dark:border-zinc-700 overflow-clip">
|
|
<div
|
|
className="w-full h-auto select-none relative flex flex-col justify-start items-start bg-white dark:bg-zinc-800"
|
|
ref={memoContainerRef}
|
|
>
|
|
<span className="w-full px-6 pt-5 pb-2 text-sm text-gray-500">{getDateTimeString(memo.displayTime)}</span>
|
|
<div className="w-full px-6 text-base pb-4 space-y-2">
|
|
<MemoContent memoName={memo.name} nodes={memo.nodes} readonly={true} disableFilter />
|
|
<MemoResourceListView resources={memo.resources} />
|
|
</div>
|
|
<div className="flex flex-row justify-between items-center w-full bg-gray-100 dark:bg-zinc-900 py-4 px-6">
|
|
<div className="flex flex-row justify-start items-center">
|
|
<UserAvatar className="mr-2" avatarUrl={user.avatarUrl} />
|
|
<div className="w-auto grow truncate flex mr-2 flex-col justify-center items-start">
|
|
<span className="w-full text truncate font-medium text-gray-600 dark:text-gray-300">
|
|
{user.nickname || user.username}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
<span className="text-gray-500 dark:text-gray-400">via memos</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</>
|
|
);
|
|
};
|
|
|
|
export default function showShareMemoDialog(memoId: number): void {
|
|
generateDialog(
|
|
{
|
|
className: "share-memo-dialog",
|
|
dialogName: "share-memo-dialog",
|
|
},
|
|
ShareMemoDialog,
|
|
{ memoId },
|
|
);
|
|
}
|