from typing import List, Tuple
import numpy as np
import random

from MOCO.problems import BiObjectiveTSP, MultiObjectiveKnapsack
from MOCO.evaluation import MOCOEvaluator

class WSLKH:
    """Weighted Sum Lin-Kernighan-Helsgaun for Multi-Objective TSP"""
    def __init__(self, problem: 'BiObjectiveTSP', num_weights: int = 10, reference_point=None):
        self.problem = problem
        self.num_weights = num_weights
        self.best_solutions = []
        self.reference_point = None
    
    def _get_weighted_distances(self, weight: float) -> np.ndarray:
        """Combine distance matrices using weight"""
        return weight * self.problem.distances1 + (1 - weight) * self.problem.distances2
    
    def _improve_tour(self, tour: List[int], distances: np.ndarray) -> List[int]:
        """2-opt improvement procedure"""
        improved = True
        while improved:
            improved = False
            best_distance = self._calculate_tour_length(tour, distances)
            
            for i in range(len(tour) - 2):
                for j in range(i + 2, len(tour)):
                    # Try 2-opt swap
                    new_tour = tour[:i+1] + tour[i+1:j+1][::-1] + tour[j+1:]
                    new_distance = self._calculate_tour_length(new_tour, distances)
                    
                    if new_distance < best_distance:
                        tour = new_tour
                        best_distance = new_distance
                        improved = True
                        break
                if improved:
                    break
        
        return tour
    
    def _calculate_tour_length(self, tour: List[int], distances: np.ndarray) -> float:
        """Calculate total tour length"""
        return sum(distances[tour[i]][tour[(i + 1) % len(tour)]] 
                  for i in range(len(tour)))
    
    def run(self) -> List[Tuple[List[int], Tuple[float, float]]]:
        """Run WS-LKH with different weights"""
        self.best_solutions = []
        weights = np.linspace(0, 1, self.num_weights)
        
        for weight in weights:
            # Get weighted distance matrix
            distances = self._get_weighted_distances(weight)
            
            # Generate initial solution
            current_tour = list(range(self.problem.n_cities))
            random.shuffle(current_tour)
            
            # Improve tour using 2-opt
            improved_tour = self._improve_tour(current_tour, distances)
            
            # Evaluate tour with original objectives
            objectives = self.problem.evaluate(improved_tour)
            
            # Add to solutions if non-dominated
            self._add_if_nondominated(improved_tour, objectives)
        
        return [(sol, obj) for sol, obj in self.best_solutions]
    
    def _add_if_nondominated(self, solution: List[int], 
                            objectives: Tuple[float, float]):
        """Add solution to best_solutions if it's non-dominated"""
        # First check if this exact solution already exists
        for existing_sol, existing_obj in self.best_solutions:
            if np.array_equal(objectives, existing_obj):
                return
        
        dominated = False
        # Remove solutions that this new solution dominates
        self.best_solutions = [
            (sol, obj) for sol, obj in self.best_solutions
            if not self._dominates(objectives, obj)
        ]
        
        # Check if new solution is dominated
        for _, obj in self.best_solutions:
            if self._dominates(obj, objectives):
                dominated = True
                break
        
        if not dominated:
            self.best_solutions.append((solution, objectives))
    
    def _dominates(self, obj1: Tuple[float, float], 
                  obj2: Tuple[float, float]) -> bool:
        """Check if obj1 dominates obj2"""
        return (obj1[0] <= obj2[0] and obj1[1] <= obj2[1] and 
                (obj1[0] < obj2[0] or obj1[1] < obj2[1]))
 
