import numpy as np

def place_points(side, space, k, base_point, min_distance):
    if k <= 0 or space < min_distance:
        return set()

    power = 1.5
    normalized_positions = np.linspace(0, 1, int(k) + 1, endpoint=True)[1:] ** power
    ideal_offsets = normalized_positions * space

    side_points = set()
    last_added_point = base_point

    for offset in ideal_offsets:
        if side == 'left':
            candidate = int(round(base_point - offset))
        else:
            candidate = int(round(base_point + offset))

        if abs(candidate - last_added_point) >= min_distance:
            side_points.add(candidate)
            last_added_point = candidate

    return side_points

def generate_dynamic_samples(next_t, interval, n_total_samples, min_distance):
    """
    Generates samples with a non-linear distribution that is
    dense around next_t, covers the entire interval, and filters points
    too close to the boundaries.
    """
    start, end = interval
    final_indices = {next_t}
    n_to_add = n_total_samples - 1

    if n_to_add <= 0:
        return final_indices

    # 1. Proportionally allocate the number of points to the left and right of next_t
    space_left = next_t - start
    space_right = end - next_t
    total_space = space_left + space_right

    if total_space <= 0:
        return final_indices

    n_left_quota = round(n_to_add * (space_left / total_space))
    n_right_quota = n_to_add - n_left_quota

    # --- Generate points for both sides ---
    left_points = place_points('left', space_left, n_left_quota, next_t, min_distance)
    right_points = place_points('right', space_right, n_right_quota, next_t, min_distance)

    final_indices.update(left_points)
    final_indices.update(right_points)

    # --- Final Filtering Step: Abandon points too close to boundaries ---

    filtered_indices = {
        p for p in final_indices
        if (p - start) >= min_distance and (end - p) >= min_distance
    }

    # Ensure next_t is always included, even if it's close to a boundary.
    filtered_indices.add(next_t)

    return filtered_indices


def _generate_samples(center_index, max_offset, num_to_generate, is_left, min_frame_length, growth_factor):
    """
    Helper function to generate samples for one side with an exponentially growing stride.
    """
    # Don't generate if there's no space or no samples requested
    if max_offset <= 0 or num_to_generate <= 0:
        return []

    samples = []
    # Initial stride is 5% of the available space, but not less than min_frame_length
    initial_stride = max(min_frame_length, 0.07 * max_offset)

    offset = 0.0
    current_gap = float(initial_stride)

    for _ in range(num_to_generate):
        offset += current_gap
        # Stop if we are about to sample past the max boundary
        if offset > max_offset:
            break

        pos = center_index - int(round(offset)) if is_left else center_index + int(round(offset))
        samples.append(pos)

        # The gap for the next sample grows, but never shrinks below min_frame_length
        current_gap = max(current_gap * growth_factor, min_frame_length)

    return samples


def sample_logic(sims, budget, min_frame_length=4, HIGH_SIM_THRESHOLD=0.6, GROWTH_FACTOR=1.5):
    """
    Samples frames by sequentially processing each side and rolling over any unused budget.
    """

    # --- Initial Setup ---
    if budget % 2 == 1:
        budget -= 1
    if budget <= 0:
        return []

    total_frames = len(sims)
    original_indices = np.where(~np.isnan(sims))[0]
    if not original_indices.any():
        return []

    max_support_frames = budget // 2
    all_peaks = sorted([(sims[i], i) for i in original_indices], key=lambda x: x[0], reverse=True)
    sorted_indices = sorted(original_indices) # FIX: Use sorted indices for neighbor search

    # --- Filter for Qualified Peaks ---
    qualified_peaks = [p for p in all_peaks if p[0] >= HIGH_SIM_THRESHOLD]
    if len(qualified_peaks) <= max_support_frames // 3:
        candidate_frame_indices = [p[1] for p in qualified_peaks] + [p[1] for p in all_peaks[len(qualified_peaks):max_support_frames+len(qualified_peaks)]]
    elif not qualified_peaks:
        candidate_frame_indices = [p[1] for p in all_peaks[:max_support_frames]]
    else:
        candidate_frame_indices = [p[1] for p in qualified_peaks[:max_support_frames]]
    if not candidate_frame_indices:
        return []
    new_sampled_frames = []
    for i, index in enumerate(candidate_frame_indices):
        # Find left and right neighbors
        current_pos = np.searchsorted(sorted_indices, index)
        left_frame = sorted_indices[current_pos - 1] if current_pos > 0 else 0
        right_frame = sorted_indices[current_pos + 1] if current_pos < len(sorted_indices) - 1 else total_frames

        # Calculate budget for this peak
        quota = (max_support_frames // len(candidate_frame_indices) + int(i < max_support_frames % len(candidate_frame_indices))) * 2
        num_samples_per_side = quota // 2

        # Define the maximum sampling offset from the center 'index'
        left_max = max(index - left_frame - min_frame_length // 2, 0) // 2
        right_max = max(right_frame - index - min_frame_length // 2, 0) // 2

        # --- Step 1: Sample the left side first ---
        left_samples = _generate_samples(index, left_max, num_samples_per_side, True, min_frame_length, GROWTH_FACTOR)
        new_sampled_frames.extend(left_samples)

        # --- Step 2: Use the extra left quota on the other side ---
        left_shortfall = num_samples_per_side - len(left_samples)
        right_target_count = num_samples_per_side + left_shortfall

        # --- Step 3: Sample the right side with the adjusted quota ---
        if right_target_count > 0:
            right_samples = _generate_samples(index, right_max, right_target_count, False, min_frame_length, GROWTH_FACTOR)
            new_sampled_frames.extend(right_samples)

    # Remove duplicates and sort the final list
    return list(set(new_sampled_frames))