Browse Source

Start moduleConfig & general import cleanup

pull/82/head
Sacha Weatherstone 3 years ago
parent
commit
e36032045f
No known key found for this signature in database GPG Key ID: 7AB2D7E206124B31
  1. 2
      .trunk/trunk.yaml
  2. 11
      src/App.tsx
  3. 5
      src/DeviceWrapper.tsx
  4. 5
      src/Nav/BottomNav.tsx
  5. 2
      src/Nav/NavSpacer.tsx
  6. 10
      src/Nav/PageNav.tsx
  7. 2
      src/PageRouter.tsx
  8. 2
      src/components/CommandPalette/GroupView.tsx
  9. 20
      src/components/CommandPalette/Index.tsx
  10. 2
      src/components/CommandPalette/NoResults.tsx
  11. 6
      src/components/CommandPalette/PaletteTransition.tsx
  12. 2
      src/components/CommandPalette/SearchBox.tsx
  13. 2
      src/components/CommandPalette/SearchResult.tsx
  14. 22
      src/components/DeviceSelector.tsx
  15. 11
      src/components/Dialog/DialogManager.tsx
  16. 18
      src/components/Dialog/ImportDialog.tsx
  17. 3
      src/components/Dialog/QRDialog.tsx
  18. 9
      src/components/Dialog/RebootDialog.tsx
  19. 9
      src/components/Dialog/ShutdownDialog.tsx
  20. 6
      src/components/Drawer/Metrics.tsx
  21. 4
      src/components/Drawer/Sensor.tsx
  22. 2
      src/components/Drawer/index.tsx
  23. 6
      src/components/NewDevice.tsx
  24. 15
      src/components/PageComponents/AppConfig/Map.tsx
  25. 3
      src/components/PageComponents/Channel.tsx
  26. 27
      src/components/PageComponents/Config/Bluetooth.tsx
  27. 20
      src/components/PageComponents/Config/Device.tsx
  28. 28
      src/components/PageComponents/Config/Display.tsx
  29. 27
      src/components/PageComponents/Config/LoRa.tsx
  30. 56
      src/components/PageComponents/Config/Network.tsx
  31. 48
      src/components/PageComponents/Config/Position.tsx
  32. 25
      src/components/PageComponents/Config/Power.tsx
  33. 26
      src/components/PageComponents/Config/User.tsx
  34. 4
      src/components/PageComponents/Connect/BLE.tsx
  35. 7
      src/components/PageComponents/Connect/HTTP.tsx
  36. 4
      src/components/PageComponents/Connect/Serial.tsx
  37. 2
      src/components/PageComponents/Map/MapControlls.tsx
  38. 2
      src/components/PageComponents/Messages/ChannelChat.tsx
  39. 2
      src/components/PageComponents/Messages/Message.tsx
  40. 6
      src/components/PageComponents/Messages/MessageInput.tsx
  41. 6
      src/components/PageComponents/Messages/NewLocationMessage.tsx
  42. 4
      src/components/PageComponents/Messages/WaypointMessage.tsx
  43. 55
      src/components/PageComponents/ModuleConfig/Audio.tsx
  44. 47
      src/components/PageComponents/ModuleConfig/CannedMessage.tsx
  45. 60
      src/components/PageComponents/ModuleConfig/ExternalNotification.tsx
  46. 47
      src/components/PageComponents/ModuleConfig/MQTT.tsx
  47. 64
      src/components/PageComponents/ModuleConfig/RangeTest.tsx
  48. 57
      src/components/PageComponents/ModuleConfig/Serial.tsx
  49. 60
      src/components/PageComponents/ModuleConfig/StoreForward.tsx
  50. 60
      src/components/PageComponents/ModuleConfig/Telemetry.tsx
  51. 10
      src/components/Sidebar.tsx
  52. 5
      src/components/Widgets/BatteryWidget.tsx
  53. 4
      src/components/Widgets/DeviceWidget.tsx
  54. 4
      src/components/Widgets/PeersWidget.tsx
  55. 2
      src/components/Widgets/PositionWidget.tsx
  56. 10
      src/components/form/BitwiseSelect.tsx
  57. 1
      src/components/form/Button.tsx
  58. 1
      src/components/form/Checkbox.tsx
  59. 5
      src/components/form/Form.tsx
  60. 4
      src/components/form/FormSection.tsx
  61. 9
      src/components/form/IPInput.tsx
  62. 1
      src/components/form/IconButton.tsx
  63. 5
      src/components/form/InfoWrapper.tsx
  64. 2
      src/components/form/Input.tsx
  65. 1
      src/components/form/Select.tsx
  66. 2
      src/components/form/Toggle.tsx
  67. 9
      src/components/generic/Dialog.tsx
  68. 2
      src/components/generic/Mono.tsx
  69. 2
      src/components/generic/TabbedContent.tsx
  70. 6
      src/components/generic/ThemeController.tsx
  71. 2
      src/core/utils/selectEnumOptions.tsx
  72. 3
      src/index.tsx
  73. 7
      src/pages/Channels.tsx
  74. 2
      src/pages/Config/AppConfig.tsx
  75. 6
      src/pages/Config/DeviceConfig.tsx
  76. 4
      src/pages/Config/ModuleConfig.tsx
  77. 7
      src/pages/Config/index.tsx
  78. 2
      src/pages/Extensions/Environment.tsx
  79. 1
      src/pages/Extensions/FileBrowser.tsx
  80. 4
      src/pages/Extensions/Index.tsx
  81. 7
      src/pages/Map.tsx
  82. 8
      src/pages/Messages.tsx
  83. 3
      src/pages/Peers.tsx
  84. 2
      src/validation/appConfig/map.ts

11
src/App.tsx