# This gave good results! 
class WSDP:
    """Consistent WSDP with adaptive scaling for different instance sizes"""
    
    def __init__(self, problem: 'MultiObjectiveKnapsack', num_weights: int = 100, reference_point=None):
        self.problem = problem
        self.num_weights = num_weights
        self.best_solutions = []
        self.reference_point = reference_point
        
    def _solve_weighted_knapsack(self, weight: float) -> List[int]:
        """Solve weighted knapsack with instance-appropriate method"""
        n = self.problem.n_items
        
        # Calculate weighted values
        weighted_values = [
            weight * self.problem.values[0][i] + (1 - weight) * self.problem.values[1][i]
            for i in range(n)
        ]
        
        # Use DP for all instances with appropriate scaling
        # The key is to use a scale that keeps W reasonable for all problem sizes
        
        # Since BiKP50 has capacity=12.5 and BiKP100/200 have capacity=25,
        # we need a scale that works for both
        scale = 30  # This gives W=375 for cap=12.5 and W=750 for cap=25
        
        # Calculate W
        W = int(self.problem.capacity * scale)
        
        # Always use DP (no switching to greedy)
        return self._solve_dp(weighted_values, scale)
    
    def _solve_dp(self, weighted_values: List[float], scale: int) -> List[int]:
        """DP with given scale factor"""
        n = self.problem.n_items
        W = int(self.problem.capacity * scale)
        
        # Use memory-efficient single array DP
        dp = np.zeros(W + 1, dtype=np.float32)
        parent = np.full((n, W + 1), -1, dtype=np.int16)
        
        # Fill DP table
        for i in range(n):
            w_i = max(1, int(round(self.problem.weights[i] * scale)))
            if w_i > W:
                continue
            
            v_i = weighted_values[i]
            
            # Backward to avoid overwriting
            for w in range(W, w_i - 1, -1):
                if dp[w - w_i] + v_i > dp[w]:
                    dp[w] = dp[w - w_i] + v_i
                    parent[i][w] = w - w_i
        
        # Reconstruct
        solution = [0] * n
        w = W
        
        for i in range(n - 1, -1, -1):
            if w > 0 and parent[i][w] >= 0:
                solution[i] = 1
                w = parent[i][w]
        
        return solution
    
    def _enhanced_greedy(self, weighted_values: List[float]) -> List[int]:
        """Enhanced greedy for large instances"""
        n = self.problem.n_items
        
        # Phase 1: Standard greedy by efficiency
        items = []
        for i in range(n):
            if self.problem.weights[i] > 0 and weighted_values[i] > 0:
                eff = weighted_values[i] / self.problem.weights[i]
                items.append((eff, i))
        
        items.sort(reverse=True)
        
        solution = [0] * n
        capacity_used = 0
        
        for _, i in items:
            if capacity_used + self.problem.weights[i] <= self.problem.capacity:
                solution[i] = 1
                capacity_used += self.problem.weights[i]
        
        # Phase 2: Try to improve with swaps (limited iterations for speed)
        improved = True
        iterations = 0
        max_iterations = 10  # Limit for large instances
        
        while improved and iterations < max_iterations:
            improved = False
            iterations += 1
            
            # Find best swap
            best_swap = None
            best_improvement = 0
            
            for i in range(n):
                if solution[i] == 1:
                    for j in range(n):
                        if solution[j] == 0:
                            new_capacity = capacity_used - self.problem.weights[i] + self.problem.weights[j]
                            if new_capacity <= self.problem.capacity:
                                improvement = weighted_values[j] - weighted_values[i]
                                if improvement > best_improvement:
                                    best_improvement = improvement
                                    best_swap = (i, j)
            
            if best_swap:
                i, j = best_swap
                solution[i] = 0
                solution[j] = 1
                capacity_used = capacity_used - self.problem.weights[i] + self.problem.weights[j]
                improved = True
        
        return solution
    
    def run(self) -> List[Tuple[List[int], Tuple[float, ...]]]:
        """Run WSDP with adaptive strategy"""
        self.best_solutions = []
        n = self.problem.n_items
        
        # Adaptive weight generation based on instance size
        if n <= 50:
            num_weights = 120  # More weights for small instances
        elif n <= 100:
            num_weights = 100  # Standard
        else:
            num_weights = 80   # Fewer weights for large instances
        
        # Generate weights
        weights = list(np.linspace(0, 1, num_weights))
        
        # Always include extreme weights
        extreme_weights = [
            0.0, 0.0001, 0.001, 0.005, 0.01, 0.02, 0.05, 0.1,
            0.9, 0.95, 0.98, 0.99, 0.995, 0.999, 0.9999, 1.0
        ]
        weights.extend(extreme_weights)
        
        # Add some random weights for diversity
        np.random.seed(42)
        weights.extend(np.random.uniform(0, 1, 20))
        
        # Remove duplicates and sort
        weights = sorted(list(set(weights)))
        
        # Track unique solutions
        seen_objectives = set()
        
        # First, get extreme solutions
        for obj_idx in range(2):
            items = []
            for i in range(n):
                if self.problem.weights[i] > 0:
                    eff = self.problem.values[obj_idx][i] / self.problem.weights[i]
                    items.append((eff, i))
            
            items.sort(reverse=True)
            
            solution = [0] * n
            capacity_used = 0
            
            for _, i in items:
                if capacity_used + self.problem.weights[i] <= self.problem.capacity:
                    solution[i] = 1
                    capacity_used += self.problem.weights[i]
            
            if sum(solution) > 0:
                objectives = self.problem.evaluate(solution)
                rounded_obj = tuple(round(v, 2) for v in objectives)
                if rounded_obj not in seen_objectives:
                    seen_objectives.add(rounded_obj)
                    self._add_if_nondominated(solution, objectives)
        
        # Solve for all weights
        for weight in weights:
            solution = self._solve_weighted_knapsack(weight)
            
            if sum(solution) == 0:
                continue
            
            objectives = self.problem.evaluate(solution)
            rounded_obj = tuple(round(v, 2) for v in objectives)
            
            if rounded_obj not in seen_objectives:
                seen_objectives.add(rounded_obj)
                self._add_if_nondominated(solution, objectives)
        
        return [(sol, obj) for sol, obj in self.best_solutions]
    
    def _add_if_nondominated(self, solution: List[int], objectives: Tuple[float, ...]):
        """Add solution if non-dominated"""
        # Remove dominated solutions
        self.best_solutions = [
            (sol, obj) for sol, obj in self.best_solutions
            if not self._dominates(objectives, obj)
        ]
        
        # Check if new solution is dominated
        for _, obj in self.best_solutions:
            if self._dominates(obj, objectives):
                return
        
        self.best_solutions.append((solution, objectives))
    
    def _dominates(self, obj1: Tuple[float, ...], obj2: Tuple[float, ...]) -> bool:
        """Check if obj1 dominates obj2 (maximization)"""
        at_least_one_better = False
        
        for v1, v2 in zip(obj1, obj2):
            if v1 < v2:
                return False
            if v1 > v2:
                at_least_one_better = True
        
        return at_least_one_better


