import useSWR from "swr"; import snarkdown from "snarkdown"; import { forwardRef, useImperativeHandle, useState, useMemo } from "react"; import { useLockFn } from "ahooks"; import { Box, LinearProgress, 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"; import { listen, Event, UnlistenFn } from "@tauri-apps/api/event"; import { portableFlag } from "@/pages/_layout"; const UpdateLog = styled(Box)(() => ({ "h1,h2,h3,ul,ol,p": { margin: "0.5em 0", color: "inherit" }, })); let eventListener: UnlistenFn | null = null; 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 }); const [downloaded, setDownloaded] = useState(0); const [buffer, setBuffer] = useState(0); const [total, setTotal] = useState(0); 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 (portableFlag) { Notice.error(t("Portable Updater Error")); return; } if (updateState) return; setUpdateState(true); if (eventListener !== null) { eventListener(); } eventListener = await listen( "tauri://update-download-progress", (e: Event) => { setTotal(e.payload.contentLength); setBuffer(e.payload.chunkLength); setDownloaded((a) => { return a + e.payload.chunkLength; }); } ); 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} > {updateState && ( )} ); });