@ -1,8 +1,5 @@
import type React from "react";
import { MapProvider } from "react-map-gl"; import { MapProvider } from "react-map-gl";
import { useAppStore } from "@core/stores/appStore.js";
import { useAppStore } from "@app/core/stores/appStore.js";
import { DeviceWrapper } from "@app/DeviceWrapper.js"; import { DeviceWrapper } from "@app/DeviceWrapper.js";
import { PageRouter } from "@app/PageRouter.js"; import { PageRouter } from "@app/PageRouter.js";
import { CommandPalette } from "@components/CommandPalette/Index.js"; import { CommandPalette } from "@components/CommandPalette/Index.js";
@ -12,9 +9,9 @@ import { NewDevice } from "@components/NewDevice.js";
import { Sidebar } from "@components/Sidebar.js"; import { Sidebar } from "@components/Sidebar.js";
import { useDeviceStore } from "@core/stores/deviceStore.js"; import { useDeviceStore } from "@core/stores/deviceStore.js";
import { Drawer } from "./components/Drawer/index.js"; import { Drawer } from "@components/Drawer/index.js";
import { ThemeController } from "./components/generic/ThemeController.js"; import { ThemeController } from "@components/generic/ThemeController.js";
import { BottomNav } from "./Nav/BottomNav.js"; import { BottomNav } from "@app/Nav/BottomNav.js";
export const App = (): JSX.Element => { export const App = (): JSX.Element => {
const { getDevice } = useDeviceStore(); const { getDevice } = useDeviceStore();

5
src/DeviceWrapper.tsx

@ -1,10 +1,9 @@
import type React from "react";
import { DeviceContext } from "@core/providers/useDevice.js"; import { DeviceContext } from "@core/providers/useDevice.js";
import type { Device } from "@core/stores/deviceStore.js"; import type { Device } from "@core/stores/deviceStore.js";
import type { ReactNode } from "react";
export interface DeviceWrapperProps { export interface DeviceWrapperProps {
children: React.ReactNode; children: ReactNode;
device?: Device; device?: Device;
} }

5
src/Nav/BottomNav.tsx

@ -1,9 +1,8 @@
import type React from "react";
import { GitBranchIcon } from "@primer/octicons-react"; import { GitBranchIcon } from "@primer/octicons-react";
import type { ReactNode } from "react";
export interface BottomNavProps { export interface BottomNavProps {
children: React.ReactNode; children: ReactNode;
} }
export const BottomNav = ({ children }: BottomNavProps): JSX.Element => { export const BottomNav = ({ children }: BottomNavProps): JSX.Element => {

2
src/Nav/NavSpacer.tsx

@ -1,5 +1,3 @@
import type React from "react";
export const NavSpacer = (): JSX.Element => { export const NavSpacer = (): JSX.Element => {
return <div className="h-1 w-10 rounded-full bg-accentMuted" />; return <div className="h-1 w-10 rounded-full bg-accentMuted" />;
}; };

10
src/Nav/PageNav.tsx

@ -1,8 +1,6 @@
import type React from "react"; import type { ComponentType, SVGProps } from "react";
import type { SVGProps } from "react"; import { useDevice } from "@core/providers/useDevice.js";
import type { Page } from "@core/stores/deviceStore.js";
import { useDevice } from "@app/core/providers/useDevice.js";
import type { Page } from "@app/core/stores/deviceStore.js";
import { import {
BeakerIcon, BeakerIcon,
ChatBubbleBottomCenterTextIcon, ChatBubbleBottomCenterTextIcon,
@ -17,7 +15,7 @@ export const PageNav = (): JSX.Element => {
interface NavLink { interface NavLink {
name: string; name: string;
icon: React.ComponentType<React.SVGProps<SVGSVGElement>>; icon: ComponentType<SVGProps<SVGSVGElement>>;
page: Page; page: Page;
} }

2
src/PageRouter.tsx

@ -1,5 +1,3 @@
import type React from "react";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
import { ChannelsPage } from "@pages/Channels.js"; import { ChannelsPage } from "@pages/Channels.js";
import { ConfigPage } from "@pages/Config/index.js"; import { ConfigPage } from "@pages/Config/index.js";

2
src/components/CommandPalette/GroupView.tsx

@ -1,5 +1,3 @@
import type React from "react";
import type { Group } from "@components/CommandPalette/Index.js"; import type { Group } from "@components/CommandPalette/Index.js";
import { Combobox } from "@headlessui/react"; import { Combobox } from "@headlessui/react";
import { ChevronRightIcon } from "@heroicons/react/24/outline"; import { ChevronRightIcon } from "@heroicons/react/24/outline";

20
src/components/CommandPalette/Index.tsx

@ -1,11 +1,8 @@
import type React from "react"; import { ComponentType, Fragment, SVGProps, useEffect, useState } from "react";
import { Fragment, useEffect, useState } from "react";
import { toast } from "react-hot-toast"; import { toast } from "react-hot-toast";
import { useDevice } from "@core/providers/useDevice.js";
import { useDevice } from "@app/core/providers/useDevice.js"; import { useAppStore } from "@core/stores/appStore.js";
import { useAppStore } from "@app/core/stores/appStore.js"; import { useDeviceStore } from "@core/stores/deviceStore.js";
import { useDeviceStore } from "@app/core/stores/deviceStore.js";
import { GroupView } from "@components/CommandPalette/GroupView.js"; import { GroupView } from "@components/CommandPalette/GroupView.js";
import { NoResults } from "@components/CommandPalette/NoResults.js"; import { NoResults } from "@components/CommandPalette/NoResults.js";
import { PaletteTransition } from "@components/CommandPalette/PaletteTransition.js"; import { PaletteTransition } from "@components/CommandPalette/PaletteTransition.js";
@ -39,18 +36,17 @@ import {
WindowIcon, WindowIcon,
XCircleIcon XCircleIcon
} from "@heroicons/react/24/outline"; } from "@heroicons/react/24/outline";
import { Blur } from "@components/generic/Blur.js";
import { Blur } from "../generic/Blur.js"; import { ThemeController } from "@components/generic/ThemeController.js";
import { ThemeController } from "../generic/ThemeController.js";
export interface Group { export interface Group {
name: string; name: string;
icon: React.ComponentType<React.SVGProps<SVGSVGElement>>; icon: ComponentType<SVGProps<SVGSVGElement>>;
commands: Command[]; commands: Command[];
} }
export interface Command { export interface Command {
name: string; name: string;
icon: React.ComponentType<React.SVGProps<SVGSVGElement>>; icon: ComponentType<SVGProps<SVGSVGElement>>;
action?: () => void; action?: () => void;
subItems?: SubItem[]; subItems?: SubItem[];
tags?: string[]; tags?: string[];

2
src/components/CommandPalette/NoResults.tsx

@ -1,5 +1,3 @@
import type React from "react";
import { Mono } from "@components/generic/Mono.js"; import { Mono } from "@components/generic/Mono.js";
import { CommandLineIcon } from "@heroicons/react/24/outline"; import { CommandLineIcon } from "@heroicons/react/24/outline";

6
src/components/CommandPalette/PaletteTransition.tsx

@ -1,10 +1,8 @@
import type React from "react"; import { Fragment, ReactNode } from "react";
import { Fragment } from "react";
import { Transition } from "@headlessui/react"; import { Transition } from "@headlessui/react";
export interface PaletteTransitionProps { export interface PaletteTransitionProps {
children: React.ReactNode; children: ReactNode;
} }
export const PaletteTransition = ({ export const PaletteTransition = ({

2
src/components/CommandPalette/SearchBox.tsx

@ -1,5 +1,3 @@
import type React from "react";
import { Combobox } from "@headlessui/react"; import { Combobox } from "@headlessui/react";
import { MagnifyingGlassIcon } from "@heroicons/react/24/outline"; import { MagnifyingGlassIcon } from "@heroicons/react/24/outline";

2
src/components/CommandPalette/SearchResult.tsx

@ -1,5 +1,3 @@
import type React from "react";
import type { Group } from "@components/CommandPalette/Index.js"; import type { Group } from "@components/CommandPalette/Index.js";
import { Combobox } from "@headlessui/react"; import { Combobox } from "@headlessui/react";
import { ChevronRightIcon } from "@heroicons/react/24/outline"; import { ChevronRightIcon } from "@heroicons/react/24/outline";

22
src/components/DeviceSelector.tsx

@ -1,17 +1,15 @@
import type React from "react"; import { useAppStore } from "@core/stores/appStore.js";
import { useDeviceStore } from "@core/stores/deviceStore.js";
import { useAppStore } from "@app/core/stores/appStore.js";
import { useDeviceStore } from "@app/core/stores/deviceStore.js";
import { NavSpacer } from "@app/Nav/NavSpacer.js"; import { NavSpacer } from "@app/Nav/NavSpacer.js";
import { PageNav } from "@app/Nav/PageNav.js"; import { PageNav } from "@app/Nav/PageNav.js";
import { Mono } from "@components/generic/Mono.js";
import { Hashicon } from "@emeraldpay/hashicon-react"; import { Hashicon } from "@emeraldpay/hashicon-react";
import { PlusIcon } from "@heroicons/react/24/outline"; import { PlusIcon } from "@heroicons/react/24/outline";
import { MoonIcon, SunIcon } from "@primer/octicons-react"; import { MoonIcon, SunIcon } from "@primer/octicons-react";
export const DeviceSelector = (): JSX.Element => { export const DeviceSelector = (): JSX.Element => {
const { getDevices } = useDeviceStore(); const { getDevices } = useDeviceStore();
const { selectedDevice, setSelectedDevice, darkMode, setDarkMode } = useAppStore(); const { selectedDevice, setSelectedDevice, darkMode, setDarkMode } =
useAppStore();
return ( return (
<div className="flex h-full w-14 items-center gap-3 bg-backgroundPrimary pt-3 [writing-mode:vertical-rl]"> <div className="flex h-full w-14 items-center gap-3 bg-backgroundPrimary pt-3 [writing-mode:vertical-rl]">
@ -55,12 +53,12 @@ export const DeviceSelector = (): JSX.Element => {
<NavSpacer /> <NavSpacer />
<div onClick={() => setDarkMode(!darkMode)} className="bg-backgroundPrimary py-5 px-4 hover:brightness-hover active:brightness-press text-textSecondary hover:text-textPrimary">{ <div
darkMode ? ( onClick={() => setDarkMode(!darkMode)}
<SunIcon className="w-4" /> className="bg-backgroundPrimary py-5 px-4 text-textSecondary hover:text-textPrimary hover:brightness-hover active:brightness-press"
) : ( >
<MoonIcon className="w-4" /> {darkMode ? <SunIcon className="w-4" /> : <MoonIcon className="w-4" />}
)}</div> </div>
<img <img
src={darkMode ? "Logo_White.svg" : "Logo_Black.svg"} src={darkMode ? "Logo_White.svg" : "Logo_Black.svg"}

11
src/components/Dialog/DialogManager.tsx

@ -1,11 +1,8 @@
import type React from "react"; import { useDevice } from "@core/providers/useDevice.js";
import { useDevice } from "@app/core/providers/useDevice.js";
import { QRDialog } from "@components/Dialog/QRDialog.js"; import { QRDialog } from "@components/Dialog/QRDialog.js";
import { RebootDialog } from "@components/Dialog/RebootDialog.js";
import { RebootDialog } from "./RebootDialog.js"; import { ShutdownDialog } from "@components/Dialog/ShutdownDialog.js";
import { ShutdownDialog } from "./ShutdownDialog.js"; import { ImportDialog } from "@components/Dialog/ImportDialog.js";
import { ImportDialog } from "./ImportDialog.js";
export const DialogManager = (): JSX.Element => { export const DialogManager = (): JSX.Element => {
const { channels, config, dialog, setDialogOpen } = useDevice(); const { channels, config, dialog, setDialogOpen } = useDevice();

18
src/components/Dialog/ImportDialog.tsx

@ -1,20 +1,14 @@
import type React from "react";
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
import { toByteArray } from "base64-js";
import { fromByteArray, toByteArray } from "base64-js";
import { toast } from "react-hot-toast";
import { QRCode } from "react-qrcode-logo";
import { Checkbox } from "@components/form/Checkbox.js"; import { Checkbox } from "@components/form/Checkbox.js";
import { Input } from "@components/form/Input.js"; import { Input } from "@components/form/Input.js";
import { Dialog } from "@components/generic/Dialog.js"; import { Dialog } from "@components/generic/Dialog.js";
import { ClipboardIcon } from "@heroicons/react/24/outline";
import { Protobuf } from "@meshtastic/meshtasticjs"; import { Protobuf } from "@meshtastic/meshtasticjs";
import { Select } from "../form/Select.js"; import { Select } from "@components/form/Select.js";
import { renderOptions } from "@app/core/utils/selectEnumOptions.js"; import { renderOptions } from "@core/utils/selectEnumOptions.js";
import { Toggle } from "../form/Toggle.js"; import { Toggle } from "@components/form/Toggle.js";
import { Button } from "../form/Button.js"; import { Button } from "@components/form/Button.js";
import { useDevice } from "@app/core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
export interface ImportDialogProps { export interface ImportDialogProps {
isOpen: boolean; isOpen: boolean;

3
src/components/Dialog/QRDialog.tsx

@ -1,10 +1,7 @@
import type React from "react";
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
import { fromByteArray } from "base64-js"; import { fromByteArray } from "base64-js";
import { toast } from "react-hot-toast"; import { toast } from "react-hot-toast";
import { QRCode } from "react-qrcode-logo"; import { QRCode } from "react-qrcode-logo";
import { Checkbox } from "@components/form/Checkbox.js"; import { Checkbox } from "@components/form/Checkbox.js";
import { Input } from "@components/form/Input.js"; import { Input } from "@components/form/Input.js";
import { Dialog } from "@components/generic/Dialog.js"; import { Dialog } from "@components/generic/Dialog.js";

9
src/components/Dialog/RebootDialog.tsx

@ -1,12 +1,9 @@
import type React from "react";
import { useState } from "react"; import { useState } from "react";
import { useDevice } from "@core/providers/useDevice.js";
import { useDevice } from "@app/core/providers/useDevice.js";
import { Dialog } from "@components/generic/Dialog.js"; import { Dialog } from "@components/generic/Dialog.js";
import { ArrowPathIcon, ClockIcon } from "@heroicons/react/24/outline"; import { ArrowPathIcon, ClockIcon } from "@heroicons/react/24/outline";
import { Button } from "@components/form/Button.js";
import { Button } from "../form/Button.js"; import { Input } from "@components/form/Input.js";
import { Input } from "../form/Input.js";
export interface RebootDialogProps { export interface RebootDialogProps {
isOpen: boolean; isOpen: boolean;

9
src/components/Dialog/ShutdownDialog.tsx

@ -1,12 +1,9 @@
import type React from "react";
import { useState } from "react"; import { useState } from "react";
import { useDevice } from "@core/providers/useDevice.js";
import { useDevice } from "@app/core/providers/useDevice.js";
import { Dialog } from "@components/generic/Dialog.js"; import { Dialog } from "@components/generic/Dialog.js";
import { ClockIcon, PowerIcon } from "@heroicons/react/24/outline"; import { ClockIcon, PowerIcon } from "@heroicons/react/24/outline";
import { Button } from "@components/form/Button.js";
import { Button } from "../form/Button.js"; import { Input } from "@components/form/Input.js";
import { Input } from "../form/Input.js";
export interface ShutdownDialogProps { export interface ShutdownDialogProps {
isOpen: boolean; isOpen: boolean;

6
src/components/Drawer/Metrics.tsx

@ -1,7 +1,4 @@
import "chartjs-adapter-date-fns"; import "chartjs-adapter-date-fns";
import type React from "react";
import { import {
Chart as ChartJS, Chart as ChartJS,
Filler, Filler,
@ -13,8 +10,7 @@ import {
Tooltip Tooltip
} from "chart.js"; } from "chart.js";
import { Line } from "react-chartjs-2"; import { Line } from "react-chartjs-2";
import { useDevice } from "@core/providers/useDevice.js";
import { useDevice } from "@app/core/providers/useDevice.js";
export const Metrics = (): JSX.Element => { export const Metrics = (): JSX.Element => {
const { nodes, hardware } = useDevice(); const { nodes, hardware } = useDevice();

4
src/components/Drawer/Sensor.tsx

@ -1,7 +1,5 @@
import "chartjs-adapter-date-fns"; import "chartjs-adapter-date-fns";
import type React from "react";
import { import {
Chart as ChartJS, Chart as ChartJS,
Filler, Filler,
@ -14,7 +12,7 @@ import {
} from "chart.js"; } from "chart.js";
import { Line } from "react-chartjs-2"; import { Line } from "react-chartjs-2";
import { useDevice } from "@app/core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
export const Sensor = (): JSX.Element => { export const Sensor = (): JSX.Element => {
const { nodes, hardware } = useDevice(); const { nodes, hardware } = useDevice();

2
src/components/Drawer/index.tsx

@ -1,6 +1,4 @@
import type React from "react";
import { useState } from "react"; import { useState } from "react";
import { Metrics } from "@components/Drawer/Metrics.js"; import { Metrics } from "@components/Drawer/Metrics.js";
import { Notifications } from "@components/Drawer/Notifications.js"; import { Notifications } from "@components/Drawer/Notifications.js";
import { Sensor } from "@components/Drawer/Sensor.js"; import { Sensor } from "@components/Drawer/Sensor.js";

6
src/components/NewDevice.tsx

@ -4,7 +4,7 @@ import { TabbedContent, TabType } from "@components/generic/TabbedContent.js";
import { BLE } from "@components/PageComponents/Connect/BLE.js"; import { BLE } from "@components/PageComponents/Connect/BLE.js";
import { HTTP } from "@components/PageComponents/Connect/HTTP.js"; import { HTTP } from "@components/PageComponents/Connect/HTTP.js";
import { Serial } from "@components/PageComponents/Connect/Serial.js"; import { Serial } from "@components/PageComponents/Connect/Serial.js";
import { useAppStore } from "@app/core/stores/appStore.js"; import { useAppStore } from "@core/stores/appStore.js";
import { MoonIcon, SunIcon } from "@heroicons/react/24/outline"; import { MoonIcon, SunIcon } from "@heroicons/react/24/outline";
export const NewDevice = () => { export const NewDevice = () => {
@ -37,9 +37,7 @@ export const NewDevice = () => {
return ( return (
<div className="m-auto h-96 w-96"> <div className="m-auto h-96 w-96">
<TabbedContent <TabbedContent tabs={tabs} />
tabs={tabs}
/>
</div> </div>
); );
}; };

15
src/components/PageComponents/AppConfig/Map.tsx

@ -1,13 +1,10 @@
import type React from "react";
import { Controller, useFieldArray, useForm } from "react-hook-form"; import { Controller, useFieldArray, useForm } from "react-hook-form";
import { Button } from "@components/form/Button.js";
import { Button } from "@app/components/form/Button.js"; import { IconButton } from "@components/form/IconButton.js";
import { IconButton } from "@app/components/form/IconButton.js"; import { InfoWrapper } from "@components/form/InfoWrapper.js";
import { InfoWrapper } from "@app/components/form/InfoWrapper.js"; import { Input } from "@components/form/Input.js";
import { Input } from "@app/components/form/Input.js"; import { Toggle } from "@components/form/Toggle.js";
import { Toggle } from "@app/components/form/Toggle.js"; import { useAppStore } from "@core/stores/appStore.js";
import { useAppStore } from "@app/core/stores/appStore.js";
import { MapValidation } from "@app/validation/appConfig/map.js"; import { MapValidation } from "@app/validation/appConfig/map.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { TrashIcon } from "@heroicons/react/24/outline"; import { TrashIcon } from "@heroicons/react/24/outline";

3
src/components/PageComponents/Channel.tsx

@ -1,10 +1,7 @@
import type React from "react";
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
import { fromByteArray, toByteArray } from "base64-js"; import { fromByteArray, toByteArray } from "base64-js";
import { Controller, useForm } from "react-hook-form"; import { Controller, useForm } from "react-hook-form";
import { toast } from "react-hot-toast"; import { toast } from "react-hot-toast";
import { ChannelSettingsValidation } from "@app/validation/channelSettings.js"; import { ChannelSettingsValidation } from "@app/validation/channelSettings.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { Input } from "@components/form/Input.js"; import { Input } from "@components/form/Input.js";

27
src/components/PageComponents/Config/Bluetooth.tsx

@ -1,12 +1,8 @@
import type React from "react";
import { useEffect } from "react"; import { useEffect } from "react";
import { Controller, useForm, useWatch } from "react-hook-form"; import { Controller, useForm, useWatch } from "react-hook-form";
import { toast } from "react-hot-toast"; import { Input } from "@components/form/Input.js";
import { Select } from "@components/form/Select.js";
import { Input } from "@app/components/form/Input.js"; import { Toggle } from "@components/form/Toggle.js";
import { Select } from "@app/components/form/Select.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { BluetoothValidation } from "@app/validation/config/bluetooth.js"; import { BluetoothValidation } from "@app/validation/config/bluetooth.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
@ -17,17 +13,12 @@ import { Protobuf } from "@meshtastic/meshtasticjs";
export const Bluetooth = (): JSX.Element => { export const Bluetooth = (): JSX.Element => {
const { config, setWorkingConfig } = useDevice(); const { config, setWorkingConfig } = useDevice();
const { const { register, handleSubmit, control, reset } =
register, useForm<BluetoothValidation>({
handleSubmit, mode: "onChange",
formState: { errors, isDirty }, defaultValues: config.bluetooth,
control, resolver: classValidatorResolver(BluetoothValidation)
reset });
} = useForm<BluetoothValidation>({
mode: "onChange",
defaultValues: config.bluetooth,
resolver: classValidatorResolver(BluetoothValidation)
});
useEffect(() => { useEffect(() => {
reset(config.bluetooth); reset(config.bluetooth);

20
src/components/PageComponents/Config/Device.tsx

@ -1,12 +1,8 @@
import type React from "react";
import { useEffect } from "react"; import { useEffect } from "react";
import { Controller, useForm } from "react-hook-form"; import { Controller, useForm } from "react-hook-form";
import { toast } from "react-hot-toast"; import { Input } from "@components/form/Input.js";
import { Select } from "@components/form/Select.js";
import { Input } from "@app/components/form/Input.js"; import { Toggle } from "@components/form/Toggle.js";
import { Select } from "@app/components/form/Select.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { DeviceValidation } from "@app/validation/config/device.js"; import { DeviceValidation } from "@app/validation/config/device.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
@ -16,13 +12,7 @@ import { Protobuf } from "@meshtastic/meshtasticjs";
export const Device = (): JSX.Element => { export const Device = (): JSX.Element => {
const { config, setWorkingConfig } = useDevice(); const { config, setWorkingConfig } = useDevice();
const { const { register, handleSubmit, control, reset } = useForm<DeviceValidation>({
register,
handleSubmit,
formState: { errors, isDirty },
control,
reset
} = useForm<DeviceValidation>({
mode: "onChange", mode: "onChange",
defaultValues: config.device, defaultValues: config.device,
resolver: classValidatorResolver(DeviceValidation) resolver: classValidatorResolver(DeviceValidation)
@ -80,14 +70,12 @@ export const Device = (): JSX.Element => {
label="Button Pin" label="Button Pin"
description="Button pin override" description="Button pin override"
type="number" type="number"
error={errors.buttonGpio?.message}
{...register("buttonGpio", { valueAsNumber: true })} {...register("buttonGpio", { valueAsNumber: true })}
/> />
<Input <Input
label="Buzzer Pin" label="Buzzer Pin"
description="Buzzer pin override" description="Buzzer pin override"
type="number" type="number"
error={errors.buzzerGpio?.message}
{...register("buzzerGpio", { valueAsNumber: true })} {...register("buzzerGpio", { valueAsNumber: true })}
/> />
</Form> </Form>

28
src/components/PageComponents/Config/Display.tsx

@ -1,12 +1,8 @@
import type React from "react";
import { useEffect } from "react"; import { useEffect } from "react";
import { Controller, useForm } from "react-hook-form"; import { Controller, useForm } from "react-hook-form";
import { toast } from "react-hot-toast"; import { Input } from "@components/form/Input.js";
import { Select } from "@components/form/Select.js";
import { Input } from "@app/components/form/Input.js"; import { Toggle } from "@components/form/Toggle.js";
import { Select } from "@app/components/form/Select.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { DisplayValidation } from "@app/validation/config/display.js"; import { DisplayValidation } from "@app/validation/config/display.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
@ -16,17 +12,13 @@ import { Protobuf } from "@meshtastic/meshtasticjs";
export const Display = (): JSX.Element => { export const Display = (): JSX.Element => {
const { config, setWorkingConfig } = useDevice(); const { config, setWorkingConfig } = useDevice();
const { const { register, handleSubmit, reset, control } = useForm<DisplayValidation>(
register, {
handleSubmit, mode: "onChange",
formState: { errors, isDirty }, defaultValues: config.display,
reset, resolver: classValidatorResolver(DisplayValidation)
control }
} = useForm<DisplayValidation>({ );
mode: "onChange",
defaultValues: config.display,
resolver: classValidatorResolver(DisplayValidation)
});
useEffect(() => { useEffect(() => {
reset(config.display); reset(config.display);

27
src/components/PageComponents/Config/LoRa.tsx

@ -1,13 +1,9 @@
import type React from "react";
import { useEffect } from "react"; import { useEffect } from "react";
import { Controller, useForm, useWatch } from "react-hook-form"; import { Controller, useForm, useWatch } from "react-hook-form";
import { toast } from "react-hot-toast"; import { FormSection } from "@components/form/FormSection.js";
import { Input } from "@components/form/Input.js";
import { FormSection } from "@app/components/form/FormSection.js"; import { Select } from "@components/form/Select.js";
import { Input } from "@app/components/form/Input.js"; import { Toggle } from "@components/form/Toggle.js";
import { Select } from "@app/components/form/Select.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { LoRaValidation } from "@app/validation/config/lora.js"; import { LoRaValidation } from "@app/validation/config/lora.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
@ -18,13 +14,7 @@ import { Protobuf } from "@meshtastic/meshtasticjs";
export const LoRa = (): JSX.Element => { export const LoRa = (): JSX.Element => {
const { config, setWorkingConfig } = useDevice(); const { config, setWorkingConfig } = useDevice();
const { const { register, handleSubmit, control, reset } = useForm<LoRaValidation>({
register,
handleSubmit,
formState: { errors, isDirty },
control,
reset
} = useForm<LoRaValidation>({
mode: "onChange", mode: "onChange",
defaultValues: config.lora, defaultValues: config.lora,
resolver: classValidatorResolver(LoRaValidation) resolver: classValidatorResolver(LoRaValidation)
@ -81,7 +71,6 @@ export const LoRa = (): JSX.Element => {
description="Channel bandwidth in MHz" description="Channel bandwidth in MHz"
type="number" type="number"
suffix="MHz" suffix="MHz"
error={errors.bandwidth?.message}
{...register("bandwidth", { {...register("bandwidth", {
valueAsNumber: true valueAsNumber: true
})} })}
@ -91,7 +80,6 @@ export const LoRa = (): JSX.Element => {
description="Indicates the number of chirps per symbol" description="Indicates the number of chirps per symbol"
type="number" type="number"
suffix="CPS" suffix="CPS"
error={errors.spreadFactor?.message}
{...register("spreadFactor", { {...register("spreadFactor", {
valueAsNumber: true valueAsNumber: true
})} })}
@ -100,7 +88,6 @@ export const LoRa = (): JSX.Element => {
label="Coding Rate" label="Coding Rate"
description="The denominator of the coding rate" description="The denominator of the coding rate"
type="number" type="number"
error={errors.codingRate?.message}
{...register("codingRate", { {...register("codingRate", {
valueAsNumber: true valueAsNumber: true
})} })}
@ -132,14 +119,12 @@ export const LoRa = (): JSX.Element => {
label="Transmit Power" label="Transmit Power"
description="Max transmit power in dBm" description="Max transmit power in dBm"
type="number" type="number"
error={errors.txPower?.message}
{...register("txPower", { valueAsNumber: true })} {...register("txPower", { valueAsNumber: true })}
/> />
<Input <Input
label="Channel Number" label="Channel Number"
description="LoRa channel number" description="LoRa channel number"
type="number" type="number"
error={errors.channelNum?.message}
{...register("channelNum", { valueAsNumber: true })} {...register("channelNum", { valueAsNumber: true })}
/> />
<Input <Input
@ -147,7 +132,6 @@ export const LoRa = (): JSX.Element => {
description="Frequency offset to correct for crystal calibration errors" description="Frequency offset to correct for crystal calibration errors"
suffix="Hz" suffix="Hz"
type="number" type="number"
error={errors.frequencyOffset?.message}
{...register("frequencyOffset", { valueAsNumber: true })} {...register("frequencyOffset", { valueAsNumber: true })}
/> />
<Controller <Controller
@ -168,7 +152,6 @@ export const LoRa = (): JSX.Element => {
description="Maximum number of hops" description="Maximum number of hops"
suffix="Hops" suffix="Hops"
type="number" type="number"
error={errors.hopLimit?.message}
{...register("hopLimit", { valueAsNumber: true })} {...register("hopLimit", { valueAsNumber: true })}
/> />
</Form> </Form>

56
src/components/PageComponents/Config/Network.tsx

@ -1,15 +1,11 @@
import type React from "react";
import { useEffect } from "react"; import { useEffect } from "react";
import { Controller, useForm, useWatch } from "react-hook-form"; import { Controller, useForm, useWatch } from "react-hook-form";
import { toast } from "react-hot-toast"; import { FormSection } from "@components/form/FormSection.js";
import { Input } from "@components/form/Input.js";
import { FormSection } from "@app/components/form/FormSection.js"; import { IPInput } from "@components/form/IPInput.js";
import { Input } from "@app/components/form/Input.js"; import { Select } from "@components/form/Select.js";
import { IPInput } from "@app/components/form/IPInput.js"; import { Toggle } from "@components/form/Toggle.js";
import { Select } from "@app/components/form/Select.js"; import { renderOptions } from "@core/utils/selectEnumOptions.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { renderOptions } from "@app/core/utils/selectEnumOptions.js";
import { NetworkValidation } from "@app/validation/config/network.js"; import { NetworkValidation } from "@app/validation/config/network.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
@ -19,17 +15,13 @@ import { Protobuf } from "@meshtastic/meshtasticjs";
export const Network = (): JSX.Element => { export const Network = (): JSX.Element => {
const { config, setWorkingConfig } = useDevice(); const { config, setWorkingConfig } = useDevice();
const { const { register, handleSubmit, control, reset } = useForm<NetworkValidation>(
register, {
handleSubmit, mode: "onChange",
formState: { errors, isDirty }, defaultValues: config.network,
control, resolver: classValidatorResolver(NetworkValidation)
reset }
} = useForm<NetworkValidation>({ );
mode: "onChange",
defaultValues: config.network,
resolver: classValidatorResolver(NetworkValidation)
});
const wifiEnabled = useWatch({ const wifiEnabled = useWatch({
control, control,
@ -66,19 +58,6 @@ export const Network = (): JSX.Element => {
return ( return (
<Form onSubmit={onSubmit}> <Form onSubmit={onSubmit}>
<ErrorMessage errors={errors} name="wifiEnabled" />
<ErrorMessage errors={errors} name="wifiMode" />
<ErrorMessage errors={errors} name="wifiSsid" />
<ErrorMessage errors={errors} name="wifiPsk" />
<ErrorMessage errors={errors} name="ntpServer" />
<ErrorMessage errors={errors} name="ethEnabled" />
<ErrorMessage errors={errors} name="addressMode" />
<ErrorMessage errors={errors} name="ethConfig" />
<ErrorMessage errors={errors} name="ip" />
<ErrorMessage errors={errors} name="gateway" />
<ErrorMessage errors={errors} name="subnet" />
<ErrorMessage errors={errors} name="dns" />
<FormSection title="WiFi Config"> <FormSection title="WiFi Config">
<Controller <Controller
name="wifiEnabled" name="wifiEnabled"
@ -95,7 +74,6 @@ export const Network = (): JSX.Element => {
<Input <Input
label="SSID" label="SSID"
description="Network name" description="Network name"
error={errors.wifiSsid?.message}
disabled={!wifiEnabled} disabled={!wifiEnabled}
{...register("wifiSsid", { disabled: !wifiEnabled })} {...register("wifiSsid", { disabled: !wifiEnabled })}
/> />
@ -103,7 +81,6 @@ export const Network = (): JSX.Element => {
label="PSK" label="PSK"
type="password" type="password"
description="Network password" description="Network password"
error={errors.wifiPsk?.message}
disabled={!wifiEnabled} disabled={!wifiEnabled}
{...register("wifiPsk", { disabled: !wifiEnabled })} {...register("wifiPsk", { disabled: !wifiEnabled })}
/> />
@ -138,26 +115,21 @@ export const Network = (): JSX.Element => {
<IPInput <IPInput
label="IP" label="IP"
description="IP Address" description="IP Address"
error={errors.ipv4Config?.ip?.message}
{...register("ipv4Config.ip", { valueAsNumber: true })} {...register("ipv4Config.ip", { valueAsNumber: true })}
/> />
<IPInput <IPInput
label="Gateway" label="Gateway"
description="Default Gateway" description="Default Gateway"
error={errors.ipv4Config?.gateway?.message}
{...register("ipv4Config.gateway", { valueAsNumber: true })} {...register("ipv4Config.gateway", { valueAsNumber: true })}
/> />
<IPInput <IPInput
label="Subnet" label="Subnet"
description="Subnet Mask" description="Subnet Mask"
error={errors.ipv4Config?.subnet?.message}
{...register("ipv4Config.subnet", { valueAsNumber: true })} {...register("ipv4Config.subnet", { valueAsNumber: true })}
/> />
<IPInput <IPInput
label="DNS" label="DNS"
// type="number" //prevent
description="DNS Server" description="DNS Server"
error={errors.ipv4Config?.dns?.message}
{...register("ipv4Config.dns", { valueAsNumber: true })} {...register("ipv4Config.dns", { valueAsNumber: true })}
/> />
</> </>
@ -166,13 +138,11 @@ export const Network = (): JSX.Element => {
<Input <Input
label="NTP Server" label="NTP Server"
description="NTP server for time synchronization" description="NTP server for time synchronization"
error={errors.ntpServer?.message}
{...register("ntpServer")} {...register("ntpServer")}
/> />
<Input <Input
label="Rsyslog Server" label="Rsyslog Server"
description="Rsyslog server for external logging" description="Rsyslog server for external logging"
error={errors.rsyslogServer?.message}
{...register("rsyslogServer")} {...register("rsyslogServer")}
/> />
</Form> </Form>

48
src/components/PageComponents/Config/Position.tsx

@ -1,13 +1,9 @@
import type React from "react";
import { useEffect } from "react"; import { useEffect } from "react";
import { Controller, useForm, useWatch } from "react-hook-form"; import { Controller, useForm, useWatch } from "react-hook-form";
import { toast } from "react-hot-toast"; import { BitwiseSelect } from "@components/form/BitwiseSelect.js";
import { FormSection } from "@components/form/FormSection.js";
import { BitwiseSelect } from "@app/components/form/BitwiseSelect.js"; import { Input } from "@components/form/Input.js";
import { FormSection } from "@app/components/form/FormSection.js"; import { Toggle } from "@components/form/Toggle.js";
import { Input } from "@app/components/form/Input.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { PositionValidation } from "@app/validation/config/position.js"; import { PositionValidation } from "@app/validation/config/position.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
@ -19,22 +15,17 @@ export const Position = (): JSX.Element => {
const myNode = nodes.find((n) => n.data.num === hardware.myNodeNum); const myNode = nodes.find((n) => n.data.num === hardware.myNodeNum);
const { const { register, handleSubmit, reset, control } =
register, useForm<PositionValidation>({
handleSubmit, mode: "onChange",
formState: { errors, isDirty }, defaultValues: {
reset, fixedAlt: myNode?.data.position?.altitude,
control fixedLat: (myNode?.data.position?.latitudeI ?? 0) / 1e7,
} = useForm<PositionValidation>({ fixedLng: (myNode?.data.position?.longitudeI ?? 0) / 1e7,
mode: "onChange", ...config.position
defaultValues: { },
fixedAlt: myNode?.data.position?.altitude, resolver: classValidatorResolver(PositionValidation)
fixedLat: (myNode?.data.position?.latitudeI ?? 0) / 1e7, });
fixedLng: (myNode?.data.position?.longitudeI ?? 0) / 1e7,
...config.position
},
resolver: classValidatorResolver(PositionValidation)
});
const fixedPositionEnabled = useWatch({ const fixedPositionEnabled = useWatch({
control, control,
@ -153,7 +144,6 @@ export const Position = (): JSX.Element => {
<BitwiseSelect <BitwiseSelect
label="Position Flags" label="Position Flags"
description="Configuration options for POSITION messages" description="Configuration options for POSITION messages"
error={error?.message}
selected={value} selected={value}
decodeEnun={Protobuf.Config_PositionConfig_PositionFlags} decodeEnun={Protobuf.Config_PositionConfig_PositionFlags}
onChange={onChange} onChange={onChange}
@ -180,7 +170,6 @@ export const Position = (): JSX.Element => {
suffix="m" suffix="m"
label="Altitude" label="Altitude"
type="number" type="number"
error={errors.fixedAlt?.message}
disabled={!fixedPositionEnabled} disabled={!fixedPositionEnabled}
{...register("fixedAlt", { valueAsNumber: true })} {...register("fixedAlt", { valueAsNumber: true })}
/> />
@ -188,7 +177,6 @@ export const Position = (): JSX.Element => {
suffix="°" suffix="°"
label="Latitude" label="Latitude"
type="number" type="number"
error={errors.fixedLat?.message}
disabled={!fixedPositionEnabled} disabled={!fixedPositionEnabled}
{...register("fixedLat", { valueAsNumber: true })} {...register("fixedLat", { valueAsNumber: true })}
/> />
@ -196,7 +184,6 @@ export const Position = (): JSX.Element => {
suffix="°" suffix="°"
label="Longitude" label="Longitude"
type="number" type="number"
error={errors.fixedLng?.message}
disabled={!fixedPositionEnabled} disabled={!fixedPositionEnabled}
{...register("fixedLng", { valueAsNumber: true })} {...register("fixedLng", { valueAsNumber: true })}
/> />
@ -209,7 +196,6 @@ export const Position = (): JSX.Element => {
label="Broadcast Interval" label="Broadcast Interval"
description="How often your position is sent out over the mesh" description="How often your position is sent out over the mesh"
type="number" type="number"
error={errors.positionBroadcastSecs?.message}
{...register("positionBroadcastSecs", { valueAsNumber: true })} {...register("positionBroadcastSecs", { valueAsNumber: true })}
/> />
<Input <Input
@ -217,7 +203,6 @@ export const Position = (): JSX.Element => {
label="GPS Update Interval" label="GPS Update Interval"
description="How often a GPS fix should be acquired" description="How often a GPS fix should be acquired"
type="number" type="number"
error={errors.gpsUpdateInterval?.message}
{...register("gpsUpdateInterval", { valueAsNumber: true })} {...register("gpsUpdateInterval", { valueAsNumber: true })}
/> />
<Input <Input
@ -225,7 +210,6 @@ export const Position = (): JSX.Element => {
label="Fix Attempt Duration" label="Fix Attempt Duration"
description="How long the device will try to get a fix for" description="How long the device will try to get a fix for"
type="number" type="number"
error={errors.gpsAttemptTime?.message}
{...register("gpsAttemptTime", { valueAsNumber: true })} {...register("gpsAttemptTime", { valueAsNumber: true })}
/> />
</FormSection> </FormSection>
@ -233,14 +217,12 @@ export const Position = (): JSX.Element => {
label="RX Pin" label="RX Pin"
description="GPS Module RX pin override" description="GPS Module RX pin override"
type="number" type="number"
error={errors.rxGpio?.message}
{...register("rxGpio", { valueAsNumber: true })} {...register("rxGpio", { valueAsNumber: true })}
/> />
<Input <Input
label="TX Pin" label="TX Pin"
description="GPS Module TX pin override" description="GPS Module TX pin override"
type="number" type="number"
error={errors.txGpio?.message}
{...register("txGpio", { valueAsNumber: true })} {...register("txGpio", { valueAsNumber: true })}
/> />
</Form> </Form>

25
src/components/PageComponents/Config/Power.tsx

@ -1,12 +1,8 @@
import type React from "react";
import { useEffect } from "react"; import { useEffect } from "react";
import { Controller, useForm } from "react-hook-form"; import { Controller, useForm } from "react-hook-form";
import { toast } from "react-hot-toast"; import { FormSection } from "@components/form/FormSection.js";
import { Input } from "@components/form/Input.js";
import { FormSection } from "@app/components/form/FormSection.js"; import { Toggle } from "@components/form/Toggle.js";
import { Input } from "@app/components/form/Input.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { PowerValidation } from "@app/validation/config/power.js"; import { PowerValidation } from "@app/validation/config/power.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
@ -15,13 +11,7 @@ import { Protobuf } from "@meshtastic/meshtasticjs";
export const Power = (): JSX.Element => { export const Power = (): JSX.Element => {
const { config, setWorkingConfig } = useDevice(); const { config, setWorkingConfig } = useDevice();
const { const { register, handleSubmit, reset, control } = useForm<PowerValidation>({
register,
handleSubmit,
formState: { errors, isDirty },
reset,
control
} = useForm<PowerValidation>({
mode: "onChange", mode: "onChange",
defaultValues: config.power, defaultValues: config.power,
resolver: classValidatorResolver(PowerValidation) resolver: classValidatorResolver(PowerValidation)
@ -49,7 +39,6 @@ export const Power = (): JSX.Element => {
description="Automatically shutdown node after this long when on battery, 0 for indefinite" description="Automatically shutdown node after this long when on battery, 0 for indefinite"
suffix="Seconds" suffix="Seconds"
type="number" type="number"
error={errors.onBatteryShutdownAfterSecs?.message}
{...register("onBatteryShutdownAfterSecs", { valueAsNumber: true })} {...register("onBatteryShutdownAfterSecs", { valueAsNumber: true })}
/> />
<Controller <Controller
@ -68,7 +57,6 @@ export const Power = (): JSX.Element => {
label="ADC Multiplier Override ratio" label="ADC Multiplier Override ratio"
description="Used for tweaking battery voltage reading" description="Used for tweaking battery voltage reading"
type="number" type="number"
error={errors.adcMultiplierOverride?.message}
{...register("adcMultiplierOverride", { valueAsNumber: true })} {...register("adcMultiplierOverride", { valueAsNumber: true })}
/> />
<FormSection title="Sleep Settings"> <FormSection title="Sleep Settings">
@ -77,7 +65,6 @@ export const Power = (): JSX.Element => {
description="Minimum amount of time the device will stay awake for after receiving a packet" description="Minimum amount of time the device will stay awake for after receiving a packet"
suffix="Seconds" suffix="Seconds"
type="number" type="number"
error={errors.minWakeSecs?.message}
{...register("minWakeSecs", { valueAsNumber: true })} {...register("minWakeSecs", { valueAsNumber: true })}
/> />
<Input <Input
@ -85,7 +72,6 @@ export const Power = (): JSX.Element => {
description="The device will enter super deep sleep after this time" description="The device will enter super deep sleep after this time"
suffix="Seconds" suffix="Seconds"
type="number" type="number"
error={errors.meshSdsTimeoutSecs?.message}
{...register("meshSdsTimeoutSecs", { valueAsNumber: true })} {...register("meshSdsTimeoutSecs", { valueAsNumber: true })}
/> />
<Input <Input
@ -93,7 +79,6 @@ export const Power = (): JSX.Element => {
description="How long the device will be in super deep sleep for" description="How long the device will be in super deep sleep for"
suffix="Seconds" suffix="Seconds"
type="number" type="number"
error={errors.sdsSecs?.message}
{...register("sdsSecs", { valueAsNumber: true })} {...register("sdsSecs", { valueAsNumber: true })}
/> />
<Input <Input
@ -101,7 +86,6 @@ export const Power = (): JSX.Element => {
description="How long the device will be in light sleep for" description="How long the device will be in light sleep for"
suffix="Seconds" suffix="Seconds"
type="number" type="number"
error={errors.lsSecs?.message}
{...register("lsSecs", { valueAsNumber: true })} {...register("lsSecs", { valueAsNumber: true })}
/> />
</FormSection> </FormSection>
@ -110,7 +94,6 @@ export const Power = (): JSX.Element => {
description="If the device does not receive a Bluetooth connection, the BLE radio will be disabled after this long" description="If the device does not receive a Bluetooth connection, the BLE radio will be disabled after this long"
suffix="Seconds" suffix="Seconds"
type="number" type="number"
error={errors.waitBluetoothSecs?.message}
{...register("waitBluetoothSecs", { valueAsNumber: true })} {...register("waitBluetoothSecs", { valueAsNumber: true })}
/> />
</Form> </Form>

26
src/components/PageComponents/Config/User.tsx

@ -1,18 +1,14 @@
import type React from "react";
import { useEffect } from "react"; import { useEffect } from "react";
import { Controller, useForm } from "react-hook-form"; import { Controller, useForm } from "react-hook-form";
import { toast } from "react-hot-toast"; import { toast } from "react-hot-toast";
import { base16 } from "rfc4648"; import { base16 } from "rfc4648";
import { Input } from "@components/form/Input.js";
import { Input } from "@app/components/form/Input.js"; import { Select } from "@components/form/Select.js";
import { Select } from "@app/components/form/Select.js"; import { Toggle } from "@components/form/Toggle.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { UserValidation } from "@app/validation/config/user.js"; import { UserValidation } from "@app/validation/config/user.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
import { renderOptions } from "@core/utils/selectEnumOptions.js"; import { renderOptions } from "@core/utils/selectEnumOptions.js";
import { ErrorMessage } from "@hookform/error-message";
import { classValidatorResolver } from "@hookform/resolvers/class-validator"; import { classValidatorResolver } from "@hookform/resolvers/class-validator";
import { Protobuf } from "@meshtastic/meshtasticjs"; import { Protobuf } from "@meshtastic/meshtasticjs";
@ -21,13 +17,7 @@ export const User = (): JSX.Element => {
const myNode = nodes.find((n) => n.data.num === hardware.myNodeNum); const myNode = nodes.find((n) => n.data.num === hardware.myNodeNum);
const { const { register, handleSubmit, reset, control } = useForm<UserValidation>({
register,
handleSubmit,
formState: { errors, isDirty },
reset,
control
} = useForm<UserValidation>({
defaultValues: myNode?.data.user, defaultValues: myNode?.data.user,
resolver: classValidatorResolver(UserValidation) resolver: classValidatorResolver(UserValidation)
}); });
@ -61,12 +51,7 @@ export const User = (): JSX.Element => {
}); });
return ( return (
<Form <Form onSubmit={onSubmit}>
onSubmit={onSubmit}
>
<ErrorMessage errors={errors} name="longName" />
<ErrorMessage errors={errors} name="shortName" />
<ErrorMessage errors={errors} name="isLicensed" />
<Input <Input
label="Device Name" label="Device Name"
description="Personalised name for this device." description="Personalised name for this device."
@ -105,7 +90,6 @@ export const User = (): JSX.Element => {
label="Device ID" label="Device ID"
disabled disabled
description="Preset unique identifier for this device." description="Preset unique identifier for this device."
error={errors.id?.message}
value={myNode?.data.user?.id} value={myNode?.data.user?.id}
/> />
<Select <Select

4
src/components/PageComponents/Connect/BLE.tsx

@ -1,7 +1,5 @@
import type React from "react";
import { useCallback, useEffect, useState } from "react"; import { useCallback, useEffect, useState } from "react";
import { Mono } from "@components/generic/Mono.js";
import { Mono } from "@app/components/generic/Mono.js";
import { Button } from "@components/form/Button.js"; import { Button } from "@components/form/Button.js";
import { useAppStore } from "@core/stores/appStore.js"; import { useAppStore } from "@core/stores/appStore.js";
import { useDeviceStore } from "@core/stores/deviceStore.js"; import { useDeviceStore } from "@core/stores/deviceStore.js";

7
src/components/PageComponents/Connect/HTTP.tsx

@ -1,9 +1,6 @@
import type React from "react";
import { Controller, useForm, useWatch } from "react-hook-form"; import { Controller, useForm, useWatch } from "react-hook-form";
import { Input } from "@components/form/Input.js";
import { Input } from "@app/components/form/Input.js"; import { Toggle } from "@components/form/Toggle.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { Button } from "@components/form/Button.js"; import { Button } from "@components/form/Button.js";
import { useAppStore } from "@core/stores/appStore.js"; import { useAppStore } from "@core/stores/appStore.js";
import { useDeviceStore } from "@core/stores/deviceStore.js"; import { useDeviceStore } from "@core/stores/deviceStore.js";

4
src/components/PageComponents/Connect/Serial.tsx

@ -1,7 +1,5 @@
import type React from "react";
import { useCallback, useEffect, useState } from "react"; import { useCallback, useEffect, useState } from "react";
import { Mono } from "@components/generic/Mono.js";
import { Mono } from "@app/components/generic/Mono.js";
import { Button } from "@components/form/Button.js"; import { Button } from "@components/form/Button.js";
import { useAppStore } from "@core/stores/appStore.js"; import { useAppStore } from "@core/stores/appStore.js";
import { useDeviceStore } from "@core/stores/deviceStore.js"; import { useDeviceStore } from "@core/stores/deviceStore.js";

2
src/components/PageComponents/Map/MapControlls.tsx

@ -2,7 +2,7 @@ import { useEffect } from "react";
import { useMap } from "react-map-gl"; import { useMap } from "react-map-gl";
import { useDevice } from "@app/core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
import { import {
MagnifyingGlassMinusIcon, MagnifyingGlassMinusIcon,
MagnifyingGlassPlusIcon, MagnifyingGlassPlusIcon,

2
src/components/PageComponents/Messages/ChannelChat.tsx

@ -1,5 +1,3 @@
import type React from "react";
import { Message } from "@components/PageComponents/Messages/Message.js"; import { Message } from "@components/PageComponents/Messages/Message.js";
import { MessageInput } from "@components/PageComponents/Messages/MessageInput.js"; import { MessageInput } from "@components/PageComponents/Messages/MessageInput.js";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";

2
src/components/PageComponents/Messages/Message.tsx

@ -1,5 +1,3 @@
import type React from "react";
import { WaypointMessage } from "@components/PageComponents/Messages/WaypointMessage.js"; import { WaypointMessage } from "@components/PageComponents/Messages/WaypointMessage.js";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
import type { AllMessageTypes } from "@core/stores/deviceStore.js"; import type { AllMessageTypes } from "@core/stores/deviceStore.js";

6
src/components/PageComponents/Messages/MessageInput.tsx

@ -1,7 +1,5 @@
import type React from "react"; import { IconButton } from "@components/form/IconButton.js";
import { Input } from "@components/form/Input.js";
import { IconButton } from "@app/components/form/IconButton.js";
import { Input } from "@app/components/form/Input.js";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
import type { Channel } from "@core/stores/deviceStore.js"; import type { Channel } from "@core/stores/deviceStore.js";
import { MapPinIcon, PaperAirplaneIcon } from "@heroicons/react/24/outline"; import { MapPinIcon, PaperAirplaneIcon } from "@heroicons/react/24/outline";

6
src/components/PageComponents/Messages/NewLocationMessage.tsx

@ -1,7 +1,5 @@
import type React from "react"; import { Input } from "@components/form/Input.js";
import { Select } from "@components/form/Select.js";
import { Input } from "@app/components/form/Input.js";
import { Select } from "@app/components/form/Select.js";
import { Button } from "@components/form/Button.js"; import { Button } from "@components/form/Button.js";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
import { renderOptions } from "@core/utils/selectEnumOptions.js"; import { renderOptions } from "@core/utils/selectEnumOptions.js";

4
src/components/PageComponents/Messages/WaypointMessage.tsx

@ -1,6 +1,4 @@
import type React from "react"; import { useDevice } from "@core/providers/useDevice.js";
import { useDevice } from "@app/core/providers/useDevice.js";
import { toMGRS } from "@core/utils/toMGRS.js"; import { toMGRS } from "@core/utils/toMGRS.js";
import { MapPinIcon } from "@heroicons/react/24/outline"; import { MapPinIcon } from "@heroicons/react/24/outline";

55
src/components/PageComponents/ModuleConfig/Audio.tsx

@ -1,12 +1,8 @@
import type React from "react";
import { useEffect } from "react"; import { useEffect } from "react";
import { Controller, useForm } from "react-hook-form"; import { Controller, useForm } from "react-hook-form";
import { toast } from "react-hot-toast"; import { Input } from "@components/form/Input.js";
import { Select } from "@components/form/Select.js";
import { Input } from "@app/components/form/Input.js"; import { Toggle } from "@components/form/Toggle.js";
import { Select } from "@app/components/form/Select.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { AudioValidation } from "@app/validation/moduleConfig/audio.js"; import { AudioValidation } from "@app/validation/moduleConfig/audio.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
@ -15,14 +11,9 @@ import { classValidatorResolver } from "@hookform/resolvers/class-validator";
import { Protobuf } from "@meshtastic/meshtasticjs"; import { Protobuf } from "@meshtastic/meshtasticjs";
export const Audio = (): JSX.Element => { export const Audio = (): JSX.Element => {
const { moduleConfig, connection, setModuleConfig } = useDevice(); const { moduleConfig, setWorkingModuleConfig } = useDevice();
const { const { register, handleSubmit, reset, control } = useForm<AudioValidation>({
register, mode: "onChange",
handleSubmit,
formState: { isDirty },
reset,
control
} = useForm<AudioValidation>({
defaultValues: moduleConfig.audio, defaultValues: moduleConfig.audio,
resolver: classValidatorResolver(AudioValidation) resolver: classValidatorResolver(AudioValidation)
}); });
@ -32,34 +23,14 @@ export const Audio = (): JSX.Element => {
}, [reset, moduleConfig.audio]); }, [reset, moduleConfig.audio]);
const onSubmit = handleSubmit((data) => { const onSubmit = handleSubmit((data) => {
if (connection) { setWorkingModuleConfig(
void toast.promise( new Protobuf.ModuleConfig({
connection payloadVariant: {
.setModuleConfig( case: "audio",
new Protobuf.ModuleConfig({ value: data
payloadVariant: {
case: "audio",
value: data
}
})
)
.then(() =>
setModuleConfig(
new Protobuf.ModuleConfig({
payloadVariant: {
case: "audio",
value: data
}
})
)
),
{
loading: "Saving...",
success: "Saved Audio Config, Restarting Node",
error: "No response received"
} }
); })
} );
}); });
return ( return (

47
src/components/PageComponents/ModuleConfig/CannedMessage.tsx

@ -1,12 +1,8 @@
import type React from "react";
import { useEffect } from "react"; import { useEffect } from "react";
import { Controller, useForm, useWatch } from "react-hook-form"; import { Controller, useForm, useWatch } from "react-hook-form";
import { toast } from "react-hot-toast"; import { Input } from "@components/form/Input.js";
import { Select } from "@components/form/Select.js";
import { Input } from "@app/components/form/Input.js"; import { Toggle } from "@components/form/Toggle.js";
import { Select } from "@app/components/form/Select.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { CannedMessageValidation } from "@app/validation/moduleConfig/cannedMessage.js"; import { CannedMessageValidation } from "@app/validation/moduleConfig/cannedMessage.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
@ -15,7 +11,7 @@ import { classValidatorResolver } from "@hookform/resolvers/class-validator";
import { Protobuf } from "@meshtastic/meshtasticjs"; import { Protobuf } from "@meshtastic/meshtasticjs";
export const CannedMessage = (): JSX.Element => { export const CannedMessage = (): JSX.Element => {
const { moduleConfig, connection, setModuleConfig } = useDevice(); const { moduleConfig, setWorkingModuleConfig } = useDevice();
const { const {
register, register,
handleSubmit, handleSubmit,
@ -23,6 +19,7 @@ export const CannedMessage = (): JSX.Element => {
reset, reset,
control control
} = useForm<CannedMessageValidation>({ } = useForm<CannedMessageValidation>({
mode: "onChange",
defaultValues: moduleConfig.cannedMessage, defaultValues: moduleConfig.cannedMessage,
resolver: classValidatorResolver(CannedMessageValidation) resolver: classValidatorResolver(CannedMessageValidation)
}); });
@ -38,34 +35,14 @@ export const CannedMessage = (): JSX.Element => {
}, [reset, moduleConfig.cannedMessage]); }, [reset, moduleConfig.cannedMessage]);
const onSubmit = handleSubmit((data) => { const onSubmit = handleSubmit((data) => {
if (connection) { setWorkingModuleConfig(
void toast.promise( new Protobuf.ModuleConfig({
connection payloadVariant: {
.setModuleConfig( case: "cannedMessage",
new Protobuf.ModuleConfig({ value: data
payloadVariant: {
case: "cannedMessage",
value: data
}
})
)
.then(() =>
setModuleConfig(
new Protobuf.ModuleConfig({
payloadVariant: {
case: "cannedMessage",
value: data
}
})
)
),
{
loading: "Saving...",
success: "Saved Canned Message Config, Restarting Node",
error: "No response received"
} }
); })
} );
}); });
return ( return (

60
src/components/PageComponents/ModuleConfig/ExternalNotification.tsx

@ -1,11 +1,7 @@
import type React from "react";
import { useEffect } from "react"; import { useEffect } from "react";
import { Controller, useForm, useWatch } from "react-hook-form"; import { Controller, useForm, useWatch } from "react-hook-form";
import { toast } from "react-hot-toast"; import { Input } from "@components/form/Input.js";
import { Toggle } from "@components/form/Toggle.js";
import { Input } from "@app/components/form/Input.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { ExternalNotificationValidation } from "@app/validation/moduleConfig/externalNotification.js"; import { ExternalNotificationValidation } from "@app/validation/moduleConfig/externalNotification.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
@ -13,50 +9,26 @@ import { classValidatorResolver } from "@hookform/resolvers/class-validator";
import { Protobuf } from "@meshtastic/meshtasticjs"; import { Protobuf } from "@meshtastic/meshtasticjs";
export const ExternalNotification = (): JSX.Element => { export const ExternalNotification = (): JSX.Element => {
const { moduleConfig, connection, setModuleConfig } = useDevice(); const { moduleConfig, setWorkingModuleConfig } = useDevice();
const { const { register, handleSubmit, reset, control } =
register, useForm<ExternalNotificationValidation>({
handleSubmit, mode: "onChange",
formState: { errors, isDirty }, defaultValues: moduleConfig.externalNotification,
reset, resolver: classValidatorResolver(ExternalNotificationValidation)
control });
} = useForm<ExternalNotificationValidation>({
defaultValues: moduleConfig.externalNotification,
resolver: classValidatorResolver(ExternalNotificationValidation)
});
useEffect(() => { useEffect(() => {
reset(moduleConfig.externalNotification); reset(moduleConfig.externalNotification);
}, [reset, moduleConfig.externalNotification]); }, [reset, moduleConfig.externalNotification]);
const onSubmit = handleSubmit((data) => { const onSubmit = handleSubmit((data) => {
if (connection) { setWorkingModuleConfig(
void toast.promise( new Protobuf.ModuleConfig({
connection payloadVariant: {
.setModuleConfig( case: "externalNotification",
new Protobuf.ModuleConfig({ value: data
payloadVariant: {
case: "externalNotification",
value: data
}
})
)
.then(() =>
setModuleConfig(
new Protobuf.ModuleConfig({
payloadVariant: {
case: "externalNotification",
value: data
}
})
)
),
{
loading: "Saving...",
success: "Saved External Notification Config, Restarting Node",
error: "No response received"
} }
); })
} );
}); });
const moduleEnabled = useWatch({ const moduleEnabled = useWatch({

47
src/components/PageComponents/ModuleConfig/MQTT.tsx

@ -1,11 +1,7 @@
import type React from "react";
import { useEffect } from "react"; import { useEffect } from "react";
import { Controller, useForm, useWatch } from "react-hook-form"; import { Controller, useForm, useWatch } from "react-hook-form";
import { toast } from "react-hot-toast"; import { Input } from "@components/form/Input.js";
import { Toggle } from "@components/form/Toggle.js";
import { Input } from "@app/components/form/Input.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { MQTTValidation } from "@app/validation/moduleConfig/mqtt.js"; import { MQTTValidation } from "@app/validation/moduleConfig/mqtt.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
@ -13,7 +9,7 @@ import { classValidatorResolver } from "@hookform/resolvers/class-validator";
import { Protobuf } from "@meshtastic/meshtasticjs"; import { Protobuf } from "@meshtastic/meshtasticjs";
export const MQTT = (): JSX.Element => { export const MQTT = (): JSX.Element => {
const { moduleConfig, connection, setModuleConfig } = useDevice(); const { moduleConfig, setWorkingModuleConfig } = useDevice();
const { const {
register, register,
handleSubmit, handleSubmit,
@ -21,6 +17,7 @@ export const MQTT = (): JSX.Element => {
reset, reset,
control control
} = useForm<MQTTValidation>({ } = useForm<MQTTValidation>({
mode: "onChange",
defaultValues: moduleConfig.mqtt, defaultValues: moduleConfig.mqtt,
resolver: classValidatorResolver(MQTTValidation) resolver: classValidatorResolver(MQTTValidation)
}); });
@ -36,34 +33,14 @@ export const MQTT = (): JSX.Element => {
}, [reset, moduleConfig.mqtt]); }, [reset, moduleConfig.mqtt]);
const onSubmit = handleSubmit((data) => { const onSubmit = handleSubmit((data) => {
if (connection) { setWorkingModuleConfig(
void toast.promise( new Protobuf.ModuleConfig({
connection payloadVariant: {
.setModuleConfig( case: "mqtt",
new Protobuf.ModuleConfig({ value: data
payloadVariant: {
case: "mqtt",
value: data
}
})
)
.then(() =>
setModuleConfig(
new Protobuf.ModuleConfig({
payloadVariant: {
case: "mqtt",
value: data
}
})
)
),
{
loading: "Saving...",
success: "Saved MQTT Config, Restarting Node",
error: "No response received"
} }
); })
} );
}); });
return ( return (
@ -82,7 +59,7 @@ export const MQTT = (): JSX.Element => {
/> />
<Input <Input
label="MQTT Server Address" label="MQTT Server Address"
//description="Description" description="Description"
disabled={!moduleEnabled} disabled={!moduleEnabled}
{...register("address")} {...register("address")}
/> />

64
src/components/PageComponents/ModuleConfig/RangeTest.tsx

@ -1,11 +1,7 @@
import type React from "react";
import { useEffect } from "react"; import { useEffect } from "react";
import { Controller, useForm, useWatch } from "react-hook-form"; import { Controller, useForm, useWatch } from "react-hook-form";
import { toast } from "react-hot-toast"; import { Input } from "@components/form/Input.js";
import { Toggle } from "@components/form/Toggle.js";
import { Input } from "@app/components/form/Input.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { RangeTestValidation } from "@app/validation/moduleConfig/rangeTest.js"; import { RangeTestValidation } from "@app/validation/moduleConfig/rangeTest.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
@ -13,51 +9,27 @@ import { classValidatorResolver } from "@hookform/resolvers/class-validator";
import { Protobuf } from "@meshtastic/meshtasticjs"; import { Protobuf } from "@meshtastic/meshtasticjs";
export const RangeTest = (): JSX.Element => { export const RangeTest = (): JSX.Element => {
const { moduleConfig, connection, setModuleConfig } = useDevice(); const { moduleConfig, setWorkingModuleConfig } = useDevice();
const { const { register, handleSubmit, reset, control } =
register, useForm<RangeTestValidation>({
handleSubmit, mode: "onChange",
formState: { errors, isDirty }, defaultValues: moduleConfig.rangeTest,
reset, resolver: classValidatorResolver(RangeTestValidation)
control });
} = useForm<RangeTestValidation>({
defaultValues: moduleConfig.rangeTest,
resolver: classValidatorResolver(RangeTestValidation)
});
useEffect(() => { useEffect(() => {
reset(moduleConfig.rangeTest); reset(moduleConfig.rangeTest);
}, [reset, moduleConfig.rangeTest]); }, [reset, moduleConfig.rangeTest]);
const onSubmit = handleSubmit((data) => { const onSubmit = handleSubmit((data) => {
if (connection) { setWorkingModuleConfig(
void toast.promise( new Protobuf.ModuleConfig({
connection payloadVariant: {
.setModuleConfig( case: "rangeTest",
new Protobuf.ModuleConfig({ value: data
payloadVariant: {
case: "rangeTest",
value: data
}
})
)
.then(() =>
setModuleConfig(
new Protobuf.ModuleConfig({
payloadVariant: {
case: "rangeTest",
value: data
}
})
)
),
{
loading: "Saving...",
success: "Saved Range Test Config, Restarting Node",
error: "No response received"
} }
); })
} );
}); });
const moduleEnabled = useWatch({ const moduleEnabled = useWatch({
@ -67,9 +39,7 @@ export const RangeTest = (): JSX.Element => {
}); });
return ( return (
<Form <Form onSubmit={onSubmit}>
onSubmit={onSubmit}
>
<Controller <Controller
name="enabled" name="enabled"
control={control} control={control}

57
src/components/PageComponents/ModuleConfig/Serial.tsx

@ -1,28 +1,19 @@
import type React from "react";
import { useEffect } from "react"; import { useEffect } from "react";
import { Controller, useForm, useWatch } from "react-hook-form"; import { Controller, useForm, useWatch } from "react-hook-form";
import { toast } from "react-hot-toast"; import { Input } from "@components/form/Input.js";
import { Toggle } from "@components/form/Toggle.js";
import { Input } from "@app/components/form/Input.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { SerialValidation } from "@app/validation/moduleConfig/serial.js"; import { SerialValidation } from "@app/validation/moduleConfig/serial.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
import { classValidatorResolver } from "@hookform/resolvers/class-validator"; import { classValidatorResolver } from "@hookform/resolvers/class-validator";
import { Protobuf } from "@meshtastic/meshtasticjs"; import { Protobuf } from "@meshtastic/meshtasticjs";
import { renderOptions } from "@app/core/utils/selectEnumOptions"; import { renderOptions } from "@core/utils/selectEnumOptions";
import { Select } from "@app/components/form/Select"; import { Select } from "@components/form/Select";
export const Serial = (): JSX.Element => { export const Serial = (): JSX.Element => {
const { moduleConfig, connection, setModuleConfig } = useDevice(); const { moduleConfig, setWorkingModuleConfig } = useDevice();
const { const { register, handleSubmit, reset, control } = useForm<SerialValidation>({
register, mode: "onChange",
handleSubmit,
formState: { errors, isDirty },
reset,
control
} = useForm<SerialValidation>({
defaultValues: moduleConfig.serial, defaultValues: moduleConfig.serial,
resolver: classValidatorResolver(SerialValidation) resolver: classValidatorResolver(SerialValidation)
}); });
@ -32,34 +23,14 @@ export const Serial = (): JSX.Element => {
}, [reset, moduleConfig.serial]); }, [reset, moduleConfig.serial]);
const onSubmit = handleSubmit((data) => { const onSubmit = handleSubmit((data) => {
if (connection) { setWorkingModuleConfig(
void toast.promise( new Protobuf.ModuleConfig({
connection payloadVariant: {
.setModuleConfig( case: "serial",
new Protobuf.ModuleConfig({ value: data
payloadVariant: {
case: "serial",
value: data
}
})
)
.then(() =>
setModuleConfig(
new Protobuf.ModuleConfig({
payloadVariant: {
case: "serial",
value: data
}
})
)
),
{
loading: "Saving...",
success: "Saved Serial Config, Restarting Node",
error: "No response received"
} }
); })
} );
}); });
const moduleEnabled = useWatch({ const moduleEnabled = useWatch({

60
src/components/PageComponents/ModuleConfig/StoreForward.tsx

@ -1,11 +1,7 @@
import type React from "react";
import { useEffect } from "react"; import { useEffect } from "react";
import { Controller, useForm, useWatch } from "react-hook-form"; import { Controller, useForm, useWatch } from "react-hook-form";
import { toast } from "react-hot-toast"; import { Input } from "@components/form/Input.js";
import { Toggle } from "@components/form/Toggle.js";
import { Input } from "@app/components/form/Input.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { StoreForwardValidation } from "@app/validation/moduleConfig/storeForward.js"; import { StoreForwardValidation } from "@app/validation/moduleConfig/storeForward.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
@ -13,51 +9,27 @@ import { classValidatorResolver } from "@hookform/resolvers/class-validator";
import { Protobuf } from "@meshtastic/meshtasticjs"; import { Protobuf } from "@meshtastic/meshtasticjs";
export const StoreForward = (): JSX.Element => { export const StoreForward = (): JSX.Element => {
const { moduleConfig, connection, setModuleConfig } = useDevice(); const { moduleConfig, setWorkingModuleConfig } = useDevice();
const { const { register, handleSubmit, reset, control } =
register, useForm<StoreForwardValidation>({
handleSubmit, mode: "onChange",
formState: { errors, isDirty }, defaultValues: moduleConfig.storeForward,
reset, resolver: classValidatorResolver(StoreForwardValidation)
control });
} = useForm<StoreForwardValidation>({
defaultValues: moduleConfig.storeForward,
resolver: classValidatorResolver(StoreForwardValidation)
});
useEffect(() => { useEffect(() => {
reset(moduleConfig.storeForward); reset(moduleConfig.storeForward);
}, [reset, moduleConfig.storeForward]); }, [reset, moduleConfig.storeForward]);
const onSubmit = handleSubmit((data) => { const onSubmit = handleSubmit((data) => {
if (connection) { setWorkingModuleConfig(
void toast.promise( new Protobuf.ModuleConfig({
connection payloadVariant: {
.setModuleConfig( case: "storeForward",
new Protobuf.ModuleConfig({ value: data
payloadVariant: {
case: "storeForward",
value: data
}
})
)
.then(() =>
setModuleConfig(
new Protobuf.ModuleConfig({
payloadVariant: {
case: "storeForward",
value: data
}
})
)
),
{
loading: "Saving...",
success: "Saved Store & Forward Config, Restarting Node",
error: "No response received"
} }
); })
} );
}); });
const moduleEnabled = useWatch({ const moduleEnabled = useWatch({

60
src/components/PageComponents/ModuleConfig/Telemetry.tsx

@ -1,11 +1,7 @@
import type React from "react";
import { useEffect } from "react"; import { useEffect } from "react";
import { Controller, useForm } from "react-hook-form"; import { Controller, useForm } from "react-hook-form";
import { toast } from "react-hot-toast"; import { Input } from "@components/form/Input.js";
import { Toggle } from "@components/form/Toggle.js";
import { Input } from "@app/components/form/Input.js";
import { Toggle } from "@app/components/form/Toggle.js";
import { TelemetryValidation } from "@app/validation/moduleConfig/telemetry.js"; import { TelemetryValidation } from "@app/validation/moduleConfig/telemetry.js";
import { Form } from "@components/form/Form"; import { Form } from "@components/form/Form";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
@ -13,51 +9,27 @@ import { classValidatorResolver } from "@hookform/resolvers/class-validator";
import { Protobuf } from "@meshtastic/meshtasticjs"; import { Protobuf } from "@meshtastic/meshtasticjs";
export const Telemetry = (): JSX.Element => { export const Telemetry = (): JSX.Element => {
const { moduleConfig, connection, setModuleConfig } = useDevice(); const { moduleConfig, setWorkingModuleConfig } = useDevice();
const { const { register, handleSubmit, reset, control } =
register, useForm<TelemetryValidation>({
handleSubmit, mode: "onChange",
formState: { errors, isDirty }, defaultValues: moduleConfig.telemetry,
reset, resolver: classValidatorResolver(TelemetryValidation)
control });
} = useForm<TelemetryValidation>({
defaultValues: moduleConfig.telemetry,
resolver: classValidatorResolver(TelemetryValidation)
});
useEffect(() => { useEffect(() => {
reset(moduleConfig.telemetry); reset(moduleConfig.telemetry);
}, [reset, moduleConfig.telemetry]); }, [reset, moduleConfig.telemetry]);
const onSubmit = handleSubmit((data) => { const onSubmit = handleSubmit((data) => {
if (connection) { setWorkingModuleConfig(
void toast.promise( new Protobuf.ModuleConfig({
connection payloadVariant: {
.setModuleConfig( case: "telemetry",
new Protobuf.ModuleConfig({ value: data
payloadVariant: {
case: "telemetry",
value: data
}
})
)
.then(() =>
setModuleConfig(
new Protobuf.ModuleConfig({
payloadVariant: {
case: "telemetry",
value: data
}
})
)
),
{
loading: "Saving...",
success: "Saved Telemetry Config, Restarting Node",
error: "No response received"
} }
); })
} );
}); });
return ( return (

10
src/components/Sidebar.tsx

@ -1,7 +1,5 @@
import type React from "react"; import { useDevice } from "@core/providers/useDevice.js";
import { toMGRS } from "@core/utils/toMGRS.js";
import { useDevice } from "@app/core/providers/useDevice.js";
import { toMGRS } from "@app/core/utils/toMGRS.js";
import { BatteryWidget } from "@components/Widgets/BatteryWidget.js"; import { BatteryWidget } from "@components/Widgets/BatteryWidget.js";
import { DeviceWidget } from "@components/Widgets/DeviceWidget.js"; import { DeviceWidget } from "@components/Widgets/DeviceWidget.js";
import { PeersWidget } from "@components/Widgets/PeersWidget.js"; import { PeersWidget } from "@components/Widgets/PeersWidget.js";
@ -10,9 +8,7 @@ import { useAppStore } from "@core/stores/appStore.js";
import { useDeviceStore } from "@core/stores/deviceStore.js"; import { useDeviceStore } from "@core/stores/deviceStore.js";
import { CommandLineIcon } from "@heroicons/react/24/outline"; import { CommandLineIcon } from "@heroicons/react/24/outline";
import { Types } from "@meshtastic/meshtasticjs"; import { Types } from "@meshtastic/meshtasticjs";
import { Input } from "@components/form/Input.js";
import { Input } from "./form/Input.js";
import { Button } from "./form/Button.js";
export const Sidebar = (): JSX.Element => { export const Sidebar = (): JSX.Element => {
const { removeDevice } = useDeviceStore(); const { removeDevice } = useDeviceStore();

5
src/components/Widgets/BatteryWidget.tsx

@ -1,9 +1,6 @@
import type React from "react";
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
import prettyMilliseconds from "pretty-ms"; import prettyMilliseconds from "pretty-ms";
import { useDevice } from "@core/providers/useDevice.js";
import { useDevice } from "@app/core/providers/useDevice.js";
import { Battery100Icon, ClockIcon } from "@heroicons/react/24/outline"; import { Battery100Icon, ClockIcon } from "@heroicons/react/24/outline";
export interface BatteryWidgetProps { export interface BatteryWidgetProps {

4
src/components/Widgets/DeviceWidget.tsx

@ -1,5 +1,3 @@
import type React from "react";
import { Button } from "@components/form/Button.js"; import { Button } from "@components/form/Button.js";
import { Hashicon } from "@emeraldpay/hashicon-react"; import { Hashicon } from "@emeraldpay/hashicon-react";
import { XCircleIcon } from "@heroicons/react/24/outline"; import { XCircleIcon } from "@heroicons/react/24/outline";
@ -21,7 +19,7 @@ export const DeviceWidget = ({
}: DeviceWidgetProps): JSX.Element => { }: DeviceWidgetProps): JSX.Element => {
return ( return (
<div className="relative flex shrink-0 flex-col overflow-hidden rounded-md text-sm text-textPrimary"> <div className="relative flex shrink-0 flex-col overflow-hidden rounded-md text-sm text-textPrimary">
<div className="bg-backgroundPrimary flex p-3"> <div className="flex bg-backgroundPrimary p-3">
<div> <div>
<Hashicon size={96} value={nodeNum} /> <Hashicon size={96} value={nodeNum} />
</div> </div>

4
src/components/Widgets/PeersWidget.tsx

@ -1,6 +1,4 @@
import type React from "react"; import { useDevice } from "@core/providers/useDevice.js";
import { useDevice } from "@app/core/providers/useDevice.js";
import { IconButton } from "@components/form/IconButton.js"; import { IconButton } from "@components/form/IconButton.js";
import { Mono } from "@components/generic/Mono.js"; import { Mono } from "@components/generic/Mono.js";
import { import {

2
src/components/Widgets/PositionWidget.tsx

@ -1,5 +1,3 @@
import type React from "react";
import { MapPinIcon } from "@heroicons/react/24/outline"; import { MapPinIcon } from "@heroicons/react/24/outline";
export interface PositionWidgetProps { export interface PositionWidgetProps {

10
src/components/form/BitwiseSelect.tsx

@ -1,10 +1,6 @@
import React, { useState } from "react"; import React, { useEffect, useState } from "react";
import { import { bitwiseDecode, bitwiseEncode, enumLike } from "@core/utils/bitwise.js";
bitwiseDecode,
bitwiseEncode,
enumLike
} from "@app/core/utils/bitwise.js";
import { InfoWrapper } from "@components/form/InfoWrapper.js"; import { InfoWrapper } from "@components/form/InfoWrapper.js";
import { Listbox } from "@headlessui/react"; import { Listbox } from "@headlessui/react";
import { Protobuf } from "@meshtastic/meshtasticjs"; import { Protobuf } from "@meshtastic/meshtasticjs";
@ -41,7 +37,7 @@ export const BitwiseSelect = ({
}; };
}); });
React.useEffect(() => { useEffect(() => {
setDecodedSelected( setDecodedSelected(
bitwiseDecode(selected, Protobuf.Config_PositionConfig_PositionFlags).map( bitwiseDecode(selected, Protobuf.Config_PositionConfig_PositionFlags).map(
(flag) => (flag) =>

1
src/components/form/Button.tsx

@ -1,4 +1,3 @@
import type React from "react";
import type { ButtonHTMLAttributes } from "react"; import type { ButtonHTMLAttributes } from "react";
export interface ButtonProps extends ButtonHTMLAttributes<HTMLButtonElement> { export interface ButtonProps extends ButtonHTMLAttributes<HTMLButtonElement> {

1
src/components/form/Checkbox.tsx

@ -1,4 +1,3 @@
import type React from "react";
import { forwardRef, InputHTMLAttributes } from "react"; import { forwardRef, InputHTMLAttributes } from "react";
export interface CheckboxProps extends InputHTMLAttributes<HTMLInputElement> { export interface CheckboxProps extends InputHTMLAttributes<HTMLInputElement> {

5
src/components/form/Form.tsx

@ -1,8 +1,7 @@
import type React from "react"; import type { FormEvent, HTMLProps } from "react";
import type { HTMLProps } from "react";
export interface FormProps extends HTMLProps<HTMLFormElement> { export interface FormProps extends HTMLProps<HTMLFormElement> {
onSubmit?: (event: React.FormEvent<HTMLFormElement>) => Promise<void>; onSubmit?: (event: FormEvent<HTMLFormElement>) => Promise<void>;
} }
export const Form = ({ export const Form = ({

4
src/components/form/FormSection.tsx

@ -1,8 +1,8 @@
import type React from "react"; import type { ReactNode } from "react";
export interface FormSectionProps { export interface FormSectionProps {
title: string; title: string;
children: React.ReactNode; children: ReactNode;
} }
export const FormSection = ({ export const FormSection = ({

9
src/components/form/IPInput.tsx

@ -1,10 +1,7 @@
import type React from "react"; import { forwardRef, useEffect } from "react";
import { forwardRef, InputHTMLAttributes, useEffect } from "react"; import { InfoWrapper } from "@components/form/InfoWrapper.js";
import { InfoWrapper, InfoWrapperProps } from "@components/form/InfoWrapper.js";
import { ExclamationCircleIcon } from "@heroicons/react/24/outline";
import { useState } from "react"; import { useState } from "react";
import type { InputProps } from "./Input.js"; import type { InputProps } from "@components/form/Input.js";
export const IPInput = forwardRef<HTMLInputElement, InputProps>(function Input( export const IPInput = forwardRef<HTMLInputElement, InputProps>(function Input(
{ {

1
src/components/form/IconButton.tsx

@ -1,4 +1,3 @@
import type React from "react";
import type { ButtonHTMLAttributes } from "react"; import type { ButtonHTMLAttributes } from "react";
export interface IconButtonProps export interface IconButtonProps

5
src/components/form/InfoWrapper.tsx

@ -1,12 +1,11 @@
import type React from "react";
import { ExclamationCircleIcon } from "@heroicons/react/24/outline"; import { ExclamationCircleIcon } from "@heroicons/react/24/outline";
import type { ReactNode } from "react";
export interface InfoWrapperProps { export interface InfoWrapperProps {
label?: string; label?: string;
description?: string; description?: string;
error?: string; error?: string;
children: React.ReactNode; children: ReactNode;
} }
export const InfoWrapper = ({ export const InfoWrapper = ({

2
src/components/form/Input.tsx

@ -1,6 +1,4 @@
import type React from "react";
import { forwardRef, InputHTMLAttributes } from "react"; import { forwardRef, InputHTMLAttributes } from "react";
import { InfoWrapper, InfoWrapperProps } from "@components/form/InfoWrapper.js"; import { InfoWrapper, InfoWrapperProps } from "@components/form/InfoWrapper.js";
import { ExclamationCircleIcon } from "@heroicons/react/24/outline"; import { ExclamationCircleIcon } from "@heroicons/react/24/outline";

1
src/components/form/Select.tsx

@ -1,4 +1,3 @@
import type React from "react";
import { forwardRef, SelectHTMLAttributes } from "react"; import { forwardRef, SelectHTMLAttributes } from "react";
import { InfoWrapper, InfoWrapperProps } from "@components/form/InfoWrapper.js"; import { InfoWrapper, InfoWrapperProps } from "@components/form/InfoWrapper.js";

2
src/components/form/Toggle.tsx

@ -1,5 +1,3 @@
import type React from "react";
import { Switch } from "@headlessui/react"; import { Switch } from "@headlessui/react";
export interface ToggleProps { export interface ToggleProps {

9
src/components/generic/Dialog.tsx

@ -1,17 +1,16 @@
import type React from "react";
import { IconButton } from "@components/form/IconButton.js"; import { IconButton } from "@components/form/IconButton.js";
import { Dialog as DialogUI } from "@headlessui/react"; import { Dialog as DialogUI } from "@headlessui/react";
import { XMarkIcon } from "@heroicons/react/24/outline"; import { XMarkIcon } from "@heroicons/react/24/outline";
import { ThemeController } from "./ThemeController.js"; import { ThemeController } from "@components/generic/ThemeController.js";
import { Blur } from "./Blur.js"; import { Blur } from "@components/generic/Blur.js";
import type { ReactNode } from "react";
export interface DialogProps { export interface DialogProps {
title: string; title: string;
description: string; description: string;
isOpen: boolean; isOpen: boolean;
close: () => void; close: () => void;
children: React.ReactNode; children: ReactNode;
} }
export const Dialog = ({ export const Dialog = ({

2
src/components/generic/Mono.tsx

@ -1,5 +1,3 @@
import type React from "react";
export const Mono = ({ export const Mono = ({
children, children,
className, className,

2
src/components/generic/TabbedContent.tsx

@ -1,6 +1,4 @@
import type React from "react";
import { Fragment } from "react"; import { Fragment } from "react";
import { Mono } from "@components/generic/Mono"; import { Mono } from "@components/generic/Mono";
import { Tab } from "@headlessui/react"; import { Tab } from "@headlessui/react";

6
src/components/generic/ThemeController.tsx

@ -1,8 +1,8 @@
import { useAppStore } from "@app/core/stores/appStore.js"; import { useAppStore } from "@core/stores/appStore.js";
import type React from "react"; import type { ReactNode } from "react";
export interface ThemeControllerProps { export interface ThemeControllerProps {
children: React.ReactNode; children: ReactNode;
} }
export const ThemeController = ({ export const ThemeController = ({

2
src/core/utils/selectEnumOptions.tsx

@ -1,5 +1,3 @@
import type React from "react";
export const renderOptions = (enumValue: { export const renderOptions = (enumValue: {
[s: string]: string | number; [s: string]: string | number;
}): JSX.Element[] => { }): JSX.Element[] => {

3
src/index.tsx

@ -1,9 +1,6 @@
import "@app/index.css"; import "@app/index.css";
import "maplibre-gl/dist/maplibre-gl.css"; import "maplibre-gl/dist/maplibre-gl.css";
import type React from "react";
import { StrictMode } from "react"; import { StrictMode } from "react";
import { enableMapSet } from "immer"; import { enableMapSet } from "immer";
import { createRoot } from "react-dom/client"; import { createRoot } from "react-dom/client";

7
src/pages/Channels.tsx

@ -1,8 +1,5 @@
import type React from "react"; import { TabbedContent, TabType } from "@components/generic/TabbedContent";
import { Channel } from "@components/PageComponents/Channel.js";
import { TabbedContent, TabType } from "@app/components/generic/TabbedContent";
import { Channel } from "@app/components/PageComponents/Channel.js";
import { Button } from "@components/form/Button.js";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
import { import {
ArrowDownOnSquareStackIcon, ArrowDownOnSquareStackIcon,

2
src/pages/Config/AppConfig.tsx

@ -1,6 +1,4 @@
import type React from "react";
import { Fragment } from "react"; import { Fragment } from "react";
import { Map } from "@components/PageComponents/AppConfig/Map.js"; import { Map } from "@components/PageComponents/AppConfig/Map.js";
import { Tab } from "@headlessui/react"; import { Tab } from "@headlessui/react";

6
src/pages/Config/DeviceConfig.tsx

@ -1,7 +1,5 @@
import type React from "react";
import { Fragment } from "react"; import { Fragment } from "react";
import { Network } from "@components/PageComponents/Config/Network.js";
import { Network } from "@app/components/PageComponents/Config/Network.js";
import { Bluetooth } from "@components/PageComponents/Config/Bluetooth.js"; import { Bluetooth } from "@components/PageComponents/Config/Bluetooth.js";
import { Device } from "@components/PageComponents/Config/Device.js"; import { Device } from "@components/PageComponents/Config/Device.js";
import { Display } from "@components/PageComponents/Config/Display.js"; import { Display } from "@components/PageComponents/Config/Display.js";
@ -12,7 +10,7 @@ import { User } from "@components/PageComponents/Config/User.js";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
import { Tab } from "@headlessui/react"; import { Tab } from "@headlessui/react";
import { ChevronRightIcon, HomeIcon } from "@heroicons/react/24/outline"; import { ChevronRightIcon, HomeIcon } from "@heroicons/react/24/outline";
import { Button } from "@app/components/form/Button.js"; import { Button } from "@components/form/Button.js";
import { CheckIcon } from "@primer/octicons-react"; import { CheckIcon } from "@primer/octicons-react";
export const DeviceConfig = (): JSX.Element => { export const DeviceConfig = (): JSX.Element => {

4
src/pages/Config/ModuleConfig.tsx

@ -1,7 +1,5 @@
import type React from "react";
import { Fragment } from "react"; import { Fragment } from "react";
import { Audio } from "@components/PageComponents/ModuleConfig/Audio.js";
import { Audio } from "@app/components/PageComponents/ModuleConfig/Audio.js";
import { CannedMessage } from "@components/PageComponents/ModuleConfig/CannedMessage"; import { CannedMessage } from "@components/PageComponents/ModuleConfig/CannedMessage";
import { ExternalNotification } from "@components/PageComponents/ModuleConfig/ExternalNotification.js"; import { ExternalNotification } from "@components/PageComponents/ModuleConfig/ExternalNotification.js";
import { MQTT } from "@components/PageComponents/ModuleConfig/MQTT.js"; import { MQTT } from "@components/PageComponents/ModuleConfig/MQTT.js";

7
src/pages/Config/index.tsx

@ -1,8 +1,5 @@
import type React from "react"; import { TabbedContent, TabType } from "@components/generic/TabbedContent";
import { useDevice } from "@core/providers/useDevice.js";
import { Button } from "@app/components/form/Button.js";
import { TabbedContent, TabType } from "@app/components/generic/TabbedContent";
import { useDevice } from "@app/core/providers/useDevice.js";
import { import {
Cog8ToothIcon, Cog8ToothIcon,
CubeTransparentIcon, CubeTransparentIcon,

2
src/pages/Extensions/Environment.tsx

@ -1,5 +1,3 @@
import type React from "react";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
export const Environment = (): JSX.Element => { export const Environment = (): JSX.Element => {

1
src/pages/Extensions/FileBrowser.tsx

@ -1,4 +1,3 @@
import type React from "react";
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
export interface File { export interface File {

4
src/pages/Extensions/Index.tsx

@ -1,6 +1,4 @@
import type React from "react"; import { TabbedContent, TabType } from "@components/generic/TabbedContent";
import { TabbedContent, TabType } from "@app/components/generic/TabbedContent";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
import { import {
CloudIcon, CloudIcon,

7
src/pages/Map.tsx

@ -1,10 +1,7 @@
import type React from "react";
import maplibregl from "maplibre-gl"; import maplibregl from "maplibre-gl";
import { Layer, Map, Marker, Source } from "react-map-gl"; import { Layer, Map, Marker, Source } from "react-map-gl";
import { MapControlls } from "@components/PageComponents/Map/MapControlls.js";
import { MapControlls } from "@app/components/PageComponents/Map/MapControlls.js"; import { useAppStore } from "@core/stores/appStore.js";
import { useAppStore } from "@app/core/stores/appStore.js";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
import { Hashicon } from "@emeraldpay/hashicon-react"; import { Hashicon } from "@emeraldpay/hashicon-react";
import { MapPinIcon } from "@heroicons/react/24/outline"; import { MapPinIcon } from "@heroicons/react/24/outline";

8
src/pages/Messages.tsx

@ -1,10 +1,4 @@
import type React from "react"; import { TabbedContent, TabType } from "@components/generic/TabbedContent.js";
import { IconButton } from "@app/components/form/IconButton.js";
import {
TabbedContent,
TabType
} from "@app/components/generic/TabbedContent.js";
import { ChannelChat } from "@components/PageComponents/Messages/ChannelChat.js"; import { ChannelChat } from "@components/PageComponents/Messages/ChannelChat.js";
import { useDevice } from "@core/providers/useDevice.js"; import { useDevice } from "@core/providers/useDevice.js";
import { PencilIcon } from "@heroicons/react/24/outline"; import { PencilIcon } from "@heroicons/react/24/outline";

3
src/pages/Peers.tsx

@ -1,7 +1,4 @@
import type React from "react";
import { base16 } from "rfc4648"; import { base16 } from "rfc4648";
import { Mono } from "@components/generic/Mono.js"; import { Mono } from "@components/generic/Mono.js";
import { Table } from "@components/generic/Table"; import { Table } from "@components/generic/Table";
import { TimeAgo } from "@components/generic/Table/tmp/TimeAgo.js"; import { TimeAgo } from "@components/generic/Table/tmp/TimeAgo.js";

2
src/validation/appConfig/map.ts

@ -1,6 +1,6 @@
import { IsArray, IsBoolean, IsNumber, IsString } from "class-validator"; import { IsArray, IsBoolean, IsNumber, IsString } from "class-validator";
import type { RasterSource } from "@app/core/stores/appStore.js"; import type { RasterSource } from "@core/stores/appStore.js";
export class MapValidation { export class MapValidation {
@IsArray() @IsArray()

Loading…
Cancel
Save