mirror of
https://github.com/clash-verge-rev/clash-verge-rev.git
synced 2026-01-28 16:30:52 +08:00
110 lines
2.5 KiB
TypeScript
110 lines
2.5 KiB
TypeScript
import { useCallback, useEffect, useReducer } from "react";
|
|
|
|
import { useProfiles } from "@/hooks/use-profiles";
|
|
|
|
import { ProxySortType } from "./use-filter-sort";
|
|
|
|
export interface HeadState {
|
|
open?: boolean;
|
|
showType: boolean;
|
|
sortType: ProxySortType;
|
|
filterText: string;
|
|
textState: "url" | "filter" | null;
|
|
testUrl: string;
|
|
}
|
|
|
|
type HeadStateStorage = Record<string, Record<string, HeadState>>;
|
|
|
|
const HEAD_STATE_KEY = "proxy-head-state";
|
|
export const DEFAULT_STATE: HeadState = {
|
|
open: false,
|
|
showType: true,
|
|
sortType: 0,
|
|
filterText: "",
|
|
textState: null,
|
|
testUrl: "",
|
|
};
|
|
|
|
type HeadStateAction =
|
|
| { type: "reset" }
|
|
| { type: "replace"; payload: Record<string, HeadState> }
|
|
| { type: "update"; groupName: string; patch: Partial<HeadState> };
|
|
|
|
function headStateReducer(
|
|
state: Record<string, HeadState>,
|
|
action: HeadStateAction,
|
|
): Record<string, HeadState> {
|
|
switch (action.type) {
|
|
case "reset":
|
|
return {};
|
|
case "replace":
|
|
return action.payload;
|
|
case "update": {
|
|
const prev = state[action.groupName] || DEFAULT_STATE;
|
|
return { ...state, [action.groupName]: { ...prev, ...action.patch } };
|
|
}
|
|
default:
|
|
return state;
|
|
}
|
|
}
|
|
|
|
export function useHeadStateNew() {
|
|
const { profiles } = useProfiles();
|
|
const current = profiles?.current || "";
|
|
|
|
const [state, dispatch] = useReducer(headStateReducer, {});
|
|
|
|
useEffect(() => {
|
|
if (!current) {
|
|
dispatch({ type: "reset" });
|
|
return;
|
|
}
|
|
|
|
try {
|
|
const data = JSON.parse(
|
|
localStorage.getItem(HEAD_STATE_KEY)!,
|
|
) as HeadStateStorage;
|
|
|
|
const value = data[current] || {};
|
|
|
|
if (value && typeof value === "object") {
|
|
dispatch({ type: "replace", payload: value });
|
|
} else {
|
|
dispatch({ type: "reset" });
|
|
}
|
|
} catch {
|
|
dispatch({ type: "reset" });
|
|
}
|
|
}, [current]);
|
|
|
|
useEffect(() => {
|
|
if (!current) return;
|
|
|
|
const timer = setTimeout(() => {
|
|
try {
|
|
const item = localStorage.getItem(HEAD_STATE_KEY);
|
|
|
|
let data = (item ? JSON.parse(item) : {}) as HeadStateStorage;
|
|
|
|
if (!data || typeof data !== "object") data = {};
|
|
|
|
data[current] = state;
|
|
|
|
localStorage.setItem(HEAD_STATE_KEY, JSON.stringify(data));
|
|
} catch {}
|
|
});
|
|
|
|
return () => clearTimeout(timer);
|
|
}, [state, current]);
|
|
|
|
const setHeadState = useCallback(
|
|
(groupName: string, obj: Partial<HeadState>) => {
|
|
if (!current) return;
|
|
dispatch({ type: "update", groupName, patch: obj });
|
|
},
|
|
[current],
|
|
);
|
|
|
|
return [state, setHeadState] as const;
|
|
}
|