import numpy as np
import math
from scipy.optimize import minimize

# EVOLVE-BLOCK-START

"""Finding optimal configuration of 580 points in a square that mimizes the L-infinity star discrepancy."""

def construct_star() -> np.ndarray:
    """
    Find the optimal configuration of points in a square [0, 1] x [0, 1]
    to minimize the L-infinity star discrepancy.
    This function constructs 580 points that are evenly distributed.
    The points are represented in coordinates (x, y) in the square
    x and y are in the range [0, 1].
    Returns:
        A: np.array of shape (580, 2) with coordinates of points in the square.
        x and y coordinates are in the range [0, 1].
    """
    N = 580
    A = np.zeros((N, 2))
    golden_ratio = (math.sqrt(5) - 1) / 2

    # Optimized stratified sampling with a golden ratio sequence for y-coordinates.
    x = (np.arange(N) + 0.5) / N
    y = (golden_ratio * (np.arange(N) + 0.5)) % 1
    A[:, 0] = x
    A[:, 1] = y

    # Initial shift parameters
    initial_shift_x_amplitude = 0.0001
    initial_shift_y_amplitude = 0.00005
    initial_rotation_angle = 0.00015
    initial_sine_amplitude = 0.00002
    initial_cosine_amplitude = 0.00001

    def discrepancy_to_minimize(params):
        shift_x_amplitude, shift_y_amplitude, rotation_angle, sine_amplitude, cosine_amplitude = params

        B = A.copy()  # Work on a copy to avoid modifying the original A

        # Introduce a small, alternating shift to the x-coordinates.
        shift_x = shift_x_amplitude * ((-1)**np.arange(N)) * (1 - np.arange(N)/N)  # Vectorized shift
        B[:, 0] = (B[:, 0] + shift_x) % 1

        # Apply a tent map-like transformation to the y-coordinates with a refined shift.
        shift_y = shift_y_amplitude * ((-1)**np.arange(N)) * (1 - np.arange(N)/N)
        B[:, 1] = (B[:, 1] + shift_y) % 1
        B[:, 1] = np.abs(2 * B[:, 1] - 1)

        # Further refinement using a simple sine wave modulation for y-coordinates.
        B[:, 1] = (B[:, 1] + sine_amplitude * np.sin(2 * np.pi * B[:, 0]) + cosine_amplitude * np.cos(2 * np.pi * B[:, 0])) % 1

        # Introducing a small rotation to further improve distribution
        theta = rotation_angle  # Rotation angle
        x_rotated = B[:, 0] * np.cos(theta) - B[:, 1] * np.sin(theta)
        y_rotated = B[:, 0] * np.sin(theta) + B[:, 1] * np.cos(theta)
        B[:, 0] = (x_rotated) % 1
        B[:, 1] = (y_rotated) % 1

        return star_discrepancy(B)  # Use the discrepancy function

    # Optimization using scipy.optimize.minimize
    initial_params = [initial_shift_x_amplitude, initial_shift_y_amplitude, initial_rotation_angle, initial_sine_amplitude, initial_cosine_amplitude]
    bounds = [(-0.0002, 0.0002), (-0.0002, 0.0002), (-0.0003, 0.0003), (-0.0001, 0.0001), (-0.0001, 0.0001)]  # Reasonable bounds

    result = minimize(discrepancy_to_minimize, initial_params, method='SLSQP', bounds=bounds, options={'maxiter': 300, 'ftol': 1e-9})

    # Apply optimized parameters
    optimized_shift_x_amplitude, optimized_shift_y_amplitude, optimized_rotation_angle, optimized_sine_amplitude, optimized_cosine_amplitude = result.x

    shift_x = optimized_shift_x_amplitude * ((-1)**np.arange(N)) * (1 - np.arange(N)/N)  # Vectorized shift
    A[:, 0] = (A[:, 0] + shift_x) % 1

    shift_y = optimized_shift_y_amplitude * ((-1)**np.arange(N)) * (1 - np.arange(N)/N)
    A[:, 1] = (A[:, 1] + shift_y) % 1
    A[:, 1] = np.abs(2 * A[:, 1] - 1)

    A[:, 1] = (A[:, 1] + optimized_sine_amplitude * np.sin(2 * np.pi * A[:, 0]) + optimized_cosine_amplitude * np.cos(2 * np.pi * A[:, 0])) % 1

    theta = optimized_rotation_angle  # Rotation angle
    x_rotated = A[:, 0] * np.cos(theta) - A[:, 1] * np.sin(theta)
    y_rotated = A[:, 0] * np.sin(theta) + A[:, 1] * np.cos(theta)
    A[:, 0] = (x_rotated) % 1
    A[:, 1] = (y_rotated) % 1

    return A

