Skip to content
Snippets Groups Projects
_layout.tsx 4.69 KiB
Newer Older
GyDi's avatar
GyDi committed
import dayjs from "dayjs";
GyDi's avatar
GyDi committed
import i18next from "i18next";
GyDi's avatar
GyDi committed
import relativeTime from "dayjs/plugin/relativeTime";
GyDi's avatar
GyDi committed
import { SWRConfig, mutate } from "swr";
GyDi's avatar
GyDi committed
import { useEffect } from "react";
GyDi's avatar
GyDi committed
import { useTranslation } from "react-i18next";
GyDi's avatar
GyDi committed
import { Route, Routes } from "react-router-dom";
GyDi's avatar
GyDi committed
import { alpha, List, Paper, ThemeProvider } from "@mui/material";
import { listen } from "@tauri-apps/api/event";
GyDi's avatar
GyDi committed
import { appWindow } from "@tauri-apps/api/window";
import { routers } from "./_routers";
GyDi's avatar
GyDi committed
import { getAxios } from "@/services/api";
GyDi's avatar
GyDi committed
import { useVerge } from "@/hooks/use-verge";
GyDi's avatar
GyDi committed
import { ReactComponent as LogoSvg } from "@/assets/image/logo.svg";
import { BaseErrorBoundary, Notice } from "@/components/base";
GyDi's avatar
GyDi committed
import LayoutItem from "@/components/layout/layout-item";
import LayoutControl from "@/components/layout/layout-control";
import LayoutTraffic from "@/components/layout/layout-traffic";
import UpdateButton from "@/components/layout/update-button";
import useCustomTheme from "@/components/layout/use-custom-theme";
import getSystem from "@/utils/get-system";
import "dayjs/locale/ru";
GyDi's avatar
GyDi committed
import "dayjs/locale/zh-cn";
GyDi's avatar
GyDi committed

GyDi's avatar
GyDi committed
dayjs.extend(relativeTime);
GyDi's avatar
GyDi committed

GyDi's avatar
GyDi committed
const OS = getSystem();
GyDi's avatar
GyDi committed

GyDi's avatar
GyDi committed
const Layout = () => {
GyDi's avatar
GyDi committed
  const { t } = useTranslation();
GyDi's avatar
GyDi committed
  const { theme } = useCustomTheme();

GyDi's avatar
GyDi committed
  const { verge } = useVerge();
  const { theme_blur, language } = verge || {};
GyDi's avatar
GyDi committed

GyDi's avatar
GyDi committed
  useEffect(() => {
    window.addEventListener("keydown", (e) => {
      // macOS有cmd+w
      if (e.key === "Escape" && OS !== "macos") {
        appWindow.close();
GyDi's avatar
GyDi committed
      }
GyDi's avatar
GyDi committed
    });
    listen("verge://refresh-clash-config", async () => {
      // the clash info may be updated
      await getAxios(true);
      mutate("getProxies");
GyDi's avatar
GyDi committed
      mutate("getVersion");
      mutate("getClashConfig");
    });

    // update the verge config
    listen("verge://refresh-verge-config", () => mutate("getVergeConfig"));
GyDi's avatar
GyDi committed

GyDi's avatar
GyDi committed
    // 设置提示监听
    listen("verge://notice-message", ({ payload }) => {
      const [status, msg] = payload as [string, string];
      switch (status) {
        case "set_config::ok":
          Notice.success("Refresh clash config");
          break;
        case "set_config::error":
          Notice.error(msg);
          break;
        default:
          break;
      }
    });
GyDi's avatar
GyDi committed
  }, []);

GyDi's avatar
GyDi committed
  useEffect(() => {
GyDi's avatar
GyDi committed
    if (language) {
      dayjs.locale(language === "zh" ? "zh-cn" : language);
      i18next.changeLanguage(language);
GyDi's avatar
GyDi committed
    }
GyDi's avatar
GyDi committed
  }, [language]);
GyDi's avatar
GyDi committed
  return (
GyDi's avatar
GyDi committed
    <SWRConfig value={{ errorRetryCount: 3 }}>
      <ThemeProvider theme={theme}>
GyDi's avatar
GyDi committed
        <Paper
          square
          elevation={0}
GyDi's avatar
GyDi committed
          className={`${OS} layout`}
GyDi's avatar
GyDi committed
          onPointerDown={(e: any) => {
            if (e.target?.dataset?.windrag) appWindow.startDragging();
          }}
          onContextMenu={(e) => {
            // only prevent it on Windows
            const validList = ["input", "textarea"];
            const target = e.currentTarget;
            if (
              OS === "windows" &&
              !(
                validList.includes(target.tagName.toLowerCase()) ||
                target.isContentEditable
              )
            ) {
              e.preventDefault();
            }
GyDi's avatar
GyDi committed
          sx={[
GyDi's avatar
GyDi committed
            ({ palette }) => ({
GyDi's avatar
GyDi committed
              bgcolor: alpha(palette.background.paper, theme_blur ? 0.8 : 1),
GyDi's avatar
GyDi committed
            }),
          ]}
        >
GyDi's avatar
GyDi committed
          <div className="layout__left" data-windrag>
            <div className="the-logo" data-windrag>
GyDi's avatar
GyDi committed
              <LogoSvg />
              {!(OS === "windows" && WIN_PORTABLE) && (
                <UpdateButton className="the-newbtn" />
              )}
GyDi's avatar
GyDi committed

GyDi's avatar
GyDi committed
            <List className="the-menu">
              {routers.map((router) => (
GyDi's avatar
GyDi committed
                <LayoutItem key={router.label} to={router.link}>
GyDi's avatar
GyDi committed
                  {t(router.label)}
GyDi's avatar
GyDi committed
                </LayoutItem>
GyDi's avatar
GyDi committed

GyDi's avatar
GyDi committed
            <div className="the-traffic" data-windrag>
GyDi's avatar
GyDi committed
              <LayoutTraffic />
GyDi's avatar
GyDi committed
          </div>
GyDi's avatar
GyDi committed

GyDi's avatar
GyDi committed
          <div className="layout__right" data-windrag>
            {OS === "windows" && (
GyDi's avatar
GyDi committed
              <div className="the-bar">
                <LayoutControl />
              </div>
            )}
GyDi's avatar
GyDi committed
            <div className="the-content">
GyDi's avatar
GyDi committed
              <Routes>
                {routers.map(({ label, link, ele: Ele }) => (
                  <Route
                    key={label}
                    path={link}
                    element={
                      <BaseErrorBoundary key={label}>
                        <Ele />
                      </BaseErrorBoundary>
                    }
                  />
                ))}
              </Routes>
GyDi's avatar
GyDi committed
            </div>
          </div>
        </Paper>
      </ThemeProvider>
    </SWRConfig>
GyDi's avatar
GyDi committed
  );
};

export default Layout;