from playwright.sync_api import sync_playwright
import os
import re
import json
from collections import defaultdict
import time
import shutil

def capture_screenshot_and_interact(html_file_path, screenshot_path="screenshot.png", 
                                    interact_type="click", interaction_index=1,
                                    click_selector="#InteractionPart", 
                                    screenshot_after_interaction_path=None, log_file=None)
    if not os.path.exists(html_file_path):
        error_msg = f"HTML file not found: {html_file_path}"
        if log_file:
            log_file.write(error_msg + "\n")
            log_file.flush()
        raise FileNotFoundError(error_msg)
    
    absolute_path = os.path.abspath(html_file_path)
    file_url = f"file://{absolute_path}"
    
    if not screenshot_after_interaction_path:
        name, ext = os.path.splitext(screenshot_path)
        screenshot_after_interaction_path = f"{name}_after_interaction{ext}"
    
    verification_selectors = get_verification_selectors(interact_type, interaction_index)
    
    with sync_playwright() as p:
        browser = p.chromium.launch(headless=True)
        context = browser.new_context(
            viewport={'width': 1280, 'height': 800}
        )
        page = context.new_page()
        
        page.set_default_timeout(50000)
        
        try:
            page.goto(file_url, wait_until="networkidle")
            
            page.wait_for_load_state("load")
            
            log_msg = f"Waiting for element {click_selector} to appear..."
            print(log_msg)
            if log_file:
                log_file.write(log_msg + "\n")
                log_file.flush()
                
            page.wait_for_selector(click_selector, state="visible", timeout=15000)
            
            log_msg = "Waiting for images to load..."
            print(log_msg)
            if log_file:
                log_file.write(log_msg + "\n")
                log_file.flush()
            
            page.evaluate("""() => {
                return new Promise((resolve) => {
                    const images = document.querySelectorAll('img');
                    if (images.length === 0) {
                        return resolve();
                    }
                    
                    let loadedImages = 0;
                    const totalImages = images.length;
                    
                    const imageLoaded = () => {
                        loadedImages++;
                        if (loadedImages === totalImages) {
                            resolve();
                        }
                    };
                    
                    for (let img of images) {
                        if (img.complete) {
                            imageLoaded();
                        } else {
                            img.addEventListener('load', imageLoaded);
                            img.addEventListener('error', imageLoaded);
                        }
                    }
                });
            }""")
            
            page.wait_for_timeout(500)
            
            log_msg = "All images have been loaded, taking screenshots..."
            print(log_msg)
            if log_file:
                log_file.write(log_msg + "\n")
                log_file.flush()
            page.screenshot(path=screenshot_path, full_page=True)
            log_msg = f"Initial screenshot saved to: {screenshot_path}"
            print(log_msg)
            if log_file:
                log_file.write(log_msg + "\n")
                log_file.flush()
            
            interaction_success = False
            
            if interact_type == "click":
                try:
                    page.click(click_selector, force=True)
                    
                    page.wait_for_timeout(1000)
                    
                    page.evaluate("""() => {
                        return new Promise((resolve) => {
                            const images = document.querySelectorAll('img');
                            
                            if (images.length === 0) {
                                return resolve();
                            }
                            
                            let loadedImages = 0;
                            const totalImages = images.length;
                            
                            const imageLoaded = () => {
                                loadedImages++;
                                if (loadedImages === totalImages) {
                                    resolve();
                                }
                            };
                            
                            for (let img of images) {
                                if (img.complete) {
                                    imageLoaded();
                                } else {
                                    img.addEventListener('load', imageLoaded);
                                    img.addEventListener('error', imageLoaded);
                                }
                            }
                        });
                    }""")
                    
                    interaction_success = verify_interaction(page, verification_selectors, interact_type, interaction_index, log_file)
                    
                    page.screenshot(path=screenshot_after_interaction_path, full_page=True)
                    log_msg = f"Screenshot after interaction saved to: {screenshot_after_interaction_path}"
                    print(log_msg)
                    if log_file:
                        log_file.write(log_msg + "\n")
                        log_file.flush()
                    
                except Exception as e:
                    error_msg = f"Error clicking element: {e}"
                    print(error_msg)
                    if log_file:
                        log_file.write(error_msg + "\n")
                        log_file.flush()
                    raise
            
            elif interact_type == "hover":
                try:
                    page.hover(click_selector, force=True)
                    
                    page.wait_for_timeout(1000)
                    
                    page.evaluate("""() => {
                        return new Promise((resolve) => {
                            const images = document.querySelectorAll('img');
                            
                            if (images.length === 0) {
                                return resolve();
                            }
                            
                            let loadedImages = 0;
                            const totalImages = images.length;
                            
                            const imageLoaded = () => {
                                loadedImages++;
                                if (loadedImages === totalImages) {
                                    resolve();
                                }
                            };
                            
                            for (let img of images) {
                                if (img.complete) {
                                    imageLoaded();
                                } else {
                                    img.addEventListener('load', imageLoaded);
                                    img.addEventListener('error', imageLoaded);
                                }
                            }
                        });
                    }""")
                    interaction_success = verify_interaction(page, verification_selectors, interact_type, interaction_index, log_file)
                    
                    page.screenshot(path=screenshot_after_interaction_path, full_page=True)
                    log_msg = f"Screenshot after hover saved to: {screenshot_after_interaction_path}"
                    print(log_msg)
                    if log_file:
                        log_file.write(log_msg + "\n")
                        log_file.flush()
                    
                except Exception as e:
                    error_msg = f"Error hovering element: {e}"
                    print(error_msg)
                    if log_file:
                        log_file.write(error_msg + "\n")
                        log_file.flush()
                    raise
                    
            return interaction_success
            
        except Exception as e:
            error_msg = f"Error processing page: {e}"
            print(error_msg)
            if log_file:
                log_file.write(error_msg + "\n")
                log_file.flush()
            raise
        finally:
            browser.close()