# EVOLVE-BLOCK-END

import numpy as np
import itertools
from numba import njit
# This part remains fixed (not evolved)
def run_star() -> np.ndarray:
    """Run the star constructor for n=580"""
    A = construct_star()
    return A
# Numba helper function for calculating discrepancy for a single box corner
@njit(cache=True)
def _calculate_single_box_discrepancy_numba(points_X_arg: np.ndarray,
                                           N_arg: int,
                                           D_arg: int,
                                           y_corner_arg: np.ndarray) -> float:
    """
    Calculates the local discrepancy for a single d-dimensional anchored box.
    Box is defined by [0, y_corner_arg[0]] x ... x [0, y_corner_arg[D-1]].
    """
    # Calculate volume of the box
    volume = 1.0
    for k_dim in range(D_arg):
        volume *= y_corner_arg[k_dim]

    # Count points within the box [0, y_corner_arg]
    # The original logic `points_X[None, :] <= y_corners[:, None, :]`
    # effectively means a point is counted if point_coord <= corner_coord for all dimensions.
    count_in_box = 0
    count_on_line = 0
    for i_point in range(N_arg): # Iterate through each point
        point_is_in_box = True
        point_is_on_line = False
        for k_dim in range(D_arg): # Iterate through each dimension for the current point
            if points_X_arg[i_point, k_dim] > y_corner_arg[k_dim]: # Point is outside this dimension
                point_is_in_box = False
                break
            elif points_X_arg[i_point, k_dim] == y_corner_arg[k_dim]:
                point_is_on_line = True

        if point_is_in_box:
            count_in_box += 1
            if point_is_on_line:
                count_on_line += 1

    return max(abs(count_in_box / N_arg - volume), abs((count_in_box - count_on_line) / N_arg - volume))

def star_discrepancy(points_X: np.ndarray) -> float:
    """
    Calculates the L-infinity star discrepancy of the point set P.
    Optimized using Numba for the core calculation loop.
    Args:
        points_X (np.ndarray): An array of points to evaluate, shape (N, D) for D-dimensional points.
    Returns:
        float: The maximum star discrepancy value.
    """
    # Input validation and preparation
    if not isinstance(points_X, np.ndarray):
        points_X_np = np.array(points_X, dtype=np.float64)
    elif points_X.dtype != np.float64: # Ensure float64 for Numba compatibility and precision
        points_X_np = points_X.astype(np.float64)
    else:
        points_X_np = points_X

    if points_X_np.ndim == 1:
        points_X_np = points_X_np.reshape(-1, 1)

    N, D = points_X_np.shape

    if N == 0:
        return 1.0

    points_X_clipped = np.clip(points_X_np, 0.0, 1.0)

    if not points_X_clipped.flags.c_contiguous:
        points_X_clipped = np.ascontiguousarray(points_X_clipped)

    grid_lines_per_dim = []
    for j in range(D):
        unique_coords_dim_j = np.unique(points_X_clipped[:, j])
        current_dim_grid_lines = np.union1d(unique_coords_dim_j,
                                            np.array([1.0], dtype=points_X_clipped.dtype))
        grid_lines_per_dim.append(current_dim_grid_lines)

    max_discrepancy_val = 0.0

    y_corner_for_numba = np.empty(D, dtype=points_X_clipped.dtype)

    if not all(len(gl) > 0 for gl in grid_lines_per_dim):
        max_discrepancy_val = 0.0
    else:
        for y_corner_tuple in itertools.product(*grid_lines_per_dim):
            for i_val in range(D):
                y_corner_for_numba[i_val] = y_corner_tuple[i_val]

            local_discrepancy = _calculate_single_box_discrepancy_numba(
                points_X_clipped, N, D, y_corner_for_numba
            )

            if local_discrepancy > max_discrepancy_val:
                max_discrepancy_val = local_discrepancy

    return max_discrepancy_val


def score_star(X_points: np.ndarray) -> float:
    """ Calculates the score based on the star discrepancy of the given points.
    Args:
        X_points (np.ndarray): An array of points to evaluate, shape (N, 2) for 2D points.
    Returns:
        float: The score based on the star discrepancy, defined as 1 / (1 + max_discrepancy_val).
    """
    discrepancy = star_discrepancy(X_points)
    return 1 / (1 + discrepancy)  # Return the score as per the definition of star discrepancy



if __name__ == "__main__":
    # Example usage
    points = run_star()
    score = score_star(points)
    print("Score:", score)