"""
PowerPoint automation utilities for converting presentations to images using PowerPoint Online.

Requirements:
    1. Install playwright: pip install playwright
    2. Download browser binaries: playwright install chromium
"""

import asyncio
import os
from pathlib import Path
from typing import Optional

from dotenv import load_dotenv
from playwright.async_api import async_playwright

from officearena.utils.onedrive import OneDriveClient  # noqa: E402


async def convert_powerpoint_to_images(
    edit_link: str,
    download_dir: Optional[str] = None,
    headless: bool = True,
    verbose: bool = True,
    wait_timeout: int = 60,
    download_timeout: int = 30,
) -> Optional[str]:
    """
    Convert a PowerPoint presentation to images using PowerPoint Online's export feature.

    Args:
        edit_link: The edit link for the PowerPoint presentation
        download_dir: Directory to save downloaded images (default: ./downloads)
        headless: Run browser in headless mode (default: True)
        verbose: Enable verbose logging (default: True)
        wait_timeout: Timeout in seconds to wait for the presentation to load
        download_timeout: Timeout in seconds to wait for the download to complete

    Returns:
        Path to the downloaded zip file containing images, or None if failed
    """
    if download_dir is None:
        download_dir = os.path.join(os.getcwd(), "downloads")

    # Ensure download directory exists
    Path(download_dir).mkdir(parents=True, exist_ok=True)

    def log(message: str):
        if verbose:
            print(f"[PowerPoint Converter] {message}")

    playwright = None
    browser = None
    context = None
    page = None

    try:
        # Step 1: Set up browser
        log("Starting browser setup...")
        playwright = await async_playwright().start()

        # Configure browser launch options for better headless compatibility
        launch_options = {
            "headless": headless,
            "args": (
                [
                    "--no-sandbox",
                    "--disable-setuid-sandbox",
                    "--disable-dev-shm-usage",
                    "--disable-accelerated-2d-canvas",
                    "--no-first-run",
                    "--no-zygote",
                    "--disable-gpu",
                    "--disable-web-security",
                    "--disable-features=VizDisplayCompositor",
                ]
                if headless
                else []
            ),
        }

        browser = await playwright.chromium.launch(**launch_options)

        # Create context with realistic user agent and settings
        context = await browser.new_context(
            accept_downloads=True,
            viewport={"width": 1920, "height": 1080},
            user_agent="Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36",
            extra_http_headers={
                "Accept": "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8",
                "Accept-Language": "en-US,en;q=0.5",
                "Accept-Encoding": "gzip, deflate",
                "Connection": "keep-alive",
                "Upgrade-Insecure-Requests": "1",
            },
        )
        page = await context.new_page()
        log("Browser setup complete")

        # Step 2: Navigate to the PowerPoint presentation
        log(f"Navigating to: {edit_link}")
        await page.goto(edit_link, wait_until="domcontentloaded")

        # Additional wait for PowerPoint Online to fully initialize
        # Headless mode often needs more time for complex apps
        actual_wait = wait_timeout * 1.5 if headless else wait_timeout
        log(f"Waiting {actual_wait} seconds for presentation to load...")
        await asyncio.sleep(actual_wait)

        # Step 3: Look for the File button
        log("Looking for File button...")

        if verbose:
            # List all buttons with data-unique-id attributes for debugging
            frame = [f for f in page.frames if "WacFrame" in f.name]
            if not frame:
                raise Exception("No WacFrame found. Check if the page is loaded correctly.")
            frame = frame[0]
        file_button = None
        file_selectors = [
            "#FileMenuFlyoutLauncher",
            '[data-unique-id*="File"]',
            'button[id*="File"]',
            'button:has-text("File")',
            '[aria-label*="File"]',
            '[data-automation-id*="File"]',
        ]

        for selector in file_selectors:
            try:
                log(f"   Trying selector: {selector}")
                element = await frame.wait_for_selector(selector, timeout=5000, state="visible")
                if element:
                    file_button = element
                    log(f"   Found File button with: {selector}")
                    break
            except Exception as e:
                log(f"   Failed with {selector}: {str(e)}")

        if not file_button:
            raise Exception("Could not find File button. Check debug screenshot for available elements.")

        await file_button.click()
        log("Clicked File button")

        # Step 4: Look for the Export menu item
        log("Looking for Export menu item...")

        # Give the menu a moment to appear (longer wait for headless)
        menu_wait = 5 if headless else 2
        await asyncio.sleep(menu_wait)

        export_item = None
        try:
            # Increase timeout for headless mode
            timeout = 15000 if headless else 10000
            element = await frame.wait_for_selector('[role="menuitem"]', timeout=timeout)
            export_element = frame.get_by_role("menuitem", name="Export")
            if await export_element.is_visible():
                export_item = export_element
                log("   Found Export item.")
                await export_element.click()
        except Exception as e:
            log(f"   Failed with get_by_role, trying fallback selectors: {str(e)}")

            # Fallback selectors for headless mode
            export_selectors = [
                '[data-unique-id="FileMenuExportSelection"]',
                '*[data-unique-id*="Export"]',
                'text="Export"',
                '[aria-label*="Export"]',
            ]

            for selector in export_selectors:
                try:
                    log(f"   Trying selector: {selector}")
                    element = await frame.wait_for_selector(selector, timeout=5000)
                    if element and await element.is_visible():
                        export_item = element
                        log(f"   Found Export item with: {selector}")
                        await element.click()
                        break
                except Exception as e:
                    log(f"   Failed with {selector}: {str(e)}")

        if not export_item:
            raise Exception("Could not find Export menu item. Check debug screenshot.")

        log("Clicked Export menu item")

        # Step 5: Look for the "Download as Images" option
        log("Looking for Download as Images option...")

        download_images_item = None
        try:
            # Increase timeout for headless mode
            timeout = 15000 if headless else 10000
            export_images_element = frame.get_by_role("menuitem", name="Export to Images")
            if await export_images_element.is_visible():
                download_images_item = export_images_element
                log("   Found Export to Images.")
                await export_images_element.click()
        except Exception as e:
            log(f"   Failed with get_by_role, trying fallback selectors: {str(e)}")

            # Fallback selectors for headless mode
            download_selectors = [
                '[data-unique-id="PptJewelDownloadAsImages"]',
                '[data-unique-id*="Download"]',
                '[data-unique-id*="Images"]',
                'button:has-text("Download")',
                'button:has-text("Images")',
                '*[aria-label*="Download"]',
                '*[aria-label*="Images"]',
            ]

            for selector in download_selectors:
                try:
                    log(f"   Trying selector: {selector}")
                    element = await frame.wait_for_selector(selector, timeout=5000)
                    if element and await element.is_visible():
                        download_images_item = element
                        log(f"   Found Download as Images with: {selector}")
                        await element.click()
                        break
                except Exception as e:
                    log(f"   Failed with {selector}: {str(e)}")

        if not download_images_item:
            raise Exception("Could not find Download as Images option. Check debug screenshot.")

        # Step 6: Wait for processing
        log(f"Waiting {download_timeout} seconds for image processing...")
        await asyncio.sleep(download_timeout)

        # Step 7: Click on the Download button
        log("Looking for Download button...")
        download_button_selector = 'button[aria-label="Download"]'
        download_button = await frame.wait_for_selector(download_button_selector, timeout=30000)

        # Set up download handler before clicking
        download_info = []

        def handle_download(download):
            download_info.append(download)

        page.on("download", handle_download)

        await download_button.click()
        log("Clicked Download button")

        # Wait for download to start and complete
        log("Waiting for download to complete...")
        await asyncio.sleep(60)  # Give download time to start

        if download_info:
            download = download_info[0]
            # Generate a unique filename
            timestamp = str(int(asyncio.get_event_loop().time()))
            filename = f"powerpoint_images_{timestamp}.zip"
            download_path = os.path.join(download_dir, filename)

            await download.save_as(download_path)
            log(f"Download completed: {download_path}")
            return download_path
        else:
            log("No download detected")
            return None

    except Exception as e:
        log(f"Error during conversion: {str(e)}")
        if verbose and page:
            try:
                await page.screenshot(path=os.path.join(download_dir, "debug_error.png"))
                log("Error screenshot saved")
            except Exception:
                pass
        return None

    finally:
        # Clean up
        try:
            if page:
                await page.close()
            if context:
                await context.close()
            if browser:
                await browser.close()
            if playwright:
                await playwright.stop()
        except Exception:
            pass


