// gomuks - A Matrix client written in Go. // Copyright (C) 2024 Tulir Asokan // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . import { useEffect, useMemo } from "react" import { ScaleLoader } from "react-spinners" import Client from "./api/client.ts" import RPCClient from "./api/rpc.ts" import WailsClient from "./api/wailsclient.ts" import WSClient from "./api/wsclient.ts" import ClientContext from "./ui/ClientContext.ts" import MainScreen from "./ui/MainScreen.tsx" import { LoginScreen, VerificationScreen } from "./ui/login" import { LightboxWrapper } from "./ui/modal" import { useEventAsState } from "./util/eventdispatcher.ts" function makeRPCClient(): RPCClient { if (window.wails || window._wails || navigator.userAgent.includes("wails.io")) { return new WailsClient() } return new WSClient("_gomuks/websocket") } function App() { const client = useMemo(() => new Client(makeRPCClient()), []) const connState = useEventAsState(client.rpc.connect) const clientState = useEventAsState(client.state) useEffect(() => { window.client = client return client.start() }, [client]) const afterConnectError = Boolean(connState?.error && connState.reconnecting && clientState?.is_verified) useEffect(() => { if (afterConnectError) { const cancelKeys = (evt: KeyboardEvent | MouseEvent) => evt.stopPropagation() document.body.addEventListener("keydown", cancelKeys, { capture: true }) document.body.addEventListener("keyup", cancelKeys, { capture: true }) document.body.addEventListener("click", cancelKeys, { capture: true }) return () => { document.body.removeEventListener("keydown", cancelKeys, { capture: true }) document.body.removeEventListener("keyup", cancelKeys, { capture: true }) document.body.removeEventListener("click", cancelKeys, { capture: true }) } } }, [afterConnectError]) const errorOverlay = connState?.error ?
{connState.error} 😿
{connState.reconnecting &&
Reconnecting to backend... {connState.nextAttempt ?
(next attempt at {connState.nextAttempt})
: null}
}
: null if (connState?.error && !afterConnectError) { return
{errorOverlay}
} else if ((!connState?.connected && !afterConnectError) || !clientState) { const msg = connState?.connected ? "Waiting for client state..." : "Connecting to backend..." return
{msg}
} else if (!clientState.is_logged_in) { return
} else if (!clientState.is_verified) { return
} else { return {errorOverlay} } } export default App