import { memo, useState } from 'react';
import { __ } from '@wordpress/i18n';
import * as Checkbox from '@radix-ui/react-checkbox';
import Icon from '@/utils/Icon';
import clsx from 'clsx';

interface CheckboxGroupInputProps {

	/** If true, the field displays an indeterminate (or overridden) state */
	indeterminate?: boolean;

	/** Label for the group used in aria-label */
	label: string;

	/** The current value; can be a boolean (for a single toggle) or an array for multi-selection */
	value: any; // eslint-disable-line @typescript-eslint/no-explicit-any

	/** Base id for the element */
	id: string;

	/** Callback when the value changes */
	onChange: ( value: any ) => void; // eslint-disable-line @typescript-eslint/no-explicit-any

	/** If true, the field is required */
	required?: boolean;

	/**
	 * If a boolean is provided then it disables the whole field; if an array is provided, only the matching option(s) are disabled
	 */
	disabled?: boolean | string[];

	/** A record of options in the format { optionId: optionLabel } */
	options: Record<string, string>;
}

/**
 * CheckboxGroupInput component
 *
 * Renders a group of checkboxes based on the options provided.
 * Supports a boolean mode when one option is provided as well as a load-more functionality.
 * @param root0
 * @param root0.indeterminate
 * @param root0.label
 * @param root0.value
 * @param root0.id
 * @param root0.onChange
 * @param root0.required
 * @param root0.disabled
 * @param root0.options
 */
const CheckboxGroupInput: React.FC<CheckboxGroupInputProps> = ({
	indeterminate,
	label,
	value,
	id,
	onChange,
	required,
	disabled,
	options = {}
}) => {

	// state for load-more expansion
	const [ loadMoreExpanded, setLoadMoreExpanded ] = useState( false );

	// Convert incoming value to an array if not already one
	let valueValidated: any = value; // eslint-disable-line @typescript-eslint/no-explicit-any
	if ( ! Array.isArray( valueValidated ) ) {
		valueValidated = '' === valueValidated ? [] : [ valueValidated ];
	}

	// Calculate isBoolean directly from options instead of using useEffect and state
	// This eliminates an unnecessary effect and state update
	const isBoolean =
		1 === Object.keys( options ).length && 'true' === Object.keys( options )[0];

	// If indeterminate is true, override the value (for display purposes)
	const displayValue = indeterminate ? true : value;
	const selected = Array.isArray( valueValidated ) ? valueValidated : [];
	const loadMoreCount = 10;
	const loadMoreEnabled = Object.keys( options ).length > loadMoreCount;

	/**
	 * Handles a change on an individual checkbox.
	 * For boolean mode, simply toggles the value.
	 * Otherwise, adds or removes the selected option.
	 * @param _checked
	 * @param option
	 */
	const handleCheckboxChange = ( _checked: boolean, option: string ) => {
		if ( isBoolean ) {
			onChange( ! displayValue );
		} else {
			const newSelected =
				selected.includes( '' + option ) ||
				selected.includes( parseInt( option ) ) ?
					selected.filter(
							( item: any ) => // eslint-disable-line @typescript-eslint/no-explicit-any
								item !== '' + option &&
								item !== parseInt( option )
						) :
					[ ...selected, option ];
			onChange( newSelected );
		}
	};

	/**
	 * Determines if an option is considered checked.
	 * @param optionId
	 */
	const isEnabled = ( optionId: string ) => {
		if ( isBoolean ) {
			return displayValue;
		}
		return (
			selected.includes( '' + optionId ) ||
			selected.includes( parseInt( optionId ) )
		);
	};

	const loadMoreHandler = () => {
		setLoadMoreExpanded( ! loadMoreExpanded );
	};

	// If disabled is not an array, treat it as a boolean disabling the entire field.
	const allDisabled = disabled && ! Array.isArray( disabled ) ? disabled : false;

	if ( 0 === Object.keys( options ).length ) {
		return <>{__( 'No options found', 'burst-statistics' )}</>;
	}

	return (
		<div className="flex flex-col gap-2">
			{Object.entries( options ).map( ([ key, optionLabel ], i ) => (
				<div
					key={key}
					className={`${! loadMoreExpanded && i > loadMoreCount - 1 ? 'hidden' : ''} flex items-center`}
				>
					<Checkbox.Root
						className={clsx( 'w-4 h-4 border border-gray-400 rounded shrink-0 disabled:opacity-50', isEnabled( key ) ? 'dark:bg-gray-200' : '' )}
						id={`${id}_${key}`}
						checked={isEnabled( key )}
						aria-label={label}
						disabled={
							allDisabled ||
							( Array.isArray( disabled ) &&
								( disabled as string[]).includes( key ) )
						}
						required={required}
						onCheckedChange={( checked ) =>
							handleCheckboxChange( !! checked, key )
						}
					>
						<Checkbox.Indicator className="flex items-center justify-center">
							<Icon
								name={indeterminate ? 'indeterminate' : 'check'}
								size={14}
								color="black"
							/>
						</Checkbox.Indicator>
					</Checkbox.Root>
					<label className="ml-2" htmlFor={`${id}_${key}`}>
						{optionLabel}
					</label>
				</div>
			) )}
			{loadMoreEnabled && ! loadMoreExpanded && (
				<div className="flex justify-start">
					<button
						onClick={loadMoreHandler}
						className="underline w-fit"
					>
						{__( 'Show more', 'burst-statistics' )}
					</button>
				</div>
			)}
			{loadMoreEnabled && loadMoreExpanded && (
				<div className="flex justify-start">
					<button
						onClick={loadMoreHandler}
						className="underline w-fit"
					>
						{__( 'Show less', 'burst-statistics' )}
					</button>
				</div>
			)}
		</div>
	);
};

export default memo( CheckboxGroupInput );
