"""
Screenshot comparison utility.

Two-stage comparison (dHash + SSIM) to decide if two screenshots show the "same" frame.
Supports cropping status/nav bar; pure Python, no OpenCV.
"""

import numpy as np
from pathlib import Path
from typing import Union, Tuple, Optional, Dict, Any
from PIL import Image
from datetime import datetime

# Constants
DEFAULT_DHASH_SIZE = (9, 8)
DEFAULT_DHASH_THRESHOLD = 5
DEFAULT_SSIM_GRAY_ZONE = (6, 12)
DEFAULT_SSIM_SIZE = (256, 256)
DEFAULT_SSIM_THRESHOLD = 0.98

# Crop params for 461x1024 scaled screenshots
STATUS_BAR_HEIGHT_461x1024 = 50  # Status bar height (px)
NAV_BAR_HEIGHT_461x1024 = 0  # Nav bar height (px; 0 = no crop)

# SSIM params
SSIM_C1 = (0.01 * 255) ** 2  # Stability constant 1
SSIM_C2 = (0.03 * 255) ** 2  # Stability constant 2
SSIM_WINDOW_SIZE = 11
SSIM_SIGMA = 1.5

# Diff analysis
DIFF_THRESHOLD = 10  # Diff visualization threshold (grayscale)


class ScreenshotComparator:
    """
    Two-stage comparator:
    1. dHash (perceptual hash) as primary (fast)
    2. SSIM (structural similarity) as fallback (gray zone only)
    """
    
    def __init__(
        self,
        dhash_size: Tuple[int, int] = DEFAULT_DHASH_SIZE,
        dhash_threshold: int = DEFAULT_DHASH_THRESHOLD,
        ssim_gray_zone: Tuple[int, int] = DEFAULT_SSIM_GRAY_ZONE,
        ssim_size: Tuple[int, int] = DEFAULT_SSIM_SIZE,
        ssim_threshold: float = DEFAULT_SSIM_THRESHOLD,
        crop_top: int = 0,
        crop_bottom: int = 0,
    ):
        """
        Args:
            dhash_size: Image size for dHash, default (9, 8)
            dhash_threshold: dHash Hamming threshold, default 5 (<=5 => same)
            ssim_gray_zone: Hamming range for SSIM, default (6, 12)
            ssim_size: Image size for SSIM, default (256, 256)
            ssim_threshold: SSIM similarity threshold, default 0.98 (>=0.98 => same)
            crop_top: Top crop in px (status bar), default 0; auto if 0 and auto_detect
            crop_bottom: Bottom crop in px (nav bar), default 0; auto if 0 and auto_detect
        """
        self.dhash_size = dhash_size
        self.dhash_threshold = dhash_threshold
        self.ssim_gray_zone = ssim_gray_zone
        self.ssim_size = ssim_size
        self.ssim_threshold = ssim_threshold
        self.crop_top = crop_top
        self.crop_bottom = crop_bottom
    
    def _load_image(self, image_input: Union[str, Path, Image.Image, np.ndarray]) -> np.ndarray:
        """
        Load image from path, PIL Image, or numpy array.
        
        Args:
            image_input: Path (str/Path), PIL Image, or HWC numpy array (RGB/RGBA)
        
        Returns:
            Grayscale numpy array, shape (H, W)
        """
        if isinstance(image_input, (str, Path)):
            img = Image.open(image_input)
        elif isinstance(image_input, Image.Image):
            img = image_input
        elif isinstance(image_input, np.ndarray):
            if len(image_input.shape) == 3:
                if image_input.shape[2] == 4:
                    img = Image.fromarray(image_input, 'RGBA')
                else:
                    img = Image.fromarray(image_input, 'RGB')
            else:
                return image_input.astype(np.float32)
        else:
            raise TypeError(f"Unsupported image type: {type(image_input)}")
        
        if img.mode == 'RGBA':
            background = Image.new('RGB', img.size, (255, 255, 255))
            background.paste(img, mask=img.split()[3])
            img = background
        elif img.mode != 'RGB' and img.mode != 'L':
            img = img.convert('RGB')
        
        if img.mode != 'L':
            img = img.convert('L')
        
        # To numpy
        img_array = np.array(img, dtype=np.float32)
        return img_array
    
    def _resize_image(self, img_array: np.ndarray, size: Tuple[int, int]) -> np.ndarray:
        """Resize image. Args: img_array, size (width, height). Returns resized array."""
        img = Image.fromarray(img_array.astype(np.uint8))
        img_resized = img.resize(size, Image.Resampling.LANCZOS)
        return np.array(img_resized, dtype=np.float32)
    
    def auto_detect_crop_bounds(
        self,
        image: Union[str, Path, Image.Image, np.ndarray],
        status_bar_height: Optional[int] = None,
        nav_bar_height: Optional[int] = None,
    ) -> Tuple[int, int]:
        """
        Auto-detect crop bounds (top status bar only; bottom nav bar not cropped by default).
        Heuristics: status bar ~20-41px for 461x1024; nav_bar_height=0 by default.
        Returns (crop_top, crop_bottom) in pixels.
        """
        img = self._load_image(image)
        h, w = img.shape
        
        if status_bar_height is None:
            if h == 1024 and w == 461:
                status_bar_height = STATUS_BAR_HEIGHT_461x1024
            else:
                status_bar_height = min(int(h * 0.02), 80)
                status_bar_height = max(status_bar_height, 20)
        
        if nav_bar_height is None:
        if nav_bar_height is None:
            nav_bar_height = NAV_BAR_HEIGHT_461x1024
        
        return status_bar_height, nav_bar_height
    
    def crop_content_area(
        self,
        image: Union[str, Path, Image.Image, np.ndarray],
        auto_detect: bool = False,
        save_cropped: Optional[Union[str, Path]] = None,
    ) -> np.ndarray:
        """Crop content area (drop top status bar). Returns cropped grayscale array."""
        img = self._load_image(image)
        
        if auto_detect and self.crop_top == 0 and self.crop_bottom == 0:
            crop_top, crop_bottom = self.auto_detect_crop_bounds(img)
        else:
            crop_top = self.crop_top
            crop_bottom = self.crop_bottom
        
        if crop_top == 0 and crop_bottom == 0:
            if save_cropped:
                cropped_img = Image.fromarray(img.astype(np.uint8), mode='L')
                cropped_img.save(save_cropped)
            return img
        
        h, w = img.shape
        top = crop_top
        bottom = h - crop_bottom if crop_bottom > 0 else h
        
        if top >= bottom:
            raise ValueError(f"Invalid crop: crop_top ({top}) >= effective bottom ({bottom})")
        
        cropped = img[top:bottom, :]
        
        if save_cropped:
            cropped_img = Image.fromarray(cropped.astype(np.uint8), mode='L')
            cropped_img.save(save_cropped)
            print(f"Saved cropped image: {save_cropped} (top {crop_top}px, bottom {crop_bottom}px)")
        
        return cropped
    
    def compute_dhash(self, image: Union[str, Path, Image.Image, np.ndarray]) -> int:
        """Compute dHash of image. Returns 64-bit hash (Python int)."""
        img = self._load_image(image)
        
        # Resize to (width+1) x height, e.g. 9x8
        width, height = self.dhash_size
        resized = Image.fromarray(img.astype(np.uint8)).resize((width + 1, height), Image.Resampling.LANCZOS)
        pixels = np.array(resized, dtype=np.float32)
        
        hash_bits = []
        for row in range(height):
            for col in range(width):
                if pixels[row, col] < pixels[row, col + 1]:
                    hash_bits.append(1)
                else:
                    hash_bits.append(0)
        
        # To 64-bit hash
        hash_value = 0
        for bit in hash_bits:
            hash_value = (hash_value << 1) | bit
        
        return hash_value
    
    def hamming_distance(self, hash1: int, hash2: int) -> int:
        """Hamming distance between two hashes (count of differing bits)."""
        # XOR then count ones
        xor = hash1 ^ hash2
        distance = bin(xor).count('1')
        return distance
    
    def compute_ssim(
        self,
        img1: Union[str, Path, Image.Image, np.ndarray],
        img2: Union[str, Path, Image.Image, np.ndarray],
    ) -> float:
        """Compute SSIM (structural similarity) of two images. Pure Python. Returns 0-1."""
        # Load and preprocess
        img1_gray = self._load_image(img1)
        img2_gray = self._load_image(img2)
        
        img1_resized = Image.fromarray(img1_gray.astype(np.uint8)).resize(
            self.ssim_size, Image.Resampling.LANCZOS
        )
        img2_resized = Image.fromarray(img2_gray.astype(np.uint8)).resize(
            self.ssim_size, Image.Resampling.LANCZOS
        )
        
        img1_array = np.array(img1_resized, dtype=np.float32)
        img2_array = np.array(img2_resized, dtype=np.float32)
        
        gaussian_window = self._create_gaussian_window(SSIM_WINDOW_SIZE, SSIM_SIGMA)
        
        # Local SSIM
        h, w = img1_array.shape
        pad = SSIM_WINDOW_SIZE // 2
        
        # Pad boundaries (reflect)
        img1_padded = np.pad(img1_array, pad, mode='reflect')
        img2_padded = np.pad(img2_array, pad, mode='reflect')
        
        ssim_map = np.zeros((h, w))
        
        for i in range(h):
            for j in range(w):
                patch1 = img1_padded[i:i+SSIM_WINDOW_SIZE, j:j+SSIM_WINDOW_SIZE]
                patch2 = img2_padded[i:i+SSIM_WINDOW_SIZE, j:j+SSIM_WINDOW_SIZE]
                
                mu1 = np.sum(patch1 * gaussian_window)
                mu2 = np.sum(patch2 * gaussian_window)
                
                sigma1_sq = np.sum(gaussian_window * (patch1 - mu1) ** 2)
                sigma2_sq = np.sum(gaussian_window * (patch2 - mu2) ** 2)
                sigma12 = np.sum(gaussian_window * (patch1 - mu1) * (patch2 - mu2))
                
                numerator = (2 * mu1 * mu2 + SSIM_C1) * (2 * sigma12 + SSIM_C2)
                denominator = (mu1 ** 2 + mu2 ** 2 + SSIM_C1) * (sigma1_sq + sigma2_sq + SSIM_C2)
                
                ssim_map[i, j] = numerator / denominator if denominator > 0 else 0.0
        
        return float(np.mean(ssim_map))
    
    def _create_gaussian_window(self, size: int, sigma: float) -> np.ndarray:
        """Create normalized Gaussian window for SSIM."""
        center = size // 2
        x, y = np.meshgrid(np.arange(size) - center, np.arange(size) - center)
        
        gaussian = np.exp(-(x ** 2 + y ** 2) / (2 * sigma ** 2))
        gaussian = gaussian / np.sum(gaussian)
        
        return gaussian
    
    def analyze_differences(
        self,
        img1: Union[str, Path, Image.Image, np.ndarray],
        img2: Union[str, Path, Image.Image, np.ndarray],
        crop: bool = True,
        auto_detect_crop: bool = False,
        save_diff: Optional[Union[str, Path]] = None,
    ) -> Dict[str, Any]:
        """Analyze difference between two images; optionally save diff visualization. Returns result dict."""
        # Preprocess
        if crop:
            img1_processed = self.crop_content_area(img1, auto_detect=auto_detect_crop)
            img2_processed = self.crop_content_area(img2, auto_detect=auto_detect_crop)
        else:
            img1_processed = self._load_image(img1)
            img2_processed = self._load_image(img2)
        
        # Ensure same size
        if img1_processed.shape != img2_processed.shape:
            target_h, target_w = min(img1_processed.shape[0], img2_processed.shape[0]), \
                                 min(img1_processed.shape[1], img2_processed.shape[1])
            img1_processed = self._resize_image(img1_processed, (target_w, target_h))
            img2_processed = self._resize_image(img2_processed, (target_w, target_h))
        
        diff = np.abs(img1_processed - img2_processed)
        
        # Stats
        diff_pixels = np.sum(diff > 0)
        total_pixels = diff.size
        diff_percentage = (diff_pixels / total_pixels) * 100
        max_diff = np.max(diff)
        mean_diff = np.mean(diff)
        
        diff_visual = np.zeros((*diff.shape, 3), dtype=np.uint8)
        img1_rgb = np.stack([img1_processed, img1_processed, img1_processed], axis=2).astype(np.uint8)
        diff_mask = diff > DIFF_THRESHOLD
        diff_visual = img1_rgb.copy()
        diff_visual[diff_mask] = [255, 0, 0]
        
        if save_diff:
            diff_img = Image.fromarray(diff_visual)
            diff_img.save(save_diff)
            print(f"Saved diff image: {save_diff}")
        
        return {
            'diff_pixels': int(diff_pixels),
            'total_pixels': int(total_pixels),
            'diff_percentage': float(diff_percentage),
            'max_diff': float(max_diff),
            'mean_diff': float(mean_diff),
            'diff_image_path': str(save_diff) if save_diff else None,
        }
    
    def compare(
        self,
        img1: Union[str, Path, Image.Image, np.ndarray],
        img2: Union[str, Path, Image.Image, np.ndarray],
        crop: bool = True,
        auto_detect_crop: bool = False,
        save_cropped: bool = False,
        output_dir: Optional[Union[str, Path]] = None,
    ) -> Dict[str, Any]:
        """
        Compare two images for "same" frame. Two-stage: dHash first, SSIM in gray zone.
        Returns dict with 'same', 'dhash_distance', 'ssim_score', 'method', etc.
        """
        if save_cropped and output_dir is None:
            if isinstance(img1, (str, Path)):
                output_dir = Path(img1).parent
            else:
                output_dir = Path.cwd()
        elif output_dir:
            output_dir = Path(output_dir)
            output_dir.mkdir(parents=True, exist_ok=True)
        
        cropped_img1_path = None
        cropped_img2_path = None
        if save_cropped and output_dir:
            timestamp = datetime.now().strftime("%Y%m%d_%H%M%S")
            cropped_img1_path = output_dir / f"cropped_img1_{timestamp}.png"
            cropped_img2_path = output_dir / f"cropped_img2_{timestamp}.png"
        
        # Preprocess (crop if needed)
        auto_detected = False
        if crop:
            if auto_detect_crop and self.crop_top == 0 and self.crop_bottom == 0:
                auto_detected = True
            img1_processed = self.crop_content_area(
                img1, 
                auto_detect=auto_detect_crop,
                save_cropped=cropped_img1_path if save_cropped else None
            )
            img2_processed = self.crop_content_area(
                img2, 
                auto_detect=auto_detect_crop,
                save_cropped=cropped_img2_path if save_cropped else None
            )
        else:
            img1_processed = self._load_image(img1)
            img2_processed = self._load_image(img2)
        
        # Stage 1: dHash
        hash1 = self.compute_dhash(img1_processed)
        hash2 = self.compute_dhash(img2_processed)
        dhash_distance = self.hamming_distance(hash1, hash2)
        
        base_result = {
            'auto_detected_crop': auto_detected,
        }
        if save_cropped:
            base_result['cropped_img1_path'] = str(cropped_img1_path) if cropped_img1_path else None
            base_result['cropped_img2_path'] = str(cropped_img2_path) if cropped_img2_path else None
        
        if dhash_distance <= self.dhash_threshold:
            return {
                'same': True,
                'dhash_distance': dhash_distance,
                'ssim_score': None,
                'method': 'dhash',
                **base_result,
            }
        
        if dhash_distance > self.ssim_gray_zone[1]:
            return {
                'same': False,
                'dhash_distance': dhash_distance,
                'ssim_score': None,
                'method': 'dhash',
                **base_result,
            }
        
        # Stage 2: SSIM (gray zone only)
        if self.ssim_gray_zone[0] <= dhash_distance <= self.ssim_gray_zone[1]:
            ssim_score = self.compute_ssim(img1_processed, img2_processed)
            
            is_same = ssim_score >= self.ssim_threshold
            
            return {
                'same': is_same,
                'dhash_distance': dhash_distance,
                'ssim_score': ssim_score,
                'method': 'ssim',
                **base_result,
            }
        
        return {
            'same': False,
            'dhash_distance': dhash_distance,
            'ssim_score': None,
            'method': 'dhash',
            **base_result,
        }


# Default singleton; create new instance for custom params
_default_comparator = ScreenshotComparator(dhash_threshold=2)  # Medium sensitivity for subtle diffs


def get_comparator() -> ScreenshotComparator:
    """Return the default screenshot comparator singleton."""
    return _default_comparator