class WSLKH_TriObjective:
    """Weighted Sum Lin-Kernighan-Helsgaun for Tri-Objective TSP - Simplified version"""
    
    def __init__(self, problem: 'TriObjectiveTSP', num_weights: int = 21, reference_point=None):
        """
        Args:
            problem: TriObjectiveTSP problem instance
            num_weights: Total number of weight combinations to generate
        """
        self.problem = problem
        self.num_weights = num_weights
        self.best_solutions = []
        self.reference_point = reference_point
        
        # Check if problem has 3 objectives
        if not hasattr(problem, 'distances3'):
            raise ValueError("Problem must be a TriObjectiveTSP with 3 distance matrices")
    
    def _get_weighted_distances(self, weights: Tuple[float, float, float]) -> np.ndarray:
        """Combine distance matrices using weight vector"""
        return (weights[0] * self.problem.distances1 + 
                weights[1] * self.problem.distances2 + 
                weights[2] * self.problem.distances3)
    
    def _generate_weight_vectors(self) -> List[Tuple[float, float, float]]:
        """Generate weight vectors for 3 objectives - simplified version"""
        weights = []
        
        # Method 1: Simple uniform distribution on simplex
        # This is similar to np.linspace but for 3D
        if self.num_weights <= 10:
            # For small num_weights, use specific strategic weights
            weights = [
                (1.0, 0.0, 0.0),    # Only obj1
                (0.0, 1.0, 0.0),    # Only obj2
                (0.0, 0.0, 1.0),    # Only obj3
                (0.5, 0.5, 0.0),    # Mix obj1 & obj2
                (0.5, 0.0, 0.5),    # Mix obj1 & obj3
                (0.0, 0.5, 0.5),    # Mix obj2 & obj3
                (1/3, 1/3, 1/3),    # Equal weights
                (0.7, 0.2, 0.1),    # Favor obj1
                (0.2, 0.7, 0.1),    # Favor obj2
                (0.1, 0.2, 0.7),    # Favor obj3
            ][:self.num_weights]
        else:
            # For larger num_weights, generate systematically
            # This is the 3D equivalent of linspace
            n_per_dim = int(np.sqrt(self.num_weights)) + 1
            
            for i in range(n_per_dim):
                for j in range(n_per_dim):
                    w1 = i / (n_per_dim - 1) if n_per_dim > 1 else 0.5
                    w2 = j / (n_per_dim - 1) if n_per_dim > 1 else 0.5
                    
                    # Ensure weights sum to 1
                    if w1 + w2 <= 1.0:
                        w3 = 1.0 - w1 - w2
                        weights.append((w1, w2, w3))
            
            # Add extreme points to ensure coverage
            weights.extend([
                (1.0, 0.0, 0.0),
                (0.0, 1.0, 0.0),
                (0.0, 0.0, 1.0),
            ])
            
            # Remove duplicates and limit to num_weights
            unique_weights = []
            seen = set()
            for w in weights:
                rounded = tuple(round(x, 4) for x in w)
                if rounded not in seen:
                    seen.add(rounded)
                    unique_weights.append(w)
            
            # Take exactly num_weights
            if len(unique_weights) > self.num_weights:
                # Sample to get exactly num_weights, keeping extremes
                essential = [(1.0, 0.0, 0.0), (0.0, 1.0, 0.0), (0.0, 0.0, 1.0)]
                others = [w for w in unique_weights if w not in essential]
                np.random.shuffle(others)
                weights = essential + others[:self.num_weights - 3]
            else:
                weights = unique_weights
        
        return weights
    
    def _improve_tour(self, tour: List[int], distances: np.ndarray) -> List[int]:
        """2-opt improvement procedure - same as original"""
        improved = True
        while improved:
            improved = False
            best_distance = self._calculate_tour_length(tour, distances)
            
            for i in range(len(tour) - 2):
                for j in range(i + 2, len(tour)):
                    # Try 2-opt swap
                    new_tour = tour[:i+1] + tour[i+1:j+1][::-1] + tour[j+1:]
                    new_distance = self._calculate_tour_length(new_tour, distances)
                    
                    if new_distance < best_distance:
                        tour = new_tour
                        best_distance = new_distance
                        improved = True
                        break
                if improved:
                    break
        
        return tour
    
    def _calculate_tour_length(self, tour: List[int], distances: np.ndarray) -> float:
        """Calculate total tour length - same as original"""
        return sum(distances[tour[i]][tour[(i + 1) % len(tour)]] 
                  for i in range(len(tour)))
    
    def run(self) -> List[Tuple[List[int], Tuple[float, float, float]]]:
        """Run WS-LKH with different weight vectors"""
        self.best_solutions = []
        weights = self._generate_weight_vectors()
        
        for weight_vector in weights:
            # Get weighted distance matrix
            distances = self._get_weighted_distances(weight_vector)
            
            # Generate initial solution - same as original
            current_tour = list(range(self.problem.n_cities))
            random.shuffle(current_tour)
            
            # Improve tour using 2-opt
            improved_tour = self._improve_tour(current_tour, distances)
            
            # Evaluate tour with original objectives
            objectives = self.problem.evaluate(improved_tour)
            
            # Add to solutions if non-dominated
            self._add_if_nondominated(improved_tour, objectives)
        
        return [(sol, obj) for sol, obj in self.best_solutions]
    
    def _add_if_nondominated(self, solution: List[int], 
                            objectives: Tuple[float, float, float]):
        """Add solution to best_solutions if it's non-dominated"""
        # First check if this exact solution already exists
        for existing_sol, existing_obj in self.best_solutions:
            if np.array_equal(objectives, existing_obj):
                return
        
        dominated = False
        # Remove solutions that this new solution dominates
        self.best_solutions = [
            (sol, obj) for sol, obj in self.best_solutions
            if not self._dominates(objectives, obj)
        ]
        
        # Check if new solution is dominated
        for _, obj in self.best_solutions:
            if self._dominates(obj, objectives):
                dominated = True
                break
        
        if not dominated:
            self.best_solutions.append((solution, objectives))
    
    def _dominates(self, obj1: Tuple[float, float, float], 
                  obj2: Tuple[float, float, float]) -> bool:
        """Check if obj1 dominates obj2 (minimization)"""
        return (obj1[0] <= obj2[0] and obj1[1] <= obj2[1] and obj1[2] <= obj2[2] and 
                (obj1[0] < obj2[0] or obj1[1] < obj2[1] or obj1[2] < obj2[2]))


