"""Independent ADB utility functions for AutoRPA.

This module provides ADB operations without depending on android_world.
It can work with both android_env interfaces and simple command-based interfaces.
"""

import re
import time
import unicodedata
from typing import Any, List, Optional, Iterable


def _get_adb_interface(env: Any):
    """Get ADB interface from environment.
    
    Tries multiple ways to access ADB:
    1. env.controller.adb (android_world AsyncEnv)
    2. env.controller.execute_adb_call (android_world AsyncEnv alternative)
    3. env.adb (direct adb attribute)
    4. env.execute_adb_call (direct method)
    
    Returns:
        ADB interface object or None if not found
    """
    # Try android_world AsyncEnv style (has controller attribute with adb)
    if hasattr(env, 'controller'):
        controller = env.controller
        if hasattr(controller, 'adb'):
            return controller.adb
        elif hasattr(controller, 'execute_adb_call'):
            return controller
    
    # Try direct adb attribute
    if hasattr(env, 'adb'):
        return env.adb
    
    # Try direct execute_adb_call method
    if hasattr(env, 'execute_adb_call'):
        return env
    
    return None


def issue_generic_request(
    command: List[str] | str,
    env: Any,
    timeout_sec: Optional[float] = 10,
) -> Any:
    """Execute a generic ADB command.
    
    Args:
        command: List of command arguments or a space-separated string
        env: Environment interface (AsyncAndroidEnv, android_env, or object with execute_adb_call/adb)
        timeout_sec: Timeout for the operation
        
    Returns:
        Command output (format depends on env implementation)
    """
    if isinstance(command, str):
        command = command.split(' ')
    
    # Get ADB interface using helper function
    adb_interface = _get_adb_interface(env)
    
    if adb_interface and hasattr(adb_interface, 'execute_adb_call'):
        try:
            from android_env.proto import adb_pb2
            # Check if we need to wrap in AdbRequest
            if not isinstance(command, adb_pb2.AdbRequest):
                request = adb_pb2.AdbRequest(
                    generic=adb_pb2.AdbRequest.GenericRequest(args=command),
                    timeout_sec=timeout_sec,
                )
                return adb_interface.execute_adb_call(request)
            else:
                return adb_interface.execute_adb_call(command)
        except (ImportError, AttributeError):
            # Fallback: try direct call
            return adb_interface.execute_adb_call(command)
    
    # Last resort: raise error
    raise AttributeError(
        f"Environment {type(env)} does not have execute_adb_call method or adb attribute. "
        f"Available attributes: {[attr for attr in dir(env) if not attr.startswith('_')]}"
    )


def tap_screen(x: int, y: int, env: Any, timeout_sec: Optional[float] = 10) -> Any:
    """Tap the screen at specified coordinates.
    
    Args:
        x: X coordinate
        y: Y coordinate
        env: Environment interface
        timeout_sec: Timeout for the operation
        
    Returns:
        Response from the ADB command
    """
    # Try using ADB interface
    adb_interface = _get_adb_interface(env)
    if adb_interface and hasattr(adb_interface, 'execute_adb_call'):
        try:
            from android_env.proto import adb_pb2
            request = adb_pb2.AdbRequest(
                tap=adb_pb2.AdbRequest.Tap(x=x, y=y),
                timeout_sec=timeout_sec,
            )
            return adb_interface.execute_adb_call(request)
        except (ImportError, AttributeError):
            pass
    
    # Fallback to shell command
    command = ['shell', 'input', 'tap', str(x), str(y)]
    return issue_generic_request(command, env, timeout_sec)


def double_tap(x: int, y: int, env: Any) -> None:
    """Double tap the screen at specified coordinates.
    
    Args:
        x: X coordinate
        y: Y coordinate
        env: Environment interface
    """
    tap_screen(x, y, env)
    time.sleep(0.1)
    tap_screen(x, y, env)


