import numpy as np
import matplotlib.pyplot as plt
import cv2
import os
import scipy.spatial
import scipy.optimize
import yaml
import copy
import shutil
import multiprocessing
from PIL import Image
import numpy as np
from skimage.metrics import structural_similarity as ssim
import scipy
import math
import time
import re
import gc
from object_recognition.post_processing.smxReader import NanonisSXM
from object_recognition.Yolo.object_detection import ObjectRecognition
from molecule_movement.matching import RandomMatching, GreedyMatching, HungarianMatching
from shapely import Point
import tkinter as tk
from tkinter import ttk
from shapely.geometry import Point, Polygon, LineString, box
from pathlib import Path

from loguru import logger

class AnalyseMoieties():

    def __init__(self):
        self.input_yaml_file = os.path.normpath(os.path.join(os.getcwd(),"input.yaml"))
        self.config_object_recognition_file = os.path.normpath(os.path.join(os.getcwd(),"object_recognition/Yolo/config.yaml"))

        self.reference_image_dir = os.path.normpath(os.path.join(os.getcwd(),"object_recognition/post_processing/reference_images/"))
        os.makedirs(self.reference_image_dir, exist_ok=True)


        self.moiety_classes = np.array(self.get_yaml_data(self.config_object_recognition_file, "classes"))

        self.object_recognition = ObjectRecognition()

        self._CONFIDENCE_THRESHOLD = 0.55
        self._INIT_CONFIDENCE_THRESHOLD = 0.99

    def load_reference_contour(self, moiety_type, at_origin=False):
        """
        Load the reference contour for a given moiety type from the reference images directory.

        Parameters:
        -----------
        moiety_type : str
            The type of the moiety (e.g., 'molecule_FePc_Au111_4K').

        Returns:
        --------
        ref_img_gray : np.ndarray
            The contour as xy-coordinates in nm.
        """
        ref_contour_file = os.path.join(self.reference_image_dir, f"{moiety_type}_contour_shape.npy")
        if os.path.exists(ref_contour_file):
            if at_origin:
                # Load the contour and shift it to the origin
                ref_contour = np.load(ref_contour_file, allow_pickle=True)
                ref_center, tgt_vecs = cv2.PCACompute(ref_contour[:, 0, :], mean=None, maxComponents=1)
                #ref_center = np.mean(ref_contour, axis=0)
                ref_contour_at_origin = ref_contour - ref_center
                return ref_contour_at_origin
            else:
                # Load the contour without shifting
                return np.load(ref_contour_file, allow_pickle=True)
        else:
            raise ValueError(f"Reference contour for {moiety_type} not found in {self.reference_image_dir}. Please create it first.")

    def determine_moiety_information_cartesian(self, ref_img_gray, tgt_img_gray, size_px, moiety_type, bbox, plot=True, pixels_per_unit=None):
        """
        Determines the moiety's position and orientation in Cartesian coordinates by matching a reference contour
        to the detected target region in a microscopy image using chamfer matching.

        Parameters:
            ref_img_gray (np.ndarray): Grayscale reference image.
            tgt_img_gray (np.ndarray): Grayscale target image.
            size_px (int): Image size in pixels (assumes square).
            moiety_type (str): Moiety identifier for config-based orientation.
            bbox (tuple): Normalized bounding box center and size (x, y, width, height).
            plot (bool): Whether to display plots.
            pixels_per_unit (float or None): Optional conversion from pixels to physical units (e.g., µm).

        Returns:
            position (np.ndarray): Cartesian (x, y) position.
            angle (float): Orientation in radians, CCW from x-axis.
            contour_target (np.ndarray): Target contour in Cartesian coordinates.
            contour_reference (np.ndarray): Transformed reference contour in Cartesian coordinates.
        """

        def preprocess_image(img):
            img_resized = cv2.resize(img, (size_px, size_px), interpolation=cv2.INTER_AREA)
            return cv2.bilateralFilter(img_resized, 9, 75, 75)

        ref_img_filtered = preprocess_image(ref_img_gray)
        tgt_img_filtered = preprocess_image(tgt_img_gray)

        # === REFERENCE CONTOUR =====================================
        ref_thresh = np.mean(ref_img_filtered) + np.std(ref_img_filtered)
        ref_bin = np.where(ref_img_filtered < ref_thresh, 0, 255).astype(np.uint8)
        ref_contours, _ = cv2.findContours(ref_bin, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_NONE)
        if not ref_contours:
            raise ValueError("No reference contour found.")
        reference_contour = max(ref_contours, key=cv2.contourArea)
        ref_pts = reference_contour[:, 0, :].astype(np.float64)

        ref_centroid, ref_vecs = cv2.PCACompute(ref_pts, mean=None, maxComponents=1)
        reference_orientation_rad = 0
        if 'molecule' in moiety_type:
            ref_orientation_deg = self.get_yaml_data(os.path.join(os.getcwd(), "input.yaml"), moiety_type, "reference_orientation_deg")
            reference_orientation_rad = np.deg2rad(ref_orientation_deg)

        if plot:
            vis = cv2.cvtColor(ref_bin, cv2.COLOR_GRAY2BGR)
            cv2.drawContours(vis, [reference_contour], -1, (0, 0, 255), 2)
            pt2 = ref_centroid[0] + 50 * np.array([np.cos(-reference_orientation_rad), np.sin(-reference_orientation_rad)])
            cv2.line(vis, tuple(ref_centroid[0].astype(int)), tuple(pt2.astype(int)), (0, 0, 255), 2)
            cv2.imshow("Reference Contour", vis)
            cv2.waitKey(0)
            cv2.destroyAllWindows()

        # === TARGET CONTOUR ========================================
        bbox_center = np.array([bbox[0], bbox[1]]) * size_px
        enlarge = 1
        x0 = max(0, int((bbox[0] - bbox[2] * enlarge) * size_px))
        x1 = min(size_px, int((bbox[0] + bbox[2] * enlarge) * size_px))
        y0 = max(0, int((bbox[1] - bbox[3] * enlarge) * size_px))
        y1 = min(size_px, int((bbox[1] + bbox[3] * enlarge) * size_px))

        cropped_target = tgt_img_filtered[y0:y1, x0:x1]
        offset = np.array([x0, y0])
        tgt_thresh = np.mean(tgt_img_filtered) + np.std(tgt_img_filtered)
        tgt_bin = np.where(cropped_target < tgt_thresh, 0, 255).astype(np.uint8)
        tgt_contours, _ = cv2.findContours(tgt_bin, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_NONE)
        if not tgt_contours:
            raise ValueError("No target contour found.")

        best_contour = min(tgt_contours, key=lambda cnt: np.mean(np.linalg.norm(cnt[:, 0, :] + offset - bbox_center, axis=1)**2))
        best_pts = best_contour[:, 0, :].astype(np.float64)
        tgt_centroid, tgt_vecs = cv2.PCACompute(best_pts, mean=None, maxComponents=1)
        tgt_orientation = np.arctan2(tgt_vecs[0, 1], tgt_vecs[0, 0])

        clean_contour = []
        for pt in best_contour + offset:
            x, y = pt[0]
            if 0 < x < size_px - 1 and 0 < y < size_px - 1:
                clean_contour.append([[x, y]])
        best_contour = np.array(clean_contour) - offset

        if plot:
            vis = cv2.cvtColor(tgt_bin, cv2.COLOR_GRAY2BGR)
            for pt in best_contour:
                cv2.circle(vis, tuple(pt[0]), 2, (0, 255, 0), -1)
            pt2 = tgt_centroid[0] + 50 * tgt_vecs[0]
            cv2.line(vis, tuple(tgt_centroid[0].astype(int)), tuple(pt2.astype(int)), (0, 255, 0), 2)
            cv2.imshow("Target Contour", vis)
            cv2.waitKey(0)
            cv2.destroyAllWindows()

        # === CONTOUR MATCHING ======================================
        angle_img, dx, dy = self.optimize_global(best_contour[:, 0], reference_contour[:, 0])
        angle = angle_img
        reference_contour_at_target = self.transform_contour(reference_contour[:, 0], angle, dx, dy)
        best_angle_before_symmetry = angle + reference_orientation_rad
        raw_angle_deg = np.rad2deg(best_angle_before_symmetry)
        print(f"Raw angle before symmetry adjustment: {np.rad2deg(best_angle_before_symmetry)} degrees")
        best_angle = self.adjust_orientation_angle_due_to_symmetry(best_angle_before_symmetry, moiety_type)
        print(f"Adjusted angle after symmetry: {np.rad2deg(best_angle)} degrees")

        ref_centroid, _ = cv2.PCACompute(reference_contour_at_target, mean=None)
        best_position = ref_centroid[0]

        # === CONVERT TO CARTESIAN COORDINATES ======================
        def to_cartesian_y(points):
            return np.stack([points[..., 0], points[..., 1]], axis=-1)

        cartesian_position = to_cartesian_y(best_position + offset)
        cartesian_contour_target = to_cartesian_y(best_contour + offset)
        cartesian_reference_contour = to_cartesian_y(reference_contour_at_target + offset)
        cartesian_angle = -best_angle  # flip Y axis implies angle sign flip

        if pixels_per_unit:
            cartesian_position /= pixels_per_unit
            cartesian_contour_target /= pixels_per_unit
            cartesian_reference_contour /= pixels_per_unit

        if plot:
            plt.figure()
            plt.scatter(*cartesian_contour_target.T, label='Target', s=100)
            plt.scatter(*cartesian_reference_contour.T, label='Reference', alpha=0.5)
            pt2 = cartesian_position + 50 * np.array([np.cos(cartesian_angle), np.sin(cartesian_angle)])
            plt.plot([cartesian_position[0], pt2[0]], [cartesian_position[1], pt2[1]], 'r-')
            plt.axis('equal')
            plt.legend()
            plt.title("Contours in Cartesian Coordinates")
            plt.show()

        return cartesian_position, cartesian_angle, cartesian_contour_target, cartesian_reference_contour, raw_angle_deg


    # def determine_moiety_information_px(self, ref_img_gray, tgt_img_gray, size_px, moiety_type, bbox, plot=True):
    #     """
    #     Determines the moiety's pixel-level position and orientation by matching a reference contour
    #     to the detected target region in a microscopy image using chamfer matching.

    #     """

    #     def preprocess_image(img):
    #         img_resized = cv2.resize(img, (size_px, size_px), interpolation=cv2.INTER_AREA)
    #         return cv2.bilateralFilter(img_resized, 9, 75, 75)

    #     ref_img_filtered = preprocess_image(ref_img_gray)
    #     tgt_img_filtered = preprocess_image(tgt_img_gray)

    #     # === REFERENCE CONTOUR & ORIENTATION ======================================
    #     ref_thresh = np.mean(ref_img_filtered) + np.std(ref_img_filtered)
    #     ref_bin = np.where(ref_img_filtered < ref_thresh, 0, 255).astype(np.uint8)
    #     ref_contours, _ = cv2.findContours(ref_bin, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_NONE)
    #     if not ref_contours:
    #         raise ValueError("No reference contour found.")
    #     reference_contour = max(ref_contours, key=cv2.contourArea)

    #     ref_pts = reference_contour[:, 0, :].astype(np.float64)
    #     ref_centroid, ref_vecs = cv2.PCACompute(ref_pts, mean=None, maxComponents=1)
    #     reference_position = np.array(ref_centroid[0], dtype=np.int32)

    #     if 'molecule' in moiety_type:
    #         ref_orientation_deg = self.get_yaml_data(os.path.join(os.getcwd(), "input.yaml"), moiety_type, "reference_orientation_deg")
    #         reference_orientation_rad = np.deg2rad(ref_orientation_deg)
    #         reference_orientation_img_rad = -reference_orientation_rad   # y-axis flip to be in image coordinates
    #     else:
    #         reference_orientation_rad = 0

    #     if plot:
    #         vis = cv2.cvtColor(ref_bin, cv2.COLOR_GRAY2BGR)
    #         cv2.drawContours(vis, [reference_contour], -1, (0, 0, 255), 2)
    #         pt2 = reference_position + 50 * np.array([np.cos(reference_orientation_img_rad), np.sin(reference_orientation_img_rad)])
    #         cv2.line(vis, tuple(reference_position), tuple(pt2.astype(int)), (0, 0, 255), 2)
    #         cv2.imshow("Reference Contour", vis)
    #         cv2.waitKey(0)
    #         cv2.destroyAllWindows()

    #     # === TARGET CONTOUR & ORIENTATION =========================================
    #     bbox_center = np.array([bbox[0], bbox[1]]) * size_px
    #     enlarge = 1 # not working due to negative cropping indices
    #     #bbox_x = np.array([bbox[0] - bbox[2] * enlarge, bbox[0] + bbox[2] * enlarge]) * size_px
    #     #bbox_y = np.array([bbox[1] - bbox[3] * enlarge, bbox[1] + bbox[3] * enlarge]) * size_px

    #     x0 = max(0, int((bbox[0] - bbox[2] * enlarge) * size_px))
    #     x1 = min(size_px, int((bbox[0] + bbox[2] * enlarge) * size_px))
    #     y0 = max(0, int((bbox[1] - bbox[3] * enlarge) * size_px))
    #     y1 = min(size_px, int((bbox[1] + bbox[3] * enlarge) * size_px))

    #     #x0, x1 = map(int, bbox_x)
    #     #y0, y1 = map(int, bbox_y)
    #     cropped_target = tgt_img_filtered[y0:y1, x0:x1]
    #     offset = np.array([x0,y0])

    #     tgt_thresh = np.mean(tgt_img_filtered) + np.std(tgt_img_filtered)
    #     tgt_bin = np.where(cropped_target < tgt_thresh, 0, 255).astype(np.uint8)

    #     tgt_contours, _ = cv2.findContours(tgt_bin, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_NONE)
    #     if not tgt_contours:
    #         raise ValueError("No target contour found.")

    #     # Choose contour closest to center
    #     best_contour = min(tgt_contours, key=lambda cnt: np.mean(np.linalg.norm(cnt[:, 0, :] + offset - bbox_center, axis=1)**2))
    #     best_pts = best_contour[:, 0, :].astype(np.float64)

    #     tgt_centroid, tgt_vecs = cv2.PCACompute(best_pts, mean=None, maxComponents=1)
    #     tgt_orientation = np.arctan2(tgt_vecs[0, 1], tgt_vecs[0, 0])

    #     # Remove border-hugging points
    #     clean_contour = []
    #     for pt in best_contour + offset:
    #         x, y = pt[0]
    #         if 0 < x < size_px - 1 and 0 < y < size_px - 1:
    #             clean_contour.append([[x, y]])
    #     best_contour = np.array(clean_contour) - offset

    #     if plot:
    #         vis = cv2.cvtColor(tgt_bin, cv2.COLOR_GRAY2BGR)
    #         for pt in best_contour:
    #             cv2.circle(vis, tuple(pt[0]), 2, (0, 255, 0), -1)
    #         pt2 = tgt_centroid[0] + 50 * tgt_vecs[0]
    #         cv2.line(vis, tuple(tgt_centroid[0].astype(int)), tuple(pt2.astype(int)), (0, 255, 0), 2)
    #         cv2.imshow("Target Contour", vis)
    #         cv2.waitKey(0)
    #         cv2.destroyAllWindows()

    #     # === CONTOUR MATCHING =====================================================
    #     angle_img, dx, dy = self.optimize_global(best_contour[:, 0], reference_contour[:, 0]) #
    #     #angle_img = angle_img #-angle_img  # Adjust for y-axis flip in image coordinates (map image coordinates to mathematical coordinates)
    #     reference_contour_at_target = self.transform_contour(reference_contour[:, 0], angle_img, dx, dy)
    #     best_angle_before_symmetry = angle_img+reference_orientation_rad

    #     # Log raw measurement angle before symmetry adjustment
    #     logger.bind(task='stats', raw_angle=int(np.rad2deg(best_angle_before_symmetry))).trace("")
    #     print(f"Raw angle before symmetry adjustment: {np.rad2deg(best_angle_before_symmetry)} degrees")
    #     best_angle = self.adjust_orientation_angle_due_to_symmetry(best_angle_before_symmetry, moiety_type)
    #     print(f"Adjusted angle after symmetry: {np.rad2deg(best_angle)} degrees")
    #     # Recompute orientation from transformed reference
    #     ref_centroid, ref_vecs = cv2.PCACompute(reference_contour_at_target, mean=None)
    #     #best_angle = np.arctan2(ref_vecs[0, 1], ref_vecs[0, 0])
    #     best_position = ref_centroid[0] #.astype(np.int32)

    #     # Convert img to cartesian coordinates
    #     best_angle = -best_angle  # Adjust for y-axis flip in image coordinates
    #     best_position[1] = size_px - best_position[1] # Flip y-axis to match image coordinates
    #     best_contour[:, :, 1] = size_px - best_contour[:, :, 1]
    #     reference_contour_at_target[:, 1] = size_px - reference_contour_at_target[:, 1]


    #     if plot:
    #         plt.scatter(*best_contour[:, 0].T, label='Target', s=100)
    #         plt.scatter(*reference_contour[:, 0].T, label='Reference')
    #         plt.scatter(*reference_contour_at_target.T, label='Reference', alpha=0.5)
    #         # Orientation vector of reference contour
    #         pt2 = best_position + 50 * np.array([np.cos(best_angle), np.sin(best_angle)]) # compensate the -angle necessary for y-axis flip
    #         plt.plot([best_position[0], pt2[0]], [best_position[1], pt2[1]], 'r-')
    #         plt.axis('equal')
    #         plt.legend()
    #         plt.show()

    #     # Return positions adjusted for original full image
    #     return (best_position + offset,
    #             best_angle,
    #             best_contour + offset,
    #             reference_contour_at_target.reshape(-1, 1, 2) + offset)

    def get_current_moiety_index(self):
        return self._current_moiety

    def get_active_moiety_information(self):
        """
        Returns the current moiety information.
        """
        return (self.moiety_types[self._current_moiety],
                self.moiety_position_nm[self._current_moiety],
                self.moiety_orientation_rad[self._current_moiety],
                self.moiety_bbox_width_height_nm[self._current_moiety],
                self.confidence[self._current_moiety],
                self.moiety_target_contour_nm[self._current_moiety],
                self.moiety_matching_reference_contour_nm[self._current_moiety],
                self.moiety_types_before[self._current_moiety],
                self.moiety_type_changes[self._current_moiety])

    def get_active_moiety_position(self):
        return self.moiety_position_nm[self._current_moiety]

    def does_action_intersect_moiety(self, start_tip_position, end_tip_position):
        """
        Check if the current action intersects with the active moiety.
        """
        intersects = False
        if start_tip_position is not None and end_tip_position is not None:
            # Convert start/end positions to np arrays if necessary
            start_tip = np.array([start_tip_position.x, start_tip_position.y])
            end_tip = np.array([end_tip_position.x, end_tip_position.y])

            # Create line from tip movement
            tip_line = LineString([start_tip, end_tip])

            # Get the contour of the manipulated moiety
            manipulated_contour = self.moiety_matching_reference_contour_nm[self._current_moiety]
            moiety_polygon = Polygon(manipulated_contour)

            # Check if the tip line intersects the moiety contour
            intersects = tip_line.intersects(moiety_polygon)
            # info = dict()
            # info['intersects'] = intersects
            # if not intersects:
            #     self.moiety_position_nm[self._current_moiety] = self.moiety_position_nm_before[self._current_moiety]
            return intersects

    def get_obstacles(self):
        # Get indices where moiety_type is not molecule or atom
        obstacles = np.where(np.array([False if 'molecule' in moiety_type or 'atom' in moiety_type else True for moiety_type in self.moiety_types]))[0]
        moieties = self.moiety_types, self.moiety_position_nm, self.moiety_orientation_rad, self.moiety_bbox_width_height_nm, self.confidence
        self.obstacles = [np.array([moiety[i] for i in obstacles]) for moiety in moieties]
        return self.obstacles

    def _set_current_moiety(self, moiety_index):
        self._current_moiety = np.asarray(moiety_index, dtype=int)

    def get_yaml_data(self, input_yaml_file, *args):
        """ Get input.yaml entry for specific arguments.

            Parameters
            ----------------
            args | str
                Keyword for which the value is read from the input.yaml file.

            Return
            ----------------
            value_read | str, float, int
        """
        with open(input_yaml_file, "r") as file:
            data = yaml.load(file, Loader=yaml.FullLoader)

        try:
            value = data
            for key in args:
                value = value[key]
        except KeyError as e:
            logger.warning(f"Could not fetch key {e=} from {input_yaml_file=}")

        return value

    @staticmethod
    def rotate_around_origin(origin, point, angle_rad):
        """
        Rotate a point counterclockwise by a given angle around a given origin.

        The angle should be given in radians.
        """
        ox, oy = origin
        px, py = point

        qx = ox + np.cos(angle_rad) * (px - ox) - np.sin(angle_rad) * (py - oy)
        qy = oy + np.sin(angle_rad) * (px - ox) + np.cos(angle_rad) * (py - oy)
        return np.array([qx, qy])


    @staticmethod
    def adjust_orientation_angle_due_to_symmetry(angle_rad, moiety_type):
        """ Adjust the orientation angle due to the adsorption geometry of the moiety on the substrate. """
        angle_rad = angle_rad % (2 * np.pi)  # Normalize angle to [0, 2*pi]

        if angle_rad >= 2*np.pi:
            angle_rad -= 2*np.pi
        elif angle_rad <= -2*np.pi:
            angle_rad += 2*np.pi

        if 'H2Pc_4K_Ag111' in moiety_type:
            # Moiety is 6-fold symmetric: discrete angles of 0, 30, 60, 90, 120, 150, 180 degrees
            if angle_rad >= np.pi:
                angle_rad -= np.pi
            elif angle_rad <= -np.pi:
                angle_rad += np.pi
            # Moiety angle discretization 30 degrees
            bined_angle_rad = np.round(angle_rad / np.pi * 6) * np.pi / 6

        elif 'FePc_Au111_4K' in moiety_type:
            # Moiety is 3-fold symmetric: discrete angles of 0, 30, 60 degrees
            angle_deg = np.rad2deg(angle_rad)
            angle = angle_deg % 90  # Normalize to [0, 90)
            logger.bind(task='stats', angle=angle_deg).warning(f"Angle before discretization: {angle_deg} degrees")
            # if angle < 15 or angle >= 75:
            #     bined_angle_deg = 0
            # elif angle < 45:
            #     bined_angle_deg = 30
            # else:
            #     bined_angle_deg = 60
            bined_angle_deg = angle
            bined_angle_rad = np.deg2rad(bined_angle_deg)
            # angle_in_sector = angle_rad % (np.pi / 2)  # Collapse to [0, π/2)
            # mapped_angle  = (angle_in_sector / (np.pi / 2)) * (np.pi / 3)  # Scale to [0, π/3)
            # bined_angle_rad = np.round(mapped_angle  / np.pi * 6) * np.pi / 6
            # if angle_rad >= np.pi/2:
            #     angle_rad -= np.pi/2
            # elif angle_rad <= -np.pi/2:
            #     angle_rad += np.pi/2
            # # Moiety angle discretization 30 degrees
            # bined_angle_rad = np.round(angle_rad / np.pi * 6) * np.pi / 6

        elif 'atom' in moiety_type:
            bined_angle_rad = 0


        return bined_angle_rad


    @staticmethod
    def convert_value(value):
        """Convert value to int, float, or str as appropriate."""
        values = value.split()
        converted_values = []

        if values:
            for val in values:
                try:
                    if '.' in val or 'E' in val or 'e' in val:
                        converted_values.append(float(val))
                    else:
                        converted_values.append(int(val))
                except ValueError:
                    converted_values.append(val)

            return converted_values if len(converted_values) > 1 else converted_values[0]
        else:
            return None


    def get_datafile_for_image(self, img_dir):

        # Check if the image file exists for the given path and correct the path if necessary
        if os.path.exists(img_dir+".dat"):
            return img_dir+".dat"
        elif os.path.exists(img_dir+".smx"):
            return img_dir+".smx"

        # Add .dat.jpeg to the image file
        # Createc
        if os.path.exists(img_dir.replace(".jpeg", ".dat")):
            dat_file = img_dir.replace(".jpeg", ".dat")
        elif os.path.exists(img_dir.replace(".dat.jpeg", ".dat")):
            dat_file = img_dir.replace(".dat.jpeg", ".dat")
        # Nanonis sxm file
        elif os.path.exists(img_dir) and img_dir.endswith(".sxm"):
            dat_file = img_dir
        else:
            raise ValueError("No dat file found for the image.")
        return dat_file

    def is_molecule_destroyed(self):
        " If confidence is below threshold the moiety could be destroyed or changed. Thus ask the user to confirm. "
        (
            moiety_type,
            moiety_position_nm,
            moiety_orientation_rad,
            moiety_bbox_width_height_nm,
            confidence,
            moiety_target_contour_nm,
            moiety_matching_reference_contour_nm,
            moiety_types_before,
            moiety_type_changes
        ) = self.get_active_moiety_information()

        # Check if the moiety is destroyed based on the confidence and type
        if confidence < self._CONFIDENCE_THRESHOLD and moiety_types_before != moiety_type:
            print(f"Moiety '{moiety_type}' destroyed since confidence {confidence} is below threshold")
            self.root = tk.Tk()
            self.selected_moiety = tk.StringVar()

            label = ttk.Label(self.root, text="Select a Moiety:")
            label.pack(pady=5)

            self.dropdown = ttk.Combobox(self.root, values=self.moiety_classes, textvariable=self.selected_moiety)
            self.dropdown.pack(pady=5)
            self.dropdown.set(moiety_type)  # Default selection

            confirm_button = ttk.Button(self.root, text="Confirm", command=self.button_confirm)
            confirm_button.pack(pady=5)

            destroy_button = ttk.Button(self.root, text="Destroyed", command=self.button_destroyed)
            destroy_button.pack(pady=5)

            self.root.mainloop()

            if self.moiety_types[self._current_moiety] == 'destroyed':
                print("Moiety destroyed")
                return True
            else:
                print(f"Moiety '{self.moiety_types[self._current_moiety]}' selected")
                return False

    def button_confirm(self):
         # remove " from the selected moiety

        self.moiety_types[self._current_moiety] = self.selected_moiety.get().strip("[]").strip("'")
        self.root.destroy()

    def button_destroyed(self):
        self.moiety_type = 'destroyed'
        self.root.destroy()

