import type { ChannelValidation } from "@app/validation/channel.js"; import { DynamicForm } from "@components/Form/DynamicForm.js"; import { useToast } from "@core/hooks/useToast.js"; import { useDevice } from "@core/stores/deviceStore.js"; import { Protobuf } from "@meshtastic/js"; import { fromByteArray, toByteArray } from "base64-js"; export interface SettingsPanelProps { channel: Protobuf.Channel.Channel; } export const Channel = ({ channel }: SettingsPanelProps): JSX.Element => { const { config, connection, addChannel } = useDevice(); const { toast } = useToast(); const onSubmit = (data: ChannelValidation) => { const channel = new Protobuf.Channel.Channel({ ...data, settings: { ...data.settings, psk: toByteArray(data.settings.psk ?? ""), moduleSettings: { positionPrecision: data.positionPrecision, } }, }); connection?.setChannel(channel).then(() => { toast({ title: `Saved Channel: ${channel.settings?.name}`, }); addChannel(channel); }); }; return ( onSubmit={onSubmit} submitType="onSubmit" hasSubmitButton={true} defaultValues={{ ...channel, ...{ settings: { ...channel?.settings, psk: fromByteArray(channel?.settings?.psk ?? new Uint8Array(0)), }, }, positionEnabled: true, preciseLocation: true, positionPrecision: 1 }} fieldGroups={[ { label: "Channel Settings", description: "Crypto, MQTT & misc settings", fields: [ { type: "select", name: "role", label: "Role", description: "Device telemetry is sent over PRIMARY. Only one PRIMARY allowed", properties: { enumValue: Protobuf.Channel.Channel_Role, }, }, { type: "password", name: "settings.psk", label: "pre-Shared Key", description: "16, or 32 bytes", properties: { // act }, }, { type: "text", name: "settings.name", label: "Name", description: "A unique name for the channel <12 bytes, leave blank for default", }, { type: "toggle", name: "settings.uplinkEnabled", label: "Uplink Enabled", description: "Send messages from the local mesh to MQTT", }, { type: "toggle", name: "settings.downlinkEnabled", label: "Downlink Enabled", description: "Send messages from MQTT to the local mesh", }, { type: "toggle", name: "positionEnabled", label: "Allow Position Requests", description: "Send position to channel", }, { type: "toggle", name: "preciseLocation", label: "Precise Location", description: "Send precise location to channel", disabledBy: [{ fieldName: "positionEnabled", }] }, { type: "select", name: "positionPrecision", label: "Approximate Location", description: "Position shared on mesh is accurate within this distance", properties: { enumValue: config.display?.units == 0 ? { "Within 23 km":2300, "Within 12 km":12000, "Within 5.8 km":5800, "Within 2.9 km":2900, "Within 1.5 km":1500, "Within 700 m":700, "Within 350 m":350, "Within 200 m":200, "Within 90 m":90, "Within 50 m":50 } : { "Within 15 miles":1, "Within 7.3 miles":2, "Within 3.6 miles":3, "Within 1.8 miles":4, "Within 0.9 miles":5, "Within 0.5 miles":6, "Within 0.2 miles":7, "Within 600 feet":8, "Within 300 feet":9, "Within 150 feet":10 } }, disabledBy: [ { fieldName: "positionEnabled", }, ], }, ], }, ]} /> ); };