from typing import Optional

from absl.flags import FLAGS

from ..prompts.general_prompt import FUNCTION_ACTIONS_PROMPT, OPERATION_GUIDANCE_PROMPT, INDEX_REPLACEMENT_PROMPT

PROMPT_HEADER = """[system]
You are an excellent web agent in a simulated environment. Given a web page and a task, you write step-by-step code to accomplish the task such as click some buttons and send an email.

[user]
"""

OUTPUT_FORMAT_TEMPLATE_base = """
[Output Format]
Your output must consist of following things:
- Observations: Summarize key input details and immediate screen observations. Include all obvious insights.
{reflection_block}
- Completed Tasks: List completed tasks, each starting with a ✅.
- Plan Justification: Briefly explain the rationale behind your plan.
- Plan List: List tasks to achieve the goal, each starting with a ⬜; if the goal is achieved, output "goal completed."
- Code Reason: Explain why the next action is chosen.
"""

OUTPUT_FORMAT_TEMPLATE_code = OUTPUT_FORMAT_TEMPLATE_base + """
- Code: Output the next action code (one action, one line). Use only [ADMISSIBLE ACTIONS].
"""

# different
OUTPUT_FORMAT_TEMPLATE_code_soft = OUTPUT_FORMAT_TEMPLATE_base + """
- Code: Output the next action code (one action, one line). Use only [ADMISSIBLE ACTIONS] and DO NOT use `env_op.find_element()`.
- Soft-coded Action: Output the code with a robust xpath according to [XPath Replacement] (two lines: xpath= and Action). Always include this section.
"""

WORKFLOW_COMMON = ("""
[Your Workflow]
1. Analyze Input: Carefully examine all input. Extract all goal-relevant info — don't miss anything useful. Directly infer and record any obvious conclusions.
2. Evaluate Progress: Check if the goal is achieved; if so, stop. Otherwise, update completed tasks.
3. Devise Plan: Break the goal into efficient, non-redundant steps. Identify and obtain any missing information.
4. Execute & Adjust: Analyze the UI info to decide actions. Adjust the plan if elements are missing or inaccessible.
5. Error Handling: Retry once on failure; if it still fails, choose an alternative.
6. Generate Next Action: Choose the next logical action that advances the goal.
""")

WORKFLOW_COMMON_soft = WORKFLOW_COMMON + '\n7. Generate Soft-coded Action: Use dynamic element matching (via `env_op.find_element`) to replace hardcoded xpath.\n'

WORKFLOW_REFLECTION = """
[Your Workflow]
1. Refer to the action history to assess whether the current progress aligns with the operations described in the last reflection. If relevant, check the accuracy of the reflection:
   - If accurate, consider how to avoid previous mistakes;
   - If inaccurate, disregard misleading content;
   If irrelevant, skip the reflection.
2. Analyze Input: Carefully examine all input. Extract all goal-relevant info — don't miss anything useful. Directly infer and record any obvious conclusions.
3. Evaluate Progress: Check if the goal is achieved; if so, stop. Otherwise, update completed tasks.
4. Devise Plan: Break the goal into efficient, non-redundant steps. Identify and obtain any missing information.
5. Execute & Adjust: Analyze the UI info to decide actions. Adjust the plan if elements are missing or inaccessible.
6. Error Handling: Retry once on failure; if it still fails, choose an alternative.
7. Generate Next Action: Choose the next logical action that advances the goal.
"""

WORKFLOW_REFLECTION_soft = WORKFLOW_REFLECTION + '\n8. Generate Soft-coded Action: Use dynamic element matching (via `env_op.find_element`) to replace hardcoded xpath.\n'

WORK_GUIDELINES = """
[Guidelines]
- After you output the action, the action will be executed. The results of each action and the new observations will be printed to you at next step.
- Maintain a holistic view by identifying the specific steps required to complete the task using the current input.
- Fully leverage provided input to reduce unnecessary follow-up actions.
- Interact only with verified UI elements.
- Adapt dynamically – adjust to screen changes and execution failures.
- Stay focused on the user goal – ignore distracting messages or popups unrelated to the actual task.
{additional_guidelines}
"""