class AnalyseMoietyCreatec(AnalyseMoieties):
    def __init__(self):
        super().__init__()

    def get_stm_image(self, img_file):
        if img_file.endswith(".dat"):
            img_file = img_file.replace(".dat", ".dat.jpeg")
        # Add .dat.jpeg to the image file
        elif not img_file.endswith(".dat.jpeg"):
            img_file = img_file + ".dat.jpeg"

        # Get number of pixels from dat file
        dat_file = self.get_datafile_for_image(img_file)
        pixels = int(self.get_data_for_image(dat_file, "Num.X / Num.X"))

        # Check if the image file exists for the given path
        if not os.path.exists(img_file):
            raise ValueError(f"File not found: {img_file}")

        # Load the STM image
        img = cv2.imread(img_file, cv2.IMREAD_COLOR)
        return img, pixels

    @staticmethod
    def remove_background(image):

        # Create an initial mask of the same size as the image, filled with zeros.
        mask = np.zeros(image.shape[:2], np.uint8)

        # Allocate temporary arrays for the background and foreground models.
        bgdModel = np.zeros((1, 65), np.float64)
        fgdModel = np.zeros((1, 65), np.float64)

        # Let the user select a ROI (Region of Interest) that encloses the foreground.
        # This interactive step ensures that the algorithm "dynamically" knows which region to keep.
        print("Select the region of interest (ROI) for the foreground object and press ENTER or SPACE when done.")
        rect = cv2.selectROI("Input Image", image, fromCenter=False, showCrosshair=True)
        cv2.destroyWindow("Input Image")

        # If no ROI was selected, exit the function.
        if rect == (0, 0, 0, 0):
            print("No ROI selected. Exiting.")
            return

        # Apply the GrabCut algorithm with the selected rectangle.
        # The number of iterations (here 5) can be adjusted depending on image complexity.
        cv2.grabCut(image, mask, rect, bgdModel, fgdModel, 5, cv2.GC_INIT_WITH_RECT)

        # Create a binary mask where sure or likely background pixels are 0 and foreground pixels are 1.
        mask2 = np.where((mask == cv2.GC_BGD) | (mask == cv2.GC_PR_BGD), 0, 1).astype('uint8')

        # Multiply the original image with the binary mask to extract the foreground.
        image_foreground = image * mask2[:, :, np.newaxis]

        # Save the output image (background removed)
        # cv2.imwrite(output_path, image_foreground)
        # print(f"Background removed image saved as {output_path}")

        # Optionally, display the result.
        cv2.imshow("Foreground", image_foreground)
        cv2.waitKey(0)
        cv2.destroyAllWindows()
        return image_foreground



    def get_moiety_information_px(self, reference_path, target_path, moiety_type, bbox, plot=False):
        """
        Get the moiety's pixel-level position and orientation by matching a reference contour
        to the detected target region in a microscopy image using chamfer matching.

        Parameters:
        -----------
        ref_img : str
            Path to the reference image file.
        img_file : str
            Path to the target image file.
        moiety_type : str
            Type of moiety (e.g., 'molecule').
        bbox : list[float]
            Bounding box around predicted object (center_x, center_y, half_width, half_height) in normalized coordinates.
        plot : bool
            Whether to visualize matching results.

        Returns:
        --------
        best_matching_position : np.ndarray
            [x, y] coordinates of moiety position in pixels.
        best_matching_orientation_rad : float
            Orientation in radians.
        best_target_contour : np.ndarray
            Target contour points.
        best_matching_reference_contour : np.ndarray
            Transformed reference contour points.
        """
        # === IMAGE LOADING & PREPROCESSING ========================================
        ref_img_gray = cv2.imread(reference_path, cv2.IMREAD_GRAYSCALE)
        tgt_img_gray = cv2.imread(target_path, cv2.IMREAD_GRAYSCALE)

        # Ensure that the reference and target images have the same size
        dat_reference_path = reference_path.replace(".jpeg", ".dat")
        if not os.path.exists(dat_reference_path):
            dat_reference_path = reference_path.replace(".dat.jpeg", ".dat")
        dat_target_path = target_path.replace(".jpeg", ".dat")
        if not os.path.exists(dat_target_path):
            dat_target_path = target_path.replace(".dat.jpeg", ".dat")

        _, img_size_px, _, _ = self.get_data_for_image(dat_reference_path, "Num.X / Num.X")
        size_px = img_size_px['x']
        assert size_px == img_size_px['y'], "Non-square image dimensions not supported."

        return self.determine_moiety_information_cartesian(ref_img_gray, tgt_img_gray, size_px, moiety_type, bbox, plot=True)

    def init_moiety_information(self, img_file, numbering=True, plot=False):
        """
        Initialize the information of the moieties based on the prediction of the object recognition.

        Parameters:
        -----------
        pred: list
            The prediciton of the object recognition:
            0: type of moiety
            1: x_center
            2: y_center
            3: width
            4: height
            5: confidence

        Returns:
        --------
        moiety_type: str
            List containing the type of the moieties
        moiety_positions: np.array
            List containing the positions of the moieties in pixel
        moiety_orientations: float
            List containing the orientations of the moieties in degrees
        moiety_bbox: np.array
            List containing the bounding boxes of the moieties in pixel
        """
        # Object detection prediction
        img, img_size = self.get_stm_image(img_file)
        pred = self.object_recognition.predict(img, confidence_threshold=self._INIT_CONFIDENCE_THRESHOLD)
        self.object_recognition.plot_predictions(img, pred, numbering=numbering, plot=plot)



        dat_file = self.get_datafile_for_image(img_file)
        real_img_size_px = self.get_data_for_image(dat_file, "Num.X / Num.X")
        img_size_nm = self.convert_image_pixel_to_nm(img_file=img_file)
        assert real_img_size_px == self.get_data_for_image(dat_file, "Num.Y / Num.Y")

        # Get the position of the overview image in the general coordinate system
        origin_position_of_overview_image_nm = self.get_origin_of_image_nm(img_file)

        # Setup or load obstacles from the object detection
        obstacles_px = self.object_recognition.get_obstacles(img, img_file, pred)
        # iterate over the obstacles and convert them to nm
        obstacles_nm = np.array([self.convert_pixel_to_nm(np.asarray(obstacles_px[i]), img_file) for i in range(len(obstacles_px)) if obstacles_px[i] is not None], dtype=object)
        # interate over the obstacles and convert them to global coordinates
        self._obstacles_nm = [obstacles_nm[i] + np.array([-img_size_nm['x']/2,0]) + np.array(origin_position_of_overview_image_nm) for i in range(len(obstacles_nm))]

        print("File: ", img_file)
        print("Origin position of overview image: ", origin_position_of_overview_image_nm)

        # Assign the moiety type, position and bounding box width and heigth to the global coordinate system
        self.moiety_types = np.array([self.moiety_classes[int(i)] for i in pred.T[0]])
        self.moiety_types_before = copy.deepcopy(self.moiety_types)
        # Init type changes with False
        self.moiety_type_changes = np.zeros(len(self.moiety_types), dtype=bool)

        moiety_positions_px = np.array([pred.T[1], pred.T[2]]).T*real_img_size_px
        self.moiety_position_nm = self.convert_pixel_to_nm(moiety_positions_px, img_file=img_file) + origin_position_of_overview_image_nm
        print("WARNING: Moiety positions correctly shifted by image offset to origin???: ", moiety_positions_px, self.moiety_position_nm )

        self.moiety_orientation_rad = np.zeros(len(self.moiety_types))*np.nan

        moiety_bbox_px = np.array(pred.T[3:5]).T*real_img_size_px
        self.moiety_bbox_width_height_nm = self.convert_pixel_to_nm(moiety_bbox_px, img_file=img_file)
        self.confidence = pred.T[5]

        self.moiety_target_contour_nm = np.empty(len(self.moiety_types),dtype=object)
        self.moiety_matching_reference_contour_nm = np.empty(len(self.moiety_types),dtype=object)

        return self.moiety_types, self.moiety_position_nm, self.moiety_orientation_rad, self.moiety_bbox_width_height_nm, self.confidence

    def get_moiety_indices(self, moiety_type, moiety_position_nm, position_threshold_nm=0.5):
        """
        Determine the index of the moiety based on the type and the bounding box.

        Parameters:
        -----------
        moiety_type: str
            The type of the moiety.
        bbox: np.array
            The bounding box of the moiety.

        Returns:
        --------
        moiety_index: int
            The index of the moiety.
        """
        # Determine the index of the moiety
        moiety_index_type = np.where((self.moiety_types == moiety_type))

        # Determine the index of the moiety based on the position in nm within a tolerance of 0.5 nm
        moiety_index_position = np.where(np.linalg.norm(self.moiety_position_nm - moiety_position_nm, axis=1) < position_threshold_nm)

        moiety_index = np.intersect1d(moiety_index_type, moiety_index_position)

        return moiety_index


    def update_moiety_information(self, img_file, start_tip_position=None, end_tip_position=None, active_moiety_index=None):
        """
        Determine the exact position of the predicted moiety in the image based on the respective reference image.
        Only for a moiety with prefix 'molecule_' or 'atom_' the exact position is determined. Other moieties
        are classified by the center of the bounding box.

        Parameters:
        -----------
        image: np.array
            The measured image containing the moieties.
        pred: list
            The prediciton of the object recognition:
            0: type of moiety
            1: x_center
            2: y_center
            3: width
            4: height
            5: confidence

        Returns:
        --------
        moiety_positions: np.array
            List containing the positions of the moieties in nm
        moiety_orientations: float
            List containing the orientations of the moieties in degrees
        moiety_type: str
            List containing the type of the moieties
        moiety_contour: np.array
            List containing the contours of the moiety
        """
        # Object detection prediction
        img, img_size = self.get_stm_image(img_file)
        pred = self.object_recognition.predict(img, confidence_threshold=self._CONFIDENCE_THRESHOLD)
        self.object_recognition.plot_predictions(img, pred)

        dat_file = self.get_datafile_for_image(img_file)
        real_img_size_px = self.get_data_for_image(dat_file, "Num.X / Num.X")
        img_size_nm = self.convert_image_pixel_to_nm(real_img_size_px, img_file=img_file)
        origin_img_nm = self.get_origin_of_image_nm(img_file)

        print("File: ", img_file)
        print("Origin position of measured image: ", origin_img_nm)

        # Assign the moiety of the overall image to the moiety of the current image
        moiety_types = np.array([self.moiety_classes[int(i)] for i in pred.T[0]])
        moiety_positions = np.array([pred.T[1], pred.T[2]]).T
        moiety_bbox = np.array(pred.T[1:5]).T
        confidence = pred.T[5]

        for i, moiety_type in enumerate(moiety_types):
            # Determine exact position for moieties with prefix 'molecule_' or 'atom_'
            if 'molecule_' in moiety_type or 'atom_' in moiety_type:
                ref_img_file = os.path.join(self.reference_image_dir, str(moiety_type)+"_"+str(real_img_size_px)+".jpeg")

                # Save reference image if it does not exist
                self.save_reference_image(img_file=img_file, moiety_type=moiety_type)

                # Check if the reference image exists
                if not os.path.exists(ref_img_file):
                    ref_img_file = os.path.join(self.reference_image_dir, str(moiety_type)+"_"+str(real_img_size_px)+".dat.jpeg")

                start_time = time.time()
                (
                    moiety_position_px,
                    moiety_orientation_rad,
                    moiety_target_contour_px,
                    moiety_matching_reference_contour_px
                ) = self.get_moiety_information_px(ref_img_file, img_file, moiety_type=moiety_type, bbox=moiety_bbox[i], plot=False)
                print("Time for determining moiety information: ", time.time()-start_time)

                # Determine index of the moiety that gets updated in the global moiety information
                moiety_position_nm = self.convert_pixel_to_nm(moiety_position_px, img_file=img_file) + origin_img_nm
                moiety_position_nm = moiety_position_nm + np.array([-img_size_nm['x']/2,0]) + np.array(origin_img_nm) # Createc center only shifted in x direction

                moiety_index = self.get_moiety_indices(moiety_type=moiety_type, moiety_position_nm=moiety_position_nm)
                moiety_bbox_width_height_nm = self.convert_pixel_to_nm(moiety_bbox[i][2:5]*real_img_size_px, img_file=img_file)

                moiety_target_contour_nm = self.convert_pixel_to_nm(moiety_target_contour_px, img_file=img_file) + origin_img_nm
                moiety_matching_reference_contour_nm = self.convert_pixel_to_nm(moiety_matching_reference_contour_px, img_file=img_file) + origin_img_nm

                assert moiety_index.size > 0, "Moiety index is empty"

                self.moiety_types_before[moiety_index] = self.moiety_types[moiety_index]
                self.moiety_types[moiety_index] = moiety_type
                self.moiety_type_changes[moiety_index] = moiety_type != self.moiety_types_before[moiety_index]
                self.moiety_position_nm[moiety_index] = moiety_position_nm
                self.moiety_orientation_rad[moiety_index] = moiety_orientation_rad
                self.moiety_bbox_width_height_nm[moiety_index] = moiety_bbox_width_height_nm
                self.confidence[moiety_index] = confidence[i]

                self.moiety_target_contour_nm[int(moiety_index)] = moiety_target_contour_nm
                self.moiety_matching_reference_contour_nm[int(moiety_index)] = moiety_matching_reference_contour_nm

        return self.moiety_types, self.moiety_position_nm, self.moiety_orientation_rad, self.moiety_bbox_width_height_nm, self.confidence, self.moiety_target_contour_nm, self.moiety_matching_reference_contour_nm, self.moiety_types_before, self.moiety_type_changes, {}

    def save_reference_image(self, img_file, moiety_type):
        """
        Save the current STM image as a new reference image if no reference image exists for the current moiety.
        If a reference image already exists, it is not overwritten.
        """
        # Get image
        img, img_size = self.get_stm_image(img_file)

        reference_filename = f"{moiety_type}_{img_size}.jpeg"
        reference_image_path = os.path.join(self.reference_image_dir, reference_filename)

        # Check if the reference image already exists
        if not os.path.exists(reference_image_path):
            cv2.imwrite(reference_image_path, img)

    def save_reference_contour(self, ref_contour_nm, moiety_type, plot=False):
        """ Saves the contour of the moiety as a reference contour file.
        """
        # Save file if it does not exist
        file = os.path.join(self.reference_image_dir, f"{moiety_type}_contour_shape.npy")
        if not os.path.exists(file):
            np.save(file, ref_contour_nm)

            # Plot the contour if requested
            if plot:
                moiety_matching_reference_contour_nm = np.load(file)
                fig = plt.figure(figsize=(10, 10))
                ax = fig.add_subplot(111)
                ax.set_title(f"Reference contour for {moiety_type}")
                ax.set_xlabel("x (nm)")
                ax.set_ylabel("y (nm)")
                ax.scatter(moiety_matching_reference_contour_nm.T[0], moiety_matching_reference_contour_nm.T[1], c='red', label='Matching Reference Contour')
                ax.legend()
                plt.show()
                print("Reference contour saved and plotted.")

    def get_center_of_image_nm(self, img_file):

        dat_file = self.get_datafile_for_image(img_file)
        global_image_offset_x_dac = -self.get_data_for_image(dat_file, 'Scanrotoffx / OffsetX')
        global_image_offset_y_dac = -self.get_data_for_image(dat_file, 'Scanrotoffy / OffsetY')
        real_img_size_y_px = self.get_data_for_image(dat_file, "Num.Y / Num.Y")
        real_img_size_y_dac = self.convert_pixel_to_dac(real_img_size_y_px, img_file=img_file)
        center_of_image = np.array([global_image_offset_x_dac, global_image_offset_y_dac + real_img_size_y_dac/2])
        return self.convert_dac_to_nm(center_of_image, img_file=img_file)


    def get_origin_of_image_nm(self, img_file):
        """
        Get the origin of the image in the global coordinate system.

        Parameters:
        -----------
        img_file: str
            The path to the image file.

        Returns:
        --------
        origin_position_of_overview_image: np.array
            The origin position of the image in the global coordinate system.
        """

        dat_file = self.get_datafile_for_image(img_file)
        global_image_offset_x_dac = -self.get_data_for_image(dat_file, 'Scanrotoffx / OffsetX')
        global_image_offset_y_dac = -self.get_data_for_image(dat_file, 'Scanrotoffy / OffsetY')
        real_img_size_x_px = self.get_data_for_image(dat_file, "Num.X / Num.X")
        real_img_size_y_px = self.get_data_for_image(dat_file, "Num.Y / Num.Y")
        real_img_size_x_dac = self.convert_pixel_to_dac(real_img_size_x_px, img_file=img_file)
        #real_img_size_y_dac = self.convert_pixel_to_dac(real_img_size_y_px, img_file=img_file)
        origin_position_of_overview_image_dac = np.array([global_image_offset_x_dac - real_img_size_x_dac/2, global_image_offset_y_dac])
        return self.convert_dac_to_nm(origin_position_of_overview_image_dac, img_file=img_file)


    def plot_moieties(self, img_file, plot=True):
        """
        Plot the overview image and the measured target image with the determined contours.
        """
        if img_file.endswith(".dat"):
            img_file = img_file.replace(".dat", ".dat.jpeg")
        dat_file = self.get_datafile_for_image(img_file)
        real_img_size = self.get_data_for_image(dat_file, "Num.X / Num.X")
        origin_img_nm = self.get_origin_of_image_nm(img_file)


        # Plot the moiety positions and contours in the overview image
        if plot:
            overview_img = cv2.imread(img_file)
            overview_img = cv2.resize(overview_img, (real_img_size, real_img_size))

            for i, moiety_position_nm in enumerate(self.moiety_position_nm):
                cv2.drawMarker(overview_img, np.array(self.convert_nm_to_pixel(moiety_position_nm-origin_img_nm, img_file=img_file)).astype(int), (106, 24, 72), markerType=cv2.MARKER_CROSS, markerSize=20, thickness=2)

                if self.moiety_target_contour_nm[i] is not None:
                    #cv2.drawContours(overview_img, np.array(self.convert_nm_to_pixel(self.moiety_target_contour_nm[i]-origin_img_nm, img_file=img_file)).astype(int), -1, (255, 0, 0), 2)
                    cv2.drawMarker(overview_img, np.array(self.convert_nm_to_pixel(moiety_position_nm-origin_img_nm, img_file=img_file)).astype(int), (96, 32, 0), markerType=cv2.MARKER_CROSS, markerSize=20, thickness=2)
                    cv2.drawContours(overview_img, np.array(self.convert_nm_to_pixel(self.moiety_matching_reference_contour_nm[i]-origin_img_nm, img_file=img_file)).astype(int), -1, (96, 32, 0), 2)
                    # Draw the orientation of the moiety
                    cv2.line(overview_img, tuple(np.array(self.convert_nm_to_pixel(moiety_position_nm-origin_img_nm, img_file=img_file)).astype(int)),
                        (int(np.array(self.convert_nm_to_pixel(moiety_position_nm-origin_img_nm, img_file=img_file)).astype(int)[0] + 50*np.cos(self.moiety_orientation_rad[i])),
                        int(np.array(self.convert_nm_to_pixel(moiety_position_nm-origin_img_nm, img_file=img_file)).astype(int)[1] + 50*np.sin(self.moiety_orientation_rad[i]))), (96, 32, 0), 2)

            cv2.imshow("Overview Image", overview_img)
            cv2.waitKey(0)
            cv2.destroyAllWindows()


    def convert_image_pixel_to_nm(self, img_file):
        # Check if the dat file exists
        if os.path.exists(img_file+".dat"):
            dat_file = img_file+".dat"
        else:
            dat_file = img_file.replace(".jpeg", ".dat")
            if not os.path.exists(dat_file):
                dat_file = img_file.replace(".dat.jpeg", ".dat")

        CONSTANT_AD_CONVERTER = 524287
        # --- These are just defaul values. They are overwritten by the values from the STM/AFM program if the connection is established.
        deltaX = self.get_data_for_image(dat_file, "Delta X / Delta X [Dac]")
        deltaY = self.get_data_for_image(dat_file, "Delta Y / Delta Y [Dac]")

        numX = self.get_data_for_image(dat_file, "Num.X")
        numY = self.get_data_for_image(dat_file, "Num.Y")

        GainX = self.get_data_for_image(dat_file, "GainX / GainX")
        GainY = self.get_data_for_image(dat_file, "GainY / GainY")

        Xpiezoconst = self.get_data_for_image(dat_file, "Xpiezoconst / Xpiezoconst")
        Ypiezoconst = self.get_data_for_image(dat_file, "YPiezoconst / YPiezoconst")

        assert deltaX*numX/CONSTANT_AD_CONVERTER*GainX*Xpiezoconst == deltaY*numY/CONSTANT_AD_CONVERTER*GainY*Ypiezoconst

        return deltaX*numX/CONSTANT_AD_CONVERTER*GainX*Xpiezoconst

    def convert_pixel_to_dac(self, pixel, img_file):
        # Check if the dat file exists
        if os.path.exists(img_file+".dat"):
            dat_file = img_file+".dat"
        else:
            dat_file = img_file.replace(".jpeg", ".dat")
            if not os.path.exists(dat_file):
                dat_file = img_file.replace(".dat.jpeg", ".dat")

        # --- These are just defaul values. They are overwritten by the values from the STM/AFM program if the connection is established.
        deltaX = self.get_data_for_image(dat_file, "Delta X / Delta X [Dac]")
        deltaY = self.get_data_for_image(dat_file, "Delta Y / Delta Y [Dac]")

        assert deltaX == deltaY

        return pixel*deltaX

    def convert_pixel_to_nm(self, pixel, img_file):
        # Check if the dat file exists
        if os.path.exists(img_file+".dat"):
            dat_file = img_file+".dat"
        else:
            dat_file = img_file.replace(".jpeg", ".dat")
            if not os.path.exists(dat_file):
                dat_file = img_file.replace(".dat.jpeg", ".dat")

        CONSTANT_AD_CONVERTER = 524287
        # --- These are just defaul values. They are overwritten by the values from the STM/AFM program if the connection is established.
        deltaX = self.get_data_for_image(dat_file, "Delta X / Delta X [Dac]")
        deltaY = self.get_data_for_image(dat_file, "Delta Y / Delta Y [Dac]")

        numX = self.get_data_for_image(dat_file, "Num.X")
        numY = self.get_data_for_image(dat_file, "Num.Y")

        GainX = self.get_data_for_image(dat_file, "GainX / GainX")
        GainY = self.get_data_for_image(dat_file, "GainY / GainY")

        Xpiezoconst = self.get_data_for_image(dat_file, "Xpiezoconst / Xpiezoconst")
        Ypiezoconst = self.get_data_for_image(dat_file, "YPiezoconst / YPiezoconst")

        assert deltaX/CONSTANT_AD_CONVERTER*GainX*Xpiezoconst == deltaY/CONSTANT_AD_CONVERTER*GainY*Ypiezoconst

        return pixel*deltaX*GainX*Xpiezoconst/CONSTANT_AD_CONVERTER

    def convert_nm_to_pixel(self, nm, img_file):
        # Check if the dat file exists
        if os.path.exists(img_file+".dat"):
            dat_file = img_file+".dat"
        else:
            dat_file = img_file.replace(".jpeg", ".dat")
            if not os.path.exists(dat_file):
                dat_file = img_file.replace(".dat.jpeg", ".dat")

        CONSTANT_AD_CONVERTER = 524287
        # --- These are just defaul values. They are overwritten by the values from the STM/AFM program if the connection is established.
        deltaX = self.get_data_for_image(dat_file, "Delta X / Delta X [Dac]")
        deltaY = self.get_data_for_image(dat_file, "Delta Y / Delta Y [Dac]")

        numX = self.get_data_for_image(dat_file, "Num.X")
        numY = self.get_data_for_image(dat_file, "Num.Y")

        GainX = self.get_data_for_image(dat_file, "GainX / GainX")
        GainY = self.get_data_for_image(dat_file, "GainY / GainY")

        Xpiezoconst = self.get_data_for_image(dat_file, "Xpiezoconst / Xpiezoconst")
        Ypiezoconst = self.get_data_for_image(dat_file, "YPiezoconst / YPiezoconst")

        assert deltaX/CONSTANT_AD_CONVERTER*GainX*Xpiezoconst == deltaY/CONSTANT_AD_CONVERTER*GainY*Ypiezoconst

        return nm*CONSTANT_AD_CONVERTER/(deltaX*GainX*Xpiezoconst)

    def convert_dac_to_nm(self, dac, img_file):
        # Check if the dat file exists
        if os.path.exists(img_file+".dat"):
            dat_file = img_file+".dat"
        else:
            dat_file = img_file.replace(".jpeg", ".dat")
            if not os.path.exists(dat_file):
                dat_file = img_file.replace(".dat.jpeg", ".dat")

        CONSTANT_AD_CONVERTER = 524287
        # --- These are just defaul values. They are overwritten by the values from the STM/AFM program if the connection is established.

        GainX = self.get_data_for_image(dat_file, "GainX / GainX")
        GainY = self.get_data_for_image(dat_file, "GainY / GainY")

        Xpiezoconst = self.get_data_for_image(dat_file, "Xpiezoconst / Xpiezoconst")
        Ypiezoconst = self.get_data_for_image(dat_file, "YPiezoconst / YPiezoconst")

        assert GainX*Xpiezoconst/CONSTANT_AD_CONVERTER == GainY*Ypiezoconst/CONSTANT_AD_CONVERTER

        return dac*GainX*Xpiezoconst/CONSTANT_AD_CONVERTER

    def get_data_for_image(self, file, *args):
        """ Get *.dat file entry for specific arguments.

            Parameters
            ----------------
            args | str
                Keyword for which the value is read from the *.dat file.

            Return
            ----------------
            value_read | str, float, int
        """

        with open(file, "r", encoding="ISO-8859-1") as f:
            lines = f.readlines()

        values = []
        for arg in args:
            for line in lines:
                if arg in line:
                    values.append(line.split("=")[1].strip())

        # Convert value to int, float, or str
        if len(values) == 1:
            try:
                return int(values[0])
            except ValueError:
                try:
                    return float(values[0])
                except ValueError:
                    return values[0]

        return np.array(values)


