import useSWR from "swr"; import snarkdown from "snarkdown"; import { forwardRef, useImperativeHandle, useState, useMemo } from "react"; import { useLockFn } from "ahooks"; import { Box, styled } from "@mui/material"; import { useRecoilState } from "recoil"; import { useTranslation } from "react-i18next"; import { relaunch } from "@tauri-apps/api/process"; import { checkUpdate, installUpdate } from "@tauri-apps/api/updater"; import { BaseDialog, DialogRef, Notice } from "@/components/base"; import { atomUpdateState } from "@/services/states"; const UpdateLog = styled(Box)(() => ({ "h1,h2,h3,ul,ol,p": { margin: "0.5em 0", color: "inherit" }, })); export const UpdateViewer = forwardRef((props, ref) => { const { t } = useTranslation(); const [open, setOpen] = useState(false); const [updateState, setUpdateState] = useRecoilState(atomUpdateState); const { data: updateInfo } = useSWR("checkUpdate", checkUpdate, { errorRetryCount: 2, revalidateIfStale: false, focusThrottleInterval: 36e5, // 1 hour }); useImperativeHandle(ref, () => ({ open: () => setOpen(true), close: () => setOpen(false), })); // markdown parser const parseContent = useMemo(() => { if (!updateInfo?.manifest?.body) { return "New Version is available"; } return snarkdown(updateInfo?.manifest?.body); }, [updateInfo]); const onUpdate = useLockFn(async () => { if (updateState) return; setUpdateState(true); try { await installUpdate(); await relaunch(); } catch (err: any) { Notice.error(err?.message || err.toString()); } finally { setUpdateState(false); } }); return ( setOpen(false)} onCancel={() => setOpen(false)} onOk={onUpdate} > ); });