"""
General Prompts for AutoRPA Building Phase

This module contains:
- OPERATION_GUIDANCE_PROMPT: Shared guidance for all agents
- AutoRPA-specific prompts: FIND_ELEMENT_PROMPT, INDEX_REPLACEMENT_PROMPT, etc.
- get_builder_action_prompt(): Function to build action prompts for any GUI agent
"""

OPERATION_GUIDANCE_PROMPT = """
[OPERATION_GUIDANCE]
Follow these guidelines:
General:
- Choose the simplest method.
- Navigate to gather necessary info (e.g., open the calendar for schedule queries) before answering.
- For file-related operations, make sure to operate only on the exact target file; do not interact with similar files. You must locate and use the precise file required.
- Remember to save the file after making any changes to its contents.
- When naming files, ensure proper file extensions.

Action Related:
- Ensure the index for click, long_press, and input_text is visible in both the screenshot and UI list.
- Use swipe('up') or swipe('down') to reveal elements not currently visible. When reached bottom, use swipe('down') to back to the top part of the screen.
- Confirm that a UI element supports the intended action before interacting.

Text Operations:
- Use standard text selection methods (long-press and selection bar) when needed.
"""

# AutoRPA-specific prompts for dynamic element finding and enhanced actions

FIND_ELEMENT_PROMPT = """
### Get Element Index
env_op.find_element(**kwargs) -> int # Use this function to find an element in the UI list using filtering criteria and return its index. If no matching element is found, the index will be -1.
- Rules for generating `kwargs`:
  - Use find_element only when an index is required (e.g., not for env_op.open_app).
  - Extract all attributes necessary to uniquely identify the element in the current page. **All field values (except target_description) must be extracted EXACTLY from the provided UI elements list—do not fabricate, infer, or modify.** Values must match exactly (case-sensitive). Omit fields not present in the UI element. For actions, include only the necessary action(s), not the full list.
  - From text, hint_text, content_description, and tooltip, extract only one—whichever is most relevant and present in the UI list. This does not limit the use of other attributes.
  - Always include target_description: briefly describe the element's role, appearance, position on screen, and any dynamic/contextual content that helps make it unique.
- `kwargs` Examples:
  - kwargs = {{"content_description": "Save Changes", "target_description": "rectangle button to save"}}
- note:
  Prefer using the find_element function to match the corresponding element.
"""

GET_UI_CONTENT_PROMPT = ("""
### Get Current UI Elements List
env_op.get_ui_content() -> list[dict] # Returns: List[Dict] – Each dictionary contains properties of a UI element on the current page.
- Return format: List of dictionaries, each dict may contain:
  - `index` (int): Element index (required)
  - `text` (str, optional): Displayed text
  - `content_description` (str, optional): Accessibility description
  - `hint_text` (str, optional): Hint text
  - `tooltip` (str, optional): Tooltip text
  - `actions` (list[str], optional): Supported actions like ['click', 'long_press', 'swipe', 'input_text']
  - `is_selected` (bool, optional): Whether the element is selected
  - `is_checked` (bool, optional): Whether a checkable element is checked
  - Other UI element state flags as available
- Example return:
  [{{'index': 0, 'actions': ['swipe']}}, {{'index': 1, 'content_description': 'Home'}}, {{'index': 2, 'text': 'Phone', 'content_description': 'Phone', 'actions': ['long_press']}}]
- Important: After executing any action (click, input_text, swipe, etc.), the UI content changes. Call `env_op.get_ui_content()` again if you need updated element information.
""")

ASK_MLLM_PROMPT = ("""
### Ask MLLM to answer one question
env_op.ask_mllm(question: str) -> str # Will return the answer in plain text.
# For complex screen understanding, you can ask mllm to return a json object, and then use json.loads(answer) to get targets.
# Example:
# answer = env_op.ask_mllm(question="...Return a json object with the name of the reviews as keys and the number of stars as values.")
# json.loads(answer) # {{'name1': '...', 'name2': '...'}}
""")

