add peer details

This commit is contained in:
Eduard Gert
2026-05-27 14:52:19 +02:00
parent 004a305e46
commit 0e83d2ad94
35 changed files with 1549 additions and 345 deletions

View File

@@ -16,6 +16,7 @@
"dependencies": {
"@radix-ui/react-dialog": "^1.1.15",
"@radix-ui/react-dropdown-menu": "^2.1.16",
"@radix-ui/react-hover-card": "^1.1.15",
"@radix-ui/react-label": "^2.1.8",
"@radix-ui/react-popover": "^1.1.15",
"@radix-ui/react-radio-group": "^1.3.8",

View File

@@ -11,6 +11,9 @@ dependencies:
'@radix-ui/react-dropdown-menu':
specifier: ^2.1.16
version: 2.1.16(@types/react-dom@18.3.7)(@types/react@18.3.28)(react-dom@18.3.1)(react@18.3.1)
'@radix-ui/react-hover-card':
specifier: ^1.1.15
version: 1.1.15(@types/react-dom@18.3.7)(@types/react@18.3.28)(react-dom@18.3.1)(react@18.3.1)
'@radix-ui/react-label':
specifier: ^2.1.8
version: 2.1.8(@types/react-dom@18.3.7)(@types/react@18.3.28)(react-dom@18.3.1)(react@18.3.1)
@@ -828,6 +831,34 @@ packages:
react-dom: 18.3.1(react@18.3.1)
dev: false
/@radix-ui/react-hover-card@1.1.15(@types/react-dom@18.3.7)(@types/react@18.3.28)(react-dom@18.3.1)(react@18.3.1):
resolution: {integrity: sha512-qgTkjNT1CfKMoP0rcasmlH2r1DAiYicWsDsufxl940sT2wHNEWWv6FMWIQXWhVdmC1d/HYfbhQx60KYyAtKxjg==}
peerDependencies:
'@types/react': '*'
'@types/react-dom': '*'
react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc
react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc
peerDependenciesMeta:
'@types/react':
optional: true
'@types/react-dom':
optional: true
dependencies:
'@radix-ui/primitive': 1.1.3
'@radix-ui/react-compose-refs': 1.1.2(@types/react@18.3.28)(react@18.3.1)
'@radix-ui/react-context': 1.1.2(@types/react@18.3.28)(react@18.3.1)
'@radix-ui/react-dismissable-layer': 1.1.11(@types/react-dom@18.3.7)(@types/react@18.3.28)(react-dom@18.3.1)(react@18.3.1)
'@radix-ui/react-popper': 1.2.8(@types/react-dom@18.3.7)(@types/react@18.3.28)(react-dom@18.3.1)(react@18.3.1)
'@radix-ui/react-portal': 1.1.9(@types/react-dom@18.3.7)(@types/react@18.3.28)(react-dom@18.3.1)(react@18.3.1)
'@radix-ui/react-presence': 1.1.5(@types/react-dom@18.3.7)(@types/react@18.3.28)(react-dom@18.3.1)(react@18.3.1)
'@radix-ui/react-primitive': 2.1.3(@types/react-dom@18.3.7)(@types/react@18.3.28)(react-dom@18.3.1)(react@18.3.1)
'@radix-ui/react-use-controllable-state': 1.2.2(@types/react@18.3.28)(react@18.3.1)
'@types/react': 18.3.28
'@types/react-dom': 18.3.7(@types/react@18.3.28)
react: 18.3.1
react-dom: 18.3.1(react@18.3.1)
dev: false
/@radix-ui/react-id@1.1.1(@types/react@18.3.28)(react@18.3.1):
resolution: {integrity: sha512-kGkGegYIdQsOb4XjsfM97rXsiHaBwco+hFI66oO4s9LU+PLAC5oJ7khdOVFxkhsmlbpUqDAvXw11CluXP+jkHg==}
peerDependencies:

View File

@@ -1,9 +1,13 @@
import { ComponentType, forwardRef, ReactNode } from "react";
import { motion, HTMLMotionProps } from "framer-motion";
import {
ButtonHTMLAttributes,
ComponentType,
forwardRef,
ReactNode,
} from "react";
import { LucideProps } from "lucide-react";
import { cn } from "@/lib/cn";
type Props = HTMLMotionProps<"button"> & {
type Props = ButtonHTMLAttributes<HTMLButtonElement> & {
icon?: ComponentType<LucideProps>;
iconNode?: ReactNode;
title: string;
@@ -28,10 +32,9 @@ export const CardNavItem = forwardRef<HTMLButtonElement, Props>(
ref,
) {
return (
<motion.button
<button
ref={ref}
type={type}
whileTap={{ scale: 0.98 }}
className={cn(
"w-full flex items-center gap-3 p-1.5 rounded-lg cursor-default outline-none text-left",
"transition-colors duration-150",
@@ -77,7 +80,7 @@ export const CardNavItem = forwardRef<HTMLButtonElement, Props>(
</p>
)}
</div>
</motion.button>
</button>
);
},
);

View File

@@ -8,6 +8,7 @@ type CopyToClipboardProps = {
size?: number;
iconAlignment?: "left" | "right";
className?: string;
iconClassName?: string;
alwaysShowIcon?: boolean;
};
@@ -17,6 +18,7 @@ export const CopyToClipboard = ({
size = 10,
iconAlignment = "right",
className,
iconClassName,
alwaysShowIcon = false,
}: CopyToClipboardProps) => {
const wrapperRef = useRef<HTMLDivElement>(null);
@@ -57,6 +59,7 @@ export const CopyToClipboard = ({
className={cn(
"shrink-0 inline-flex relative top-[2px] right-[1px]",
iconAlignment === "left" ? "order-first" : "order-last",
iconClassName,
)}
>
<Check

View File

@@ -0,0 +1,70 @@
import { ComponentType } from "react";
import { useTranslation } from "react-i18next";
import { Browser } from "@wailsio/runtime";
import { ExternalLinkIcon, LucideProps } from "lucide-react";
import { cn } from "@/lib/cn";
import { SquareIcon } from "./SquareIcon";
type Props = {
icon: ComponentType<LucideProps>;
title: string;
description?: string;
learnMoreUrl?: string;
learnMoreTopic?: string;
className?: string;
};
const openUrl = (url: string) => {
void Browser.OpenURL(url).catch(() => window.open(url, "_blank"));
};
export const EmptyState = ({
icon,
title,
description,
learnMoreUrl,
learnMoreTopic,
className,
}: Props) => {
const { t } = useTranslation();
return (
<div className={cn("py-12 text-center", className)}>
<div
className={
"flex flex-col items-center justify-center max-w-sm mx-auto"
}
>
<SquareIcon icon={icon} className={"mb-3"} />
<p className={"text-base font-semibold text-nb-gray-200 mb-1"}>
{title}
</p>
{description && (
<p className={"text-sm text-nb-gray-350"}>{description}</p>
)}
{learnMoreUrl && learnMoreTopic && (
<p className={"text-sm text-nb-gray-350"}>
{t("common.learnMoreAbout")}{" "}
<a
href={learnMoreUrl}
onClick={(e) => {
e.preventDefault();
openUrl(learnMoreUrl);
}}
className={cn(
"text-netbird hover:underline underline-offset-4",
"cursor-pointer wails-no-draggable",
"inline-flex items-center gap-1",
)}
>
{learnMoreTopic}
<ExternalLinkIcon
size={12}
className={"shrink-0"}
/>
</a>
</p>
)}
</div>
</div>
);
};

View File

@@ -0,0 +1,34 @@
import * as HoverCardPrimitive from "@radix-ui/react-hover-card";
import * as React from "react";
import { cn } from "@/lib/cn";
const HoverCard = HoverCardPrimitive.Root;
const HoverCardTrigger = HoverCardPrimitive.Trigger;
const HoverCardPortal = HoverCardPrimitive.Portal;
const HoverCardContent = React.forwardRef<
React.ElementRef<typeof HoverCardPrimitive.Content>,
React.ComponentPropsWithoutRef<typeof HoverCardPrimitive.Content>
>(({ className, align = "start", sideOffset = 6, ...props }, ref) => (
<HoverCardPrimitive.Portal>
<HoverCardPrimitive.Content
ref={ref}
align={align}
sideOffset={sideOffset}
className={cn(
"z-50 w-80 rounded-lg border border-nb-gray-900 bg-nb-gray-935",
"p-3 text-nb-gray-200 shadow-lg outline-none",
"data-[state=open]:animate-in data-[state=closed]:animate-out",
"data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0",
"data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95",
"data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2",
"data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2",
className,
)}
{...props}
/>
</HoverCardPrimitive.Portal>
));
HoverCardContent.displayName = HoverCardPrimitive.Content.displayName;
export { HoverCard, HoverCardContent, HoverCardPortal, HoverCardTrigger };

View File

@@ -0,0 +1,25 @@
import { ComponentType } from "react";
import { FunnelXIcon, LucideProps } from "lucide-react";
import { useTranslation } from "react-i18next";
import { EmptyState } from "./EmptyState";
type Props = {
icon?: ComponentType<LucideProps>;
title?: string;
description?: string;
};
export const NoResults = ({
icon = FunnelXIcon,
title,
description,
}: Props) => {
const { t } = useTranslation();
return (
<EmptyState
icon={icon}
title={title ?? t("common.noResults.title")}
description={description ?? t("common.noResults.description")}
/>
);
};

View File

@@ -0,0 +1,20 @@
import { UnplugIcon } from "lucide-react";
import { useTranslation } from "react-i18next";
import { EmptyState } from "./EmptyState";
export const NotConnectedState = () => {
const { t } = useTranslation();
return (
<div
className={
"h-full min-h-[260px] flex items-center justify-center px-6"
}
>
<EmptyState
icon={UnplugIcon}
title={t("notConnected.title")}
description={t("notConnected.description")}
/>
</div>
);
};

View File

@@ -1,15 +1,24 @@
import { forwardRef, InputHTMLAttributes } from "react";
import { forwardRef, InputHTMLAttributes, ReactNode } from "react";
import { SearchIcon } from "lucide-react";
import { cn } from "@/lib/cn";
type Props = InputHTMLAttributes<HTMLInputElement> & {
iconSize?: number;
shortcut?: ReactNode;
};
export const SearchInput = forwardRef<HTMLInputElement, Props>(
function SearchInput({ iconSize = 16, className, ...props }, ref) {
function SearchInput(
{ iconSize = 16, className, disabled, shortcut, ...props },
ref,
) {
return (
<div className={"flex items-center gap-2 px-1 h-10"}>
<div
className={cn(
"flex items-center gap-2 px-1 h-10",
disabled && "opacity-50",
)}
>
<SearchIcon
size={iconSize}
className={"text-nb-gray-300 shrink-0"}
@@ -17,13 +26,29 @@ export const SearchInput = forwardRef<HTMLInputElement, Props>(
<input
ref={ref}
type={"text"}
disabled={disabled}
{...props}
className={cn(
"w-full bg-transparent text-sm text-nb-gray-200 placeholder:text-nb-gray-400",
"outline-none border-none",
disabled && "cursor-not-allowed",
className,
)}
/>
{shortcut && (
<span
className={cn(
"shrink-0 select-none",
"inline-flex items-center justify-center",
"h-5 min-w-[20px] px-1.5 rounded",
"border border-nb-gray-850 bg-nb-gray-920",
"text-[10px] font-medium text-nb-gray-400",
"wails-no-draggable",
)}
>
{shortcut}
</span>
)}
</div>
);
},

View File

@@ -14,7 +14,7 @@ type SquareIconProps = {
export const SquareIcon = ({ icon: Icon, iconSize = 20, className }: SquareIconProps) => (
<div
className={cn(
"h-11 w-11 rounded-xl flex items-center justify-center bg-nb-gray-920 border border-nb-gray-900 text-white",
"h-11 w-11 rounded-lg flex items-center justify-center bg-nb-gray-920 border border-nb-gray-900 text-white",
className,
)}
>

View File

@@ -5,6 +5,7 @@ import { cn } from "@/lib/cn";
type SwitchItemGroupContextValue = {
value: string;
layoutId: string;
disabled: boolean;
};
const SwitchItemGroupContext = createContext<SwitchItemGroupContextValue | null>(null);
@@ -22,18 +23,27 @@ type Props = {
onChange: (value: string) => void;
children: ReactNode;
className?: string;
disabled?: boolean;
};
export const SwitchItemGroup = ({ value, onChange, children, className }: Props) => {
export const SwitchItemGroup = ({
value,
onChange,
children,
className,
disabled = false,
}: Props) => {
const layoutId = useId();
return (
<SwitchItemGroupContext.Provider value={{ value, layoutId }}>
<SwitchItemGroupContext.Provider value={{ value, layoutId, disabled }}>
<RadioGroup.Root
value={value}
onValueChange={onChange}
disabled={disabled}
className={cn(
"flex shrink-0 rounded-lg border border-nb-gray-850 bg-nb-gray-910 p-1 overflow-hidden",
disabled && "opacity-50 pointer-events-none",
className,
)}
>

View File

@@ -1,5 +1,6 @@
import { Outlet } from "react-router-dom";
import { Header } from "@/layouts/Header.tsx";
import { NavSectionProvider } from "@/lib/navSection";
import { ViewModeProvider } from "@/lib/viewMode";
import { ClientVersionProvider } from "@/modules/auto-update/ClientVersionContext.tsx";
import { StatusProvider } from "@/modules/daemon-status/StatusContext.tsx";
@@ -14,8 +15,10 @@ export const AppLayout = () => {
<DebugBundleProvider>
<ClientVersionProvider>
<ViewModeProvider>
<Header />
<Outlet />
<NavSectionProvider>
<Header />
<Outlet />
</NavSectionProvider>
</ViewModeProvider>
</ClientVersionProvider>
</DebugBundleProvider>

View File

@@ -319,7 +319,7 @@ export const ConnectionStatusSwitch = () => {
<div className={"flex flex-col items-center"}>
<h1
className={
"text-sm font-medium text-nb-gray-200 tracking-wide transition-colors duration-300 select-none wails-no-draggable"
"text-sm font-medium text-nb-gray-200 tracking-wide transition-colors duration-300 select-none wails-no-draggable mb-1"
}
>
{t(STATUS_KEY[connState])}
@@ -339,7 +339,7 @@ export const ConnectionStatusSwitch = () => {
<CopyToClipboard
message={ip}
className={cn(
"min-h-[1em] transition-opacity duration-300 ",
"min-h-[1em] transition-opacity duration-300",
"relative left-[0.55rem]",
showLocal && ip ? "opacity-100" : "opacity-0 pointer-events-none",
)}

View File

@@ -48,91 +48,97 @@ export const Header = () => {
setViewMode(mode);
};
return (
<div
className={cn(
"shrink-0 cursor-default wails-draggable grid grid-cols-3 items-center",
//"bg-gradient-to-b from-nb-gray-850/30",
//"bg-nb-gray-935 border border-b border-nb-gray-850",
"py-3 px-3",
)}
>
<div />
<div className={"flex justify-center ml-4"}>
<ProfileDropdown onManageProfiles={openManageProfiles} />
</div>
<div className={"flex justify-end"}>
<div className={"relative"}>
<DropdownMenu modal={false} open={menuOpen} onOpenChange={setMenuOpen}>
<DropdownMenuTrigger asChild className={"wails-no-draggable"}>
<IconButton
icon={MoreVertical}
iconClassName={"text-nb-gray-200 wails-no-draggable"}
className={"select-none"}
/>
</DropdownMenuTrigger>
<DropdownMenuContent
align="end"
sideOffset={8}
className="min-w-52 select-none data-[state=closed]:!animate-none data-[state=closed]:!duration-0"
>
{updateAvailable && (
<>
<DropdownMenuItem onClick={openAbout}>
<div className="flex items-center gap-2">
<ArrowUpCircleIcon
size={14}
className={"text-netbird"}
/>
<span className={"text-netbird"}>
{t("header.menu.updateAvailable")}
</span>
</div>
</DropdownMenuItem>
<DropdownMenuSeparator />
</>
)}
<DropdownMenuItem onClick={openSettings}>
const profileSlot = <ProfileDropdown onManageProfiles={openManageProfiles} />;
const settingsSlot = (
<div className={"relative"}>
<DropdownMenu modal={false} open={menuOpen} onOpenChange={setMenuOpen}>
<DropdownMenuTrigger asChild className={"wails-no-draggable"}>
<IconButton
icon={MoreVertical}
iconClassName={"text-nb-gray-200 wails-no-draggable"}
className={"select-none"}
/>
</DropdownMenuTrigger>
<DropdownMenuContent
align="end"
sideOffset={8}
className="min-w-52 select-none data-[state=closed]:!animate-none data-[state=closed]:!duration-0"
>
{updateAvailable && (
<>
<DropdownMenuItem onClick={openAbout}>
<div className="flex items-center gap-2">
<Settings size={14} />
{t("header.menu.settings")}
<ArrowUpCircleIcon size={14} className={"text-netbird"} />
<span className={"text-netbird"}>
{t("header.menu.updateAvailable")}
</span>
</div>
</DropdownMenuItem>
<DropdownMenuSeparator />
<ViewModeItem
icon={RectangleVertical}
label={t("header.menu.defaultView")}
selected={viewMode === "default"}
onSelect={() => selectMode("default")}
/>
<ViewModeItem
icon={PanelsRightBottom}
label={t("header.menu.advancedView")}
selected={viewMode === "advanced"}
onSelect={() => selectMode("advanced")}
/>
</DropdownMenuContent>
</DropdownMenu>
{updateAvailable && (
<span
className={
"pointer-events-none absolute top-1.5 right-1.5 flex h-2.5 w-2.5 items-center justify-center"
}
>
<span
className={
"absolute inset-0 rounded-full bg-netbird opacity-60 animate-ping"
}
/>
<span
className={
"relative h-1.5 w-1.5 rounded-full bg-netbird"
}
/>
</span>
</>
)}
</div>
<DropdownMenuItem onClick={openSettings}>
<div className="flex items-center gap-2">
<Settings size={14} />
{t("header.menu.settings")}
</div>
</DropdownMenuItem>
<DropdownMenuSeparator />
<ViewModeItem
icon={RectangleVertical}
label={t("header.menu.defaultView")}
selected={viewMode === "default"}
onSelect={() => selectMode("default")}
/>
<ViewModeItem
icon={PanelsRightBottom}
label={t("header.menu.advancedView")}
selected={viewMode === "advanced"}
onSelect={() => selectMode("advanced")}
/>
</DropdownMenuContent>
</DropdownMenu>
{updateAvailable && (
<span
className={
"pointer-events-none absolute top-1.5 right-1.5 flex h-2.5 w-2.5 items-center justify-center"
}
>
<span
className={
"absolute inset-0 rounded-full bg-netbird opacity-60 animate-ping"
}
/>
<span className={"relative h-1.5 w-1.5 rounded-full bg-netbird"} />
</span>
)}
</div>
);
// The inner grid is locked to 356px (the default-mode content width:
// 380px window 12px px-3 each side). It stays left-anchored regardless
// of window size, so the profile keeps the exact same absolute X
// position when the user flips to advanced view. The settings button is
// pulled out as an absolute, right-anchored element so it tracks the
// window's right edge in both modes.
// Header height matches the Settings window's top traffic-light strip
// so the right panel ends up the same height in both windows. The h-10
// of the inner buttons (profile trigger, more-vertical) defines the
// natural height; the strip in SettingsLayout is sized to mirror it.
return (
<div
className={cn(
"shrink-0 cursor-default wails-draggable relative",
"flex items-center h-12 px-3 top-2",
)}
>
<div className={"grid grid-cols-3 items-center w-[356px] shrink-0"}>
<div />
<div className={"flex justify-center ml-4"}>{profileSlot}</div>
<div />
</div>
<div className={"absolute right-3 top-1/2 -translate-y-1/2"}>{settingsSlot}</div>
</div>
);
};

View File

@@ -1,47 +1,56 @@
import { useState } from "react";
import { ConnectionStatusSwitch } from "@/layouts/ConnectionStatusSwitch.tsx";
import { MainRightSide } from "@/layouts/MainRightSide.tsx";
import { Navigation, NavSection } from "@/layouts/Navigation.tsx";
import { Navigation } from "@/layouts/Navigation.tsx";
import { useNavSection } from "@/lib/navSection";
import { useViewMode } from "@/lib/viewMode";
import { Peers } from "@/modules/peers/Peers";
import { Networks } from "@/modules/networks/Networks";
import { ExitNodes } from "@/modules/exit-nodes/ExitNodes";
import { NetworksProvider } from "@/modules/networks/NetworksContext";
import {
PeerDetailProvider,
usePeerDetail,
} from "@/modules/peers/PeerDetailContext";
import { PeerDetailPanel } from "@/modules/peers/PeerDetailPanel";
export const Main = () => {
const { viewMode } = useViewMode();
const isAdvanced = viewMode === "advanced";
const [section, setSection] = useState<NavSection>("peers");
return (
<NetworksProvider>
<div className={"wails-draggable flex flex-1 min-h-0 p-4 gap-4"}>
{/* Fixed-width column for the connection switch. Navigation
is rendered absolutely at the bottom in advanced view so
it doesn't reshape the column and shift the switch up. */}
<div
className={
"relative flex flex-col items-center shrink-0 w-[348px]"
}
>
<ConnectionStatusSwitch />
{isAdvanced && (
<div className={"absolute inset-x-0 bottom-0 px-1"}>
<Navigation
active={section}
onSelect={setSection}
/>
</div>
)}
</div>
{isAdvanced && (
<MainRightSide>
{section === "peers" && <Peers />}
{section === "networks" && <Networks />}
{section === "exitNode" && <ExitNodes />}
</MainRightSide>
)}
</div>
<PeerDetailProvider>
<MainBody />
</PeerDetailProvider>
</NetworksProvider>
);
};
const MainBody = () => {
const { viewMode } = useViewMode();
const isAdvanced = viewMode === "advanced";
const { section } = useNavSection();
const { selected } = usePeerDetail();
return (
<div className={"wails-draggable flex flex-1 min-h-0 p-4 gap-4"}>
<div
className={
"flex flex-col items-center shrink-0 w-[348px]"
}
>
<ConnectionStatusSwitch />
</div>
{isAdvanced && (
<MainRightSide
overlay={<PeerDetailPanel />}
overlayOpen={selected !== null}
>
<Navigation />
<div className={"flex-1 min-h-0 flex flex-col"}>
{section === "peers" && <Peers />}
{section === "networks" && <Networks />}
{section === "exitNode" && <ExitNodes />}
</div>
</MainRightSide>
)}
</div>
);
};

View File

@@ -1,20 +1,39 @@
import { ReactNode } from "react";
import { motion } from "framer-motion";
import { cn } from "@/lib/cn.ts";
type Props = {
children: ReactNode;
overlay?: ReactNode;
overlayOpen?: boolean;
};
export const MainRightSide = ({ children }: Props) => {
// iOS-style push transition: incoming pane slides in from the right while
// the outgoing pane shifts slightly left. Same easing on both sides so
// they feel like one motion.
const PANEL_TRANSITION = {
duration: 0.32,
ease: [0.32, 0.72, 0, 1] as [number, number, number, number],
};
export const MainRightSide = ({ children, overlay, overlayOpen = false }: Props) => {
return (
<div
className={cn(
"wails-no-draggable",
"bg-nb-gray-935 border border-nb-gray-910",
"flex-1 min-h-0 min-w-0 flex flex-col rounded-xl rounded-br-2xl overflow-hidden",
"wails-no-draggable relative",
"bg-nb-gray-940 border border-nb-gray-920",
"flex-1 min-h-0 min-w-0 flex flex-col rounded-xl rounded-br-2xl overflow-hidden",
)}
>
{children}
<motion.div
animate={{ x: overlayOpen ? -48 : 0 }}
transition={PANEL_TRANSITION}
className={"flex-1 min-h-0 min-w-0 flex flex-col"}
style={{ pointerEvents: overlayOpen ? "none" : "auto" }}
>
{children}
</motion.div>
{overlay}
</div>
);
};

View File

@@ -1,75 +1,79 @@
import { useMemo } from "react";
import { ComponentType } from "react";
import { useTranslation } from "react-i18next";
import { Layers3Icon, MonitorSmartphoneIcon, SquareArrowUpRight } from "lucide-react";
import { CardNavItem } from "@/components/CardNavItem.tsx";
import { Layers3Icon, LucideProps, MonitorSmartphoneIcon, SquareArrowUpRight } from "lucide-react";
import { cn } from "@/lib/cn";
import { useNavSection, type NavSection } from "@/lib/navSection";
import { useStatus } from "@/modules/daemon-status/StatusContext";
import { useNetworks } from "@/modules/networks/NetworksContext";
export type NavSection = "peers" | "networks" | "exitNode";
type Props = {
active: NavSection;
onSelect: (section: NavSection) => void;
type TabEntry = {
value: NavSection;
label: string;
icon: ComponentType<LucideProps>;
};
export const Navigation = ({ active, onSelect }: Props) => {
export const Navigation = () => {
const { t } = useTranslation();
const { section, setSection } = useNavSection();
const { status } = useStatus();
const { networkRoutes, exitNodes, activeExitNode } = useNetworks();
const isConnected = status?.status === "Connected";
const peerCounts = useMemo(() => {
const peers = status?.peers ?? [];
const online = peers.filter((p) => p.connStatus === "Connected").length;
return { online, total: peers.length };
}, [status?.peers]);
const networkCounts = useMemo(
() => ({
active: networkRoutes.filter((r) => r.selected).length,
total: networkRoutes.length,
}),
[networkRoutes],
);
const exitNodeDescription = activeExitNode
? activeExitNode.id
: t("nav.exitNode.none", { total: exitNodes.length });
const tabs: TabEntry[] = [
{
value: "peers",
label: t("nav.peers.title"),
icon: MonitorSmartphoneIcon,
},
{
value: "networks",
label: t("nav.resources.title"),
icon: Layers3Icon,
},
{
value: "exitNode",
label: t("nav.exitNode.title"),
icon: ExitNodeIcon,
},
];
return (
<nav className={"w-full flex flex-col gap-1 mt-auto"}>
<CardNavItem
icon={MonitorSmartphoneIcon}
title={t("nav.peers.title")}
description={t("nav.peers.description", peerCounts)}
active={active === "peers"}
onClick={() => onSelect("peers")}
/>
<CardNavItem
icon={Layers3Icon}
title={t("nav.resources.title")}
description={t("nav.resources.description", networkCounts)}
iconSize={14}
active={active === "networks"}
onClick={() => onSelect("networks")}
/>
<CardNavItem
iconNode={
<SquareArrowUpRight
size={14}
<div className={"wails-no-draggable shrink-0 flex items-stretch"}>
{tabs.map((tab) => {
const isActive = tab.value === section;
const isDisabled = !isConnected && !isActive;
const Icon = tab.icon;
return (
<button
key={tab.value}
type={"button"}
onClick={() => setSection(tab.value)}
disabled={isDisabled}
className={cn(
"transition-colors duration-150 rotate-45",
active === "exitNode"
? "text-nb-gray-200"
: "text-nb-gray-400",
"group relative flex flex-1 items-center justify-center",
"gap-2.5 px-5 py-3",
"outline-none transition-all",
isActive ? "text-netbird" : "text-nb-gray-500 hover:text-nb-gray-400",
isDisabled ? "opacity-50 cursor-not-allowed" : "cursor-pointer",
)}
/>
}
title={t("nav.exitNode.title")}
description={exitNodeDescription}
active={active === "exitNode"}
onClick={() => onSelect("exitNode")}
/>
</nav>
>
<Icon size={14} />
<span className={"text-sm font-normal"}>{tab.label}</span>
<span
className={cn(
"absolute inset-x-0 bottom-0 h-px transition-all",
isActive
? "bg-netbird"
: "bg-nb-gray-900 group-hover:bg-nb-gray-600",
)}
/>
</button>
);
})}
</div>
);
};
const ExitNodeIcon = (props: LucideProps) => (
<SquareArrowUpRight {...props} className={cn("rotate-45", props.className)} />
);
export type { NavSection } from "@/lib/navSection";

View File

@@ -9,12 +9,11 @@ import { ProfileProvider } from "@/modules/profile/ProfileContext.tsx";
// settings window has its own native title bar and doesn't show the profile
// selector / panel toggle / settings icon.
//
// The 38px placeholder strip at the top accounts for the macOS
// The h-10 placeholder strip at the top accounts for the macOS
// `MacTitleBarHiddenInset` setting in services/windowmanager.go: the native
// title bar is invisible but the traffic-light buttons still float in the
// top-left corner. Without this strip the buttons would overlap the settings
// content. The strip is `wails-draggable` so users can move the window by
// dragging it.
// top-left corner. The height also mirrors the main window's Header so the
// MainRightSide panel ends up the same height in both windows.
export const SettingsLayout = () => {
return (
<div className={"relative flex h-full flex-col select-none"}>
@@ -24,7 +23,7 @@ export const SettingsLayout = () => {
<ClientVersionProvider>
<div
className={
"wails-draggable cursor-default select-none h-[38px] shrink-0"
"wails-draggable cursor-default select-none h-12 shrink-0"
}
/>
<Outlet />

View File

@@ -0,0 +1,29 @@
import { createContext, useContext, useState, type ReactNode } from "react";
export type NavSection = "peers" | "networks" | "exitNode";
type NavSectionContextValue = {
section: NavSection;
setSection: (s: NavSection) => void;
};
const NavSectionContext = createContext<NavSectionContextValue | null>(null);
export const useNavSection = (): NavSectionContextValue => {
const ctx = useContext(NavSectionContext);
if (!ctx) {
throw new Error(
"useNavSection must be used inside NavSectionProvider",
);
}
return ctx;
};
export const NavSectionProvider = ({ children }: { children: ReactNode }) => {
const [section, setSection] = useState<NavSection>("peers");
return (
<NavSectionContext.Provider value={{ section, setSection }}>
{children}
</NavSectionContext.Provider>
);
};

View File

@@ -0,0 +1,56 @@
import { useEffect } from "react";
export type Shortcut = {
key: string; // e.g. "k", "Escape", "/"
cmd?: boolean; // requires Cmd (mac) / Ctrl (win/linux)
shift?: boolean;
alt?: boolean;
// When true (default), preventDefault is called on a match.
preventDefault?: boolean;
};
// Listens for a keyboard shortcut on the window and invokes `callback` on
// match. Disable conditionally via `enabled` to avoid stealing keys while a
// dialog/panel is in the foreground.
export const useKeyboardShortcut = (
shortcut: Shortcut,
callback: () => void,
enabled = true,
) => {
useEffect(() => {
if (!enabled) return;
const onKey = (e: KeyboardEvent) => {
if (e.key.toLowerCase() !== shortcut.key.toLowerCase()) return;
const mod = e.metaKey || e.ctrlKey;
if (!!shortcut.cmd !== mod) return;
if (!!shortcut.shift !== e.shiftKey) return;
if (!!shortcut.alt !== e.altKey) return;
if (shortcut.preventDefault !== false) e.preventDefault();
callback();
};
window.addEventListener("keydown", onKey);
return () => window.removeEventListener("keydown", onKey);
}, [
shortcut.key,
shortcut.cmd,
shortcut.shift,
shortcut.alt,
shortcut.preventDefault,
callback,
enabled,
]);
};
// True on macOS — use the ⌘ glyph; otherwise show "Ctrl".
export const isMac =
typeof navigator !== "undefined" &&
/Mac|iPhone|iPad|iPod/i.test(navigator.platform);
export const formatShortcut = (shortcut: Shortcut): string => {
const parts: string[] = [];
if (shortcut.cmd) parts.push(isMac ? "⌘" : "Ctrl");
if (shortcut.shift) parts.push(isMac ? "⇧" : "Shift");
if (shortcut.alt) parts.push(isMac ? "⌥" : "Alt");
parts.push(shortcut.key.length === 1 ? shortcut.key.toUpperCase() : shortcut.key);
return parts.join(isMac ? "" : "+");
};

View File

@@ -1,13 +1,26 @@
import { useEffect, useMemo, useRef, useState } from "react";
import { useTranslation } from "react-i18next";
import * as ScrollArea from "@radix-ui/react-scroll-area";
import { WaypointsIcon } from "lucide-react";
import { cn } from "@/lib/cn";
import { SearchInput } from "@/components/SearchInput";
import { EmptyState } from "@/components/EmptyState";
import { NoResults } from "@/components/NoResults";
import { NotConnectedState } from "@/components/NotConnectedState";
import { useStatus } from "@/modules/daemon-status/StatusContext";
import {
formatShortcut,
useKeyboardShortcut,
} from "@/lib/useKeyboardShortcut";
const SEARCH_SHORTCUT = { key: "k", cmd: true } as const;
import { useNetworks } from "@/modules/networks/NetworksContext";
import { ExitNodesList } from "./ExitNodesList";
export const ExitNodes = () => {
const { t } = useTranslation();
const { status } = useStatus();
const isConnected = status?.status === "Connected";
const { exitNodes, toggleExitNode } = useNetworks();
const [search, setSearch] = useState("");
const searchRef = useRef<HTMLInputElement>(null);
@@ -16,14 +29,16 @@ export const ExitNodes = () => {
searchRef.current?.focus();
}, []);
useKeyboardShortcut(SEARCH_SHORTCUT, () => {
searchRef.current?.focus();
searchRef.current?.select();
});
const filtered = useMemo(() => {
const q = search.trim().toLowerCase();
const matches = exitNodes.filter((r) => {
if (!q) return true;
return (
r.id.toLowerCase().includes(q) ||
r.range.toLowerCase().includes(q)
);
return r.id.toLowerCase().includes(q);
});
return matches.sort((a, b) => {
if (a.selected !== b.selected) return a.selected ? -1 : 1;
@@ -31,6 +46,32 @@ export const ExitNodes = () => {
});
}, [exitNodes, search]);
if (!isConnected) {
return (
<div className={"flex flex-col w-full h-full min-h-0"}>
<NotConnectedState />
</div>
);
}
if (exitNodes.length === 0) {
return (
<div
className={
"flex-1 flex items-center justify-center px-6 w-full h-full min-h-0"
}
>
<EmptyState
icon={WaypointsIcon}
title={t("exitNodes.empty.title")}
description={t("exitNodes.empty.description")}
learnMoreUrl={"https://docs.netbird.io/how-to/exit-node"}
learnMoreTopic={t("nav.exitNode.title")}
/>
</div>
);
}
return (
<div className={"flex flex-col w-full h-full min-h-0 pt-4"}>
<div className={"flex flex-col gap-3 px-6"}>
@@ -39,6 +80,7 @@ export const ExitNodes = () => {
placeholder={t("exitNodes.search.placeholder")}
value={search}
onChange={(e) => setSearch(e.target.value)}
shortcut={formatShortcut(SEARCH_SHORTCUT)}
/>
</div>
<ScrollArea.Root
@@ -46,7 +88,11 @@ export const ExitNodes = () => {
className={"flex-1 min-h-0 overflow-hidden mt-3"}
>
<ScrollArea.Viewport className={"h-full w-full"}>
<ExitNodesList data={filtered} onToggle={toggleExitNode} />
{filtered.length === 0 ? (
<NoResults />
) : (
<ExitNodesList data={filtered} onToggle={toggleExitNode} />
)}
</ScrollArea.Viewport>
<ScrollArea.Scrollbar
orientation={"vertical"}

View File

@@ -1,10 +1,9 @@
import * as RadioGroup from "@radix-ui/react-radio-group";
import { useTranslation } from "react-i18next";
import type { Network } from "@bindings/services/models.js";
import { cn } from "@/lib/cn";
import { CopyToClipboard } from "@/components/CopyToClipboard";
const dotClass = (selected: boolean): string =>
selected ? "bg-green-400" : "bg-nb-gray-500";
const NONE_VALUE = "__none__";
type Props = {
data: Network[];
@@ -13,49 +12,64 @@ type Props = {
export const ExitNodesList = ({ data, onToggle }: Props) => {
const { t } = useTranslation();
if (data.length === 0) {
return (
<div className={"py-12 text-center text-sm text-nb-gray-400"}>
{t("exitNodes.empty")}
</div>
);
}
const active = data.find((n) => n.selected) ?? null;
const value = active?.id ?? NONE_VALUE;
const handleChange = (next: string) => {
if (next === value) return;
if (next === NONE_VALUE) {
if (active) onToggle(active.id, true);
return;
}
onToggle(next, false);
};
return (
<ul className={"flex flex-col"}>
<RadioGroup.Root
value={value}
onValueChange={handleChange}
className={"flex flex-col"}
>
<Row value={NONE_VALUE} label={t("exitNodes.none")} />
{data.map((n) => (
<li
key={n.id}
className={"flex items-center gap-3 px-7 py-3 min-w-0"}
>
<button
type={"button"}
onClick={() => onToggle(n.id, n.selected)}
className={cn(
"h-2 w-2 rounded-full shrink-0 cursor-pointer",
dotClass(n.selected),
)}
title={
n.selected
? t("exitNodes.active")
: t("exitNodes.inactive")
}
/>
<CopyToClipboard message={n.id} className={"min-w-0 flex-1"}>
<span className={"text-[0.81rem] font-medium text-nb-gray-100"}>
{n.id}
</span>
</CopyToClipboard>
<CopyToClipboard
message={n.range}
className={cn("ml-auto shrink-0", "relative left-2.5")}
>
<span className={"text-xs font-mono text-nb-gray-400"}>
{n.range}
</span>
</CopyToClipboard>
</li>
<Row key={n.id} value={n.id} label={n.id} />
))}
</ul>
</RadioGroup.Root>
);
};
type RowProps = {
value: string;
label: string;
};
const Row = ({ value, label }: RowProps) => (
<RadioGroup.Item
value={value}
className={cn(
"group flex items-center gap-3 px-7 py-3 min-w-0 text-left outline-none",
"cursor-pointer wails-no-draggable",
"hover:bg-nb-gray-900/40",
)}
>
<span
className={cn(
"h-4 w-4 shrink-0 rounded-full border",
"border-nb-gray-700 bg-nb-gray-900",
"flex items-center justify-center",
"group-data-[state=checked]:border-netbird group-data-[state=checked]:bg-netbird",
)}
>
<RadioGroup.Indicator
className={"h-2 w-2 rounded-full bg-white"}
/>
</span>
<span
className={
"text-[0.81rem] font-medium text-nb-gray-100 truncate min-w-0 flex-1"
}
>
{label}
</span>
</RadioGroup.Item>
);

View File

@@ -2,19 +2,20 @@ import { useTranslation } from "react-i18next";
import { SwitchItem } from "@/components/SwitchItem";
import { SwitchItemGroup } from "@/components/SwitchItemGroup";
export type NetworkFilter = "all" | "selected" | "overlapping";
export type NetworkFilter = "all" | "active" | "overlapping";
type Props = {
value: NetworkFilter;
onChange: (value: NetworkFilter) => void;
counts: Record<NetworkFilter, number>;
disabled?: boolean;
};
export const NetworkFilters = ({ value, onChange, counts }: Props) => {
export const NetworkFilters = ({ value, onChange, counts, disabled }: Props) => {
const { t, i18n } = useTranslation();
const filters: { value: NetworkFilter; label: string }[] = [
{ value: "all", label: t("networks.filter.all") },
{ value: "selected", label: t("networks.filter.selected") },
{ value: "active", label: t("networks.filter.active") },
{ value: "overlapping", label: t("networks.filter.overlapping") },
];
@@ -23,6 +24,7 @@ export const NetworkFilters = ({ value, onChange, counts }: Props) => {
key={i18n.language}
value={value}
onChange={(v) => onChange(v as NetworkFilter)}
disabled={disabled}
className={"w-full"}
>
{filters.map((f) => (

View File

@@ -1,13 +1,29 @@
import { useEffect, useMemo, useRef, useState } from "react";
import { useTranslation } from "react-i18next";
import * as ScrollArea from "@radix-ui/react-scroll-area";
import { NetworkIcon } from "lucide-react";
import { cn } from "@/lib/cn";
import { SearchInput } from "@/components/SearchInput";
import { NetworkFilters, NetworkFilter } from "./NetworkFilters";
import { EmptyState } from "@/components/EmptyState";
import { NoResults } from "@/components/NoResults";
import { NotConnectedState } from "@/components/NotConnectedState";
import { useStatus } from "@/modules/daemon-status/StatusContext";
import {
formatShortcut,
useKeyboardShortcut,
} from "@/lib/useKeyboardShortcut";
const SEARCH_SHORTCUT = { key: "k", cmd: true } as const;
import { NetworkFilter, NetworkFilters } from "./NetworkFilters";
import { NetworksList } from "./NetworksList";
import { useNetworks } from "./NetworksContext";
const collectOverlapping = (routes: { id: string; range: string; domains: string[] }[]): Set<string> => {
// Map every range string -> ids of CIDR routes that share it. Domain routes
// are skipped (they overlap on domain, not prefix). Single-entry buckets
// aren't overlaps.
const buildOverlapMap = (
routes: { id: string; range: string; domains: string[] }[],
): Map<string, string[]> => {
const byRange = new Map<string, string[]>();
for (const r of routes) {
if (r.domains.length > 0) continue;
@@ -15,15 +31,17 @@ const collectOverlapping = (routes: { id: string; range: string; domains: string
arr.push(r.id);
byRange.set(r.range, arr);
}
const out = new Set<string>();
for (const ids of byRange.values()) {
if (ids.length > 1) ids.forEach((id) => out.add(id));
const out = new Map<string, string[]>();
for (const [range, ids] of byRange) {
if (ids.length > 1) out.set(range, ids);
}
return out;
};
export const Networks = () => {
const { t } = useTranslation();
const { status } = useStatus();
const isConnected = status?.status === "Connected";
const { networkRoutes, toggleNetwork } = useNetworks();
const [search, setSearch] = useState("");
const [filter, setFilter] = useState<NetworkFilter>("all");
@@ -33,25 +51,38 @@ export const Networks = () => {
searchRef.current?.focus();
}, []);
const overlappingIds = useMemo(
() => collectOverlapping(networkRoutes),
useKeyboardShortcut(SEARCH_SHORTCUT, () => {
searchRef.current?.focus();
searchRef.current?.select();
});
const overlapGroups = useMemo(
() => buildOverlapMap(networkRoutes),
[networkRoutes],
);
const overlapById = useMemo(() => {
const map = new Map<string, string[]>();
for (const ids of overlapGroups.values()) {
for (const id of ids) map.set(id, ids);
}
return map;
}, [overlapGroups]);
const counts = useMemo<Record<NetworkFilter, number>>(
() => ({
all: networkRoutes.length,
selected: networkRoutes.filter((r) => r.selected).length,
overlapping: overlappingIds.size,
active: networkRoutes.filter((r) => r.selected).length,
overlapping: overlapById.size,
}),
[networkRoutes, overlappingIds],
[networkRoutes, overlapById],
);
const filtered = useMemo(() => {
const q = search.trim().toLowerCase();
const matches = networkRoutes.filter((r) => {
if (filter === "selected" && !r.selected) return false;
if (filter === "overlapping" && !overlappingIds.has(r.id)) return false;
if (filter === "active" && !r.selected) return false;
if (filter === "overlapping" && !overlapById.has(r.id)) return false;
if (q) {
const haystack = [r.id, r.range, ...r.domains]
.join(" ")
@@ -64,7 +95,33 @@ export const Networks = () => {
if (a.selected !== b.selected) return a.selected ? -1 : 1;
return a.id.localeCompare(b.id);
});
}, [networkRoutes, search, filter, overlappingIds]);
}, [networkRoutes, search, filter, overlapById]);
if (!isConnected) {
return (
<div className={"flex flex-col w-full h-full min-h-0"}>
<NotConnectedState />
</div>
);
}
if (networkRoutes.length === 0) {
return (
<div
className={
"flex-1 flex items-center justify-center px-6 w-full h-full min-h-0"
}
>
<EmptyState
icon={NetworkIcon}
title={t("networks.empty.title")}
description={t("networks.empty.description")}
learnMoreUrl={"https://docs.netbird.io/how-to/networks"}
learnMoreTopic={t("nav.resources.title")}
/>
</div>
);
}
return (
<div className={"flex flex-col w-full h-full min-h-0 pt-4"}>
@@ -74,6 +131,7 @@ export const Networks = () => {
placeholder={t("networks.search.placeholder")}
value={search}
onChange={(e) => setSearch(e.target.value)}
shortcut={formatShortcut(SEARCH_SHORTCUT)}
/>
<NetworkFilters
value={filter}
@@ -86,7 +144,14 @@ export const Networks = () => {
className={"flex-1 min-h-0 overflow-hidden mt-3"}
>
<ScrollArea.Viewport className={"h-full w-full"}>
<NetworksList data={filtered} onToggle={toggleNetwork} />
{filtered.length === 0 ? (
<NoResults />
) : (
<NetworksList
data={filtered}
onToggle={toggleNetwork}
/>
)}
</ScrollArea.Viewport>
<ScrollArea.Scrollbar
orientation={"vertical"}

View File

@@ -1,10 +1,47 @@
import * as Popover from "@radix-ui/react-popover";
import { GlobeIcon, NetworkIcon, WorkflowIcon } from "lucide-react";
import { useTranslation } from "react-i18next";
import type { Network } from "@bindings/services/models.js";
import { cn } from "@/lib/cn";
import { CopyToClipboard } from "@/components/CopyToClipboard";
const dotClass = (selected: boolean): string =>
selected ? "bg-green-400" : "bg-nb-gray-500";
// The daemon stringifies route.Network via netip.Prefix.String(). For
// DNS-based routes the prefix is the zero value, which Go renders as
// "invalid Prefix". Those rows render their domain + resolved IPs instead.
const INVALID_PREFIX = "invalid Prefix";
const isDnsRoute = (n: Network): boolean =>
n.domains.length > 0 && (!n.range || n.range === INVALID_PREFIX);
// Mirror management's NetworkResourceType (resource.go GetResourceType):
// a CIDR is a host when its prefix length equals the address width
// (32 for IPv4, 128 for IPv6); anything broader is a subnet. Routes with
// domains attached are domain resources.
type ResourceType = "host" | "subnet" | "domain";
const isHostCidr = (cidr: string): boolean => {
const [addr, bitsStr] = cidr.split("/");
if (!addr || !bitsStr) return false;
const bits = Number(bitsStr);
// IPv6 prefixes always contain ':'; IPv4 prefixes always contain '.'.
const isV6 = addr.includes(":");
return isV6 ? bits === 128 : bits === 32;
};
const resourceTypeOf = (n: Network): ResourceType => {
if (isDnsRoute(n)) return "domain";
// n.range is a single CIDR for resource routes. Exit-node v4+v6 pairs
// come comma-joined, but those are filtered out upstream — guard
// defensively by inspecting only the first segment.
const primary = n.range.split(",")[0].trim();
return isHostCidr(primary) ? "host" : "subnet";
};
const ResourceIcon = ({ type }: { type: ResourceType }) => {
if (type === "host") return <WorkflowIcon size={15} />;
if (type === "domain") return <GlobeIcon size={15} />;
return <NetworkIcon size={15} />;
};
type Props = {
data: Network[];
@@ -13,13 +50,6 @@ type Props = {
export const NetworksList = ({ data, onToggle }: Props) => {
const { t } = useTranslation();
if (data.length === 0) {
return (
<div className={"py-12 text-center text-sm text-nb-gray-400"}>
{t("networks.empty")}
</div>
);
}
return (
<ul className={"flex flex-col"}>
@@ -28,30 +58,177 @@ export const NetworksList = ({ data, onToggle }: Props) => {
key={n.id}
className={"flex items-center gap-3 px-7 py-3 min-w-0"}
>
<button
type={"button"}
onClick={() => onToggle(n.id, n.selected)}
className={cn(
"h-2 w-2 rounded-full shrink-0 cursor-pointer",
dotClass(n.selected),
)}
title={n.selected ? t("networks.selected") : t("networks.unselected")}
<NetworkToggle
checked={n.selected}
onChange={() => onToggle(n.id, n.selected)}
label={
n.selected
? t("networks.selected")
: t("networks.unselected")
}
/>
<CopyToClipboard message={n.id} className={"min-w-0 flex-1"}>
<span className={"text-[0.81rem] font-medium text-nb-gray-100"}>
{n.id}
</span>
</CopyToClipboard>
<CopyToClipboard
message={n.range}
className={cn("ml-auto shrink-0", "relative left-2.5")}
>
<span className={"text-xs font-mono text-nb-gray-400"}>
{n.range}
</span>
</CopyToClipboard>
<span className={"shrink-0 text-nb-gray-400"}>
<ResourceIcon type={resourceTypeOf(n)} />
</span>
<div className={"min-w-0 flex-1 flex flex-col gap-0.5"}>
<CopyToClipboard message={n.id}>
<span
className={
"text-[0.81rem] font-medium text-nb-gray-100"
}
>
{n.id}
</span>
</CopyToClipboard>
<Subtitle network={n} />
</div>
</li>
))}
</ul>
);
};
const Subtitle = ({ network }: { network: Network }) => {
if (isDnsRoute(network)) {
const domain = network.domains[0];
const ips = network.resolvedIps[domain] ?? [];
return <DomainSubtitle domain={domain} ips={ips} />;
}
if (network.range && network.range !== INVALID_PREFIX) {
return (
<CopyToClipboard message={network.range}>
<span
className={
"text-xs font-mono text-nb-gray-400 truncate"
}
>
{network.range}
</span>
</CopyToClipboard>
);
}
return null;
};
type DomainSubtitleProps = {
domain: string;
ips: string[];
};
const DomainSubtitle = ({ domain, ips }: DomainSubtitleProps) => {
const first = ips[0];
const extra = ips.length - 1;
return (
<div className={"flex items-center gap-1.5 text-xs min-w-0"}>
<CopyToClipboard message={domain}>
<span className={"font-mono text-nb-gray-400 truncate"}>
{domain}
</span>
</CopyToClipboard>
{first && (
<>
<span className={"text-nb-gray-600"}>·</span>
<CopyToClipboard message={first}>
<span
className={"font-mono text-nb-gray-500 truncate"}
>
{first}
</span>
</CopyToClipboard>
{extra > 0 && <ResolvedIpsPopover ips={ips} />}
</>
)}
</div>
);
};
const ResolvedIpsPopover = ({ ips }: { ips: string[] }) => {
const { t } = useTranslation();
const extra = ips.length - 1;
return (
<Popover.Root>
<Popover.Trigger asChild>
<button
type={"button"}
className={cn(
"shrink-0 rounded bg-nb-gray-900 hover:bg-nb-gray-850",
"px-1.5 py-0.5 text-[10px] font-medium text-nb-gray-300",
"wails-no-draggable cursor-pointer outline-none",
)}
>
{t("networks.ips.more", { count: extra })}
</button>
</Popover.Trigger>
<Popover.Portal>
<Popover.Content
side={"bottom"}
align={"start"}
sideOffset={6}
className={cn(
"z-50 w-64 max-h-72 overflow-auto",
"rounded-lg border border-nb-gray-900 bg-nb-gray-935",
"p-2 shadow-lg outline-none",
)}
>
<div
className={
"px-1 pb-1 text-[10px] uppercase tracking-wide text-nb-gray-500"
}
>
{t("networks.ips.heading")}
</div>
<ul className={"flex flex-col"}>
{ips.map((ip) => (
<li key={ip}>
<CopyToClipboard
message={ip}
className={"px-1 py-0.5"}
>
<span
className={
"font-mono text-[0.72rem] text-nb-gray-200 break-all"
}
>
{ip}
</span>
</CopyToClipboard>
</li>
))}
</ul>
</Popover.Content>
</Popover.Portal>
</Popover.Root>
);
};
type ToggleProps = {
checked: boolean;
onChange: () => void;
label: string;
};
const NetworkToggle = ({ checked, onChange, label }: ToggleProps) => (
<button
type={"button"}
role={"switch"}
aria-checked={checked}
aria-label={label}
onClick={onChange}
className={cn(
"shrink-0 inline-flex h-4 w-7 items-center rounded-full",
"transition-colors cursor-pointer wails-no-draggable",
checked ? "bg-netbird" : "bg-nb-gray-700",
)}
>
<span
className={cn(
"inline-block h-3 w-3 rounded-full bg-white transition-transform",
checked ? "translate-x-3.5" : "translate-x-0.5",
)}
/>
</button>
);

View File

@@ -0,0 +1,28 @@
import { createContext, useContext, useState, type ReactNode } from "react";
import type { PeerStatus } from "@bindings/services/models.js";
type PeerDetailContextValue = {
selected: PeerStatus | null;
setSelected: (p: PeerStatus | null) => void;
};
const PeerDetailContext = createContext<PeerDetailContextValue | null>(null);
export const usePeerDetail = (): PeerDetailContextValue => {
const ctx = useContext(PeerDetailContext);
if (!ctx) {
throw new Error(
"usePeerDetail must be used inside PeerDetailProvider",
);
}
return ctx;
};
export const PeerDetailProvider = ({ children }: { children: ReactNode }) => {
const [selected, setSelected] = useState<PeerStatus | null>(null);
return (
<PeerDetailContext.Provider value={{ selected, setSelected }}>
{children}
</PeerDetailContext.Provider>
);
};

View File

@@ -0,0 +1,122 @@
import { useEffect } from "react";
import { useTranslation } from "react-i18next";
import { AnimatePresence, motion, type Transition } from "framer-motion";
import * as ScrollArea from "@radix-ui/react-scroll-area";
import { ArrowLeftIcon } from "lucide-react";
import { cn } from "@/lib/cn";
import { useStatus } from "@/modules/daemon-status/StatusContext";
import { PeerDetails } from "./PeerDetails";
import { usePeerDetail } from "./PeerDetailContext";
const DEFAULT_TRANSITION: Transition = {
duration: 0.32,
ease: [0.32, 0.72, 0, 1],
};
const dotClass = (connStatus: string): string => {
switch (connStatus) {
case "Connected":
return "bg-green-400";
case "Connecting":
return "bg-yellow-300 animate-pulse-slow";
default:
return "bg-nb-gray-500";
}
};
type Props = {
transition?: Transition;
};
export const PeerDetailPanel = ({ transition = DEFAULT_TRANSITION }: Props) => {
const { t } = useTranslation();
const { selected, setSelected } = usePeerDetail();
const { status } = useStatus();
// Keep `selected` in sync with the live peer list so the panel reflects
// status / latency / byte updates without re-opening. If the peer
// disappears, close the panel.
useEffect(() => {
if (!selected) return;
const peers = status?.peers ?? [];
const fresh = peers.find((p) => p.pubKey === selected.pubKey);
if (!fresh) {
setSelected(null);
return;
}
if (fresh !== selected) setSelected(fresh);
}, [status, selected, setSelected]);
// Esc closes the panel.
useEffect(() => {
if (!selected) return;
const onKey = (e: KeyboardEvent) => {
if (e.key === "Escape") setSelected(null);
};
window.addEventListener("keydown", onKey);
return () => window.removeEventListener("keydown", onKey);
}, [selected, setSelected]);
return (
<AnimatePresence>
{selected && (
<motion.div
initial={{ x: "100%" }}
animate={{ x: 0 }}
exit={{ x: "100%" }}
transition={transition}
className={cn("absolute inset-0 z-20 flex flex-col", "bg-nb-gray-940")}
>
<div
className={cn(
"shrink-0 flex items-center gap-3",
"px-3 h-12 border-b border-nb-gray-910",
)}
>
<button
type={"button"}
onClick={() => setSelected(null)}
aria-label={t("common.close")}
className={cn(
"shrink-0 h-8 w-8 rounded-md flex items-center justify-center",
"text-nb-gray-300 hover:bg-nb-gray-910 hover:text-nb-gray-100",
"transition-colors outline-none cursor-default",
"wails-no-draggable",
)}
>
<ArrowLeftIcon size={16} />
</button>
<span
className={cn(
"h-2 w-2 rounded-full shrink-0",
dotClass(selected.connStatus),
)}
title={selected.connStatus}
/>
<span className={"min-w-0 text-sm font-medium text-nb-gray-100 truncate"}>
{selected.fqdn || selected.ip}
</span>
</div>
<ScrollArea.Root type={"auto"} className={"flex-1 min-h-0 overflow-hidden"}>
<ScrollArea.Viewport className={"h-full w-full"}>
<PeerDetails peer={selected} />
</ScrollArea.Viewport>
<ScrollArea.Scrollbar
orientation={"vertical"}
className={cn(
"flex select-none touch-none transition-colors",
"w-1.5 bg-transparent py-1",
)}
>
<ScrollArea.Thumb
className={
"flex-1 rounded-full bg-nb-gray-800 hover:bg-nb-gray-700 relative"
}
/>
</ScrollArea.Scrollbar>
</ScrollArea.Root>
</motion.div>
)}
</AnimatePresence>
);
};

View File

@@ -0,0 +1,198 @@
import { ComponentType, ReactNode } from "react";
import { useTranslation } from "react-i18next";
import {
ArrowDownIcon,
ArrowUpDownIcon,
ArrowUpIcon,
CableIcon,
ClockIcon,
GaugeIcon,
HandshakeIcon,
KeyRoundIcon,
Layers3Icon,
LucideProps,
MapPinIcon,
MonitorIcon,
NetworkIcon,
Radio,
ZapIcon,
} from "lucide-react";
import type { PeerStatus } from "@bindings/services/models.js";
import { cn } from "@/lib/cn";
import { CopyToClipboard } from "@/components/CopyToClipboard";
import { formatBytes, formatRelative, latencyColor } from "./format";
type Props = {
peer: PeerStatus;
};
const DASH = "-";
export const PeerDetails = ({ peer }: Props) => {
const { t } = useTranslation();
const lastHandshake = formatRelative(peer.lastHandshakeUnix) ?? t("peers.details.never");
const statusSince = formatRelative(peer.connStatusUpdateUnix) ?? DASH;
const isConnected = peer.connStatus === "Connected";
const ConnectionIcon = peer.relayed ? NetworkIcon : ZapIcon;
const connectionLabel = peer.relayed ? t("peers.details.relayed") : t("peers.details.p2p");
return (
<ul className={"flex flex-col divide-y divide-nb-gray-920"}>
<Row icon={MapPinIcon} label={t("peers.details.netbirdIp")}>
{peer.ip ? (
<CopyToClipboard
message={peer.ip}
alwaysShowIcon
className={"max-w-full"}
iconClassName={"top-0"}
>
<span className={"font-mono"}>{peer.ip}</span>
</CopyToClipboard>
) : (
DASH
)}
</Row>
{isConnected && (
<Row icon={CableIcon} label={t("peers.details.connection")}>
<span className={"inline-flex items-center gap-1.5 whitespace-nowrap"}>
<ConnectionIcon size={13} />
{connectionLabel}
</span>
</Row>
)}
{peer.latencyMs > 0 && (
<Row icon={GaugeIcon} label={t("peers.details.latency")}>
<span className={cn("tabular-nums", latencyColor(peer.latencyMs))}>
{peer.latencyMs} ms
</span>
</Row>
)}
{(peer.bytesRx > 0 || peer.bytesTx > 0) && (
<Row icon={ArrowUpDownIcon} label={t("peers.details.bytes")}>
<div
className={
"flex items-center gap-3 justify-end text-nb-gray-300 font-medium"
}
>
<div className={"flex gap-1.5 items-center whitespace-nowrap"}>
<ArrowDownIcon size={13} className={"text-sky-400"} />
<span className={"sr-only"}>{t("peers.details.bytesReceived")}:</span>
<span className={"tabular-nums"}>{formatBytes(peer.bytesRx)}</span>
</div>
<div className={"flex gap-1.5 items-center whitespace-nowrap"}>
<ArrowUpIcon size={13} className={"text-netbird"} />
<span className={"sr-only"}>{t("peers.details.bytesSent")}:</span>
<span className={"tabular-nums"}>{formatBytes(peer.bytesTx)}</span>
</div>
</div>
</Row>
)}
<Row icon={HandshakeIcon} label={t("peers.details.lastHandshake")}>
{lastHandshake}
</Row>
<Row icon={ClockIcon} label={t("peers.details.statusSince")}>
{statusSince}
</Row>
<IceRow
icon={MonitorIcon}
baseLabel={t("peers.details.localIce")}
type={peer.localIceCandidateType}
endpoint={peer.localIceCandidateEndpoint}
/>
<IceRow
icon={Radio}
baseLabel={t("peers.details.remoteIce")}
type={peer.remoteIceCandidateType}
endpoint={peer.remoteIceCandidateEndpoint}
/>
{peer.relayed && (
<Row icon={NetworkIcon} label={t("peers.details.relayAddress")}>
{peer.relayAddress ? (
<CopyToClipboard
message={peer.relayAddress}
alwaysShowIcon
className={"max-w-full"}
iconClassName={"top-0"}
>
<span className={"font-mono truncate"}>{peer.relayAddress}</span>
</CopyToClipboard>
) : (
DASH
)}
</Row>
)}
{peer.networks.length > 0 && (
<Row icon={Layers3Icon} label={t("peers.details.networks")}>
<span className={"break-words"}>{peer.networks.join(", ")}</span>
</Row>
)}
<Row icon={KeyRoundIcon} label={t("peers.details.publicKey")}>
{peer.pubKey ? (
<CopyToClipboard
message={peer.pubKey}
alwaysShowIcon
className={"max-w-full"}
iconClassName={"top-0"}
>
<span className={"font-mono truncate"}>{peer.pubKey}</span>
</CopyToClipboard>
) : (
DASH
)}
</Row>
</ul>
);
};
type RowProps = {
icon: ComponentType<LucideProps>;
iconClassName?: string;
label: string;
children: ReactNode;
};
type IceRowProps = {
icon: ComponentType<LucideProps>;
baseLabel: string;
type: string;
endpoint: string;
};
const capitalize = (s: string): string => (s ? s[0].toUpperCase() + s.slice(1) : s);
const IceRow = ({ icon, baseLabel, type, endpoint }: IceRowProps) => {
if (!type && !endpoint) return null;
const label = type ? `${baseLabel} (${capitalize(type)})` : baseLabel;
return (
<Row icon={icon} label={label}>
{endpoint ? (
<CopyToClipboard
message={endpoint}
alwaysShowIcon
className={"max-w-full"}
iconClassName={"top-0"}
>
<span className={"font-mono truncate"}>{endpoint}</span>
</CopyToClipboard>
) : (
<span className={"truncate"}>{capitalize(type)}</span>
)}
</Row>
);
};
const Row = ({ icon: Icon, iconClassName, label, children }: RowProps) => (
<li className={"flex items-center gap-2 px-5 py-4 text-xs text-nb-gray-100 min-w-0"}>
<Icon size={14} className={cn("text-nb-gray-100 shrink-0", iconClassName)} />
<span className={"text-nb-gray-200 shrink-0 font-semibold"}>{label}</span>
<span
className={cn(
"min-w-0 flex-1 text-right pl-8",
"text-nb-gray-350 font-medium",
"flex justify-end items-center",
)}
>
{children}
</span>
</li>
);

View File

@@ -8,9 +8,10 @@ type Props = {
value: StatusFilter;
onChange: (value: StatusFilter) => void;
counts: Record<StatusFilter, number>;
disabled?: boolean;
};
export const PeerFilters = ({ value, onChange, counts }: Props) => {
export const PeerFilters = ({ value, onChange, counts, disabled }: Props) => {
const { t, i18n } = useTranslation();
const filters: { value: StatusFilter; label: string }[] = [
{ value: "all", label: t("peers.filter.all") },
@@ -23,6 +24,7 @@ export const PeerFilters = ({ value, onChange, counts }: Props) => {
key={i18n.language}
value={value}
onChange={(v) => onChange(v as StatusFilter)}
disabled={disabled}
className={"w-full"}
>
{filters.map((f) => (

View File

@@ -1,14 +1,24 @@
import { useEffect, useMemo, useRef, useState } from "react";
import { useTranslation } from "react-i18next";
import * as ScrollArea from "@radix-ui/react-scroll-area";
import { LaptopIcon } from "lucide-react";
import { cn } from "@/lib/cn";
import { SearchInput } from "@/components/SearchInput";
import { EmptyState } from "@/components/EmptyState";
import { NoResults } from "@/components/NoResults";
import { NotConnectedState } from "@/components/NotConnectedState";
import { useStatus } from "@/modules/daemon-status/StatusContext";
import {
formatShortcut,
useKeyboardShortcut,
} from "@/lib/useKeyboardShortcut";
import { PeerFilters, StatusFilter } from "./PeerFilters";
import { PeersList } from "./PeersList";
const isOnline = (connStatus: string) => connStatus === "Connected";
const SEARCH_SHORTCUT = { key: "k", cmd: true } as const;
export const Peers = () => {
const { t } = useTranslation();
const { status } = useStatus();
@@ -23,6 +33,12 @@ export const Peers = () => {
searchRef.current?.focus();
}, []);
useKeyboardShortcut(SEARCH_SHORTCUT, () => {
searchRef.current?.focus();
searchRef.current?.select();
});
const isConnected = status?.status === "Connected";
const peers = status?.peers ?? [];
const counts = useMemo<Record<StatusFilter, number>>(() => {
@@ -54,6 +70,32 @@ export const Peers = () => {
});
}, [peers, search, statusFilter]);
if (!isConnected) {
return (
<div className={"flex flex-col w-full h-full min-h-0"}>
<NotConnectedState />
</div>
);
}
if (peers.length === 0) {
return (
<div
className={
"flex-1 flex items-center justify-center px-6 w-full h-full min-h-0"
}
>
<EmptyState
icon={LaptopIcon}
title={t("peers.empty.title")}
description={t("peers.empty.description")}
learnMoreUrl={"https://docs.netbird.io/how-to/getting-started"}
learnMoreTopic={t("nav.peers.title")}
/>
</div>
);
}
return (
<div className={"flex flex-col w-full h-full min-h-0 pt-4"}>
<div className={"flex flex-col gap-3 px-6"}>
@@ -62,6 +104,7 @@ export const Peers = () => {
placeholder={t("peers.search.placeholder")}
value={search}
onChange={(e) => setSearch(e.target.value)}
shortcut={formatShortcut(SEARCH_SHORTCUT)}
/>
<PeerFilters
value={statusFilter}
@@ -74,7 +117,11 @@ export const Peers = () => {
className={"flex-1 min-h-0 overflow-hidden mt-3"}
>
<ScrollArea.Viewport className={"h-full w-full"}>
<PeersList data={filtered} />
{filtered.length === 0 ? (
<NoResults />
) : (
<PeersList data={filtered} />
)}
</ScrollArea.Viewport>
<ScrollArea.Scrollbar
orientation={"vertical"}

View File

@@ -1,7 +1,9 @@
import { useTranslation } from "react-i18next";
import { ChevronRightIcon } from "lucide-react";
import type { PeerStatus } from "@bindings/services/models.js";
import { cn } from "@/lib/cn";
import { CopyToClipboard } from "@/components/CopyToClipboard";
import { latencyColor } from "./format";
import { usePeerDetail } from "./PeerDetailContext";
const dotClass = (connStatus: string): string => {
switch (connStatus) {
@@ -15,34 +17,69 @@ const dotClass = (connStatus: string): string => {
};
export const PeersList = ({ data }: { data: PeerStatus[] }) => {
const { t } = useTranslation();
if (data.length === 0) {
return (
<div className={"py-12 text-center text-sm text-nb-gray-400"}>{t("peers.empty")}</div>
);
}
const { setSelected } = usePeerDetail();
return (
<ul className={"flex flex-col"}>
{data.map((peer) => (
<li key={peer.pubKey} className={"flex items-center gap-3 px-7 py-3 min-w-0"}>
<span
className={cn("h-2 w-2 rounded-full shrink-0", dotClass(peer.connStatus))}
title={peer.connStatus}
/>
<CopyToClipboard message={peer.fqdn} className={"min-w-0 flex-1"}>
<span className={"text-[0.81rem] font-medium text-nb-gray-100"}>
{peer.fqdn}
</span>
</CopyToClipboard>
<CopyToClipboard
message={peer.ip}
className={cn("ml-auto shrink-0", "relative left-2.5")}
{data.map((peer) => {
const isConnected = peer.connStatus === "Connected";
return (
<li
key={peer.pubKey}
onClick={() => setSelected(peer)}
className={cn(
"group flex items-start gap-2.5 px-7 py-3 min-w-0",
"hover:bg-nb-gray-900/40 transition-colors",
"wails-no-draggable cursor-pointer",
)}
>
<span className={"text-xs font-mono text-nb-gray-400"}>{peer.ip}</span>
</CopyToClipboard>
</li>
))}
<span
className={cn(
"h-2 w-2 rounded-full shrink-0 mt-2",
dotClass(peer.connStatus),
)}
title={peer.connStatus}
/>
<div className={"min-w-0 flex-1 flex flex-col leading-tight"}>
<div>
<CopyToClipboard message={peer.fqdn}>
<span
className={
"text-[0.81rem] font-medium text-nb-gray-100 truncate"
}
>
{peer.fqdn}
</span>
</CopyToClipboard>
</div>
<div>
<CopyToClipboard message={peer.ip}>
<span className={"text-xs font-mono text-nb-gray-400 truncate"}>
{peer.ip}
</span>
</CopyToClipboard>
</div>
</div>
{isConnected && peer.latencyMs > 0 && (
<span
className={cn(
"shrink-0 self-center text-xs tabular-nums",
latencyColor(peer.latencyMs),
)}
>
{peer.latencyMs} ms
</span>
)}
<ChevronRightIcon
size={16}
className={cn(
"shrink-0 self-center text-nb-gray-300",
"opacity-0 group-hover:opacity-100 transition-opacity",
)}
/>
</li>
);
})}
</ul>
);
};

View File

@@ -0,0 +1,35 @@
export const formatBytes = (bytes: number, decimals: number = 2): string => {
try {
if (bytes === 0) return "0 B";
const k = 1024;
const sizes = ["B", "KB", "MB", "GB", "TB"];
const i = Math.floor(Math.log(bytes) / Math.log(k));
return (
parseFloat((bytes / Math.pow(k, i)).toFixed(decimals)) +
" " +
sizes[i]
);
} catch {
return "0 B";
}
};
export const latencyColor = (ms: number): string => {
if (ms <= 0) return "text-nb-gray-400";
if (ms < 100) return "text-green-400";
return "text-yellow-400";
};
export const formatRelative = (
unixSeconds: number,
nowMs: number = Date.now(),
): string | null => {
if (!Number.isFinite(unixSeconds) || unixSeconds <= 0) return null;
const diff = Math.max(0, Math.floor(nowMs / 1000 - unixSeconds));
if (diff < 60) return `${diff}s ago`;
if (diff < 3600) return `${Math.floor(diff / 60)}m ago`;
if (diff < 86400) return `${Math.floor(diff / 3600)}h ago`;
return `${Math.floor(diff / 86400)}d ago`;
};

View File

@@ -61,6 +61,11 @@
"common.refresh": "Aktualisieren",
"common.loading": "Lädt…",
"common.netbird": "NetBird",
"common.learnMoreAbout": "Mehr erfahren über",
"common.noResults.title": "Keine Ergebnisse gefunden",
"common.noResults.description": "Es konnten keine Ergebnisse gefunden werden. Bitte versuchen Sie es mit einem anderen Suchbegriff oder ändern Sie Ihre Filter.",
"notConnected.title": "Nicht verbunden",
"notConnected.description": "Verbinden Sie sich mit NetBird, um Peers, Netzwerke und Exit Nodes anzuzeigen.",
"connect.status.disconnected": "Getrennt",
"connect.status.connecting": "Verbindet…",
@@ -74,11 +79,12 @@
"connect.error.disconnectTitle": "Trennen fehlgeschlagen",
"nav.peers.title": "Peers",
"nav.peers.description": "{online} von {total} online",
"nav.peers.description": "{connected} von {total} verbunden",
"nav.resources.title": "Ressourcen",
"nav.resources.description": "{active} von {total} aktiv",
"nav.exitNode.title": "Exit-Node",
"nav.exitNode.none": "{total} verfügbar",
"nav.exitNode.none": "Nicht aktiv",
"nav.exitNode.using": "Über {name}",
"header.openSettings": "Einstellungen öffnen",
"header.togglePanel": "Seitenleiste umschalten",
@@ -303,20 +309,42 @@
"peers.filter.all": "Alle",
"peers.filter.online": "Online",
"peers.filter.offline": "Offline",
"peers.empty": "Keine Peers entsprechen den aktuellen Filtern.",
"peers.empty.title": "Noch keine Peers",
"peers.empty.description": "Es sind keine Peers mit Ihrem Netzwerk verbunden. Fügen Sie einen Peer hinzu, um zu beginnen.",
"peers.details.domain": "Domain",
"peers.details.netbirdIp": "NetBird-IP",
"peers.details.publicKey": "Öffentlicher Schlüssel",
"peers.details.connection": "Verbindung",
"peers.details.latency": "Latenz",
"peers.details.lastHandshake": "Letzter Handshake",
"peers.details.statusSince": "Letzte Verbindungsaktualisierung",
"peers.details.bytes": "Bytes",
"peers.details.bytesSent": "Gesendet",
"peers.details.bytesReceived": "Empfangen",
"peers.details.localIce": "Lokales ICE",
"peers.details.remoteIce": "Remote ICE",
"peers.details.never": "Nie",
"peers.details.relayAddress": "Relay",
"peers.details.networks": "Netzwerke",
"peers.details.relayed": "Relayed",
"peers.details.p2p": "P2P",
"peers.details.rosenpass": "Rosenpass aktiviert",
"networks.search.placeholder": "Nach Netzwerk, Bereich oder Domain suchen",
"networks.filter.all": "Alle",
"networks.filter.selected": "Ausgewählt",
"networks.filter.active": "Aktiv",
"networks.filter.overlapping": "Überlappend",
"networks.empty": "Keine Netzwerke entsprechen den aktuellen Filtern.",
"networks.empty.title": "Keine Netzwerke verfügbar",
"networks.empty.description": "Für diesen Peer wurden keine geleiteten Netzwerke freigegeben.",
"networks.selected": "Ausgewählt",
"networks.unselected": "Nicht ausgewählt",
"networks.ips.more": "+{count} weitere",
"networks.ips.heading": "Aufgelöste IPs",
"exitNodes.search.placeholder": "Exit Nodes suchen",
"exitNodes.empty": "Keine Exit Nodes verfügbar.",
"exitNodes.active": "Aktiv",
"exitNodes.inactive": "Inaktiv",
"exitNodes.none": "Keiner",
"exitNodes.empty.title": "Keine Exit Nodes verfügbar",
"exitNodes.empty.description": "Für diesen Peer wurden keine Exit Nodes freigegeben.",
"quickActions.connect": "Verbinden",
"quickActions.disconnect": "Trennen",

View File

@@ -61,6 +61,11 @@
"common.refresh": "Refresh",
"common.loading": "Loading…",
"common.netbird": "NetBird",
"common.learnMoreAbout": "Learn more about",
"common.noResults.title": "Could not find any results",
"common.noResults.description": "We couldn't find any results. Please try a different search term or change your filters.",
"notConnected.title": "Not connected",
"notConnected.description": "Connect to NetBird to view peers, networks, and exit nodes.",
"connect.status.disconnected": "Disconnected",
"connect.status.connecting": "Connecting...",
@@ -74,11 +79,12 @@
"connect.error.disconnectTitle": "Disconnect Failed",
"nav.peers.title": "Peers",
"nav.peers.description": "{online} of {total} Online",
"nav.peers.description": "{connected} of {total} connected",
"nav.resources.title": "Resources",
"nav.resources.description": "{active} of {total} Active",
"nav.resources.description": "{active} of {total} active",
"nav.exitNode.title": "Exit Node",
"nav.exitNode.none": "{total} available",
"nav.exitNode.none": "Not active",
"nav.exitNode.using": "Via {name}",
"header.openSettings": "Open settings",
"header.togglePanel": "Toggle side panel",
@@ -319,20 +325,42 @@
"peers.filter.all": "All",
"peers.filter.online": "Online",
"peers.filter.offline": "Offline",
"peers.empty": "No peers match the current filters.",
"peers.empty.title": "No peers yet",
"peers.empty.description": "No peers are connected to your network. Add a peer to get started.",
"peers.details.domain": "Domain",
"peers.details.netbirdIp": "NetBird IP",
"peers.details.publicKey": "Public key",
"peers.details.connection": "Connection",
"peers.details.latency": "Latency",
"peers.details.lastHandshake": "Last handshake",
"peers.details.statusSince": "Last connection update",
"peers.details.bytes": "Bytes",
"peers.details.bytesSent": "Sent",
"peers.details.bytesReceived": "Received",
"peers.details.localIce": "Local ICE",
"peers.details.remoteIce": "Remote ICE",
"peers.details.never": "Never",
"peers.details.relayAddress": "Relay",
"peers.details.networks": "Networks",
"peers.details.relayed": "Relayed",
"peers.details.p2p": "P2P",
"peers.details.rosenpass": "Rosenpass enabled",
"networks.search.placeholder": "Search by network, range or domain",
"networks.filter.all": "All",
"networks.filter.selected": "Selected",
"networks.filter.active": "Active",
"networks.filter.overlapping": "Overlapping",
"networks.empty": "No networks match the current filters.",
"networks.empty.title": "No networks available",
"networks.empty.description": "No routed networks have been shared with this peer.",
"networks.selected": "Selected",
"networks.unselected": "Not selected",
"networks.ips.more": "+{count} more",
"networks.ips.heading": "Resolved IPs",
"exitNodes.search.placeholder": "Search exit nodes",
"exitNodes.empty": "No exit nodes available.",
"exitNodes.active": "Active",
"exitNodes.inactive": "Inactive",
"exitNodes.none": "None",
"exitNodes.empty.title": "No exit nodes available",
"exitNodes.empty.description": "No exit nodes have been shared with this peer.",
"quickActions.connect": "Connect",
"quickActions.disconnect": "Disconnect",

View File

@@ -61,6 +61,11 @@
"common.refresh": "Frissítés",
"common.loading": "Betöltés…",
"common.netbird": "NetBird",
"common.learnMoreAbout": "Tudjon meg többet erről:",
"common.noResults.title": "Nincs találat",
"common.noResults.description": "Nem találtunk eredményt. Próbáljon meg másik keresési kifejezést, vagy módosítsa a szűrőket.",
"notConnected.title": "Nincs csatlakozva",
"notConnected.description": "Csatlakozzon a NetBirdhöz a társak, hálózatok és kilépő csomópontok megtekintéséhez.",
"connect.status.disconnected": "Lekapcsolva",
"connect.status.connecting": "Csatlakozás…",
@@ -74,11 +79,12 @@
"connect.error.disconnectTitle": "Bontás sikertelen",
"nav.peers.title": "Társak",
"nav.peers.description": "{online} / {total} online",
"nav.peers.description": "{connected} / {total} csatlakoztatva",
"nav.resources.title": "Erőforrások",
"nav.resources.description": "{active} / {total} aktív",
"nav.exitNode.title": "Kilépő csomópont",
"nav.exitNode.none": "{total} elérhető",
"nav.exitNode.none": "Nem aktív",
"nav.exitNode.using": "Ezen át: {name}",
"header.openSettings": "Beállítások megnyitása",
"header.togglePanel": "Oldalsó panel váltása",
@@ -303,20 +309,42 @@
"peers.filter.all": "Összes",
"peers.filter.online": "Online",
"peers.filter.offline": "Offline",
"peers.empty": "Egyetlen társ sem felel meg a jelenlegi szűrőknek.",
"peers.empty.title": "Még nincsenek társak",
"peers.empty.description": "Egyetlen társ sem csatlakozott a hálózatához. Adjon hozzá egy társat a kezdéshez.",
"peers.details.domain": "Domain",
"peers.details.netbirdIp": "NetBird IP",
"peers.details.publicKey": "Publikus kulcs",
"peers.details.connection": "Kapcsolat",
"peers.details.latency": "Késleltetés",
"peers.details.lastHandshake": "Utolsó kézfogás",
"peers.details.statusSince": "Utolsó kapcsolati frissítés",
"peers.details.bytes": "Bájtok",
"peers.details.bytesSent": "Küldve",
"peers.details.bytesReceived": "Fogadva",
"peers.details.localIce": "Helyi ICE",
"peers.details.remoteIce": "Távoli ICE",
"peers.details.never": "Soha",
"peers.details.relayAddress": "Relay",
"peers.details.networks": "Hálózatok",
"peers.details.relayed": "Relayed",
"peers.details.p2p": "P2P",
"peers.details.rosenpass": "Rosenpass engedélyezve",
"networks.search.placeholder": "Keresés hálózat, tartomány vagy domain alapján",
"networks.filter.all": "Összes",
"networks.filter.selected": "Kiválasztott",
"networks.filter.active": "Aktív",
"networks.filter.overlapping": "Átfedő",
"networks.empty": "Egyetlen hálózat sem felel meg a jelenlegi szűrőknek.",
"networks.empty.title": "Nincs elérhető hálózat",
"networks.empty.description": "Ehhez a társhoz nem osztottak meg útvonalas hálózatokat.",
"networks.selected": "Kiválasztva",
"networks.unselected": "Nincs kiválasztva",
"networks.ips.more": "+{count} további",
"networks.ips.heading": "Feloldott IP-címek",
"exitNodes.search.placeholder": "Keresés a kilépő csomópontok között",
"exitNodes.empty": "Nincs elérhető kilépő csomópont.",
"exitNodes.active": "Aktív",
"exitNodes.inactive": "Inaktív",
"exitNodes.none": "Egyik sem",
"exitNodes.empty.title": "Nincs elérhető kilépő csomópont",
"exitNodes.empty.description": "Ehhez a társhoz nem osztottak meg kilépő csomópontokat.",
"quickActions.connect": "Csatlakozás",
"quickActions.disconnect": "Bontás",