def long_press(x: int, y: int, env: Any, duration_ms: int = 1000) -> None:
    """Long press the screen at specified coordinates.
    
    Args:
        x: X coordinate
        y: Y coordinate
        env: Environment interface
        duration_ms: Duration of the press in milliseconds
    """
    command = ['shell', 'input', 'swipe', str(x), str(y), str(x), str(y), str(duration_ms)]
    issue_generic_request(command, env)


def _adb_text_format(text: str) -> str:
    """Prepares text for use with adb (escapes special characters)."""
    to_escape = [
        '\\', ';', '|', '`', '\r', ' ', "'", '"', '&', '<', '>',
        '(', ')', '#', '$',
    ]
    for char in to_escape:
        text = text.replace(char, '\\' + char)
    normalized_text = unicodedata.normalize('NFKD', text)
    return normalized_text.encode('ascii', 'ignore').decode('ascii')


def _split_words_and_newlines(text: str) -> Iterable[str]:
    """Split lines of text into individual words and newline chars."""
    lines = text.split('\n')
    for i, line in enumerate(lines):
        words = line.split(' ')
        for j, word in enumerate(words):
            if word:
                yield word
            if j < len(words) - 1:
                yield '%s'
        if i < len(lines) - 1:
            yield '\n'


def type_text(text: str, env: Any, timeout_sec: int = 10) -> None:
    """Type text on the device word-by-word.
    
    This types word-by-word to prevent long text strings from being typed
    out of order at the character level.
    
    Args:
        text: Text to type
        env: Environment interface
        timeout_sec: Timeout in seconds per word
    """
    words = _split_words_and_newlines(text)
    for word in words:
        if word == '\n':
            press_enter_button(env)
            continue
        formatted = _adb_text_format(word)
        
        # Try using ADB interface
        adb_interface = _get_adb_interface(env)
        if adb_interface and hasattr(adb_interface, 'execute_adb_call'):
            try:
                from android_env.proto import adb_pb2
                request = adb_pb2.AdbRequest(
                    input_text=adb_pb2.AdbRequest.InputText(text=formatted),
                    timeout_sec=timeout_sec,
                )
                adb_interface.execute_adb_call(request)
                continue
            except (ImportError, AttributeError):
                pass
        
        # Fallback to shell command
        command = ['shell', 'input', 'text', formatted]
        issue_generic_request(command, env, timeout_sec)


def press_enter_button(env: Any, timeout_sec: Optional[float] = 10) -> Any:
    """Press the Enter/Return key.
    
    Args:
        env: Environment interface
        timeout_sec: Timeout for the operation
        
    Returns:
        Response from the ADB command
    """
    # Try using ADB interface
    adb_interface = _get_adb_interface(env)
    if adb_interface and hasattr(adb_interface, 'execute_adb_call'):
        try:
            from android_env.proto import adb_pb2
            request = adb_pb2.AdbRequest(
                press_button=adb_pb2.AdbRequest.PressButton(
                    button=adb_pb2.AdbRequest.PressButton.ENTER
                ),
                timeout_sec=timeout_sec,
            )
            return adb_interface.execute_adb_call(request)
        except (ImportError, AttributeError):
            pass
    
    # Fallback to shell command
    command = ['shell', 'input', 'keyevent', '66']  # KEYCODE_ENTER
    return issue_generic_request(command, env, timeout_sec)


def press_home_button(env: Any, timeout_sec: Optional[float] = 10) -> Any:
    """Press the Home button.
    
    Args:
        env: Environment interface
        timeout_sec: Timeout for the operation
        
    Returns:
        Response from the ADB command
    """
    # Try using ADB interface
    adb_interface = _get_adb_interface(env)
    if adb_interface and hasattr(adb_interface, 'execute_adb_call'):
        try:
            from android_env.proto import adb_pb2
            request = adb_pb2.AdbRequest(
                press_button=adb_pb2.AdbRequest.PressButton(
                    button=adb_pb2.AdbRequest.PressButton.HOME
                ),
                timeout_sec=timeout_sec,
            )
            return adb_interface.execute_adb_call(request)
        except (ImportError, AttributeError):
            pass
    
    # Fallback to shell command
    command = ['shell', 'input', 'keyevent', '3']  # KEYCODE_HOME
    return issue_generic_request(command, env, timeout_sec)