def convert_powerpoint_to_images_sync(
    edit_link: str,
    download_dir: Optional[str] = None,
    headless: bool = True,
    verbose: bool = True,
    wait_timeout: int = 15,
    download_timeout: int = 120,
) -> Optional[str]:
    """
    Synchronous wrapper for convert_powerpoint_to_images.

    Args:
        edit_link: The edit link for the PowerPoint presentation
        download_dir: Directory to save downloaded images
        headless: Run browser in headless mode
        verbose: Enable verbose logging
        wait_timeout: Timeout in seconds to wait for the presentation to load
        download_timeout: Timeout in seconds to wait for the download to complete

    Returns:
        Path to the downloaded zip file containing images, or None if failed
    """
    return asyncio.run(
        convert_powerpoint_to_images(
            edit_link=edit_link,
            download_dir=download_dir,
            headless=headless,
            verbose=verbose,
            wait_timeout=wait_timeout,
            download_timeout=download_timeout,
        )
    )


def ensure_classic_ribbon_always_show(
    sandbox,
    verbose: bool = True,
    attempts: int = 2,
    wait_between: float = 1.0,
) -> bool:
    """Switch PowerPoint Online to Classic Ribbon + Always show (sync, for ScreenEnv).

    Args:
        sandbox: ScreenEnv sandbox providing a sync Playwright page.
        verbose: If True, print progress logs.
        attempts: How many times to retry opening the toggle if radios not found.
        wait_between: Sleep (seconds) between attempts.

    Returns:
        True if we believe Classic + Always show were applied (or already set), else False.
    """

    def log(msg: str):
        if verbose:
            print(f"[Ribbon] {msg}")

    context = sandbox.chromium_context
    page = context.pages[0] if context.pages else None
    if page is None:
        log("No page attribute on sandbox.")
        return False

    try:
        # Locate frame
        log("Locating WacFrame...")
        frame = None
        for f in page.frames:
            try:
                name = getattr(f, "name", lambda: "")()
            except Exception:
                # Some Playwright versions expose name as property not callable
                try:
                    name = f.name
                except Exception:
                    name = ""
            if "WacFrame" in str(name):
                frame = f
                log(f"Found frame: {name}")
                break
        if frame is None:
            log("WacFrame not found; can't configure ribbon.")
            return False

        def click_home_tab():
            """Click on the Home tab to ensure ribbon is visible."""
            try:
                # Try to find and click the Home tab button
                home_selectors = [
                    "button#Home",
                    '[data-unique-id*="Home"]',
                    'button[id*="Home"]',
                    'button:has-text("Home")',
                    '[aria-label*="Home"]',
                    '[data-automation-id*="Home"]',
                ]

                for selector in home_selectors:
                    try:
                        log(f"   Trying Home tab selector: {selector}")
                        home_button = frame.wait_for_selector(selector, timeout=2500, state="visible")
                        if home_button:
                            home_button.click()
                            log(f"   Clicked Home tab with: {selector}")
                            return True
                    except Exception as e:
                        log(f"   Failed with Home selector {selector}: {e}")

                log("Home tab button not found")
                return False
            except Exception as e:
                log(f"Home tab click failed: {e}")
                return False

        def safe_click_ribbon_toggle(tag: str):
            try:
                # First click on Home tab to ensure ribbon is visible
                click_home_tab()

                el = frame.wait_for_selector("#RibbonModeToggle", timeout=2500, state="visible")
                if el:
                    el.click()
                    log(f"Clicked Ribbon toggle ({tag}).")
                    return True
                log(f"Ribbon toggle not returned ({tag}).")
            except Exception as e:
                log(f"Ribbon toggle click failed ({tag}): {e}")
            return False

        def try_click_radio(label: str) -> bool:
            try:
                radio = frame.get_by_role("menuitemradio", name=label)
                if radio:
                    radio.click()
                    log(f"Clicked radio '{label}'.")
                    return True
            except Exception as e:
                log(f"Radio '{label}' not clickable: {e}")
            return False

        classic_done = False
        always_done = False

        for attempt in range(1, attempts + 1):
            log(f"Attempt {attempt}/{attempts} - setting Classic Ribbon...")
            safe_click_ribbon_toggle(f"classic-attempt-{attempt}")
            if try_click_radio("Classic Ribbon"):
                classic_done = True
            else:
                log("Classic Ribbon radio not found this attempt.")
            # Always show
            log(f"Attempt {attempt}/{attempts} - setting Always show ribbon...")
            safe_click_ribbon_toggle(f"always-attempt-{attempt}")
            if try_click_radio("Always show ribbon"):
                always_done = True
            else:
                log("Always show ribbon radio not found this attempt.")

            if classic_done and always_done:
                log("Both Classic and Always show applied.")
                return True
            if attempt < attempts:
                import time

                time.sleep(wait_between)

        log(f"Finished attempts. classic_done={classic_done} always_done={always_done}")
        return classic_done or always_done  # partial success acceptable
    except Exception as e:
        log(f"Unexpected failure: {e}")
        return False


