"""Independent data models for React* agent.

This module provides data models without depending on android_world.
"""

from dataclasses import dataclass
from typing import Optional, Tuple, Any


# Action type constants
UNKNOWN = 'unknown'


@dataclass
class JSONAction:
    """Represents an action in JSON format.
    
    This class supports various action types and their parameters.
    """
    action_type: str
    index: Optional[int] = None
    x: Optional[int] = None
    y: Optional[int] = None
    text: Optional[str] = None
    clear_text: bool = False
    direction: Optional[str] = None  # 'up', 'down', 'left', 'right'
    start_x: Optional[int] = None  # For coordinate-based swipe
    start_y: Optional[int] = None  # For coordinate-based swipe
    end_x: Optional[int] = None  # For coordinate-based swipe
    end_y: Optional[int] = None  # For coordinate-based swipe
    app_name: Optional[str] = None
    keycode: Optional[int] = None
    touch_xy: Optional[Tuple[int, int]] = None
    lift_xy: Optional[Tuple[int, int]] = None
    activity_nickname: Optional[str] = None
    orientation: Optional[str] = None
    command: Optional[str] = None  # For shell command
    seconds: Optional[float] = None  # For wait action duration
    
    def __post_init__(self):
        """Validate action parameters after initialization."""
        valid_action_types = [
            'click', 'double_tap', 'long_press', 'input_text',
            'keyboard_enter', 'navigate_home', 'navigate_back',
            'press_keyboard', 'drag_and_drop', 'scroll', 'swipe',
            'open_app', 'wait', 'launch_adb_activity', 'change_orientation',
            'stop', 'answer', 'ask_mllm', 'shell', UNKNOWN
        ]
        
        if self.action_type not in valid_action_types:
            self.action_type = UNKNOWN
    
    def to_dict(self):
        """Convert action to dictionary format."""
        result = {'action_type': self.action_type}
        
        if self.index is not None:
            result['index'] = self.index
        if self.x is not None:
            result['x'] = self.x
        if self.y is not None:
            result['y'] = self.y
        if self.text is not None:
            result['text'] = self.text
        if self.clear_text:
            result['clear_text'] = self.clear_text
        if self.direction is not None:
            result['direction'] = self.direction
        if self.start_x is not None:
            result['start_x'] = self.start_x
        if self.start_y is not None:
            result['start_y'] = self.start_y
        if self.end_x is not None:
            result['end_x'] = self.end_x
        if self.end_y is not None:
            result['end_y'] = self.end_y
        if self.app_name is not None:
            result['app_name'] = self.app_name
        if self.keycode is not None:
            result['keycode'] = self.keycode
        if self.touch_xy is not None:
            result['touch_xy'] = self.touch_xy
        if self.lift_xy is not None:
            result['lift_xy'] = self.lift_xy
        if self.activity_nickname is not None:
            result['activity_nickname'] = self.activity_nickname
        if self.orientation is not None:
            result['orientation'] = self.orientation
        if self.command is not None:
            result['command'] = self.command
        if self.seconds is not None:
            result['seconds'] = self.seconds
            
        return result


class AndroidController:
    """Base interface for Android environment controllers."""
    
    def get_ui_elements(self):
        """Get current UI elements on the screen."""
        raise NotImplementedError("Subclasses must implement get_ui_elements()")
    
    def execute_adb_call(self, command):
        """Execute an ADB command."""
        raise NotImplementedError("Subclasses must implement execute_adb_call()")

