"""Independent ADB utility functions for React* agent.

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, Collection, Iterable


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 with execute_adb_call method
        timeout_sec: Timeout for the operation
        
    Returns:
        Command output (format depends on env implementation)
    """
    if isinstance(command, str):
        command = command.split(' ')
    
    # Try android_env style first (with AdbRequest)
    try:
        from android_env.proto import adb_pb2
        if hasattr(env, 'execute_adb_call'):
            # 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 env.execute_adb_call(request)
    except (ImportError, AttributeError):
        pass
    
    # Fallback to simple interface
    return env.execute_adb_call(command)


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 android_env style first
    try:
        from android_env.proto import adb_pb2
        if hasattr(env, 'execute_adb_call'):
            request = adb_pb2.AdbRequest(
                tap=adb_pb2.AdbRequest.Tap(x=x, y=y),
                timeout_sec=timeout_sec,
            )
            return env.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 android_env style first
        try:
            from android_env.proto import adb_pb2
            if hasattr(env, 'execute_adb_call'):
                request = adb_pb2.AdbRequest(
                    input_text=adb_pb2.AdbRequest.InputText(text=formatted),
                    timeout_sec=timeout_sec,
                )
                env.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 android_env style first
    try:
        from android_env.proto import adb_pb2
        if hasattr(env, 'execute_adb_call'):
            request = adb_pb2.AdbRequest(
                press_button=adb_pb2.AdbRequest.PressButton(
                    button=adb_pb2.AdbRequest.PressButton.ENTER
                ),
                timeout_sec=timeout_sec,
            )
            return env.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 android_env style first
    try:
        from android_env.proto import adb_pb2
        if hasattr(env, 'execute_adb_call'):
            request = adb_pb2.AdbRequest(
                press_button=adb_pb2.AdbRequest.PressButton(
                    button=adb_pb2.AdbRequest.PressButton.HOME
                ),
                timeout_sec=timeout_sec,
            )
            return env.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 android_env style first
    try:
        from android_env.proto import adb_pb2
        if hasattr(env, 'execute_adb_call'):
            request = adb_pb2.AdbRequest(
                press_button=adb_pb2.AdbRequest.PressButton(
                    button=adb_pb2.AdbRequest.PressButton.BACK
                ),
                timeout_sec=timeout_sec,
            )
            return env.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 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:
            from android_env.proto import adb_pb2
            if hasattr(env, 'execute_adb_call'):
                request = adb_pb2.AdbRequest(
                    start_activity=adb_pb2.AdbRequest.StartActivity(
                        full_activity=activity,
                        extra_args=[],
                    ),
                    timeout_sec=5,
                )
                env.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
    
    # Try to launch by package name using monkey
    command = ['shell', 'monkey', '-p', app_name, '1']
    issue_generic_request(command, env, timeout_sec=5)
    return app_name


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)