ELEMENT_FINDING_STRATEGY = """
### Element Finding Priority
When locating UI elements, follow this priority order:
1. **env_op.find_element(**kwargs)** - Preferred method for direct element matching by attributes.
2. **env_op.get_ui_content()** - Use as fallback to manually parse the UI list when the above methods are insufficient.
3. **env_op.ask_mllm(question)** - Use when you need visual/language understanding to identify elements (e.g., complex layouts, contextual reasoning).

### Element Finding in Loops
When searching for elements in a loop:
- Before each iteration, perform swipe/scroll actions to change the page state and reveal new content.
- After each action (click, swipe, etc.), call `env_op.get_ui_content()` again to get updated UI information.
- Use swipe('up') or swipe('down') to explore content not visible in the current view before re-attempting find_element.
- Allow consecutive swipe('up') to search, counting only effective swipe-ups that cause a visible UI change; after reaching the bottom, swipe('down') step by step and keep searching during rollback.
"""


def get_action_space_by_agent(agent_name: str) -> str:
  """
  Get action space prompt for a specific GUI agent.
  
  Args:
    agent_name: Name of the agent (e.g., 'react_star', 'droidrun', 'askui')
  
  Returns:
    Action space prompt string
  """
  agent_name_lower = agent_name.lower()
  
  try:
    if agent_name_lower == 'react_star' or agent_name_lower == 'react*':
      from gui_agents.react_star.prompts.action_space import FUNCTION_ACTIONS_PROMPT
      return FUNCTION_ACTIONS_PROMPT
    elif agent_name_lower == 'droidrun':
      from gui_agents.droidrun.prompts.action_space import FUNCTION_ACTIONS_PROMPT
      return FUNCTION_ACTIONS_PROMPT
    elif agent_name_lower == 'askui':
      from gui_agents.askui.prompts.action_space import FUNCTION_ACTIONS_PROMPT
      return FUNCTION_ACTIONS_PROMPT
    else:
      # Fallback to ReAct* actions for unknown agents
      print(f"Warning: Unknown agent '{agent_name}', using ReAct* actions as fallback")
      from gui_agents.react_star.prompts.action_space import FUNCTION_ACTIONS_PROMPT
      return FUNCTION_ACTIONS_PROMPT
  except ImportError as e:
    print(f"Warning: Failed to import action_space for agent '{agent_name}': {e}")
    print("Using ReAct* actions as fallback")
    from gui_agents.react_star.prompts.action_space import FUNCTION_ACTIONS_PROMPT
    return FUNCTION_ACTIONS_PROMPT


def get_builder_action_prompt(action_space_prompt: str = None, agent_name: str = None) -> str:
  """
  Build action prompt for AutoRPA builder.
  
  This function accepts action space from any GUI agent and combines it with
  AutoRPA-specific features (find_element, get_ui_content, ask_mllm).
  
  Args:
    action_space_prompt: Action space prompt from GUI agent's action_space.py
                        If None, will use agent_name to determine action space
    agent_name: Name of the agent (used if action_space_prompt is None)
                If both are None, uses default ReAct* actions for backward compatibility
  
  Returns:
    Combined prompt with actions + AutoRPA features
  """
  if action_space_prompt is None:
    if agent_name is not None:
      # Get action space by agent name
      action_space_prompt = get_action_space_by_agent(agent_name)
    else:
      # Default: Use ReAct* actions for backward compatibility
      from gui_agents.react_star.prompts.action_space import FUNCTION_ACTIONS_PROMPT
      action_space_prompt = FUNCTION_ACTIONS_PROMPT
  
  return (
    action_space_prompt
    + FIND_ELEMENT_PROMPT
    + GET_UI_CONTENT_PROMPT
    + ASK_MLLM_PROMPT
    + ELEMENT_FINDING_STRATEGY
  )


# Backward compatibility: Keep old name as alias
def FUNCTION_ACTIONS_FIND_ELEMENT_PROMPT(action_space_prompt: str = None) -> str:
  """[DEPRECATED] Use get_builder_action_prompt() instead."""
  return get_builder_action_prompt(action_space_prompt)