def get_verification_selectors(interact_type, interaction_index):
    selectors = {
        "click": {
            1: [".dropdown-menu", ".dropdown-content", "[role='menu']", "[aria-expanded='true']"],
            2: ["input[type='checkbox']:checked", "[role='checkbox'][aria-checked='true']"],
            3: [],
            4: [".modal", ".dialog", "[role='dialog']", "[aria-modal='true']"],
            5: [".tooltip", ".tip", "[role='tooltip']", "[data-tooltip]"],
            6: ["input[type='text']", "textarea", "[contenteditable='true']", "[role='textbox']"]
        },
        "hover": {
            1: [".dropdown-menu", ".dropdown-content", "[role='menu']", "[aria-expanded='true']"],
            2: [],
            3: [],
            4: [".tooltip", ".tip", "[role='tooltip']", "[data-tooltip]"]
        }
    }
    
    return selectors.get(interact_type, {}).get(interaction_index, [])


def verify_interaction(page, selectors, interact_type, interaction_index, log_file=None):
    success = False
    log_msg = ""
    
    try:
        if interact_type == "click" and interaction_index == 6:
            js_result = page.evaluate("""() => {
                const newsletterContainer = document.getElementById('newsletterContainer');
                if (newsletterContainer && 
                    (newsletterContainer.style.display === 'block' || 
                     getComputedStyle(newsletterContainer).display !== 'none')) {
                    return true;
                }
                
                const inputs = document.querySelectorAll('input[type="text"], input[type="email"], textarea, [contenteditable="true"], [role="textbox"]');
                for (let input of inputs) {
                    if (getComputedStyle(input).display !== 'none') {
                        return true;
                    }
                }
                
                return false;
            }""")
            success = js_result
            log_msg = "Form displayed after click" if success else "Form not displayed after click"
        
        elif interact_type == "click" and interaction_index == 5:
            js_result = page.evaluate("""() => {
                const tooltipText = document.querySelector('#InteractionPart .tooltip-text');
                if (tooltipText) {
                    const style = window.getComputedStyle(tooltipText);
                    if (style.visibility === 'visible' || style.opacity > 0 || style.display === 'block') {
                        return true;
                    }
                }
                
                const tooltipIds = document.querySelectorAll('[id$="Tooltip"], [id$="tooltip"]');
                for (let tooltipEl of tooltipIds) {
                    const style = window.getComputedStyle(tooltipEl);
                    if (style.display === 'block' || style.visibility === 'visible' || style.opacity > 0) {
                        return true;
                    }
                }
                
                const tooltipElements = document.querySelectorAll('.tooltip, .tip, [role="tooltip"], [data-tooltip]');
                for (let tooltip of tooltipElements) {
                    const style = window.getComputedStyle(tooltip);
                    if (style.display !== 'none' && style.visibility !== 'hidden' && style.opacity > 0) {
                        return true;
                    }
                }
                
                return false;
            }""")
            success = js_result
            log_msg = "Tooltip displayed" if success else "Tooltip not displayed"
            
        elif interact_type == "click" and interaction_index == 3:
            js_result = page.evaluate("""() => {
                const element = document.querySelector('#InteractionPart');
                if (!element) return false;
                const style = window.getComputedStyle(element);
                return style.backgroundColor !== 'rgba(0, 0, 0, 0)' && 
                       style.backgroundColor !== 'transparent' &&
                       style.backgroundColor !== 'rgb(255, 255, 255)';
            }""")
            success = js_result
            log_msg = "Background color changed" if success else "Background color not changed"
            
        elif interact_type == "hover" and interaction_index == 2:
            js_result = page.evaluate("""() => {
                const element = document.querySelector('#InteractionPart');
                if (!element) return false;
                const style = window.getComputedStyle(element);
                return style.fontWeight >= 600 || 
                       style.fontWeight === 'bold' || 
                       style.fontWeight === 'bolder';
            }""")
            success = js_result
            log_msg = "Text bolded" if success else "Text not bolded"
            
        elif interact_type == "hover" and interaction_index == 3:
            js_result = page.evaluate("""() => {
                const element = document.querySelector('#InteractionPart');
                if (!element) return false;
                const style = window.getComputedStyle(element);
                return style.textDecoration.includes('underline');
            }""")
            success = js_result
            log_msg = "Text underlined" if success else "Text not underlined"
            
        elif interact_type == "hover" and interaction_index == 1:
            js_result = page.evaluate("""() => {
                const dropdownInside = document.querySelector('#InteractionPart .dropdown-menu');,
                if (dropdownInside) {
                    const style = window.getComputedStyle(dropdownInside);
                    if (style.display !== 'none') {
                        return true;
                    }
                }
                
                const dropdownById = document.getElementById('placementDropdown');
                if (dropdownById) {
                    const style = window.getComputedStyle(dropdownById);
                    if (style.display !== 'none') {
                        return true;
                    }
                }
                
                const allDropdowns = document.querySelectorAll('.dropdown-menu');
                for (const dropdown of allDropdowns) {
                    const style = window.getComputedStyle(dropdown);
                    if (style.display !== 'none' && style.visibility !== 'hidden') {
                        return true;
                    }
                }
                
                return false;
            }""")
            success = js_result
            log_msg = "Dropdown menu displayed" if success else "Dropdown menu not displayed"
            
        elif interact_type == "hover" and interaction_index == 4:
            js_result = page.evaluate("""() => {
                const tooltipText = document.querySelector('#InteractionPart .tooltip-text');
                if (tooltipText) {
                    const style = window.getComputedStyle(tooltipText);
                    if (style.visibility === 'visible' || style.opacity > 0) {
                        return true;
                    }
                }
                
                const helpTooltip = document.getElementById('helpTooltip');
                if (helpTooltip) {
                    const style = window.getComputedStyle(helpTooltip);
                    return style.display === 'block';
                }
                
                const generalTooltips = document.querySelectorAll('.tooltip, .tip, [role="tooltip"], [data-tooltip]');
                for (let tooltip of generalTooltips) {
                    const style = window.getComputedStyle(tooltip);
                    if (style.display !== 'none' && style.visibility !== 'hidden' && style.opacity > 0) {
                        return true;
                    }
                }
                
                return false;
            }""")
            success = js_result
            log_msg = "Tooltip displayed" if success else "Tooltip not displayed"
            
        elif selectors:
            if interact_type == "hover" and interaction_index == 1:
                css_visible = page.evaluate("""() => {
                    const dropdowns = document.querySelectorAll('.dropdown-menu, .dropdown-content, [role="menu"]');
                    for (const dropdown of dropdowns) {
                        const style = window.getComputedStyle(dropdown);
                        if (style.display !== 'none' && style.visibility !== 'hidden') {
                            return true;
                        }
                    }
                    return false;
                }""")
                
                if css_visible:
                    success = True
                    log_msg = "Interaction success: Dropdown menu detected as visible through computed style"
                
            if not success:
                for selector in selectors:
                    if page.query_selector(selector):
                        success = True
                        log_msg = f"Interaction success: Found element {selector}"
                        break
            
                if not success:
                    if interact_type == "hover" and (interaction_index == 1 or interaction_index == 4):
                        fallback_check = page.evaluate("""() => {
                            const elements = document.querySelectorAll('.dropdown-menu, .dropdown-content, [role="menu"], .tooltip, .tip, [role="tooltip"], [data-tooltip]');
                            for (const el of elements) {
                                const style = window.getComputedStyle(el);
                                if (style.display !== 'none' && style.visibility !== 'hidden' && style.opacity > 0) {
                                    return true;
                                }
                            }
                            return false;
                        }""")
                        
                        if fallback_check:
                            success = True
                            log_msg = "Interaction success: Found visible interactive elements"
                        else:
                            log_msg = f"Interaction verification failed: Expected element not found {selectors}"
                    else:
                        log_msg = f"Interaction verification failed: Expected element not found {selectors}"
    except Exception as e:
        log_msg = f"Error verifying interaction: {e}"
        success = False
    
    if log_file:
        log_file.write(log_msg + "\n")
        log_file.flush()
    print(log_msg)
    
    return success