def download_powerpoint_as_images_sync(
    sandbox,
    download_dir: Optional[str] = None,
    verbose: bool = True,
    download_timeout: int = 30,
) -> Optional[str]:
    """
    Convert current PowerPoint presentation to images using existing sandbox.

    Args:
        sandbox: ScreenEnv sandbox with PowerPoint already open
        download_dir: Directory to save downloaded images (default: ./downloads)
        verbose: Enable verbose logging (default: True)
        download_timeout: Timeout in seconds to wait for image processing

    Returns:
        Path to the downloaded zip file containing images, or None if failed
    """
    if download_dir is None:
        download_dir = os.path.join(os.getcwd(), "downloads")

    # Ensure download directory exists
    Path(download_dir).mkdir(parents=True, exist_ok=True)

    def log(message: str):
        if verbose:
            print(f"[PowerPoint Converter] {message}")

    try:
        context = sandbox.chromium_context
        page = context.pages[0] if context.pages else None
        if page is None:
            raise Exception("No page found in sandbox")

        # Locate WacFrame
        log("Locating WacFrame...")
        frame = None
        for f in page.frames:
            try:
                name = getattr(f, "name", lambda: "")()
            except Exception:
                try:
                    name = f.name
                except Exception:
                    name = ""
            if "WacFrame" in str(name):
                frame = f
                log(f"Found frame: {name}")
                break

        if frame is None:
            raise Exception("No WacFrame found. Check if the page is loaded correctly.")

        # Step 1: Look for the File button
        log("Looking for File button...")
        file_button = None
        file_selectors = [
            "#FileMenuFlyoutLauncher",
            '[data-unique-id*="File"]',
            'button[id*="File"]',
            'button:has-text("File")',
            '[aria-label*="File"]',
            '[data-automation-id*="File"]',
        ]

        for selector in file_selectors:
            try:
                log(f"   Trying selector: {selector}")
                element = frame.wait_for_selector(selector, timeout=5000, state="visible")
                if element:
                    file_button = element
                    log(f"   Found File button with: {selector}")
                    break
            except Exception as e:
                log(f"   Failed with {selector}: {str(e)}")

        if not file_button:
            raise Exception("Could not find File button. Check debug screenshot for available elements.")

        file_button.click()
        log("Clicked File button")

        # Step 2: Look for the Export menu item
        log("Looking for Export menu item...")

        # Give the menu a moment to appear
        import time

        time.sleep(2)

        export_item = None
        try:
            export_element = frame.get_by_role("menuitem", name="Export")
            if export_element.is_visible():
                export_item = export_element
                log("   Found Export item.")
                export_element.click()
        except Exception as e:
            log(f"   Failed with get_by_role, trying fallback selectors: {str(e)}")

            # Fallback selectors
            export_selectors = [
                '[data-unique-id="FileMenuExportSelection"]',
                '*[data-unique-id*="Export"]',
                'text="Export"',
                '[aria-label*="Export"]',
            ]

            for selector in export_selectors:
                try:
                    log(f"   Trying selector: {selector}")
                    element = frame.wait_for_selector(selector, timeout=5000)
                    if element and element.is_visible():
                        export_item = element
                        log(f"   Found Export item with: {selector}")
                        element.click()
                        break
                except Exception as e:
                    log(f"   Failed with {selector}: {str(e)}")

        if not export_item:
            raise Exception("Could not find Export menu item. Check debug screenshot.")

        log("Clicked Export menu item")

        # Step 3: Look for the "Download as Images" option
        log("Looking for Download as Images option...")

        download_images_item = None
        try:
            export_images_element = frame.get_by_role("menuitem", name="Export to Images")
            if export_images_element.is_visible():
                download_images_item = export_images_element
                log("   Found Export to Images.")
                export_images_element.click()
        except Exception as e:
            log(f"   Failed with get_by_role, trying fallback selectors: {str(e)}")

            # Fallback selectors
            download_selectors = [
                '[data-unique-id="PptJewelDownloadAsImages"]',
                '[data-unique-id*="Download"]',
                '[data-unique-id*="Images"]',
                'button:has-text("Download")',
                'button:has-text("Images")',
                '*[aria-label*="Download"]',
                '*[aria-label*="Images"]',
            ]

            for selector in download_selectors:
                try:
                    log(f"   Trying selector: {selector}")
                    element = frame.wait_for_selector(selector, timeout=5000)
                    if element and element.is_visible():
                        download_images_item = element
                        log(f"   Found Download as Images with: {selector}")
                        element.click()
                        break
                except Exception as e:
                    log(f"   Failed with {selector}: {str(e)}")

        if not download_images_item:
            raise Exception("Could not find Download as Images option. Check debug screenshot.")

        # Step 4: Wait for processing
        log(f"Waiting {download_timeout} seconds for image processing...")
        time.sleep(download_timeout)

        # Step 5: Click on the Download button
        log("Looking for Download button...")
        download_button_selector = 'button[aria-label="Download"]'
        download_button = frame.wait_for_selector(download_button_selector, timeout=30000)

        # Wait for download to start and complete
        log("Waiting for download to complete...")
        with page.expect_download() as download_info:
            download_button.click()
            log("Clicked Download button")
        download = download_info.value
        # Save to a known remote location and return that absolute path
        return str(download.path())
    except Exception as e:
        log(f"Error during conversion: {str(e)}")
        if verbose:
            try:
                page.screenshot(path=os.path.join(download_dir, "debug_error.png"))
                log("Error screenshot saved")
            except Exception:
                pass
        return None


if __name__ == "__main__":
    # Simple test
    load_dotenv()
    onedrive_client = OneDriveClient(client_id=os.getenv("CLIENT_ID"))
    link = onedrive_client.get_edit_link("PowerPoint/sample.pptx")
    # convert link to images
    # result = convert_powerpoint_to_images_sync(
    #     edit_link=link,
    #     download_dir=".",
    #     headless=True,  # Use visible browser for testing
    #     verbose=True,
    #     download_timeout=30,  # Increase for larger presentations
    # )

    # print("-" * 50)
    # if result:
    #     print(f"SUCCESS! Images downloaded to: {result}")
    # else:
    #     print("FAILED to convert presentation to images")
    #     print(
    #         "Check the debug screenshots in the download directory for troubleshooting"
    #     )

    # # try ensure classic ribbon
    from screenenv import Sandbox

    sandbox = Sandbox(
        headless=False,
        resolution=(1024, 768),
    )
    # Open the presentation
    sandbox.open(link)
    ensure_classic_ribbon_always_show(sandbox, verbose=True)
    input("Press Enter to continue...")
    sandbox.close()