PLANNER_EXAMPLE_soft = """
{{{{
"observation": "The current screen displays an HTML page with various elements, including a button labeled \"Ok.\" The button appears clickable based on its styling and position, as seen in the screenshot.",
{reflection_example}
"completed_tasks": "...",
"plan_reason": "The task is straightforward: click the \"Ok\" button. The button is clearly visible and accessible.",
"plan_list": "⬜ Click the \"Ok\" button.\n⬜ Task complete.",
"code_reason": "Clicking the \"Ok\" button directly satisfies the user's request and is the simplest action required to complete the task.",
"code": "env_op.click_xpath(\"//*[name()='button' and normalize-space(text())='Ok']\")",
"soft_action": "xpath = env_op.find_element(\"//*[name()='button' and text()='Ok']\", \"Button labeled 'Ok' for confirming actions\")\nenv_op.click_xpath(xpath)"
}}}}
"""

PLANNER_EXAMPLE = """
{{{{
"observation": "The current screen displays an HTML page with various elements, including a button labeled \"Ok.\" The button appears clickable based on its styling and position, as seen in the screenshot.",
{reflection_example}
"completed_tasks": "...",
"plan_reason": "The task is straightforward: click the \"Ok\" button. The button is clearly visible and accessible.",
"plan_list": "⬜ Click the \"Ok\" button.\n⬜ Task complete.",
"code_reason": "Clicking the \"Ok\" button directly satisfies the user's request and is the simplest action required to complete the task.",
"code": "env_op.click_xpath(\"//*[name()='button' and normalize-space(text())='Ok']\")"
}}}}
"""

INPUT_TEMPLATE = ("""
# User goal/request: {goal}
{reflection}

[Input Provided]
- Action History & Execution Result:
{action_history}

- Main changes on the page after the previous action is executed:
{screen_changes}

- Completed Tasks:
{completed_plan}

- Previous Plan:
{previous_plan}

- Current screenshot, annotated screenshot, and UI elements list:
{ui_content}
""")


def get_prompt_template(reflection: Optional[str] = None) -> str:
  workflow_common = WORKFLOW_COMMON_soft + INDEX_REPLACEMENT_PROMPT
  workflow_reflection = WORKFLOW_REFLECTION_soft
  action_space = FUNCTION_ACTIONS_PROMPT
  output_template = OUTPUT_FORMAT_TEMPLATE_code_soft
  planner_example = PLANNER_EXAMPLE_soft
  if FLAGS.use_action_translator:
    workflow_common = WORKFLOW_COMMON
    workflow_reflection = WORKFLOW_REFLECTION
    output_template = OUTPUT_FORMAT_TEMPLATE_code
    planner_example = PLANNER_EXAMPLE
  
  if reflection:
    reflection_block = "- Reflection Consideration: Explain how the previous reflection influences the next action."
    workflow = workflow_reflection
    reflection_example = '"consider_reflection": "...",'
  else:
    reflection_block = ""
    workflow = workflow_common
    reflection_example = ""
  
  output_template = output_template.format(reflection_block=reflection_block)
  planner_example = planner_example.format(reflection_example=reflection_example)
  
  return PROMPT_HEADER + action_space + OPERATION_GUIDANCE_PROMPT + output_template + workflow + WORK_GUIDELINES + planner_example + INPUT_TEMPLATE


def get_planner_prompt(
  previous_plan: Optional[str],
  completed_plan: Optional[str],
  goal: str,
  action_history: list,
  reflection: Optional[str] = None,
  ui_content: str = "",
  screen_changes: str = '',
  additional_guidelines: Optional[str] = '',
) -> str:
  if action_history:
    action_history = '\n'.join(action_history)
  else:
    action_history = 'You just started, no action has been performed yet.'
  
  return get_prompt_template(reflection).format(
    goal=goal,
    previous_plan=previous_plan,
    completed_plan=completed_plan,
    action_history=action_history,
    ui_content=ui_content if ui_content else 'Not available',
    screen_changes=screen_changes,
    reflection='\n## The following is the reflection of the previous fail attempt:\n' + reflection if reflection else '',
    additional_guidelines=additional_guidelines,
  )