mirror of
https://github.com/netbirdio/netbird.git
synced 2026-05-20 15:49:55 +00:00
add window manager
This commit is contained in:
@@ -10,7 +10,7 @@ import * as ProfileSwitcher from "./profileswitcher.js";
|
|||||||
import * as Profiles from "./profiles.js";
|
import * as Profiles from "./profiles.js";
|
||||||
import * as Settings from "./settings.js";
|
import * as Settings from "./settings.js";
|
||||||
import * as Update from "./update.js";
|
import * as Update from "./update.js";
|
||||||
import * as Windows from "./windows.js";
|
import * as WindowManager from "./windowmanager.js";
|
||||||
export {
|
export {
|
||||||
Connection,
|
Connection,
|
||||||
Debug,
|
Debug,
|
||||||
@@ -21,7 +21,7 @@ export {
|
|||||||
Profiles,
|
Profiles,
|
||||||
Settings,
|
Settings,
|
||||||
Update,
|
Update,
|
||||||
Windows
|
WindowManager
|
||||||
};
|
};
|
||||||
|
|
||||||
export {
|
export {
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ import { Settings } from "@/modules/settings/Settings.tsx";
|
|||||||
import { SkeletonTheme } from "react-loading-skeleton";
|
import { SkeletonTheme } from "react-loading-skeleton";
|
||||||
import "react-loading-skeleton/dist/skeleton.css";
|
import "react-loading-skeleton/dist/skeleton.css";
|
||||||
import { welcome } from "@/lib/welcome";
|
import { welcome } from "@/lib/welcome";
|
||||||
import Login from "@/pages/Login.tsx";
|
import BrowserLogin from "@/pages/BrowserLogin.tsx";
|
||||||
|
|
||||||
welcome();
|
welcome();
|
||||||
|
|
||||||
@@ -22,7 +22,7 @@ ReactDOM.createRoot(document.getElementById("root") as HTMLElement).render(
|
|||||||
<HashRouter>
|
<HashRouter>
|
||||||
<Routes>
|
<Routes>
|
||||||
<Route path="/quick" element={<QuickActions />} />
|
<Route path="/quick" element={<QuickActions />} />
|
||||||
<Route path="/login" element={<Login />} />
|
<Route path="/browser-login" element={<BrowserLogin />} />
|
||||||
<Route path="/update" element={<Update />} />
|
<Route path="/update" element={<Update />} />
|
||||||
<Route path="/session-expired" element={<SessionExpired />} />
|
<Route path="/session-expired" element={<SessionExpired />} />
|
||||||
<Route element={<SettingsLayout />}>
|
<Route element={<SettingsLayout />}>
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
import { useMemo, useState } from "react";
|
import { useMemo, useState } from "react";
|
||||||
import { useNavigate } from "react-router-dom";
|
import { Dialogs, Events } from "@wailsio/runtime";
|
||||||
import { Dialogs } from "@wailsio/runtime";
|
import { Connection, WindowManager } from "@bindings/services";
|
||||||
import { Connection } from "@bindings/services";
|
|
||||||
import { ConnectionState } from "@/components/NetBirdConnectToggle.tsx";
|
import { ConnectionState } from "@/components/NetBirdConnectToggle.tsx";
|
||||||
import { ToggleSwitch } from "@/components/ToggleSwitch.tsx";
|
import { ToggleSwitch } from "@/components/ToggleSwitch.tsx";
|
||||||
import { useStatus } from "@/hooks/useStatus";
|
import { useStatus } from "@/hooks/useStatus";
|
||||||
@@ -16,13 +15,87 @@ const STATUS_LABEL: Record<ConnectionState, string> = {
|
|||||||
[ConnectionState.Disconnecting]: "Disconnecting...",
|
[ConnectionState.Disconnecting]: "Disconnecting...",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const EVENT_BROWSER_LOGIN_CANCEL = "browser-login:cancel";
|
||||||
|
|
||||||
const errorMessage = (e: unknown) =>
|
const errorMessage = (e: unknown) =>
|
||||||
e instanceof Error ? e.message : String(e);
|
e instanceof Error ? e.message : String(e);
|
||||||
|
|
||||||
|
// startLogin drives the daemon's SSO login end-to-end. The BrowserLogin
|
||||||
|
// popup window is the only login UI; errors surface as a native
|
||||||
|
// Dialogs.Error. Concurrent calls are dropped via the inFlight guard.
|
||||||
|
let loginInFlight = false;
|
||||||
|
async function startLogin(): Promise<void> {
|
||||||
|
if (loginInFlight) return;
|
||||||
|
loginInFlight = true;
|
||||||
|
|
||||||
|
let cancelled = false;
|
||||||
|
let offCancel: (() => void) | undefined;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = await Connection.Login({
|
||||||
|
profileName: "",
|
||||||
|
username: "",
|
||||||
|
managementUrl: "",
|
||||||
|
setupKey: "",
|
||||||
|
preSharedKey: "",
|
||||||
|
hostname: "",
|
||||||
|
hint: "",
|
||||||
|
});
|
||||||
|
|
||||||
|
if (result.needsSsoLogin) {
|
||||||
|
const uri = result.verificationUriComplete || result.verificationUri;
|
||||||
|
if (uri) {
|
||||||
|
Connection.OpenURL(uri).catch(console.error);
|
||||||
|
WindowManager.OpenBrowserLogin(uri).catch(console.error);
|
||||||
|
}
|
||||||
|
|
||||||
|
const cancelPromise = new Promise<void>((resolve) => {
|
||||||
|
offCancel = Events.On(EVENT_BROWSER_LOGIN_CANCEL, () => {
|
||||||
|
cancelled = true;
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const waitPromise = Connection.WaitSSOLogin({
|
||||||
|
userCode: result.userCode,
|
||||||
|
hostname: "",
|
||||||
|
});
|
||||||
|
|
||||||
|
try {
|
||||||
|
await Promise.race([waitPromise, cancelPromise]);
|
||||||
|
} finally {
|
||||||
|
WindowManager.CloseBrowserLogin().catch(console.error);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cancelled) {
|
||||||
|
// Tell the daemon to drop the in-flight WaitSSOLogin so a
|
||||||
|
// future Login starts fresh; see services/connection.go:74.
|
||||||
|
try {
|
||||||
|
await Connection.Down();
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
await Connection.Up({ profileName: "", username: "" });
|
||||||
|
} catch (e) {
|
||||||
|
WindowManager.CloseBrowserLogin().catch(console.error);
|
||||||
|
if (cancelled) return;
|
||||||
|
await Dialogs.Error({
|
||||||
|
Title: "Login Failed",
|
||||||
|
Message: errorMessage(e),
|
||||||
|
});
|
||||||
|
} finally {
|
||||||
|
offCancel?.();
|
||||||
|
loginInFlight = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export const ConnectionStatusSwitch = () => {
|
export const ConnectionStatusSwitch = () => {
|
||||||
const { status, refresh } = useStatus();
|
const { status, refresh } = useStatus();
|
||||||
const { activeProfile, username } = useProfile();
|
const { activeProfile, username } = useProfile();
|
||||||
const navigate = useNavigate();
|
|
||||||
|
|
||||||
const daemonState = status?.status ?? "Idle";
|
const daemonState = status?.status ?? "Idle";
|
||||||
const needsLogin =
|
const needsLogin =
|
||||||
@@ -89,7 +162,7 @@ export const ConnectionStatusSwitch = () => {
|
|||||||
const handleSwitch = (next: boolean) => {
|
const handleSwitch = (next: boolean) => {
|
||||||
if (unreachable || action !== null) return;
|
if (unreachable || action !== null) return;
|
||||||
if (needsLogin) {
|
if (needsLogin) {
|
||||||
navigate("/login");
|
void startLogin().finally(() => refresh());
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (next && connState === ConnectionState.Disconnected) {
|
if (next && connState === ConnectionState.Disconnected) {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { useEffect, useRef } from "react";
|
import { useEffect, useRef } from "react";
|
||||||
import { PanelRightCloseIcon, PanelRightOpenIcon, SettingsIcon } from "lucide-react";
|
import { PanelRightCloseIcon, PanelRightOpenIcon, SettingsIcon } from "lucide-react";
|
||||||
import { Window } from "@wailsio/runtime";
|
import { Window } from "@wailsio/runtime";
|
||||||
import { Windows as WindowsSvc } from "@bindings/services";
|
import { WindowManager } from "@bindings/services";
|
||||||
import { ProfileSelector } from "@/components/ProfileSelector.tsx";
|
import { ProfileSelector } from "@/components/ProfileSelector.tsx";
|
||||||
import { IconButton } from "@/components/IconButton.tsx";
|
import { IconButton } from "@/components/IconButton.tsx";
|
||||||
import { useAppearance } from "@/modules/appearance/AppearanceContext.tsx";
|
import { useAppearance } from "@/modules/appearance/AppearanceContext.tsx";
|
||||||
@@ -47,7 +47,7 @@ export const Header = () => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const openSettings = () => {
|
const openSettings = () => {
|
||||||
void WindowsSvc.OpenSettings().catch(() => {});
|
void WindowManager.OpenSettings().catch(() => {});
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
54
client/ui/frontend/src/pages/BrowserLogin.tsx
Normal file
54
client/ui/frontend/src/pages/BrowserLogin.tsx
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
import { useCallback } from "react";
|
||||||
|
import { useSearchParams } from "react-router-dom";
|
||||||
|
import { Events } from "@wailsio/runtime";
|
||||||
|
import { Loader2 } from "lucide-react";
|
||||||
|
import { Connection } from "@bindings/services";
|
||||||
|
import { Button } from "../components/Button";
|
||||||
|
import netbirdFull from "@/assets/logos/netbird-full.svg";
|
||||||
|
|
||||||
|
const EVENT_CANCEL = "browser-login:cancel";
|
||||||
|
|
||||||
|
export default function BrowserLogin() {
|
||||||
|
const [params] = useSearchParams();
|
||||||
|
const uri = params.get("uri") ?? "";
|
||||||
|
|
||||||
|
const tryAgain = useCallback(() => {
|
||||||
|
if (!uri) return;
|
||||||
|
Connection.OpenURL(uri).catch(console.error);
|
||||||
|
}, [uri]);
|
||||||
|
|
||||||
|
const cancel = useCallback(() => {
|
||||||
|
void Events.Emit(EVENT_CANCEL);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex h-screen flex-col items-center justify-center gap-3 p-8 text-center">
|
||||||
|
<img src={netbirdFull} alt="NetBird" className="mb-2 h-9" />
|
||||||
|
<h1 className="text-lg font-semibold text-white">
|
||||||
|
Continue in your browser to complete the login
|
||||||
|
</h1>
|
||||||
|
<p className="max-w-sm text-sm text-nb-gray-400">
|
||||||
|
Please complete the account authentication process in the browser tab
|
||||||
|
and continue from there.
|
||||||
|
</p>
|
||||||
|
<div className="flex items-center gap-2 text-sm text-nb-gray-400">
|
||||||
|
<Loader2 className="h-4 w-4 animate-spin" strokeWidth={1.5} />
|
||||||
|
Waiting for sign-in…
|
||||||
|
</div>
|
||||||
|
<p className="text-sm text-nb-gray-400">
|
||||||
|
Not seeing the browser tab?{" "}
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
onClick={tryAgain}
|
||||||
|
disabled={!uri}
|
||||||
|
className="text-netbird hover:underline disabled:opacity-40 disabled:cursor-not-allowed"
|
||||||
|
>
|
||||||
|
Try again
|
||||||
|
</button>
|
||||||
|
</p>
|
||||||
|
<Button variant="secondary" onClick={cancel} className="mt-2">
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,159 +0,0 @@
|
|||||||
import { useCallback, useEffect, useRef, useState } from "react";
|
|
||||||
import { useNavigate } from "react-router-dom";
|
|
||||||
import { ExternalLink, Loader2, AlertTriangle, X, RotateCcw } from "lucide-react";
|
|
||||||
import { Connection } from "../../bindings/github.com/netbirdio/netbird/client/ui/services";
|
|
||||||
import { Button } from "../components/Button";
|
|
||||||
|
|
||||||
type Phase = "starting" | "browser" | "connecting" | "error";
|
|
||||||
|
|
||||||
export default function Login() {
|
|
||||||
const navigate = useNavigate();
|
|
||||||
const [phase, setPhase] = useState<Phase>("starting");
|
|
||||||
const [verificationUri, setVerificationUri] = useState<string>("");
|
|
||||||
const [errorMsg, setErrorMsg] = useState<string>("");
|
|
||||||
// attempt is bumped every time the user asks for a fresh start, which
|
|
||||||
// re-arms the useEffect below so the daemon's Login RPC is dialed again.
|
|
||||||
const [attempt, setAttempt] = useState(0);
|
|
||||||
const cancelledRef = useRef(false);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
cancelledRef.current = false;
|
|
||||||
setPhase("starting");
|
|
||||||
setVerificationUri("");
|
|
||||||
setErrorMsg("");
|
|
||||||
|
|
||||||
(async () => {
|
|
||||||
try {
|
|
||||||
const result = await Connection.Login({
|
|
||||||
profileName: "",
|
|
||||||
username: "",
|
|
||||||
managementUrl: "",
|
|
||||||
setupKey: "",
|
|
||||||
preSharedKey: "",
|
|
||||||
hostname: "",
|
|
||||||
hint: "",
|
|
||||||
});
|
|
||||||
if (cancelledRef.current) return;
|
|
||||||
|
|
||||||
if (result.needsSsoLogin) {
|
|
||||||
const uri = result.verificationUriComplete || result.verificationUri;
|
|
||||||
setVerificationUri(uri);
|
|
||||||
setPhase("browser");
|
|
||||||
if (uri) Connection.OpenURL(uri).catch(console.error);
|
|
||||||
|
|
||||||
await Connection.WaitSSOLogin({
|
|
||||||
userCode: result.userCode,
|
|
||||||
hostname: "",
|
|
||||||
});
|
|
||||||
if (cancelledRef.current) return;
|
|
||||||
}
|
|
||||||
|
|
||||||
setPhase("connecting");
|
|
||||||
await Connection.Up({ profileName: "", username: "" });
|
|
||||||
if (cancelledRef.current) return;
|
|
||||||
|
|
||||||
navigate("/", { replace: true });
|
|
||||||
} catch (e) {
|
|
||||||
if (cancelledRef.current) return;
|
|
||||||
setErrorMsg(String(e));
|
|
||||||
setPhase("error");
|
|
||||||
}
|
|
||||||
})();
|
|
||||||
|
|
||||||
return () => {
|
|
||||||
cancelledRef.current = true;
|
|
||||||
};
|
|
||||||
}, [navigate, attempt]);
|
|
||||||
|
|
||||||
// restart aborts any in-flight wait by toggling the cancellation flag,
|
|
||||||
// tells the daemon to drop whatever it's holding (a stale WaitSSOLogin
|
|
||||||
// can wedge the daemon for a previous UserCode), and then bumps attempt
|
|
||||||
// so the effect re-runs with a clean slate.
|
|
||||||
const restart = useCallback(async () => {
|
|
||||||
cancelledRef.current = true;
|
|
||||||
try {
|
|
||||||
await Connection.Down();
|
|
||||||
} catch (e) {
|
|
||||||
console.error(e);
|
|
||||||
}
|
|
||||||
setAttempt((n) => n + 1);
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
// Cancel must also tell the daemon to abandon the in-flight WaitSSOLogin.
|
|
||||||
// Without Down(), the daemon stays parked on the OAuth flow's UserCode
|
|
||||||
// forever; subsequent Login calls re-use the cached flow but the user has
|
|
||||||
// no way out. Down() triggers the daemon's actCancel(), which unblocks
|
|
||||||
// WaitSSOLogin with a context-canceled error so our promise settles.
|
|
||||||
const cancel = useCallback(async () => {
|
|
||||||
cancelledRef.current = true;
|
|
||||||
try {
|
|
||||||
await Connection.Down();
|
|
||||||
} catch (e) {
|
|
||||||
console.error(e);
|
|
||||||
}
|
|
||||||
navigate("/", { replace: true });
|
|
||||||
}, [navigate]);
|
|
||||||
|
|
||||||
if (phase === "error") {
|
|
||||||
return (
|
|
||||||
<div className="flex h-full flex-col items-center justify-center gap-4 p-6 text-center">
|
|
||||||
<AlertTriangle className="h-8 w-8 text-red-500" strokeWidth={1.5} />
|
|
||||||
<h1 className="text-xl font-semibold">Login failed</h1>
|
|
||||||
<p className="max-w-sm break-words text-sm text-nb-gray-500">{errorMsg}</p>
|
|
||||||
<div className="flex gap-2">
|
|
||||||
<Button onClick={restart}>
|
|
||||||
<RotateCcw className="h-4 w-4" strokeWidth={1.5} /> Try again
|
|
||||||
</Button>
|
|
||||||
<Button variant="secondary" onClick={cancel}>
|
|
||||||
Back
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (phase === "browser") {
|
|
||||||
return (
|
|
||||||
<div className="flex h-full flex-col items-center justify-center gap-4 p-6 text-center">
|
|
||||||
<h1 className="text-xl font-semibold">Continue in your browser</h1>
|
|
||||||
<p className="max-w-sm text-sm text-nb-gray-500">
|
|
||||||
A browser tab should have opened. Sign in there — this window will
|
|
||||||
continue automatically once you're done.
|
|
||||||
</p>
|
|
||||||
{verificationUri && (
|
|
||||||
<Button onClick={() => Connection.OpenURL(verificationUri).catch(console.error)}>
|
|
||||||
<ExternalLink className="h-4 w-4" strokeWidth={1.5} />
|
|
||||||
Reopen URL
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
<p className="max-w-sm break-all font-mono text-xs text-nb-gray-500">
|
|
||||||
{verificationUri}
|
|
||||||
</p>
|
|
||||||
<div className="flex items-center gap-2 text-sm text-nb-gray-500">
|
|
||||||
<Loader2 className="h-4 w-4 animate-spin" strokeWidth={1.5} />
|
|
||||||
Waiting for sign-in…
|
|
||||||
</div>
|
|
||||||
<div className="flex gap-2 pt-2">
|
|
||||||
<Button variant="secondary" onClick={restart}>
|
|
||||||
<RotateCcw className="h-4 w-4" strokeWidth={1.5} /> Restart
|
|
||||||
</Button>
|
|
||||||
<Button variant="ghost" onClick={cancel}>
|
|
||||||
<X className="h-4 w-4" strokeWidth={1.5} /> Cancel
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const message =
|
|
||||||
phase === "connecting" ? "Bringing the connection up…" : "Starting login…";
|
|
||||||
return (
|
|
||||||
<div className="flex h-full flex-col items-center justify-center gap-3 p-6 text-center">
|
|
||||||
<Loader2 className="h-8 w-8 animate-spin text-netbird" strokeWidth={1.5} />
|
|
||||||
<p className="text-sm text-nb-gray-500">{message}</p>
|
|
||||||
<Button variant="ghost" onClick={cancel}>
|
|
||||||
<X className="h-4 w-4" strokeWidth={1.5} /> Cancel
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -152,22 +152,11 @@ func main() {
|
|||||||
window.Hide()
|
window.Hide()
|
||||||
})
|
})
|
||||||
|
|
||||||
// Pre-create the settings window AFTER the main window so the OS treats
|
// The settings and browser-login windows are created lazily and
|
||||||
// the main window as the primary one. The settings window stays hidden
|
// destroyed on close, so they don't linger as hidden windows that
|
||||||
// until the user clicks the Settings icon — preloading it here keeps the
|
// Wails's macOS dock-reopen handler would pop back up.
|
||||||
// first-open instant.
|
windowManager := services.NewWindowManager(app)
|
||||||
windows := services.NewWindows(app)
|
app.RegisterService(application.NewService(windowManager))
|
||||||
|
|
||||||
app.RegisterService(application.NewService(connection))
|
|
||||||
app.RegisterService(application.NewService(settings))
|
|
||||||
app.RegisterService(application.NewService(services.NewNetworks(conn)))
|
|
||||||
app.RegisterService(application.NewService(services.NewForwarding(conn)))
|
|
||||||
app.RegisterService(application.NewService(profiles))
|
|
||||||
app.RegisterService(application.NewService(services.NewDebug(conn)))
|
|
||||||
app.RegisterService(application.NewService(update))
|
|
||||||
app.RegisterService(application.NewService(peers))
|
|
||||||
app.RegisterService(application.NewService(windows))
|
|
||||||
app.RegisterService(application.NewService(notifier))
|
|
||||||
|
|
||||||
// Register an in-process StatusNotifierWatcher so the tray works on
|
// Register an in-process StatusNotifierWatcher so the tray works on
|
||||||
// minimal WMs (Fluxbox, OpenBox, i3, dwm, vanilla GNOME without the
|
// minimal WMs (Fluxbox, OpenBox, i3, dwm, vanilla GNOME without the
|
||||||
|
|||||||
132
client/ui/services/window_manager.go
Normal file
132
client/ui/services/window_manager.go
Normal file
@@ -0,0 +1,132 @@
|
|||||||
|
//go:build !android && !ios && !freebsd && !js
|
||||||
|
|
||||||
|
package services
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/url"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"github.com/wailsapp/wails/v3/pkg/application"
|
||||||
|
"github.com/wailsapp/wails/v3/pkg/events"
|
||||||
|
)
|
||||||
|
|
||||||
|
// EventTriggerLogin asks the frontend's startLogin() orchestrator to begin
|
||||||
|
// an SSO flow. Emitted by the tray (Login menu item, session expired) since
|
||||||
|
// the tray can't call JS directly.
|
||||||
|
const EventTriggerLogin = "trigger-login"
|
||||||
|
|
||||||
|
// EventBrowserLoginCancel is emitted by the BrowserLogin popup window when
|
||||||
|
// the user clicks Cancel or closes the window. startLogin() listens for it
|
||||||
|
// and tears down the daemon's pending SSO wait.
|
||||||
|
const EventBrowserLoginCancel = "browser-login:cancel"
|
||||||
|
|
||||||
|
// WindowManager opens auxiliary application windows on demand from the
|
||||||
|
// frontend. The main window is created up-front in main.go; this service is
|
||||||
|
// for secondary, on-demand surfaces (Settings, BrowserLogin).
|
||||||
|
//
|
||||||
|
// Secondary windows are created on first open and destroyed on close —
|
||||||
|
// the Wails-recommended singleton pattern (see Multiple Windows docs:
|
||||||
|
// "Cleanup on close"). Destroying rather than hiding means the dock-reopen
|
||||||
|
// handler doesn't find a hidden window to resurrect.
|
||||||
|
type WindowManager struct {
|
||||||
|
app *application.App
|
||||||
|
settings *application.WebviewWindow
|
||||||
|
browserLogin *application.WebviewWindow
|
||||||
|
mu sync.Mutex
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewWindowManager(app *application.App) *WindowManager {
|
||||||
|
return &WindowManager{app: app}
|
||||||
|
}
|
||||||
|
|
||||||
|
// OpenSettings shows the settings window, creating it on first use (and
|
||||||
|
// after the user has closed a previous instance).
|
||||||
|
func (s *WindowManager) OpenSettings() {
|
||||||
|
s.mu.Lock()
|
||||||
|
defer s.mu.Unlock()
|
||||||
|
if s.settings == nil {
|
||||||
|
s.settings = s.app.Window.NewWithOptions(application.WebviewWindowOptions{
|
||||||
|
Name: "settings",
|
||||||
|
Title: "NetBird Settings",
|
||||||
|
Width: 900,
|
||||||
|
Height: 640,
|
||||||
|
DisableResize: true,
|
||||||
|
MinimiseButtonState: application.ButtonHidden,
|
||||||
|
MaximiseButtonState: application.ButtonHidden,
|
||||||
|
CloseButtonState: application.ButtonEnabled,
|
||||||
|
BackgroundColour: application.NewRGB(24, 26, 29),
|
||||||
|
URL: "/#/settings",
|
||||||
|
Mac: application.MacWindow{
|
||||||
|
InvisibleTitleBarHeight: 38,
|
||||||
|
Backdrop: application.MacBackdropTranslucent,
|
||||||
|
TitleBar: application.MacTitleBarHiddenInset,
|
||||||
|
CollectionBehavior: application.MacWindowCollectionBehaviorFullScreenNone,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
s.settings.OnWindowEvent(events.Common.WindowClosing, func(_ *application.WindowEvent) {
|
||||||
|
s.mu.Lock()
|
||||||
|
s.settings = nil
|
||||||
|
s.mu.Unlock()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
s.settings.Show()
|
||||||
|
s.settings.Focus()
|
||||||
|
}
|
||||||
|
|
||||||
|
// OpenBrowserLogin shows the SSO popup window, creating it on first use (and
|
||||||
|
// after the user has closed a previous instance). The URI is encoded into
|
||||||
|
// the window's start URL so the React page reads it via useSearchParams.
|
||||||
|
func (s *WindowManager) OpenBrowserLogin(uri string) {
|
||||||
|
s.mu.Lock()
|
||||||
|
defer s.mu.Unlock()
|
||||||
|
if s.browserLogin == nil {
|
||||||
|
startURL := "/#/browser-login"
|
||||||
|
if uri != "" {
|
||||||
|
startURL = "/#/browser-login?uri=" + url.QueryEscape(uri)
|
||||||
|
}
|
||||||
|
s.browserLogin = s.app.Window.NewWithOptions(application.WebviewWindowOptions{
|
||||||
|
Name: "browser-login",
|
||||||
|
Title: "NetBird Sign-in",
|
||||||
|
Width: 460,
|
||||||
|
Height: 440,
|
||||||
|
DisableResize: true,
|
||||||
|
MinimiseButtonState: application.ButtonHidden,
|
||||||
|
MaximiseButtonState: application.ButtonHidden,
|
||||||
|
CloseButtonState: application.ButtonEnabled,
|
||||||
|
BackgroundColour: application.NewRGB(24, 26, 29),
|
||||||
|
URL: startURL,
|
||||||
|
Mac: application.MacWindow{
|
||||||
|
InvisibleTitleBarHeight: 38,
|
||||||
|
Backdrop: application.MacBackdropTranslucent,
|
||||||
|
TitleBar: application.MacTitleBarHiddenInset,
|
||||||
|
CollectionBehavior: application.MacWindowCollectionBehaviorFullScreenNone,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
bl := s.browserLogin
|
||||||
|
// User-initiated close (red X) means cancel. Emit the event so
|
||||||
|
// startLogin() can tear the SSO wait down, then let the window
|
||||||
|
// destroy naturally — no hide trickery.
|
||||||
|
bl.OnWindowEvent(events.Common.WindowClosing, func(_ *application.WindowEvent) {
|
||||||
|
s.app.Event.Emit(EventBrowserLoginCancel)
|
||||||
|
s.mu.Lock()
|
||||||
|
s.browserLogin = nil
|
||||||
|
s.mu.Unlock()
|
||||||
|
})
|
||||||
|
} else if uri != "" {
|
||||||
|
s.browserLogin.SetURL("/#/browser-login?uri=" + url.QueryEscape(uri))
|
||||||
|
}
|
||||||
|
s.browserLogin.Show()
|
||||||
|
s.browserLogin.Focus()
|
||||||
|
}
|
||||||
|
|
||||||
|
// CloseBrowserLogin destroys the SSO popup window if it exists. Called from
|
||||||
|
// startLogin() when the flow completes or cancels programmatically.
|
||||||
|
func (s *WindowManager) CloseBrowserLogin() {
|
||||||
|
s.mu.Lock()
|
||||||
|
w := s.browserLogin
|
||||||
|
s.browserLogin = nil
|
||||||
|
s.mu.Unlock()
|
||||||
|
if w != nil {
|
||||||
|
w.Close()
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,66 +0,0 @@
|
|||||||
//go:build !android && !ios && !freebsd && !js
|
|
||||||
|
|
||||||
package services
|
|
||||||
|
|
||||||
import (
|
|
||||||
"github.com/wailsapp/wails/v3/pkg/application"
|
|
||||||
"github.com/wailsapp/wails/v3/pkg/events"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Windows opens auxiliary application windows on demand from the frontend.
|
|
||||||
// The main window is created up-front in main.go; this service is for
|
|
||||||
// secondary, on-demand surfaces (Settings).
|
|
||||||
//
|
|
||||||
// The settings window is created hidden at app startup so its React bundle is
|
|
||||||
// already loaded by the time the user clicks the Settings icon — OpenSettings
|
|
||||||
// then just shows and focuses the pre-warmed window. Closing the window hides
|
|
||||||
// it instead of destroying it, so reopening is also instant.
|
|
||||||
type Windows struct {
|
|
||||||
app *application.App
|
|
||||||
settings *application.WebviewWindow
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewWindows(app *application.App) *Windows {
|
|
||||||
w := &Windows{app: app}
|
|
||||||
w.settings = w.buildSettings()
|
|
||||||
return w
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *Windows) buildSettings() *application.WebviewWindow {
|
|
||||||
w := s.app.Window.NewWithOptions(application.WebviewWindowOptions{
|
|
||||||
Title: "NetBird Settings",
|
|
||||||
Width: 900,
|
|
||||||
Height: 640,
|
|
||||||
Hidden: true,
|
|
||||||
DisableResize: true,
|
|
||||||
MinimiseButtonState: application.ButtonHidden,
|
|
||||||
MaximiseButtonState: application.ButtonHidden,
|
|
||||||
CloseButtonState: application.ButtonEnabled,
|
|
||||||
BackgroundColour: application.NewRGB(24, 26, 29),
|
|
||||||
URL: "/#/settings",
|
|
||||||
Mac: application.MacWindow{
|
|
||||||
InvisibleTitleBarHeight: 38,
|
|
||||||
Backdrop: application.MacBackdropTranslucent,
|
|
||||||
TitleBar: application.MacTitleBarHiddenInset,
|
|
||||||
CollectionBehavior: application.MacWindowCollectionBehaviorFullScreenNone,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
// Hide instead of close so the React bundle stays warm and the next
|
|
||||||
// OpenSettings is instant — same trick the main window uses.
|
|
||||||
w.RegisterHook(events.Common.WindowClosing, func(e *application.WindowEvent) {
|
|
||||||
e.Cancel()
|
|
||||||
w.Hide()
|
|
||||||
})
|
|
||||||
|
|
||||||
return w
|
|
||||||
}
|
|
||||||
|
|
||||||
// OpenSettings shows the pre-warmed settings window.
|
|
||||||
func (s *Windows) OpenSettings() {
|
|
||||||
if s.settings == nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
s.settings.Show()
|
|
||||||
s.settings.Focus()
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user