def test_weighted_sum_methods(num_runs=3, num_weights=20):
    """Test both weighted sum methods with multiple runs on different instances"""
    # Set up initial instances for single run demonstration
    print("\nInitial Single Run Results:")
    print("===========================")
    
    print("\nTesting Weighted Sum LKH on Bi-Objective TSP:")
    # bitsp = BiObjectiveTSP(n_cities=20)
    # wslkh = WSLKH(bitsp, num_weights=num_weights)
    
    # Show results from a single run first
    # solutions_lkh = wslkh.run()
    # print(f"Found {len(solutions_lkh)} non-dominated solutions")
    # for _, objectives in solutions_lkh:
    #     print(f"Objectives: ({objectives[0]:.0f}, {objectives[1]:.0f})")
    
    # Test WS-DP on Knapsack
    print("\nTesting Weighted Sum DP on Multi-Objective Knapsack:")
    mokp = MultiObjectiveKnapsack(n_items=50, n_objectives=2, capacity=12.5)
    wsdp = WSDP(mokp, num_weights=num_weights)
    solutions_dp = wsdp.run()

    print(f"Found {len(solutions_dp)} non-dominated solutions")
    for _, objectives in solutions_dp:
        print(f"Objectives: ({objectives[0]:.2f}, {objectives[1]:.2f})")

    print("\nMultiple Runs Evaluation:")
    print("========================")
    
    # Set up parameters
    tsp_problem_params = {'n_cities': 20}
    knapsack_problem_params = {
        'n_items': 50,#50,
        'n_objectives': 2,
        'capacity': 12.5#12.5
    }
    
    wslkh_params = {'num_weights': num_weights}
    wsdp_params = {'num_weights': num_weights}

    # Initialize evaluator with normalized reference points
    evaluator = MOCOEvaluator(reference_point=(1.2, 1.2))

    print(f"\nRunning each algorithm {num_runs} times on different instances...")
    
    # Evaluate TSP with multiple instances
    # tsp_result = evaluator.evaluate_algorithm(
    #     algorithm_class=WSLKH,
    #     problem_class=BiObjectiveTSP,
    #     algorithm_name="WS-LKH",
    #     parameters=wslkh_params,
    #     problem_params=tsp_problem_params,
    #     num_runs=num_runs
    # )
    
    # Evaluate Knapsack with multiple instances
    knapsack_result = evaluator.evaluate_algorithm(
        algorithm_class=WSDP,
        problem_class=MultiObjectiveKnapsack,
        algorithm_name="WS-DP",
        parameters=wsdp_params,
        problem_params=knapsack_problem_params,
        num_runs=num_runs
    )
    
    # Generate report
    evaluator.generate_report()
    
    print("\nAggregated Results:")
    for result in evaluator.results:
        print(f"\n{result.algorithm_name} on {result.problem_name}:")
        print(f"Average Runtime: {result.runtime:.2f} seconds")
        print(f"Final Hypervolume: {result.hypervolume:.4f}")
        print(f"Final Non-dominated solutions: {result.num_nondominated}")
        print(f"Algorithm parameters: {result.parameters['algorithm']}")
        print(f"Problem parameters: {result.parameters['problem']}")

