# defualt
def tsp_crossover(parent_chromosome):

    pop_size = parent_chromosome.shape[0]  # 种群大小
    chrom_length = parent_chromosome.shape[1]  # 染色体长度（城市数量）
    offspring_chromosome = np.zeros_like(parent_chromosome)  # 存储子代

    # 1. 随机打乱种群，形成配对（避免自交）
    indices = np.random.permutation(pop_size)
    # 如果种群大小是奇数，最后一个个体不参与交叉（或采用其他策略）
    if pop_size % 2 != 0:
        indices = indices[:-1]

    # 2. 遍历每一对父代
    for i in range(0, len(indices), 2):  # 步长为2
        idx1, idx2 = indices[i], indices[i + 1]
        parent1 = parent_chromosome[idx1].copy()
        parent2 = parent_chromosome[idx2].copy()

        # 3. 为这对父代生成两个子代
        for child_index, (p1, p2) in enumerate(
            ((parent1, parent2), (parent2, parent1))
        ):
            # 随机选择交叉的起止点
            start, end = sorted(np.random.choice(chrom_length, 2, replace=False))

            # 子代初始化为无效值（例如-1），便于调试
            child = -np.ones(chrom_length, dtype=parent_chromosome.dtype)

            # 4. 从父代p1继承中间段[start, end]的城市
            child[start : end + 1] = p1[start : end + 1]

            # 5. 从父代p2填充剩余城市（保持顺序）
            # 5.1 确定填充起始位置和p2的遍历起始位置
            fill_pos = (end + 1) % chrom_length
            current_pos_in_p2 = (end + 1) % chrom_length

            # 5.2 遍历p2（从end+1开始，环回），将不位于继承段中的城市填入子代
            for _ in range(chrom_length - (end - start + 1)):
                # 找到下一个不在子代继承段中的城市
                while p2[current_pos_in_p2] in child:
                    current_pos_in_p2 = (current_pos_in_p2 + 1) % chrom_length
                # 将该城市填入子代
                child[fill_pos] = p2[current_pos_in_p2]
                fill_pos = (fill_pos + 1) % chrom_length

            # 6. 将生成的孩子放入子代种群
            offspring_index = idx1 if child_index == 0 else idx2
            offspring_chromosome[offspring_index] = child

    # 处理可能未参与交叉的个体（如种群大小为奇数），可以原样保留或进行变异
    if pop_size % 2 != 0:
        offspring_chromosome[indices[-1]] = parent_chromosome[indices[-1]]

    return offspring_chromosome


# defualt 2-swap
def tsp_mutation(offspring_chromosome):
    """Swap Mutation Operator - 修复版本"""
    pop_size = offspring_chromosome.shape[0]  # 种群大小
    chrom_length = offspring_chromosome.shape[1]  # 染色体长度（城市数量）

    for i in range(pop_size):
        # 随机选择两个不同的位置进行交换
        swap_idx = np.random.choice(chrom_length, 2, replace=False)
        city1, city2 = swap_idx

        # 交换城市
        offspring_chromosome[i, city1], offspring_chromosome[i, city2] = (
            offspring_chromosome[i, city2],
            offspring_chromosome[i, city1],
        )

    return offspring_chromosome


tsp_2opt_template = '''

import numpy as np
from typing import Tuple
import random


def tsp_2opt(offspring_chromosome):
    """2-opt Mutation Operator。
    约束：
    1. 输入：唯一参数offspring_chromosome为2D numpy数组，形状(pop_size, chrom_length)
    2. 输出：返回形状、数据类型完全相同的数组，严禁直接修改输入数组
    3. 必须保证输出为有效排列（无重复城市编号）
    4. 所有异常必须被捕获，异常时返回原始个体副本
    5. 仅使用 `numpy`，并给出标准导入语句，尽量避免低效的循环，善用NumPy的向量化操作。

    """
    pop_size = offspring_chromosome.shape[0]  # 种群大小
    chrom_length = offspring_chromosome.shape[1]  # 染色体长度（城市数量）

    for i in range(pop_size):
        # 随机选择两个不相邻的城市位置
        swap_idx = np.random.choice(chrom_length, 2, replace=False)
        city1, city2 = sorted(swap_idx)

        # 2-opt：交换路径上的两个部分
        offspring_chromosome[i, city1 : city2 + 1] = np.flip(
            offspring_chromosome[i, city1 : city2 + 1]
        )

    return offspring_chromosome
'''


# max
def tsp_2opt(offspring_chromosome):
    """2-opt Mutation Operator"""
    pop_size = offspring_chromosome.shape[0]  # 种群大小
    chrom_length = offspring_chromosome.shape[1]  # 染色体长度（城市数量）

    for i in range(pop_size):
        # 随机选择两个不相邻的城市位置
        swap_idx = np.random.choice(chrom_length, 2, replace=False)
        city1, city2 = sorted(swap_idx)

        # 2-opt：交换路径上的两个部分
        offspring_chromosome[i, city1 : city2 + 1] = np.flip(
            offspring_chromosome[i, city1 : city2 + 1]
        )

    return offspring_chromosome