def press_back_button(env: Any, timeout_sec: Optional[float] = 10) -> Any:
    """Press the Back button.
    
    Args:
        env: Environment interface
        timeout_sec: Timeout for the operation
        
    Returns:
        Response from the ADB command
    """
    # Try using ADB interface
    adb_interface = _get_adb_interface(env)
    if adb_interface and hasattr(adb_interface, 'execute_adb_call'):
        try:
            from android_env.proto import adb_pb2
            request = adb_pb2.AdbRequest(
                press_button=adb_pb2.AdbRequest.PressButton(
                    button=adb_pb2.AdbRequest.PressButton.BACK
                ),
                timeout_sec=timeout_sec,
            )
            return adb_interface.execute_adb_call(request)
        except (ImportError, AttributeError):
            pass
    
    # Fallback to shell command
    command = ['shell', 'input', 'keyevent', '4']  # KEYCODE_BACK
    return issue_generic_request(command, env, timeout_sec)


def press_keyboard_generic(keycode: int, env: Any) -> None:
    """Press a generic keyboard key by keycode.
    
    Args:
        keycode: Android keycode number
        env: Environment interface
    """
    command = ['shell', 'input', 'keyevent', str(keycode)]
    issue_generic_request(command, env)


def generate_swipe_command(
    start_x: int,
    start_y: int,
    end_x: int,
    end_y: int,
    duration_ms: int = 300
) -> List[str]:
    """Generate a swipe command.
    
    Args:
        start_x: Starting X coordinate
        start_y: Starting Y coordinate
        end_x: Ending X coordinate
        end_y: Ending Y coordinate
        duration_ms: Duration of swipe in milliseconds
        
    Returns:
        Command as list of strings
    """
    return [
        'shell', 'input', 'swipe',
        str(start_x), str(start_y),
        str(end_x), str(end_y),
        str(duration_ms)
    ]


def generate_drag_and_drop_command(
    start_x: int,
    start_y: int,
    end_x: int,
    end_y: int,
    duration_ms: int = 1000
) -> List[str]:
    """Generate a drag and drop command.
    
    Args:
        start_x: Starting X coordinate
        start_y: Starting Y coordinate
        end_x: Ending X coordinate
        end_y: Ending Y coordinate
        duration_ms: Duration of drag in milliseconds
        
    Returns:
        Command as list of strings
    """
    return generate_swipe_command(start_x, start_y, end_x, end_y, duration_ms)


