#!/usr/bin/env python3
"""
RealWorldPlotter: high-quality geometric figure renderer.

Based on OpenCV / NumPy, renders plotting_code generated by LLMPlotter into realistic PNG images.
"""

from __future__ import annotations

import copy
import math
import re
from collections import defaultdict
from dataclasses import dataclass
from pathlib import Path
from typing import Any, Dict, Iterable, List, Optional, Sequence, Tuple

import cv2
import numpy as np
from numpy.core.defchararray import isdigit

from PIL import Image, ImageDraw, ImageFont  # type: ignore


PointName = str
RightAngleKey = Tuple[Tuple[str, int, int], Tuple[str, int, int]]


@dataclass
class PlotStyle:
    """Rendering style configuration (highly tunable)."""
    
    background_color: Tuple[int, int, int] = (255, 255, 255)
    background_gradient: bool = False
    background_gradient_color: Tuple[int, int, int] = (255, 255, 255)

    grid_spacing: int = 0
    grid_color: Tuple[int, int, int] = (220, 220, 220)
    grid_alpha: float = 0.0

    fill_color: Tuple[int, int, int] = (0, 0, 0)
    fill_palette: Tuple[Tuple[int, int, int], ...] = ((0, 0, 0),)
    fill_opacity: float = 0.0

    line_color: Tuple[int, int, int] = (0, 0, 0)
    line_palette: Tuple[Tuple[int, int, int], ...] = ((0, 0, 0),)
    line_width: int = 2
    line_glow_color: Optional[Tuple[int, int, int]] = None
    line_glow_width: int = 0

    circle_color: Tuple[int, int, int] = (0, 0, 0)
    circle_fill_alpha: float = 0.0

    point_color: Tuple[int, int, int] = (0, 0, 0)
    point_outline_color: Optional[Tuple[int, int, int]] = None
    point_radius: int = 6
    point_outline_width: int = 0

    label_color: Tuple[int, int, int] = (0, 0, 0)
    label_shadow_color: Optional[Tuple[int, int, int]] = None
    label_shadow_offset: Tuple[int, int] = (1, 1)
    label_background_color: Optional[Tuple[int, int, int]] = None
    label_border_color: Optional[Tuple[int, int, int]] = None
    label_padding: int = 4
    label_font_face: int = cv2.FONT_HERSHEY_SIMPLEX
    label_font_min_scale: float = 0.9
    label_font_max_scale: float = 2.0
    label_font_scale_multiplier: float = 1.0
    label_font_thickness: int = 2
    label_distance: int = 22

    annotation_color: Tuple[int, int, int] = (0, 0, 0)
    annotation_width: int = 3
    equal_tick_length: int = 12
    annotation_label_background_color: Optional[Tuple[int, int, int]] = None
    annotation_label_border_color: Optional[Tuple[int, int, int]] = None
    annotation_label_padding: int = 4
    annotation_font_scale_multiplier: float = 1.0
    annotation_font_min_size: int = 12
    annotation_font_divisor: float = 0.0
    
    # Debug mode
    debug_show_boxes: bool = False
    debug_box_color: Tuple[int, int, int] = (255, 0, 0)  # red
    debug_box_thickness: int = 1
    
    # Canvas rotation
    rotation_angle: float = 0  # rotation angle (degrees); positive values are counter-clockwise