tsp_3opt_template = '''

import numpy as np
from typing import Tuple
import random

def tsp_3opt(offspring_chromosome):
    """3-opt Mutation Operator。
    约束：
    1. 输入：唯一参数offspring_chromosome为2D numpy数组，形状(pop_size, chrom_length)
    2. 输出：返回形状、数据类型完全相同的数组，严禁直接修改输入数组
    3. 必须保证输出为有效排列（无重复城市编号）
    4. 所有异常必须被捕获，异常时返回原始个体副本
    5. 仅使用 `numpy`，并给出标准导入语句，尽量避免低效的循环，善用NumPy的向量化操作。

    """
    pop_size = offspring_chromosome.shape[0]  # 种群大小
    chrom_length = offspring_chromosome.shape[1]  # 染色体长度（城市数量）

    for i in range(pop_size):
        # 随机选择三个不相邻的城市位置
        swap_idx = np.random.choice(chrom_length, 3, replace=False)
        city1, city2, city3 = sorted(swap_idx)

        # 3-opt：根据不同的选择交换三个路径段
        # 这里展示一种常见的3-opt策略（实际应用中有多个策略）
        temp = np.copy(offspring_chromosome[i])
        offspring_chromosome[i, city1 : city3 + 1] = np.concatenate(
            (temp[city1 : city2 + 1], np.flip(temp[city2 + 1 : city3 + 1]))
        )

    return offspring_chromosome
'''


def tsp_3opt(offspring_chromosome):

    pop_size = offspring_chromosome.shape[0]  # 种群大小
    chrom_length = offspring_chromosome.shape[1]  # 染色体长度（城市数量）

    for i in range(pop_size):
        # 随机选择三个不相邻的城市位置
        swap_idx = np.random.choice(chrom_length, 3, replace=False)
        city1, city2, city3 = sorted(swap_idx)

        # 3-opt：根据不同的选择交换三个路径段
        # 这里展示一种常见的3-opt策略（实际应用中有多个策略）
        temp = np.copy(offspring_chromosome[i])
        offspring_chromosome[i, city1 : city3 + 1] = np.concatenate(
            (temp[city1 : city2 + 1], np.flip(temp[city2 + 1 : city3 + 1]))
        )

    return offspring_chromosome


tsp_oropt_template = '''

import numpy as np
from typing import Tuple
import random

def tsp_oropt(offspring_chromosome, dist_matrix=None, apply_if_improves=False):
    """tsp_oropt Operator。
    约束：
    1. 输入：唯一参数offspring_chromosome为2D numpy数组，形状(pop_size, chrom_length)
    2. 输出：返回形状、数据类型完全相同的数组，严禁直接修改输入数组
    3. 必须保证输出为有效排列（无重复城市编号）
    4. 所有异常必须被捕获，异常时返回原始个体副本
    5. 仅使用 `numpy`，并给出标准导入语句，尽量避免低效的循环，善用NumPy的向量化操作。

    """
    offspring_chromosome = offspring_chromosome.copy()  # 关键修正：避免修改原输入
    pop_size, chrom_length = offspring_chromosome.shape
    
    # 如果需要计算路径长度
    def tour_length(tour, dist_m):
        return sum(dist_m[tour[i], tour[(i+1) % len(tour)]] for i in range(len(tour)))
    
    for i in range(pop_size):
        tour = offspring_chromosome[i]
        start, end = sorted(np.random.choice(chrom_length, 2, replace=False))
        segment = tour[start:end+1]
        remaining = np.concatenate([tour[:start], tour[end+1:]])
        
        insert_pos = np.random.randint(len(remaining) + 1)
        new_tour = np.insert(remaining, insert_pos, segment)
        
        if apply_if_improves and dist_matrix is not None:
            # 经典Or-Opt：仅在接受移动能缩短路径时应用
            current_len = tour_length(tour, dist_matrix)
            new_len = tour_length(new_tour, dist_matrix)
            if new_len < current_len:
                offspring_chromosome[i] = new_tour
        else:
            # 随机变异：无条件应用移动
            offspring_chromosome[i] = new_tour
    
    return offspring_chromosome
'''


import numpy as np


def tsp_oropt(offspring_chromosome, dist_matrix=None, apply_if_improves=False):
    """
    Or-Opt style mutation operator.

    Args:
        offspring_chromosome: A 2D numpy array of shape (pop_size, chrom_length).
        dist_matrix: Distance matrix for TSP. Required if `apply_if_improves` is True.
        apply_if_improves: If True, only apply the move if it improves the tour length.
                           If False, apply the move randomly (simple mutation).

    Returns:
        Modified offspring_chromosome.
    """
    offspring_chromosome = offspring_chromosome.copy()  # 关键修正：避免修改原输入
    pop_size, chrom_length = offspring_chromosome.shape

    # 如果需要计算路径长度
    def tour_length(tour, dist_m):
        return sum(dist_m[tour[i], tour[(i + 1) % len(tour)]] for i in range(len(tour)))

    for i in range(pop_size):
        tour = offspring_chromosome[i]
        start, end = sorted(np.random.choice(chrom_length, 2, replace=False))
        segment = tour[start : end + 1]
        remaining = np.concatenate([tour[:start], tour[end + 1 :]])

        insert_pos = np.random.randint(len(remaining) + 1)
        new_tour = np.insert(remaining, insert_pos, segment)

        if apply_if_improves and dist_matrix is not None:
            # 经典Or-Opt：仅在接受移动能缩短路径时应用
            current_len = tour_length(tour, dist_matrix)
            new_len = tour_length(new_tour, dist_matrix)
            if new_len < current_len:
                offspring_chromosome[i] = new_tour
        else:
            # 随机变异：无条件应用移动
            offspring_chromosome[i] = new_tour

    return offspring_chromosome
