mirror of https://github.com/jetkvm/kvm.git
140 lines
4.9 KiB
TypeScript
140 lines
4.9 KiB
TypeScript
import { useEffect } from "react";
|
|
import validator from "validator";
|
|
import { LuPlus, LuX } from "react-icons/lu";
|
|
import { useFieldArray, useFormContext } from "react-hook-form";
|
|
import { cx } from "cva";
|
|
|
|
import { NetworkSettings } from "@hooks/stores";
|
|
import { Button } from "@components/Button";
|
|
import { GridCard } from "@components/Card";
|
|
import { InputFieldWithLabel } from "@components/InputField";
|
|
import { netMaskFromCidr4 } from "@/utils/ip";
|
|
import { m } from "@localizations/messages.js";
|
|
|
|
export default function StaticIpv4Card() {
|
|
const formMethods = useFormContext<NetworkSettings>();
|
|
const { register, formState, watch, setValue } = formMethods;
|
|
|
|
const { fields, append, remove } = useFieldArray({ name: "ipv4_static.dns" });
|
|
|
|
useEffect(() => {
|
|
if (fields.length === 0) append("");
|
|
}, [append, fields.length]);
|
|
|
|
const dns = watch("ipv4_static.dns");
|
|
|
|
const ipv4StaticAddress = watch("ipv4_static.address");
|
|
const hideSubnetMask = ipv4StaticAddress?.includes("/");
|
|
|
|
useEffect(() => {
|
|
const parts = ipv4StaticAddress?.split("/", 2);
|
|
if (parts?.length !== 2) return;
|
|
|
|
const cidrNotation = parseInt(parts?.[1] ?? "");
|
|
if (isNaN(cidrNotation) || cidrNotation < 0 || cidrNotation > 32) return;
|
|
|
|
const mask = netMaskFromCidr4(cidrNotation);
|
|
setValue("ipv4_static.netmask", mask);
|
|
}, [ipv4StaticAddress, setValue]);
|
|
|
|
const ipv4Validation = (value: string) => {
|
|
if (!validator.isIP(value, 4)) return m.network_ipv4_invalid()
|
|
return true;
|
|
};
|
|
|
|
const validateIsIPOrCIDR4 = (value: string) => {
|
|
if (!validator.isIP(value) && !validator.isIPRange(value, 4)) return m.network_ipv4_invalid_cidr();
|
|
return true;
|
|
};
|
|
|
|
return (
|
|
<GridCard>
|
|
<div className="animate-fadeIn p-4 text-black opacity-0 animation-duration-500 dark:text-white">
|
|
<div className="space-y-4">
|
|
<h3 className="text-base font-bold text-slate-900 dark:text-white">
|
|
{m.network_static_ipv4_header()}
|
|
</h3>
|
|
|
|
<div className={cx("grid grid-cols-1 gap-4", hideSubnetMask ? "md:grid-cols-1" : "md:grid-cols-2")}>
|
|
<InputFieldWithLabel
|
|
label={m.network_ipv4_address()}
|
|
type="text"
|
|
size="SM"
|
|
placeholder="192.168.1.100"
|
|
{
|
|
...register("ipv4_static.address", {
|
|
validate: (value: string | undefined) => validateIsIPOrCIDR4(value ?? "")
|
|
})}
|
|
error={formState.errors.ipv4_static?.address?.message}
|
|
/>
|
|
|
|
{!hideSubnetMask && <InputFieldWithLabel
|
|
label={m.network_ipv4_netmask()}
|
|
type="text"
|
|
size="SM"
|
|
placeholder="255.255.255.0"
|
|
{...register("ipv4_static.netmask", { validate: (value: string | undefined) => ipv4Validation(value ?? "") })}
|
|
error={formState.errors.ipv4_static?.netmask?.message}
|
|
/>}
|
|
</div>
|
|
|
|
<InputFieldWithLabel
|
|
label={m.network_ipv4_gateway()}
|
|
type="text"
|
|
size="SM"
|
|
placeholder="192.168.1.1"
|
|
{...register("ipv4_static.gateway", { validate: (value: string | undefined) => ipv4Validation(value ?? "") })}
|
|
error={formState.errors.ipv4_static?.gateway?.message}
|
|
/>
|
|
|
|
{/* DNS server fields */}
|
|
<div className="space-y-4">
|
|
{fields.map((dns, index) => {
|
|
return (
|
|
<div key={dns.id}>
|
|
<div className="flex items-start gap-x-2">
|
|
<div className="flex-1">
|
|
<InputFieldWithLabel
|
|
label={index === 0 ? m.network_ipv4_dns() : null}
|
|
type="text"
|
|
size="SM"
|
|
placeholder="1.1.1.1"
|
|
{...register(
|
|
`ipv4_static.dns.${index}`,
|
|
{ validate: (value: string | undefined) => ipv4Validation(value ?? "") }
|
|
)}
|
|
error={formState.errors.ipv4_static?.dns?.[index]?.message}
|
|
/>
|
|
</div>
|
|
{index > 0 && (
|
|
<div className="flex-shrink-0">
|
|
<Button
|
|
size="SM"
|
|
theme="light"
|
|
type="button"
|
|
onClick={() => remove(index)}
|
|
LeadingIcon={LuX}
|
|
/>
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
})}
|
|
</div>
|
|
|
|
<Button
|
|
size="SM"
|
|
theme="light"
|
|
onClick={() => append("", { shouldFocus: true })}
|
|
LeadingIcon={LuPlus}
|
|
type="button"
|
|
text={m.network_settings_add_dns()}
|
|
disabled={dns?.[0] === ""}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</GridCard>
|
|
);
|
|
}
|