{
  "root": {
    "name": "change_all_text_slide_27_center_aligned",
    "description": "Evaluates if all text on slide 27 is center-aligned, and only slide 27 is affected.",
    "is_critical": true,
    "metadata": {},
    "children": [
      {
        "name": "all_text_on_slide_27_is_center_aligned",
        "description": "Checks that every text element on slide 27 is center-aligned.",
        "is_critical": true,
        "metadata": {},
        "children": [
          {
            "name": "verify_text_center_alignment_slide_27",
            "description": "Verifies all textboxes/shapes on slide 27 have center alignment.",
            "is_critical": true,
            "metadata": {},
            "scorer": {
              "type": "function",
              "function_code": "def compute_score():\n    \n    if len(ppt_diff.modified_slides) == 0:\n        return \"No slides were modified.\", 0.0\n    \n    from pptx import Presentation\n    from pptx.enum.text import PP_ALIGN\n    \n    def get_effective_alignment(paragraph, shape_or_cell):\n        \"\"\"Get the effective alignment using multiple detection methods\"\"\"\n        from pptx.enum.text import PP_ALIGN\n        \n        # Method 1: Direct paragraph alignment\n        if paragraph.alignment is not None:\n            return paragraph.alignment\n        \n        # Method 2: Check paragraph properties in XML more thoroughly\n        try:\n            if hasattr(paragraph, '_p') and paragraph._p is not None:\n                # Try multiple XML namespace approaches\n                namespaces = {\n                    'a': 'http://schemas.openxmlformats.org/drawingml/2006/main',\n                    'p': 'http://schemas.openxmlformats.org/presentationml/2006/main'\n                }\n                \n                # Look for pPr (paragraph properties) with different approaches\n                pPr_elements = [\n                    paragraph._p.find('.//a:pPr', namespaces),\n                    paragraph._p.find('.//pPr'),\n                    paragraph._p.find('.//{http://schemas.openxmlformats.org/drawingml/2006/main}pPr'),\n                ]\n                \n                for pPr in pPr_elements:\n                    if pPr is not None:\n                        algn = pPr.get('algn')\n                        if algn:\n                            if algn == 'ctr':\n                                return PP_ALIGN.CENTER\n                            elif algn == 'l':\n                                return PP_ALIGN.LEFT\n                            elif algn == 'r':\n                                return PP_ALIGN.RIGHT\n                        \n                        # Also check for marL, marR attributes which might indicate center\n                        marL = pPr.get('marL')\n                        marR = pPr.get('marR')\n                        if marL and marR and marL == marR:\n                            return PP_ALIGN.CENTER\n        except Exception as e:\n            pass\n        \n        # Method 3: Check placeholder hierarchy more thoroughly\n        try:\n            if hasattr(shape_or_cell, 'placeholder_format') and shape_or_cell.placeholder_format:\n                placeholder_idx = shape_or_cell.placeholder_format.idx\n                \n                # Get the slide layout\n                if hasattr(shape_or_cell, 'part'):\n                    slide_layout = shape_or_cell.part.slide_layout\n                    \n                    # Check layout placeholders more thoroughly\n                    for layout_shape in slide_layout.placeholders:\n                        if layout_shape.placeholder_format.idx == placeholder_idx:\n                            if layout_shape.has_text_frame:\n                                # Check all paragraphs and their properties\n                                for layout_para in layout_shape.text_frame.paragraphs:\n                                    # Direct alignment check\n                                    if layout_para.alignment is not None:\n                                        return layout_para.alignment\n                                    \n                                    # Check XML properties of layout paragraph\n                                    if hasattr(layout_para, '_p') and layout_para._p is not None:\n                                        pPr = layout_para._p.find('.//{http://schemas.openxmlformats.org/drawingml/2006/main}pPr')\n                                        if pPr is not None:\n                                            algn = pPr.get('algn')\n                                            if algn:\n                                                if algn == 'ctr':\n                                                    return PP_ALIGN.CENTER\n                            break\n                    \n                    # Check master placeholders more thoroughly\n                    slide_master = slide_layout.slide_master\n                    for master_shape in slide_master.placeholders:\n                        if master_shape.placeholder_format.idx == placeholder_idx:\n                            if master_shape.has_text_frame:\n                                # Check all paragraphs and their properties\n                                for master_para in master_shape.text_frame.paragraphs:\n                                    # Direct alignment check\n                                    if master_para.alignment is not None:\n                                        return master_para.alignment\n                                    \n                                    # Check XML properties of master paragraph\n                                    if hasattr(master_para, '_p') and master_para._p is not None:\n                                        pPr = master_para._p.find('.//{http://schemas.openxmlformats.org/drawingml/2006/main}pPr')\n                                        if pPr is not None:\n                                            algn = pPr.get('algn')\n                                            if algn:\n                                                if algn == 'ctr':\n                                                    return PP_ALIGN.CENTER\n                            break\n        except Exception as e:\n            pass\n        # Method 4: Check text frame level properties\n        try:\n            if hasattr(shape_or_cell, 'text_frame'):\n                text_frame = shape_or_cell.text_frame\n                # Check if there are any text frame level alignment properties\n                if hasattr(text_frame, '_txBody') and text_frame._txBody is not None:\n                    # Look for bodyPr or lstStyle elements that might contain alignment\n                    bodyPr = text_frame._txBody.find('.//{http://schemas.openxmlformats.org/drawingml/2006/main}bodyPr')\n                    if bodyPr is not None:\n                        anchor = bodyPr.get('anchor')\n                        if anchor == 'ctr':\n                            return PP_ALIGN.CENTER\n        except Exception as e:\n            pass\n        # Method 5: Check if shape position suggests center alignment\n        try:\n            if hasattr(shape_or_cell, 'left') and hasattr(shape_or_cell, 'width'):\n                slide_width = 9144000  # Standard PowerPoint slide width in EMUs\n                shape_left = shape_or_cell.left\n                shape_width = shape_or_cell.width\n                shape_center_x = shape_left + (shape_width / 2)\n                slide_center_x = slide_width / 2\n                \n                # If shape is roughly centered horizontally, likely center-aligned\n                center_tolerance = slide_width * 0.15  # 15% tolerance\n                if abs(shape_center_x - slide_center_x) < center_tolerance:\n                    return PP_ALIGN.CENTER\n        except Exception as e:\n            pass\n        return PP_ALIGN.LEFT\n    \n    try:\n        prs = Presentation(modified_ppt_path)\n        slide = prs.slides[26] if len(prs.slides) >= 27 else None\n        if not slide:\n            return \"Slide 27 not found in the presentation.\", 0.0\n        \n        all_center = True\n        non_center_cnt = 0\n        total_text_items = 0\n        \n        for shape in slide.shapes:\n            # Handle regular text frames\n            if shape.has_text_frame and shape.text_frame.text.strip():  # Only process non-empty text frames\n                # Check if this text frame is center-aligned (check all paragraphs)\n                shape_is_center = True\n                shape_has_content = False\n                \n                for paragraph in shape.text_frame.paragraphs:\n                    if paragraph.text.strip():  # Only count non-empty paragraphs\n                        shape_has_content = True\n                        effective_alignment = get_effective_alignment(paragraph, shape)\n                        \n                        # If any paragraph in the shape is not center-aligned, the shape fails\n                        if effective_alignment != PP_ALIGN.CENTER:\n                            shape_is_center = False\n                            break\n                \n                if shape_has_content:  # Only count shapes that have actual content\n                    total_text_items += 1\n                    \n                    # Debug info\n                    para_align = shape.text_frame.paragraphs[0].alignment if shape.text_frame.paragraphs else None\n                    is_placeholder = hasattr(shape, 'placeholder_format') and shape.placeholder_format\n                    placeholder_idx = shape.placeholder_format.idx if is_placeholder else \"N/A\"\n                    \n                    \n                    if not shape_is_center:\n                        all_center = False\n                        non_center_cnt += 1\n            \n            # Handle tables\n            elif shape.shape_type == 19:  # MSO_SHAPE_TYPE.TABLE = 19\n                for row_idx, row in enumerate(shape.table.rows):\n                    for col_idx, cell in enumerate(row.cells):\n                        if cell.text_frame and cell.text_frame.text.strip():  # Only process non-empty cells\n                            # Check if this cell is center-aligned (check all paragraphs)\n                            cell_is_center = True\n                            \n                            for paragraph in cell.text_frame.paragraphs:\n                                if paragraph.text.strip():  # Only count non-empty paragraphs\n                                    effective_alignment = get_effective_alignment(paragraph, cell)\n                                    \n                                    # If any paragraph in the cell is not center-aligned, the cell fails\n                                    if effective_alignment != PP_ALIGN.CENTER:\n                                        cell_is_center = False\n                                        break\n                            \n                            total_text_items += 1\n                            \n                            \n                            if not cell_is_center:\n                                all_center = False\n                                non_center_cnt += 1\n        \n        if total_text_items == 0:\n            return \"No text elements found on slide 27.\", 0.0\n        \n        if all_center:\n            return \"All text on slide 27 is center-aligned.\", 1.0\n        else:\n            score = (total_text_items - non_center_cnt) / total_text_items\n            return f\"{non_center_cnt} out of {total_text_items} text element(s) not center-aligned on slide 27.\", score\n            \n    except Exception as e:\n        return f\"Error: {e}\", 0.0\n"
            }
          }
        ]
      },
      {
        "name": "no_unintended_changes_to_other_slides",
        "description": "Checks that no unintended changes were made to slides other than 27.",
        "is_critical": false,
        "metadata": {},
        "children": [
          {
            "name": "verify_only_slide_27_modified",
            "description": "Verifies only slide 27 is modified (content, layout, elements, text, etc.).",
            "is_critical": true,
            "metadata": {},
            "scorer": {
              "type": "function",
              "function_code": "def compute_score():\n    # Check for modifications or additions/removals on slides other than 27\n    changed = set()\n    for s1, s2 in ppt_diff.modified_slides:\n        if s1.slide_number != 27:\n            changed.add(s1.slide_number)\n    for s in ppt_diff.added_slides:\n        if s.slide_number != 27:\n            changed.add(s.slide_number)\n    for s in ppt_diff.removed_slides:\n        if s.slide_number != 27:\n            changed.add(s.slide_number)\n    if changed:\n        return f\"Slides other than 27 were changed: {sorted(changed)}\", 0.0\n    return \"Only slide 27 was modified.\", 1.0\n"
            }
          }
        ]
      },
      {
        "name": "no_extraneous_formatting_or_effects_slide_27",
        "description": "Checks no other formatting, animations, or transitions were added/removed on slide 27.",
        "is_critical": false,
        "metadata": {},
        "children": [
          {
            "name": "verify_no_new_animations_on_slide_27",
            "description": "Verifies no new or removed animations on slide 27.",
            "is_critical": false,
            "metadata": {},
            "scorer": {
              "type": "function",
              "function_code": "def compute_score():\n    slide27_id = None\n    for s in ppt_diff.modified_slides + [(s, s) for s in ppt_diff.added_slides]:\n        if s[1].slide_number == 27:\n            slide27_id = s[1].slide_id\n            break\n    if slide27_id is None:\n        # Try to infer: fallback by index if slide_id missing\n        try:\n            slide27_id = ppt_diff.modified_slides[26][1].slide_id\n        except:\n            pass\n    added = [a for a in ppt_diff.added_animations if a.slide_id == slide27_id]\n    removed = [a for a in ppt_diff.removed_animations if a.slide_id == slide27_id]\n    if added or removed:\n        return f\"Animations added/removed on slide 27: added={len(added)}, removed={len(removed)}\", 0.0\n    return \"No animations added or removed on slide 27.\", 1.0\n"
            }
          },
          {
            "name": "verify_no_new_transitions_on_slide_27",
            "description": "Verifies no new or removed transitions on slide 27.",
            "is_critical": false,
            "metadata": {},
            "scorer": {
              "type": "function",
              "function_code": "def compute_score():\n    slide27_id = None\n    for s in ppt_diff.modified_slides + [(s, s) for s in ppt_diff.added_slides]:\n        if s[1].slide_number == 27:\n            slide27_id = s[1].slide_id\n            break\n    if slide27_id is None:\n        # Try to infer: fallback by index if slide_id missing\n        try:\n            slide27_id = ppt_diff.modified_slides[26][1].slide_id\n        except:\n            pass\n    added = [t for t in ppt_diff.added_transitions if t.slide_id == slide27_id]\n    removed = [t for t in ppt_diff.removed_transitions if t.slide_id == slide27_id]\n    if added or removed:\n        return f\"Transitions added/removed on slide 27: added={len(added)}, removed={len(removed)}\", 0.0\n    return \"No transitions added or removed on slide 27.\", 1.0\n"
            }
          },
          {
            "name": "verify_no_other_formatting_changes_slide_27",
            "description": "Checks no extraneous formatting changes (font, color, etc.) except alignment on slide 27.",
            "is_critical": false,
            "metadata": {},
            "scorer": {
              "type": "function",
              "function_code": "def compute_score():\n    # Focus on major formatting changes. We'll compare text properties other than alignment.\n    from pptx import Presentation\n    import difflib\n    try:\n        prs_orig = Presentation(original_ppt_path)\n        prs_mod = Presentation(modified_ppt_path)\n        slide_orig = None\n        slide_mod = None\n        for idx, s in enumerate(prs_orig.slides):\n            if idx == 26: slide_orig = s\n        for idx, s in enumerate(prs_mod.slides):\n            if idx == 26: slide_mod = s\n        if slide_orig is None or slide_mod is None:\n            return \"Slide 27 missing in original or modified.\", 1.0  # Don't penalize if can't compare\n        changes = 0\n        total = 0\n        for sh_o, sh_m in zip(slide_orig.shapes, slide_mod.shapes):\n            if not (sh_o.has_text_frame and sh_m.has_text_frame):\n                continue\n            para_o = sh_o.text_frame.paragraphs[0]\n            para_m = sh_m.text_frame.paragraphs[0]\n            # Check font size, color, bold, italic, underline, font name\n            run_o = para_o.runs[0] if para_o.runs else None\n            run_m = para_m.runs[0] if para_m.runs else None\n            if not (run_o and run_m):\n                continue\n            # Compare properties except alignment\n            diff = False\n            props = [\n                'font.size','font.bold','font.italic','font.underline','font.color.rgb','font.name'\n            ]\n            if run_o.font.size != run_m.font.size: diff = True\n            if run_o.font.bold != run_m.font.bold: diff = True\n            if run_o.font.italic != run_m.font.italic: diff = True\n            if run_o.font.underline != run_m.font.underline: diff = True\n            if (run_o.font.color and run_m.font.color and run_o.font.color.rgb != run_m.font.color.rgb): diff = True\n            if run_o.font.name != run_m.font.name: diff = True\n            if diff: changes += 1\n            total += 1\n        if total == 0:\n            return \"No comparable text runs found on slide 27.\", 1.0\n        if changes == 0:\n            return \"No extraneous formatting changes on slide 27.\", 1.0\n        else:\n            score = 1.0 - (changes/total)\n            return f\"{changes} of {total} text elements have new formatting (not alignment)\", score\n    except Exception as e:\n        return f\"Formatting check error: {e}\", 1.0\n"
            }
          }
        ]
      }
    ]
  },
  "metadata": {
    "task": "Change all text on slide 27 to be center-aligned"
  }
}