setting-clash.tsx 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. import { useRef } from "react";
  2. import { useTranslation } from "react-i18next";
  3. import { useLockFn } from "ahooks";
  4. import {
  5. TextField,
  6. Select,
  7. MenuItem,
  8. Typography,
  9. IconButton,
  10. Tooltip,
  11. } from "@mui/material";
  12. import { InfoRounded, Settings, Shuffle } from "@mui/icons-material";
  13. import { DialogRef, Notice, Switch } from "@/components/base";
  14. import { useClash } from "@/hooks/use-clash";
  15. import { GuardState } from "./mods/guard-state";
  16. import { WebUIViewer } from "./mods/web-ui-viewer";
  17. import { ClashPortViewer } from "./mods/clash-port-viewer";
  18. import { ControllerViewer } from "./mods/controller-viewer";
  19. import { SettingList, SettingItem } from "./mods/setting-comp";
  20. import { ClashCoreViewer } from "./mods/clash-core-viewer";
  21. import { invoke_uwp_tool } from "@/services/cmds";
  22. import getSystem from "@/utils/get-system";
  23. import { useVerge } from "@/hooks/use-verge";
  24. import { updateGeoData } from "@/services/api";
  25. const isWIN = getSystem() === "windows";
  26. interface Props {
  27. onError: (err: Error) => void;
  28. }
  29. const SettingClash = ({ onError }: Props) => {
  30. const { t } = useTranslation();
  31. const { clash, version, mutateClash, patchClash } = useClash();
  32. const { verge, mutateVerge, patchVerge } = useVerge();
  33. const { ipv6, "allow-lan": allowLan, "log-level": logLevel } = clash ?? {};
  34. const { enable_random_port = false, verge_mixed_port } = verge ?? {};
  35. const webRef = useRef<DialogRef>(null);
  36. const portRef = useRef<DialogRef>(null);
  37. const ctrlRef = useRef<DialogRef>(null);
  38. const coreRef = useRef<DialogRef>(null);
  39. const onSwitchFormat = (_e: any, value: boolean) => value;
  40. const onChangeData = (patch: Partial<IConfigData>) => {
  41. mutateClash((old) => ({ ...(old! || {}), ...patch }), false);
  42. };
  43. const onChangeVerge = (patch: Partial<IVergeConfig>) => {
  44. mutateVerge({ ...verge, ...patch }, false);
  45. };
  46. const onUpdateGeo = async () => {
  47. try {
  48. await updateGeoData();
  49. Notice.success(t("GeoData Updated"));
  50. } catch (err: any) {
  51. Notice.error(err?.response.data.message || err.toString());
  52. }
  53. };
  54. return (
  55. <SettingList title={t("Clash Setting")}>
  56. <WebUIViewer ref={webRef} />
  57. <ClashPortViewer ref={portRef} />
  58. <ControllerViewer ref={ctrlRef} />
  59. <ClashCoreViewer ref={coreRef} />
  60. <SettingItem label={t("Allow Lan")}>
  61. <GuardState
  62. value={allowLan ?? false}
  63. valueProps="checked"
  64. onCatch={onError}
  65. onFormat={onSwitchFormat}
  66. onChange={(e) => onChangeData({ "allow-lan": e })}
  67. onGuard={(e) => patchClash({ "allow-lan": e })}
  68. >
  69. <Switch edge="end" />
  70. </GuardState>
  71. </SettingItem>
  72. <SettingItem label={t("IPv6")}>
  73. <GuardState
  74. value={ipv6 ?? false}
  75. valueProps="checked"
  76. onCatch={onError}
  77. onFormat={onSwitchFormat}
  78. onChange={(e) => onChangeData({ ipv6: e })}
  79. onGuard={(e) => patchClash({ ipv6: e })}
  80. >
  81. <Switch edge="end" />
  82. </GuardState>
  83. </SettingItem>
  84. <SettingItem label={t("Log Level")}>
  85. <GuardState
  86. // clash premium 2022.08.26 值为warn
  87. value={logLevel === "warn" ? "warning" : logLevel ?? "info"}
  88. onCatch={onError}
  89. onFormat={(e: any) => e.target.value}
  90. onChange={(e) => onChangeData({ "log-level": e })}
  91. onGuard={(e) => patchClash({ "log-level": e })}
  92. >
  93. <Select size="small" sx={{ width: 100, "> div": { py: "7.5px" } }}>
  94. <MenuItem value="debug">Debug</MenuItem>
  95. <MenuItem value="info">Info</MenuItem>
  96. <MenuItem value="warning">Warn</MenuItem>
  97. <MenuItem value="error">Error</MenuItem>
  98. <MenuItem value="silent">Silent</MenuItem>
  99. </Select>
  100. </GuardState>
  101. </SettingItem>
  102. <SettingItem
  103. label={t("Port Config")}
  104. extra={
  105. <Tooltip title={t("Random Port")}>
  106. <IconButton
  107. color={enable_random_port ? "primary" : "inherit"}
  108. size="small"
  109. onClick={() => {
  110. Notice.success(
  111. t("Restart Application to Apply Modifications"),
  112. 1000
  113. );
  114. onChangeVerge({ enable_random_port: !enable_random_port });
  115. patchVerge({ enable_random_port: !enable_random_port });
  116. }}
  117. >
  118. <Shuffle
  119. fontSize="inherit"
  120. style={{ cursor: "pointer", opacity: 0.75 }}
  121. />
  122. </IconButton>
  123. </Tooltip>
  124. }
  125. >
  126. <TextField
  127. disabled={enable_random_port}
  128. autoComplete="off"
  129. size="small"
  130. value={verge_mixed_port ?? 7897}
  131. sx={{ width: 100, input: { py: "7.5px", cursor: "pointer" } }}
  132. onClick={(e) => {
  133. portRef.current?.open();
  134. (e.target as any).blur();
  135. }}
  136. />
  137. </SettingItem>
  138. <SettingItem
  139. onClick={() => ctrlRef.current?.open()}
  140. label={t("External")}
  141. />
  142. <SettingItem onClick={() => webRef.current?.open()} label={t("Web UI")} />
  143. <SettingItem
  144. label={t("Clash Core")}
  145. extra={
  146. <IconButton
  147. color="inherit"
  148. size="small"
  149. onClick={() => coreRef.current?.open()}
  150. >
  151. <Settings
  152. fontSize="inherit"
  153. style={{ cursor: "pointer", opacity: 0.75 }}
  154. />
  155. </IconButton>
  156. }
  157. >
  158. <Typography sx={{ py: "7px", pr: 1 }}>{version}</Typography>
  159. </SettingItem>
  160. {isWIN && (
  161. <SettingItem
  162. onClick={invoke_uwp_tool}
  163. label={t("Open UWP tool")}
  164. extra={
  165. <>
  166. <Tooltip title={t("Open UWP tool Info")} placement="top">
  167. <IconButton color="inherit" size="small">
  168. <InfoRounded
  169. fontSize="inherit"
  170. style={{ cursor: "pointer", opacity: 0.75 }}
  171. />
  172. </IconButton>
  173. </Tooltip>
  174. </>
  175. }
  176. />
  177. )}
  178. <SettingItem onClick={onUpdateGeo} label={t("Update GeoData")} />
  179. </SettingList>
  180. );
  181. };
  182. export default SettingClash;