def diagnose_negative_hv(n_items=50, capacity=12.5, num_instances=5):
    """Diagnose why hypervolume can be negative"""
    import numpy as np
    
    print(f"\n=== Diagnosing Negative HV for BiKP{n_items} ===")
    print(f"Running {num_instances} instances...\n")
    
    negative_hv_count = 0
    
    for instance in range(num_instances):
        print(f"\n--- Instance {instance + 1} ---")
        
        # Create problem
        mokp = MultiObjectiveKnapsack(n_items=n_items, n_objectives=2, capacity=capacity)
        
        # Run WSDP
        wsdp = WSDP(mokp, num_weights=40)
        solutions = wsdp.run()
        
        print(f"Found {len(solutions)} non-dominated solutions")
        
        if not solutions:
            print("WARNING: No solutions found!")
            negative_hv_count += 1
            continue
        
        # Check all solutions for feasibility
        infeasible_count = 0
        for sol, obj in solutions:
            weight = sum(mokp.weights[i] * sol[i] for i in range(n_items))
            if weight > capacity * 1.001:  # Small tolerance
                infeasible_count += 1
                print(f"  INFEASIBLE: weight {weight:.3f} > capacity {capacity}")
        
        if infeasible_count > 0:
            print(f"WARNING: {infeasible_count} infeasible solutions!")
        
        # Analyze objectives
        objectives = [obj for _, obj in solutions]
        obj1_values = [obj[0] for obj in objectives]
        obj2_values = [obj[1] for obj in objectives]
        
        min_obj1, max_obj1 = min(obj1_values), max(obj1_values)
        min_obj2, max_obj2 = min(obj2_values), max(obj2_values)
        
        print(f"Objective ranges:")
        print(f"  Obj1: [{min_obj1:.2f}, {max_obj1:.2f}]")
        print(f"  Obj2: [{min_obj2:.2f}, {max_obj2:.2f}]")
        
        # Check for invalid objectives
        if any(obj[0] < 0 or obj[1] < 0 for obj in objectives):
            print("WARNING: Negative objective values found!")
            negative_hv_count += 1
        
        # Simulate hypervolume calculation with typical reference points
        ref_points = [(0, 0), (5, 5), (10, 10)]
        
        for ref in ref_points:
            print(f"\nWith reference point {ref}:")
            
            # Check if any solution dominates reference
            dominates_ref = 0
            for obj in objectives:
                if obj[0] > ref[0] and obj[1] > ref[1]:
                    dominates_ref += 1
            
            print(f"  Solutions dominating ref: {dominates_ref}/{len(objectives)}")
            
            if dominates_ref == 0:
                print(f"  WARNING: No solution dominates reference point!")
                print(f"  This would cause HV = 0 or negative!")
                negative_hv_count += 1
        
        # Find extreme solutions
        print(f"\nExtreme solutions:")
        
        # Best for objective 1
        best_obj1_idx = max(range(len(objectives)), key=lambda i: objectives[i][0])
        print(f"  Best Obj1: {objectives[best_obj1_idx]}")
        
        # Best for objective 2  
        best_obj2_idx = max(range(len(objectives)), key=lambda i: objectives[i][1])
        print(f"  Best Obj2: {objectives[best_obj2_idx]}")
        
        # Check if solutions are too similar
        if len(objectives) > 1:
            # Calculate spread
            obj1_spread = max_obj1 - min_obj1
            obj2_spread = max_obj2 - min_obj2
            
            if obj1_spread < 1.0 or obj2_spread < 1.0:
                print(f"\nWARNING: Very small spread in objectives!")
                print(f"  Obj1 spread: {obj1_spread:.3f}")
                print(f"  Obj2 spread: {obj2_spread:.3f}")
    
    print(f"\n=== Summary ===")
    print(f"Instances with potential HV issues: {negative_hv_count}/{num_instances}")
    
    # Test with known good solution
    print(f"\n=== Testing with known good approach ===")
    mokp = MultiObjectiveKnapsack(n_items=n_items, n_objectives=2, capacity=capacity)
    
    # Extreme weight solutions
    print("\nExtreme weight solutions:")
    
    # Weight = 0 (only objective 2)
    greedy_obj2 = [0] * n_items
    items_by_obj2 = sorted(range(n_items), key=lambda i: mokp.values[1][i]/mokp.weights[i] 
                          if mokp.weights[i] > 0 else 0, reverse=True)
    used = 0
    for i in items_by_obj2:
        if used + mokp.weights[i] <= capacity:
            greedy_obj2[i] = 1
            used += mokp.weights[i]
    
    obj2_result = mokp.evaluate(greedy_obj2)
    print(f"  Weight 0.0 (max obj2): {obj2_result}")
    
    # Weight = 1 (only objective 1)
    greedy_obj1 = [0] * n_items
    items_by_obj1 = sorted(range(n_items), key=lambda i: mokp.values[0][i]/mokp.weights[i] 
                          if mokp.weights[i] > 0 else 0, reverse=True)
    used = 0
    for i in items_by_obj1:
        if used + mokp.weights[i] <= capacity:
            greedy_obj1[i] = 1
            used += mokp.weights[i]
    
    obj1_result = mokp.evaluate(greedy_obj1)
    print(f"  Weight 1.0 (max obj1): {obj1_result}")
    
    # Expected ranges
    print(f"\nExpected objective ranges for BiKP{n_items}:")
    avg_items_fit = capacity / 0.5  # Average weight is ~0.5
    print(f"  Average items that fit: ~{avg_items_fit:.0f}")
    print(f"  Expected max objective: ~{avg_items_fit * 0.5:.0f}")
    
    return negative_hv_count