class RealWorldPlotter:
    """Render realistic PNG images according to plotting_code."""

    def __init__(
        self,
        min_canvas: int = 480,
        max_canvas: int = 640,
        style: Optional[PlotStyle] = None,
        debug: bool = False,
    ) -> None:
        self.min_canvas = min_canvas
        self.max_canvas = max_canvas
        self.style = style or PlotStyle()
        if debug:
            self.style.debug_show_boxes = True
        self._point_centroid: Optional[Tuple[float, float]] = None
        self._label_boxes: List[Tuple[int, int, int, int]] = []
        self._annotation_boxes: List[Tuple[int, int, int, int]] = []
        # Geometric element boxes used for overlap detection
        self._segment_boxes: List[Tuple[int, int, int, int]] = []  # narrow rectangles for segments
        self._circle_boxes: List[Tuple[int, int, int, int, int]] = []  # circle boxes (x, y, radius)
        self.last_processed_plotting_code: Optional[Dict[str, Any]] = None
        # Segment chains including implicit intersection points (used to determine atomic segments)
        self._segment_chains_with_implicit: Optional[List[List[str]]] = None

    # ------------------------------------------------------------------ #
    # Public API                                                         #
    # ------------------------------------------------------------------ #
    def render_image(
        self,
        plotting_code: Dict,
        output_file: Path,
        return_plotting_code: bool = False,
    ) -> bool | Tuple[bool, Dict[str, Any]]:
        """
        Args:
            plotting_code: Dictionary containing points / segments / circles / annotations.
            output_file: Output PNG path.
            return_plotting_code: When True, returns a snapshot of the rendering result, including normalized segments and actual annotations.
        """
        points_data = plotting_code.get("points") or {}
        if not points_data:
            raise ValueError("plotting_code['points'] is required")

        segments = list(plotting_code.get("segments") or [])
        circles = list(plotting_code.get("circles") or [])
        annotations = plotting_code.get("annotations") or {}

        canonical_names = self._build_canonical_name_map(points_data)

        points, canvas, scale = self._normalize_coordinates(points_data)
        
        # Check whether there are points with identical coordinates
        point_list = list(points.items())
        for i in range(len(point_list)):
            name1, coord1 = point_list[i]
            for j in range(i + 1, len(point_list)):
                name2, coord2 = point_list[j]
                if coord1 == coord2:
                    # Found points with identical coordinates; abort rendering
                    if return_plotting_code:
                        return False, {}
                    return False
        
        float_points = {name.lower(): (float(value[0]), float(value[1])) for name, value in points_data.items()}
        
        # Part 1: compute centers and radii for all circles
        circle_info_list = self._calculate_circle_info(points, circles, scale) if circles else []
        
        # Part 2: expand the canvas according to circle radii to ensure circles do not go out of bounds
        if circle_info_list:
            height, width = canvas.shape[:2]
            min_padding = 10  # minimum padding
            
            # Compute required expansion in each direction
            expand_left = 0
            expand_right = 0
            expand_top = 0
            expand_bottom = 0
            
            for center, radius in circle_info_list:
                radius_int = int(radius)
                # Compute required extra space in each direction
                left_needed = radius_int - center[0] + min_padding
                right_needed = (center[0] + radius_int) - (width - 1) + min_padding
                top_needed = radius_int - center[1] + min_padding
                bottom_needed = (center[1] + radius_int) - (height - 1) + min_padding
                
                expand_left = max(expand_left, max(0, left_needed))
                expand_right = max(expand_right, max(0, right_needed))
                expand_top = max(expand_top, max(0, top_needed))
                expand_bottom = max(expand_bottom, max(0, bottom_needed))
            
            # If expansion is needed, extend the canvas
            if expand_left > 0 or expand_right > 0 or expand_top > 0 or expand_bottom > 0:
                new_height = height + expand_top + expand_bottom
                new_width = width + expand_left + expand_right
                new_canvas = np.full(
                    (new_height, new_width, 3),
                    self.style.background_color,
                    dtype=np.uint8
                )
                # Copy the original canvas into the appropriate position of the new canvas
                new_canvas[expand_top:expand_top+height, expand_left:expand_left+width] = canvas
                # Update coordinates of all points
                for name in points:
                    px, py = points[name]
                    points[name] = (px + expand_left, py + expand_top)
                # Update centers of all circles
                circle_info_list = [
                    ((center[0] + expand_left, center[1] + expand_top), radius)
                    for center, radius in circle_info_list
                ]
                canvas = new_canvas
        
        normalized_segments = self._normalize_segment_list(points, segments)
        
        # Before building segment-chains, check whether the two sides of each right angle exist; if not, add them
        normalized_segments, added_segments = self._add_missing_right_angle_segments(
            points, normalized_segments, annotations
        )
        
        # Also append newly added segments to the original `segments` list
        if added_segments:
            segments.extend(added_segments)
        
        processed_annotations, segment_chains, annotation_summary = self._prepare_annotations(
            points,
            float_points,
            normalized_segments,
            annotations,
        )
        annotations = processed_annotations
        # IMPORTANT:
        # - `points` are pixel-grid coordinates after normalization (used for rendering)
        # - `points_data` are the original coordinate system from plotting_code (often floats)
        # For `actual_data` we want the final pixel-grid coordinates, so snapshot uses `points`.
        self.last_processed_plotting_code = self._build_render_snapshot(
            canonical_names,
            points,
            segments,
            circles,
            annotations,
            segment_chains,
            annotation_summary,
        )

        self._label_boxes.clear()
        self._annotation_boxes.clear()
        self._segment_boxes.clear()
        self._circle_boxes.clear()
        
        # Step 1: draw basic shapes and segments
        self._apply_region_fills(canvas, points, plotting_code, annotations)
        # First create detection boxes for segments and circles (before drawing)
        self._create_segment_boxes(points, segments)
        self._create_circle_boxes(points, circles, circle_info_list)
        # Then draw
        self._draw_segments(canvas, points, segments)
        self._draw_circles(canvas, circle_info_list, points=points, circles=circles)
        
        # Step 2: draw points (without labels)
        self._draw_points(canvas, points)
        
        # Step 3: draw other annotations (lengths, angles, right-angle markers, etc.)
        self._draw_annotations(canvas, points, annotations, segments, segment_chains)
        
        # Step 4: finally draw point labels (this allows detecting overlaps with all drawn content, including right-angle markers)
        self._draw_point_labels(canvas, points, segments)
        
        # Debug mode: draw all annotation rectangles
        if self.style.debug_show_boxes:
            self._draw_debug_boxes(canvas)

        # Rotate the canvas (if a rotation angle is set)
        rotated_points_snapshot: Optional[Dict[str, Tuple[int, int]]] = None
        if abs(self.style.rotation_angle) > 1e-6:  # avoid floating-point precision issues
            canvas, rotated_points_snapshot = self._rotate_canvas_with_points(
                canvas, points, self.style.rotation_angle
            )
        else:
            rotated_points_snapshot = dict(points)

        # Auto-crop the canvas and record the offset to align coordinates (based on the rotated coordinates)
        canvas, crop_offset = self._auto_crop(canvas)
        if rotated_points_snapshot is not None and crop_offset is not None:
            ox, oy = crop_offset
            rotated_points_snapshot = {
                name: (int(px - ox), int(py - oy))
                for name, (px, py) in rotated_points_snapshot.items()
            }

        output_file.parent.mkdir(parents=True, exist_ok=True)
        success = cv2.imwrite(str(output_file), canvas)
        if return_plotting_code:
            snapshot = copy.deepcopy(self.last_processed_plotting_code or {})
            if rotated_points_snapshot is not None:
                # Overwrite points with rotated pixel coordinates
                snapshot["points"] = {
                    canonical_names.get(str(name).lower(), str(name)): (int(x), int(y))
                    for name, (x, y) in rotated_points_snapshot.items()
                }
            return bool(success), snapshot
        return bool(success)

    # ------------------------------------------------------------------ #
    # Plotting-data preprocessing                                       #
    # ------------------------------------------------------------------ #
    def _build_canonical_name_map(
        self,
        points: Dict[str, Sequence[float]],
    ) -> Dict[str, str]:
        mapping: Dict[str, str] = {}
        for raw_name in points.keys():
            key = str(raw_name).lower()
            if key not in mapping:
                mapping[key] = str(raw_name)
        return mapping

    def _prepare_annotations(
        self,
        points: Dict[str, Tuple[int, int]],
        float_points: Dict[str, Tuple[float, float]],
        normalized_segments: List[Tuple[str, str]],
        annotations: Dict,
    ) -> Tuple[Dict, List[List[str]], Dict[str, Any]]:
        segment_chains, segment_chains_with_implicit = self._build_segment_chains(float_points, normalized_segments)
        filtered_annotations, annotation_summary = self._filter_annotations_for_atomic_elements(
            points,
            normalized_segments,
            annotations,
            segment_chains_with_implicit,
        )
        return filtered_annotations, segment_chains, annotation_summary

    def _add_missing_right_angle_segments(
        self,
        points: Dict[str, Tuple[int, int]],
        normalized_segments: List[Tuple[str, str]],
        annotations: Dict,
    ) -> Tuple[List[Tuple[str, str]], List[List[str]]]:
        """
        Check whether the two sides of each right-angle annotation exist in `segments`;
        if not, add the missing segments.

        This step should be executed before building segment-chains.
        
        Args:
            points: Dictionary of point coordinates.
            normalized_segments: List of normalized segments.
            annotations: Annotation dictionary containing `right_angles`.
        
        Returns:
            (updated list of segments, newly added raw-format segments)
        """
        # Create a set of segments for fast lookup (using ordered pairs so that (a, b) and (b, a) are both recognized)
        segment_set: set[Tuple[str, str]] = set()
        for a, b in normalized_segments:
            segment_set.add((a, b))
            segment_set.add((b, a))  # add the reverse edge for easier lookup
        
        # Get all right-angle annotations
        right_angles = annotations.get("right_angles") or []
        new_segments: List[Tuple[str, str]] = []
        added_segments_raw: List[List[str]] = []  # raw format, to be appended to `segments`
        
        for entry in right_angles:
            if not isinstance(entry, (list, tuple)) or len(entry) != 3:
                continue
            
            a, b, c = (str(p).lower() for p in entry)
            
            # Check that the points exist
            if any(p not in points for p in (a, b, c)):
                continue
            
            # Check whether the first edge (a, b) exists
            edge1 = (a, b)
            edge1_reverse = (b, a)
            if edge1 not in segment_set and edge1_reverse not in segment_set:
                # Edge does not exist; add it to the new segment list
                new_segments.append(edge1)
                added_segments_raw.append([a, b])  # raw format
                segment_set.add(edge1)
                segment_set.add(edge1_reverse)
            
            # Check whether the second edge (b, c) exists
            edge2 = (b, c)
            edge2_reverse = (c, b)
            if edge2 not in segment_set and edge2_reverse not in segment_set:
                # Edge does not exist; add it to the new segment list
                new_segments.append(edge2)
                added_segments_raw.append([b, c])  # raw format
                segment_set.add(edge2)
                segment_set.add(edge2_reverse)
        
        # Add newly created segments to the original normalized_segments list
        if new_segments:
            normalized_segments = normalized_segments + new_segments
        
        return normalized_segments, added_segments_raw

    def _normalize_segment_list(
        self,
        points: Dict[str, Tuple[int, int]],
        segments: Iterable[Sequence[PointName]],
    ) -> List[Tuple[str, str]]:
        normalized: List[Tuple[str, str]] = []
        for segment in segments:
            if not segment or len(segment) != 2:
                continue
            a_raw, b_raw = segment
            a = str(a_raw).lower()
            b = str(b_raw).lower()
            if a == b or a not in points or b not in points:
                continue
            normalized.append((a, b))
        return normalized

    def _build_render_snapshot(
        self,
        canonical_names: Dict[str, str],
        points: Dict[str, Tuple[int, int]],
        segments: Iterable[Sequence[PointName]],
        circles: Iterable[Sequence],
        annotations: Dict,
        segment_chains: List[List[str]],
        annotation_summary: Dict[str, Any],
    ) -> Dict[str, Any]:
        points_snapshot: Dict[str, Tuple[int, int]] = {}
        for name, coord in points.items():
            canonical = canonical_names.get(str(name).lower(), str(name))
            # points here are already in pixel-grid coordinates (int, int) after normalization
            points_snapshot[canonical] = (int(coord[0]), int(coord[1]))

        segments_snapshot = [copy.deepcopy(segment) for segment in list(segments)]
        circles_snapshot = copy.deepcopy(list(circles))
        annotations_snapshot = copy.deepcopy(annotations)
        chains_snapshot = [
            [canonical_names.get(name, name) for name in chain]
            for chain in segment_chains
        ]

        result = {
            "points": points_snapshot,
            "segments": segments_snapshot,
            "circles": circles_snapshot,
            "annotations": annotations_snapshot,
            "segment_chains": chains_snapshot,
            "annotation_summary": copy.deepcopy(annotation_summary),
        }
        
        # `right_angles` information is already contained in `annotation_summary`; no need to duplicate it.
        
        return result

    def _segment_intersect(
        self,
        p1: Tuple[float, float],
        p2: Tuple[float, float],
        p3: Tuple[float, float],
        p4: Tuple[float, float],
        tolerance: float = 1e-6,
    ) -> Optional[Tuple[float, float]]:
        """
        Determine whether two line segments intersect and return their intersection point
        (using pre-scale float coordinates).
        
        Args:
            p1, p2: Endpoints of the first segment.
            p3, p4: Endpoints of the second segment.
            tolerance: Tolerance for floating-point errors.
        
        Returns:
            Intersection coordinates (x, y); returns None if they do not intersect or are collinear.
        """
        x1, y1 = p1
        x2, y2 = p2
        x3, y3 = p3
        x4, y4 = p4
        
        # Compute direction vectors
        dx1 = x2 - x1
        dy1 = y2 - y1
        dx2 = x4 - x3
        dy2 = y4 - y3
        
        # Compute cross product
        denom = dx1 * dy2 - dy1 * dx2
        
        # If the denominator is close to 0, the segments are parallel or collinear
        if abs(denom) < tolerance:
            return None
        
        # Compute parameters t and u
        t = ((x3 - x1) * dy2 - (y3 - y1) * dx2) / denom
        u = ((x3 - x1) * dy1 - (y3 - y1) * dx1) / denom
        
        # Check whether the intersection lies on both segments (including endpoints)
        if 0 <= t <= 1 and 0 <= u <= 1:
            # Compute intersection coordinates
            x = x1 + t * dx1
            y = y1 + t * dy1
            return (x, y)
        
        return None

    def _point_exists(
        self,
        point: Tuple[float, float],
        existing_points: Dict[str, Tuple[float, float]],
        tolerance: float = 1e-3,
    ) -> Optional[str]:
        """
        Check whether a point already exists in the original data (explicit intersection),
        using pre-scale float coordinates for comparison.
        
        Args:
            point: Coordinates of the point to check.
            existing_points: All points in the original data (pre-scale float coordinates).
            tolerance: Tolerance for coordinate comparison.
        
        Returns:
            If the point exists, return its name; otherwise return None.
        """
        px, py = point
        for name, (ex, ey) in existing_points.items():
            if abs(px - ex) < tolerance and abs(py - ey) < tolerance:
                return name
        return None

    def _point_on_segment(
        self,
        point: Tuple[float, float],
        p1: Tuple[float, float],
        p2: Tuple[float, float],
        tolerance: float = 1e-3,
    ) -> bool:
        """
        Check whether a point lies on a segment (excluding endpoints),
        using pre-scale float coordinates.
        
        Args:
            point: Point to check.
            p1, p2: Endpoints of the segment.
            tolerance: Tolerance for floating-point errors.
        
        Returns:
            True if the point lies in the interior of the segment (excluding endpoints), otherwise False.
        """
        px, py = point
        x1, y1 = p1
        x2, y2 = p2
        
        # Check whether the point lies on the extended line of the segment
        cross = (px - x1) * (y2 - y1) - (py - y1) * (x2 - x1)
        if abs(cross) > tolerance:
            return False
        
        # Check whether the point lies inside the segment (excluding endpoints)
        dot = (px - x1) * (x2 - x1) + (py - y1) * (y2 - y1)
        length_sq = (x2 - x1) ** 2 + (y2 - y1) ** 2
        
        if length_sq < tolerance:
            return False
        
        t = dot / length_sq
        return tolerance < t < 1 - tolerance

    def _find_all_intersections(
        self,
        points: Dict[str, Tuple[float, float]],
        normalized_segments: List[Tuple[str, str]],
    ) -> Dict[Tuple[Tuple[str, str], Tuple[str, str]], List[Tuple[str, Tuple[float, float], str]]]:
        """
        Find all intersection points between segments (using pre-scale float coordinates).
        
        Args:
            points: Dictionary of point coordinates (pre-scale float coordinates).
            normalized_segments: List of normalized segments.
        
        Returns:
            Dict: {(seg1, seg2): [(point_id, (x, y), 'explicit'|'implicit'), ...]}
                - point_id: For explicit intersections, the point name; for implicit intersections, a unique ID.
                - (x, y): Intersection coordinates (pre-scale float coordinates).
                - 'explicit'|'implicit': Intersection type.
        """
        intersections: Dict[Tuple[Tuple[str, str], Tuple[str, str]], List[Tuple[str, Tuple[float, float], str]]] = {}
        implicit_point_counter = 0
        
        n = len(normalized_segments)
        for i in range(n):
            seg1 = normalized_segments[i]
            a1, b1 = seg1
            if a1 not in points or b1 not in points:
                continue
            p1 = points[a1]
            p2 = points[b1]
            
            for j in range(i + 1, n):
                seg2 = normalized_segments[j]
                a2, b2 = seg2
                if a2 not in points or b2 not in points:
                    continue
                
                # Skip segments sharing endpoints (they are already connected in the components)
                if a1 == a2 or a1 == b2 or b1 == a2 or b1 == b2:
                    continue
                
                p3 = points[a2]
                p4 = points[b2]
                
                # Compute intersection
                intersection = self._segment_intersect(p1, p2, p3, p4)
                if intersection is not None:
                    # Check whether the intersection is an explicit point (already exists in the original data)
                    point_name = self._point_exists(intersection, points)
                    
                    if point_name is not None:
                        # Explicit intersection
                        point_id = point_name
                        point_type = 'explicit'
                    else:
                        # Implicit intersection; generate a unique ID
                        point_id = f"__implicit_{implicit_point_counter}"
                        implicit_point_counter += 1
                        point_type = 'implicit'
                    
                    key = (seg1, seg2) if seg1 < seg2 else (seg2, seg1)
                    if key not in intersections:
                        intersections[key] = []
                    intersections[key].append((point_id, intersection, point_type))
        
        return intersections

    def _build_segment_chains(
        self,
        points: Dict[str, Tuple[float, float]],
        normalized_segments: List[Tuple[str, str]],
    ) -> List[List[str]]:
        """
        Build segment chains that include explicit and implicit intersections for identifying atomic segments,
        but ultimately return chains containing only explicit intersections, using pre-scale float coordinates.
        """
        # Step 1: build a base adjacency list (only original segments)
        line_groups: Dict[Tuple[float, float, float], Dict[str, Any]] = {}
        for a, b in normalized_segments:
            if a not in points or b not in points:
                continue
            signature = self._line_signature(points[a], points[b])
            if signature is None:
                continue
            line_key, direction = signature
            info = line_groups.setdefault(
                line_key,
                {
                    "direction": direction,
                    "adjacency": defaultdict(set),
                },
            )
            info["adjacency"][a].add(b)
            info["adjacency"][b].add(a)
        
        # Step 2: find all intersections (explicit and implicit)
        all_intersections = self._find_all_intersections(points, normalized_segments)
        
        # Create a point dictionary including implicit intersections (used to build components that include implicit points)
        points_with_implicit = dict(points)  # copy original points
        implicit_points: Dict[str, Tuple[float, float]] = {}  # dictionary of implicit intersection points
        
        # Collect all implicit intersections
        for (seg1, seg2), point_list in all_intersections.items():
            for point_id, point_coord, point_type in point_list:
                if point_type == 'implicit':
                    implicit_points[point_id] = point_coord
                    points_with_implicit[point_id] = point_coord
        
        # Step 3: for each collinear group, add intersection points into the adjacency list
        for info in line_groups.values():
            adjacency = info["adjacency"]
            direction = info["direction"]
            
            # Find the segments belonging to the current collinear group
            group_segments = set()
            for seg in normalized_segments:
                a, b = seg
                if a in adjacency and b in adjacency.get(a, set()):
                    group_segments.add(seg)
            
            # Add intersections into the adjacency list
            for (seg1, seg2), point_list in all_intersections.items():
                # Check whether both segments belong to the current collinear group
                if seg1 not in group_segments and seg2 not in group_segments:
                    continue
                
                for point_id, point_coord, point_type in point_list:
                    # Check whether the point lies on some segment of the current collinear group
                    for seg in [seg1, seg2]:
                        if seg in group_segments:
                            a, b = seg
                            if a not in points or b not in points:
                                continue
                            # Check whether the point lies on the segment (not at an endpoint)
                            if self._point_on_segment(point_coord, points[a], points[b]):
                                # Add to the adjacency list (including implicit intersections)
                                adjacency[a].add(point_id)
                                adjacency[point_id].add(a)
                                adjacency[b].add(point_id)
                                adjacency[point_id].add(b)
                                break
        
        # Step 4: build connected components including implicit intersections (for determining atomic segments)
        chains_with_implicit: List[List[str]] = []
        for info in line_groups.values():
            adjacency = info["adjacency"]
            direction = info["direction"]
            visited: set[str] = set()
            for node in list(adjacency.keys()):
                if node in visited:
                    continue
                component = self._collect_line_component(node, adjacency, visited)
                if len(component) < 2:
                    continue
                # Sort using a point dictionary that includes implicit intersection points
                ordered = sorted(
                    component,
                    key=lambda name: self._project_onto_direction(
                        points_with_implicit[name], direction
                    ),
                )
                chains_with_implicit.append(ordered)
        
        # Step 5: from connected components including implicit intersections, keep only explicit points
        chains_explicit_only: List[List[str]] = []
        for chain in chains_with_implicit:
            # Filter out implicit intersection points (those whose names start with "__implicit_")
            explicit_chain = [point for point in chain if not point.startswith("__implicit_")]
            if len(explicit_chain) >= 2:
                chains_explicit_only.append(explicit_chain)
        
        # Save the version including implicit intersection points (used later when determining atomic segments)
        self._segment_chains_with_implicit = chains_with_implicit
        # Return the version that contains explicit intersection points only
        return chains_explicit_only, chains_with_implicit

    def _line_signature(
        self,
        p1: Tuple[float, float],
        p2: Tuple[float, float],
    ) -> Optional[Tuple[Tuple[float, float, float], Tuple[float, float]]]:
        dx = float(p2[0] - p1[0])
        dy = float(p2[1] - p1[1])
        length = math.hypot(dx, dy)
        if length < 1e-6:
            return None
        dir_x = dx / length
        dir_y = dy / length
        normal_x = -dir_y
        normal_y = dir_x
        offset = normal_x * float(p1[0]) + normal_y * float(p1[1])
        if normal_x < 0 or (abs(normal_x) < 1e-9 and normal_y < 0):
            normal_x *= -1
            normal_y *= -1
            offset *= -1
        line_key = (round(normal_x, 6), round(normal_y, 6), round(offset, 3))
        return line_key, (dir_x, dir_y)

    def _collect_line_component(
        self,
        start: str,
        adjacency: Dict[str, set[str]],
        visited: set[str],
    ) -> List[str]:
        stack = [start]
        component: List[str] = []
        while stack:
            node = stack.pop()
            if node in visited:
                continue
            visited.add(node)
            component.append(node)
            for neighbor in adjacency.get(node, set()):
                if neighbor not in visited:
                    stack.append(neighbor)
        return component

    def get_segment_chains_with_implicit(self) -> Optional[List[List[str]]]:
        """
        Get the segment chains that include implicit intersection points (used to determine atomic segments).
        
        Returns:
            A list of segment chains containing both explicit and implicit intersection points,
            or None if they have not been built yet.
        """
        return self._segment_chains_with_implicit

    def _project_onto_direction(
        self,
        point: Tuple[float, float],
        direction: Tuple[float, float],
    ) -> float:
        return point[0] * direction[0] + point[1] * direction[1]

    def _filter_annotations_for_atomic_elements(
        self,
        points: Dict[str, Tuple[int, int]],
        normalized_segments: List[Tuple[str, str]],
        annotations: Dict,
        segment_chains_with_implicit: List[List[str]],
    ) -> Tuple[Dict, Dict[str, Any]]:
        raw_annotations = annotations or {}
        sanitized_annotations = copy.deepcopy(raw_annotations)

        length_entries = list((raw_annotations.get("length_of_line") or []))
        filtered_lengths, skipped_lengths = self._filter_length_annotations(points, length_entries, segment_chains_with_implicit)
        sanitized_annotations["length_of_line"] = filtered_lengths

        direction_map = self._build_direction_map(points, normalized_segments)
        right_angle_keys = self._collect_right_angle_keys(points, raw_annotations)
        angle_entries = list((raw_annotations.get("measure_of_angle") or []))
        filtered_angles, skipped_angles = self._filter_angle_annotations(
            points,
            angle_entries,
            direction_map,
            right_angle_keys,
        )
        sanitized_annotations["measure_of_angle"] = filtered_angles

        # Apply filtering for `right_angles` as well
        right_angle_entries = list((raw_annotations.get("right_angles") or []))
        filtered_right_angles, skipped_right_angles = self._filter_right_angle_annotations(
            points, right_angle_entries
        )
        sanitized_annotations["right_angles"] = filtered_right_angles

        summary = {
            "length_of_line": {
                "applied": copy.deepcopy(filtered_lengths),
                "skipped": skipped_lengths,
            },
            "measure_of_angle": {
                "applied": copy.deepcopy(filtered_angles),
                "skipped": skipped_angles,
            },
            "right_angles": {
                "applied": copy.deepcopy(filtered_right_angles),
                "skipped": skipped_right_angles,
            },
        }
        return sanitized_annotations, summary

    def _filter_length_annotations(
        self,
        points: Dict[str, Tuple[int, int]],
        entries: List[Sequence],
        segment_chains_with_implicit: List[List[str]],
    ) -> Tuple[List[Sequence], List[Dict[str, Any]]]:
        applied: List[Sequence] = []
        skipped: List[Dict[str, Any]] = []
        for entry in entries:
            if not isinstance(entry, (list, tuple)) or len(entry) != 2:
                skipped.append({"entry": copy.deepcopy(entry), "reason": "invalid_entry"})
                continue
            segment, value = entry
            if not isinstance(segment, (list, tuple)) or len(segment) != 2:
                skipped.append({"entry": copy.deepcopy(entry), "reason": "invalid_segment"})
                continue
            a = str(segment[0]).lower()
            b = str(segment[1]).lower()
            if a not in points or b not in points:
                skipped.append({"entry": copy.deepcopy(entry), "reason": "missing_points"})
                continue
            if self._is_segment_atomic(a, b, points, segment_chains_with_implicit):
                applied.append(entry)
            else:
                skipped.append({"entry": copy.deepcopy(entry), "reason": "non_atomic_segment"})
        return applied, skipped

    def _filter_angle_annotations(
        self,
        points: Dict[str, Tuple[int, int]],
        entries: List[Sequence],
        direction_map: Dict[str, List[float]],
        right_angle_keys: set[RightAngleKey],
    ) -> Tuple[List[Sequence], List[Dict[str, Any]]]:
        applied: List[Sequence] = []
        skipped: List[Dict[str, Any]] = []
        for entry in entries:
            if not isinstance(entry, (list, tuple)) or len(entry) != 2:
                skipped.append({"entry": copy.deepcopy(entry), "reason": "invalid_entry"})
                continue
            triple = entry[0]
            if not isinstance(triple, (list, tuple)) or len(triple) != 3:
                skipped.append({"entry": copy.deepcopy(entry), "reason": "invalid_angle"})
                continue
            a, b, c = (str(triple[0]).lower(), str(triple[1]).lower(), str(triple[2]).lower())
            if any(name not in points for name in (a, b, c)):
                skipped.append({"entry": copy.deepcopy(entry), "reason": "missing_points"})
                continue
            pair_key = self._build_right_angle_pair_key(points, a, b, c)
            if pair_key and pair_key in right_angle_keys:
                applied.append(entry)
                continue
            if self._is_angle_atomic(a, b, c, points, direction_map):
                applied.append(entry)
            else:
                skipped.append({"entry": copy.deepcopy(entry), "reason": "non_atomic_angle"})
        return applied, skipped

    def _filter_right_angle_annotations(
        self,
        points: Dict[str, Tuple[int, int]],
        entries: List[Sequence],
    ) -> Tuple[List[Sequence], List[Dict[str, Any]]]:
        """
        Filter right-angle annotations so that, for each vertex, only the first encountered right angle is kept
        (deduplication).
        
        Returns:
            (applied, skipped): list of applied right angles and list of skipped ones (with reasons).
        """
        applied: List[Sequence] = []
        skipped: List[Dict[str, Any]] = []
        seen_vertices: set[str] = set()  # Track processed vertices (for deduplication)
        
        for entry in entries:
            if not isinstance(entry, (list, tuple)) or len(entry) != 3:
                skipped.append({"entry": copy.deepcopy(entry), "reason": "invalid_entry"})
                continue
            
            a, b, c = (str(p).lower() for p in entry)
            if any(p not in points for p in (a, b, c)):
                skipped.append({"entry": copy.deepcopy(entry), "reason": "missing_points"})
                continue
            
            # Check whether `pair_key` is valid
            pair_key = self._build_right_angle_pair_key(points, a, b, c)
            if pair_key is None:
                skipped.append({"entry": copy.deepcopy(entry), "reason": "invalid_angle"})
                continue
            
            # Check vector lengths
            xa, ya = points[a]
            xb, yb = points[b]
            xc, yc = points[c]
            vba = np.array([xa - xb, ya - yb], dtype=float)
            vbc = np.array([xc - xb, yc - yb], dtype=float)
            norm_ba = np.linalg.norm(vba)
            norm_bc = np.linalg.norm(vbc)
            if norm_ba < 1 or norm_bc < 1:
                skipped.append({"entry": copy.deepcopy(entry), "reason": "zero_length_vector"})
                continue
            
            # Deduplicate: for each vertex, keep only the first encountered right angle
            if b in seen_vertices:
                skipped.append({"entry": copy.deepcopy(entry), "reason": "duplicate_vertex"})
                continue
            
            seen_vertices.add(b)
            applied.append(copy.deepcopy(entry))
        
        return applied, skipped

    def _is_segment_atomic(
        self,
        a: str,
        b: str,
        points: Dict[str, Tuple[int, int]],
        segment_chains_with_implicit: List[List[str]],
    ) -> bool:
        start = points.get(a)
        end = points.get(b)
        if start is None or end is None:
            return False
        for chain in segment_chains_with_implicit:
            if a in chain and b in chain:
                return ",".join(a) + "," + ",".join(b) in ",".join(chain) or ",".join(b) + "," + ",".join(a) in ",".join(chain)
        return False

    def _build_direction_map(
        self,
        points: Dict[str, Tuple[int, int]],
        normalized_segments: List[Tuple[str, str]],
    ) -> Dict[str, List[float]]:
        adjacency: Dict[str, List[str]] = defaultdict(list)
        for a, b in normalized_segments:
            adjacency[a].append(b)
            adjacency[b].append(a)
        direction_map: Dict[str, List[float]] = {}
        for vertex, neighbors in adjacency.items():
            directions: Dict[Tuple[float, float], Dict[str, float]] = {}
            for neighbor in neighbors:
                vx = float(points[neighbor][0] - points[vertex][0])
                vy = float(points[neighbor][1] - points[vertex][1])
                length = math.hypot(vx, vy)
                if length < 1e-6:
                    continue
                dir_x = vx / length
                dir_y = vy / length
                key = (round(dir_x, 6), round(dir_y, 6))
                angle = math.atan2(dir_y, dir_x) % (2 * math.pi)
                existing = directions.get(key)
                if existing is None or length < existing.get("distance", float("inf")):
                    directions[key] = {"angle": angle, "distance": length}
            direction_map[vertex] = [info["angle"] for info in directions.values()]
        return direction_map

    def _is_angle_atomic(
        self,
        a: str,
        b: str,
        c: str,
        points: Dict[str, Tuple[int, int]],
        direction_map: Dict[str, List[float]],
    ) -> bool:
        if any(name not in points for name in (a, b, c)):
            return False
        ba = (points[a][0] - points[b][0], points[a][1] - points[b][1])
        bc = (points[c][0] - points[b][0], points[c][1] - points[b][1])
        len_ba = math.hypot(*ba)
        len_bc = math.hypot(*bc)
        if len_ba < 1e-6 or len_bc < 1e-6:
            return True
        angle_a = math.atan2(ba[1], ba[0]) % (2 * math.pi)
        angle_c = math.atan2(bc[1], bc[0]) % (2 * math.pi)
        diff = (angle_c - angle_a) % (2 * math.pi)
        if diff < 1e-6:
            return True
        if diff > math.pi:
            angle_a, angle_c = angle_c, angle_a
            diff = (angle_c - angle_a) % (2 * math.pi)
        angle_tol = math.radians(1.0)
        for other in direction_map.get(b, []):
            rel = (other - angle_a) % (2 * math.pi)
            if rel < angle_tol or abs(rel - diff) < angle_tol:
                continue
            if rel < diff - angle_tol:
                return False
        return True

    # ------------------------------------------------------------------ #
    # Normalisation                                                      #
    # ------------------------------------------------------------------ #
    def _normalize_coordinates(
        self, points: Dict[str, Sequence[float]]
    ) -> Tuple[Dict[str, Tuple[int, int]], np.ndarray, float]:
        pts: List[Tuple[str, float, float]] = []
        for name, value in points.items():
            if (
                isinstance(value, (list, tuple))
                and len(value) == 2
                and all(isinstance(v, (int, float)) for v in value)
            ):
                pts.append((name, float(value[0]), float(value[1])))

        if not pts:
            raise ValueError("points must contain valid numeric points")

        xs = [p[1] for p in pts]
        ys = [p[2] for p in pts]
        min_x, max_x = min(xs), max(xs)
        min_y, max_y = min(ys), max(ys)
        span_x = max(max_x - min_x, 1e-6)
        span_y = max(max_y - min_y, 1e-6)
        span = max(span_x, span_y)

        canvas_size = int(max(self.min_canvas, min(self.max_canvas, span * 120)))
        padding = int(canvas_size * 0.12)
        width = height = canvas_size + 2 * padding

        scale = canvas_size / span
        occupied_w = span_x * scale
        occupied_h = span_y * scale
        offset_x = (canvas_size - occupied_w) / 2.0
        offset_y = (canvas_size - occupied_h) / 2.0

        norm_points: Dict[str, Tuple[int, int]] = {}
        for name, x, y in pts:
            px = int(
                round(padding + offset_x + (x - min_x) * scale)
            )
            py = int(
                round(padding + offset_y + (max_y - y) * scale)
            )
            norm_points[name.lower()] = (px, py)

        canvas = self._create_canvas(height, width)
        self._point_centroid = self._compute_point_centroid(
            norm_points, (width / 2.0, height / 2.0)
        )
        return norm_points, canvas, scale

    def _create_canvas(self, height: int, width: int) -> np.ndarray:
        base = np.full((height, width, 3), self.style.background_color, dtype=np.uint8)
        if self.style.background_gradient:
            top = np.array(self.style.background_gradient_color, dtype=np.float32)
            bottom = np.array(self.style.background_color, dtype=np.float32)
            gradient = np.linspace(0.0, 1.0, height, dtype=np.float32).reshape(height, 1, 1)
            blended = (top * (1.0 - gradient) + bottom * gradient).astype(np.uint8)
            base[:] = blended

        if self.style.grid_spacing > 0 and self.style.grid_alpha > 0:
            self._draw_grid(base)
        return base

    def _draw_grid(self, canvas: np.ndarray) -> None:
        spacing = max(12, int(self.style.grid_spacing))
        overlay = canvas.copy()
        height, width = canvas.shape[:2]
        for x in range(spacing, width, spacing):
            cv2.line(overlay, (x, 0), (x, height), self.style.grid_color, 1, cv2.LINE_AA)
        for y in range(spacing, height, spacing):
            cv2.line(overlay, (0, y), (width, y), self.style.grid_color, 1, cv2.LINE_AA)
        alpha = float(np.clip(self.style.grid_alpha, 0.0, 1.0))
        cv2.addWeighted(overlay, alpha, canvas, 1 - alpha, 0, dst=canvas)

    def _compute_point_centroid(
        self,
        points: Dict[str, Tuple[int, int]],
        fallback: Tuple[float, float],
    ) -> Tuple[float, float]:
        if not points:
            return fallback
        xs = [pt[0] for pt in points.values()]
        ys = [pt[1] for pt in points.values()]
        return (float(sum(xs) / len(xs)), float(sum(ys) / len(ys)))

    # ------------------------------------------------------------------ #
    # Region fills                                                       #
    # ------------------------------------------------------------------ #
    def _apply_region_fills(
        self,
        canvas: np.ndarray,
        points: Dict[str, Tuple[int, int]],
        plotting_code: Dict,
        annotations: Dict,
    ) -> None:
        fill_specs = self._resolve_fill_specs(plotting_code, annotations)
        if fill_specs:
            for idx, spec in enumerate(fill_specs):
                color = spec.get("color")
                palette = self.style.fill_palette or (self.style.fill_color,)
                if isinstance(color, str):
                    fill_color = self._hex_to_bgr(color)
                elif isinstance(color, (list, tuple)):
                    fill_color = tuple(int(c) for c in color)
                else:
                    fill_color = palette[idx % len(palette)]
                opacity = float(spec.get("opacity", self.style.fill_opacity))
                kind = (spec.get("type") or "polygon").lower()
                if kind == "polygon":
                    self._fill_polygon(canvas, points, spec.get("points") or [], fill_color, opacity)
                elif kind == "circle":
                    self._fill_circle(canvas, points, spec, fill_color, opacity)
        elif len(points) >= 3:
            self._fill_convex_hull(canvas, points, self.style.fill_color, self.style.fill_opacity)

    def _resolve_fill_specs(self, plotting_code: Dict, annotations: Dict) -> List[Dict]:
        raw = (
            plotting_code.get("fill_entities")
            or plotting_code.get("filled_entities")
            or (annotations or {}).get("filled_entities")
            or (annotations or {}).get("filled_entity")
        )
        if not raw:
            return []

        def _as_list(candidate) -> List:
            if isinstance(candidate, list):
                return candidate
            if isinstance(candidate, tuple):
                return list(candidate)
            return [candidate]

        specs: List[Dict] = []
        for entry in _as_list(raw):
            if isinstance(entry, dict):
                spec = entry.copy()
                spec["type"] = (spec.get("type") or "polygon").lower()
                specs.append(spec)
            elif isinstance(entry, (list, tuple)):
                if entry and all(isinstance(i, str) for i in entry):
                    specs.append({"type": "polygon", "points": list(entry)})
                else:
                    for poly in entry:
                        if isinstance(poly, (list, tuple)) and len(poly) >= 3:
                            specs.append({"type": "polygon", "points": list(poly)})
            elif isinstance(entry, str):
                specs.append({"type": "polygon", "points": [entry]})
        return specs

    def _fill_polygon(
        self,
        canvas: np.ndarray,
        points: Dict[str, Tuple[int, int]],
        names: Sequence[str],
        color: Tuple[int, int, int],
        opacity: float,
    ) -> None:
        coords = []
        for name in names:
            key = str(name).lower()
            if key in points:
                coords.append(points[key])
        if len(coords) < 3:
            return
        overlay = canvas.copy()
        cv2.fillPoly(overlay, [np.array(coords, dtype=np.int32)], color)
        self._blend_overlay(canvas, overlay, opacity)

    def _fill_circle(
        self,
        canvas: np.ndarray,
        points: Dict[str, Tuple[int, int]],
        spec: Dict,
        color: Tuple[int, int, int],
        opacity: float,
    ) -> None:
        center_name = spec.get("center")
        if center_name is None:
            return
        key = str(center_name).lower()
        if key not in points:
            return
        center = points[key]
        radius = spec.get("radius")
        if radius is None:
            through = spec.get("through")
            if through is None:
                return
            t_key = str(through).lower()
            if t_key not in points:
                return
            radius = math.dist(points[t_key], center)
        radius = max(3, int(radius))
        overlay = canvas.copy()
        cv2.circle(overlay, center, radius, color, thickness=-1, lineType=cv2.LINE_AA)
        self._blend_overlay(canvas, overlay, opacity)

    def _fill_convex_hull(
        self,
        canvas: np.ndarray,
        points: Dict[str, Tuple[int, int]],
        color: Tuple[int, int, int],
        opacity: float,
    ) -> None:
        if len(points) < 3:
            return
        pts = np.array(list(points.values()), dtype=np.int32)
        hull = cv2.convexHull(pts)
        overlay = canvas.copy()
        cv2.fillPoly(overlay, [hull], color)
        self._blend_overlay(canvas, overlay, opacity)

    def _blend_overlay(
        self, canvas: np.ndarray, overlay: np.ndarray, alpha: float
    ) -> None:
        alpha = float(np.clip(alpha, 0.0, 1.0))
        cv2.addWeighted(overlay, alpha, canvas, 1 - alpha, 0, dst=canvas)

    # ------------------------------------------------------------------ #
    # Base drawing                                                       #
    # ------------------------------------------------------------------ #
    def _create_segment_boxes(
        self,
        points: Dict[str, Tuple[int, int]],
        segments: Iterable[Sequence[PointName]],
    ) -> None:
        """Create narrow rectangle boxes for segments for overlap detection."""
        for segment in segments:
            if not segment or len(segment) != 2:
                continue
            p1, p2 = segment
            key1, key2 = str(p1).lower(), str(p2).lower()
            if key1 not in points or key2 not in points:
                continue
            
            x1, y1 = points[key1]
            x2, y2 = points[key2]
            # Compute the segment direction and its perpendicular direction
            dx = x2 - x1
            dy = y2 - y1
            length = math.sqrt(dx * dx + dy * dy)
            if length > 0:
                # Detection rectangle for the segment: use a relatively large width to keep annotations
                # from overlapping with the segment. The visual line width is usually thin, but a wider
                # detection width (e.g. 20px) gives better avoidance behavior.
                detection_width = max(self.style.line_width * 2, 20.0)
                
                # Perpendicular direction
                perp_x = -dy / length
                perp_y = dx / length
                # Compute the four corner points of the rectangle
                half_width = detection_width / 2.0
                corners = [
                    (x1 + perp_x * half_width, y1 + perp_y * half_width),
                    (x1 - perp_x * half_width, y1 - perp_y * half_width),
                    (x2 - perp_x * half_width, y2 - perp_y * half_width),
                    (x2 + perp_x * half_width, y2 + perp_y * half_width),
                ]
                # Compute the bounding box
                xs = [int(c[0]) for c in corners]
                ys = [int(c[1]) for c in corners]
                segment_box = (min(xs), min(ys), max(xs), max(ys))
                self._segment_boxes.append(segment_box)
    
    def _draw_segments(
        self,
        canvas: np.ndarray,
        points: Dict[str, Tuple[int, int]],
        segments: Iterable[Sequence[PointName]],
    ) -> None:
        palette = self.style.line_palette or (self.style.line_color,)
        for idx, segment in enumerate(segments):
            if not segment or len(segment) != 2:
                continue
            p1, p2 = segment
            key1, key2 = str(p1).lower(), str(p2).lower()
            if key1 not in points or key2 not in points:
                continue
            color = palette[idx % len(palette)]
            self._draw_line_with_glow(canvas, points[key1], points[key2], color)

    def _create_circle_boxes(
        self,
        points: Dict[str, Tuple[int, int]],
        circles: Iterable[Sequence],
        circle_info_list: Optional[List[Tuple[Tuple[int, int], float]]] = None,
    ) -> None:
        """Create circle boxes for circles for overlap detection."""
        # If `circle_info_list` is provided, use it directly
        if circle_info_list:
            for center, radius in circle_info_list:
                radius_int = max(3, int(radius))
                self._circle_boxes.append((center[0], center[1], radius_int))
            return
        
        # Otherwise, parse from `circles` (backwards compatibility)
        for circle in circles:
            if not circle or len(circle) < 2:
                continue
            center_name = str(circle[0]).lower()
            if center_name not in points:
                continue
            center = points[center_name]
            radius = None
            descriptor = circle[1]
            if isinstance(descriptor, (int, float)):
                radius = max(3, int(descriptor))
            else:
                candidate = str(descriptor).lower()
                if candidate in points:
                    radius = max(3, int(math.dist(points[candidate], center)))
            if radius is None:
                continue
            
            # Create a circle box for overlap detection (on the circle boundary)
            # Here `radius` should be the visual radius of the circle; no need to add extra padding,
            # because the detection logic will handle padding.
            circle_box_radius = radius
            self._circle_boxes.append((center[0], center[1], circle_box_radius))
    
    def _calculate_circle_info(
        self,
        points: Dict[str, Tuple[int, int]],
        circles: Iterable[Sequence],
        scale: float,
    ) -> List[Tuple[Tuple[int, int], float]]:
        """
        Compute centers and radii for all circles (first stage).
        
        Note: the order of checks is important; circles defined by diameter must be processed
        before circles defined by three points.
        
        Returns:
            List[Tuple[Tuple[int, int], float]]: list of (center, radius).
        """
        circle_info_list: List[Tuple[Tuple[int, int], float]] = []
        for circle in circles:
            if not circle or not isinstance(circle, (list, tuple)):
                continue
            center = None
            radius = None
            circle = circle[1:]
            # Format 1: diameter [["A", "B", "diameter"]] - must be checked first
            if len(circle) == 3 and circle[2] == "diameter":
                p1_name = str(circle[0]).lower()
                p2_name = str(circle[1]).lower()
                if p1_name in points and p2_name in points:
                    p1 = points[p1_name]
                    p2 = points[p2_name]
                    center = (
                        int((p1[0] + p2[0]) / 2),
                        int((p1[1] + p2[1]) / 2)
                    )
                    radius = float(math.dist(p1, p2) / 2)
            
            # Format 2: three-point circle [["A", "B", "C"]] - second check
            elif len(circle) == 3 and all(isinstance(p, str) for p in circle):
                p1, p2, p3 = [str(p).lower() for p in circle]
                if p1 in points and p2 in points and p3 in points:
                    center_tuple, radius_float = self._calculate_circle_from_three_points(
                        points[p1], points[p2], points[p3]
                    )
                    if center_tuple and radius_float:
                        center = (int(center_tuple[0]), int(center_tuple[1]))
                        radius = float(radius_float)
            # Format 3/4: center + radius [["O", 5]] or [["O", "P"]] - third check
            elif len(circle) == 2:
                center_name = str(circle[0]).lower()
                if center_name in points:
                    center = points[center_name]
                    descriptor = circle[1]
                    if isinstance(descriptor, (int, float)) or (isinstance(descriptor, str) and isdigit(descriptor)):
                        # Scale a numeric radius so that it matches the canvas size
                        radius = float(descriptor) * scale
                    else:
                        candidate = str(descriptor).lower()
                        if candidate in points:
                            radius = float(math.dist(points[candidate], center))
            
            if center and radius is not None:
                circle_info_list.append((center, radius))
        return circle_info_list

    def _draw_circles(
        self,
        canvas: np.ndarray,
        circle_info_list: List[Tuple[Tuple[int, int], float]],
        points: Optional[Dict[str, Tuple[int, int]]] = None,
        circles: Optional[Iterable[Sequence]] = None,
    ) -> None:
        """
        Draw circles (third stage).
        
        Args:
            canvas: Canvas image.
            circle_info_list: List of circle centers and radii [(center, radius), ...].
            points: Dictionary of point coordinates (used to draw circle centers).
            circles: Original circle data (used to obtain center point names).
        """
        # Collect the circle-center points that need to be drawn
        center_points_to_draw: List[Tuple[int, int]] = []
        
        if points is not None and circles is not None:
            # Extract center point names from `circles`
            for circle in circles:
                if not circle or not isinstance(circle, (list, tuple)) or len(circle) < 2:
                    continue
                # Format: [["C1", "E", 6]] or [["C2", "A", 10]]
                # The second element is the center point name
                center_name = str(circle[1]).lower()
                if center_name in points:
                    center_points_to_draw.append(points[center_name])
        
        for center, radius in circle_info_list:
            radius_int = max(3, int(radius))
            
            # Draw filled circle (if transparency is set)
            if self.style.circle_fill_alpha > 0:
                overlay = canvas.copy()
                cv2.circle(
                    overlay,
                    center,
                    radius_int,
                    self.style.circle_color,
                    thickness=-1,
                    lineType=cv2.LINE_AA,
                )
                self._blend_overlay(canvas, overlay, self.style.circle_fill_alpha)
            
            # Draw circle outline
            cv2.circle(
                canvas,
                center,
                radius_int,
                self.style.circle_color,
                self.style.line_width,
                lineType=cv2.LINE_AA,
            )
        
        # Draw center points (on top of the circles to ensure visibility)
        for center_point in center_points_to_draw:
            if self.style.point_outline_color:
                cv2.circle(
                    canvas,
                    center_point,
                    self.style.point_radius + self.style.point_outline_width,
                    self.style.point_outline_color,
                    thickness=-1,
                    lineType=cv2.LINE_AA,
                )
            cv2.circle(
                canvas,
                center_point,
                self.style.point_radius,
                self.style.point_color,
                thickness=-1,
                lineType=cv2.LINE_AA,
            )
    
    @staticmethod
    def _calculate_circle_from_three_points(
        p1: Tuple[int, int],
        p2: Tuple[int, int],
        p3: Tuple[int, int],
    ) -> Tuple[Optional[Tuple[float, float]], Optional[float]]:
        """Compute the center and radius of a circle defined by three points."""
        # Use the intersection of two perpendicular bisectors as the circle center.
        # Perpendicular bisector 1: passes through the midpoint of p1p2 and is perpendicular to p1p2
        mid1 = [(p1[0] + p2[0]) / 2, (p1[1] + p2[1]) / 2]
        dir1 = [p2[0] - p1[0], p2[1] - p1[1]]
        # Perpendicular direction
        perp1 = [-dir1[1], dir1[0]]
        
        # Perpendicular bisector 2: passes through the midpoint of p2p3 and is perpendicular to p2p3
        mid2 = [(p2[0] + p3[0]) / 2, (p2[1] + p3[1]) / 2]
        dir2 = [p3[0] - p2[0], p3[1] - p2[1]]
        perp2 = [-dir2[1], dir2[0]]
        
        # Solve for the intersection of the two lines
        det = perp1[0] * perp2[1] - perp1[1] * perp2[0]
        if abs(det) < 1e-10:
            # Use the perpendicular bisector of p1p3 instead
            mid2 = [(p1[0] + p3[0]) / 2, (p1[1] + p3[1]) / 2]
            dir2 = [p3[0] - p1[0], p3[1] - p1[1]]
            perp2 = [-dir2[1], dir2[0]]
            det = perp1[0] * perp2[1] - perp1[1] * perp2[0]
        
        if abs(det) < 1e-10:
            # The three points are collinear; the circle is undefined
            return None, None
        
        # Solve for t
        dx = mid2[0] - mid1[0]
        dy = mid2[1] - mid1[1]
        t = (dx * perp2[1] - dy * perp2[0]) / det
        
        center = (mid1[0] + t * perp1[0], mid1[1] + t * perp1[1])
        radius = math.sqrt((center[0] - p1[0])**2 + (center[1] - p1[1])**2)
        
        return center, radius

    def _get_text_size(
        self,
        text: str,
        font_face: int,
        font_scale: float,
        thickness: int,
    ) -> Tuple[int, int, int]:
        """
        Calculate text size, preferring PIL when available.
        Returns: (width, full_height, baseline_offset_from_top)
        """
        font_size = max(10, int(18 * font_scale))
        try:
            font = ImageFont.truetype("DejaVuSans.ttf", font_size)
        except Exception:
            font = ImageFont.load_default()
        
        bbox = font.getbbox(text)
        width = bbox[2] - bbox[0] + 2
        height = bbox[3] - bbox[1] + 2
        
        # Roughly estimate the baseline position
        return width, height, int(height * 0.8)

    def _draw_points(
        self,
        canvas: np.ndarray,
        points: Dict[str, Tuple[int, int]],
    ) -> None:
        """Draw all points (without labels)."""
        for name, (px, py) in points.items():
            if self.style.point_outline_color:
                cv2.circle(
                    canvas,
                    (px, py),
                    self.style.point_radius + self.style.point_outline_width,
                    self.style.point_outline_color,
                    thickness=-1,
                    lineType=cv2.LINE_AA,
                )
            cv2.circle(
                canvas,
                (px, py),
                self.style.point_radius,
                self.style.point_color,
                thickness=-1,
                lineType=cv2.LINE_AA,
            )

    def _draw_point_labels(
        self,
        canvas: np.ndarray,
        points: Dict[str, Tuple[int, int]],
        segments: Iterable[Sequence[PointName]],
    ) -> None:
        """Draw point labels (should be called after all other content to allow overlap detection)."""
        centroid = self._point_centroid or (
            canvas.shape[1] / 2.0,
            canvas.shape[0] / 2.0,
        )
        min_dim = float(min(canvas.shape[:2]))
        base_scale = max(
            self.style.label_font_min_scale,
            min(self.style.label_font_max_scale, min_dim / 420.0),
        )
        base_scale *= max(0.1, self.style.label_font_scale_multiplier)
        font_face = self.style.label_font_face
        thickness = max(1, int(round(base_scale * self.style.label_font_thickness)))

        for name, (px, py) in points.items():
            label = name.upper()
            # If the label is of the form O_xxx, keep only 'O'
            if label.startswith('O_'):
                label = 'O'
            text_w, text_h, baseline = self._get_text_size(label, font_face, base_scale, thickness)
            
            pad = max(0, self.style.label_padding)
            point_pos = (px, py)
            
            # Use pixel-based sampling to find the best label position (similar to plotter_reference.py)
            base_distance = self.style.point_radius + max(8, self.style.label_distance * 0.6)
            radiuses = [
                int(base_distance),
                int(base_distance * 1.2),
                int(base_distance * 1.5),
            ]
            
            text_pos, rect = self._find_best_position_by_pixels(
                canvas=canvas,
                text_size=(text_w, text_h),
                center_pos=point_pos,
                radiuses=radiuses,
                n_samples_per_radius=20,
                background_color=self.style.background_color,
                padding=pad,
            )
            
            anchor_x, anchor_y = text_pos
            
            anchor_x = rect[0] + pad
            anchor_y = rect[1] + pad
            baseline_adjusted = anchor_y + text_h
            self._label_boxes.append(rect)

            if self.style.label_background_color is not None:
                top_left = (rect[0], rect[1])
                bottom_right = (rect[2], rect[3])
                cv2.rectangle(
                    canvas,
                    top_left,
                    bottom_right,
                    self.style.label_background_color,
                    thickness=-1,
                )
                if self.style.label_border_color is not None:
                    cv2.rectangle(
                        canvas,
                        top_left,
                        bottom_right,
                        self.style.label_border_color,
                        thickness=1,
                    )

            # Draw text
            self._draw_text_common(
                canvas, 
                label, 
                (anchor_x, anchor_y), 
                font_face, 
                base_scale, 
                self.style.label_color, 
                thickness,
                shadow_color=self.style.label_shadow_color,
                shadow_offset=self.style.label_shadow_offset
            )


    # ------------------------------------------------------------------ #
    # Annotation rendering                                               #
    # ------------------------------------------------------------------ #
    def _draw_annotations(
        self,
        canvas: np.ndarray,
        points: Dict[str, Tuple[int, int]],
        annotations: Dict,
        segments: Optional[Iterable[Sequence]] = None,
        segment_chains: Optional[List[List[str]]] = None,
    ) -> None:
        if not annotations:
            return

        right_angle_keys = self._collect_right_angle_keys(points, annotations)
        # Draw right angles directly; no need to return applied/skipped (they were filtered earlier)
        self._plot_right_angles(
            canvas, points, annotations.get("right_angles") or [], segment_chains
        )
        # Remove eq_lines annotation handling
        # self._plot_equal_lines(canvas, points, annotations.get("eq_lines") or [])
        self._plot_lengths(canvas, points, annotations.get("length_of_line") or [])
        self._plot_angle_measures(
            canvas,
            points,
            annotations.get("measure_of_angle") or [],
            right_angle_keys,
            segments=segments,
        )

    def _plot_right_angles(
        self,
        canvas: np.ndarray,
        points: Dict[str, Tuple[int, int]],
        right_angles: Iterable[Sequence[PointName]],
        segment_chains: Optional[List[List[str]]] = None,
    ) -> None:
        """
        Draw right-angle markers (note: `right_angles` have already been filtered earlier).
        """
        min_dim = float(min(canvas.shape[:2]))
        # Slightly reduce the size of right-angle markers to make them look more refined
        arm = max(12, int(min_dim / 25))

        for triple in right_angles:
            if len(triple) != 3:
                continue
            a, b, c = (str(p).lower() for p in triple)
            if any(p not in points for p in (a, b, c)):
                continue
            xa, ya = points[a]
            xb, yb = points[b]
            xc, yc = points[c]
            vba = np.array([xa - xb, ya - yb], dtype=float)
            vbc = np.array([xc - xb, yc - yb], dtype=float)
            pair_key = self._angle_direction_pair_key_from_vectors(b, vba, vbc)
            if pair_key is None:
                continue
            norm_ba = np.linalg.norm(vba)
            norm_bc = np.linalg.norm(vbc)
            if norm_ba < 1 or norm_bc < 1:
                continue
            vba /= norm_ba
            vbc /= norm_bc
            
            p1 = (int(xb + vba[0] * arm), int(yb + vba[1] * arm))
            p2 = (int(xb + vbc[0] * arm), int(yb + vbc[1] * arm))
            p3 = (int(p1[0] + vbc[0] * arm), int(p1[1] + vbc[1] * arm))
            cv2.line(canvas, p1, p3, self.style.annotation_color, self.style.annotation_width, cv2.LINE_AA)
            cv2.line(canvas, p2, p3, self.style.annotation_color, self.style.annotation_width, cv2.LINE_AA)

    def _plot_equal_lines(
        self,
        canvas: np.ndarray,
        points: Dict[str, Tuple[int, int]],
        equal_lines: Iterable[Sequence[Sequence[PointName]]],
    ) -> None:
        for idx, group in enumerate(equal_lines):
            segments = self._normalize_segment_group(group)
            if not segments:
                continue
            tick_pattern = min(3, (idx % 3) + 1)
            tick_offsets = self._equal_tick_offsets(tick_pattern)
            mark = max(6, self.style.equal_tick_length)
            for segment in segments:
                p1, p2 = (str(p).lower() for p in segment)
                if p1 not in points or p2 not in points:
                    continue
                x1, y1 = points[p1]
                x2, y2 = points[p2]
                vec = np.array([x2 - x1, y2 - y1], dtype=float)
                length = np.linalg.norm(vec)
                if length < 5:
                    continue
                direction = vec / length
                normal = np.array([-direction[1], direction[0]])
                mid = np.array([(x1 + x2) / 2.0, (y1 + y2) / 2.0])
                for offset_along_line in tick_offsets:
                    base = mid + direction * offset_along_line
                    start = (
                        int(base[0] + normal[0] * mark),
                        int(base[1] + normal[1] * mark),
                    )
                    end = (
                        int(base[0] - normal[0] * mark),
                        int(base[1] - normal[1] * mark),
                    )
                    cv2.line(
                        canvas,
                        start,
                        end,
                        self.style.annotation_color,
                        self.style.annotation_width,
                        lineType=cv2.LINE_AA,
                    )

    def _plot_equal_angles(
        self,
        canvas: np.ndarray,
        points: Dict[str, Tuple[int, int]],
        equal_angles: Iterable[Sequence[Sequence[PointName]]],
    ) -> None:
        for idx, group in enumerate(equal_angles):
            triples = self._normalize_angle_group(group)
            if not triples:
                continue
            for local_offset, triple in enumerate(triples):
                a, b, c = (str(p).lower() for p in triple)
                if any(p not in points for p in (a, b, c)):
                    continue
                shrink = idx * 2 + local_offset
                self._draw_angle_arc(canvas, points[a], points[b], points[c], shrink=shrink)

    def _draw_angle_arc(
        self,
        canvas: np.ndarray,
        pa: Tuple[int, int],
        pb: Tuple[int, int],
        pc: Tuple[int, int],
        shrink: float = 0,
    ) -> None:
        """
        Draw the angle ∠ABC, where B = pb, A = pa, C = pc.
        The arc lies strictly within the angle between BA and BC, and the smaller arc is chosen.
        """
        xa, ya = pa
        xb, yb = pb
        xc, yc = pc
        
        # Vectors BA and BC
        BA = np.array([xa - xb, ya - yb], dtype=float)
        BC = np.array([xc - xb, yc - yb], dtype=float)
        len_BA = np.linalg.norm(BA)
        len_BC = np.linalg.norm(BC)
        if len_BA < 1e-3 or len_BC < 1e-3:
            return

        radius = self._angle_arc_radius_from_lengths(len_BA, len_BC, shrink)
        if radius <= 8:
            return

        # Use the cross product to decide the order of A and C, ensuring BA × BC >= 0
        # (consistent with the OpenCV coordinate direction)
        cross = (xa - xb) * (yc - yb) - (ya - yb) * (xc - xb)
        if cross < 0:
            xa, ya, xc, yc = xc, yc, xa, ya

        angle_BC = math.degrees(math.atan2(yc - yb, xc - xb))
        angle_BA = math.degrees(math.atan2(ya - yb, xa - xb))

        # Normalize angles into [0, 360)
        if angle_BC < 0:
            angle_BC += 360
        if angle_BA < 0:
            angle_BA += 360

        # Ensure that we go from BA to BC in the counter-clockwise direction
        if angle_BC < angle_BA:
            angle_BA -= 360

        start_angle = angle_BA
        end_angle = angle_BC

        # Choose the smaller arc to avoid wrapping around the large side
        sweep = abs(end_angle - start_angle)
        if sweep > 180:
            start_angle, end_angle = end_angle, start_angle

        cv2.ellipse(
            canvas,
            (int(xb), int(yb)),
            (radius, radius),
            0,
            start_angle,
            end_angle,
            self.style.annotation_color,
            self.style.annotation_width,
            lineType=cv2.LINE_AA,
        )

    def _plot_lengths(
        self,
        canvas: np.ndarray,
        points: Dict[str, Tuple[int, int]],
        length_annotations: Iterable[Sequence],
    ) -> None:
        text_jobs: List[Tuple[str, Tuple[float, float], Tuple[int, int, int, int]]] = []

        for entry in length_annotations:
            if not entry or len(entry) != 2:
                continue
            segment, value = entry
            if not isinstance(segment, (list, tuple)) or len(segment) != 2:
                continue
            p1, p2 = (str(p).lower() for p in segment)
            if p1 not in points or p2 not in points:
                continue
            x1, y1 = points[p1]
            x2, y2 = points[p2]
            vec = np.array([x2 - x1, y2 - y1], dtype=float)
            length = np.linalg.norm(vec)
            if length < 1:
                continue
            direction = vec / length
            normal = np.array([-direction[1], direction[0]])
            # Ensure that the annotation lies exactly at the segment midpoint
            mid = np.array([(x1 + x2) / 2.0, (y1 + y2) / 2.0])
            self._draw_dimension_ticks(canvas, (x1, y1), (x2, y2))
            
            # Decide on which side of the segment to place the annotation: the side away from the figure center
            centroid = np.array(
                self._point_centroid
                or (canvas.shape[1] / 2.0, canvas.shape[0] / 2.0)
            )
            # Vector from the segment midpoint to the figure center
            to_centroid = centroid - mid
            # If the normal points toward the center, flip it (so the annotation is away from the center)
            if np.dot(normal, to_centroid) > 0:
                normal = -normal

            # Initial offset distance (perpendicular to the segment).
            # Use a slightly smaller distance so the annotation is closer to the segment.
            base_offset = max(20, self.style.label_distance * 0.8)
            
            # Pre-compute the text rectangle used for overlap detection
            text = self._sanitize_length_text(str(value))
            # Compute text size using the same logic as `_draw_annotation_text`
            min_dim = float(min(canvas.shape[:2]))
            base_scale = max(
                self.style.label_font_min_scale,
                min(self.style.label_font_max_scale, min_dim / 420.0),
            )
            base_scale *= max(0.1, self.style.annotation_font_scale_multiplier)
            font_face = self.style.label_font_face
            thickness = max(1, int(round(base_scale * self.style.label_font_thickness)))
            text_w, text_h, baseline = self._get_text_size(text, font_face, base_scale, thickness)
            
            # Enforce a minimum font size
            min_height_target = float(self.style.annotation_font_min_size or 0)
            divisor = float(self.style.annotation_font_divisor or 0)
            if divisor > 0:
                min_height_target = max(min_height_target, min_dim / divisor)
            if min_height_target > 0 and text_h < min_height_target:
                scale_factor = min_height_target / max(text_h, 1)
                base_scale *= scale_factor
                text_w, text_h, baseline = self._get_text_size(text, font_face, base_scale, thickness)
            
            pad = max(0, self.style.annotation_label_padding)
            
            # Use pixel-based detection to find the best position.
            # Limit the maximum distance so the annotation stays near the segment.
            # Use a narrow range of distances so the annotation is closer to the segment.
            offset_multipliers = [0.8, 0.9, 1.0, 1.05, 1.1]  # smaller range, closer to the segment
            
            # Try both sides of the segment (normal direction and its opposite)
            best_pos = None
            best_rect = None
            best_score = float('inf')
            found_no_overlap = False  # Flag: whether we have found a non-overlapping position
            
            # First try along the normal direction, then the opposite (both sides)
            for direction_mult in [1.0, -1.0]:
                for offset_mult in offset_multipliers:
                    candidate_center = mid + normal * (base_offset * offset_mult * direction_mult)
                    cx, cy = candidate_center
                    origin_x = int(round(cx - text_w / 2))
                    origin_y = int(round(cy - text_h / 2))
                    candidate_rect = (
                        origin_x - pad,
                        origin_y - pad,
                        origin_x + text_w + pad,
                        origin_y + text_h + pad,
                    )
                    # Ensure the rectangle lies within the canvas bounds
                    candidate_rect = self._clamp_rect(candidate_rect, canvas.shape)
                    pixel_val = self._get_pixel_overlap_value(canvas, candidate_rect, self.style.background_color)
                    
                    # Compute a combined score: prioritize non-overlap, then closeness.
                    # Distance penalty: the farther from 1.0, the larger the penalty
                    # (use absolute value because offset_mult can be less than 1).
                    distance_penalty = abs(offset_mult - 1.0) * 100.0  # increase weight of distance penalty
                    if abs(pixel_val) < 1e-5:
                        # No overlap: consider distance only
                        score = distance_penalty
                        if score < best_score:
                            best_score = score
                            best_pos = candidate_center
                            best_rect = candidate_rect
                            found_no_overlap = True
                            # If we find a non-overlapping position with offset_mult close to 1.0,
                            # we can stop searching early.
                            if abs(offset_mult - 1.0) < 0.05:  # very close to base_offset
                                break
                    else:
                        # There is overlap; prefer smaller overlap (only if we have not
                        # already found a non-overlapping position).
                        if not found_no_overlap:
                            score = pixel_val * 10000.0 + distance_penalty
                            if score < best_score:
                                best_score = score
                                best_pos = candidate_center
                                best_rect = candidate_rect
                
                # If we have found a non-overlapping and nearby position, we can stop early
                if found_no_overlap and best_pos is not None:
                    # Check whether the non-overlapping position is already very close to the base offset
                    current_offset = np.linalg.norm(np.array(best_pos) - mid) / base_offset
                    if abs(current_offset - 1.0) < 0.05:  # very close to base_offset
                        break
            
            if best_pos is None:
                # Fall back to the default position
                best_pos = mid + normal * base_offset
                cx, cy = best_pos
                origin_x = int(round(cx - text_w / 2))
                origin_y = int(round(cy - text_h / 2))
                best_rect = (
                    origin_x - pad,
                    origin_y - pad,
                    origin_x + text_w + pad,
                    origin_y + text_h + pad,
                )
                best_rect = self._clamp_rect(best_rect, canvas.shape)
            
            text_jobs.append((text, best_pos, best_rect))
            
        # Record rectangles of length annotations to avoid overlap with later annotations
        for text, pos, text_rect in text_jobs:
            if text_rect:
                # Record angle-annotation rectangles to avoid overlaps with later annotations
                self._annotation_boxes.append(text_rect)

        for text, pos, _ in text_jobs:
            self._draw_annotation_text(canvas, text, pos)

    def _plot_angle_measures(
        self,
        canvas: np.ndarray,
        points: Dict[str, Tuple[int, int]],
        angle_annotations: Iterable[Sequence],
        right_angle_keys: set[RightAngleKey],
        segments: Optional[Iterable[Sequence]] = None,
    ) -> None:
        text_jobs: List[Tuple[str, Tuple[float, float], Tuple[int, int, int, int]]] = []
        import re

        for entry in angle_annotations:
            if not entry or len(entry) != 2:
                continue
            triple, value = entry
            if len(triple) != 3:
                continue
            a, b, c = (str(p).lower() for p in triple)
            if any(p not in points for p in (a, b, c)):
                continue
            pair_key = self._build_right_angle_pair_key(points, a, b, c)
            if pair_key and pair_key in right_angle_keys:
                continue
            
            # Skip angles that are (approximately) 90 degrees; they should be handled via `right_angles`
            try:
                value_str = str(value).strip()
                # Extract numeric value
                match = re.search(r"-?\d+(?:\.\d+)?", value_str)
                    if match:
                        angle_value = float(match.group(0))
                        # If the angle is close to 90 degrees (within a small tolerance), skip it
                    if abs(angle_value - 90.0) < 1.0:
                        continue
            except (ValueError, AttributeError):
                pass
            
            pa, pb, pc = points[a], points[b], points[c]

            va = np.array([pa[0] - pb[0], pa[1] - pb[1]], dtype=float)
            vc = np.array([pc[0] - pb[0], pc[1] - pb[1]], dtype=float)
            if np.linalg.norm(va) < 1 or np.linalg.norm(vc) < 1:
                continue
            
            # Normalize vectors
            va_norm = va / np.linalg.norm(va)
            vc_norm = vc / np.linalg.norm(vc)
            
            # Compute the angle bisector (va_norm + vc_norm already points inside the angle)
            bisector = va_norm + vc_norm
            if np.linalg.norm(bisector) < 1e-6:
                continue
            bisector /= np.linalg.norm(bisector)
            
            # No need to flip: va_norm + vc_norm always points inside the angle,
            # regardless of clockwise or counter-clockwise orientation.
            
            arc_radius = self._compute_angle_arc_radius(pa, pb, pc, shrink=2)
            
            # Pre-compute text size
            value_str = self._sanitize_angle_text(str(value))
            min_dim = float(min(canvas.shape[:2]))
            base_scale = max(
                self.style.label_font_min_scale,
                min(self.style.label_font_max_scale, min_dim / 420.0),
            )
            base_scale *= max(0.1, self.style.annotation_font_scale_multiplier)
            font_face = self.style.label_font_face
            thickness = max(1, int(round(base_scale * self.style.label_font_thickness)))
            text_w, text_h, baseline = self._get_text_size(value_str, font_face, base_scale, thickness)
            pad = max(0, self.style.annotation_label_padding)
            
            # Strategy: place the annotation along the angle bisector, close to the arc but not intersecting it.
            # Compute the minimum distance from the annotation to the arc edge:
            # arc radius + half of the text height + a small gap.
            # This keeps the annotation outside the arc while remaining visually close.
            text_radius = text_h / 2.0  # "Radius" of the text (center to edge distance, only considering height)
            # Use a smaller gap so the annotation is closer to the arc
            min_distance_from_arc = arc_radius + text_radius + max(4.0, pad + 2)
            
            # Use pixel-based detection to find the best position, sampling multiple points
            # along the angle bisector.
            dist_multipliers = [1.0, 1.15, 1.3, 1.6, 2.0]
            
            best_pos = None
            best_rect = None
            min_pixel_val = float('inf')
            
            pb_array = np.array([pb[0], pb[1]], dtype=float)
            for dist_multiplier in dist_multipliers:
                candidate_center = pb_array + bisector * (min_distance_from_arc * dist_multiplier)
                cx, cy = candidate_center
                origin_x = int(round(cx - text_w / 2))
                origin_y = int(round(cy - text_h / 2))
                candidate_rect = (
                    origin_x - pad,
                    origin_y - pad,
                    origin_x + text_w + pad,
                    origin_y + text_h + pad,
                )
                # Ensure the rectangle lies within the canvas bounds
                candidate_rect = self._clamp_rect(candidate_rect, canvas.shape)
                pixel_val = self._get_pixel_overlap_value(canvas, candidate_rect, self.style.background_color)
                
                if abs(pixel_val) < 1e-5:
                    # Completely non-overlapping position found; use it immediately
                    best_pos = candidate_center
                    best_rect = candidate_rect
                    min_pixel_val = 0.0
                    break
                
                if pixel_val < min_pixel_val:
                    min_pixel_val = pixel_val
                    best_pos = candidate_center
                    best_rect = candidate_rect
            
            if best_pos is None:
                # Fall back to the default position
                best_pos = pb_array + bisector * min_distance_from_arc
                cx, cy = best_pos
                origin_x = int(round(cx - text_w / 2))
                origin_y = int(round(cy - text_h / 2))
                best_rect = (
                    origin_x - pad,
                    origin_y - pad,
                    origin_x + text_w + pad,
                    origin_y + text_h + pad,
                )
                best_rect = self._clamp_rect(best_rect, canvas.shape)
            
            # Draw a slightly shrunken angle arc so that it is visually separated from the numeric label
            self._draw_angle_arc(canvas, pa, pb, pc, shrink=2)

            text_jobs.append((value_str, best_pos, best_rect))

        # Record angle-annotation rectangles to avoid overlap with later annotations
        for text, pos, text_rect in text_jobs:
            if text_rect:
                self._annotation_boxes.append(text_rect)
        
        for text, pos, _ in text_jobs:
            self._draw_annotation_text(canvas, text, pos)

    # ------------------------------------------------------------------ #
    # Utilities                                                          #
    # ------------------------------------------------------------------ #
    def _label_anchor_from_center(
        self,
        point: Tuple[int, int],
        centroid: Tuple[float, float],
        text_w: int,
        text_h: int,
        canvas_shape: Tuple[int, int, int],
    ) -> Tuple[int, int]:
        px, py = point
        cx, cy = centroid
        vec = np.array([px - cx, py - cy], dtype=float)
        norm = np.linalg.norm(vec)
        if norm < 1e-6:
            vec = np.array([1.0, -1.0])
            norm = np.linalg.norm(vec)
        vec /= norm
        # Increase the distance between the point and its label to reduce overlap
        distance = max(self.style.label_distance + self.style.point_radius + 12, 35)
        target_x = px + vec[0] * distance
        target_y = py + vec[1] * distance
        width = canvas_shape[1]
        height = canvas_shape[0]
        anchor_x = int(self._clamp(target_x - text_w / 2, 2, width - text_w - 2))
        anchor_y = int(self._clamp(target_y - text_h / 2, 2, height - text_h - 2))
        return anchor_x, anchor_y

    def _draw_line_with_glow(
        self,
        canvas: np.ndarray,
        start: Tuple[int, int],
        end: Tuple[int, int],
        color: Tuple[int, int, int],
    ) -> None:
        if self.style.line_glow_color and self.style.line_glow_width > 0:
            cv2.line(
                canvas,
                start,
                end,
                self.style.line_glow_color,
                max(self.style.line_width + self.style.line_glow_width, 1),
                lineType=cv2.LINE_AA,
            )
        cv2.line(
            canvas,
            start,
            end,
            color,
            self.style.line_width,
            lineType=cv2.LINE_AA,
        )

    def _draw_dimension_ticks(
        self,
        canvas: np.ndarray,
        start: Tuple[int, int],
        end: Tuple[int, int],
    ) -> None:
        start_pt = np.array(start, dtype=float)
        end_pt = np.array(end, dtype=float)
        direction = end_pt - start_pt
        length = np.linalg.norm(direction)
        if length < 1:
            return
        direction /= length
        normal = np.array([-direction[1], direction[0]])
        tick = max(4, int(self.style.equal_tick_length * 0.6))
        for base in (start_pt, end_pt):
            tick_start = base + normal * tick
            tick_end = base - normal * tick
            cv2.line(
                canvas,
                (int(tick_start[0]), int(tick_start[1])),
                (int(tick_end[0]), int(tick_end[1])),
                self.style.annotation_color,
                1,
                lineType=cv2.LINE_AA,
            )

    def _draw_annotation_text(
        self,
        canvas: np.ndarray,
        text: str,
        center: Tuple[float, float],
        base_scale: Optional[float] = None,
        thickness: Optional[int] = None,
    ) -> None:
        if base_scale is None or thickness is None:
            min_dim = float(min(canvas.shape[:2]))
            base_scale = max(
                self.style.label_font_min_scale,
                min(self.style.label_font_max_scale, min_dim / 420.0),
            )
            base_scale *= max(0.1, self.style.annotation_font_scale_multiplier)
            thickness = max(1, int(round(base_scale * self.style.label_font_thickness)))

        font_face = self.style.label_font_face
        width, height, baseline = self._get_text_size(text, font_face, base_scale, thickness)
        min_dim = float(min(canvas.shape[:2]))
        min_height_target = float(self.style.annotation_font_min_size or 0)
        divisor = float(self.style.annotation_font_divisor or 0)
        if divisor > 0:
            min_height_target = max(min_height_target, min_dim / divisor)
        if min_height_target > 0 and height < min_height_target:
            scale_factor = min_height_target / max(height, 1)
            base_scale *= scale_factor
            width, height, baseline = self._get_text_size(text, font_face, base_scale, thickness)

        cx, cy = center
        pad = max(0, self.style.annotation_label_padding)
        
        # Compute the top-left coordinate so that the text is centered at (cx, cy)
        origin_x = int(round(cx - width / 2))
        origin_y = int(round(cy - height / 2))
        
        bg_color = self.style.annotation_label_background_color
        border_color = self.style.annotation_label_border_color
        
        # Compute the text rectangle (including padding)
        text_rect = (
            origin_x - pad,
            origin_y - pad,
            origin_x + width + pad,
            origin_y + height + pad,
        )
        
        # For angle annotations, try to keep the text center at the desired position,
        # avoiding large movements.
        original_center = (cx, cy)
        
        # First check whether the original position lies within the canvas
        min_x = pad
        max_x = canvas.shape[1] - width - pad
        min_y = pad
        max_y = canvas.shape[0] - height - pad
        
        # If the original position is inside the canvas, try not to move it much
        if (min_x <= origin_x <= max_x and min_y <= origin_y <= max_y):
            # Try to resolve overlap, but limit how far we move the label
            resolved_rect = self._resolve_non_overlapping_rect(text_rect, canvas.shape, padding=6)
            
            # Compute the center of the adjusted rectangle
            resolved_center_x = (resolved_rect[0] + resolved_rect[2]) / 2.0
            resolved_center_y = (resolved_rect[1] + resolved_rect[3]) / 2.0
            
            # If the center shifts too much (more than 2px), keep the original position
            offset_x = resolved_center_x - original_center[0]
            offset_y = resolved_center_y - original_center[1]
            
            if abs(offset_x) > 2 or abs(offset_y) > 2:
                # Keep the original center position
                origin_x = int(round(original_center[0] - width / 2))
                origin_y = int(round(original_center[1] - height / 2))
                rect = text_rect
            else:
                # Use the adjusted rectangle
                origin_x = resolved_rect[0] + pad
                origin_y = resolved_rect[1] + pad
                rect = resolved_rect
        else:
            # If the original position is outside the canvas, make minimal adjustments
            if origin_x < min_x:
                origin_x = min_x
            elif origin_x > max_x:
                origin_x = max_x
                
            if origin_y < min_y:
                origin_y = min_y
            elif origin_y > max_y:
                origin_y = max_y
            
            rect = (
                origin_x - pad,
                origin_y - pad,
                origin_x + width + pad,
                origin_y + height + pad,
            )
        if bg_color is not None or border_color is not None:
            top_left = (rect[0], rect[1])
            bottom_right = (rect[2], rect[3])
            if bg_color is not None:
                cv2.rectangle(
                    canvas,
                    top_left,
                    bottom_right,
                    bg_color,
                    thickness=-1,
                )
            if border_color is not None:
                cv2.rectangle(
                    canvas,
                    top_left,
                    bottom_right,
                    border_color,
                    thickness=1,
                )

        self._draw_text_common(
            canvas,
            text,
            (origin_x, origin_y),
            font_face,
            base_scale,
            self.style.annotation_color,
            thickness,
            baseline_offset=baseline
        )
        
        self._annotation_boxes.append(rect)

    def _normalize_angle_group(
        self,
        group: Sequence[Sequence[PointName]],
    ) -> List[Tuple[PointName, PointName, PointName]]:
        if not group:
            return []
        if isinstance(group, dict):
            source = (
                group.get("angles")
                or group.get("items")
                or group.get("segments")
                or []
            )
            return self._normalize_angle_group(source)  # type: ignore[arg-type]
        normalized: List[Tuple[PointName, PointName, PointName]] = []
        if len(group) == 3 and all(isinstance(item, (str, int)) for item in group):
            normalized.append(
                (group[0], group[1], group[2])  # type: ignore[arg-type]
            )
            return normalized
        for candidate in group:
            if isinstance(candidate, (list, tuple)) and len(candidate) == 3:
                normalized.append((candidate[0], candidate[1], candidate[2]))
        return normalized

    def _normalize_segment_group(
        self,
        group: Sequence[Sequence[PointName]],
    ) -> List[Tuple[PointName, PointName]]:
        if not group:
            return []
        if isinstance(group, dict):
            source = (
                group.get("segments")
                or group.get("lines")
                or group.get("items")
                or []
            )
            return self._normalize_segment_group(source)  # type: ignore[arg-type]
        normalized: List[Tuple[PointName, PointName]] = []
        if len(group) == 2 and all(isinstance(item, (str, int)) for item in group):
            normalized.append((group[0], group[1]))  # type: ignore[arg-type]
            return normalized
        for candidate in group:
            if isinstance(candidate, (list, tuple)) and len(candidate) == 2:
                normalized.append((candidate[0], candidate[1]))
        return normalized

    def _equal_tick_offsets(self, tick_count: int) -> List[float]:
        spacing = max(6.0, self.style.equal_tick_length * 0.7)
        if tick_count <= 1:
            return [0.0]
        if tick_count == 2:
            return [-spacing, spacing]
        return [-spacing, 0.0, spacing]

    @staticmethod
    def _clamp(value: float, low: float, high: float) -> float:
        if low > high:
            low, high = high, low
        return max(low, min(high, value))

    @staticmethod
    def _hex_to_bgr(value: str) -> Tuple[int, int, int]:
        stripped = value.strip().lstrip("#")
        if len(stripped) == 3:
            stripped = "".join(ch * 2 for ch in stripped)
        if len(stripped) != 6:
            return (255, 255, 255)
        r = int(stripped[0:2], 16)
        g = int(stripped[2:4], 16)
        b = int(stripped[4:6], 16)
        return (b, g, r)

    @staticmethod
    def _sanitize_length_text(raw: str) -> str:
        """
        Clean up length-annotation text:
        - Convert sqrt(x) into the mathematical symbol form √x
        - Remove trailing uncertainty markers such as ? / fullwidth '?'
        - Remove unnecessary .0 decimals (e.g. 5.0 -> 5)
        - Keep the original content (including √), rendered by a Unicode-capable font
        """
        text = raw.strip()
        
        # Convert sqrt expressions into radical symbols.
        # Handle cases of the form n*sqrt(x) (n can be integer/float and may have spaces).
        # Examples: 2*sqrt(3) -> 2√3, 2* sqrt(3) -> 2√3
        text = re.sub(r'(\d+(?:\.\d+)?)\s*\*\s*sqrt\((\d+(?:\.\d+)?)\)', r'\1√\2', text)
        # Handle standalone sqrt(x). Example: sqrt(3) -> √3
        text = re.sub(r'sqrt\((\d+(?:\.\d+)?)\)', r'√\1', text)
        
        text = re.sub(r"[?\uFF1F]+$", "", text)
        # Remove unnecessary .0 decimals
        text = re.sub(r"(\d+)\.0+(?!\d)", r"\1", text)
        return text

    @staticmethod
    def _sanitize_angle_text(raw: str) -> str:
        """
        Clean up angle-annotation text:
        - If the text contains "pi", treat it as an expression with π and only beautify it
          (pi -> π) without forcing conversion to degrees. For example, "pi/6" -> "π/6",
          "2*pi/3" -> "2*π/3".
        - Otherwise:
          - Remove trailing uncertainty markers such as ? / fullwidth '?'
          - Try to extract a numeric value and normalize it to '<value>°'
          - Remove unnecessary .0 decimals (e.g. 90.0° -> 90°)
        """
        text = raw.strip()
        # First strip trailing uncertainty markers
        text = re.sub(r"[?\uFF1F]+$", "", text)
        # If it contains "pi", show it in π form without numeric normalization
        if "pi" in text:
            # Simple replacement for nicer visual appearance
            text = text.replace("pi", "π")
            return text
        # Numeric angle: normalize to '<value>°'
        match = re.search(r"-?\d+(?:\.\d+)?", text)
        if match:
            value = match.group(0)
            # Remove unnecessary .0 decimals
            value = re.sub(r"(\d+)\.0+$", r"\1", value)
            return f"{value}°"
        return text

    def _auto_crop(
        self, canvas: np.ndarray
    ) -> Tuple[np.ndarray, Optional[Tuple[int, int]]]:
        """
        Automatically crop away outer background-only regions to make
        the figure tighter while keeping a small margin.
        Returns (cropped_canvas, (offset_x, offset_y)) where offsets are
        how much the top-left corner moved. If no crop performed, offset is (0,0).
        """
        bg = np.array(self.style.background_color, dtype=np.uint8)
        if canvas.ndim != 3 or canvas.shape[2] != 3:
            return canvas, (0, 0)

        # mask of any pixel that differs from pure background
        diff = np.any(canvas != bg, axis=2)
        if not np.any(diff):
            return canvas, (0, 0)

        ys, xs = np.where(diff)
        y_min, y_max = int(ys.min()), int(ys.max())
        x_min, x_max = int(xs.min()), int(xs.max())

        h, w = canvas.shape[:2]
        margin = int(max(6, min(h, w) * 0.04))

        y_min_c = max(0, y_min - margin)
        y_max_c = min(h - 1, y_max + margin)
        x_min_c = max(0, x_min - margin)
        x_max_c = min(w - 1, x_max + margin)

        cropped = canvas[y_min_c : y_max_c + 1, x_min_c : x_max_c + 1].copy()
        return cropped, (x_min_c, y_min_c)

    def _rotate_canvas_with_points(
        self,
        canvas: np.ndarray,
        points: Dict[str, Tuple[int, int]],
        angle: float,
    ) -> Tuple[np.ndarray, Dict[str, Tuple[int, int]]]:
        """
        Rotate the canvas and apply the same transformation to point coordinates (in pixel space).
        Returns (rotated_canvas, rotated_points).
        """
        if canvas.ndim != 3 or canvas.shape[2] != 3:
            return canvas, dict(points)
        
        height, width = canvas.shape[:2]
        center = (width / 2.0, height / 2.0)
        
        rotation_matrix = cv2.getRotationMatrix2D(center, angle, 1.0)
        
        cos_val = abs(rotation_matrix[0, 0])
        sin_val = abs(rotation_matrix[0, 1])
        new_width = int((height * sin_val) + (width * cos_val))
        new_height = int((height * cos_val) + (width * sin_val))
        
        rotation_matrix[0, 2] += (new_width / 2) - center[0]
        rotation_matrix[1, 2] += (new_height / 2) - center[1]
        
        rotated = cv2.warpAffine(
            canvas,
            rotation_matrix,
            (new_width, new_height),
            flags=cv2.INTER_LINEAR,
            borderMode=cv2.BORDER_CONSTANT,
            borderValue=self.style.background_color
        )
        
        rotated_points: Dict[str, Tuple[int, int]] = {}
        for name, (x, y) in points.items():
            vec = np.array([x, y, 1.0], dtype=float)
            rx, ry = rotation_matrix.dot(vec)
            rotated_points[name] = (int(round(rx)), int(round(ry)))
        
        return rotated, rotated_points

    def _resolve_non_overlapping_rect(
        self,
        rect: Tuple[int, int, int, int],
        canvas_shape: Tuple[int, int, int],
        padding: int = 2,
        max_attempts: int = 20,
        step: int = 14,
    ) -> Tuple[int, int, int, int]:
        # Include all elements that need to be checked for overlap
        existing_rects = self._label_boxes + self._annotation_boxes + self._segment_boxes
        existing_circles = self._circle_boxes
        rect = self._clamp_rect(rect, canvas_shape)
        if not existing_rects and not existing_circles:
            return rect

        directions = [
            (0, 1),    # down
            (0, -1),   # up
            (1, 0),    # right
            (-1, 0),   # left
            (1, 1),
            (-1, 1),
            (1, -1),
            (-1, -1),
        ]
        for radius in range(max_attempts):
            candidates = [(0, 0)] if radius == 0 else directions
            for dx, dy in candidates:
                offset_x = dx * radius * step
                offset_y = dy * radius * step
                candidate = (
                    rect[0] + offset_x,
                    rect[1] + offset_y,
                    rect[2] + offset_x,
                    rect[3] + offset_y,
                )
                candidate = self._clamp_rect(candidate, canvas_shape)
                
                # Check overlap with all rectangles
                # Use a larger padding for segments
                overlaps_segment = any(
                    self._rectangles_overlap(candidate, other, padding=padding + 4)
                    for other in self._segment_boxes
                )
                
                if overlaps_segment:
                    continue
                    
                overlaps_rect = any(
                    self._rectangles_overlap(candidate, other, padding=padding)
                    for other in self._label_boxes + self._annotation_boxes
                )
                # Check overlap with all circle boxes
                overlaps_circle = any(
                    self._rectangle_overlaps_circle(candidate, circle, padding=padding)
                    for circle in existing_circles
                )
                if not overlaps_rect and not overlaps_circle:
                    return candidate
        return rect

    def _clamp_rect(
        self,
        rect: Tuple[int, int, int, int],
        canvas_shape: Tuple[int, int, int],
    ) -> Tuple[int, int, int, int]:
        height, width = canvas_shape[:2]
        x1, y1, x2, y2 = rect
        w = max(1, x2 - x1)
        h = max(1, y2 - y1)
        if w >= width:
            x1 = 0
            x2 = width
        else:
            x1 = int(round(self._clamp(x1, 0, width - w)))
            x2 = x1 + w
        if h >= height:
            y1 = 0
            y2 = height
        else:
            y1 = int(round(self._clamp(y1, 0, height - h)))
            y2 = y1 + h
        return (int(x1), int(y1), int(x2), int(y2))

    def _adjust_rect_away_from_segments(
        self,
        rect: Tuple[int, int, int, int],
        canvas_shape: Tuple[int, int, int],
        points: Dict[str, Tuple[int, int]],
        segments: Iterable[Sequence[PointName]],
        min_distance: float = 20.0,
    ) -> Tuple[int, int, int, int]:
        """
        Push label rectangles away from nearby segments to reduce overlap.
        """
        cx = (rect[0] + rect[2]) / 2.0
        cy = (rect[1] + rect[3]) / 2.0
        closest_dist = float("inf")
        closest_mid = None

        for seg in segments:
            if not isinstance(seg, (list, tuple)) or len(seg) != 2:
                continue
            a, b = (str(seg[0]).lower(), str(seg[1]).lower())
            if a not in points or b not in points:
                continue
            p1 = np.array(points[a], dtype=float)
            p2 = np.array(points[b], dtype=float)
            if np.allclose(p1, p2):
                continue
            dist, mid = self._point_segment_distance_with_midpoint(
                np.array([cx, cy], dtype=float),
                p1,
                p2,
            )
            if dist < closest_dist:
                closest_dist = dist
                closest_mid = mid

        if closest_mid is None or closest_dist >= min_distance:
            return rect

        direction = np.array([cx, cy], dtype=float) - closest_mid
        if np.linalg.norm(direction) < 1e-6:
            direction = np.array([1.0, -1.0], dtype=float)
        direction /= np.linalg.norm(direction)
        push = (min_distance - closest_dist) + min_distance * 0.4
        dx = direction[0] * push
        dy = direction[1] * push

        shifted = (
            rect[0] + dx,
            rect[1] + dy,
            rect[2] + dx,
            rect[3] + dy,
        )
        return self._clamp_rect(
            (int(round(shifted[0])), int(round(shifted[1])), int(round(shifted[2])), int(round(shifted[3]))),
            canvas_shape,
        )

    @staticmethod
    def _point_segment_distance_with_midpoint(
        p: np.ndarray,
        a: np.ndarray,
        b: np.ndarray,
    ) -> Tuple[float, np.ndarray]:
        ab = b - a
        ab_len2 = float(np.dot(ab, ab))
        if ab_len2 <= 1e-6:
            return float(np.linalg.norm(p - a)), a
        t = float(np.dot(p - a, ab) / ab_len2)
        t = max(0.0, min(1.0, t))
        proj = a + t * ab
        return float(np.linalg.norm(p - proj)), proj

    def _draw_text_common(
        self,
        canvas: np.ndarray,
        text: str,
        pos: Tuple[int, int],
        font_face: int,
        font_scale: float,
        color: Tuple[int, int, int],
        thickness: int,
        baseline_offset: Optional[int] = None,
        shadow_color: Optional[Tuple[int, int, int]] = None,
        shadow_offset: Optional[Tuple[int, int]] = None
    ) -> None:
        """General-purpose text drawing helper."""
        x, y = pos  # top-left
        
        if baseline_offset is None:
            _, _, baseline_offset = self._get_text_size(text, font_face, font_scale, thickness)

        if shadow_color:
            offset = shadow_offset or (1, 1)
            sx, sy = x + offset[0], y + offset[1]
            self._draw_text_pil(canvas, text, (sx, sy), font_scale, shadow_color)

        self._draw_text_pil(canvas, text, (x, y), font_scale, color)
            
    def _draw_text_pil(
        self,
        canvas: np.ndarray,
        text: str,
        pos: Tuple[int, int],
        font_scale: float,
        color: Tuple[int, int, int]
    ) -> None:
        x, y = pos
        rgb = cv2.cvtColor(canvas, cv2.COLOR_BGR2RGB)
        img = Image.fromarray(rgb)
        draw = ImageDraw.Draw(img)
        font_size = max(10, int(18 * font_scale))
        try:
            font = ImageFont.truetype("DejaVuSans.ttf", font_size)
        except Exception:
            font = ImageFont.load_default()
        
        draw.text(
            (x, y),
            text,
            font=font,
            fill=(int(color[2]), int(color[1]), int(color[0])),
        )
        canvas[:, :] = cv2.cvtColor(np.asarray(img), cv2.COLOR_RGB2BGR)

    def _draw_text_cv2(
        self,
        canvas: np.ndarray,
        text: str,
        pos: Tuple[int, int],
        font_face: int,
        font_scale: float,
        color: Tuple[int, int, int],
        thickness: int,
        baseline_offset: int
    ) -> None:
        x, y = pos
        cv2.putText(
            canvas,
            text,
            (int(x), int(y + baseline_offset)),
            font_face,
            font_scale,
            color,
            thickness,
            lineType=cv2.LINE_AA,
        )

    def _draw_debug_boxes(self, canvas: np.ndarray) -> None:
        """Debug mode: draw all annotation bounding boxes."""
        thickness = self.style.debug_box_thickness
        
        # Draw segment rectangles (magenta)
        for box in self._segment_boxes:
            x1, y1, x2, y2 = box
            cv2.rectangle(canvas, (x1, y1), (x2, y2), (255, 0, 255), thickness)
        
        # Draw circle detection regions (cyan/yellow)
        for circle in self._circle_boxes:
            cx, cy, radius = circle
            # Draw only the circle (no rectangle) to avoid visual clutter
            cv2.circle(canvas, (int(cx), int(cy)), int(radius), (0, 255, 255), thickness)
        
        # Draw point-label rectangles (green)
        for box in self._label_boxes:
            x1, y1, x2, y2 = box
            cv2.rectangle(canvas, (x1, y1), (x2, y2), (0, 255, 0), thickness)
        
        # Draw annotation rectangles (red).
        # Note: OpenCV uses BGR order, so red is (0, 0, 255).
        for box in self._annotation_boxes:
            x1, y1, x2, y2 = box
            cv2.rectangle(canvas, (x1, y1), (x2, y2), (0, 0, 255), thickness)
    
    @staticmethod
    def _rectangles_overlap(
        a: Tuple[int, int, int, int],
        b: Tuple[int, int, int, int],
        padding: int = 2,
    ) -> bool:
        ax1, ay1, ax2, ay2 = a
        bx1, by1, bx2, by2 = b
        return not (
            ax2 + padding <= bx1
            or bx2 + padding <= ax1
            or ay2 + padding <= by1
            or by2 + padding <= ay1
        )
    
    def _rectangle_overlap_area(
        self,
        a: Tuple[int, int, int, int],
        b: Tuple[int, int, int, int],
        padding: int = 0,
    ) -> float:
        """Return the overlap area between two rectangles (taking padding into account)."""
        ax1, ay1, ax2, ay2 = a
        bx1, by1, bx2, by2 = b
        left = max(ax1, bx1 - padding)
        right = min(ax2, bx2 + padding)
        top = max(ay1, by1 - padding)
        bottom = min(ay2, by2 + padding)
        if right <= left or bottom <= top:
            return 0.0
        return float(max(0, right - left) * max(0, bottom - top))
    
    def _point_in_circle(
        self,
        point: Tuple[float, float],
        circle: Tuple[int, int, int],
        padding: int = 2,
    ) -> bool:
        """Check whether a point lies inside a circle (with padding)."""
        px, py = point
        cx, cy, radius = circle
        dist = math.sqrt((px - cx) ** 2 + (py - cy) ** 2)
        return dist <= radius + padding
    
    def _rectangle_overlaps_circle(
        self,
        rect: Tuple[int, int, int, int],
        circle: Tuple[int, int, int],
        padding: int = 2,
    ) -> bool:
        """Check whether a rectangle overlaps the circumference of a circle."""
        x1, y1, x2, y2 = rect
        cx, cy, radius = circle
        
        # Compute the minimum distance from the rectangle to the circle center (fully outside check).
        closest_x = max(x1, min(cx, x2))
        closest_y = max(y1, min(cy, y2))
        min_dist = math.sqrt((closest_x - cx) ** 2 + (closest_y - cy) ** 2)
        
        if min_dist > radius + padding:
            return False  # fully outside the circle
            
        # Compute the maximum distance from the rectangle to the circle center (fully inside check).
        # The farthest point must be one of the four corners.
        corners = [(x1, y1), (x2, y1), (x2, y2), (x1, y2)]
        max_dist = 0.0
        for px, py in corners:
            d = math.sqrt((px - cx) ** 2 + (py - cy) ** 2)
            if d > max_dist:
                max_dist = d
        
        if max_dist < radius - padding:
            return False  # fully inside the circle (no contact with circumference)
            
        # Otherwise, the rectangle intersects the annulus around the circle
        return True
    
    def _rectangle_circle_penalty(
        self,
        rect: Tuple[int, int, int, int],
        circle: Tuple[int, int, int],
        padding: int = 2,
    ) -> float:
        """
        Return a continuous penalty value based on how deeply a rectangle touches a circle's circumference,
        used for occlusion scoring.
        """
        x1, y1, x2, y2 = rect
        cx, cy, radius = circle
        closest_x = max(x1, min(cx, x2))
        closest_y = max(y1, min(cy, y2))
        min_dist = math.sqrt((closest_x - cx) ** 2 + (closest_y - cy) ** 2)
        if min_dist > radius + padding:
            return 0.0
        corners = [(x1, y1), (x2, y1), (x2, y2), (x1, y2)]
        max_dist = 0.0
        for px, py in corners:
            d = math.sqrt((px - cx) ** 2 + (py - cy) ** 2)
            if d > max_dist:
                max_dist = d
        if max_dist < radius - padding:
            return 0.0
        # Larger penalties mean closer to or deeper into the circular annulus
        penetration = max(0.0, (radius + padding) - min_dist)
        coverage = max(0.0, max_dist - (radius - padding))
        return max(1.0, penetration + coverage)
    
    def _get_pixel_overlap_value(
        self,
        canvas: np.ndarray,
        bbox: Tuple[int, int, int, int],
        background_color: Tuple[int, int, int],
    ) -> float:
        """
        Measure overlap based on pixel values, returning how much the given bbox region
        overlaps with already drawn content.
        
        Returns:
            0.0 for no overlap (all background color); larger values mean more overlap.
        
        Args:
            canvas: Current canvas (BGR format).
            bbox: (x1, y1, x2, y2) rectangle.
            background_color: Background color (B, G, R).
        
        Returns:
            Overlap value: sum of differences for all non-background pixels in the region.
        """
        x1, y1, x2, y2 = bbox
        # Clamp bbox to stay inside the canvas
        h, w = canvas.shape[:2]
        x1 = max(0, min(x1, w - 1))
        y1 = max(0, min(y1, h - 1))
        x2 = max(x1 + 1, min(x2, w))
        y2 = max(y1 + 1, min(y2, h))
        
        if x2 <= x1 or y2 <= y1:
            return 0.0
        
        # Extract pixel region inside the bbox
        region = canvas[y1:y2, x1:x2, :]
        
        # Compute difference from the background color (BGR)
        bg = np.array(background_color, dtype=np.uint8)
        diff = np.abs(region.astype(np.int32) - bg.astype(np.int32))
        
        # Sum differences over all pixels
        pixel_val = np.sum(diff)
        
        return float(pixel_val)
    
    def _find_best_position_by_pixels(
        self,
        canvas: np.ndarray,
        text_size: Tuple[int, int],
        center_pos: Tuple[float, float],
        radiuses: List[int],
        n_samples_per_radius: int = 20,
        background_color: Tuple[int, int, int] = (255, 255, 255),
        padding: int = 2,
    ) -> Tuple[Tuple[int, int], Tuple[int, int, int, int]]:
        """
        Use pixel-based overlap detection to find the best annotation position
        (similar to the method used in plotter_reference.py).
        
        Args:
            canvas: Current canvas.
            text_size: (width, height) of the text.
            center_pos: (x, y) center position (e.g. the point position).
            radiuses: List of search radii.
            n_samples_per_radius: Number of samples per radius.
            background_color: Background color.
            padding: Padding for the bbox.
        
        Returns:
            (best_position, bbox) where best_position is the top-left of the text
            and bbox is (x1, y1, x2, y2).
        """
        import random
        import math
        
        text_w, text_h = text_size
        cx, cy = center_pos
        
        position_pixel_vals = []
        position_to_select = []
        bbox_to_select = []
        
        for radius in radiuses:
            for _ in range(n_samples_per_radius):
                # Randomly sample points on the circle of the given radius
                theta = math.radians(random.uniform(0, 360))
                x = cx + math.cos(theta) * radius
                y = cy + math.sin(theta) * radius
                
                # Compute the text bbox (text centered at (x, y))
                anchor_x = int(round(x - text_w / 2))
                anchor_y = int(round(y - text_h / 2))
                
                # Add padding
                bbox = (
                    max(0, anchor_x - padding),
                    max(0, anchor_y - padding),
                    anchor_x + text_w + padding,
                    anchor_y + text_h + padding,
                )
                
                # Compute pixel-overlap score
                pixel_val = self._get_pixel_overlap_value(canvas, bbox, background_color)
                
                # If there is essentially no overlap, return immediately
                if abs(pixel_val) < 1e-5:
                    return ((anchor_x, anchor_y), bbox)
                
                position_pixel_vals.append(pixel_val)
                position_to_select.append((anchor_x, anchor_y))
        bbox_to_select.append(bbox)
        
        # If all positions have overlap, choose the one with the smallest overlap value
        if position_pixel_vals:
            idx = np.argmin(position_pixel_vals)
            return (position_to_select[idx], bbox_to_select[idx])
        
        # Extreme fallback: fall back to a default position
        anchor_x = int(round(cx + radiuses[0] if radiuses else 10))
        anchor_y = int(round(cy - text_h / 2))
        bbox = (
            max(0, anchor_x - padding),
            max(0, anchor_y - padding),
            anchor_x + text_w + padding,
            anchor_y + text_h + padding,
        )
        return ((anchor_x, anchor_y), bbox)

    def _ray_direction_signature(
        self,
        vertex: str,
        vec: Sequence[float],
        precision: int = 10_000,
    ) -> Optional[Tuple[str, int, int]]:
        dx = float(vec[0]) if len(vec) > 0 else 0.0
        dy = float(vec[1]) if len(vec) > 1 else 0.0
        length = math.hypot(dx, dy)
        if length < 1e-6:
            return None
        dx /= length
        dy /= length
        # Treat opposite directions as the same infinite line (AB == BA)
        if dx < -1e-9 or (abs(dx) <= 1e-9 and dy < 0):
            dx = -dx
            dy = -dy
        return (
            vertex,
            int(round(dx * precision)),
            int(round(dy * precision)),
        )

    def _angle_direction_pair_key_from_vectors(
        self,
        vertex: str,
        vec1: Sequence[float],
        vec2: Sequence[float],
    ) -> Optional[RightAngleKey]:
        key1 = self._ray_direction_signature(vertex, vec1)
        key2 = self._ray_direction_signature(vertex, vec2)
        if key1 is None or key2 is None:
            return None
        if key1 == key2:
            return None
        ordered = sorted((key1, key2))
        return (ordered[0], ordered[1])

    def _build_right_angle_pair_key(
        self,
        points: Dict[str, Tuple[int, int]],
        a: str,
        b: str,
        c: str,
    ) -> Optional[RightAngleKey]:
        if any(name not in points for name in (a, b, c)):
            return None
        ba = (
            float(points[a][0] - points[b][0]),
            float(points[a][1] - points[b][1]),
        )
        bc = (
            float(points[c][0] - points[b][0]),
            float(points[c][1] - points[b][1]),
        )
        return self._angle_direction_pair_key_from_vectors(b, ba, bc)

    def _collect_right_angle_keys(
        self,
        points: Dict[str, Tuple[int, int]],
        annotations: Dict,
    ) -> set[RightAngleKey]:
        keys: set[RightAngleKey] = set()
        right_angles = (annotations or {}).get("right_angles") or []
        for triple in right_angles:
            if not isinstance(triple, (list, tuple)) or len(triple) != 3:
                continue
            a, b, c = (str(v).lower() for v in triple)
            key = self._build_right_angle_pair_key(points, a, b, c)
            if key is None:
                continue
            keys.add(key)
        return keys

    def _angle_arc_radius_from_lengths(
        self,
        len_ba: float,
        len_bc: float,
        shrink: float = 0.0,
    ) -> int:
        line_len = min(len_ba, len_bc)
        if line_len < 1e-3:
            return 0
        # Angle-arc radius: based on one quarter of the segment length (slightly larger).
        radius = int(line_len / 4)
        # Clamp radius between 12 and 30, then shrink slightly.
        radius = max(12, min(radius, 30) - int(shrink * 3))
        return max(radius, 0)
    
    def _compute_angle_arc_radius(
        self,
        pa: Tuple[int, int],
        pb: Tuple[int, int],
        pc: Tuple[int, int],
        shrink: float = 0.0,
    ) -> int:
        vba = np.array([pa[0] - pb[0], pa[1] - pb[1]], dtype=float)
        vbc = np.array([pc[0] - pb[0], pc[1] - pb[1]], dtype=float)
        norm_ba = np.linalg.norm(vba)
        norm_bc = np.linalg.norm(vbc)
        if norm_ba < 1 or norm_bc < 1:
            return 0
        return self._angle_arc_radius_from_lengths(norm_ba, norm_bc, shrink)
