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]
Please respond using English typography conventions only. 
Your output must consist of following things:
- Summarize and Analyze: **Because future workers and builders cannot see the original observation, your summary is their only window into the page state.** Structure your description naturally in three parts:
  (1) **Action result** - what happened after the last action (e.g., "Clicking 'Forums' successfully opened the forums directory page", or "Starting from homepage" if first step);
  (2) **Page layout** - concisely describe where you are and the overall page structure, just the page type and main sections (e.g., "The page shows heading 'List of forums' with navigation tabs and sorting controls");
  (3) **Critical details and analysis** - provide rich details ONLY for task-relevant UI elements, including their types, states (hasPopup, expanded, selected, focused, etc.), ids, and parent-child relationships. For example: "The page shows button 'Sort by: Hot' (id 3242) hasPopup: menu expanded: True, which displays a dropdown menu containing link '* Hot' (selected, id 3782), link 'New' (id 3785)...". Do NOT describe every element - only those relevant to deciding the next action.
{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 the replaced index according to [Index Replacement] (three lines: kwargs=, index=, 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 index.\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 index.\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.
{additional_guidelines}
"""


PLANNER_EXAMPLE_soft = """
[Sample Output]
User goal/request is: Tell me the count of comments that have received more downvotes than upvotes for the user who made the latest post on the Showerthoughts forum.
----- Step 1
OBSERVING...
{{{{
"summarize_and_analyze": "**Action result**: Starting from Postmill (Reddit-like) homepage (no prior action). **Page layout**: The page displays the standard Reddit-like header with navigation links and a main content area. **Critical details and analysis**: I can see link 'Forums' (id 66) in the header navigation. To find the latest post on the Showerthoughts forum as required by the task, I need to first navigate to the forums directory by clicking the Forums link.",
{reflection_example}
"completed_tasks": None,
"plan_reason": "To find the latest post on the Showerthoughts forum, I must first navigate to the forum. The 'Forums' link is the entry point for this.",
"plan_list": "⬜ Click the 'Forums' link to locate the Showerthoughts forum.\\n⬜ Find the latest post there.\\n⬜ Identify the user who made the latest post to view his profile.\\n⬜ Count his comments as required.",
"code_reason": "Clicking the 'Forums' link is the first logical step to find the list of forums.",
"code": "env_op.click(66)",
"soft_action": "kwargs = {{{{\"type\": \"link\", \"text\": \"Forums\", \"target_description\": \"the link to the forums page\"}}}}\nindex = env_op.find_element(**kwargs)\nenv_op.click(index)"
}}}}

----- Step 2
OBSERVING...
{{{{
"summarize_and_analyze": "**Action result**: Clicking 'Forums' successfully opened the forums directory page. **Page layout**: The page shows heading 'List of forums' with navigation tabs and sorting controls. **Critical details and analysis**: Several forum entries are visible (AskReddit, relationship_advice, worldnews, news, movies, memes), but Showerthoughts is not in the current list. The scroll bar indicates 65% of content remains below. I can see link 'Alphabetical' (id 749) which will show a complete alphabetical listing of all forums.",
{reflection_example}
"completed_tasks": "✅ Clicked the 'Forums' link.",
"plan_reason": "Since the target forum isn't visible, the next logical action is to view all forums alphabetically to find it.",
"plan_list": "⬜ Click the 'Alphabetical' link to see all forums.\\n⬜ Find the Showerthoughts forum.\\n⬜ Find the latest post there.\\n⬜ Identify the user who made the latest post to view his profile.\\n⬜ Count his comments as required.",
"code_reason": "Clicking the 'Alphabetical' link will show all forums and help locate the 'Showerthoughts' forum.",
"code": "env_op.click(749)",
"soft_action": "kwargs = {{{{\"type\": \"link\", \"text\": \"Alphabetical\", \"target_description\": \"the link to see all forums alphabetically\"}}}}\nindex = env_op.find_element(**kwargs)\nenv_op.click(index)"
}}}}

----- Step 4
OBSERVING...
{{{{
"summarize_and_analyze": "**Action result**: Successfully entered the Showerthoughts forum page. **Page layout**: The page shows heading '/f/Showerthoughts' with navigation tabs and a list of posts. **Critical details and analysis**: The page displays button 'Sort by: Hot' (id 1767) hasPopup: menu, indicating the current sort order. The visible posts show dates from 2022-2023 (e.g., 'October 31, 2022', 'March 31, 2023'), confirming Hot sort is active rather than chronological order. To find the latest post, I need to change the sort order to 'New'.",
{reflection_example}
"completed_tasks": "✅ Clicked the 'Forums' link.\\n✅ Clicked the 'Alphabetical' link.\\n✅ Entered the 'Showerthoughts' forum.",
"plan_reason": "The task requires finding the latest post, so I need to change the sorting from 'Hot' to 'New'.",
"plan_list": "⬜ Click the button 'Sort by: Hot'.\\n⬜ Sort the posts by the newest.\\n⬜ Find the user who made the latest post to view his profile.\\n⬜ Count his comments as required.",
"code_reason": "Clicking the 'Sort by: Hot' button is the way to change the sorting order.",
"code": "env_op.click(1767)",
"soft_action": "kwargs = {{{{\"type\": \"button\", \"text\": \"Sort by: Hot\", \"target_description\": \"the button to change post sorting\"}}}}\nindex = env_op.find_element(**kwargs)\nenv_op.click(index)"
}}}}

----- Step 7
OBSERVING...
{{{{
"summarize_and_analyze": "**Action result**: Successfully opened the user profile page for 'nirinaron'. **Page layout**: The page shows heading 'nirinaron' with navigation tabs 'Overview', 'Submissions', and 'Comments', plus user information. **Critical details and analysis**: The visible content shows one submission article with link 'No comments' displayed, and scanning the page reveals no comment entries. This indicates the user has not made any comments, so the count of comments with more downvotes than upvotes is 0.",
{reflection_example}
"completed_tasks": "✅ Clicked the 'Forums' link.\\n✅ Clicked the 'Alphabetical' link to see all forums.\\n✅ Clicked the 'Showerthoughts' link.\\n✅ Sorted posts by 'New'.\\n✅ Navigated to the latest post author's profile.",
"plan_reason": "The task is to count comments with more downvotes than upvotes, and I've found there are no comments. The task is complete.",
"plan_list": "⬜ No more actions. Complete the task now.",
"code_reason": "The answer has been found, so I will stop and provide the answer.",
"code": "env_op.stop('0')",
"soft_action": "env_op.stop('0')"
}}}}
"""

PLANNER_EXAMPLE = """
[Sample Output]
User goal/request is: Tell me the count of comments that have received more downvotes than upvotes for the user who made the latest post on the Showerthoughts forum.
----- Step 1
OBSERVING...
{{{{
"summarize_and_analyze": "**Action result**: Starting from Postmill (Reddit-like) homepage (no prior action). **Page layout**: The page displays the standard Reddit-like header with navigation links and a main content area. **Critical details and analysis**: I can see link 'Forums' (id 66) in the header navigation. To find the latest post on the Showerthoughts forum as required by the task, I need to first navigate to the forums directory by clicking the Forums link.",
{reflection_example}
"completed_tasks": None,
"plan_reason": "To find the latest post on the Showerthoughts forum, I must first navigate to the forum. The 'Forums' link is the entry point for this.",
"plan_list": "⬜ Click the 'Forums' link to locate the Showerthoughts forum.\\n⬜ Find the latest post there.\\n⬜ Identify the user who made the latest post to view his profile.\\n⬜ Count his comments as required.",
"code_reason": "Clicking the 'Forums' link is the first logical step to find the list of forums.",
"code": "env_op.click(66)"
}}}}

----- Step 2
OBSERVING...
{{{{
"summarize_and_analyze": "**Action result**: Clicking 'Forums' successfully opened the forums directory page. **Page layout**: The page shows heading 'List of forums' with navigation tabs and sorting controls. **Critical details and analysis**: Several forum entries are visible (AskReddit, relationship_advice, worldnews, news, movies, memes), but Showerthoughts is not in the current list. The scroll bar indicates 65% of content remains below. I can see link 'Alphabetical' (id 749) which will show a complete alphabetical listing of all forums.",
{reflection_example}
"completed_tasks": "✅ Clicked the 'Forums' link.",
"plan_reason": "Since the target forum isn't visible, the next logical action is to view all forums alphabetically to find it.",
"plan_list": "⬜ Click the 'Alphabetical' link to see all forums.\\n⬜ Find the Showerthoughts forum.\\n⬜ Find the latest post there.\\n⬜ Identify the user who made the latest post to view his profile.\\n⬜ Count his comments as required.",
"code_reason": "Clicking the 'Alphabetical' link will show all forums and help locate the 'Showerthoughts' forum.",
"code": "env_op.click(749)"
}}}}

----- Step 4
OBSERVING...
{{{{
"summarize_and_analyze": "**Action result**: Successfully entered the Showerthoughts forum page. **Page layout**: The page shows heading '/f/Showerthoughts' with navigation tabs and a list of posts. **Critical details and analysis**: The page displays button 'Sort by: Hot' (id 1767) hasPopup: menu, indicating the current sort order. The visible posts show dates from 2022-2023 (e.g., 'October 31, 2022', 'March 31, 2023'), confirming Hot sort is active rather than chronological order. To find the latest post, I need to change the sort order to 'New'.",
{reflection_example}
"completed_tasks": "✅ Clicked the 'Forums' link.\\n✅ Clicked the 'Alphabetical' link.\\n✅ Entered the 'Showerthoughts' forum.",
"plan_reason": "The task requires finding the latest post, so I need to change the sorting from 'Hot' to 'New'.",
"plan_list": "⬜ Click the button 'Sort by: Hot'.\\n⬜ Sort the posts by the newest.\\n⬜ Find the user who made the latest post to view his profile.\\n⬜ Count his comments as required.",
"code_reason": "Clicking the 'Sort by: Hot' button is the way to change the sorting order.",
"code": "env_op.click(1767)"
}}}}

----- Step 7
OBSERVING...
{{{{
"summarize_and_analyze": "**Action result**: Successfully opened the user profile page for 'nirinaron'. **Page layout**: The page shows heading 'nirinaron' with navigation tabs 'Overview', 'Submissions', and 'Comments', plus user information. **Critical details and analysis**: The visible content shows one submission article with link 'No comments' displayed, and scanning the page reveals no comment entries. This indicates the user has not made any comments, so the count of comments with more downvotes than upvotes is 0.",
{reflection_example}
"completed_tasks": "✅ Clicked the 'Forums' link.\\n✅ Clicked the 'Alphabetical' link to see all forums.\\n✅ Clicked the 'Showerthoughts' link.\\n✅ Sorted posts by 'New'.\\n✅ Navigated to the latest post author's profile.",
"plan_reason": "The task is to count comments with more downvotes than upvotes, and I've found there are no comments. The task is complete.",
"plan_list": "⬜ No more actions. Complete the task now.",
"code_reason": "The answer has been found, so I will stop and provide the answer.",
"code": "env_op.stop('0')"
}}}}
"""

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 UI elements list:
{ui_content}
""")


def get_prompt_template(reflection: Optional[str] | None = 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 = 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,
  )