# Also create a function to verify the WSDP implementation
def verify_wsdp_correctness():
    """Verify WSDP is working correctly on a simple instance"""
    print("\n=== Verifying WSDP Correctness ===")
    
    # Create a simple 5-item problem
    mokp = MultiObjectiveKnapsack(n_items=5, n_objectives=2, capacity=2.5)
    
    # Print problem details
    print(f"\nProblem details:")
    print(f"Weights: {[f'{w:.3f}' for w in mokp.weights]}")
    print(f"Values Obj1: {[f'{v:.3f}' for v in mokp.values[0]]}")
    print(f"Values Obj2: {[f'{v:.3f}' for v in mokp.values[1]]}")
    print(f"Capacity: {mokp.capacity}")
    
    # Run WSDP
    wsdp = WSDP(mokp, num_weights=10)
    solutions = wsdp.run()
    
    print(f"\nFound {len(solutions)} solutions:")
    for i, (sol, obj) in enumerate(solutions):
        weight = sum(mokp.weights[j] * sol[j] for j in range(5))
        items = [j for j in range(5) if sol[j] == 1]
        print(f"  Sol {i+1}: items={items}, weight={weight:.3f}, obj={obj}")
    
    # Verify all 2^5 = 32 possible solutions to find true Pareto front
    print(f"\nBrute force verification:")
    all_feasible = []
    
    for mask in range(32):
        sol = [(mask >> i) & 1 for i in range(5)]
        weight = sum(mokp.weights[i] * sol[i] for i in range(5))
        
        if weight <= mokp.capacity:
            obj = mokp.evaluate(sol)
            all_feasible.append((sol, obj))
    
    # Find non-dominated
    true_pareto = []
    for sol1, obj1 in all_feasible:
        dominated = False
        for sol2, obj2 in all_feasible:
            if obj2[0] >= obj1[0] and obj2[1] >= obj1[1] and (obj2[0] > obj1[0] or obj2[1] > obj1[1]):
                dominated = True
                break
        if not dominated:
            true_pareto.append((sol1, obj1))
    
    print(f"\nTrue Pareto front has {len(true_pareto)} solutions")
    
    # Compare
    print(f"WSDP found {len(solutions)} vs true {len(true_pareto)}")
    
    return solutions, true_pareto

if __name__ == "__main__":
    test_weighted_sum_methods(num_runs=10, num_weights=80)

    # First verify on small instance
    # verify_wsdp_correctness()
    
    # # Then diagnose larger instances
    # diagnose_negative_hv(n_items=50, capacity=12.5, num_instances=3)
    # diagnose_negative_hv(n_items=100, capacity=25, num_instances=3)