def get_adb_activity(app_name: str) -> Optional[str]:
    """Get the ADB activity for common app names.
    
    Maps app names to their activities. Returns None if not found.
    
    Args:
        app_name: The app name to look up
        
    Returns:
        The activity string or None
    """
    # Common app mappings (subset from android_world)
    # You can add more app mappings here to avoid monkey command issues with spaces
    pattern_to_activity = {
        # System apps
        'google chrome|chrome': 'com.android.chrome/com.google.android.apps.chrome.Main',
        'settings|system settings': 'com.android.settings/.Settings',
        'camera': 'com.android.camera2/com.android.camera.CameraLauncher',
        'gmail': 'com.google.android.gm/.ConversationListActivityGmail',
        'google maps|maps': 'com.google.android.apps.maps/com.google.android.maps.MapsActivity',
        'google photos|photos': 'com.google.android.apps.photos/com.google.android.apps.photos.home.HomeActivity',
        'messages': 'com.google.android.apps.messaging/com.google.android.apps.messaging.ui.ConversationListActivity',
        'contacts': 'com.google.android.contacts/com.android.contacts.activities.PeopleActivity',
        'files': 'com.google.android.documentsui/com.android.documentsui.files.FilesActivity',
        'clock': 'com.google.android.deskclock/com.android.deskclock.DeskClock',
        
        # Simple Mobile Tools Suite (all have spaces in display names!)
        'simple gallery pro|gallery pro': 'com.simplemobiletools.gallery.pro/.activities.SplashActivity',
        'simple gallery': 'com.simplemobiletools.gallery.pro/.activities.SplashActivity',
        'simple calendar pro|calendar pro|simple calendar': 'com.simplemobiletools.calendar.pro/.activities.SplashActivity',
        'simple sms messenger|simple sms|sms messenger': 'com.simplemobiletools.smsmessenger/.activities.SplashActivity',
        'simple contacts pro|simple contacts': 'com.simplemobiletools.contacts.pro/.activities.SplashActivity',
        'simple notes pro|simple notes': 'com.simplemobiletools.notes.pro/.activities.SplashActivity',
        
        # Third-party apps commonly used in AndroidWorld (synced with official android_world)
        'audio recorder': 'com.dimowner.audiorecorder/com.dimowner.audiorecorder.app.welcome.WelcomeActivity',
        'markor': 'net.gsantner.markor/net.gsantner.markor.activity.MainActivity',
        'tasks|tasks app|tasks\\.org:': 'org.tasks/com.todoroo.astrid.activity.MainActivity',
        'osmand': 'net.osmand/net.osmand.plus.activities.MapActivity',
        'retro music|retro|retro player': 'code.name.monkey.retromusic/.activities.MainActivity',
        'pro expense|pro expense app': 'com.arduia.expense/com.arduia.expense.ui.MainActivity',
        'broccoli|broccoli app|broccoli recipe app|recipe app': 'com.flauschcode.broccoli/com.flauschcode.broccoli.MainActivity',
        'open tracks sports tracker|activity tracker|open tracks|opentracks': 'de.dennisguse.opentracks/de.dennisguse.opentracks.TrackListActivity',
        'vlc|vlc app|vlc player': 'org.videolan.vlc/.gui.MainActivity',
        'joplin|joplin app': 'net.cozic.joplin/.MainActivity',
    }
    
    for pattern, activity in pattern_to_activity.items():
        if re.match(pattern.lower(), app_name.lower()):
            return activity
    
    return None


def _get_package_name_from_display_name(app_name: str, env: Any) -> Optional[str]:
    """Try to find the package name for an app by its display name.
    
    This function uses multiple strategies to find the package name:
    1. Query launcher activities and match app labels
    2. Match package name patterns against the app name
    
    Args:
        app_name: Display name of the app (e.g., "Simple Gallery Pro")
        env: Environment interface
        
    Returns:
        Package name if found, None otherwise
    """
    try:
        # Strategy 1: Use pm list packages and match name parts in package
        command = ['shell', 'pm', 'list', 'packages', '-3']  # -3 for third-party apps only
        result = issue_generic_request(command, env, timeout_sec=10)
        
        # Parse result to extract package names
        result_str = str(result)
        if 'output:' in result_str:
            match = re.search(r'output:\s*"([^"]*)"', result_str)
            if match:
                result_str = match.group(1)
        
        # Clean up escape sequences
        result_str = result_str.replace('\\n', '\n')
        
        # Extract all packages
        packages = []
        for line in result_str.split('\n'):
            if line.startswith('package:'):
                package = line.replace('package:', '').strip()
                if package:
                    packages.append(package)
        
        # Try to match app name with package names
        # Strategy: check if all words from app name appear in package name
        app_words = app_name.lower().replace(' ', '').replace('-', '').replace('_', '')
        
        best_match = None
        best_score = 0
        
        for package in packages:
            package_lower = package.lower()
            
            # Count how many characters from app name appear in package name (in order)
            score = 0
            pkg_idx = 0
            for char in app_words:
                try:
                    pkg_idx = package_lower.index(char, pkg_idx) + 1
                    score += 1
                except ValueError:
                    break
            
            # If we matched a significant portion, consider it
            if score > best_score and score >= len(app_words) * 0.6:  # At least 60% match
                best_score = score
                best_match = package
        
        return best_match
    except Exception as e:
        # Log error but don't crash
        print(f"Warning: Failed to find package name for '{app_name}': {e}")
        return None