def process_all_html_files_in_directory(directory_path, output_path, log_file_path="output.txt", folder_name=None):

    interaction_stats = defaultdict(lambda: {"total": 0, "success": 0, "failed_files": []})
    
    os.makedirs(output_path, exist_ok=True)
    
    with open(log_file_path, 'a', encoding='utf-8') as log_file:
        log_msg = f"\nProcessing directory: {directory_path}"
        print(log_msg)
        log_file.write(log_msg + "\n")
        log_file.flush()
        
        files = os.listdir(directory_path)
        
        html_files = [file for file in files if re.match(r"(click|hover)_\d+_\d+\.html", file)]
        
        failed_files = []
        
        for html_file in html_files:
            try:
                match = re.match(r"(click|hover)_(\d+)_\d+\.html", html_file)
                if match:
                    interact_type = match.group(1)
                    interaction_index = int(match.group(2))
                    interaction_key = f"{interact_type}_{interaction_index}"
                    
                    interaction_stats[interaction_key]["total"] += 1
                    
                    html_file_path = os.path.join(directory_path, html_file)
                    
                    screenshot_before_path = os.path.join(output_path, f"{html_file[:-5]}(before).png")
                    screenshot_after_path = os.path.join(output_path, f"{html_file[:-5]}(after).png")
                    
                    if os.path.exists(screenshot_before_path) and os.path.exists(screenshot_after_path):
                        log_msg = f"Skip {html_file}, screenshot already exists"
                        print(log_msg)
                        log_file.write(log_msg + "\n")
                        log_file.flush()
                        interaction_stats[interaction_key]["success"] += 1
                        continue
                    
                    success = capture_screenshot_and_interact(
                        html_file_path, 
                        screenshot_before_path, 
                        interact_type=interact_type,
                        interaction_index=interaction_index,
                        click_selector="#InteractionPart", 
                        screenshot_after_interaction_path=screenshot_after_path,
                        log_file=log_file
                    )
                    
                    if success:
                        interaction_stats[interaction_key]["success"] += 1
                    else:
                        interaction_stats[interaction_key]["failed_files"].append(html_file)
            except Exception as e:
                error_msg = f"Error processing {html_file}: {e}"
                print(error_msg)
                log_file.write(error_msg + "\n")
                log_file.flush()
                failed_files.append(html_file)
                
                match = re.match(r"(click|hover)_(\d+)_\d+\.html", html_file)
                if match:
                    interact_type = match.group(1)
                    interaction_index = int(match.group(2))
                    interaction_key = f"{interact_type}_{interaction_index}"
                    interaction_stats[interaction_key]["failed_files"].append(html_file)
        
        if failed_files:
            log_msg = "\nFiles that threw exceptions during processing:"
            print(log_msg)
            log_file.write(log_msg + "\n")
            for file in failed_files:
                log_msg = f"- {file}"
                print(log_msg)
                log_file.write(log_msg + "\n")
        else:
            log_msg = "\nAll files processed without exceptions!"
            print(log_msg)
            log_file.write(log_msg + "\n")
        
        log_msg = "\nInteraction success rate statistics:"
        print(log_msg)
        log_file.write(log_msg + "\n")
        
        for interaction_key, stats in sorted(interaction_stats.items()):
            if stats["total"] > 0:
                success_rate = (stats["success"] / stats["total"]) * 100
                log_msg = f"Interaction type {interaction_key}: Success rate {success_rate:.2f}% ({stats['success']}/{stats['total']})"
                print(log_msg)
                log_file.write(log_msg + "\n")
                
                if len(stats["failed_files"]) > 0:
                    log_msg = f"\nInteraction type {interaction_key} failed files list:"
                    print(log_msg)
                    log_file.write(log_msg + "\n")
                    for file in stats["failed_files"]:
                        log_msg = f"- {file}"
                        print(log_msg)
                        log_file.write(log_msg + "\n")
        
        if folder_name:
            parent_dir = os.path.dirname(output_path)
            stats_path = os.path.join(parent_dir, f"{folder_name}_stats.json")
            with open(stats_path, 'w', encoding='utf-8') as stats_file:
                json.dump(interaction_stats, stats_file, indent=2)
                log_msg = f"Interaction statistics saved to: {stats_path}"
                print(log_msg)
                log_file.write(log_msg + "\n")
    
    return interaction_stats


