{
  "root": {
    "name": "Slide numbering is added to the bottom right of all slides",
    "description": "Evaluates whether slide numbers are added to every slide and positioned at the bottom right, with no extraneous or incorrect modifications.",
    "is_critical": true,
    "metadata": {},
    "children": [
      {
        "name": "All slides in the modified PPT have slide numbers present",
        "description": "Checks that every slide contains a visible and correct slide number element.",
        "is_critical": true,
        "metadata": {},
        "scorer": {
          "type": "function",
          "function_code": "def compute_score() -> tuple[str, float]:\n    from pptx import Presentation\n    import re\n    present = 0\n    total = 0\n    prs = Presentation(modified_ppt_path)\n    for idx, slide in enumerate(prs.slides):\n        total += 1\n        found = False\n        for shape in slide.shapes:\n            if not shape.has_text_frame:\n                continue\n            text = shape.text_frame.text.strip()\n            # Accept 1 or '1', sometimes 'Slide 1'\n            if re.fullmatch(r'(Slide )?%d' % (idx+1), text):\n                found = True\n                break\n        if found:\n            present += 1\n    if total == 0:\n        return (\"No slides found in the presentation.\", 0.0)\n    score = present / total\n    if score == 1.0:\n        return (\"All slides have their slide number present.\", 1.0)\n    else:\n        return (f\"{present} out of {total} slides have a visible slide number.\", score)\n"
        }
      },
      {
        "name": "Slide number is correctly positioned at the bottom right of each slide",
        "description": "Checks that the slide number appears at the correct bottom right location for every slide.",
        "is_critical": true,
        "metadata": {},
        "scorer": {
          "type": "function",
          "function_code": "def compute_score() -> tuple[str, float]:\n    from pptx import Presentation\n    import re\n    prs = Presentation(modified_ppt_path)\n    total = 0\n    correct_pos = 0\n    \n    for idx, slide in enumerate(prs.slides):\n        slide_num_ok = False\n        for shape in slide.shapes:\n            if not shape.has_text_frame:\n                continue\n            text = shape.text_frame.text.strip()\n            \n            # Check if this looks like a slide number for this slide\n            expected_numbers = [str(idx+1), f\"Slide {idx+1}\"]\n            if text in expected_numbers or re.fullmatch(r'(Slide )?%d' % (idx+1), text):\n                # Assess shape position - use more reasonable thresholds\n                SLIDE_W = prs.slide_width\n                SLIDE_H = prs.slide_height\n                \n                # Calculate position percentages\n                right_pct = ((shape.left + shape.width) / SLIDE_W) * 100\n                bottom_pct = ((shape.top + shape.height) / SLIDE_H) * 100\n                \n                # Accept if shape is in bottom-right area (flexible thresholds)\n                # Right side: at least 70% from left edge\n                # Bottom: at least 85% from top edge  \n                if right_pct >= 70 and bottom_pct >= 85:\n                    slide_num_ok = True\n                    break\n        \n        total += 1\n        if slide_num_ok:\n            correct_pos += 1\n    \n    if total == 0:\n        return (\"No slides in presentation.\", 0.0)\n    \n    score = correct_pos / total\n    if score == 1.0:\n        return (\"All slide numbers are positioned in the bottom right area.\", 1.0)\n    else:\n        return (f\"{correct_pos} of {total} slides have the slide number at the correct position.\", score)\n"
        }
      },
      {
        "name": "No extraneous or unintended modifications",
        "description": "Checks that only slide numbers were added and no other changes (e.g., animations, transitions, elements, or slide modifications unrelated to slide numbering) were made.",
        "is_critical": false,
        "metadata": {},
        "scorer": {
          "type": "function",
          "function_code": "def compute_score() -> tuple[str, float]:\n    # For slide numbering task, allow text boxes and slide number placeholders\n    # Only reject major content changes, slide additions/removals\n    unexpected = False\n    messages = []\n    \n    # 1. Any added/removed slides?\n    if ppt_diff.added_slides or ppt_diff.removed_slides:\n        unexpected = True\n        messages.append(\"Slides were added or removed.\")\n    \n    # 2. Check for major content changes (ignore minor animation/transition changes \n    # that happen naturally when adding slide numbers)\n    from pptx import Presentation\n    import re\n    original = Presentation(original_ppt_path)\n    modified = Presentation(modified_ppt_path)\n    \n    if len(original.slides) != len(modified.slides):\n        unexpected = True\n        messages.append(\"Number of slides changed.\")\n    \n    # Precompile slide number text regex\n    import re\n    slide_number_re = re.compile(r'^(slide\\s+)?(\\d+|\\*)$')\n\n    # Check for shapes that aren't slide numbers and ensure existing content not modified\n    for i, (mslide, oslide) in enumerate(zip(modified.slides, original.slides)):\n        added_shapes = []\n        \n        # Build lookup of original shapes by name -> text (normalized)\n        original_shape_text = {}\n        for oshape in oslide.shapes:\n            oname = getattr(oshape, \"name\", \"\")\n            if oname and hasattr(oshape, \"has_text_frame\") and oshape.has_text_frame and oshape.text_frame.text:\n                original_shape_text[oname] = re.sub(r\"\\s+\", \" \", oshape.text_frame.text.strip()).lower()\n\n        # Track if existing shape text changed\n        # Find shapes that are new in modified slide & detect modifications\n        for shape in mslide.shapes:\n            shape_name = getattr(shape, \"name\", \"\")\n            shape_type = getattr(shape, \"shape_type\", None)\n            \n            # Skip if it's a slide number placeholder or similar\n            if (\"slide number\" in shape_name.lower() or \n                \"placeholder\" in shape_name.lower()):\n                continue\n                \n            # Check if this shape has text that looks like a slide number\n            if (hasattr(shape, \"has_text_frame\") and shape.has_text_frame and shape.text_frame.text):\n                text = shape.text_frame.text.strip()\n                # Match various slide number formats\n                if slide_number_re.match(text.lower()) or text.isdigit():\n                    continue\n            \n            # Check if this shape exists in original (by name/content)\n            found_in_original = False\n            for oshape in oslide.shapes:\n                if (getattr(oshape, \"name\", \"\") == shape_name and shape_name):\n                    found_in_original = True\n                    break\n            \n            if not found_in_original:\n                added_shapes.append(shape)\n            else:\n                # Potential modification: compare text content if both have text frames\n                if (shape_name in original_shape_text and hasattr(shape, \"has_text_frame\") and shape.has_text_frame):\n                    m_text_raw = shape.text_frame.text or \"\"\n                    m_text_norm = re.sub(r\"\\s+\", \" \", m_text_raw.strip()).lower()\n                    o_text_norm = original_shape_text.get(shape_name, \"\")\n                    if m_text_norm != o_text_norm:\n                        # Allow if original empty and modified only added a slide number text\n                        if not (o_text_norm == \"\" and slide_number_re.match(m_text_norm)):\n                            unexpected = True\n                            messages.append(f\"Content of shape '{shape_name}' modified on slide {i+1}.\")\n        \n        # Only flag non-slide-number additions as unexpected\n        significant_additions = []\n        for shape in added_shapes:\n            shape_name = getattr(shape, \"name\", \"\")\n            if not (\"slide number\" in shape_name.lower() or \"placeholder\" in shape_name.lower()):\n                # Check if the shape contains slide number text\n                if hasattr(shape, \"has_text_frame\") and shape.has_text_frame:\n                    text = shape.text_frame.text.strip()\n                    if not slide_number_re.match(text.lower()) and not text.isdigit():\n                        significant_additions.append(shape)\n                else:\n                    significant_additions.append(shape)\n        \n        if significant_additions:\n            for shape in significant_additions:\n                unexpected = True\n                shape_name = getattr(shape, \"name\", \"unnamed\")\n                messages.append(f\"Unexpected non-slide-number shape '{shape_name}' added on slide {i+1}.\")\n    \n    if unexpected:\n        return (\"; \".join(messages), 0.0)\n    return (\"No significant extraneous modifications detected (slide number additions are expected).\", 1.0)\n"
        }
      }
    ]
  },
  "metadata": {
    "task": "Add slide numbering at the bottom right corner of all slides"
  }
}