import numpy as np
from typing import List
from ..mixture_gen import Component


def diagonal_cube_generator(
    n_components: int,
    n_features: int,
    distributions: List[str],
    cube_size: float = 0.5,
    spacing_factor: float = 1.0,
    vary_size: bool = False,
    vary_factor: float = 0.5,
    randomization: bool = True,
) -> List[Component]:
    """
    Generate components as a series of cubes arranged diagonally in feature space.

    Args:
        n_components: Number of components to generate
        n_features: Number of features
        distributions: List of distribution types to use
        cube_size: Base size of each cube (between 0.1 and 0.9)
        spacing_factor: Controls spacing between cubes (1.0 means corners touch)
                        Values > 1.0 create gaps, values < 1.0 create overlap
        vary_size: Whether to vary the size of individual cubes
        vary_factor: How much to vary cube sizes (if vary_size is True)

    Returns:
        List of Component objects
    """
    components = []

    # Ensure cube_size is reasonable
    cube_size = min(max(0.1, cube_size), 0.9)

    # Calculate the stride between cube centers
    stride = cube_size * spacing_factor

    # Calculate the offset to center the arrangement in feature space
    offset = -((n_components - 1) * stride) / 2

    # Generate components
    for i in range(n_components):
        # Calculate the center of this cube
        center = offset + i * stride

        # Vary the cube size if requested
        if vary_size:
            this_size = cube_size * (1.0 + vary_factor * (np.random.random() - 0.5))
        else:
            this_size = cube_size

        # Define rules as intervals around the center (same for all dimensions)
        rules = {}
        for j in range(n_features):
            half_size = this_size / 2
            rules[j] = (center - half_size, center + half_size)

        # Select distribution
        dist = (
            np.random.choice(distributions)
            if randomization
            else distributions[i % len(distributions)]
        )

        # Define distribution parameters based on the distribution type
        if dist == "normal":
            dist_params = {
                "loc": 0.0,
                "scale": np.random.uniform(0.1, 0.5) if randomization else 0.25,
            }
        elif dist == "uniform":
            dist_params = {
                "low": np.random.uniform(-1, -0.2) if randomization else -0.5,
                "high": np.random.uniform(0.2, 1) if randomization else 0.5,
            }
        elif dist == "gamma":
            dist_params = {
                "shape": 2.0,
                "scale": np.random.uniform(0.1, 0.3) if randomization else 0.2,
            }
        elif dist == "exponential":
            dist_params = {
                "scale": np.random.uniform(0.1, 0.4) if randomization else 0.3
            }

        # Equal weights for all components
        weight = 1.0 / n_components

        components.append(
            Component(
                rules=rules, distribution=dist, dist_params=dist_params, weight=weight
            )
        )

    return components


def diagonal_component_generator(
    n_components: int,
    n_features: int,
    distributions: List[str],
    base_size: float = 0.5,
    spacing_factor: float = 1.0,
    shape_type: str = "cube",  # "cube", "rectangle", or "random"
    adaptive_sizing: bool = True,  # Automatically adjust size based on n_components
    vary_size: bool = False,
    vary_factor: float = 0.2,
) -> List[Component]:
    """
    Generate components arranged diagonally in feature space.

    Args:
        n_components: Number of components to generate
        n_features: Number of features
        distributions: List of distribution types to use
        base_size: Base size of each component (between 0.1 and 0.9)
        spacing_factor: Controls spacing between components (1.0 means corners touch)
        shape_type: Type of shape to generate:
                   - "cube": Equal size in all dimensions
                   - "rectangle": Varying sizes across dimensions
                   - "random": Random sizes in each dimension
        adaptive_sizing: Automatically adjust size based on number of components
        vary_size: Whether to vary the size of individual components
        vary_factor: How much to vary component sizes (if vary_size is True)

    Returns:
        List of Component objects
    """
    components = []

    # Adapt base size if needed
    if adaptive_sizing:
        # Smaller components when we have more of them
        # This helps keep them within the feature space bounds
        adapted_size = base_size * (1.0 - 0.05 * (n_components - 2))
        base_size = max(0.1, min(0.9, adapted_size))
    else:
        base_size = min(max(0.1, base_size), 0.9)

    # Calculate the stride between component centers
    stride = base_size * spacing_factor

    # Calculate the offset to center the arrangement in feature space
    offset = -((n_components - 1) * stride) / 2

    # Generate components
    for i in range(n_components):
        # Calculate the center of this component
        center = offset + i * stride

        # Generate sizes for each dimension
        if shape_type == "cube":
            # Same size in all dimensions
            if vary_size:
                sizes = [
                    base_size * (1.0 + vary_factor * (np.random.random() - 0.5))
                ] * n_features
            else:
                sizes = [base_size] * n_features

        elif shape_type == "rectangle":
            # Primary dimension (diagonal) uses base_size, others are smaller
            sizes = []
            for j in range(n_features):
                if j == 0:  # Primary dimension
                    if vary_size:
                        sizes.append(
                            base_size * (1.0 + vary_factor * (np.random.random() - 0.5))
                        )
                    else:
                        sizes.append(base_size)
                else:  # Secondary dimensions
                    factor = 0.5 + 0.3 * np.random.random()  # 50-80% of base size
                    sizes.append(base_size * factor)

        elif shape_type == "random":
            # Random size in each dimension
            sizes = [
                base_size * (0.5 + 0.7 * np.random.random()) for _ in range(n_features)
            ]

        # Define rules as intervals for each dimension
        rules = {}
        for j in range(n_features):
            half_size = sizes[j] / 2
            # All dimensions use the same center position for diagonal arrangement
            rules[j] = (center - half_size, center + half_size)

        # Select distribution
        dist = distributions[i % len(distributions)]

        # Define distribution parameters based on the distribution type
        if dist == "normal":
            dist_params = {"loc": 0.0, "scale": 0.1}
        elif dist == "uniform":
            dist_params = {"low": -0.5, "high": 0.5}
        elif dist == "gamma":
            dist_params = {"shape": 2.0, "scale": 0.2}
        elif dist == "exponential":
            dist_params = {"scale": 0.3}

        # Equal weights for all components
        weight = 1.0 / n_components

        components.append(
            Component(
                rules=rules, distribution=dist, dist_params=dist_params, weight=weight
            )
        )

    return components