def launch_app(app_name: str, env: Any) -> Optional[str]:
    """Launch an application by name, package name, or activity.
    
    Args:
        app_name: App name, package name, or activity name
        env: Environment interface
        
    Returns:
        The app name that was launched, or None if failed
    """
    # Try to get activity from common app names
    activity = get_adb_activity(app_name)
    
    if activity:
        # Launch using activity
        # Try using ADB interface
        adb_interface = _get_adb_interface(env)
        if adb_interface and hasattr(adb_interface, 'execute_adb_call'):
            try:
                from android_env.proto import adb_pb2
                request = adb_pb2.AdbRequest(
                    start_activity=adb_pb2.AdbRequest.StartActivity(
                        full_activity=activity,
                        extra_args=[],
                    ),
                    timeout_sec=5,
                )
                adb_interface.execute_adb_call(request)
                return app_name
            except (ImportError, AttributeError):
                pass
        
        # Fallback: use am start
        command = ['shell', 'am', 'start', '-n', activity]
        issue_generic_request(command, env, timeout_sec=5)
        return app_name
    
    # If it's a full activity name (contains /)
    if '/' in app_name:
        command = ['shell', 'am', 'start', '-n', app_name]
        issue_generic_request(command, env, timeout_sec=5)
        return app_name
    
    # Check if app_name contains spaces - if so, try to find package name
    if ' ' in app_name:
        package_name = _get_package_name_from_display_name(app_name, env)
        if package_name:
            # Use the found package name with monkey
            try:
                command = ['shell', 'monkey', '-p', package_name, '1']
                issue_generic_request(command, env, timeout_sec=5)
                return app_name
            except Exception:
                # If monkey fails, try using am start with launcher intent
                try:
                    command = ['shell', 'am', 'start', '-a', 'android.intent.action.MAIN', 
                               '-c', 'android.intent.category.LAUNCHER', package_name]
                    issue_generic_request(command, env, timeout_sec=5)
                    return app_name
                except Exception as e:
                    raise ValueError(
                        f"Failed to launch app '{app_name}' with package name '{package_name}': {e}"
                    )
        else:
            # If we can't find the package, provide helpful error
            raise ValueError(
                f"Cannot find package name for '{app_name}'. "
                f"Please use the exact package name (e.g., 'com.simplemobiletools.gallery.pro') "
                f"instead of the display name. You can find package names by running: "
                f"'adb shell pm list packages -3'"
            )
    
    # Try to launch by package name using monkey (for package names without spaces)
    try:
        command = ['shell', 'monkey', '-p', app_name, '1']
        issue_generic_request(command, env, timeout_sec=5)
        return app_name
    except Exception:
        # If monkey fails, try using am start with launcher intent as fallback
        try:
            command = ['shell', 'am', 'start', '-a', 'android.intent.action.MAIN', 
                       '-c', 'android.intent.category.LAUNCHER', app_name]
            issue_generic_request(command, env, timeout_sec=5)
            return app_name
        except Exception as e:
            raise ValueError(
                f"Failed to launch app '{app_name}'. "
                f"Please ensure the package name is correct or use the full activity name "
                f"in format 'package.name/package.name.ActivityName'"
            ) from e


def change_orientation(orientation: str, env: Any) -> None:
    """Change device orientation.
    
    Args:
        orientation: 'portrait' or 'landscape'
        env: Environment interface
    """
    if orientation.lower() == 'portrait':
        # Disable auto-rotate and set to portrait
        issue_generic_request(['shell', 'settings', 'put', 'system', 'accelerometer_rotation', '0'], env)
        issue_generic_request(['shell', 'settings', 'put', 'system', 'user_rotation', '0'], env)
    elif orientation.lower() == 'landscape':
        # Disable auto-rotate and set to landscape
        issue_generic_request(['shell', 'settings', 'put', 'system', 'accelerometer_rotation', '0'], env)
        issue_generic_request(['shell', 'settings', 'put', 'system', 'user_rotation', '1'], env)

