123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138 |
- import { useEffect, useRef, useState } from "react";
- import { Box, Typography } from "@mui/material";
- import {
- ArrowDownward,
- ArrowUpward,
- MemoryOutlined,
- } from "@mui/icons-material";
- import { useClashInfo } from "@/hooks/use-clash";
- import { useVerge } from "@/hooks/use-verge";
- import { TrafficGraph, type TrafficRef } from "./traffic-graph";
- import { useLogSetup } from "./use-log-setup";
- import { useVisibility } from "@/hooks/use-visibility";
- import { useWebsocket } from "@/hooks/use-websocket";
- import parseTraffic from "@/utils/parse-traffic";
- // setup the traffic
- export const LayoutTraffic = () => {
- const { clashInfo } = useClashInfo();
- const { verge } = useVerge();
- // whether hide traffic graph
- const trafficGraph = verge?.traffic_graph ?? true;
- const trafficRef = useRef<TrafficRef>(null);
- const [traffic, setTraffic] = useState({ up: 0, down: 0 });
- const [memory, setMemory] = useState({ inuse: 0 });
- const pageVisible = useVisibility();
- // setup log ws during layout
- useLogSetup();
- const trafficWs = useWebsocket(
- (event) => {
- const data = JSON.parse(event.data) as ITrafficItem;
- trafficRef.current?.appendData(data);
- setTraffic(data);
- },
- { onError: () => setTraffic({ up: 0, down: 0 }), errorCount: 10 }
- );
- useEffect(() => {
- if (!clashInfo || !pageVisible) return;
- const { server = "", secret = "" } = clashInfo;
- trafficWs.connect(
- `ws://${server}/traffic?token=${encodeURIComponent(secret)}`
- );
- return () => trafficWs.disconnect();
- }, [clashInfo, pageVisible]);
- /* --------- meta memory information --------- */
- const isMetaCore = verge?.clash_core?.includes("clash-meta");
- const displayMemory = isMetaCore && (verge?.enable_memory_usage ?? true);
- const memoryWs = useWebsocket(
- (event) => {
- setMemory(JSON.parse(event.data));
- },
- { onError: () => setMemory({ inuse: 0 }), errorCount: 10 }
- );
- useEffect(() => {
- if (!clashInfo || !pageVisible || !displayMemory) return;
- const { server = "", secret = "" } = clashInfo;
- memoryWs.connect(
- `ws://${server}/memory?token=${encodeURIComponent(secret)}`
- );
- return () => memoryWs.disconnect();
- }, [clashInfo, pageVisible, displayMemory]);
- const [up, upUnit] = parseTraffic(traffic.up);
- const [down, downUnit] = parseTraffic(traffic.down);
- const [inuse, inuseUnit] = parseTraffic(memory.inuse);
- const iconStyle: any = {
- sx: { mr: "8px", fontSize: 16 },
- };
- const valStyle: any = {
- component: "span",
- // color: "primary",
- textAlign: "center",
- sx: { flex: "1 1 56px", userSelect: "none" },
- };
- const unitStyle: any = {
- component: "span",
- color: "grey.500",
- fontSize: "12px",
- textAlign: "right",
- sx: { flex: "0 1 27px", userSelect: "none" },
- };
- return (
- <Box position="relative" onClick={trafficRef.current?.toggleStyle}>
- {trafficGraph && pageVisible && (
- <div style={{ width: "100%", height: 60, marginBottom: 6 }}>
- <TrafficGraph ref={trafficRef} />
- </div>
- )}
- <Box display="flex" flexDirection="column" gap={0.75}>
- <Box display="flex" alignItems="center" whiteSpace="nowrap">
- <ArrowUpward
- {...iconStyle}
- color={+up > 0 ? "secondary" : "disabled"}
- />
- <Typography {...valStyle} color="secondary">
- {up}
- </Typography>
- <Typography {...unitStyle}>{upUnit}/s</Typography>
- </Box>
- <Box display="flex" alignItems="center" whiteSpace="nowrap">
- <ArrowDownward
- {...iconStyle}
- color={+down > 0 ? "primary" : "disabled"}
- />
- <Typography {...valStyle} color="primary">
- {down}
- </Typography>
- <Typography {...unitStyle}>{downUnit}/s</Typography>
- </Box>
- {displayMemory && (
- <Box
- display="flex"
- alignItems="center"
- whiteSpace="nowrap"
- title="Memory Usage"
- >
- <MemoryOutlined {...iconStyle} color="disabled" />
- <Typography {...valStyle}>{inuse}</Typography>
- <Typography {...unitStyle}>{inuseUnit}</Typography>
- </Box>
- )}
- </Box>
- </Box>
- );
- };
|