class AnalyseMoietyNanonis(AnalyseMoieties):
    def __init__(self):
        super().__init__()

    def get_data_from_sxm(self, img_file, specific_key=None):
        """
        Reads a Nanonis data file and extracts a specific key-value pair.

        :param file_path: Path to the file.
        :param specific_key: Key to extract (None for all).
        :return: Dictionary containing extracted data or the specific key's value.
        """
        data = {}
        key = None
        value_lines = []

        with open(img_file, 'r', encoding='ISO-8859-1') as file:
            for line in file:
                line = line.strip()

                if line.startswith(':'):
                    # Save the previous key-value pair
                    if key is not None:
                        processed_value = " ".join(value_lines).strip()
                        data[key] = self.convert_value(processed_value)

                        # If a specific key is requested, return its value
                        if specific_key and key == specific_key:
                            return {key: data[key]}

                    # Start a new key
                    key = re.sub(r'[:]', '', line)  # Remove leading ':'
                    value_lines = []
                else:
                    # Accumulate values for the current key
                    value_lines.append(line)

        # Save the last key-value pair
        if key is not None:
            processed_value = " ".join(value_lines).strip()
            data[key] = self.convert_value(processed_value)

        return data if not specific_key else {specific_key: data.get(specific_key, None)}

    def get_stm_image(self, img_file):
        # Make sure image file ends with .sxm
        # assert img_file.endswith('.sxm')
        # Get the Z channel of the image and the number of pixels
        image_data = self.get_image_data_from_sxm(img_file)
        img_Z_m, pixels = image_data[0][1], image_data[1]['x']
        # Rescale values of m to pixel values between 0 and 255
        img_Z_m_min = np.min(img_Z_m)
        img_Z_m_max = np.max(img_Z_m)
        if img_Z_m_max == img_Z_m_min:
            return np.zeros_like(img_Z_m, dtype=np.uint8)
        img_Z = ((img_Z_m - img_Z_m_min) / (img_Z_m_max - img_Z_m_min) * 255).astype(np.uint8)
        # Convert to RGB image
        img_Z_rgb = cv2.cvtColor(img_Z, cv2.COLOR_GRAY2RGB)
        return img_Z_rgb, pixels


    @staticmethod
    def detect_horizontal_lines(img_rgb, min_line_length=50, max_line_gap=10, margin_ratio=0.1):
        """
        Detects horizontal lines in the image, excluding lines at the top and bottom edges.

        Parameters:
            img_rgb (np.ndarray): The RGB image obtained from STM data.
            min_line_length (int): Minimum line length for Hough Transform.
            max_line_gap (int): Maximum allowed gap between line segments.
            margin_ratio (float): Fraction of image height to exclude from top and bottom.

        Returns:
            List of lines [(x1, y1, x2, y2)], where each line is horizontal.
        """
        # Convert to grey scale
        img_gray = cv2.cvtColor(img_rgb, cv2.COLOR_RGB2GRAY
                                )
        # Apply edge detection (Canny)
        edges = cv2.Canny(img_gray, 50, 150, apertureSize=3)

        # Get image dimensions
        height, width = img_gray.shape

        # Define top and bottom margins to exclude
        top_margin = int(height * margin_ratio)
        bottom_margin = height - top_margin

        # Detect lines using Probabilistic Hough Transform
        lines = cv2.HoughLinesP(edges, rho=1, theta=np.pi/180, threshold=50,
                                minLineLength=min_line_length, maxLineGap=max_line_gap)

        horizontal_lines = []

        if lines is not None:
            for line in lines:
                x1, y1, x2, y2 = line[0]
                # Check if line is horizontal (within a small angle tolerance)
                if abs(y2 - y1) <= 2:
                    # Exclude lines at the top and bottom margins
                    if top_margin < y1 < bottom_margin and top_margin < y2 < bottom_margin:
                        horizontal_lines.append((x1, y1, x2, y2))

        contains_horizontal_lines = False
        if horizontal_lines:
            contains_horizontal_lines = True

        return horizontal_lines, contains_horizontal_lines


    def get_image_data_from_sxm(self, img_file):
        if not img_file.endswith(".sxm"):
            img_file += '.sxm'
        load = NanonisSXM(img_file)
        assert len(load.channels_name) >= 4, "The measurement does not contain the necessary channels: Current, Z, Y, X"

        current = load.retrieve_channel_data('Current')
        zz = load.retrieve_channel_data('Z')
        yy = load.retrieve_channel_data('Y')
        xx = load.retrieve_channel_data('X')

        scan_dir = load.header['SCAN_DIR'][0][0]
        pixels = {'x': int(load.header['SCAN_PIXELS'][0][0]),
                  'y': int(load.header['SCAN_PIXELS'][0][1])}
        real_nm = {'x': 1e9 * float(load.header['SCAN_RANGE'][0][0]),
                   'y': 1e9 * float(load.header['SCAN_RANGE'][0][1])}
        offset_nm = {'x': 1e9 * float(load.header['SCAN_OFFSET'][0][0]),
                     'y': 1e9 * float(load.header['SCAN_OFFSET'][0][1])}
        offset_nm = (offset_nm['x'], offset_nm['y'])
        if scan_dir == 'up':
            data = np.flip(current, axis=0), np.flip(zz, axis=0), np.flip(yy, axis=0), np.flip(xx, axis=0)
        else:
            data = current, zz, yy, xx
        return data, pixels, real_nm, offset_nm

    def init_moiety_information(self, img_file, numbering=True, plot=False):
        """
        Initialize the information of the moieties based on the prediction of the object recognition.

        Parameters:
        -----------
        pred: list
            The prediciton of the object recognition:
            0: type of moiety
            1: x_center
            2: y_center
            3: width
            4: height
            5: confidence

        Returns:
        --------
        moiety_type: str
            List containing the type of the moieties
        moiety_positions: np.array
            List containing the positions of the moieties in pixel
        moiety_orientations: float
            List containing the orientations of the moieties in degrees
        moiety_bbox: np.array
            List containing the bounding boxes of the moieties in pixel
        """
        # Object detection prediction
        img, img_info = self.get_stm_image(img_file)
        pred = self.object_recognition.predict(img, confidence_threshold=self._INIT_CONFIDENCE_THRESHOLD)
        self.object_recognition.plot_predictions(img, pred, numbering=numbering, plot=plot)

        data, img_size_px, img_size_nm, offset  = self.get_image_data_from_sxm(img_file)
        real_img_size = img_size_px['x']
        assert real_img_size == img_size_px['y']

        # Get the position of the overview image in the general coordinate system
        self.size_overview_image_nm = round(img_size_nm['x'],0), round(img_size_nm['y'],0)
        self.origin_position_of_overview_image_nm = np.array(offset)

        # Setup or load obstacles from the object detection
        obstacles_px = self.object_recognition.get_obstacles(img, img_file, pred)
        # iterate over the obstacles and convert them to nm
        obstacles_nm = np.array([self.convert_pixel_to_nm(np.asarray(obstacles_px[i]), img_file) for i in range(len(obstacles_px)) if obstacles_px[i] is not None], dtype=object)
        # interate over the obstacles and convert them to global coordinates
        self._obstacles_nm = [obstacles_nm[i] + np.array([-img_size_nm['x']/2,img_size_nm['y']/2]) + np.array(self.origin_position_of_overview_image_nm) for i in range(len(obstacles_nm))]

        print("File: ", img_file)
        print("Origin position of overview image: ", self.origin_position_of_overview_image_nm)

        # Assign the moiety type, position and bounding box width and heigth to the global coordinate system
        self.moiety_types = np.array([self.moiety_classes[int(i)] for i in pred.T[0]])
        self.moiety_types_before = copy.deepcopy(self.moiety_types)
        # Init type changes with False
        self.moiety_type_changes = np.zeros(len(self.moiety_types), dtype=bool)

        moiety_positions_px = np.atleast_2d([pred.T[1], pred.T[2]]).T*real_img_size
        moiety_positions_nm = np.asarray(self.convert_pixel_to_nm(moiety_positions_px, img_file=img_file))
        #self.moiety_position_nm = np.empty((0, 2))
        self.moiety_position_nm = (moiety_positions_nm + np.array([-img_size_nm['x']/2,img_size_nm['y']/2]) + np.array(self.origin_position_of_overview_image_nm) )
        #self.moiety_position_nm =  np.vstack((self.moiety_position_nm, new_moiety_positions[np.newaxis, :]))

        self.moiety_orientation_rad = np.zeros(len(self.moiety_types))*np.nan

        moiety_bbox_px = np.array(pred.T[3:5]).T*real_img_size
        self.moiety_bbox_width_height_nm = self.convert_pixel_to_nm(moiety_bbox_px, img_file=img_file)
        self.confidence = get_confidence(pred)

        self.moiety_target_contour_nm = np.empty(len(self.moiety_types),dtype=object)
        self.moiety_matching_reference_contour_nm = np.empty(len(self.moiety_types),dtype=object)

        return self.moiety_types, self.moiety_position_nm, self.moiety_orientation_rad, self.moiety_bbox_width_height_nm, self.confidence

    def convert_pixel_to_nm(self, pixel, size_px, size_nm):
        data, img_size_px, img_size_nm, offset  = self.get_image_data_from_sxm(img_file)
        assert img_size_nm['x'] == img_size_nm['y']
        assert img_size_px['x'] == img_size_px['y']

        # Set y-axis negative to match the global coordinate system
        pixel = np.array([pixel.T[0], -pixel.T[1]]).T

        return pixel*img_size_nm['x']/img_size_px['x']

    def get_moiety_indices(self, moiety_type, moiety_position_nm, position_threshold_nm=0.6):
        """
        Determine the index of the moiety based on the type and bounding box.

        Parameters:
        -----------
        moiety_type: str
            The type of the moiety.
        bbox: np.array
            The bounding box of the moiety.

        Returns:
        --------
        moiety_index: int
            The index of the moiety.
        """
        # Determine the index of the moiety
        moiety_index_type = np.where((self.moiety_types == moiety_type))

        # Determine the index of the moiety based on the position in nm within a tolerance of 0.5 nm.
        # If multiple moieties are close to each other, the closest one is chosen.
        # closest_index = np.where(np.linalg.norm(self.moiety_position_nm - moiety_position_nm, axis=1) < position_threshold_nm)
        min_distance = float('inf')
        for idx, pos in enumerate(self.moiety_position_nm):
            distance = np.linalg.norm(np.array(moiety_position_nm) - np.array(pos))

            if distance < position_threshold_nm and distance < min_distance:
                min_distance = distance
                closest_moiety = pos
                closest_index = idx
                print("Minimum distance in index search: ", min_distance, closest_index)

        moiety_index = np.intersect1d(moiety_index_type, closest_index)

        return moiety_index

    def save_reference_image(self, img_file, moiety_type):
        """
        Save the current STM image as a new reference image if no reference image exists for the current moiety.
        If a reference image already exists, it is not overwritten.
        """
        # Get image
        img, img_size = self.get_stm_image(img_file)

        reference_filename = f"{moiety_type}_{img_size}"
        reference_image_path = os.path.join(self.reference_image_dir, reference_filename + ".jpeg")

        # Check if the reference image already exists
        if not os.path.exists(reference_image_path):
            cv2.imwrite(reference_image_path, img)
            # Copy img_file to the reference image directory with reference_filename
            shutil.copy(img_file+'.sxm', os.path.join(self.reference_image_dir, reference_filename+'.sxm'))

            print(f"Reference imagse saved in {self.reference_image_dir}")

    def save_reference_contour(self, ref_contour_nm, moiety_type, plot=False):
        """ Saves the contour of the moiety as a reference contour file.
        """
        # Save file if it does not exist
        file = os.path.join(self.reference_image_dir, f"{moiety_type}_contour_shape.npy")
        if not os.path.exists(file):
            np.save(file, ref_contour_nm)

            # Plot the contour if requested
            if plot:
                moiety_matching_reference_contour_nm = np.load(file)
                fig = plt.figure(figsize=(10, 10))
                ax = fig.add_subplot(111)
                ax.set_title(f"Reference contour for {moiety_type}")
                ax.set_xlabel("x (nm)")
                ax.set_ylabel("y (nm)")
                ax.scatter(moiety_matching_reference_contour_nm.T[0], moiety_matching_reference_contour_nm.T[1], c='red', label='Matching Reference Contour')
                ax.legend()
                plt.show()
                print("Reference contour saved and plotted.")

    def update_moiety_information(self, img_file, start_tip_position=None, end_tip_position=None, active_moiety_index=None, plot=True):
        """
        Determine the exact position of the predicted moiety in the image based on the respective reference image.
        Only for a moiety with prefix 'molecule_' or 'atom_' the exact position is determined. Other moieties
        are classified by the center of the bounding box.

        Parameters:
        -----------
        image: np.array
            The measured image containing the moieties.
        pred: list
            The prediciton of the object recognition:
            0: type of moiety
            1: x_center
            2: y_center
            3: width
            4: height
            5: confidence

        Returns:
        --------
        moiety_positions: np.array
            List containing the positions of the moieties in nm
        moiety_orientations: float
            List containing the orientations of the moieties in degrees
        moiety_type: str
            List containing the type of the moieties
        moiety_contour: np.array
            List containing the contours of the moiety
        """
        assert active_moiety_index != None
        # Save previous moiety type
        self.moiety_types_before = copy.deepcopy(self.moiety_types)
        self.moiety_position_nm_before = copy.deepcopy(self.moiety_position_nm)
        self.moiety_orientation_rad_before = copy.deepcopy(self.moiety_orientation_rad)
        self.moiety_bbox_width_height_nm_before = copy.deepcopy(self.moiety_bbox_width_height_nm)
        self.confidence_before = copy.deepcopy(self.confidence)
        self.moiety_target_contour_nm_before = copy.deepcopy(self.moiety_target_contour_nm)
        self.moiety_matching_reference_contour_nm_before = copy.deepcopy(self.moiety_matching_reference_contour_nm)

        # Object detection prediction
        img, img_size = self.get_stm_image(img_file)

        # Detect horizontal lines in the image
        horizontal_lines, contains_horizontal_lines = self.detect_horizontal_lines(img)

        # If horizontal lines are detected, set confidence threshold to 1 to prompt the user to check the results
        # If no horizontal lines are detected, use the default confidence threshold
        info = dict()
        if contains_horizontal_lines:
            info['perfect_scan'] = False
            confidence_threshold = 1
            return (), info
        else:
            info['perfect_scan'] = True
            confidence_threshold = self._CONFIDENCE_THRESHOLD

        pred = self.object_recognition.predict(img, confidence_threshold=confidence_threshold)
        self.object_recognition.plot_predictions(img, pred)


        data, img_size_px, img_size_nm, origin_img_nm  = self.get_image_data_from_sxm(img_file)
        real_img_size_px = img_size_px['x']
        real_img_size_nm = img_size_nm['x']
        assert img_size_px['x'] == img_size_px['y']
        assert img_size_nm['x'] == img_size_nm['y']

        print("File: ", img_file)
        print("Origin position of measured image: ", origin_img_nm)

        # Assign the moiety of the overall image to the moiety of the current image
        scanned_moiety_types = np.array([self.moiety_classes[int(i)] for i in pred.T[0]])
        moiety_positions = np.array([pred.T[1], pred.T[2]]).T
        moiety_bbox = np.array(pred.T[1:5]).T
        confidence = pred.T[5]
        number_of_scanned_moieties = len(scanned_moiety_types)

        # Determine position of manipulated moieties
        scanned_moiety_position_nm = np.empty((0, 2))  # Shape (N, 2)
        scanned_moiety_orientation_rad = np.empty((0,))  # 1D array of floats
        scanned_moiety_bbox_width_height_nm = np.empty((0, 2))  # Shape (N, 2)
        scanned_moiety_contour_nm = np.empty(len(scanned_moiety_types),dtype=object)
        scanned_moiety_matching_reference_contour_nm = np.empty(len(scanned_moiety_types),dtype=object)

        for i, moiety_type in enumerate(scanned_moiety_types):
            # Determine exact position for moieties with prefix 'molecule_' or 'atom_'
            if 'molecule_' in moiety_type or 'atom_' in moiety_type:
                # Save reference image if it does not exist
                self.save_reference_image(img_file=img_file, moiety_type=moiety_type)

                # Load the reference image
                ref_img_file = self.load_reference_image(moiety_type=moiety_type, size_px=real_img_size_px)

                # Determine moiety information from bounding boxes
                start_time = time.time()
                (
                    moiety_position_xy,
                    moiety_orientation_rad,
                    moiety_target_contour_px,
                    moiety_matching_reference_contour_px,
                    raw_angle_deg
                ) = self.get_moiety_information_cartesian(ref_img_file, img_file, moiety_type=moiety_type, bbox=moiety_bbox[i], plot=False)
                print("Time for determining moiety information: ", time.time()-start_time)

                # Determine index of the moiety that gets updated in the global moiety information
                moiety_position_nm = self.convert_pixel_to_nm(moiety_position_xy, img_file=img_file)
                moiety_position_nm = moiety_position_nm + np.array([-img_size_nm['x']/2,img_size_nm['y']/2]) + np.array(origin_img_nm)

                moiety_bbox_width_height_nm = self.convert_pixel_to_nm(moiety_bbox[i][2:5]*real_img_size_px, img_file=img_file)

                # Save the position and orientation of the scanned moiety
                # Append using np.vstack or np.append
                scanned_moiety_position_nm = np.vstack((scanned_moiety_position_nm, moiety_position_nm[np.newaxis, :]))
                scanned_moiety_orientation_rad = np.append(scanned_moiety_orientation_rad, moiety_orientation_rad)

                scanned_moiety_contour_nm[i] = self.convert_pixel_to_nm(moiety_target_contour_px, img_file=img_file) + np.array([-img_size_nm['x']/2,img_size_nm['y']/2]) + np.array(origin_img_nm)
                scanned_moiety_matching_reference_contour_nm[i] = self.convert_pixel_to_nm(moiety_matching_reference_contour_px, img_file=img_file) + np.array([-img_size_nm['x']/2,img_size_nm['y']/2]) + np.array(origin_img_nm)
                scanned_moiety_bbox_width_height_nm = np.vstack((scanned_moiety_bbox_width_height_nm, moiety_bbox_width_height_nm[np.newaxis, :]))

                # Save reference contour in not existing
                self.save_reference_contour(ref_contour_nm=scanned_moiety_matching_reference_contour_nm[i], moiety_type=moiety_type)

        scanned_moiety_position_nm = np.asarray(scanned_moiety_position_nm)
        scanned_moiety_bbox_width_height_nm = np.asarray(scanned_moiety_bbox_width_height_nm)

        # Determine the global moieties that are within the scan frame
        def check_if_point_is_within_scan_frame(points, scaling_factor=1):
            # Get the scan frame in nm
            scan_frame = np.array([
                origin_img_nm[0]-real_img_size_nm/2*scaling_factor, origin_img_nm[1]+real_img_size_nm/2*scaling_factor,
                origin_img_nm[0]+real_img_size_nm/2*scaling_factor, origin_img_nm[1]-real_img_size_nm/2*scaling_factor
            ])

            is_inside_scan_frame, intersects_scan_frame = self.is_point_in_bbox(points, bbox=scan_frame)
            return points[is_inside_scan_frame], points[intersects_scan_frame], is_inside_scan_frame, intersects_scan_frame, scan_frame

        def check_if_contour_are_within_scan_frame(points, scaling_factor=1):
            # Get the scan frame in nm
            scan_frame = np.array([
                origin_img_nm[0]-real_img_size_nm/2*scaling_factor, origin_img_nm[1]+real_img_size_nm/2*scaling_factor,
                origin_img_nm[0]+real_img_size_nm/2*scaling_factor, origin_img_nm[1]-real_img_size_nm/2*scaling_factor
            ])

            is_inside_scan_frame, intersects_scan_frame = self.is_contour_in_bbox(points, bbox=scan_frame)
            return points[is_inside_scan_frame], points[intersects_scan_frame], is_inside_scan_frame, intersects_scan_frame, scan_frame

        # Check global positions in scan frame
        global_moieties_position_is_within_scan_frame = []
        are_global_moiety_inside_scan_frame=[]
        for global_moiety in self.moiety_position_nm:
            global_moiety_position_is_within_scan_frame, global_moiety_position_intersects_scan_frame, is_global_moiety_inside_scan_frame, is_global_moiety_intersecting_scan_frame, scan_frame = check_if_point_is_within_scan_frame(global_moiety, scaling_factor=2)
            global_moieties_position_is_within_scan_frame.append(global_moiety_position_is_within_scan_frame)
            are_global_moiety_inside_scan_frame.append(is_global_moiety_inside_scan_frame)
        number_of_global_moieties_within_scan_frame = np.sum(are_global_moiety_inside_scan_frame)


        # row ... scanned
        # col ... global
        dist_matrix = scipy.spatial.distance.cdist(scanned_moiety_position_nm, self.moiety_position_nm)

        unclear_assignment = list()
        distance_for_matching_threshold = 1.1
        for scanned_moiety_i, distances_for_scanned_positions in enumerate(dist_matrix):
            min_distance = min(distances_for_scanned_positions)

            if min_distance > distance_for_matching_threshold:
                unclear_assignment.append(scanned_moiety_i)

        if len(scanned_moiety_types) == 1:
            self.moiety_position_nm[active_moiety_index] = scanned_moiety_position_nm[0]
            self.moiety_types[active_moiety_index] = scanned_moiety_types[0]
            self.moiety_orientation_rad[active_moiety_index] = scanned_moiety_orientation_rad[0]
            self.confidence[active_moiety_index] = confidence[0]
            self.moiety_target_contour_nm[active_moiety_index] = scanned_moiety_contour_nm[0]
            self.moiety_matching_reference_contour_nm[active_moiety_index] = scanned_moiety_matching_reference_contour_nm[0]
            self.moiety_bbox_width_height_nm[active_moiety_index] = scanned_moiety_bbox_width_height_nm[0]
        else:
            # One moiety got manipulated
            if len(unclear_assignment) == 1:
                self.moiety_position_nm[active_moiety_index] = scanned_moiety_position_nm[unclear_assignment[0]]
                self.moiety_types[active_moiety_index] = scanned_moiety_types[unclear_assignment[0]]
                self.moiety_orientation_rad[active_moiety_index] = scanned_moiety_orientation_rad[unclear_assignment[0]]
                self.confidence[active_moiety_index] = confidence[unclear_assignment[0]]
                self.moiety_target_contour_nm[active_moiety_index] = scanned_moiety_contour_nm[unclear_assignment[0]]
                self.moiety_matching_reference_contour_nm[active_moiety_index] = scanned_moiety_matching_reference_contour_nm[unclear_assignment[0]]
                self.moiety_bbox_width_height_nm[active_moiety_index] = scanned_moiety_bbox_width_height_nm[unclear_assignment[0]]
            elif len(unclear_assignment) > 1:
                # More than one moiety cannot be assigned to global positoins -> more than one got manipulated and we scanned them
                global_moiety_position_without_active_moiety = copy.deepcopy(self.moiety_position_nm)
                global_moiety_position_without_active_moiety[active_moiety_index] = np.array([ np.inf, np.inf])
                dist_matrix = scipy.spatial.distance.cdist(scanned_moiety_position_nm, global_moiety_position_without_active_moiety)

                scan_idx, global_indices = scipy.optimize.linear_sum_assignment(dist_matrix)







        # already_updated = set()
        # scaling_factor = 1
        # are_scanned_reference_inside_scan_frame = []
        # are_scanned_reference_intersecting_scan_frame = []
        # for i, single_moiety in enumerate(scanned_moiety_position_nm):
        #     # Check if the scanned moiety is within the scan frame
        #     scanned_reference_contour_is_within_scan_frame, scanned_reference_contour_intersects_scan_frame, is_scanned_reference_inside_scan_frame, is_scanned_reference_intersecting_scan_frame, scan_frame = check_if_contour_are_within_scan_frame(scanned_moiety_matching_reference_contour_nm[i])
        #     are_scanned_reference_inside_scan_frame.append(is_scanned_reference_inside_scan_frame)
        #     are_scanned_reference_intersecting_scan_frame.append(is_scanned_reference_intersecting_scan_frame)

        #     scanned_moiety_position_is_within_scan_frame, scanned_moiety_position_intersect_scan_frame, is_position_within_scan_frame, is_position_intersecting_scan_frame, scan_frame = check_if_point_is_within_scan_frame(single_moiety)

        #     single_moiety = np.atleast_2d(single_moiety)
        #     # Set partly scanned moieties to global moieties
        #     if is_scanned_reference_intersecting_scan_frame and not is_scanned_reference_inside_scan_frame:
        #         # Check which global moiety is intersecting with the scan frame

        #         # Compute distance matrix
        #         dist_matrix = scipy.spatial.distance.cdist(single_moiety, self.moiety_position_nm)  # Shape (M, N)

        #         # Find best matches using Hungarian algorithm
        #         scan_idx, global_indices = scipy.optimize.linear_sum_assignment(dist_matrix)
        #         for global_index in global_indices:
        #             if global_index in already_updated:
        #                 continue
        #             else:
        #                 if global_indices[global_index] == self._current_moiety:
        #                     print(f"Current moiety matches global incdex")

        #                 # Update the global moiety information
        #                 # global_idx_before = global_idx
        #                 # assert global_idx_before != global_idx

        #                 self.moiety_types[global_indices] = scanned_moiety_types[i]
        #                 self.moiety_position_nm[global_indices] = scanned_moiety_position_is_within_scan_frame
        #                 self.moiety_orientation_rad[global_indices] = scanned_moiety_orientation_rad[i]
        #                 self.confidence[global_indices] = confidence[i]
        #                 self.moiety_target_contour_nm[global_indices] = [scanned_moiety_contour_nm[i]]
        #                 self.moiety_matching_reference_contour_nm[global_indices] = [scanned_moiety_matching_reference_contour_nm[i]]
        #                 self.moiety_bbox_width_height_nm[global_indices] = scanned_moiety_bbox_width_height_nm[i]

        #             already_updated.add(global_index)




        # # XXX Creep may be necessary sometimes
        # # Correct piezo creep in 1st scan before manipulation
        # # if np.isnan(self.moiety_orientation_rad_before).all():
        # #     # Check if the number of global moieties within the scan frame is equal to the number of scanned moieties
        # #     if number_of_global_moieties_within_scan_frame == number_of_scanned_moieties:
        # #         # Compute the shift of the global moieties to the scanned image
        # #         x_shift_nm, y_shift_nm = self.optimize_global_without_rotation(scanned_moiety_position_nm, self.moiety_position_nm[is_within_scan_frame])
        # #         # Update the position of the global moieties with the optimized chamfer distance of the scanned image and which are not scanned yet (check if orientation is NaN)

        # #         # self.moiety_position_nm[global_moieties_orientation_is_nan] += np.array([x_shift_nm, y_shift_nm])

        # #         # Determine global moieties position within the scan frame
        # #         global_moiety_positions_within_scan_frame, is_within_scan_frame, scan_frame, scan_frame = check_if_points_are_within_scan_frame(self.moiety_position_nm, scaling_factor=scaling_factor)
        # #         global_moieties_orientation_is_nan = np.isnan(self.moiety_orientation_rad)
        # #         first_measured_global_moieties_within_scan_frame = is_within_scan_frame & global_moieties_orientation_is_nan
        # #         number_of_global_moieties_within_scan_frame = np.sum(is_within_scan_frame)

        # if plot:
        #     plt.figure(figsize=(10, 10))
        #     # All moieties
        #     plt.scatter(self.moiety_position_nm[:, 0], self.moiety_position_nm[:, 1], c='green', label='Moieties')
        #     plt.scatter(self.moiety_position_nm[:, 0][is_global_moiety_inside_scan_frame], self.moiety_position_nm[:, 0][is_global_moiety_inside_scan_frame], c='blue', label='Moieties in end frame')
        #     for i in range(len(scanned_moiety_orientation_rad)):
        #         plt.scatter(scanned_moiety_position_nm[i][0], scanned_moiety_position_nm[i][1], c='orange', label='Scanned moieties')
        #         plt.scatter(scanned_moiety_contour_nm[i].T[0], scanned_moiety_contour_nm[i].T[1], c='red', label='Scanned moieties', marker='.', s=1)
        #         plt.scatter(scanned_moiety_matching_reference_contour_nm[i].T[0], scanned_moiety_matching_reference_contour_nm[i].T[1], c='k', label='ref Scanned moieties', marker='.', s=1)
        #     # Draw bbox
        #     plt.axis('equal')
        #     plt.xlim(scan_frame[0]-10, scan_frame[0]+10)
        #     plt.ylim(scan_frame[1]-10, scan_frame[1]+10)

        #     plt.legend()
        #     plt.gca().add_patch(plt.Rectangle((scan_frame[0], scan_frame[1]), scan_frame[2]-scan_frame[0], scan_frame[3]-scan_frame[1], fill=False, edgecolor='red', linewidth=2, label='End frame'))
        #     plt.savefig(f"{img_file}_moieties_in_scan_frame_.png")
        #     plt.close()
        #     gc.collect()


        # # --- CASE 1: One extra scanned moiety (manipulation) ---
        # if number_of_global_moieties_within_scan_frame != 0 and number_of_scanned_moieties >= number_of_global_moieties_within_scan_frame + 1:
        #     dist_matrix = scipy.spatial.distance.cdist(
        #         self.moiety_position_nm[is_global_moiety_inside_scan_frame],
        #         scanned_moiety_position_nm
        #     )   # Shape (M, N)

        #     # For each scanned moiety, find the closest global moiety
        #     matched_indices = np.argmin(dist_matrix, axis=0)
        #     min_distances = np.min(dist_matrix, axis=0)

        #     # Which scanned moiety is furthest from its global one?
        #     furthest_scanned_index = np.argmax(min_distances)
        #     furthest_distance = min_distances[furthest_scanned_index]

        #     # Keep track of that manipulated moiety
        #     manipulated_pos = scanned_moiety_position_nm[furthest_scanned_index]
        #     manipulated_global = self.moiety_position_nm[is_global_moiety_inside_scan_frame][matched_indices[furthest_scanned_index]]

        #     print(f"Manipulated moiety → scan#{furthest_scanned_index} @ {manipulated_pos}, "
        #           f"nearest global @ {manipulated_global}, dist={furthest_distance:.2f} nm")

        #     # --- ASSIGN THE REMAINING SCANNED MOIETIES ---
        #     # Map indices in scan frame to global indices
        #     global_indices_within = np.where(is_global_moiety_inside_scan_frame)[0]

        #     for scan_idx, local_g_idx in enumerate(matched_indices):
        #         abs_idx = global_indices_within[local_g_idx]
        #         new_pos = scanned_moiety_position_nm[scan_idx]

        #         if scan_idx == furthest_scanned_index:
        #             self._set_current_moiety(abs_idx) # XXX Check if this is correct
        #             continue  # skip the manipulated one

        #         # Update the global moiety information
        #         self.moiety_types[abs_idx] = scanned_moiety_types[scan_idx]
        #         self.moiety_position_nm[abs_idx] = new_pos
        #         self.moiety_orientation_rad[abs_idx] = scanned_moiety_orientation_rad[scan_idx]
        #         self.confidence[abs_idx] = confidence[scan_idx]
        #         self.moiety_target_contour_nm[abs_idx] = scanned_moiety_contour_nm[scan_idx]
        #         self.moiety_matching_reference_contour_nm[abs_idx] = scanned_moiety_matching_reference_contour_nm[scan_idx]
        #         self.moiety_bbox_width_height_nm[abs_idx] = scanned_moiety_bbox_width_height_nm[scan_idx]

        #     self.manipulated_moiety_index = furthest_scanned_index
        #     self.moiety_position_nm[self._current_moiety] = scanned_moiety_position_nm[furthest_scanned_index]

        #     # Plot for verification
        #     if plot:
        #         plt.figure(figsize=(8,8))
        #         plt.scatter(self.moiety_position_nm[:,0], self.moiety_position_nm[:,1],
        #                     c='green', s=200, label='Global (updated)')
        #         plt.scatter(self.moiety_position_nm_before[self._current_moiety][0], self.moiety_position_nm_before[self._current_moiety][1],
        #                     c='red', s=100, marker='x', label='Moiety before manip.')
        #         plt.scatter(scanned_moiety_position_nm[:,0], scanned_moiety_position_nm[:,1],
        #                     c='blue', label='Scanned')
        #         # # Highlight manipulated
        #         # plt.scatter(*manipulated_pos, c='red', s=100, marker='x',
        #         #             label='Manipulated (largest distance to global moieties)')
        #         # plt.scatter(*manipulated_global, c='purple', s=80, marker='D',
        #         #             label='Matched global')
        #         plt.axis('equal')
        #         plt.legend()
        #         plt.title("Case 1: Assign remaining scanned moieties")
        #         plt.savefig(f"{img_file}_moieties_case1.png")
        #         plt.close()
        #         gc.collect()


        # # --- CASE 2: Perfect match (no extra moiety) ---
        # elif number_of_scanned_moieties == number_of_global_moieties_within_scan_frame:
        #     print("Perfect match: All scanned moieties correspond to global moieties.")

        #     # Set current moiety
        #     self._set_current_moiety(active_moiety_index)

        #     # Compute distance matrix
        #     dist_matrix = scipy.spatial.distance.cdist(self.moiety_position_nm, scanned_moiety_position_nm)  # Shape (M, N)

        #     # Find best matches using Hungarian algorithm
        #     row_ind, col_ind = scipy.optimize.linear_sum_assignment(dist_matrix)

        #     print("Assignments between global and scanned moieties:")
        #     for global_indices, scanned_idx in zip(row_ind, col_ind):
        #         g_pos = self.moiety_position_nm[global_indices]
        #         s_pos = scanned_moiety_position_nm[scanned_idx]
        #         dist = dist_matrix[global_indices, scanned_idx]
        #         print(f"Global moiety {global_indices} at {g_pos} matched to scanned moiety {scanned_idx} at {s_pos}, distance = {dist:.2f} nm")

        #         # Update the global moiety position to the scanned position
        #         #global_absolute_idx = np.where(are_global_moiety_inside_scan_frame)[global_idx][0]
        #         self.moiety_types[global_indices] = scanned_moiety_types[scanned_idx]
        #         self.moiety_position_nm[global_indices] = scanned_moiety_position_nm[scanned_idx]
        #         self.moiety_orientation_rad[global_indices] = scanned_moiety_orientation_rad[scanned_idx]
        #         self.confidence[global_indices] = confidence[scanned_idx]
        #         self.moiety_target_contour_nm[global_indices] = np.squeeze(scanned_moiety_contour_nm[scanned_idx])
        #         self.moiety_matching_reference_contour_nm[global_indices] = scanned_moiety_matching_reference_contour_nm[scanned_idx]
        #         self.moiety_bbox_width_height_nm[global_indices] = scanned_moiety_bbox_width_height_nm[scanned_idx]

        #     if plot:
        #         plt.figure(figsize=(10, 10))
        #         plt.scatter(self.moiety_position_nm[:, 0], self.moiety_position_nm[:, 1],
        #                     c='green', s=100, label='All Global Moieties (Updated)')
        #         plt.scatter(self.moiety_position_nm[self._current_moiety][0], self.moiety_position_nm[self._current_moiety][1],
        #                     c='k', s=100, marker='x', label='Manipulated Moiety (After)')
        #         plt.scatter(self.moiety_position_nm_before[self._current_moiety][0], self.moiety_position_nm_before[self._current_moiety][1],
        #                     c='lightgrey', s=100, marker='x', label='Manipulated Moiety (Before)')
        #         plt.scatter(scanned_moiety_position_nm[:, 0], scanned_moiety_position_nm[:, 1],
        #                      c='blue', label='Scanned Moieties', alpha=0.7)
        #         for g_idx, s_idx in zip(row_ind, col_ind):
        #             g_pos = self.moiety_position_nm[are_global_moiety_inside_scan_frame[g_idx]][0][0]
        #             s_pos = scanned_moiety_position_nm[s_idx]
        #             plt.plot([g_pos[0], s_pos[0]], [g_pos[1], s_pos[1]], 'k--', alpha=0.5)
        #         plt.axis('equal')
        #         plt.legend()
        #         plt.title('Perfect Matching of Global and Scanned Moieties')
        #         plt.savefig(f"{img_file}_moieties_case2.png")
        #         plt.close()
        #         gc.collect()

        #     self.matching_indices = (row_ind, col_ind)

        # # --- CASE 3: Single moiety manipulated ---   # if moiety gets defect error
        # elif number_of_global_moieties_within_scan_frame == 0 and number_of_scanned_moieties == 1:
        #     print("Single moiety manipulated: No global moieties in scan frame, single scanned moiety.")
        #     # Update the global moiety position to the scanned position

        #     self.moiety_types[self._current_moiety] = scanned_moiety_types[0]
        #     self.moiety_position_nm[self._current_moiety] = scanned_moiety_position_nm[0]
        #     self.moiety_orientation_rad[self._current_moiety] = scanned_moiety_orientation_rad[0]
        #     self.confidence[self._current_moiety] = confidence[0]
        #     self.moiety_target_contour_nm[self._current_moiety] = scanned_moiety_contour_nm[0]
        #     self.moiety_matching_reference_contour_nm[self._current_moiety] = scanned_moiety_matching_reference_contour_nm[0]
        #     self.moiety_bbox_width_height_nm[self._current_moiety] = scanned_moiety_bbox_width_height_nm[0]



        #     # Plot for verification
        #     if plot:
        #         plt.figure(figsize=(8,8))
        #         plt.scatter(self.moiety_position_nm[:,0], self.moiety_position_nm[:,1],
        #                     c='green', s=200, label='Global (updated)')
        #         plt.scatter(self.moiety_position_nm_before[self._current_moiety][0], self.moiety_position_nm_before[self._current_moiety][1],
        #                     c='red', s=100, marker='x', label='Moiety before manip.')
        #         plt.scatter(scanned_moiety_position_nm[:,0], scanned_moiety_position_nm[:,1],
        #                     c='blue', label='Scanned')
        #         plt.axis('equal')
        #         plt.legend()
        #         plt.title("Case 1: Assign remaining scanned moieties")
        #         plt.savefig(f"{img_file}_moieties_case1.png")
        #         plt.close()
        #         gc.collect()

        # elif number_of_scanned_moieties == 0:
        #     self.moiety_types[self._current_moiety] = 'defect'

        #                 # Plot for verification
        #     if plot:
        #         plt.figure(figsize=(8,8))
        #         plt.scatter(self.moiety_position_nm[:,0], self.moiety_position_nm[:,1],
        #                     c='green', s=200, label='Global (updated)')
        #         plt.scatter(self.moiety_position_nm_before[self._current_moiety][0], self.moiety_position_nm_before[self._current_moiety][1],
        #                     c='red', s=100, marker='x', label='Moiety before manip.')
        #         plt.axis('equal')
        #         plt.legend()
        #         plt.title("Case 1: Assign remaining scanned moieties")
        #         plt.savefig(f"{img_file}_moieties_case1.png")
        #         plt.close()
        #         gc.collect()


        # do_final_check = True # XXX read from input.json file
        # # --- Final geometric validation ---
        # if do_final_check and start_tip_position is not None and end_tip_position is not None:
        #     # Convert start/end positions to np arrays if necessary
        #     start_tip = np.array([start_tip_position.x, start_tip_position.y])
        #     end_tip = np.array([end_tip_position.x, end_tip_position.y])

        #     # Create line from tip movement
        #     tip_line = LineString([start_tip, end_tip])

        #     # Get the contour of the manipulated moiety
        #     manipulated_contour = self.moiety_matching_reference_contour_nm[self._current_moiety]

        #     # if manipulated_contour.ndim == 3:  # shape: (1, N, 2) → reduce
        #     #     manipulated_contour = manipulated_contour

        #     moiety_polygon = Polygon(manipulated_contour)

        #     # Check if the tip line intersects the moiety contour
        #     intersects = tip_line.intersects(moiety_polygon)
        #     # info = dict()
        #     # info['intersects'] = intersects
        #     if not intersects:
        #         self.moiety_position_nm[self._current_moiety] = self.moiety_position_nm_before[self._current_moiety]
        #     #     print("Final check failed: Tip trajectory does NOT intersect the manipulated moiety contour.")
        #     #     print(f"Tip Start: {start_tip}, Tip End: {end_tip}")
        #     #     print(f"Manipulated Moiety Index: {self._current_moiety}")
        #     #     input("Press enter to continue...")
        #     # else:
        #     #     print("Final check passed: Tip movement intersects manipulated moiety.")

        # self.moiety_type_changes = self.moiety_types_before != self.moiety_types

        logger.bind(task='stats', raw_angle=int(np.rad2deg(raw_angle_deg))).trace("")

        return ((self.moiety_types,
                self.moiety_position_nm,
                self.moiety_orientation_rad,
                self.moiety_bbox_width_height_nm,
                self.confidence,
                self.moiety_target_contour_nm,
                self.moiety_matching_reference_contour_nm,
                self.moiety_types_before,
                self.moiety_type_changes),
                info
        )

    # Helper methods:
    def load_reference_image(self, moiety_type, size_px):
        for ext in ('sxm','jpg','dat.jpg','jpeg','dat.jpeg'):
            candidate = os.path.join(
                self.reference_image_dir,
                f"{moiety_type}_{size_px}.{ext}"
            )
            if os.path.exists(candidate):
                return candidate
        raise FileNotFoundError(f"Ref image for {moiety_type} not found")

    def _assign_scanned(self, global_idx, scan_idx, scanned, scanned_positions):
        """Update global arrays for a matched scanned moiety."""
        self.moiety_position_nm[global_idx] = scanned_positions[scan_idx]
        self.moiety_orientation_rad[global_idx] = scanned[scan_idx]['ori']
        self.moiety_target_contour_nm[global_idx] = scanned[scan_idx]['contour']
        self.moiety_matching_reference_contour_nm[global_idx] = scanned[scan_idx]['ref_contour']
        # track type changes if necessary
        new_type = scanned[scan_idx]['type']
        if new_type != self.moiety_types[global_idx]:
            self.moiety_type_changes[global_idx] = True
            self.moiety_types[global_idx] = new_type

    def _check_tip_proximity(self, original_positions, start, end):
        """Ensure the manipulated moiety matches tip positions before/after."""
        manip = self._current_moiety
        d_start = np.linalg.norm(original_positions[manip] - start)
        d_end   = np.linalg.norm(self.moiety_position_nm[manip] - end)
        print(f"Tip distances: start={d_start:.2f}, end={d_end:.2f} nm")
        # optional warning if mismatch

    def _plot_alignment(self, scanned_positions, frame, skip_index=None):
        plt.figure(figsize=(8,8))
        plt.scatter(self.moiety_position_nm[:,0], self.moiety_position_nm[:,1], c='green', label='Global')
        plt.scatter(scanned_positions[:,0], scanned_positions[:,1], c='blue', label='Scanned')
        if skip_index is not None:
            plt.scatter(*scanned_positions[skip_index], c='red', s=100, marker='x', label='Manipulated')
        plt.gca().add_patch(
            plt.Rectangle((frame[0], frame[1]), frame[2]-frame[0], frame[3]-frame[1],
                          fill=False, edgecolor='red', linewidth=2)
        )
        plt.axis('equal')
        plt.legend()
        plt.show()


    # return moiety_types, moiety_position_nm, moiety_orientation_rad, moiety_bbox_width_height_nm, self.moiety_target_contour_nm, self.moiety_matching_reference_contour_nm
    @staticmethod
    def is_point_in_bbox(geometry, bbox):
        """
        Check if a point, list of points, or a contour intersects a bounding box.

        Args:
            geometry (ndarray): Either:
                - A single point: shape (2,)
                - Multiple points: shape (N, 2)
                - A contour (polygon): shape (>=3, 2)
            bbox (tuple): (xmin, ymin, xmax, ymax)

        Returns:
            bool or ndarray:
                - If input is a single point or polygon: returns True/False.
                - If input is multiple points: returns a boolean array.
        """
        bbox_poly = box(*bbox)

        geometry = np.asarray(geometry)
        #poly = Polygon(geometry)

        #if geometry.ndim == 1 and geometry.shape[0] == 2:
            # Single point
        return bbox_poly.contains(Point(geometry)), bbox_poly.intersects(Point(geometry))

        # elif geometry.ndim == 2:
        #     if len(geometry) < 25:
        #         # Multiple points
        #         return np.array([bbox_poly.contains(Point(p)) for p in geometry])
        #     else:
        #         # Treat as polygon
        #         poly = Polygon(geometry)
        #         # return if poly is completely within the bbox or intersects the bbox
        #         return bbox_poly.contains(poly), bbox_poly.intersects(poly)
        #    return bbox_poly.contains(Polygon(geometry)), bbox_poly.intersects(Polygon(geometry))
        #raise ValueError("Input geometry must be a point, list of points, or a contour (polygon).")
        # """
        # Check if multiple points are inside a bounding box.

        # Args:
        #     points (ndarray): A 2D array of points, where each row is a point (x, y).
        #     bbox (tuple): A tuple (xmin, ymin, xmax, ymax) representing the bounding box.

        # Returns:
        #     ndarray: A boolean array indicating if each point is inside the bounding box.
        # """
        # xmin, ymin, xmax, ymax = bbox

        # # Element-wise check if each point is within the bounding box
        # return np.all((points >= [xmin, ymin]) & (points <= [xmax, ymax]), axis=1)

    @staticmethod
    def is_contour_in_bbox(geometry, bbox):
        bbox_poly = box(*bbox)
        geometry = np.asarray(geometry)
        return bbox_poly.contains(Polygon(geometry)), bbox_poly.intersects(Polygon(geometry))

    def get_observation(self):
        return self.moiety_types, self.moiety_position_nm, self.moiety_orientation_rad, self.moiety_bbox_width_height_nm, self.confidence, self.moiety_target_contour_nm, self.moiety_matching_reference_contour_nm, self.moiety_types_before, self.moiety_type_changes

    def get_moiety_information_cartesian(self, ref_img, img_file, moiety_type, bbox, plot=True):
        """
        Get the moiety's pixel-level position and orientation by matching a reference contour
        to the detected target region in a microscopy image using chamfer matching.

        Parameters:
        -----------
        ref_img : str
            Path to the reference image file.
        img_file : str
            Path to the target image file.
        moiety_type : str
            Type of moiety (e.g., 'molecule').
        bbox : list[float]
            Bounding box around predicted object (center_x, center_y, half_width, half_height) in normalized coordinates.
        plot : bool
            Whether to visualize matching results.

        Returns:
        --------
        best_matching_position : np.ndarray
            [x, y] coordinates of moiety position in pixels.
        best_matching_orientation_rad : float
            Orientation in radians.
        best_target_contour : np.ndarray
            Target contour points.
        best_matching_reference_contour : np.ndarray
            Transformed reference contour points.
        """
        # === IMAGE LOADING & PREPROCESSING ========================================
        ref_img_gray = cv2.cvtColor(self.get_stm_image(ref_img)[0], cv2.COLOR_BGR2GRAY)
        tgt_img_gray = cv2.cvtColor(self.get_stm_image(img_file)[0], cv2.COLOR_BGR2GRAY)

        _, img_size_px, _, _ = self.get_image_data_from_sxm(img_file)
        size_px = img_size_px['x']
        assert size_px == img_size_px['y'], "Non-square image dimensions not supported."

        # (
        #     moiety_position_px,
        #     moiety_orientation_rad,
        #     moiety_target_contour_px,
        #     moiety_matching_reference_contour_px
        # ) =

        return self.determine_moiety_information_cartesian(ref_img_gray, tgt_img_gray, size_px, moiety_type, bbox, plot=False)

        # # swap x and y coordinates to match the expected output format
        # best_matching_position = np.array([moiety_position_px[1], moiety_position_px[0]])
        # best_matching_orientation_rad = moiety_orientation_rad
        # best_target_contour = np.array(moiety_target_contour_px.T[1], moiety_target_contour_px.T[0]).T
        # best_matching_reference_contour = np.array(moiety_matching_reference_contour_px.T[1], moiety_matching_reference_contour_px.T[0]).T

        # return (
        #     best_matching_position,
        #     best_matching_orientation_rad,
        #     best_target_contour,
        #     best_matching_reference_contour
        # )


    @staticmethod
    def transform_contour(cont, theta, x, y):
        """Rotate and translate a contour

        :param cont: _description_
        :param theta: _description_
        :param x: _description_
        :param y: _description_
        :return: _description_
        """
        offset = np.array([x, y])
        rot_mat = np.array(
            [[np.cos(theta), -np.sin(theta)],
            [np.sin(theta), np.cos(theta)]]
        )
        center_cont = np.sum(cont, axis=0) / len(cont)
        rel_cont = cont - center_cont
        rot_cont = np.sum(rot_mat[None, :, :] * rel_cont[:, None, :], axis=2)
        trans_cont = rot_cont + center_cont + offset[None, :]
        return trans_cont

    @staticmethod
    def chamfer_dist(contour_, reference_):
        """
        Very inefficient implementation of the modified chamfer distance, where we only measure in contour to reference and not vice versa.
        This would introduce an offset.
        """
        dist_mat = np.linalg.norm(reference_[:, None] - contour_[None, :], axis=-1)
        dist1 = np.min(dist_mat, axis=0)
        return np.mean(dist1)

    def chamfer_dist_transform_no_theta(self, z, cont, ref):
        """Calculate the Chamfer distance based on the given transformation z = (x, y)

        :param z: ndarray(2), array with the two inputs (x, y)
        :param cont: ndarray(n, 2), contour which should be matched to ref
        :param ref: ndarray(m, 2), reference contour
        :return: float, chamfer distance between contour and reference
        """
        x, y = z
        theta = 0
        trans_ref = self.transform_contour(ref, theta, x, y)
        return self.chamfer_dist(cont, trans_ref)


    def optimize_global_without_rotation(self, cont, ref):
        """Optimize the global position of the contour to match the reference contour
        """
        ref_pos = np.sum(ref, axis=0) / len(ref)
        cont_pos = np.sum(cont, axis=0) / len(cont)
        off = cont_pos - ref_pos
        ref_off = ref + off

        res = scipy.optimize.basinhopping(
                self.chamfer_dist_transform_no_theta,
                x0=np.array([0, 0]),
                minimizer_kwargs={'args': (cont, ref_off), 'method': 'Nelder-Mead'},
                T=200,
                stepsize=50,
                niter=10
            )

        x_shift = res.x[0] + off[0]
        y_shift = res.x[1] + off[1]
        return x_shift, y_shift

    def chamfer_dist_transform(self, z, cont, ref):
        """Calculate the Chamfer distance based on the given transformation z = (theta, x, y)

        :param z: ndarray(3), array with the three inputs (angle, x, y)
        :param cont: ndarray(n, 2), contour which should be matched to ref
        :param ref: ndarray(m, 2), reference contour
        :return: float, chamfer distance between contour and reference
        """
        theta, x, y = z
        trans_ref = self.transform_contour(ref, theta, x, y)
        return self.chamfer_dist(cont, trans_ref)

    def optimize_global(self, cont, ref):
        """Optimize the global position of the contour to match the reference contour
        """
        ref_pos = np.sum(ref, axis=0) / len(ref)
        cont_pos = np.sum(cont, axis=0) / len(cont)
        off = cont_pos - ref_pos
        ref_off = ref + off

        res = scipy.optimize.basinhopping(
                self.chamfer_dist_transform,
                x0=np.array([0, 0, 0]),
                minimizer_kwargs={'args': (cont, ref_off), 'method': 'Nelder-Mead'},
                T=200,
                stepsize=50,
                niter=10
            )
        # Match reference to target using a minus sign
        # res.x[1] += off[0][0]
        # res.x[2] += off[0][1]
        rot = res.x[0]
        x_shift = res.x[1] + off[0]
        y_shift = res.x[2] + off[1]
        return rot, x_shift, y_shift

    def optimize_local(self, cont, ref):
        res = scipy.optimize.minimize(
                self.chamfer_dist_transform,
                x0=np.array([0, 0, 0]),
                args=(cont, ref),
                method='BFGS'
            )
        return res.x

    @staticmethod
    def filter_lonely_points(contour, min_distance=10):
        """
        Filters out "lonely" points from a contour based on a minimum distance threshold.

        A point is considered lonely if its distance to both its immediate neighbors
        exceeds the given min_distance. This is useful for removing noise or spurious points
        that do not lie close to the continuous border of a segmented object.

        Parameters:
        -----------
        contour : np.array
            Array of contour points. Expected shape can be either (N, 1, 2) as returned by
            cv2.findContours or (N, 2).
        min_distance : float
            The threshold distance in pixels. If a point’s distance to both its
            previous and next neighbor exceeds this value, it is removed.

        Returns:
        --------
        filtered_contour : np.array
            The contour after removing lonely points. The returned contour has the shape
            (M, 1, 2) with M <= N.
        """
        import numpy as np

        # Ensure the contour is in shape (N, 2)
        if contour.ndim == 3:
            pts = contour[:, 0, :]
        else:
            pts = contour.copy()

        N = pts.shape[0]
        # If the contour is too short, just return as is.
        if N < 3:
            return contour.copy()

        keep_indices = []
        # Iterate through each point. We assume the contour is closed.
        for i in range(N):
            # Use modulo indexing for closed contour:
            prev_pt = pts[(i - 1) % N]
            curr_pt = pts[i]
            next_pt = pts[(i + 1) % N]

            # Compute Euclidean distances between the current point and its neighbors.
            dist_prev = np.linalg.norm(curr_pt - prev_pt)
            dist_next = np.linalg.norm(curr_pt - next_pt)

            # If the point is close to at least one neighbor, we keep it.
            if dist_prev <= min_distance or dist_next <= min_distance:
                keep_indices.append(i)

        if len(keep_indices) < 3:
            # If too many points are removed, return the original contour.
            filtered_pts = pts
        else:
            filtered_pts = pts[keep_indices]

        # Reshape to the format (M, 1, 2) expected for contours.
        filtered_contour = filtered_pts.reshape(-1, 1, 2).astype(np.int32)
        return filtered_contour

    @staticmethod
    def filter_border_points(contour, curvature_threshold_deg=2):
        """
        Removes points from a contour that lie on nearly straight segments.

        For each point in the (closed) contour, the function computes the angle
        between the vector from the previous point to the current point and the
        vector from the current point to the next point. If the turning angle
        (i.e., deviation from a perfectly straight line) is less than the threshold,
        that point is considered to lie on a straight segment and is removed.

        Parameters:
        -----------
        contour : np.array
            Array of contour points. It can be of shape (N, 1, 2) (as returned by OpenCV)
            or (N, 2).
        curvature_threshold : float
            Minimum turning angle (in degrees) required to keep the point. Points with
            smaller angles (i.e., nearly collinear) are discarded.

        Returns:
        --------
        filtered_contour : np.array
            The filtered contour as an array of shape (M, 1, 2), where M <= N.
        """
        # Convert to (N,2) shape if needed.
        if contour.ndim == 3:
            pts = contour[:, 0, :]
        else:
            pts = contour.copy()

        N = pts.shape[0]
        if N < 3:
            # Not enough points to filter, return original contour in correct shape.
            return contour.copy()

        # Convert threshold from degrees to radians.
        thresh_rad = np.deg2rad(curvature_threshold_deg)

        # List to hold indices of points to keep.
        keep_idx = []

        # We assume the contour is closed.
        for i in range(N):
            prev = pts[(i - 1) % N]
            curr = pts[i]
            next_pt = pts[(i + 1) % N]

            # Compute vectors.
            v1 = curr - prev
            v2 = next_pt - curr

            # Check for nearly zero-length segments.
            norm_v1 = np.linalg.norm(v1)
            norm_v2 = np.linalg.norm(v2)
            if norm_v1 < 1e-5 or norm_v2 < 1e-5:
                # Skip computation if any segment is too short.
                continue

            # Compute the angle between v1 and v2.
            # For collinear points (straight line), the angle between v1 and v2 is ~0.
            dot_prod = np.dot(v1, v2)
            cos_angle = np.clip(dot_prod / (norm_v1 * norm_v2), -1.0, 1.0)
            angle = np.arccos(cos_angle)

            # For a perfectly straight line the turning angle is nearly zero.
            # Therefore, if the computed angle is smaller than our threshold, we discard the point.
            if angle >= thresh_rad:
                keep_idx.append(i)

        # Ensure that we keep at least a minimal set of points.
        if len(keep_idx) < 3:
            # Fallback: if too few points remain, just return the original contour.
            filtered_pts = pts
        else:
            filtered_pts = pts[keep_idx]

        # Reshape to have the same structure as a contour returned by cv2.findContours.
        filtered_contour = filtered_pts.reshape(-1, 1, 2).astype(np.int32)
        return filtered_contour

    def plot_moieties(self, img_file, plot=True):
        """
        Plot the overview image and the measured target image with the determined contours.
        """

        # Resize and blur images
        data, img_size_px, img_size_nm, origin_img_nm  = self.get_image_data_from_sxm(img_file)
        real_img_size = img_size_px['x']
        assert img_size_px['x'] == img_size_px['y']

        # Plot the moiety positions and contours in the overview image
        if plot:
            overview_img = self.get_stm_image(img_file)[0]
            overview_img = cv2.resize(overview_img, (real_img_size, real_img_size))

            for i, moiety_position_nm in enumerate(self.moiety_position_nm):
                moiety_position_px = moiety_position_nm - (origin_img_nm + np.array([-img_size_nm['x']/2,img_size_nm['y']/2]))
                moiety_position_px_for_image = np.array([self.convert_nm_to_pixel(moiety_position_px, img_file=img_file)]).astype(int)
                # make y coordinates negative
                moiety_position_px_for_image[:, 1] = -moiety_position_px_for_image[:, 1]
                moiety_position_px_for_image = moiety_position_px_for_image[0]
                cv2.drawMarker(overview_img,
                               moiety_position_px_for_image,
                               (0, 255, 0), markerType=cv2.MARKER_CROSS, markerSize=20, thickness=2)

                if self.moiety_target_contour_nm[i] is not None:
                    # Convert the contour from nm to pixel coordinates
                    contour_px = self.moiety_target_contour_nm[i] - (origin_img_nm + np.array([-img_size_nm['x']/2,img_size_nm['y']/2]))
                    contour_px_for_image = self.convert_nm_to_pixel(contour_px, img_file=img_file).astype(int)
                    # make y coordinates negative
                    contour_px_for_image[:,0,1] = -contour_px_for_image[:,0,1]
                    cv2.drawContours(overview_img,
                                    contour_px_for_image,
                                    -1, (255, 0, 0), 2)

                    # Draw the orientation of the moiety
                    cv2.line(overview_img, tuple(moiety_position_px_for_image),
                        (int(moiety_position_px_for_image[0] + 50*np.cos(-self.moiety_orientation_rad[i])),
                        int(moiety_position_px_for_image[1] + 50*np.sin(-self.moiety_orientation_rad[i]))), (0, 0, 255), 2)

            cv2.imshow("Overview Image", overview_img)
            cv2.waitKey(0)
            cv2.destroyAllWindows()

    def convert_nm_to_pixel(self, nm, img_file):
        data, img_size_px, img_size_nm, origin_img_nm  = self.get_image_data_from_sxm(img_file)
        assert img_size_nm['x'] == img_size_nm['y']

        px = nm * img_size_px['x'] / img_size_nm['x']
        # Do NOT invert y axis
        return px
