diff --git a/src/components/Form/FormInput.tsx b/src/components/Form/FormInput.tsx index 6ecbfcdc1..45597149d 100644 --- a/src/components/Form/FormInput.tsx +++ b/src/components/Form/FormInput.tsx @@ -3,8 +3,6 @@ import type { GenericFormElementProps, } from "@components/Form/DynamicForm.tsx"; import { Input } from "@components/UI/Input.tsx"; -import type { LucideIcon } from "lucide-react"; -import { Eye, EyeOff } from "lucide-react"; import type { ChangeEventHandler } from "react"; import { useState } from "react"; import { useController, type FieldValues } from "react-hook-form"; @@ -23,10 +21,8 @@ export interface InputFieldProps extends BaseFormBuilderProps { currentValueLength?: number; showCharacterCount?: boolean; }, - action?: { - icon: LucideIcon; - onClick: () => void; - }; + showPasswordToggle?: boolean; + showCopyButton?: boolean; }; } @@ -36,8 +32,6 @@ export function GenericInput({ field, }: GenericFormElementProps>) { const { fieldLength, ...restProperties } = field.properties || {}; - - const [passwordShown, setPasswordShown] = useState(false); const [currentLength, setCurrentLength] = useState(fieldLength?.currentValueLength || 0); const { field: controllerField } = useController({ @@ -45,10 +39,6 @@ export function GenericInput({ control, }); - const togglePasswordVisiblity = () => { - setPasswordShown(!passwordShown); - }; - const handleInputChange = (e: React.ChangeEvent) => { const newValue = e.target.value; @@ -66,25 +56,19 @@ export function GenericInput({ return (
{fieldLength?.showCharacterCount && fieldLength?.max && ( -
+
{currentLength ?? fieldLength?.currentValueLength}/{fieldLength?.max}
)} diff --git a/src/components/Form/FormPasswordGenerator.tsx b/src/components/Form/FormPasswordGenerator.tsx index f4b8fbcb0..f1d0b9f44 100644 --- a/src/components/Form/FormPasswordGenerator.tsx +++ b/src/components/Form/FormPasswordGenerator.tsx @@ -4,10 +4,9 @@ import type { } from "@components/Form/DynamicForm.tsx"; import type { ButtonVariant } from "../UI/Button.tsx"; import { Generator } from "@components/UI/Generator.tsx"; -import { Eye, EyeOff } from "lucide-react"; import type { ChangeEventHandler } from "react"; -import { useState } from "react"; import { Controller, type FieldValues } from "react-hook-form"; +import { usePasswordVisibilityToggle } from "@core/hooks/usePasswordVisibilityToggle.ts"; export interface PasswordGeneratorProps extends BaseFormBuilderProps { type: "passwordGenerator"; @@ -15,7 +14,7 @@ export interface PasswordGeneratorProps extends BaseFormBuilderProps { hide?: boolean; bits?: { text: string; value: string; key: string }[]; devicePSKBitCount: number; - inputChange: ChangeEventHandler; + inputChange: ChangeEventHandler | undefined; selectChange: (event: string) => void; actionButtons: { text: string; @@ -23,6 +22,8 @@ export interface PasswordGeneratorProps extends BaseFormBuilderProps { variant: ButtonVariant; className?: string; }[]; + showPasswordToggle?: boolean; + showCopyButton?: boolean; } export function PasswordGenerator({ @@ -30,10 +31,7 @@ export function PasswordGenerator({ field, disabled, }: GenericFormElementProps>) { - const [passwordShown, setPasswordShown] = useState(false); - const togglePasswordVisiblity = () => { - setPasswordShown(!passwordShown); - }; + const { isVisible } = usePasswordVisibilityToggle() return ( ({ control={control} render={({ field: { value, ...rest } }) => ( ({ value={value} variant={field.validationText ? "invalid" : "default"} actionButtons={field.actionButtons} + showPasswordToggle={field.showPasswordToggle} + showCopyButton={field.showCopyButton} {...field.properties} {...rest} disabled={disabled} diff --git a/src/components/Form/FormWrapper.tsx b/src/components/Form/FormWrapper.tsx index 32c84a0d3..27cc01efb 100644 --- a/src/components/Form/FormWrapper.tsx +++ b/src/components/Form/FormWrapper.tsx @@ -21,10 +21,10 @@ export const FieldWrapper = ({
{/* first column = labels/heading, second column = fields, third column = gutter */} -
+
-

{description}

+

{description}

diff --git a/src/components/PageComponents/Channel.tsx b/src/components/PageComponents/Channel.tsx index f9d8a35d1..282e8aecd 100644 --- a/src/components/PageComponents/Channel.tsx +++ b/src/components/PageComponents/Channel.tsx @@ -143,6 +143,8 @@ export const Channel = ({ channel }: SettingsPanelProps) => { hide: true, properties: { value: pass, + showPasswordToggle: true, + showCopyButton: true, }, }, { diff --git a/src/components/PageComponents/Config/Security/Security.tsx b/src/components/PageComponents/Config/Security/Security.tsx index 68446e4da..21613a3fe 100644 --- a/src/components/PageComponents/Config/Security/Security.tsx +++ b/src/components/PageComponents/Config/Security/Security.tsx @@ -195,11 +195,8 @@ export const Security = () => { ], properties: { value: state.privateKey, - action: { - icon: state.privateKeyVisible ? EyeOff : Eye, - onClick: () => - dispatch({ type: "TOGGLE_PRIVATE_KEY_VISIBILITY" }), - }, + showCopyButton: true, + showPasswordToggle: true, }, }, { @@ -211,6 +208,7 @@ export const Security = () => { "Sent out to other nodes on the mesh to allow them to compute a shared secret key", properties: { value: state.publicKey, + showCopyButton: true, }, }, ], @@ -271,6 +269,7 @@ export const Security = () => { ], properties: { value: state.adminKey, + showCopyButton: true, action: { icon: state.adminKeyVisible ? EyeOff : Eye, onClick: () => diff --git a/src/components/UI/Button.tsx b/src/components/UI/Button.tsx index bb7e68e13..17c2d350c 100644 --- a/src/components/UI/Button.tsx +++ b/src/components/UI/Button.tsx @@ -1,10 +1,9 @@ import { cva, type VariantProps } from "class-variance-authority"; import * as React from "react"; - import { cn } from "@core/utils/cn.ts"; const buttonVariants = cva( - "inline-flex items-center justify-center rounded-md text-sm font-medium transition-colors focus:outline-hidden focus:ring-2 focus:ring-slate-400 focus:ring-offset-2 disabled:opacity-50 dark:focus:ring-slate-400 disabled:pointer-events-none dark:focus:ring-offset-slate-900 cursor-pointer", + "inline-flex items-center justify-center rounded-md text-sm font-medium transition-colors focus:outline-none focus:ring-2 focus:ring-slate-400 focus:ring-offset-2 disabled:opacity-50 dark:focus:ring-slate-400 disabled:cursor-not-allowed dark:focus:ring-offset-slate-900 cursor-pointer", { variants: { variant: { @@ -27,6 +26,7 @@ const buttonVariants = cva( default: "h-10 py-2 px-4", sm: "h-9 px-2 rounded-md", lg: "h-11 px-8 rounded-md", + icon: "h-10 w-10", }, }, defaultVariants: { @@ -39,26 +39,50 @@ const buttonVariants = cva( export type ButtonVariant = VariantProps["variant"]; export interface ButtonProps - extends - React.ButtonHTMLAttributes, - VariantProps { } + extends React.ButtonHTMLAttributes, + VariantProps { + icon?: React.ReactNode; + iconAlignment?: "left" | "right"; +} const Button = React.forwardRef( - ({ className, variant, size, disabled, ...props }, ref) => { + ( + { + className, + variant, + size, + disabled, + icon, + iconAlignment = "left", + children, + ...props + }, + ref, + ) => { return ( ); }, ); Button.displayName = "Button"; -export { Button, buttonVariants }; +export { Button, buttonVariants }; \ No newline at end of file diff --git a/src/components/UI/Command.tsx b/src/components/UI/Command.tsx index d230d326e..28d0114c7 100644 --- a/src/components/UI/Command.tsx +++ b/src/components/UI/Command.tsx @@ -41,7 +41,7 @@ const CommandInput = React.forwardRef< className="flex items-center border-b border-b-slate-100 px-4 dark:border-b-slate-700" cmdk-input-wrapper="" > - + ; + variant: ButtonVariant; + className?: string; +}[] export interface GeneratorProps extends React.BaseHTMLAttributes { type: "text" | "password"; @@ -17,19 +23,12 @@ export interface GeneratorProps extends React.BaseHTMLAttributes { value: string; id: string; variant: "default" | "invalid"; - actionButtons: { - text: string; - onClick: React.MouseEventHandler; - variant: ButtonVariant; - className?: string; - }[]; + actionButtons: ActionButton[]; bits?: { text: string; value: string; key: string }[]; selectChange: (event: string) => void; - inputChange: (event: React.ChangeEvent) => void; - action?: { - icon: LucideIcon; - onClick: () => void; - }; + inputChange: (event: React.ChangeEventHandler | undefined) => void; + showPasswordToggle?: boolean; + showCopyButton?: boolean; disabled?: boolean; } @@ -50,8 +49,9 @@ const Generator = ], selectChange, inputChange, - action, disabled, + showPasswordToggle, + showCopyButton, ...props }: GeneratorProps ) => { @@ -78,27 +78,28 @@ const Generator = variant={variant} value={value} onChange={inputChange} - action={action} disabled={disabled} ref={inputRef} + showCopyButton={showCopyButton} + showPasswordToggle={showPasswordToggle} /> -
+
{actionButtons?.map(({ text, onClick, variant, className }) => ( + ))} +
+ )}
)} - {action && ( - - )}
); - }, + } ); Input.displayName = "Input"; -export { Input, inputVariants }; +export { Input, inputVariants }; \ No newline at end of file diff --git a/src/components/UI/Sidebar/sidebarButton.tsx b/src/components/UI/Sidebar/sidebarButton.tsx index 7a4ce2db9..b55952db9 100644 --- a/src/components/UI/Sidebar/sidebarButton.tsx +++ b/src/components/UI/Sidebar/sidebarButton.tsx @@ -6,7 +6,7 @@ export interface SidebarButtonProps { count?: number; active?: boolean; Icon?: LucideIcon; - children: React.ReactNode; + children?: React.ReactNode; onClick?: () => void; disabled?: boolean; } diff --git a/src/core/hooks/useCopyToClipboard.ts b/src/core/hooks/useCopyToClipboard.ts new file mode 100644 index 000000000..19292bcd2 --- /dev/null +++ b/src/core/hooks/useCopyToClipboard.ts @@ -0,0 +1,51 @@ +import { useCallback, useEffect, useRef, useState } from "react"; + +interface UseCopyToClipboardProps { + timeout?: number; +} + +export function useCopyToClipboard({ timeout = 2000 }: UseCopyToClipboardProps = {}) { + const [isCopied, setIsCopied] = useState(false); + const timeoutRef = useRef(null); + + useEffect(() => { + return () => { + if (timeoutRef.current) { + globalThis.clearTimeout(timeoutRef.current); + } + }; + }, []); + + const copy = useCallback( + async (text: string) => { + if (!navigator?.clipboard) { + console.warn('Clipboard API not available'); + setIsCopied(false); + return false; + } + + if (timeoutRef.current) { + globalThis.clearTimeout(timeoutRef.current); + } + + try { + await navigator.clipboard.writeText(text); + setIsCopied(true); + + timeoutRef.current = globalThis.setTimeout(() => { + setIsCopied(false); + timeoutRef.current = null; + }, timeout); + + return true; + } catch (error) { + console.error('Failed to copy text to clipboard:', error); + setIsCopied(false); + return false; + } + }, + [timeout] + ); + + return { isCopied, copy }; +} \ No newline at end of file diff --git a/src/core/hooks/usePasswordVisibilityToggle.test.ts b/src/core/hooks/usePasswordVisibilityToggle.test.ts new file mode 100644 index 000000000..e48e5e172 --- /dev/null +++ b/src/core/hooks/usePasswordVisibilityToggle.test.ts @@ -0,0 +1,66 @@ +import { renderHook, act } from '@testing-library/react'; +import { describe, it, expect } from 'vitest'; +import { usePasswordVisibilityToggle } from './usePasswordVisibilityToggle.ts'; + +describe('usePasswordVisibilityToggle Hook', () => { + it('should initialize with visibility set to false by default', () => { + const { result } = renderHook(() => usePasswordVisibilityToggle()); + expect(result.current.isVisible).toBe(false); + expect(typeof result.current.toggleVisibility).toBe('function'); + }); + + it('should initialize with visibility set to true if initialVisible is true', () => { + const { result } = renderHook(() => + usePasswordVisibilityToggle({ initialVisible: true }) + ); + expect(result.current.isVisible).toBe(true); + }); + + it('should toggle visibility from false to true when toggleVisibility is called', () => { + const { result } = renderHook(() => usePasswordVisibilityToggle()); + expect(result.current.isVisible).toBe(false); + act(() => { + result.current.toggleVisibility(); + }); + expect(result.current.isVisible).toBe(true); + }); + + it('should toggle visibility from true to false when toggleVisibility is called', () => { + const { result } = renderHook(() => + usePasswordVisibilityToggle({ initialVisible: true }) + ); + expect(result.current.isVisible).toBe(true); + act(() => { + result.current.toggleVisibility(); + }); + expect(result.current.isVisible).toBe(false); + }); + + it('should toggle visibility correctly multiple times', () => { + const { result } = renderHook(() => usePasswordVisibilityToggle()); + expect(result.current.isVisible).toBe(false); + act(() => { + result.current.toggleVisibility(); + }); + expect(result.current.isVisible).toBe(true); + act(() => { + result.current.toggleVisibility(); + }); + expect(result.current.isVisible).toBe(false); + act(() => { + result.current.toggleVisibility(); + }); + expect(result.current.isVisible).toBe(true); + }); + + it('should return a stable toggleVisibility function reference (due to useCallback)', () => { + const { result, rerender } = renderHook(() => usePasswordVisibilityToggle()); + const initialToggleFunc = result.current.toggleVisibility; + rerender(); + expect(result.current.toggleVisibility).toBe(initialToggleFunc); + act(() => { + result.current.toggleVisibility(); + }); + expect(result.current.isVisible).toBe(true); + }); +}); \ No newline at end of file diff --git a/src/core/hooks/usePasswordVisibilityToggle.ts b/src/core/hooks/usePasswordVisibilityToggle.ts new file mode 100644 index 000000000..2d02403e3 --- /dev/null +++ b/src/core/hooks/usePasswordVisibilityToggle.ts @@ -0,0 +1,20 @@ +import { useState, useCallback } from 'react'; + +interface UsePasswordVisibilityToggleProps { + initialVisible?: boolean; +} +/** + * Manages the state for toggling password visibility. + * + * @param {boolean} [options.initialVisible=false] + * @returns {{isVisible: boolean, toggleVisibility: () => void}} + */ +export function usePasswordVisibilityToggle({ initialVisible = false }: UsePasswordVisibilityToggleProps = {}) { + const [isVisible, setIsVisible] = useState(initialVisible); + + const toggleVisibility = useCallback(() => { + setIsVisible(prev => !prev); + }, []); + + return { isVisible, toggleVisibility }; +} \ No newline at end of file