"""
Example usage of the CoBET package.

This script demonstrates how to use all three methods:
- CoBET (identity weights)
- dCoBET (J weights)
- wa_dCoBET (adaptive weights)
"""
import numpy as np
from cobet import run_test, CoBET, dCoBET, wa_dCoBET

# Example 1: Quick start with run_test() API
def example_run_test():
    """Example using the unified run_test() function."""
    print("\n" + "="*60)
    print("Example 1: Using run_test() unified API")
    print("="*60)
    
    # Define configuration
    n_list = [250, 500]
    b_config_by_n = {
        250: {
            "trigU": [0.05, 0.08],
            "linear": [0.1, 0.2],
        },
        500: {
            "trigU": [0.03, 0.05],
            "linear": [0.05, 0.1],
        }
    }
    
    # Run CoBET
    results = run_test(
        method='cobet',
        n_list=n_list,
        K=4,
        d=2,
        theta=2,
        b_config_by_n=b_config_by_n,
        R_eval=100,  # Use small number for quick demo
        alpha=0.05,
        seed=123,
        output='example_cobet_results.xlsx'
    )
    
    print("\nFirst few results:")
    print(results.head())
    
    return results


# Example 2: Using method classes directly for single test
def example_direct_test():
    """Example using method classes directly."""
    print("\n" + "="*60)
    print("Example 2: Direct use of method classes")
    print("="*60)
    
    # Generate some data
    np.random.seed(42)
    n = 100
    X = np.random.randn(n, 2)
    Y = 0.5 * X + 0.5 * np.random.randn(n, 2)
    
    # Test 1: CoBET
    print("\n--- Testing with CoBET ---")
    cobet = CoBET(K=4, d=2, alpha=0.05)
    result = cobet.test(X, Y)
    print(f"Test statistic: {result['statistic']:.4f}")
    print(f"Z-score: {result['Z']:.4f}")
    print(f"P-value: {result['p_value']:.4f}")
    print(f"Reject null: {result['reject']}")
    
    # Test 2: dCoBET
    print("\n--- Testing with dCoBET ---")
    dcobet = dCoBET(K=4, d=2, alpha=0.05, reuse_J=True)
    result = dcobet.test(X, Y)
    print(f"Test statistic: {result['statistic']:.4f}")
    print(f"Z-score: {result['Z']:.4f}")
    print(f"P-value: {result['p_value']:.4f}")
    print(f"Reject null: {result['reject']}")
    
    # Test 3: wa_dCoBET
    print("\n--- Testing with wa_dCoBET ---")
    wa_dcobet = wa_dCoBET(K=4, d=2, alpha=0.05, n_folds=10)
    result = wa_dcobet.test(X, Y)
    print(f"Test statistic: {result['statistic']:.4f}")
    print(f"Z-score: {result['Z']:.4f}")
    print(f"P-value: {result['p_value']:.4f}")
    print(f"Reject null: {result['reject']}")
    print(f"Weight on identity: {result['w_identity']:.3f}")
    print(f"Weight on J: {result['w_J']:.3f}")


# Example 3: Comparing methods on simulated data
def example_compare_methods():
    """Example comparing all three methods."""
    print("\n" + "="*60)
    print("Example 3: Comparing all three methods")
    print("="*60)
    
    # Small configuration for demo
    n_list = [250]
    b_config_by_n = {
        250: {
            "linear": [0.1, 0.2],
        }
    }
    
    methods = ['cobet', 'dcobet', 'wa_dcobet']
    all_results = []
    
    for method in methods:
        print(f"\nRunning {method}...")
        results = run_test(
            method=method,
            n_list=n_list,
            K=4,
            d=2,
            theta=2,
            b_config_by_n=b_config_by_n,
            transforms=("linear",),  # Just one transform for speed
            R_eval=50,  # Small number for demo
            alpha=0.05,
            seed=123,
            print_results=False
        )
        all_results.append(results)
    
    # Compare results
    import pandas as pd
    comparison = pd.concat(all_results, ignore_index=True)
    print("\n--- Comparison across methods ---")
    print(comparison[['method', 'transform', 'b', 'metric', 'value']])


# Example 4: Custom data generation and testing
def example_custom_data():
    """Example with custom data generation."""
    print("\n" + "="*60)
    print("Example 4: Custom data generation")
    print("="*60)
    
    # Create method instance
    method = CoBET(K=4, d=3, theta=2, seed=42)
    
    # Generate data using built-in generator
    X, Y = method.generate_data(n=200, transform_key='trigU', b=0.1)
    
    print(f"Generated data shape: X={X.shape}, Y={Y.shape}")
    print(f"X mean: {X.mean(axis=0)}")
    print(f"Y mean: {Y.mean(axis=0)}")
    
    # Test for independence
    result = method.test(X, Y)
    print(f"\nTest result:")
    print(f"  P-value: {result['p_value']:.4f}")
    print(f"  Reject: {result['reject']}")


if __name__ == "__main__":
    print("\n" + "="*60)
    print("CoBET Package Examples")
    print("="*60)
    
    # Run examples
    try:
        # Example 1: Run full simulation (commented out by default for speed)
        # example_run_test()
        
        # Example 2: Direct method usage
        example_direct_test()
        
        # Example 3: Compare methods (commented out by default for speed)
        # example_compare_methods()
        
        # Example 4: Custom data
        example_custom_data()
        
        print("\n" + "="*60)
        print("Examples completed successfully!")
        print("="*60 + "\n")
        
    except Exception as e:
        print(f"\nError running examples: {e}")
        import traceback
        traceback.print_exc()
