2727 lines
118 KiB
TypeScript
2727 lines
118 KiB
TypeScript
import { zodResolver } from "@hookform/resolvers/zod";
|
||
import { Controller, useForm } from "react-hook-form";
|
||
import { z } from "zod";
|
||
|
||
import { Button } from "@/components/ui/button.tsx";
|
||
import {
|
||
Form,
|
||
FormControl,
|
||
FormDescription,
|
||
FormField,
|
||
FormItem,
|
||
FormLabel,
|
||
} from "@/components/ui/form.tsx";
|
||
import { Input } from "@/components/ui/input.tsx";
|
||
import { PasswordInput } from "@/components/ui/password-input.tsx";
|
||
import { ScrollArea } from "@/components/ui/scroll-area.tsx";
|
||
import { Separator } from "@/components/ui/separator.tsx";
|
||
import {
|
||
Tabs,
|
||
TabsContent,
|
||
TabsList,
|
||
TabsTrigger,
|
||
} from "@/components/ui/tabs.tsx";
|
||
import React, { useEffect, useRef, useState } from "react";
|
||
import { Switch } from "@/components/ui/switch.tsx";
|
||
import { Alert, AlertDescription } from "@/components/ui/alert.tsx";
|
||
import { toast } from "sonner";
|
||
import {
|
||
createSSHHost,
|
||
getCredentials,
|
||
getSSHHosts,
|
||
updateSSHHost,
|
||
enableAutoStart,
|
||
disableAutoStart,
|
||
getSnippets,
|
||
} from "@/ui/main-axios.ts";
|
||
import { useTranslation } from "react-i18next";
|
||
import { CredentialSelector } from "@/ui/desktop/apps/credentials/CredentialSelector.tsx";
|
||
import CodeMirror from "@uiw/react-codemirror";
|
||
import { oneDark } from "@codemirror/theme-one-dark";
|
||
import { EditorView } from "@codemirror/view";
|
||
import type { StatsConfig } from "@/types/stats-widgets";
|
||
import { DEFAULT_STATS_CONFIG } from "@/types/stats-widgets";
|
||
import { Checkbox } from "@/components/ui/checkbox.tsx";
|
||
import {
|
||
Select,
|
||
SelectContent,
|
||
SelectItem,
|
||
SelectTrigger,
|
||
SelectValue,
|
||
} from "@/components/ui/select.tsx";
|
||
import { Slider } from "@/components/ui/slider.tsx";
|
||
import {
|
||
Accordion,
|
||
AccordionContent,
|
||
AccordionItem,
|
||
AccordionTrigger,
|
||
} from "@/components/ui/accordion.tsx";
|
||
import {
|
||
TERMINAL_THEMES,
|
||
TERMINAL_FONTS,
|
||
CURSOR_STYLES,
|
||
BELL_STYLES,
|
||
FAST_SCROLL_MODIFIERS,
|
||
DEFAULT_TERMINAL_CONFIG,
|
||
} from "@/constants/terminal-themes";
|
||
import { TerminalPreview } from "@/ui/desktop/apps/terminal/TerminalPreview.tsx";
|
||
import type { TerminalConfig } from "@/types";
|
||
import { Plus, X } from "lucide-react";
|
||
|
||
interface SSHHost {
|
||
id: number;
|
||
name: string;
|
||
ip: string;
|
||
port: number;
|
||
username: string;
|
||
folder: string;
|
||
tags: string[];
|
||
pin: boolean;
|
||
authType: string;
|
||
password?: string;
|
||
key?: string;
|
||
keyPassword?: string;
|
||
keyType?: string;
|
||
enableTerminal: boolean;
|
||
enableTunnel: boolean;
|
||
enableFileManager: boolean;
|
||
defaultPath: string;
|
||
tunnelConnections: Array<{
|
||
sourcePort: number;
|
||
endpointPort: number;
|
||
endpointHost: string;
|
||
maxRetries: number;
|
||
retryInterval: number;
|
||
autoStart: boolean;
|
||
}>;
|
||
statsConfig?: StatsConfig;
|
||
terminalConfig?: TerminalConfig;
|
||
createdAt: string;
|
||
updatedAt: string;
|
||
credentialId?: number;
|
||
}
|
||
|
||
interface SSHManagerHostEditorProps {
|
||
editingHost?: SSHHost | null;
|
||
onFormSubmit?: (updatedHost?: SSHHost) => void;
|
||
}
|
||
|
||
export function HostManagerEditor({
|
||
editingHost,
|
||
onFormSubmit,
|
||
}: SSHManagerHostEditorProps) {
|
||
const { t } = useTranslation();
|
||
const [folders, setFolders] = useState<string[]>([]);
|
||
const [sshConfigurations, setSshConfigurations] = useState<string[]>([]);
|
||
const [credentials, setCredentials] = useState<
|
||
Array<{ id: number; username: string; authType: string }>
|
||
>([]);
|
||
const [snippets, setSnippets] = useState<
|
||
Array<{ id: number; name: string; content: string }>
|
||
>([]);
|
||
const [snippetSearch, setSnippetSearch] = useState("");
|
||
|
||
const [authTab, setAuthTab] = useState<
|
||
"password" | "key" | "credential" | "none"
|
||
>("password");
|
||
const [keyInputMethod, setKeyInputMethod] = useState<"upload" | "paste">(
|
||
"upload",
|
||
);
|
||
const isSubmittingRef = useRef(false);
|
||
|
||
const [statusIntervalUnit, setStatusIntervalUnit] = useState<
|
||
"seconds" | "minutes"
|
||
>("seconds");
|
||
const [metricsIntervalUnit, setMetricsIntervalUnit] = useState<
|
||
"seconds" | "minutes"
|
||
>("seconds");
|
||
|
||
const ipInputRef = useRef<HTMLInputElement>(null);
|
||
|
||
useEffect(() => {
|
||
const fetchData = async () => {
|
||
try {
|
||
const [hostsData, credentialsData, snippetsData] = await Promise.all([
|
||
getSSHHosts(),
|
||
getCredentials(),
|
||
getSnippets(),
|
||
]);
|
||
setCredentials(credentialsData);
|
||
setSnippets(Array.isArray(snippetsData) ? snippetsData : []);
|
||
|
||
const uniqueFolders = [
|
||
...new Set(
|
||
hostsData
|
||
.filter((host) => host.folder && host.folder.trim() !== "")
|
||
.map((host) => host.folder),
|
||
),
|
||
].sort();
|
||
|
||
const uniqueConfigurations = [
|
||
...new Set(
|
||
hostsData
|
||
.filter((host) => host.name && host.name.trim() !== "")
|
||
.map((host) => host.name),
|
||
),
|
||
].sort();
|
||
|
||
setFolders(uniqueFolders);
|
||
setSshConfigurations(uniqueConfigurations);
|
||
} catch {}
|
||
};
|
||
|
||
fetchData();
|
||
}, []);
|
||
|
||
useEffect(() => {
|
||
const handleCredentialChange = async () => {
|
||
try {
|
||
const hostsData = await getSSHHosts();
|
||
|
||
const uniqueFolders = [
|
||
...new Set(
|
||
hostsData
|
||
.filter((host) => host.folder && host.folder.trim() !== "")
|
||
.map((host) => host.folder),
|
||
),
|
||
].sort();
|
||
|
||
const uniqueConfigurations = [
|
||
...new Set(
|
||
hostsData
|
||
.filter((host) => host.name && host.name.trim() !== "")
|
||
.map((host) => host.name),
|
||
),
|
||
].sort();
|
||
|
||
setFolders(uniqueFolders);
|
||
setSshConfigurations(uniqueConfigurations);
|
||
} catch {}
|
||
};
|
||
|
||
window.addEventListener("credentials:changed", handleCredentialChange);
|
||
|
||
return () => {
|
||
window.removeEventListener("credentials:changed", handleCredentialChange);
|
||
};
|
||
}, []);
|
||
|
||
const formSchema = z
|
||
.object({
|
||
name: z.string().optional(),
|
||
ip: z.string().min(1),
|
||
port: z.coerce.number().min(1).max(65535),
|
||
username: z.string().min(1),
|
||
folder: z.string().optional(),
|
||
tags: z.array(z.string().min(1)).default([]),
|
||
pin: z.boolean().default(false),
|
||
authType: z.enum(["password", "key", "credential", "none"]),
|
||
credentialId: z.number().optional().nullable(),
|
||
overrideCredentialUsername: z.boolean().optional(),
|
||
password: z.string().optional(),
|
||
key: z.any().optional().nullable(),
|
||
keyPassword: z.string().optional(),
|
||
keyType: z
|
||
.enum([
|
||
"auto",
|
||
"ssh-rsa",
|
||
"ssh-ed25519",
|
||
"ecdsa-sha2-nistp256",
|
||
"ecdsa-sha2-nistp384",
|
||
"ecdsa-sha2-nistp521",
|
||
"ssh-dss",
|
||
"ssh-rsa-sha2-256",
|
||
"ssh-rsa-sha2-512",
|
||
])
|
||
.optional(),
|
||
enableTerminal: z.boolean().default(true),
|
||
enableTunnel: z.boolean().default(true),
|
||
tunnelConnections: z
|
||
.array(
|
||
z.object({
|
||
sourcePort: z.coerce.number().min(1).max(65535),
|
||
endpointPort: z.coerce.number().min(1).max(65535),
|
||
endpointHost: z.string().min(1),
|
||
maxRetries: z.coerce.number().min(0).max(100).default(3),
|
||
retryInterval: z.coerce.number().min(1).max(3600).default(10),
|
||
autoStart: z.boolean().default(false),
|
||
}),
|
||
)
|
||
.default([]),
|
||
enableFileManager: z.boolean().default(true),
|
||
defaultPath: z.string().optional(),
|
||
statsConfig: z
|
||
.object({
|
||
enabledWidgets: z
|
||
.array(
|
||
z.enum([
|
||
"cpu",
|
||
"memory",
|
||
"disk",
|
||
"network",
|
||
"uptime",
|
||
"processes",
|
||
"system",
|
||
]),
|
||
)
|
||
.default(["cpu", "memory", "disk", "network", "uptime", "system"]),
|
||
statusCheckEnabled: z.boolean().default(true),
|
||
statusCheckInterval: z.number().min(5).max(3600).default(30),
|
||
metricsEnabled: z.boolean().default(true),
|
||
metricsInterval: z.number().min(5).max(3600).default(30),
|
||
})
|
||
.default({
|
||
enabledWidgets: [
|
||
"cpu",
|
||
"memory",
|
||
"disk",
|
||
"network",
|
||
"uptime",
|
||
"system",
|
||
],
|
||
statusCheckEnabled: true,
|
||
statusCheckInterval: 30,
|
||
metricsEnabled: true,
|
||
metricsInterval: 30,
|
||
}),
|
||
terminalConfig: z
|
||
.object({
|
||
cursorBlink: z.boolean(),
|
||
cursorStyle: z.enum(["block", "underline", "bar"]),
|
||
fontSize: z.number().min(8).max(24),
|
||
fontFamily: z.string(),
|
||
letterSpacing: z.number().min(-2).max(10),
|
||
lineHeight: z.number().min(1.0).max(2.0),
|
||
theme: z.string(),
|
||
scrollback: z.number().min(1000).max(50000),
|
||
bellStyle: z.enum(["none", "sound", "visual", "both"]),
|
||
rightClickSelectsWord: z.boolean(),
|
||
fastScrollModifier: z.enum(["alt", "ctrl", "shift"]),
|
||
fastScrollSensitivity: z.number().min(1).max(10),
|
||
minimumContrastRatio: z.number().min(1).max(21),
|
||
backspaceMode: z.enum(["normal", "control-h"]),
|
||
agentForwarding: z.boolean(),
|
||
environmentVariables: z.array(
|
||
z.object({
|
||
key: z.string(),
|
||
value: z.string(),
|
||
}),
|
||
),
|
||
startupSnippetId: z.number().nullable(),
|
||
autoMosh: z.boolean(),
|
||
moshCommand: z.string(),
|
||
})
|
||
.optional(),
|
||
forceKeyboardInteractive: z.boolean().optional(),
|
||
})
|
||
.superRefine((data, ctx) => {
|
||
if (data.authType === "none") {
|
||
return;
|
||
}
|
||
|
||
if (data.authType === "password") {
|
||
if (
|
||
!data.password ||
|
||
(typeof data.password === "string" && data.password.trim() === "")
|
||
) {
|
||
ctx.addIssue({
|
||
code: z.ZodIssueCode.custom,
|
||
message: t("hosts.passwordRequired"),
|
||
path: ["password"],
|
||
});
|
||
}
|
||
} else if (data.authType === "key") {
|
||
if (
|
||
!data.key ||
|
||
(typeof data.key === "string" && data.key.trim() === "")
|
||
) {
|
||
ctx.addIssue({
|
||
code: z.ZodIssueCode.custom,
|
||
message: t("hosts.sshKeyRequired"),
|
||
path: ["key"],
|
||
});
|
||
}
|
||
if (!data.keyType) {
|
||
ctx.addIssue({
|
||
code: z.ZodIssueCode.custom,
|
||
message: t("hosts.keyTypeRequired"),
|
||
path: ["keyType"],
|
||
});
|
||
}
|
||
} else if (data.authType === "credential") {
|
||
if (
|
||
!data.credentialId ||
|
||
(typeof data.credentialId === "string" &&
|
||
data.credentialId.trim() === "")
|
||
) {
|
||
ctx.addIssue({
|
||
code: z.ZodIssueCode.custom,
|
||
message: t("hosts.credentialRequired"),
|
||
path: ["credentialId"],
|
||
});
|
||
}
|
||
}
|
||
|
||
data.tunnelConnections.forEach((connection, index) => {
|
||
if (
|
||
connection.endpointHost &&
|
||
!sshConfigurations.includes(connection.endpointHost)
|
||
) {
|
||
ctx.addIssue({
|
||
code: z.ZodIssueCode.custom,
|
||
message: t("hosts.mustSelectValidSshConfig"),
|
||
path: ["tunnelConnections", index, "endpointHost"],
|
||
});
|
||
}
|
||
});
|
||
});
|
||
|
||
type FormData = z.infer<typeof formSchema>;
|
||
|
||
const form = useForm<FormData>({
|
||
resolver: zodResolver(formSchema),
|
||
defaultValues: {
|
||
name: "",
|
||
ip: "",
|
||
port: 22,
|
||
username: "",
|
||
folder: "",
|
||
tags: [],
|
||
pin: false,
|
||
authType: "password" as const,
|
||
credentialId: null,
|
||
overrideCredentialUsername: false,
|
||
password: "",
|
||
key: null,
|
||
keyPassword: "",
|
||
keyType: "auto" as const,
|
||
enableTerminal: true,
|
||
enableTunnel: true,
|
||
enableFileManager: true,
|
||
defaultPath: "/",
|
||
tunnelConnections: [],
|
||
statsConfig: DEFAULT_STATS_CONFIG,
|
||
terminalConfig: DEFAULT_TERMINAL_CONFIG,
|
||
forceKeyboardInteractive: false,
|
||
},
|
||
});
|
||
|
||
useEffect(() => {
|
||
if (authTab === "credential") {
|
||
const currentCredentialId = form.getValues("credentialId");
|
||
const overrideUsername = form.getValues("overrideCredentialUsername");
|
||
if (currentCredentialId && !overrideUsername) {
|
||
const selectedCredential = credentials.find(
|
||
(c) => c.id === currentCredentialId,
|
||
);
|
||
if (selectedCredential) {
|
||
form.setValue("username", selectedCredential.username);
|
||
}
|
||
}
|
||
}
|
||
}, [authTab, credentials, form]);
|
||
|
||
useEffect(() => {
|
||
if (editingHost) {
|
||
const cleanedHost = { ...editingHost };
|
||
if (cleanedHost.credentialId && cleanedHost.key) {
|
||
cleanedHost.key = undefined;
|
||
cleanedHost.keyPassword = undefined;
|
||
cleanedHost.keyType = undefined;
|
||
} else if (cleanedHost.credentialId && cleanedHost.password) {
|
||
cleanedHost.password = undefined;
|
||
} else if (cleanedHost.key && cleanedHost.password) {
|
||
cleanedHost.password = undefined;
|
||
}
|
||
|
||
const defaultAuthType = cleanedHost.credentialId
|
||
? "credential"
|
||
: cleanedHost.key
|
||
? "key"
|
||
: cleanedHost.password
|
||
? "password"
|
||
: "none";
|
||
setAuthTab(defaultAuthType);
|
||
|
||
let parsedStatsConfig = DEFAULT_STATS_CONFIG;
|
||
try {
|
||
if (cleanedHost.statsConfig) {
|
||
parsedStatsConfig =
|
||
typeof cleanedHost.statsConfig === "string"
|
||
? JSON.parse(cleanedHost.statsConfig)
|
||
: cleanedHost.statsConfig;
|
||
}
|
||
} catch (error) {
|
||
console.error("Failed to parse statsConfig:", error);
|
||
}
|
||
|
||
parsedStatsConfig = { ...DEFAULT_STATS_CONFIG, ...parsedStatsConfig };
|
||
|
||
const formData = {
|
||
name: cleanedHost.name || "",
|
||
ip: cleanedHost.ip || "",
|
||
port: cleanedHost.port || 22,
|
||
username: cleanedHost.username || "",
|
||
folder: cleanedHost.folder || "",
|
||
tags: cleanedHost.tags || [],
|
||
pin: Boolean(cleanedHost.pin),
|
||
authType: defaultAuthType as "password" | "key" | "credential" | "none",
|
||
credentialId: null,
|
||
overrideCredentialUsername: Boolean(
|
||
cleanedHost.overrideCredentialUsername,
|
||
),
|
||
password: "",
|
||
key: null,
|
||
keyPassword: "",
|
||
keyType: "auto" as const,
|
||
enableTerminal: Boolean(cleanedHost.enableTerminal),
|
||
enableTunnel: Boolean(cleanedHost.enableTunnel),
|
||
enableFileManager: Boolean(cleanedHost.enableFileManager),
|
||
defaultPath: cleanedHost.defaultPath || "/",
|
||
tunnelConnections: cleanedHost.tunnelConnections || [],
|
||
statsConfig: parsedStatsConfig,
|
||
terminalConfig: cleanedHost.terminalConfig || DEFAULT_TERMINAL_CONFIG,
|
||
forceKeyboardInteractive: Boolean(cleanedHost.forceKeyboardInteractive),
|
||
};
|
||
|
||
if (defaultAuthType === "password") {
|
||
formData.password = cleanedHost.password || "";
|
||
} else if (defaultAuthType === "key") {
|
||
formData.key = editingHost.id ? "existing_key" : editingHost.key;
|
||
formData.keyPassword = cleanedHost.keyPassword || "";
|
||
formData.keyType =
|
||
(cleanedHost.keyType as
|
||
| "auto"
|
||
| "ssh-rsa"
|
||
| "ssh-ed25519"
|
||
| "ecdsa-sha2-nistp256"
|
||
| "ecdsa-sha2-nistp384"
|
||
| "ecdsa-sha2-nistp521"
|
||
| "ssh-dss"
|
||
| "ssh-rsa-sha2-256"
|
||
| "ssh-rsa-sha2-512") || "auto";
|
||
} else if (defaultAuthType === "credential") {
|
||
formData.credentialId =
|
||
cleanedHost.credentialId || "existing_credential";
|
||
}
|
||
|
||
form.reset(formData);
|
||
} else {
|
||
setAuthTab("password");
|
||
const defaultFormData = {
|
||
name: "",
|
||
ip: "",
|
||
port: 22,
|
||
username: "",
|
||
folder: "",
|
||
tags: [],
|
||
pin: false,
|
||
authType: "password" as const,
|
||
credentialId: null,
|
||
overrideCredentialUsername: false,
|
||
password: "",
|
||
key: null,
|
||
keyPassword: "",
|
||
keyType: "auto" as const,
|
||
enableTerminal: true,
|
||
enableTunnel: true,
|
||
enableFileManager: true,
|
||
defaultPath: "/",
|
||
tunnelConnections: [],
|
||
statsConfig: DEFAULT_STATS_CONFIG,
|
||
terminalConfig: DEFAULT_TERMINAL_CONFIG,
|
||
forceKeyboardInteractive: false,
|
||
};
|
||
|
||
form.reset(defaultFormData);
|
||
}
|
||
}, [editingHost?.id]);
|
||
|
||
useEffect(() => {
|
||
const focusTimer = setTimeout(() => {
|
||
if (ipInputRef.current) {
|
||
ipInputRef.current.focus();
|
||
}
|
||
}, 300);
|
||
|
||
return () => clearTimeout(focusTimer);
|
||
}, [editingHost]);
|
||
|
||
const onSubmit = async (data: FormData) => {
|
||
try {
|
||
isSubmittingRef.current = true;
|
||
|
||
if (!data.name || data.name.trim() === "") {
|
||
data.name = `${data.username}@${data.ip}`;
|
||
}
|
||
|
||
if (data.statsConfig) {
|
||
const statusInterval = data.statsConfig.statusCheckInterval || 30;
|
||
const metricsInterval = data.statsConfig.metricsInterval || 30;
|
||
|
||
if (statusInterval < 5 || statusInterval > 3600) {
|
||
toast.error(t("hosts.intervalValidation"));
|
||
isSubmittingRef.current = false;
|
||
return;
|
||
}
|
||
|
||
if (metricsInterval < 5 || metricsInterval > 3600) {
|
||
toast.error(t("hosts.intervalValidation"));
|
||
isSubmittingRef.current = false;
|
||
return;
|
||
}
|
||
}
|
||
|
||
const submitData: Record<string, unknown> = {
|
||
name: data.name,
|
||
ip: data.ip,
|
||
port: data.port,
|
||
username: data.username,
|
||
folder: data.folder || "",
|
||
tags: data.tags || [],
|
||
pin: Boolean(data.pin),
|
||
authType: data.authType,
|
||
overrideCredentialUsername: Boolean(data.overrideCredentialUsername),
|
||
enableTerminal: Boolean(data.enableTerminal),
|
||
enableTunnel: Boolean(data.enableTunnel),
|
||
enableFileManager: Boolean(data.enableFileManager),
|
||
defaultPath: data.defaultPath || "/",
|
||
tunnelConnections: data.tunnelConnections || [],
|
||
statsConfig: data.statsConfig || DEFAULT_STATS_CONFIG,
|
||
terminalConfig: data.terminalConfig || DEFAULT_TERMINAL_CONFIG,
|
||
forceKeyboardInteractive: Boolean(data.forceKeyboardInteractive),
|
||
};
|
||
|
||
submitData.credentialId = null;
|
||
submitData.password = null;
|
||
submitData.key = null;
|
||
submitData.keyPassword = null;
|
||
submitData.keyType = null;
|
||
|
||
if (data.authType === "credential") {
|
||
if (
|
||
data.credentialId === "existing_credential" &&
|
||
editingHost &&
|
||
editingHost.id
|
||
) {
|
||
delete submitData.credentialId;
|
||
} else {
|
||
submitData.credentialId = data.credentialId;
|
||
}
|
||
} else if (data.authType === "password") {
|
||
submitData.password = data.password;
|
||
} else if (data.authType === "key") {
|
||
if (data.key instanceof File) {
|
||
const keyContent = await data.key.text();
|
||
submitData.key = keyContent;
|
||
} else if (data.key === "existing_key") {
|
||
delete submitData.key;
|
||
} else {
|
||
submitData.key = data.key;
|
||
}
|
||
submitData.keyPassword = data.keyPassword;
|
||
submitData.keyType = data.keyType;
|
||
}
|
||
|
||
let savedHost;
|
||
if (editingHost && editingHost.id) {
|
||
savedHost = await updateSSHHost(editingHost.id, submitData);
|
||
toast.success(t("hosts.hostUpdatedSuccessfully", { name: data.name }));
|
||
} else {
|
||
savedHost = await createSSHHost(submitData);
|
||
toast.success(t("hosts.hostAddedSuccessfully", { name: data.name }));
|
||
}
|
||
|
||
if (savedHost && savedHost.id && data.tunnelConnections) {
|
||
const hasAutoStartTunnels = data.tunnelConnections.some(
|
||
(tunnel) => tunnel.autoStart,
|
||
);
|
||
|
||
if (hasAutoStartTunnels) {
|
||
try {
|
||
await enableAutoStart(savedHost.id);
|
||
} catch (error) {
|
||
console.warn(
|
||
`Failed to enable AutoStart plaintext cache for SSH host ${savedHost.id}:`,
|
||
error,
|
||
);
|
||
toast.warning(
|
||
t("hosts.autoStartEnableFailed", { name: data.name }),
|
||
);
|
||
}
|
||
} else {
|
||
try {
|
||
await disableAutoStart(savedHost.id);
|
||
} catch (error) {
|
||
console.warn(
|
||
`Failed to disable AutoStart plaintext cache for SSH host ${savedHost.id}:`,
|
||
error,
|
||
);
|
||
}
|
||
}
|
||
}
|
||
|
||
if (onFormSubmit) {
|
||
onFormSubmit(savedHost);
|
||
}
|
||
|
||
window.dispatchEvent(new CustomEvent("ssh-hosts:changed"));
|
||
|
||
const { refreshServerPolling } = await import("@/ui/main-axios.ts");
|
||
refreshServerPolling();
|
||
} catch {
|
||
toast.error(t("hosts.failedToSaveHost"));
|
||
} finally {
|
||
isSubmittingRef.current = false;
|
||
}
|
||
};
|
||
|
||
const [tagInput, setTagInput] = useState("");
|
||
|
||
const [folderDropdownOpen, setFolderDropdownOpen] = useState(false);
|
||
const folderInputRef = useRef<HTMLInputElement>(null);
|
||
const folderDropdownRef = useRef<HTMLDivElement>(null);
|
||
|
||
const folderValue = form.watch("folder");
|
||
const filteredFolders = React.useMemo(() => {
|
||
if (!folderValue) return folders;
|
||
return folders.filter((f) =>
|
||
f.toLowerCase().includes(folderValue.toLowerCase()),
|
||
);
|
||
}, [folderValue, folders]);
|
||
|
||
const handleFolderClick = (folder: string) => {
|
||
form.setValue("folder", folder);
|
||
setFolderDropdownOpen(false);
|
||
};
|
||
|
||
useEffect(() => {
|
||
function handleClickOutside(event: MouseEvent) {
|
||
if (
|
||
folderDropdownRef.current &&
|
||
!folderDropdownRef.current.contains(event.target as Node) &&
|
||
folderInputRef.current &&
|
||
!folderInputRef.current.contains(event.target as Node)
|
||
) {
|
||
setFolderDropdownOpen(false);
|
||
}
|
||
}
|
||
|
||
if (folderDropdownOpen) {
|
||
document.addEventListener("mousedown", handleClickOutside);
|
||
} else {
|
||
document.removeEventListener("mousedown", handleClickOutside);
|
||
}
|
||
|
||
return () => {
|
||
document.removeEventListener("mousedown", handleClickOutside);
|
||
};
|
||
}, [folderDropdownOpen]);
|
||
|
||
const keyTypeOptions = [
|
||
{ value: "auto", label: t("hosts.autoDetect") },
|
||
{ value: "ssh-rsa", label: t("hosts.rsa") },
|
||
{ value: "ssh-ed25519", label: t("hosts.ed25519") },
|
||
{ value: "ecdsa-sha2-nistp256", label: t("hosts.ecdsaNistP256") },
|
||
{ value: "ecdsa-sha2-nistp384", label: t("hosts.ecdsaNistP384") },
|
||
{ value: "ecdsa-sha2-nistp521", label: t("hosts.ecdsaNistP521") },
|
||
{ value: "ssh-dss", label: t("hosts.dsa") },
|
||
{ value: "ssh-rsa-sha2-256", label: t("hosts.rsaSha2256") },
|
||
{ value: "ssh-rsa-sha2-512", label: t("hosts.rsaSha2512") },
|
||
];
|
||
|
||
const [keyTypeDropdownOpen, setKeyTypeDropdownOpen] = useState(false);
|
||
const keyTypeButtonRef = useRef<HTMLButtonElement>(null);
|
||
const keyTypeDropdownRef = useRef<HTMLDivElement>(null);
|
||
|
||
useEffect(() => {
|
||
function onClickOutside(event: MouseEvent) {
|
||
if (
|
||
keyTypeDropdownOpen &&
|
||
keyTypeDropdownRef.current &&
|
||
!keyTypeDropdownRef.current.contains(event.target as Node) &&
|
||
keyTypeButtonRef.current &&
|
||
!keyTypeButtonRef.current.contains(event.target as Node)
|
||
) {
|
||
setKeyTypeDropdownOpen(false);
|
||
}
|
||
}
|
||
|
||
document.addEventListener("mousedown", onClickOutside);
|
||
return () => document.removeEventListener("mousedown", onClickOutside);
|
||
}, [keyTypeDropdownOpen]);
|
||
|
||
const [sshConfigDropdownOpen, setSshConfigDropdownOpen] = useState<{
|
||
[key: number]: boolean;
|
||
}>({});
|
||
const sshConfigInputRefs = useRef<{ [key: number]: HTMLInputElement | null }>(
|
||
{},
|
||
);
|
||
const sshConfigDropdownRefs = useRef<{
|
||
[key: number]: HTMLDivElement | null;
|
||
}>({});
|
||
|
||
const getFilteredSshConfigs = (index: number) => {
|
||
const value = form.watch(`tunnelConnections.${index}.endpointHost`);
|
||
|
||
const currentHostName =
|
||
form.watch("name") || `${form.watch("username")}@${form.watch("ip")}`;
|
||
|
||
let filtered = sshConfigurations.filter(
|
||
(config) => config !== currentHostName,
|
||
);
|
||
|
||
if (value) {
|
||
filtered = filtered.filter((config) =>
|
||
config.toLowerCase().includes(value.toLowerCase()),
|
||
);
|
||
}
|
||
|
||
return filtered;
|
||
};
|
||
|
||
const handleSshConfigClick = (config: string, index: number) => {
|
||
form.setValue(`tunnelConnections.${index}.endpointHost`, config);
|
||
setSshConfigDropdownOpen((prev) => ({ ...prev, [index]: false }));
|
||
};
|
||
|
||
useEffect(() => {
|
||
function handleSshConfigClickOutside(event: MouseEvent) {
|
||
const openDropdowns = Object.keys(sshConfigDropdownOpen).filter(
|
||
(key) => sshConfigDropdownOpen[parseInt(key)],
|
||
);
|
||
|
||
openDropdowns.forEach((indexStr: string) => {
|
||
const index = parseInt(indexStr);
|
||
if (
|
||
sshConfigDropdownRefs.current[index] &&
|
||
!sshConfigDropdownRefs.current[index]?.contains(
|
||
event.target as Node,
|
||
) &&
|
||
sshConfigInputRefs.current[index] &&
|
||
!sshConfigInputRefs.current[index]?.contains(event.target as Node)
|
||
) {
|
||
setSshConfigDropdownOpen((prev) => ({ ...prev, [index]: false }));
|
||
}
|
||
});
|
||
}
|
||
|
||
const hasOpenDropdowns = Object.values(sshConfigDropdownOpen).some(
|
||
(open) => open,
|
||
);
|
||
|
||
if (hasOpenDropdowns) {
|
||
document.addEventListener("mousedown", handleSshConfigClickOutside);
|
||
} else {
|
||
document.removeEventListener("mousedown", handleSshConfigClickOutside);
|
||
}
|
||
|
||
return () => {
|
||
document.removeEventListener("mousedown", handleSshConfigClickOutside);
|
||
};
|
||
}, [sshConfigDropdownOpen]);
|
||
|
||
return (
|
||
<div className="flex-1 flex flex-col h-full min-h-0 w-full">
|
||
<Form {...form}>
|
||
<form
|
||
onSubmit={form.handleSubmit(onSubmit)}
|
||
className="flex flex-col flex-1 min-h-0 h-full"
|
||
>
|
||
<ScrollArea className="flex-1 min-h-0 w-full my-1 pb-2">
|
||
<div className="pr-4">
|
||
<Tabs defaultValue="general" className="w-full">
|
||
<TabsList>
|
||
<TabsTrigger value="general">
|
||
{t("hosts.general")}
|
||
</TabsTrigger>
|
||
<TabsTrigger value="terminal">
|
||
{t("hosts.terminal")}
|
||
</TabsTrigger>
|
||
<TabsTrigger value="tunnel">{t("hosts.tunnel")}</TabsTrigger>
|
||
<TabsTrigger value="file_manager">
|
||
{t("hosts.fileManager")}
|
||
</TabsTrigger>
|
||
<TabsTrigger value="statistics">
|
||
{t("hosts.statistics")}
|
||
</TabsTrigger>
|
||
</TabsList>
|
||
<TabsContent value="general" className="pt-2">
|
||
<FormLabel className="mb-3 font-bold">
|
||
{t("hosts.connectionDetails")}
|
||
</FormLabel>
|
||
<div className="grid grid-cols-12 gap-4">
|
||
<FormField
|
||
control={form.control}
|
||
name="ip"
|
||
render={({ field }) => (
|
||
<FormItem className="col-span-5">
|
||
<FormLabel>{t("hosts.ipAddress")}</FormLabel>
|
||
<FormControl>
|
||
<Input
|
||
placeholder={t("placeholders.ipAddress")}
|
||
{...field}
|
||
ref={(e) => {
|
||
field.ref(e);
|
||
ipInputRef.current = e;
|
||
}}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="port"
|
||
render={({ field }) => (
|
||
<FormItem className="col-span-1">
|
||
<FormLabel>{t("hosts.port")}</FormLabel>
|
||
<FormControl>
|
||
<Input
|
||
placeholder={t("placeholders.port")}
|
||
{...field}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="username"
|
||
render={({ field }) => {
|
||
const isCredentialAuth = authTab === "credential";
|
||
const hasCredential = !!form.watch("credentialId");
|
||
const overrideEnabled = !!form.watch(
|
||
"overrideCredentialUsername",
|
||
);
|
||
const shouldDisable =
|
||
isCredentialAuth && hasCredential && !overrideEnabled;
|
||
|
||
return (
|
||
<FormItem className="col-span-6">
|
||
<FormLabel>{t("hosts.username")}</FormLabel>
|
||
<FormControl>
|
||
<Input
|
||
placeholder={t("placeholders.username")}
|
||
disabled={shouldDisable}
|
||
{...field}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
);
|
||
}}
|
||
/>
|
||
</div>
|
||
<FormLabel className="mb-3 mt-3 font-bold">
|
||
{t("hosts.organization")}
|
||
</FormLabel>
|
||
<div className="grid grid-cols-26 gap-4">
|
||
<FormField
|
||
control={form.control}
|
||
name="name"
|
||
render={({ field }) => (
|
||
<FormItem className="col-span-10">
|
||
<FormLabel>{t("hosts.name")}</FormLabel>
|
||
<FormControl>
|
||
<Input
|
||
placeholder={t("placeholders.hostname")}
|
||
{...field}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="folder"
|
||
render={({ field }) => (
|
||
<FormItem className="col-span-10 relative">
|
||
<FormLabel>{t("hosts.folder")}</FormLabel>
|
||
<FormControl>
|
||
<Input
|
||
ref={folderInputRef}
|
||
placeholder={t("placeholders.folder")}
|
||
className="min-h-[40px]"
|
||
autoComplete="off"
|
||
value={field.value}
|
||
onFocus={() => setFolderDropdownOpen(true)}
|
||
onChange={(e) => {
|
||
field.onChange(e);
|
||
setFolderDropdownOpen(true);
|
||
}}
|
||
/>
|
||
</FormControl>
|
||
{folderDropdownOpen && filteredFolders.length > 0 && (
|
||
<div
|
||
ref={folderDropdownRef}
|
||
className="absolute top-full left-0 z-50 mt-1 w-full bg-dark-bg border border-input rounded-md shadow-lg max-h-40 overflow-y-auto p-1"
|
||
>
|
||
<div className="grid grid-cols-1 gap-1 p-0">
|
||
{filteredFolders.map((folder) => (
|
||
<Button
|
||
key={folder}
|
||
type="button"
|
||
variant="ghost"
|
||
size="sm"
|
||
className="w-full justify-start text-left rounded px-2 py-1.5 hover:bg-white/15 focus:bg-white/20 focus:outline-none"
|
||
onClick={() => handleFolderClick(folder)}
|
||
>
|
||
{folder}
|
||
</Button>
|
||
))}
|
||
</div>
|
||
</div>
|
||
)}
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="tags"
|
||
render={({ field }) => (
|
||
<FormItem className="col-span-10 overflow-visible">
|
||
<FormLabel>{t("hosts.tags")}</FormLabel>
|
||
<FormControl>
|
||
<div className="flex flex-wrap items-center gap-1 border border-input rounded-md px-3 py-2 bg-dark-bg-input focus-within:ring-2 ring-ring min-h-[40px]">
|
||
{field.value.map((tag: string, idx: number) => (
|
||
<span
|
||
key={tag + idx}
|
||
className="flex items-center bg-gray-200 text-gray-800 rounded-full px-2 py-0.5 text-xs"
|
||
>
|
||
{tag}
|
||
<button
|
||
type="button"
|
||
className="ml-1 text-gray-500 hover:text-red-500 focus:outline-none"
|
||
onClick={() => {
|
||
const newTags = field.value.filter(
|
||
(_: string, i: number) => i !== idx,
|
||
);
|
||
field.onChange(newTags);
|
||
}}
|
||
>
|
||
×
|
||
</button>
|
||
</span>
|
||
))}
|
||
<input
|
||
type="text"
|
||
className="flex-1 min-w-[60px] border-none outline-none bg-transparent p-0 h-6"
|
||
value={tagInput}
|
||
onChange={(e) => setTagInput(e.target.value)}
|
||
onKeyDown={(e) => {
|
||
if (e.key === " " && tagInput.trim() !== "") {
|
||
e.preventDefault();
|
||
if (
|
||
!field.value.includes(tagInput.trim())
|
||
) {
|
||
field.onChange([
|
||
...field.value,
|
||
tagInput.trim(),
|
||
]);
|
||
}
|
||
setTagInput("");
|
||
} else if (
|
||
e.key === "Backspace" &&
|
||
tagInput === "" &&
|
||
field.value.length > 0
|
||
) {
|
||
field.onChange(field.value.slice(0, -1));
|
||
}
|
||
}}
|
||
placeholder={t("hosts.addTagsSpaceToAdd")}
|
||
/>
|
||
</div>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="pin"
|
||
render={({ field }) => (
|
||
<FormItem className="col-span-6">
|
||
<FormLabel>{t("hosts.pin")}</FormLabel>
|
||
<FormControl>
|
||
<Switch
|
||
checked={field.value}
|
||
onCheckedChange={field.onChange}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
</div>
|
||
<FormLabel className="mb-3 mt-3 font-bold">
|
||
{t("hosts.authentication")}
|
||
</FormLabel>
|
||
<Tabs
|
||
value={authTab}
|
||
onValueChange={(value) => {
|
||
const newAuthType = value as
|
||
| "password"
|
||
| "key"
|
||
| "credential"
|
||
| "none";
|
||
setAuthTab(newAuthType);
|
||
form.setValue("authType", newAuthType);
|
||
}}
|
||
className="flex-1 flex flex-col h-full min-h-0"
|
||
>
|
||
<TabsList>
|
||
<TabsTrigger value="password">
|
||
{t("hosts.password")}
|
||
</TabsTrigger>
|
||
<TabsTrigger value="key">{t("hosts.key")}</TabsTrigger>
|
||
<TabsTrigger value="credential">
|
||
{t("hosts.credential")}
|
||
</TabsTrigger>
|
||
<TabsTrigger value="none">{t("hosts.none")}</TabsTrigger>
|
||
</TabsList>
|
||
<TabsContent value="password">
|
||
<FormField
|
||
control={form.control}
|
||
name="password"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>{t("hosts.password")}</FormLabel>
|
||
<FormControl>
|
||
<PasswordInput
|
||
placeholder={t("placeholders.password")}
|
||
{...field}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
</TabsContent>
|
||
<TabsContent value="key">
|
||
<Tabs
|
||
value={keyInputMethod}
|
||
onValueChange={(value) => {
|
||
setKeyInputMethod(value as "upload" | "paste");
|
||
if (value === "upload") {
|
||
form.setValue("key", null);
|
||
} else {
|
||
form.setValue("key", "");
|
||
}
|
||
}}
|
||
className="w-full"
|
||
>
|
||
<TabsList className="inline-flex items-center justify-center rounded-md bg-muted p-1 text-muted-foreground">
|
||
<TabsTrigger value="upload">
|
||
{t("hosts.uploadFile")}
|
||
</TabsTrigger>
|
||
<TabsTrigger value="paste">
|
||
{t("hosts.pasteKey")}
|
||
</TabsTrigger>
|
||
</TabsList>
|
||
<TabsContent value="upload" className="mt-4">
|
||
<Controller
|
||
control={form.control}
|
||
name="key"
|
||
render={({ field }) => (
|
||
<FormItem className="mb-4">
|
||
<FormLabel>
|
||
{t("hosts.sshPrivateKey")}
|
||
</FormLabel>
|
||
<FormControl>
|
||
<div className="relative inline-block">
|
||
<input
|
||
id="key-upload"
|
||
type="file"
|
||
accept=".pem,.key,.txt,.ppk"
|
||
onChange={(e) => {
|
||
const file = e.target.files?.[0];
|
||
field.onChange(file || null);
|
||
}}
|
||
className="absolute inset-0 w-full h-full opacity-0 cursor-pointer"
|
||
/>
|
||
<Button
|
||
type="button"
|
||
variant="outline"
|
||
className="justify-start text-left"
|
||
>
|
||
<span
|
||
className="truncate"
|
||
title={
|
||
field.value?.name || t("hosts.upload")
|
||
}
|
||
>
|
||
{field.value === "existing_key"
|
||
? t("hosts.existingKey")
|
||
: field.value
|
||
? editingHost
|
||
? t("hosts.updateKey")
|
||
: field.value.name
|
||
: t("hosts.upload")}
|
||
</span>
|
||
</Button>
|
||
</div>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
</TabsContent>
|
||
<TabsContent value="paste" className="mt-4">
|
||
<Controller
|
||
control={form.control}
|
||
name="key"
|
||
render={({ field }) => (
|
||
<FormItem className="mb-4">
|
||
<FormLabel>
|
||
{t("hosts.sshPrivateKey")}
|
||
</FormLabel>
|
||
<FormControl>
|
||
<CodeMirror
|
||
value={
|
||
typeof field.value === "string"
|
||
? field.value
|
||
: ""
|
||
}
|
||
onChange={(value) => field.onChange(value)}
|
||
placeholder={t(
|
||
"placeholders.pastePrivateKey",
|
||
)}
|
||
theme={oneDark}
|
||
className="border border-input rounded-md"
|
||
minHeight="120px"
|
||
basicSetup={{
|
||
lineNumbers: true,
|
||
foldGutter: false,
|
||
dropCursor: false,
|
||
allowMultipleSelections: false,
|
||
highlightSelectionMatches: false,
|
||
searchKeymap: false,
|
||
scrollPastEnd: false,
|
||
}}
|
||
extensions={[
|
||
EditorView.theme({
|
||
".cm-scroller": {
|
||
overflow: "auto",
|
||
},
|
||
}),
|
||
]}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
</TabsContent>
|
||
</Tabs>
|
||
<div className="grid grid-cols-15 gap-4 mt-4">
|
||
<FormField
|
||
control={form.control}
|
||
name="keyPassword"
|
||
render={({ field }) => (
|
||
<FormItem className="col-span-8">
|
||
<FormLabel>{t("hosts.keyPassword")}</FormLabel>
|
||
<FormControl>
|
||
<PasswordInput
|
||
placeholder={t("placeholders.keyPassword")}
|
||
{...field}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
<FormField
|
||
control={form.control}
|
||
name="keyType"
|
||
render={({ field }) => (
|
||
<FormItem className="relative col-span-3">
|
||
<FormLabel>{t("hosts.keyType")}</FormLabel>
|
||
<FormControl>
|
||
<div className="relative">
|
||
<Button
|
||
ref={keyTypeButtonRef}
|
||
type="button"
|
||
variant="outline"
|
||
className="w-full justify-start text-left rounded-md px-2 py-2 bg-dark-bg border border-input text-foreground"
|
||
onClick={() =>
|
||
setKeyTypeDropdownOpen((open) => !open)
|
||
}
|
||
>
|
||
{keyTypeOptions.find(
|
||
(opt) => opt.value === field.value,
|
||
)?.label || t("hosts.autoDetect")}
|
||
</Button>
|
||
{keyTypeDropdownOpen && (
|
||
<div
|
||
ref={keyTypeDropdownRef}
|
||
className="absolute bottom-full left-0 z-50 mb-1 w-full bg-dark-bg border border-input rounded-md shadow-lg max-h-40 overflow-y-auto p-1"
|
||
>
|
||
<div className="grid grid-cols-1 gap-1 p-0">
|
||
{keyTypeOptions.map((opt) => (
|
||
<Button
|
||
key={opt.value}
|
||
type="button"
|
||
variant="ghost"
|
||
size="sm"
|
||
className="w-full justify-start text-left rounded-md px-2 py-1.5 bg-dark-bg text-foreground hover:bg-white/15 focus:bg-white/20 focus:outline-none"
|
||
onClick={() => {
|
||
field.onChange(opt.value);
|
||
setKeyTypeDropdownOpen(false);
|
||
}}
|
||
>
|
||
{opt.label}
|
||
</Button>
|
||
))}
|
||
</div>
|
||
</div>
|
||
)}
|
||
</div>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
</div>
|
||
</TabsContent>
|
||
<TabsContent value="credential">
|
||
<div className="space-y-4">
|
||
<FormField
|
||
control={form.control}
|
||
name="credentialId"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<CredentialSelector
|
||
value={field.value}
|
||
onValueChange={field.onChange}
|
||
onCredentialSelect={(credential) => {
|
||
if (
|
||
credential &&
|
||
!form.getValues(
|
||
"overrideCredentialUsername",
|
||
)
|
||
) {
|
||
form.setValue(
|
||
"username",
|
||
credential.username,
|
||
);
|
||
}
|
||
}}
|
||
/>
|
||
<FormDescription>
|
||
{t("hosts.credentialDescription")}
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
{form.watch("credentialId") && (
|
||
<FormField
|
||
control={form.control}
|
||
name="overrideCredentialUsername"
|
||
render={({ field }) => (
|
||
<FormItem className="flex flex-row items-center justify-between rounded-lg border p-3">
|
||
<div className="space-y-0.5">
|
||
<FormLabel>
|
||
{t("hosts.overrideCredentialUsername")}
|
||
</FormLabel>
|
||
<FormDescription>
|
||
{t("hosts.overrideCredentialUsernameDesc")}
|
||
</FormDescription>
|
||
</div>
|
||
<FormControl>
|
||
<Switch
|
||
checked={field.value}
|
||
onCheckedChange={field.onChange}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
)}
|
||
</div>
|
||
</TabsContent>
|
||
<TabsContent value="none">
|
||
<Alert className="mt-2">
|
||
<AlertDescription>
|
||
<strong>{t("hosts.noneAuthTitle")}</strong>
|
||
<div className="mt-2">
|
||
{t("hosts.noneAuthDescription")}
|
||
</div>
|
||
<div className="mt-2 text-sm">
|
||
{t("hosts.noneAuthDetails")}
|
||
</div>
|
||
</AlertDescription>
|
||
</Alert>
|
||
</TabsContent>
|
||
</Tabs>
|
||
<FormField
|
||
control={form.control}
|
||
name="forceKeyboardInteractive"
|
||
render={({ field }) => (
|
||
<FormItem className="flex flex-row items-center justify-between rounded-lg border p-3 mt-4">
|
||
<div className="space-y-0.5">
|
||
<FormLabel>
|
||
{t("hosts.forceKeyboardInteractive")}
|
||
</FormLabel>
|
||
<FormDescription>
|
||
{t("hosts.forceKeyboardInteractiveDesc")}
|
||
</FormDescription>
|
||
</div>
|
||
<FormControl>
|
||
<Switch
|
||
checked={field.value}
|
||
onCheckedChange={field.onChange}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
</TabsContent>
|
||
<TabsContent value="terminal" className="space-y-1">
|
||
<FormField
|
||
control={form.control}
|
||
name="enableTerminal"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>{t("hosts.enableTerminal")}</FormLabel>
|
||
<FormControl>
|
||
<Switch
|
||
checked={field.value}
|
||
onCheckedChange={field.onChange}
|
||
/>
|
||
</FormControl>
|
||
<FormDescription>
|
||
{t("hosts.enableTerminalDesc")}
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
<Alert className="mt-4 mb-4">
|
||
<AlertDescription>
|
||
{t("hosts.terminalCustomizationNotice")}
|
||
</AlertDescription>
|
||
</Alert>
|
||
<h1 className="text-xl font-semibold mt-7">
|
||
Terminal Customization
|
||
</h1>
|
||
<Accordion type="multiple" className="w-full">
|
||
<AccordionItem value="appearance">
|
||
<AccordionTrigger>Appearance</AccordionTrigger>
|
||
<AccordionContent className="space-y-4 pt-4">
|
||
<div className="space-y-2">
|
||
<label className="text-sm font-medium">
|
||
Theme Preview
|
||
</label>
|
||
<TerminalPreview
|
||
theme={form.watch("terminalConfig.theme")}
|
||
fontSize={form.watch("terminalConfig.fontSize")}
|
||
fontFamily={form.watch("terminalConfig.fontFamily")}
|
||
cursorStyle={form.watch(
|
||
"terminalConfig.cursorStyle",
|
||
)}
|
||
cursorBlink={form.watch(
|
||
"terminalConfig.cursorBlink",
|
||
)}
|
||
letterSpacing={form.watch(
|
||
"terminalConfig.letterSpacing",
|
||
)}
|
||
lineHeight={form.watch("terminalConfig.lineHeight")}
|
||
/>
|
||
</div>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.theme"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>Theme</FormLabel>
|
||
<Select
|
||
onValueChange={field.onChange}
|
||
value={field.value}
|
||
>
|
||
<FormControl>
|
||
<SelectTrigger>
|
||
<SelectValue placeholder="Select theme" />
|
||
</SelectTrigger>
|
||
</FormControl>
|
||
<SelectContent>
|
||
{Object.entries(TERMINAL_THEMES).map(
|
||
([key, theme]) => (
|
||
<SelectItem key={key} value={key}>
|
||
{theme.name}
|
||
</SelectItem>
|
||
),
|
||
)}
|
||
</SelectContent>
|
||
</Select>
|
||
<FormDescription>
|
||
Choose a color theme for the terminal
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.fontFamily"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>Font Family</FormLabel>
|
||
<Select
|
||
onValueChange={field.onChange}
|
||
value={field.value}
|
||
>
|
||
<FormControl>
|
||
<SelectTrigger>
|
||
<SelectValue placeholder="Select font" />
|
||
</SelectTrigger>
|
||
</FormControl>
|
||
<SelectContent>
|
||
{TERMINAL_FONTS.map((font) => (
|
||
<SelectItem
|
||
key={font.value}
|
||
value={font.value}
|
||
>
|
||
{font.label}
|
||
</SelectItem>
|
||
))}
|
||
</SelectContent>
|
||
</Select>
|
||
<FormDescription>
|
||
Select the font to use in the terminal
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.fontSize"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>Font Size: {field.value}px</FormLabel>
|
||
<FormControl>
|
||
<Slider
|
||
min={8}
|
||
max={24}
|
||
step={1}
|
||
value={[field.value]}
|
||
onValueChange={([value]) =>
|
||
field.onChange(value)
|
||
}
|
||
/>
|
||
</FormControl>
|
||
<FormDescription>
|
||
Adjust the terminal font size
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.letterSpacing"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>
|
||
Letter Spacing: {field.value}px
|
||
</FormLabel>
|
||
<FormControl>
|
||
<Slider
|
||
min={-2}
|
||
max={10}
|
||
step={0.5}
|
||
value={[field.value]}
|
||
onValueChange={([value]) =>
|
||
field.onChange(value)
|
||
}
|
||
/>
|
||
</FormControl>
|
||
<FormDescription>
|
||
Adjust spacing between characters
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.lineHeight"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>Line Height: {field.value}</FormLabel>
|
||
<FormControl>
|
||
<Slider
|
||
min={1}
|
||
max={2}
|
||
step={0.1}
|
||
value={[field.value]}
|
||
onValueChange={([value]) =>
|
||
field.onChange(value)
|
||
}
|
||
/>
|
||
</FormControl>
|
||
<FormDescription>
|
||
Adjust spacing between lines
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.cursorStyle"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>Cursor Style</FormLabel>
|
||
<Select
|
||
onValueChange={field.onChange}
|
||
value={field.value}
|
||
>
|
||
<FormControl>
|
||
<SelectTrigger>
|
||
<SelectValue placeholder="Select cursor style" />
|
||
</SelectTrigger>
|
||
</FormControl>
|
||
<SelectContent>
|
||
<SelectItem value="block">Block</SelectItem>
|
||
<SelectItem value="underline">
|
||
Underline
|
||
</SelectItem>
|
||
<SelectItem value="bar">Bar</SelectItem>
|
||
</SelectContent>
|
||
</Select>
|
||
<FormDescription>
|
||
Choose the cursor appearance
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.cursorBlink"
|
||
render={({ field }) => (
|
||
<FormItem className="flex flex-row items-center justify-between rounded-lg border p-3">
|
||
<div className="space-y-0.5">
|
||
<FormLabel>Cursor Blink</FormLabel>
|
||
<FormDescription>
|
||
Enable cursor blinking animation
|
||
</FormDescription>
|
||
</div>
|
||
<FormControl>
|
||
<Switch
|
||
checked={field.value}
|
||
onCheckedChange={field.onChange}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
</AccordionContent>
|
||
</AccordionItem>
|
||
|
||
<AccordionItem value="behavior">
|
||
<AccordionTrigger>Behavior</AccordionTrigger>
|
||
<AccordionContent className="space-y-4 pt-4">
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.scrollback"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>
|
||
Scrollback Buffer: {field.value} lines
|
||
</FormLabel>
|
||
<FormControl>
|
||
<Slider
|
||
min={1000}
|
||
max={100000}
|
||
step={1000}
|
||
value={[field.value]}
|
||
onValueChange={([value]) =>
|
||
field.onChange(value)
|
||
}
|
||
/>
|
||
</FormControl>
|
||
<FormDescription>
|
||
Number of lines to keep in scrollback history
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.bellStyle"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>Bell Style</FormLabel>
|
||
<Select
|
||
onValueChange={field.onChange}
|
||
value={field.value}
|
||
>
|
||
<FormControl>
|
||
<SelectTrigger>
|
||
<SelectValue placeholder="Select bell style" />
|
||
</SelectTrigger>
|
||
</FormControl>
|
||
<SelectContent>
|
||
<SelectItem value="none">None</SelectItem>
|
||
<SelectItem value="sound">Sound</SelectItem>
|
||
<SelectItem value="visual">Visual</SelectItem>
|
||
<SelectItem value="both">Both</SelectItem>
|
||
</SelectContent>
|
||
</Select>
|
||
<FormDescription>
|
||
How to handle terminal bell (BEL character,
|
||
\x07). Programs trigger this when completing
|
||
tasks, encountering errors, or for
|
||
notifications. "Sound" plays an audio beep,
|
||
"Visual" flashes the screen briefly, "Both" does
|
||
both, "None" disables bell alerts.
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.rightClickSelectsWord"
|
||
render={({ field }) => (
|
||
<FormItem className="flex flex-row items-center justify-between rounded-lg border p-3">
|
||
<div className="space-y-0.5">
|
||
<FormLabel>Right Click Selects Word</FormLabel>
|
||
<FormDescription>
|
||
Right-clicking selects the word under cursor
|
||
</FormDescription>
|
||
</div>
|
||
<FormControl>
|
||
<Switch
|
||
checked={field.value}
|
||
onCheckedChange={field.onChange}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.fastScrollModifier"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>Fast Scroll Modifier</FormLabel>
|
||
<Select
|
||
onValueChange={field.onChange}
|
||
value={field.value}
|
||
>
|
||
<FormControl>
|
||
<SelectTrigger>
|
||
<SelectValue placeholder="Select modifier" />
|
||
</SelectTrigger>
|
||
</FormControl>
|
||
<SelectContent>
|
||
<SelectItem value="alt">Alt</SelectItem>
|
||
<SelectItem value="ctrl">Ctrl</SelectItem>
|
||
<SelectItem value="shift">Shift</SelectItem>
|
||
</SelectContent>
|
||
</Select>
|
||
<FormDescription>
|
||
Modifier key for fast scrolling
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.fastScrollSensitivity"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>
|
||
Fast Scroll Sensitivity: {field.value}
|
||
</FormLabel>
|
||
<FormControl>
|
||
<Slider
|
||
min={1}
|
||
max={10}
|
||
step={1}
|
||
value={[field.value]}
|
||
onValueChange={([value]) =>
|
||
field.onChange(value)
|
||
}
|
||
/>
|
||
</FormControl>
|
||
<FormDescription>
|
||
Scroll speed multiplier when modifier is held
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.minimumContrastRatio"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>
|
||
Minimum Contrast Ratio: {field.value}
|
||
</FormLabel>
|
||
<FormControl>
|
||
<Slider
|
||
min={1}
|
||
max={21}
|
||
step={1}
|
||
value={[field.value]}
|
||
onValueChange={([value]) =>
|
||
field.onChange(value)
|
||
}
|
||
/>
|
||
</FormControl>
|
||
<FormDescription>
|
||
Automatically adjust colors for better
|
||
readability
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
</AccordionContent>
|
||
</AccordionItem>
|
||
|
||
<AccordionItem value="advanced">
|
||
<AccordionTrigger>Advanced</AccordionTrigger>
|
||
<AccordionContent className="space-y-4 pt-4">
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.agentForwarding"
|
||
render={({ field }) => (
|
||
<FormItem className="flex flex-row items-center justify-between rounded-lg border p-3">
|
||
<div className="space-y-0.5">
|
||
<FormLabel>SSH Agent Forwarding</FormLabel>
|
||
<FormDescription>
|
||
Forward SSH authentication agent to remote
|
||
host
|
||
</FormDescription>
|
||
</div>
|
||
<FormControl>
|
||
<Switch
|
||
checked={field.value}
|
||
onCheckedChange={field.onChange}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.backspaceMode"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>Backspace Mode</FormLabel>
|
||
<Select
|
||
onValueChange={field.onChange}
|
||
value={field.value}
|
||
>
|
||
<FormControl>
|
||
<SelectTrigger>
|
||
<SelectValue placeholder="Select backspace mode" />
|
||
</SelectTrigger>
|
||
</FormControl>
|
||
<SelectContent>
|
||
<SelectItem value="normal">
|
||
Normal (DEL)
|
||
</SelectItem>
|
||
<SelectItem value="control-h">
|
||
Control-H (^H)
|
||
</SelectItem>
|
||
</SelectContent>
|
||
</Select>
|
||
<FormDescription>
|
||
Backspace key behavior for compatibility
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.startupSnippetId"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>Startup Snippet</FormLabel>
|
||
<Select
|
||
onValueChange={(value) => {
|
||
field.onChange(
|
||
value === "none" ? null : parseInt(value),
|
||
);
|
||
setSnippetSearch("");
|
||
}}
|
||
value={field.value?.toString() || "none"}
|
||
>
|
||
<FormControl>
|
||
<SelectTrigger>
|
||
<SelectValue placeholder="Select snippet" />
|
||
</SelectTrigger>
|
||
</FormControl>
|
||
<SelectContent>
|
||
<div className="px-2 pb-2 sticky top-0 bg-popover z-10">
|
||
<Input
|
||
placeholder="Search snippets..."
|
||
value={snippetSearch}
|
||
onChange={(e) =>
|
||
setSnippetSearch(e.target.value)
|
||
}
|
||
className="h-8"
|
||
onClick={(e) => e.stopPropagation()}
|
||
onKeyDown={(e) => e.stopPropagation()}
|
||
/>
|
||
</div>
|
||
<div className="max-h-[200px] overflow-y-auto">
|
||
<SelectItem value="none">None</SelectItem>
|
||
{snippets
|
||
.filter((snippet) =>
|
||
snippet.name
|
||
.toLowerCase()
|
||
.includes(
|
||
snippetSearch.toLowerCase(),
|
||
),
|
||
)
|
||
.map((snippet) => (
|
||
<SelectItem
|
||
key={snippet.id}
|
||
value={snippet.id.toString()}
|
||
>
|
||
{snippet.name}
|
||
</SelectItem>
|
||
))}
|
||
{snippets.filter((snippet) =>
|
||
snippet.name
|
||
.toLowerCase()
|
||
.includes(snippetSearch.toLowerCase()),
|
||
).length === 0 &&
|
||
snippetSearch && (
|
||
<div className="px-2 py-6 text-center text-sm text-muted-foreground">
|
||
No snippets found
|
||
</div>
|
||
)}
|
||
</div>
|
||
</SelectContent>
|
||
</Select>
|
||
<FormDescription>
|
||
Execute a snippet when the terminal connects
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.autoMosh"
|
||
render={({ field }) => (
|
||
<FormItem className="flex flex-row items-center justify-between rounded-lg border p-3">
|
||
<div className="space-y-0.5">
|
||
<FormLabel>Auto-MOSH</FormLabel>
|
||
<FormDescription>
|
||
Automatically run MOSH command on connect
|
||
</FormDescription>
|
||
</div>
|
||
<FormControl>
|
||
<Switch
|
||
checked={field.value}
|
||
onCheckedChange={field.onChange}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
{form.watch("terminalConfig.autoMosh") && (
|
||
<FormField
|
||
control={form.control}
|
||
name="terminalConfig.moshCommand"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>MOSH Command</FormLabel>
|
||
<FormControl>
|
||
<Input
|
||
placeholder="mosh user@server"
|
||
{...field}
|
||
/>
|
||
</FormControl>
|
||
<FormDescription>
|
||
The MOSH command to execute
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
)}
|
||
|
||
<div className="space-y-2">
|
||
<label className="text-sm font-medium">
|
||
Environment Variables
|
||
</label>
|
||
<FormDescription>
|
||
Set custom environment variables for the terminal
|
||
session
|
||
</FormDescription>
|
||
{form
|
||
.watch("terminalConfig.environmentVariables")
|
||
?.map((_, index) => (
|
||
<div key={index} className="flex gap-2">
|
||
<FormField
|
||
control={form.control}
|
||
name={`terminalConfig.environmentVariables.${index}.key`}
|
||
render={({ field }) => (
|
||
<FormItem className="flex-1">
|
||
<FormControl>
|
||
<Input
|
||
placeholder="Variable name"
|
||
{...field}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
<FormField
|
||
control={form.control}
|
||
name={`terminalConfig.environmentVariables.${index}.value`}
|
||
render={({ field }) => (
|
||
<FormItem className="flex-1">
|
||
<FormControl>
|
||
<Input placeholder="Value" {...field} />
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
<Button
|
||
type="button"
|
||
variant="outline"
|
||
size="icon"
|
||
onClick={() => {
|
||
const current = form.getValues(
|
||
"terminalConfig.environmentVariables",
|
||
);
|
||
form.setValue(
|
||
"terminalConfig.environmentVariables",
|
||
current.filter((_, i) => i !== index),
|
||
);
|
||
}}
|
||
>
|
||
<X className="h-4 w-4" />
|
||
</Button>
|
||
</div>
|
||
))}
|
||
<Button
|
||
type="button"
|
||
variant="outline"
|
||
size="sm"
|
||
onClick={() => {
|
||
const current =
|
||
form.getValues(
|
||
"terminalConfig.environmentVariables",
|
||
) || [];
|
||
form.setValue(
|
||
"terminalConfig.environmentVariables",
|
||
[...current, { key: "", value: "" }],
|
||
);
|
||
}}
|
||
>
|
||
<Plus className="h-4 w-4 mr-2" />
|
||
Add Variable
|
||
</Button>
|
||
</div>
|
||
</AccordionContent>
|
||
</AccordionItem>
|
||
</Accordion>
|
||
</TabsContent>
|
||
<TabsContent value="tunnel">
|
||
<FormField
|
||
control={form.control}
|
||
name="enableTunnel"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>{t("hosts.enableTunnel")}</FormLabel>
|
||
<FormControl>
|
||
<Switch
|
||
checked={field.value}
|
||
onCheckedChange={field.onChange}
|
||
/>
|
||
</FormControl>
|
||
<FormDescription>
|
||
{t("hosts.enableTunnelDesc")}
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
{form.watch("enableTunnel") && (
|
||
<>
|
||
<Alert className="mt-4">
|
||
<AlertDescription>
|
||
<strong>{t("hosts.sshpassRequired")}</strong>
|
||
<div>
|
||
{t("hosts.sshpassRequiredDesc")}{" "}
|
||
<code className="bg-muted px-1 rounded inline">
|
||
sudo apt install sshpass
|
||
</code>{" "}
|
||
{t("hosts.debianUbuntuEquivalent")}
|
||
</div>
|
||
<div className="mt-2">
|
||
<strong>{t("hosts.otherInstallMethods")}</strong>
|
||
<div>
|
||
• {t("hosts.centosRhelFedora")}{" "}
|
||
<code className="bg-muted px-1 rounded inline">
|
||
sudo yum install sshpass
|
||
</code>{" "}
|
||
{t("hosts.or")}{" "}
|
||
<code className="bg-muted px-1 rounded inline">
|
||
sudo dnf install sshpass
|
||
</code>
|
||
</div>
|
||
<div>
|
||
• {t("hosts.macos")}{" "}
|
||
<code className="bg-muted px-1 rounded inline">
|
||
brew install hudochenkov/sshpass/sshpass
|
||
</code>
|
||
</div>
|
||
<div>• {t("hosts.windows")}</div>
|
||
</div>
|
||
</AlertDescription>
|
||
</Alert>
|
||
|
||
<Alert className="mt-4">
|
||
<AlertDescription>
|
||
<strong>{t("hosts.sshServerConfigRequired")}</strong>
|
||
<div>{t("hosts.sshServerConfigDesc")}</div>
|
||
<div>
|
||
•{" "}
|
||
<code className="bg-muted px-1 rounded inline">
|
||
GatewayPorts yes
|
||
</code>{" "}
|
||
{t("hosts.gatewayPortsYes")}
|
||
</div>
|
||
<div>
|
||
•{" "}
|
||
<code className="bg-muted px-1 rounded inline">
|
||
AllowTcpForwarding yes
|
||
</code>{" "}
|
||
{t("hosts.allowTcpForwardingYes")}
|
||
</div>
|
||
<div>
|
||
•{" "}
|
||
<code className="bg-muted px-1 rounded inline">
|
||
PermitRootLogin yes
|
||
</code>{" "}
|
||
{t("hosts.permitRootLoginYes")}
|
||
</div>
|
||
<div className="mt-2">{t("hosts.editSshConfig")}</div>
|
||
</AlertDescription>
|
||
</Alert>
|
||
<div className="mt-3 flex justify-between">
|
||
<Button
|
||
variant="outline"
|
||
size="sm"
|
||
className="h-8 px-3 text-xs"
|
||
onClick={() =>
|
||
window.open(
|
||
"https://docs.termix.site/tunnels",
|
||
"_blank",
|
||
)
|
||
}
|
||
>
|
||
{t("common.documentation")}
|
||
</Button>
|
||
</div>
|
||
<FormField
|
||
control={form.control}
|
||
name="tunnelConnections"
|
||
render={({ field }) => (
|
||
<FormItem className="mt-4">
|
||
<FormLabel>
|
||
{t("hosts.tunnelConnections")}
|
||
</FormLabel>
|
||
<FormControl>
|
||
<div className="space-y-4">
|
||
{field.value.map((connection, index) => (
|
||
<div
|
||
key={index}
|
||
className="p-4 border rounded-lg bg-muted/50"
|
||
>
|
||
<div className="flex items-center justify-between mb-3">
|
||
<h4 className="text-sm font-bold">
|
||
{t("hosts.connection")} {index + 1}
|
||
</h4>
|
||
<Button
|
||
type="button"
|
||
variant="ghost"
|
||
size="sm"
|
||
onClick={() => {
|
||
const newConnections =
|
||
field.value.filter(
|
||
(_, i) => i !== index,
|
||
);
|
||
field.onChange(newConnections);
|
||
}}
|
||
>
|
||
{t("hosts.remove")}
|
||
</Button>
|
||
</div>
|
||
<div className="grid grid-cols-12 gap-4">
|
||
<FormField
|
||
control={form.control}
|
||
name={`tunnelConnections.${index}.sourcePort`}
|
||
render={({
|
||
field: sourcePortField,
|
||
}) => (
|
||
<FormItem className="col-span-4">
|
||
<FormLabel>
|
||
{t("hosts.sourcePort")}
|
||
{t("hosts.sourcePortDesc")}
|
||
</FormLabel>
|
||
<FormControl>
|
||
<Input
|
||
placeholder="22"
|
||
{...sourcePortField}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
<FormField
|
||
control={form.control}
|
||
name={`tunnelConnections.${index}.endpointPort`}
|
||
render={({
|
||
field: endpointPortField,
|
||
}) => (
|
||
<FormItem className="col-span-4">
|
||
<FormLabel>
|
||
{t("hosts.endpointPort")}
|
||
</FormLabel>
|
||
<FormControl>
|
||
<Input
|
||
placeholder="224"
|
||
{...endpointPortField}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
<FormField
|
||
control={form.control}
|
||
name={`tunnelConnections.${index}.endpointHost`}
|
||
render={({
|
||
field: endpointHostField,
|
||
}) => (
|
||
<FormItem className="col-span-4 relative">
|
||
<FormLabel>
|
||
{t("hosts.endpointSshConfig")}
|
||
</FormLabel>
|
||
<FormControl>
|
||
<Input
|
||
ref={(el) => {
|
||
sshConfigInputRefs.current[
|
||
index
|
||
] = el;
|
||
}}
|
||
placeholder={t(
|
||
"placeholders.sshConfig",
|
||
)}
|
||
className="min-h-[40px]"
|
||
autoComplete="off"
|
||
value={endpointHostField.value}
|
||
onFocus={() =>
|
||
setSshConfigDropdownOpen(
|
||
(prev) => ({
|
||
...prev,
|
||
[index]: true,
|
||
}),
|
||
)
|
||
}
|
||
onChange={(e) => {
|
||
endpointHostField.onChange(e);
|
||
setSshConfigDropdownOpen(
|
||
(prev) => ({
|
||
...prev,
|
||
[index]: true,
|
||
}),
|
||
);
|
||
}}
|
||
/>
|
||
</FormControl>
|
||
{sshConfigDropdownOpen[index] &&
|
||
getFilteredSshConfigs(index)
|
||
.length > 0 && (
|
||
<div
|
||
ref={(el) => {
|
||
sshConfigDropdownRefs.current[
|
||
index
|
||
] = el;
|
||
}}
|
||
className="absolute top-full left-0 z-50 mt-1 w-full bg-dark-bg border border-input rounded-md shadow-lg max-h-40 overflow-y-auto p-1"
|
||
>
|
||
<div className="grid grid-cols-1 gap-1 p-0">
|
||
{getFilteredSshConfigs(
|
||
index,
|
||
).map((config) => (
|
||
<Button
|
||
key={config}
|
||
type="button"
|
||
variant="ghost"
|
||
size="sm"
|
||
className="w-full justify-start text-left rounded px-2 py-1.5 hover:bg-white/15 focus:bg-white/20 focus:outline-none"
|
||
onClick={() =>
|
||
handleSshConfigClick(
|
||
config,
|
||
index,
|
||
)
|
||
}
|
||
>
|
||
{config}
|
||
</Button>
|
||
))}
|
||
</div>
|
||
</div>
|
||
)}
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
</div>
|
||
|
||
<p className="text-sm text-muted-foreground mt-2">
|
||
{t("hosts.tunnelForwardDescription", {
|
||
sourcePort:
|
||
form.watch(
|
||
`tunnelConnections.${index}.sourcePort`,
|
||
) || "22",
|
||
endpointPort:
|
||
form.watch(
|
||
`tunnelConnections.${index}.endpointPort`,
|
||
) || "224",
|
||
})}
|
||
</p>
|
||
|
||
<div className="grid grid-cols-12 gap-4 mt-4">
|
||
<FormField
|
||
control={form.control}
|
||
name={`tunnelConnections.${index}.maxRetries`}
|
||
render={({
|
||
field: maxRetriesField,
|
||
}) => (
|
||
<FormItem className="col-span-4">
|
||
<FormLabel>
|
||
{t("hosts.maxRetries")}
|
||
</FormLabel>
|
||
<FormControl>
|
||
<Input
|
||
placeholder={t(
|
||
"placeholders.maxRetries",
|
||
)}
|
||
{...maxRetriesField}
|
||
/>
|
||
</FormControl>
|
||
<FormDescription>
|
||
{t("hosts.maxRetriesDescription")}
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
<FormField
|
||
control={form.control}
|
||
name={`tunnelConnections.${index}.retryInterval`}
|
||
render={({
|
||
field: retryIntervalField,
|
||
}) => (
|
||
<FormItem className="col-span-4">
|
||
<FormLabel>
|
||
{t("hosts.retryInterval")}
|
||
</FormLabel>
|
||
<FormControl>
|
||
<Input
|
||
placeholder={t(
|
||
"placeholders.retryInterval",
|
||
)}
|
||
{...retryIntervalField}
|
||
/>
|
||
</FormControl>
|
||
<FormDescription>
|
||
{t(
|
||
"hosts.retryIntervalDescription",
|
||
)}
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
<FormField
|
||
control={form.control}
|
||
name={`tunnelConnections.${index}.autoStart`}
|
||
render={({ field }) => (
|
||
<FormItem className="col-span-4">
|
||
<FormLabel>
|
||
{t("hosts.autoStartContainer")}
|
||
</FormLabel>
|
||
<FormControl>
|
||
<Switch
|
||
checked={field.value}
|
||
onCheckedChange={field.onChange}
|
||
/>
|
||
</FormControl>
|
||
<FormDescription>
|
||
{t("hosts.autoStartDesc")}
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
</div>
|
||
</div>
|
||
))}
|
||
<Button
|
||
type="button"
|
||
variant="outline"
|
||
onClick={() => {
|
||
field.onChange([
|
||
...field.value,
|
||
{
|
||
sourcePort: 22,
|
||
endpointPort: 224,
|
||
endpointHost: "",
|
||
maxRetries: 3,
|
||
retryInterval: 10,
|
||
autoStart: false,
|
||
},
|
||
]);
|
||
}}
|
||
>
|
||
{t("hosts.addConnection")}
|
||
</Button>
|
||
</div>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
</>
|
||
)}
|
||
</TabsContent>
|
||
<TabsContent value="file_manager">
|
||
<FormField
|
||
control={form.control}
|
||
name="enableFileManager"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>{t("hosts.enableFileManager")}</FormLabel>
|
||
<FormControl>
|
||
<Switch
|
||
checked={field.value}
|
||
onCheckedChange={field.onChange}
|
||
/>
|
||
</FormControl>
|
||
<FormDescription>
|
||
{t("hosts.enableFileManagerDesc")}
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
{form.watch("enableFileManager") && (
|
||
<div className="mt-4">
|
||
<FormField
|
||
control={form.control}
|
||
name="defaultPath"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>{t("hosts.defaultPath")}</FormLabel>
|
||
<FormControl>
|
||
<Input
|
||
placeholder={t("placeholders.homePath")}
|
||
{...field}
|
||
/>
|
||
</FormControl>
|
||
<FormDescription>
|
||
{t("hosts.defaultPathDesc")}
|
||
</FormDescription>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
</div>
|
||
)}
|
||
</TabsContent>
|
||
<TabsContent value="statistics" className="space-y-6">
|
||
<div className="space-y-4">
|
||
<div className="space-y-3">
|
||
<FormField
|
||
control={form.control}
|
||
name="statsConfig.statusCheckEnabled"
|
||
render={({ field }) => (
|
||
<FormItem className="flex flex-row items-center justify-between rounded-lg border p-3">
|
||
<div className="space-y-0.5">
|
||
<FormLabel>
|
||
{t("hosts.statusCheckEnabled")}
|
||
</FormLabel>
|
||
<FormDescription>
|
||
{t("hosts.statusCheckEnabledDesc")}
|
||
</FormDescription>
|
||
</div>
|
||
<FormControl>
|
||
<Switch
|
||
checked={field.value}
|
||
onCheckedChange={field.onChange}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
{form.watch("statsConfig.statusCheckEnabled") && (
|
||
<FormField
|
||
control={form.control}
|
||
name="statsConfig.statusCheckInterval"
|
||
render={({ field }) => {
|
||
const displayValue =
|
||
statusIntervalUnit === "minutes"
|
||
? Math.round((field.value || 30) / 60)
|
||
: field.value || 30;
|
||
|
||
const handleIntervalChange = (value: string) => {
|
||
const numValue = parseInt(value) || 0;
|
||
const seconds =
|
||
statusIntervalUnit === "minutes"
|
||
? numValue * 60
|
||
: numValue;
|
||
field.onChange(seconds);
|
||
};
|
||
|
||
return (
|
||
<FormItem>
|
||
<FormLabel>
|
||
{t("hosts.statusCheckInterval")}
|
||
</FormLabel>
|
||
<div className="flex gap-2">
|
||
<FormControl>
|
||
<Input
|
||
type="number"
|
||
value={displayValue}
|
||
onChange={(e) =>
|
||
handleIntervalChange(e.target.value)
|
||
}
|
||
className="flex-1"
|
||
/>
|
||
</FormControl>
|
||
<Select
|
||
value={statusIntervalUnit}
|
||
onValueChange={(
|
||
value: "seconds" | "minutes",
|
||
) => {
|
||
setStatusIntervalUnit(value);
|
||
const currentSeconds = field.value || 30;
|
||
if (value === "minutes") {
|
||
const minutes = Math.round(
|
||
currentSeconds / 60,
|
||
);
|
||
field.onChange(minutes * 60);
|
||
}
|
||
}}
|
||
>
|
||
<SelectTrigger className="w-[120px]">
|
||
<SelectValue />
|
||
</SelectTrigger>
|
||
<SelectContent>
|
||
<SelectItem value="seconds">
|
||
{t("hosts.intervalSeconds")}
|
||
</SelectItem>
|
||
<SelectItem value="minutes">
|
||
{t("hosts.intervalMinutes")}
|
||
</SelectItem>
|
||
</SelectContent>
|
||
</Select>
|
||
</div>
|
||
<FormDescription>
|
||
{t("hosts.statusCheckIntervalDesc")}
|
||
</FormDescription>
|
||
</FormItem>
|
||
);
|
||
}}
|
||
/>
|
||
)}
|
||
</div>
|
||
|
||
<div className="space-y-3">
|
||
<FormField
|
||
control={form.control}
|
||
name="statsConfig.metricsEnabled"
|
||
render={({ field }) => (
|
||
<FormItem className="flex flex-row items-center justify-between rounded-lg border p-3">
|
||
<div className="space-y-0.5">
|
||
<FormLabel>{t("hosts.metricsEnabled")}</FormLabel>
|
||
<FormDescription>
|
||
{t("hosts.metricsEnabledDesc")}
|
||
</FormDescription>
|
||
</div>
|
||
<FormControl>
|
||
<Switch
|
||
checked={field.value}
|
||
onCheckedChange={field.onChange}
|
||
/>
|
||
</FormControl>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
|
||
{form.watch("statsConfig.metricsEnabled") && (
|
||
<FormField
|
||
control={form.control}
|
||
name="statsConfig.metricsInterval"
|
||
render={({ field }) => {
|
||
const displayValue =
|
||
metricsIntervalUnit === "minutes"
|
||
? Math.round((field.value || 30) / 60)
|
||
: field.value || 30;
|
||
|
||
const handleIntervalChange = (value: string) => {
|
||
const numValue = parseInt(value) || 0;
|
||
const seconds =
|
||
metricsIntervalUnit === "minutes"
|
||
? numValue * 60
|
||
: numValue;
|
||
field.onChange(seconds);
|
||
};
|
||
|
||
return (
|
||
<FormItem>
|
||
<FormLabel>
|
||
{t("hosts.metricsInterval")}
|
||
</FormLabel>
|
||
<div className="flex gap-2">
|
||
<FormControl>
|
||
<Input
|
||
type="number"
|
||
value={displayValue}
|
||
onChange={(e) =>
|
||
handleIntervalChange(e.target.value)
|
||
}
|
||
className="flex-1"
|
||
/>
|
||
</FormControl>
|
||
<Select
|
||
value={metricsIntervalUnit}
|
||
onValueChange={(
|
||
value: "seconds" | "minutes",
|
||
) => {
|
||
setMetricsIntervalUnit(value);
|
||
const currentSeconds = field.value || 30;
|
||
if (value === "minutes") {
|
||
const minutes = Math.round(
|
||
currentSeconds / 60,
|
||
);
|
||
field.onChange(minutes * 60);
|
||
}
|
||
}}
|
||
>
|
||
<SelectTrigger className="w-[120px]">
|
||
<SelectValue />
|
||
</SelectTrigger>
|
||
<SelectContent>
|
||
<SelectItem value="seconds">
|
||
{t("hosts.intervalSeconds")}
|
||
</SelectItem>
|
||
<SelectItem value="minutes">
|
||
{t("hosts.intervalMinutes")}
|
||
</SelectItem>
|
||
</SelectContent>
|
||
</Select>
|
||
</div>
|
||
<FormDescription>
|
||
{t("hosts.metricsIntervalDesc")}
|
||
</FormDescription>
|
||
</FormItem>
|
||
);
|
||
}}
|
||
/>
|
||
)}
|
||
</div>
|
||
</div>
|
||
|
||
{form.watch("statsConfig.metricsEnabled") && (
|
||
<>
|
||
<FormField
|
||
control={form.control}
|
||
name="statsConfig.enabledWidgets"
|
||
render={({ field }) => (
|
||
<FormItem>
|
||
<FormLabel>{t("hosts.enabledWidgets")}</FormLabel>
|
||
<FormDescription>
|
||
{t("hosts.enabledWidgetsDesc")}
|
||
</FormDescription>
|
||
<div className="space-y-3 mt-3">
|
||
{(
|
||
[
|
||
"cpu",
|
||
"memory",
|
||
"disk",
|
||
"network",
|
||
"uptime",
|
||
"processes",
|
||
"system",
|
||
] as const
|
||
).map((widget) => (
|
||
<div
|
||
key={widget}
|
||
className="flex items-center space-x-2"
|
||
>
|
||
<Checkbox
|
||
checked={field.value?.includes(widget)}
|
||
onCheckedChange={(checked) => {
|
||
const currentWidgets = field.value || [];
|
||
if (checked) {
|
||
field.onChange([
|
||
...currentWidgets,
|
||
widget,
|
||
]);
|
||
} else {
|
||
field.onChange(
|
||
currentWidgets.filter(
|
||
(w) => w !== widget,
|
||
),
|
||
);
|
||
}
|
||
}}
|
||
/>
|
||
<label className="text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70">
|
||
{widget === "cpu" &&
|
||
t("serverStats.cpuUsage")}
|
||
{widget === "memory" &&
|
||
t("serverStats.memoryUsage")}
|
||
{widget === "disk" &&
|
||
t("serverStats.diskUsage")}
|
||
{widget === "network" &&
|
||
t("serverStats.networkInterfaces")}
|
||
{widget === "uptime" &&
|
||
t("serverStats.uptime")}
|
||
{widget === "processes" &&
|
||
t("serverStats.processes")}
|
||
{widget === "system" &&
|
||
t("serverStats.systemInfo")}
|
||
</label>
|
||
</div>
|
||
))}
|
||
</div>
|
||
</FormItem>
|
||
)}
|
||
/>
|
||
</>
|
||
)}
|
||
</TabsContent>
|
||
</Tabs>
|
||
</div>
|
||
</ScrollArea>
|
||
<footer className="shrink-0 w-full pb-0">
|
||
<Separator className="p-0.25" />
|
||
<Button className="translate-y-2" type="submit" variant="outline">
|
||
{editingHost
|
||
? editingHost.id
|
||
? t("hosts.updateHost")
|
||
: t("hosts.cloneHost")
|
||
: t("hosts.addHost")}
|
||
</Button>
|
||
</footer>
|
||
</form>
|
||
</Form>
|
||
</div>
|
||
);
|
||
}
|