def process_nested_folders(source_root, target_root, log_file_path="output.txt"):

    os.makedirs(os.path.dirname(log_file_path), exist_ok=True)
    
    with open(log_file_path, 'w', encoding='utf-8') as f:
        f.write(f"Processing folder: {source_root}\n")
    
    os.makedirs(target_root, exist_ok=True)
    
    start_time = time.time()
    
    for folder1 in os.listdir(source_root):
        folder1_path = os.path.join(source_root, folder1)
        
        if not os.path.isdir(folder1_path):
            continue
        
        target_folder1 = os.path.join(target_root, folder1)
        os.makedirs(target_folder1, exist_ok=True)
        
        inter_folder_path = os.path.join(folder1_path, "Interaction_to_code")
        
        if os.path.exists(inter_folder_path) and os.path.isdir(inter_folder_path):
            target_inter_folder = os.path.join(target_folder1, "Interaction_to_code")
            os.makedirs(target_inter_folder, exist_ok=True)
            
            with open(log_file_path, 'a', encoding='utf-8') as log_file:
                log_msg = f"\nProcessing folder: {folder1}, Interaction folder: {inter_folder_path}"
                print(log_msg)
                log_file.write(log_msg + "\n")
            
            process_all_html_files_in_directory(
                inter_folder_path, 
                target_inter_folder, 
                log_file_path,
                folder_name=folder1
            )
    
    end_time = time.time()
    with open(log_file_path, 'a', encoding='utf-8') as log_file:
        log_msg = f"\nTotal running time: {end_time - start_time:.2f} seconds"
        print(log_msg)
        log_file.write(log_msg + "\n")


if __name__ == "__main__":

    # source_root = "./pro_htmls"  
    source_root = "gpt5htmls"
    target_root = "./interaction_results"  # target folder path
    log_file_path = "./interaction_log.txt"  # log file path

    process_nested_folders(source_root, target_root, log_file_path)

