import React, { useState, useRef, useEffect, CSSProperties } from "react";
import { SelectOption, SelectProps } from "@app/models/components";
import "./select-styles.scss";
import Option from "../option/Option";

const Select: React.FC<SelectProps> = ({
  id,
  name,
  defaultValue,
  placeholder,
  value,
  onChange,
  disabled,
  label,
  options,
  multiple,
  onClear,
  externalShowOptions = true,
  handleSelectAll,
}) => {
  const [showOptions, setShowOptions] = useState(false);
  const selectRef = useRef<HTMLButtonElement>(null);
  const optionsContainerRef = useRef<HTMLDivElement>(null);

  const handleClickOutside = (event: MouseEvent) => {
    if (
      selectRef.current &&
      optionsContainerRef.current &&
      !selectRef.current.contains(event.target as Node) &&
      !optionsContainerRef.current.contains(event.target as Node)
    ) {
      setShowOptions(false);
    }
  };

  const handleSelectAllClick = () => {
    if (!multiple || !options) return;

    let newValue = "";
    if (!checkIsSelectAllSelected()) {
      const allSelectedValues = options.map((option) => option.value);
      newValue = allSelectedValues.join(",");
    } else if (defaultValue) {
      newValue = defaultValue;
    }

    onChange?.(name, newValue);
  };

  const checkIsSelectAllSelected = () => {
    if (multiple && value) {
       const selectedValues = value.split(",");
     if (handleSelectAll) {
      const anySelectedValueInOptions = selectedValues.some(value => {
        return options?.some(option => option.value === value);
      });
    
      return anySelectedValueInOptions;
    }
     
      return options?.every((option) => selectedValues.includes(option.value));
    }
    return false;
  };

  const renderOptions = () => {
    if (!options?.length) return;

    const optionsToRender = [];

    // Render "Select All" option for multiple select
    if (multiple) {
      optionsToRender.push(
        <Option
          key="select-all"
          value="select-all"
          text="Select All"
          onClick={handleSelectAll ? handleSelectAll : handleSelectAllClick}
          selected={checkIsSelectAllSelected()}
          hasCheckbox={true}
        />
      );
    }

    optionsToRender.push(
      options.map((option: SelectOption) => (
        <Option
          key={option.value}
          value={option.value}
          text={
            option.value === defaultValue
              ? `${option.text} (default)`
              : option.text
          }
          onClick={handleOptionClick}
          selected={checkIsOptionSelected(option.value)}
          hasCheckbox={multiple}
          icon={option?.icon}
          iconPosition={"end"}
        />
      ))
    );

    return optionsToRender;
  };

  const handleButtonClick = () => {
    setShowOptions(!showOptions);
  };

  const handleOptionClick = (selectedValue: string) => {
    if (multiple) {
      const selectedValues = value ? value.split(",") : [];

      const index = selectedValues.indexOf(selectedValue);

      if (index !== -1) {
        selectedValues.splice(index, 1);
      } else {
        selectedValues.push(selectedValue);
      }

      const updatedValue = selectedValues.join(",");

      onChange?.(name, updatedValue);

      setShowOptions(externalShowOptions);
    } else {
      onChange?.(name, selectedValue);
    }

    if (!multiple) {
      setShowOptions(false);
    }
  };

  const getOptionsContainerStyle = (): CSSProperties => {
    if (selectRef.current && optionsContainerRef.current) {
      const selectRect = selectRef.current.getBoundingClientRect();
      optionsContainerRef.current.style.display = "block";
      const optionsContainerHeight = optionsContainerRef.current.offsetHeight;
      const windowHeight = window.innerHeight;

      // Calculate space above and below the selectRect
      const spaceAbove = selectRect.top;
      const spaceBelow = windowHeight - selectRect.bottom;

      const defaultStyles = {
        left: 0,
        width: selectRect.width,
      };
      // Set the position based on space availability
      if (spaceBelow >= optionsContainerHeight) {
        return { top: "100%", position: "absolute", ...defaultStyles };
      } else if (spaceAbove >= optionsContainerHeight) {
        return { bottom: "100%", position: "absolute", ...defaultStyles };
      }
    }

    // Default style if calculations fail
    return { display: "none" };
  };

  const renderValue = () => {
    if (value) {
      if (multiple) {
        const selectedValues = value.split(",");
        const selectedOptions = options?.filter((option) =>
          selectedValues.includes(option.value)
        );

        if (selectedOptions) {
          return selectedOptions.map((option) => option.text).join(", ");
        }
      } else {
        // Display the selected value for single selection
        const option = options?.find((option) => option.value === value);
        return option?.value === defaultValue
          ? `${option?.text} (default)`
          : option?.text;
      }
    }

    return defaultValue ?? placeholder;
  };

  const checkIsOptionSelected = (optionValue: string) => {
    if (multiple && value) {
      const selectedValues = value.split(",");
      return selectedValues.includes(optionValue);
    } else {
      return optionValue === value;
    }
  };

  useEffect(() => {
    document.addEventListener("mousedown", handleClickOutside);

    return () => {
      document.removeEventListener("mousedown", handleClickOutside);
    };
  }, []);

  useEffect(() => {
    if (
      !value &&
      defaultValue &&
      options?.find((option) => option.value === defaultValue) &&
      onChange
    ) {
      onChange(name, defaultValue);
    }
  }, []);

  return (
    <>
      {options && onChange && (
        <div className="select-root">
          {(label || onClear) && (
            <div className="select-labelWrapper">
              {label && <label htmlFor={id}>{label}</label>}
              {onClear && value && (
                <button
                  type="button"
                  onClick={onClear}
                  className="select-clearButton"
                  disabled={disabled}
                >
                  Clear
                </button>
              )}
            </div>
          )}
          <div className="select-buttonWrapper">
            <button
              type="button"
              className={`select-button ${
                !value ? "select-button--placeholder" : ""
              }`}
              onClick={disabled ? undefined : handleButtonClick}
              ref={selectRef}
              id={id}
              name={name}
              disabled={disabled}
            >
              {renderValue()}
              <svg focusable="false" aria-hidden="true" viewBox="0 0 24 24">
                <path d="m12 5.83 2.46 2.46c.39.39 1.02.39 1.41 0 .39-.39.39-1.02 0-1.41L12.7 3.7a.9959.9959 0 0 0-1.41 0L8.12 6.88c-.39.39-.39 1.02 0 1.41.39.39 1.02.39 1.41 0L12 5.83zm0 12.34-2.46-2.46a.9959.9959 0 0 0-1.41 0c-.39.39-.39 1.02 0 1.41l3.17 3.18c.39.39 1.02.39 1.41 0l3.17-3.17c.39-.39.39-1.02 0-1.41a.9959.9959 0 0 0-1.41 0L12 18.17z"></path>
              </svg>
            </button>

            {options && options.length > 0 && (
              <div
                className="select-optionsContainer"
                style={
                  showOptions ? getOptionsContainerStyle() : { display: "none" }
                }
                ref={optionsContainerRef}
              >
                <ul role="listbox" tabIndex={-1} className="select-optionsList">
                  {renderOptions()}
                </ul>
              </div>
            )}
          </div>
        </div>
      )}
    </>
  );
};

export default Select;
