mirror of
https://github.com/streamyfin/streamyfin.git
synced 2026-01-15 23:59:08 +00:00
Some checks failed
🕒 Handle Stale Issues / 🗑️ Cleanup Stale Issues (push) Has been cancelled
🛡️ CodeQL Analysis / 🔎 Analyze with CodeQL (actions) (push) Has been cancelled
🛡️ CodeQL Analysis / 🔎 Analyze with CodeQL (javascript-typescript) (push) Has been cancelled
🔒 Lockfile Consistency Check / 🔍 Check bun.lock and package.json consistency (push) Has been cancelled
🏗️ Build Apps / 🤖 Build Android APK (Phone) (push) Has been cancelled
🏗️ Build Apps / 🤖 Build Android APK (TV) (push) Has been cancelled
🏗️ Build Apps / 🍎 Build iOS IPA (Phone) (push) Has been cancelled
🏷️🔀Merge Conflict Labeler / 🏷️ Labeling Merge Conflicts (push) Has been cancelled
🚦 Security & Quality Gate / 📝 Validate PR Title (push) Has been cancelled
🚦 Security & Quality Gate / 🔍 Vulnerable Dependencies (push) Has been cancelled
🚦 Security & Quality Gate / 🚑 Expo Doctor Check (push) Has been cancelled
🚦 Security & Quality Gate / 🔍 Lint & Test (check) (push) Has been cancelled
🚦 Security & Quality Gate / 🔍 Lint & Test (format) (push) Has been cancelled
🚦 Security & Quality Gate / 🔍 Lint & Test (lint) (push) Has been cancelled
🚦 Security & Quality Gate / 🔍 Lint & Test (typecheck) (push) Has been cancelled
93 lines
2.3 KiB
TypeScript
93 lines
2.3 KiB
TypeScript
import NetInfo from "@react-native-community/netinfo";
|
|
import { useAtom } from "jotai";
|
|
import {
|
|
createContext,
|
|
type ReactNode,
|
|
useCallback,
|
|
useContext,
|
|
useEffect,
|
|
useState,
|
|
} from "react";
|
|
import { apiAtom } from "@/providers/JellyfinProvider";
|
|
|
|
interface NetworkStatusContextType {
|
|
isConnected: boolean;
|
|
serverConnected: boolean | null;
|
|
loading: boolean;
|
|
retryCheck: () => Promise<void>;
|
|
}
|
|
|
|
const NetworkStatusContext = createContext<NetworkStatusContextType | null>(
|
|
null,
|
|
);
|
|
|
|
async function checkApiReachable(basePath?: string): Promise<boolean> {
|
|
if (!basePath) return false;
|
|
try {
|
|
const response = await fetch(basePath, { method: "HEAD" });
|
|
return response.ok;
|
|
} catch {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
export function NetworkStatusProvider({ children }: { children: ReactNode }) {
|
|
const [isConnected, setIsConnected] = useState(false);
|
|
const [serverConnected, setServerConnected] = useState<boolean | null>(null);
|
|
const [loading, setLoading] = useState(false);
|
|
const [api] = useAtom(apiAtom);
|
|
|
|
const validateConnection = useCallback(async () => {
|
|
if (!api?.basePath) return false;
|
|
const reachable = await checkApiReachable(api.basePath);
|
|
setServerConnected(reachable);
|
|
return reachable;
|
|
}, [api?.basePath]);
|
|
|
|
const retryCheck = useCallback(async () => {
|
|
setLoading(true);
|
|
await validateConnection();
|
|
setLoading(false);
|
|
}, [validateConnection]);
|
|
|
|
useEffect(() => {
|
|
const unsubscribe = NetInfo.addEventListener(async (state) => {
|
|
setIsConnected(!!state.isConnected);
|
|
if (state.isConnected) {
|
|
await validateConnection();
|
|
} else {
|
|
setServerConnected(false);
|
|
}
|
|
});
|
|
|
|
// Initial check
|
|
NetInfo.fetch().then((state) => {
|
|
if (state.isConnected) {
|
|
validateConnection();
|
|
} else {
|
|
setServerConnected(false);
|
|
}
|
|
});
|
|
|
|
return () => unsubscribe();
|
|
}, [validateConnection]);
|
|
|
|
return (
|
|
<NetworkStatusContext.Provider
|
|
value={{ isConnected, serverConnected, loading, retryCheck }}
|
|
>
|
|
{children}
|
|
</NetworkStatusContext.Provider>
|
|
);
|
|
}
|
|
|
|
export function useNetworkStatus(): NetworkStatusContextType {
|
|
const context = useContext(NetworkStatusContext);
|
|
if (!context) {
|
|
throw new Error(
|
|
"useNetworkStatus must be used within NetworkStatusProvider",
|
|
);
|
|
}
|
|
return context;
|
|
}
|