
import numpy as np
import math
import glob
import os
from os.path import join
import time
from datetime import datetime
import yaml
import matplotlib.pyplot as plt
import logging
import pytest
import cv2
from shapely import Point
import gc

try:
    import nanonisTCP
    import nanonisTCP.Bias
    import nanonisTCP.Current
    import nanonisTCP.FolMe
    import nanonisTCP.SafeTip
    import nanonisTCP.Scan
    import nanonisTCP.ZController
    import nanonisTCP.Signals
    import nanonisTCP.UserOut
    import nanonisTCP.BiasSpectr
    from object_recognition.post_processing.smxReader import NanonisSXM

except ImportError:
    print('Nanonis packages are not installed. This is only required for Nanonis STMs.')

try:
    import win32com.client
    import pythoncom
except ImportError:
    print('Win32 packages are either not installed or the user is on a Linus machine. This is only required for Createc STMs.')

class STM(object):

    """ Create an instance of the STM class dependent on the hardware used. The hardware is either nanonis or createc. """
    def __init__(self):
        self.input_yaml_file = os.path.normpath(os.path.join(os.getcwd(),"input.yaml"))

        self.directory_main = self.get_yaml_data("main_dir")
        self.directory_for_stm_data = os.path.normpath(os.path.join(self.directory_main, "STM_data"))  
        self.directory_for_stm_reference_topography = os.path.normpath(os.path.join(os.path.normpath(os.path.join(self.directory_main, "STM_data")), "reference_images"))
        os.makedirs(self.directory_for_stm_data, exist_ok=True)
        os.makedirs(self.directory_for_stm_reference_topography, exist_ok=True)
        self.experiment_name = self.get_yaml_data("experiment_name")
        self.filename_per_timestep = 'not_defined'
        self.current_image_file = 'not_defined'

        


    def get_yaml_data(self, *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(os.path.normpath(os.path.join(os.getcwd(),"input.yaml")), "r") as ymlfile:
            data = yaml.load(ymlfile, Loader=yaml.FullLoader)

        for val in args:
            if 'molecule_' in val or 'atom_' in val:
                moiety_type = val
            else:
                moiety_val = val

        if len(args) == 1:
            return data[args[0]]
        else:
            return data[moiety_type][moiety_val]

class NanonisSTM(STM):
    def __init__(self):
        super().__init__()
        self.logger = logging.getLogger("STM")
        
        self.nanonisTCP = nanonisTCP.nanonisTCP(version=10300)
        self.nanonisZCtrl = nanonisTCP.ZController.ZController(self.nanonisTCP)
        self.nanonisCurrent = nanonisTCP.Current.Current(self.nanonisTCP)
        self.nanonisBias = nanonisTCP.Bias.Bias(self.nanonisTCP)
        self.nanonisFolMe = nanonisTCP.FolMe.FolMe(self.nanonisTCP)
        self.nanonisSetSpeed = nanonisTCP.Scan.Scan(self.nanonisTCP)
        self.nanonisScan = nanonisTCP.Scan.Scan(self.nanonisTCP)
        self.nanonisSafeTip = nanonisTCP.SafeTip.SafeTip(self.nanonisTCP)
        self.nanonisSignals = nanonisTCP.Signals.Signals(self.nanonisTCP)
        self.nanonisUserOut = nanonisTCP.UserOut.UserOut(self.nanonisTCP)
        self.nanonisBiasSpectr = nanonisTCP.BiasSpectr.BiasSpectr(self.nanonisTCP)

        # Reference current to set the tip at a certain height above the surface. The value is defined in the input.yaml file.
        self.reference_current_A = self.get_yaml_data('set_reference_current_A')
        self.reference_voltage_mV = self.get_yaml_data('set_reference_voltage_mV')

        (
            self.overview_image_center_x_m,
            self.overview_image_center_y_m,
            self.overview_image_width_m,
            self.overview_image_height_m,
            self.overview_image_scan_angle_deg
        ) = self.get_parameters_from_overview_image()

        #print('WARNING: Set the size of the overview image to 30 nm')
        #self.overview_image_width_m = self.overview_image_width_m = 30

        # convert values to nm
        self.overview_image_center_x_nm = round(self.overview_image_center_x_m*1e9,3)
        self.overview_image_center_y_nm = round(self.overview_image_center_y_m*1e9,3)
        self.overview_image_width_nm = round(self.overview_image_width_m*1e9,3)
        self.overview_image_height_nm = round(self.overview_image_height_m*1e9,3)

        self.overview_image_size_m = self.overview_image_width_m
        self.overview_image_size_nm = round(self.overview_image_size_m*1e9,3)

        # get error if overview image width and height are not equal
        assert self.overview_image_width_m == self.overview_image_height_m, "Overview image width and height are not equal"

        self.overview_image_path = max(glob.glob(os.path.join(os.path.abspath(self.directory_for_stm_data), '*Au(111)-FePc-5K*.sxm')), key=os.path.getmtime)

        self.init_ZController()
        #self.unit_test_STM(False)

    def get_parameters_from_overview_image(self):
        """
        Reads the parameters from the overview image and returns the values of the parameters.
        """
        return self.nanonisScan.FrameGet()
    
    def set_parameters_for_overview_image(self):
        """
        Sets the parameters for the overview image.
        """
        self.nanonisScan.FrameSet(x=self.overview_image_center_x_m, 
                                  y=self.overview_image_center_y_m, 
                                  w=self.overview_image_width_m, 
                                  h=self.overview_image_height_m, 
                                  angle=self.overview_image_scan_angle_deg)

    def unit_test_STM(self,do_unit_test=False):
        """
        Checks the interface variables and returns the values of the variables.
        """

        z_approach_nm = 1
        voltage_mV = 3185
        pos_STM_nm_x = 8
        pos_STM_nm_y = 1
        pulse_time_in_s = 1
        self.perform_lateral_manipulation(x_start_position_nm=pos_STM_nm_x, y_start_position_nm=pos_STM_nm_y, x_end_position_nm=pos_STM_nm_x+1, y_end_position_nm=pos_STM_nm_y+0, z_position_nm=z_approach_nm, voltage_mV=voltage_mV)
        
        if do_unit_test==False:
            print("No Unit-Tests done")
            return
        
        pos_STM_nm = [4,-8]
        speed_nm_s = 5
        self.set_position(pos_STM_nm=pos_STM_nm, speed_nm_s=speed_nm_s)
        assert pytest.approx( self.nanonisFolMe.XYPosGet(Wait_for_newest_data=1), 1e-15) == (pos_STM_nm[0]*1e-9,pos_STM_nm[1]*1e-9), "Position is not set correctly"
        assert pytest.approx( self.nanonisFolMe.SpeedGet()[0], 1e-15) == speed_nm_s*1e-9, "Speed is not set correctly"

        z_approach_nm = 1
        voltage_mV = 3185
        pos_STM_nm_x = 8
        pos_STM_nm_y = 1
        pulse_time_in_s = 1
        self.perform_vertical_manipulation(z_approach_nm=z_approach_nm, voltage_mV=voltage_mV, x_position_nm=pos_STM_nm_x, y_position_nm=pos_STM_nm_y, duration_pulse_s=pulse_time_in_s)
        assert pytest.approx( self.nanonisZCtrl.ZPosGet()*1e-9, 1e-15) == self.reference_z_position_nm*1e-9, f"Z-approach is not set correctly: get: {self.nanonisZCtrl.ZPosGet()+z_approach_nm*1e-9} set: {z_approach_nm*1e-9}"
        assert pytest.approx( self.nanonisFolMe.XYPosGet(Wait_for_newest_data=1), 1e-15) == (pos_STM_nm_x*1e-9,pos_STM_nm_y*1e-9), "Position is not set correctly"

        # not (math.isclose(round(self.nanonisFolMe.XYPosGet(Wait_for_newest_data=False)[0],15), x_start_position_nm*1e-9, rel_tol=1e-15) and
        #      math.isclose(round(self.nanonisFolMe.XYPosGet(Wait_for_newest_data=False)[1],15), y_start_position_nm*1e-9, rel_tol=1e-15))
        # ):

        self.set_bias_voltage(voltage_mV)
        assert pytest.approx( self.nanonisBias.Get(), 1e-3) == voltage_mV*1e-3, "Bias voltage is not set correctly"

        

        print("Unit-Tests passed")

    def connect_to_stm(self):
        """
        Initializes the connection to the STM/AFM program.
        """
        pass


    def init_ZController(self):
        """
        Measures the current of some value above the clearn surface whose coodinates are given in the input file. Initializes the z-distance of the tip
        based on a given bias voltage and current value within the tunneling junction.
        """
        clean_surface_position_nm = self.get_yaml_data('init_Zcontroller_position_x'), self.get_yaml_data('init_Zcontroller_position_y')
        self.nanonisZCtrl.TipLiftSet(0)
        self.nanonisZCtrl.OnOffSet(0)
        self.set_bias_voltage(voltage_mV=self.reference_voltage_mV)
        self.nanonisZCtrl.SetpntGet()
        self.nanonisZCtrl.SetpntSet(self.reference_current_A)
        self.nanonisZCtrl.OnOffSet(1)
        self.set_position(pos_STM_nm=clean_surface_position_nm, speed_nm_s=5)
        self.nanonisZCtrl.OnOffSet(0)
        self.reference_z_position_nm = self.nanonisZCtrl.ZPosGet()*1e9

        # print('Warning for testing purposes set reference current to fixed value')
        # self.reference_z_position_nm = 1.0486109758289786

        print(f'Z-Control setup done. The reference z position is {self.reference_z_position_nm} [nm]')

    def update_parameters(self):
        """ 
        Updates all parameters and synchronizes the parameters with the DSP (dual digital feedback 
        controller).
        """
        pass

    def beep(self):
        """
        Makes a beep sound and writes 'Beep' into the log-file.
        """
        pass

    def get_float_param(self, name):
        """ 
        Reads the parameter specified by the argument and tries to convert it to float.
        """
        pass

    def set_position(self, pos_STM_nm, speed_nm_s, wait_end_of_move=True):
        """ 
        Moves STM-tip to new position. 
        """
        self.nanonisFolMe.SpeedSet(speed=speed_nm_s*1e-9, custom_speed=True)
        #self.nanonisTipRec.BufferClear()
        self.nanonisFolMe.XYPosSet(pos_STM_nm[0]*1e-9, pos_STM_nm[1]*1e-9, Wait_end_of_move=wait_end_of_move)
        #self.nanonisTipRec.DataGet()

    def get_relative_position(self):  
        """ 
        Gets the relative position of the STM in DAC units.
        """
        return self.nanonisFolMe.XYPosGet(Wait_for_newest_data=1)

    def get_relative_position_nm(self):  
        """ 
        Gets the relative position of the STM in DAC units.
        """
        # return np.array(self.nanonisFolMe.XYPosGet(Wait_for_newest_data=1))*1e9
        return (self.nanonisFolMe.XYPosGet(Wait_for_newest_data=1)[0]*1e9, self.nanonisFolMe.XYPosGet(Wait_for_newest_data=1)[1]*1e9)

    def get_absolute_position(self):
        """ 
        Gets the absolute position of the STM in DAC units.
        """
        pass

    def set_bias_voltage(self, voltage_mV):
        """
        Sets the bias voltage of the STM.
        """
        self.nanonisBias.Set(bias=voltage_mV*1e-3)
        print(self.nanonisBias.Get())

    def set_filename_per_timestep(self, filename_per_timestep):
        self.filename_per_timestep=filename_per_timestep
        self.current_image_file = os.path.normpath(os.path.join(self.directory_for_stm_data, self.filename_per_timestep+'.sxm'))

    def scan_topography(self, center_nm, topography_size_nm, number_of_topography_points=256, speed_m_s=50e-9, measure_molecule_at_const_current=True, overview=False):
        """
        Images the surface and determines the topography of the moiety. The topography is saved and the channels I,x,y,z are returned.
        """
        #print('WARNING!!! scan speed fast for testing')
        #speed_m_s = 50000e-9

        # Channels need to be manually selected in the GUI. XXX Is there a way to set these witihin the software?
        # Correspond to I, X, Y, Z
        center_m_x, center_m_y = center_nm.x*1e-9, center_nm.y*1e-9
        topography_size_m = topography_size_nm*1e-9
        channel_indexes = [0,28,29,30] # [0,12,13,14] these were used in the experiments
        self.nanonisSignals.NamesGet()
        self.set_bias_voltage(voltage_mV=self.reference_voltage_mV)
        self.nanonisZCtrl.SetpntSet(self.reference_current_A)
        self.nanonisScan.PropsSet(continuous_scan=2)
        self.nanonisScan.FrameSet(x=center_m_x, y=center_m_y, w=topography_size_m, h=topography_size_m)
        self.nanonisScan.BufferSet(channel_indexes=channel_indexes, lines=number_of_topography_points, pixels=number_of_topography_points)
        if overview:
            time_stamp = datetime.datetime.now().strftime("%Y-%m-%d_%H-%M-%S")
            self.nanonisScan.PropsSet(autosave=1, series_name=time_stamp+'_overview') # XXX autosave is off for testing turn on for real experiments
        else:
            nanonis_filename = self.filename_per_timestep[:-4]
            self.nanonisScan.PropsSet(autosave=1, series_name=nanonis_filename) # XXX autosave is off for testing turn on for real experiments
            # Get latest sxm-file from STM_data directory
           
            #os.rename(join(self.directory_for_stm_data,latest_sxm_file), join(self.directory_for_stm_data,self.filename_per_timestep+'.sxm'))

        self.nanonisScan.SpeedSet(fwd_speed=speed_m_s, bwd_speed=speed_m_s)
        if measure_molecule_at_const_current:
            self.nanonisZCtrl.OnOffSet(1)
        self.nanonisScan.Action(scan_action="start", scan_direction="down") # Arguments need to be within " ... "
        self.nanonisScan.WaitEndOfScan()

        # Remove the last 4 digits of the filename added by the nanonis software
        if not overview:
            file_saved = False
            while not file_saved:
                try:
                    sxm_file = min(glob.glob(os.path.join(self.directory_for_stm_data, str(nanonis_filename)+'*')), key=os.path.getctime)
                    latest_img_file = max(glob.glob(os.path.join(self.directory_for_stm_data, str(nanonis_filename)+'*')), key=os.path.getctime)
                    os.rename(os.path.join(self.directory_for_stm_data,sxm_file), os.path.join(self.directory_for_stm_data,self.filename_per_timestep+'.sxm'))
                    #os.remove(os.path.join(self.directory_for_stm_data,sxm_file))
                    file_saved = True
                except:
                    latest_img_file = max(glob.glob(os.path.join(self.directory_for_stm_data, str(nanonis_filename)+'*')), key=os.path.getctime)
                    os.remove(os.path.join(self.directory_for_stm_data,sxm_file))
                    os.rename(os.path.join(self.directory_for_stm_data,latest_img_file), os.path.join(self.directory_for_stm_data,self.filename_per_timestep+'.sxm'))
                    file_saved = False
            self.current_image_file = max(glob.glob(os.path.join(self.directory_for_stm_data, str(nanonis_filename)+'*')), key=os.path.getctime)

                # Get last written file  STM data directory and rename it to the filename_per_timestep
                # sxm_files = glob.glob(join(self.directory_for_stm_data, self.filename_per_timestep[:-4]+'*.sxm'))
                # if sxm_files:
                #     last_sxm_file = max(sxm_files, key=os.path.getmtime)
                #     os.rename(last_sxm_file, join(self.directory_for_stm_data, self.filename_per_timestep[:-4]+'.sxm'))

                # if os.path.isfile(join(self.directory_for_stm_data,self.filename_per_timestep+'0001.sxm')):
                #     os.rename(join(self.directory_for_stm_data,self.filename_per_timestep+'0001.sxm'), join(self.directory_for_stm_data,self.filename_per_timestep+'.sxm'))

        self.set_parameters_for_overview_image()

        (I, x, y, z) = (
                        self.nanonisScan.FrameDataGrab(channel_index=0 , data_direction=1)[1], # current I is held constant?
                        self.nanonisScan.FrameDataGrab(channel_index=28, data_direction=1)[1], # x Sim: 12
                        self.nanonisScan.FrameDataGrab(channel_index=29, data_direction=1)[1], # y Sim: 13
                        self.nanonisScan.FrameDataGrab(channel_index=30, data_direction=1)[1]  # z Sim: 14
                        )

        return (I, x, y, z)
      
    def convert_action_to_DAC(self, action):
        pass

    def set_manipulation_parameters(self, voltage_mV: int, z_approach_nm: float, time_in_s=5):
        """
        Sets the shape of the voltage pulse for controlling the nanocar. Voltage and time parameters
        are set individually to generate the voltage pulse.
        
        Parameter
        ---------

        Functions
        ---------
        stm.setparam('name', 'value')
            Sets the parameter called name to the desired value.
        """
        pass

    def perform_vertical_manipulation(self, z_approach_nm: float, voltage_mV: float, x_position_nm: float, y_position_nm: float, current_limit: float, duration_pulse_s=5, speed_nm_s=20):
        """ 
        Takes a vertical manipulation spectrum at the current image point X,Y. Control is returned
        after the spectrum has been completely finished. The tip remains at the current lateral 
        position and the current signal is captured.

        Functions
        ---------
        stm.vertspectrum
            Takes a Vert.Spectrum at the current image point X,Y. Control is returned after the 
            spectrum has been completely finished. The tip remains at the current lateral position.

        stm.vertsave
            Saves the current vertspecdata.
        """
        self.nanonisZCtrl.ZPosSet(zpos=(self.reference_z_position_nm)*1e-9)
        self.nanonisZCtrl.OnOffSet(0)
        self.set_position(pos_STM_nm=[x_position_nm,y_position_nm], speed_nm_s=speed_nm_s)
        self.nanonisZCtrl.TipLiftSet(z_approach_nm*1e-9)
        self.nanonisBias.Pulse(bias_pulse_width=duration_pulse_s, bias_value=voltage_mV*1e-3, z_hold=1, wait_until_done=True)
        self.nanonisZCtrl.OnOffSet(1)
        time.sleep(0.01)
        self.nanonisZCtrl.TipLiftSet(0)
        
        print('End - vert. manipulation')


    def perform_lateral_manipulation(self, x_start_position_nm, y_start_position_nm, x_end_position_nm, y_end_position_nm, z_position_nm, voltage_mV, speed_nm_s=5, plot_data=True):
        """
        Takes a lateral manipulation spectrum between start and end point where steps defines the 
        number of measured points.

        Functions
        ---------
        latmanipxymove(Xstart, Ystart, Xend, Yend, steps, delay, preampgain, 
                    biasvoltage, currentset)
        1 | x_start          | integer | X start position in relative DAC units
        2 | y_start          | integer | Y start position in relative DAC units
        3 | x_end            | integer | X end position in relative DAC units
        4 | y_end            | integer | Y end position in relative DAC units
        5 | z_approach_nm    | float | Z approach form the imaging position in nm
        6 | voltage_mV       | integer | bias voltage in mV

        Returns
        -------
        data : list([steps])
            Contains the Z-topography between start and end.
        """
        
        # Setup start of lat. manipulation
        print(f'Z approached: {z_position_nm} nm | reference: {self.reference_z_position_nm} nm')
        self.nanonisZCtrl.OnOffSet(1)
        self.nanonisZCtrl.ZPosSet(zpos=self.reference_z_position_nm*1e-9)
        self.set_position(pos_STM_nm=[x_start_position_nm,y_start_position_nm], speed_nm_s=20)
        

        # Manipulation at const. height
        print('Const. height lat. manipulation')
        self.nanonisZCtrl.TipLiftSet(-z_position_nm*1e-9)
        self.nanonisZCtrl.OnOffSet(0)
        time.sleep(0.1)
        self.nanonisBias.Set(bias=voltage_mV*1e-3)

        # # Add additional distance to const current measurement
        # dx = x_end_position_nm-x_start_position_nm
        # dy = y_end_position_nm-y_start_position_nm
        # vector = np.array([dx, dy])
        # norm = np.linalg.norm(vector)
        # new_vector = vector/norm * (norm+0.15)
        # x_end_position_nm = x_end_position_nm + new_vector[0]
        # y_end_position_nm = y_end_position_nm + new_vector[1]

        self.set_position(pos_STM_nm=[x_end_position_nm,y_end_position_nm], speed_nm_s=speed_nm_s, wait_end_of_move=False)
        # Get current data while moving tip
        measure_current_at_const_height = []
        while(not (math.isclose(round(self.nanonisFolMe.XYPosGet(Wait_for_newest_data=0)[0],15), x_end_position_nm*1e-9, abs_tol=1e-12) and 
                   math.isclose(round(self.nanonisFolMe.XYPosGet(Wait_for_newest_data=0)[1],15), y_end_position_nm*1e-9, abs_tol=1e-12))
                   ):
            measure_current_at_const_height.append(self.nanonisSignals.ValGet(signal_index=0))
        

        # Manipulation at const. current
        print('Const. current lat. manipulation')
        self.nanonisZCtrl.TipLiftSet(0)
        self.nanonisBias.Set(bias=self.reference_voltage_mV*1e-3)
        self.nanonisZCtrl.OnOffSet(1)
        time.sleep(0.1)
        #self.nanonisZCtrl.ZPosSet(zpos=(self.reference_z_position_nm)*1e-9)
        self.set_position(pos_STM_nm=[x_start_position_nm,y_start_position_nm], speed_nm_s=speed_nm_s, wait_end_of_move=False)
        # Get current data while moving tip
        measure_x_at_const_current = []
        measure_y_at_const_current = []
        measure_height_at_const_current = []
        while(not (math.isclose(round(self.nanonisFolMe.XYPosGet(Wait_for_newest_data=0)[0],15), x_start_position_nm*1e-9, abs_tol=1e-12) and
                   math.isclose(round(self.nanonisFolMe.XYPosGet(Wait_for_newest_data=0)[1],15), y_start_position_nm*1e-9, abs_tol=1e-12))
                   ):
            measure_x_at_const_current.append(self.nanonisFolMe.XYPosGet(Wait_for_newest_data=0)[0])
            measure_y_at_const_current.append(self.nanonisFolMe.XYPosGet(Wait_for_newest_data=0)[1])
            measure_height_at_const_current.append(self.nanonisZCtrl.ZPosGet())
            
        
        # Measurement of height at constant current is done in reverse direction
        measure_height_at_const_current = measure_height_at_const_current[::-1]
        measure_x_at_const_current = measure_x_at_const_current[::-1]
        measure_y_at_const_current = measure_y_at_const_current[::-1]


        # Save lateral manipulation data
        np.save(join(self.directory_for_stm_data, self.filename_per_timestep+'_lat_mani_I_at_const_h.npy'), measure_current_at_const_height)
        np.save(join(self.directory_for_stm_data, self.filename_per_timestep+'_lat_mani_h_at_consz_I.npy'), measure_height_at_const_current)

        # Determine rough molecule position after lateral manipulation
        number_datapoints = len(measure_height_at_const_current)
        if measure_height_at_const_current:
            topography_max_index = np.argmax(measure_height_at_const_current)
            rough_moiety_position_x = measure_x_at_const_current[topography_max_index]
            rough_moiety_position_y = measure_y_at_const_current[topography_max_index]
            distance_of_lat_manipulation = np.sqrt((x_end_position_nm-x_start_position_nm)**2+(y_end_position_nm-y_start_position_nm)**2)
            movement_of_moitey_nm = distance_of_lat_manipulation * topography_max_index / number_datapoints


            self.rough_moiety_position_for_exact_search_nm = Point(rough_moiety_position_x*1e9,
                                                                rough_moiety_position_y*1e9)
        else:
            self.rough_moiety_position_for_exact_search_nm = Point(x_start_position_nm, y_end_position_nm)
            topography_max_index = 0
            distance_of_lat_manipulation = 0
            movement_of_moitey_nm = 0
            measure_current_at_const_height = [0]
            measure_current_at_const_height = [0]
        print(f'Rough moiety position: {self.rough_moiety_position_for_exact_search_nm}')

        # Plot height and current data in a (1x2) plot
        if plot_data:
            print(f'Number of datapoints {number_datapoints}')
            print(f'Topography max index {topography_max_index}')
            print(f'Distance of lateral manipulation {distance_of_lat_manipulation} nm')
            print(f'Movement of moiety {movement_of_moitey_nm} nm')
            print(f'x_start_position_nm {x_start_position_nm} | y_start_position_nm {y_start_position_nm}, x_end_position_nm {x_end_position_nm} | y_end_position_nm {y_end_position_nm}')
            print(f'x rough moiety position {self.rough_moiety_position_for_exact_search_nm.x} | y rough moiety position {self.rough_moiety_position_for_exact_search_nm.y}')
        
            fig, axs = plt.subplots(1, 2, figsize=(10, 5))
            axs[0].plot(measure_current_at_const_height)
            axs[0].set_title('Const. height manipulation')
            axs[0].set_xlabel('Step')
            axs[0].set_ylabel('Current [A]')
            axs[1].plot(measure_height_at_const_current)
            axs[1].set_title('Const. current manipulation')
            axs[1].set_xlabel('Step')
            axs[1].set_ylabel('Height [m]')
            plt.savefig(join(self.directory_for_stm_data, self.filename_per_timestep+'_lat_manip_const_h_and_I.png'))
            plt.close()
            gc.collect()

        print('End - lat. manipulation')
        return measure_current_at_const_height, measure_height_at_const_current

    def _get_rough_lat_position_for_exact_search(self):
        return self.rough_moiety_position_for_exact_search_nm

    def get_current_tip_position(self):
        return self.nanonisFolMe.XYPosGet(Wait_for_newest_data=0)

    def get_current_spectrum(self):
        """
        Reads the current spectrum from the ADC channels of the STM/AFM program.

        stm.vertdata(channel,units)
            1 | channel | integer   | 0:Time in sec == 1:X == 2:Y == 3:Current_I
            2 | units   | integer   | 0:Default == 1:Volt == 2:DAC == 3:Ampere ==
                                    4:nm == 5:Hz

        Returns
        -------
        val_I : list([number of datapoints])
            Contains the current spectrum.
        """
        pass

    def is_idle(self):
        """
        Checks the status of the STM and returns true when idle.
        """
        pass

    def is_busy(self):
        """
        Checks the status of the STM and returns true when busy.
        """
        return not bool(self.nanonisScan.WaitEndOfScan())

    def scan_topography_extension_for_testing(self, dipole_angle, action_x, action_y, mode):
        pass

class CreatecSTM(STM):


    """
    The class sends commands to the STM by using the OLE control protocol and interacts with the 
    STMAFM software.

    Comment: If you want to see the available methods in python use dir(stm) and for properties use stm._prop_map_get_

    Methods
    -------
    connect()
       Initializes the connection to the STM/AFM program.
    
    update_parameters()
        Updates all parameters and synchronizes the parameters with the DSP (dual digital feedback
        controller).
    
    get_date()
        Reads the date from the STM.
    
    beep()
        Makes a beep sound and writes 'Beep' into the log-file.
    
    get_float_param(name)
        Reads the parameter specified by name and tries to convert it to float. The parameter is a
        string and has to be within the 'Basic Parameter'-Frame of the STM/AFM software.
    
    set_position()
        This sets the new position of the STM.
    
    get_relative_position()
        Returns the actual relative STM position.
    
    get_absolute_position()
        Returns the actual absolute STM position.
    
    define_voltage_pulse()
        Defines the voltage pulse.
    
    perform_vertical_manipulation()
        This performs a vertical manipulation and generates a current spectrum.
    
    perform_lateral_manipulation()
        This performs a lateral manipulation and creates a Z-topography. This is used for searching.
    
    get_current_spectrum()
        Returns the current spectrum.
    
    is_idle()
        Checks the status of the STM and returns true when idle.
    
    is_busy()
        Checks the status of the STM and returns true when busy.
    """
 
    def __init__(self):
        super().__init__()
        self.logger = logging.getLogger("STM")

        self.connect_to_stm()

        self.CONSTANT_AD_CONVERTER = 524287
        self.convert_DAC_to_nm = None
        self.convert_nm_to_DAC = None
        
        self.convert_h_to_DAC = None
        self.convert_v_to_DAC = 1 # No conversion neecessary

        self.offsetX_init_dac = None
        self.offsetY_init_dac = None

        # Reference current to set the tip at a certain height above the surface. The value is defined in the input.yaml file.
        self.reference_current_A = self.get_yaml_data('set_reference_current_A')
        self.reference_voltage_mV = self.get_yaml_data('set_reference_voltage_mV')

        (
            self.overview_image_center_x_m,
            self.overview_image_center_y_m,
            self.overview_image_width_m,
            self.overview_image_height_m,
            self.overview_image_scan_angle_deg
        ) = self.get_parameters_from_overview_image()

        # convert values to nm
        self.overview_image_center_x_nm = round(self.overview_image_center_x_m*1e9,3)
        self.overview_image_center_y_nm = round(self.overview_image_center_y_m*1e9,3)
        self.overview_image_center_nm = np.array([self.overview_image_center_x_nm, self.overview_image_center_y_nm])
        self.overview_image_width_nm = round(self.overview_image_width_m*1e9,3)
        self.overview_image_height_nm = round(self.overview_image_height_m*1e9,3)

        self.overview_image_size_m = self.overview_image_width_m
        self.overview_image_size_nm = round(self.overview_image_size_m*1e9,3)

        
        # Environment variables
        self.number_of_episodes = 0
        self.number_of_manipulations=0
        self.number_of_translation_manipulations=0
        self.number_of_assembly_manipulations=0
        self.operation_mode=None
        self.filename_per_timestep = ''

        self.SCAN_VOLTAGE_V = 0.25
        self.SCAN_CURRENT = 1.0E-10
        self.SCAN_SPEED = 40

        self.vertman_current_limit = 1.0E-8

        self.overview_image_path = max(glob.glob(os.path.join(os.path.abspath(self.directory_for_stm_data), '*overview*.dat.jpeg')), key=os.path.getmtime)
        
        self.init_ZController()

    def get_parameters_from_overview_image(self):
        """
        Reads the parameters from the overview image and returns the values of the parameters.
        """

        # Initialize STM parameters
        # Offset is the top center pixel of the image
        self.offsetX_init_dac = -self.get_float_param('OffsetX')
        self.offsetY_init_dac = -self.get_float_param('OffsetY')
       
        self.deltaX = self.get_float_param('Delta X [DAC]')
        self.deltaY = self.get_float_param('Delta Y [DAC]')
        if self.deltaX != self.deltaY: 
            raise Exception('deltaX and deltaY are different. No universal conversion betweem DAC and nm is possible.')

        self.numX = self.get_float_param('Num.X')
        self.numY = self.get_float_param('Num.Y')
        if self.numX != self.numY:
            raise Exception('numX and numY are different. No universal conversion betweem pixel and DAC or nm is possible.')

        self.GainX = self.get_float_param('GainX')
        self.GainY = self.get_float_param('GainY')
        if self.GainX != self.GainY:
            raise Exception('GainX and GainY are different. No universal conversion betweem DAC and nm is possible.')

        self.Xpiezoconst = self.get_float_param('Xpiezoconst')
        self.Ypiezoconst = self.get_float_param('Ypiezoconst')
        self.Zpiezoconst = self.get_float_param('Zpiezoconst')

        self.convert_DAC_to_nm = self.GainX*self.Xpiezoconst/self.CONSTANT_AD_CONVERTER
        self.convert_nm_to_DAC = 1/self.convert_DAC_to_nm

        self.offsetX_init_nm = self.offsetX_init_dac*self.convert_DAC_to_nm
        self.offsetY_init_nm = self.offsetY_init_dac*self.convert_DAC_to_nm

        overview_image_center_x_nm = self.deltaX*self.numX/self.CONSTANT_AD_CONVERTER*self.GainX*self.Xpiezoconst + self.offsetX_init_nm
        overview_image_center_y_nm = self.deltaY*self.numY/self.CONSTANT_AD_CONVERTER*self.GainY*self.Xpiezoconst/2 + self.offsetY_init_nm

        overview_image_width_nm = self.deltaX*self.numX/self.CONSTANT_AD_CONVERTER*self.GainX*self.Xpiezoconst
        overview_image_height_nm = self.deltaY*self.numY/self.CONSTANT_AD_CONVERTER*self.GainY*self.Ypiezoconst

        # convert dac to m
        overview_image_center_x_m = overview_image_center_x_nm*1e-9
        overview_image_center_y_m = overview_image_center_y_nm*1e-9
        overview_image_width_m = overview_image_width_nm*1e-9
        overview_image_height_m = overview_image_height_nm*1e-9

        overview_image_scan_angle_deg = self.get_float_param('Rotation')

        self.logger.info(f'Image size [nm]: {np.round(overview_image_width_m,2)} by {np.round(overview_image_height_m,2)}')

        self.update_parameters()

        return (overview_image_center_x_m,
                overview_image_center_y_m,
                overview_image_width_m,
                overview_image_height_m,
                overview_image_scan_angle_deg)

    def init_ZController(self, speed_nm_s=20):
        """
        Measures the current of some value above the clearn surface whose coodinates are given in the input file. Initializes the z-distance of the tip
        based on a given bias voltage and current value within the tunneling junction.
        """
        self.clean_surface_position_nm = self.get_yaml_data('init_Zcontroller_position_x'), self.get_yaml_data('init_Zcontroller_position_y')
        self.clean_surface_position_dac = np.array(self.clean_surface_position_nm)*self.convert_nm_to_DAC
        self.stm.setp('STMAFM.CMD.MOVE_TIP.FBZ.NM','0.0') 
        self.stm.setp('DSP.CHK.FBON', 'ON')
        self.stm.setparam('BiasVolt.[mV]', str(self.reference_voltage_mV))
        self.stm.setp('SCAN.SETPOINT.AMPERE', self.reference_current_A)
        self.stm.move_tip_relofs(self.clean_surface_position_dac[0], self.clean_surface_position_dac[1], speed_nm_s*self.convert_nm_to_DAC,0)
        self.reference_z_position_nm = self.stm.getdacvalfb()*0.1 # convert from Angstrom to nm

        print(f'Z-Control setup done. The reference z position is {self.reference_z_position_nm} [nm]')


    def connect_to_stm(self):
        """
        Initializes the connection to the STM/AFM program.
        """
        # Connect to STM
        self.logger.info("Connecting to Createc STM")
        # Initializes the COM libraries for the calling thread
        pythoncom.CoInitialize()
        self.stm = win32com.client.Dispatch("pstmafm.stmafmrem")
        time.sleep(0.001)
        self.stm.serverneverclose()

        self.update_parameters()

    def update_parameters(self):
        """ 
        Updates all parameters and synchronizes the parameters with the DSP (dual digital feedback 
        controller).
        """
        self.logger.info("Synchronize all parameters with DSP")
        self.stm.updatedspfbparam()
        self.stm.updatedspparam()
        
    def beep(self):
        """
        Makes a beep sound and writes 'Beep' into the log-file.
        """
        self.logger.info("Beep!!!")
        self.stm.stmbeep()

    def get_float_param(self, name):
        """ 
        Reads the parameter specified by the argument and tries to convert it to float. The 
        parameter is a string as it appears in the Basic Parameter form.

        Parameters
        ----------
        name : str
            String given by the Basic Parameter in the STM/AFM program and the menu
            bar under 'Forms' -> 'Basic Parameters'.

        Returns
        -------
        value : float
            The requested value from the STM as float variable - if possible.            
        """
        value = self.stm.getparam(name)
        try:
            value = float(value)
        except:
            self.logger.error("{0} cannot be read".format(name))
        
        return value

    def set_position(self, pos_STM, speed_nm_s=20):
        """ 
        Moves STM-tip to new position. Coordinates are given in relative DAC units (relative: X,Y 
        Offset and rotation are added afterwards) Control is returned after the move has been 
        completely finished.

        Attributes
        ----------
        pos_STM : np.array(2)
            The position from the environment in DAC units.

        Functions
        ---------
        stm.move_tip_relofs(x_dac, y_dac, 2000.0, 0.0))
            1 | x_dac | single | X new position in relative DAC units
            2 | y_dac | single | Y new position in relative DAC units
            3 | Speed | single | Speed in DAC units/s
            4 | Units | integer| reserved
        """
        x_dac, y_dac = float(pos_STM[0]), float(pos_STM[1])
        x_nm = x_dac*self.convert_DAC_to_nm
        y_nm = y_dac*self.convert_DAC_to_nm
        self.stm.setp('STMAFM.CMD.MOVE_TIP.NEWPOS.NM', (x_nm, y_nm, speed_nm_s))

    def get_relative_position_nm(self):  
        """ 
        Gets the relative position of the STM in DAC units.
    
        Functions
        ---------
        get_float_param('name')
            Returns the value of the standard parameter you passed over to the STM/AFM program.

        Return
        ------
        relative_stm_position : np.array(2)
            The relative position of the STM-tip.
        """
        #convert the two floats to a np.array

        relative_stm_position_nm = np.array([self.get_float_param('VertSpecPosX'), self.get_float_param('VertSpecPosY')])*self.convert_DAC_to_nm
        return relative_stm_position_nm

    def get_absolute_position(self):
        """ 
        Gets the absolute position of the STM in DAC units.
    
        Functions
        ---------
        get_float_param('name')
            Returns the value of the standard parameter you passed over to the STM/AFM program.

        Return
        ------
        absolute_stm_position : np.array(2)
            The absolute position of the STM-tip.
        """
        X_Offset = self.offsetX_init_dac
        Y_Offset = self.offsetY_init_dac
        X_Relativ = self.get_float_param('VertSpecPosX')
        Y_Relativ = self.get_float_param('VertSpecPosY')
        absolute_stm_position = np.array(np.zeros(2))
        absolute_stm_position = np.array([X_Offset+X_Relativ, Y_Offset+Y_Relativ])
        return absolute_stm_position

    def set_filename_per_timestep(self,filename_per_timestep):
        self.filename_per_timestep=filename_per_timestep
        self.current_image_file = os.path.normpath(os.path.join(self.directory_for_stm_data, self.filename_per_timestep))


    def scan_topography(self, center_nm, topography_size_nm, number_of_topography_points=256, speed_m_s=5000e-9, measure_molecule_at_const_current=True, overview=False): #start_image_x_DAC, start_image_y_DAC, num_pixels, topography_size_nm, operation_mode=None):
        """
        Images the molecule and measures its topography. 
        The STM image is saved and the topography returned.

        Return
        ------
            topography_measurement_I: np.array([number_of_topography_points, number_of_topography_points])
            topography_measurement_x: np.array([number_of_topography_points, number_of_topography_points])
            topography_measurement_y: np.array([number_of_topography_points, number_of_topography_points])
            topography_measurement_z: np.array([number_of_topography_points, number_of_topography_points])
        """
       
        center_x_nm = center_nm[0]
        center_y_nm = center_nm.y
        center_x_m = center_x_nm*1e-9
        center_y_m = center_y_nm*1e-9
        speed_nm_s = speed_m_s*1e9

        topography_size_m = topography_size_nm*1e-9
        half_topography_size_m = topography_size_m/2
        half_topography_size_nm = topography_size_nm/2
        half_topography_size_dac = half_topography_size_nm*self.convert_nm_to_DAC

        # start_image_x_dac = center_x_nm*self.convert_nm_to_DAC
        # start_image_y_dac = center_y_nm*1e9*self.convert_nm_to_DAC - half_topography_size_dac

        # # Is the offset required here?
        # x_DAC = self.offsetX_init_dac+start_image_x_dac
        # y_DAC = self.offsetY_init_dac+start_image_y_dac

        # x_pixel_off = x_DAC*524287/self.deltaX/10/self.GainX/self.Xpiezoconst
        # y_pixel_off = y_DAC*524287/self.deltaY/10/self.GainY/self.Ypiezoconst

        # x_nm = float(x_DAC*self.convert_DAC_to_nm) 
        # y_nm = float(y_DAC*self.convert_DAC_to_nm)


        # Measure molecule topography
        self.stm.setp('STMAFM.CMD.SETXYOFF.NM', (center_x_nm,center_y_nm-half_topography_size_nm))
        while (self.stm.getp('STMAFM.SCANSTATUS','') != 0):
            time.sleep(0.01)
        self.stm.setp('SCAN.IMAGESIZE.NM', (float(topography_size_nm),float(topography_size_nm)))
        self.stm.setp('SCAN.IMAGESIZE.PIXEL', (number_of_topography_points, number_of_topography_points))
        self.stm.setp('SCAN.SPEED.NM/SEC',speed_nm_s)
        self.stm.setp('SCAN.BIASVOLTAGE.VOLT', self.SCAN_VOLTAGE_V)
        self.stm.setp('SCAN.SETPOINT.AMPERE', self.SCAN_CURRENT)
        time.sleep(0.025)

        self.stm.scanstart()
        self.stm.scanwaitfinished()

        dat_save_done = False
        while not dat_save_done:
            try:
                if overview:
                    time_stamp = datetime.datetime.now().strftime("%Y-%m-%d_%H-%M-%S")
                    self.stm.savedatfilename = time_stamp+'_overview.dat'
                else:
                    self.stm.savedatfilename = self.filename_per_timestep+'.dat' 
                self.stm.setp('STMAFM.FILE.NAME.SAVE.DAT', str(self.stm.savedatfilename))
                self.stm.setp('STMAFM.FILE.SAVE.DAT',join(self.directory_for_stm_data,self.stm.savedatfilename))
                self.stm.quicksave()
                dat_save_done = True
            except OSError:
                    print("Image molecule could not be saved!")
                    dat_save_done = False
                    continue
        
        # Get topography data
        # I in A
        topography_measurement_I = np.array(self.stm.getp('DATA.SCAN', (2,3)))
        # x and y in m
        linspace_x_m = np.linspace(-half_topography_size_m, half_topography_size_m, number_of_topography_points)+center_x_m
        linspace_y_m = np.linspace(-half_topography_size_m, half_topography_size_m, number_of_topography_points)+center_y_m
        topography_measurement_x_m, topography_measurement_y_m = np.meshgrid(linspace_x_m, linspace_y_m)
        # Topography in m
        topography_measurement_z_nm = np.array(self.stm.getp('DATA.SCAN', (1,4)))
        topography_measurement_z_m = topography_measurement_z_nm*1e-9
        
        offsetX_init_nm = self.offsetX_init_dac*self.convert_DAC_to_nm
        offsetY_init_nm = self.offsetY_init_dac*self.convert_DAC_to_nm
        set_settings = False
        while not set_settings:
            try:
                self.stm.setp('STMAFM.CMD.SETXYOFF.NM', (offsetX_init_nm, offsetY_init_nm))
                self.stm.setp('SCAN.IMAGESIZE.NM', (float(self.overview_image_width_nm),float(self.overview_image_height_nm)))
                self.stm.setp('SCAN.IMAGESIZE.PIXEL', (self.numX, self.numY))
                self.stm.setp('SCAN.SETPOINT.AMPERE', self.SCAN_CURRENT)
                set_settings = True
            except OSError:
                    print("Image molecule could not be saved!")
                    set_settings = False
                    continue
        return (topography_measurement_I, topography_measurement_x_m, topography_measurement_y_m, topography_measurement_z_m)

    def convert_action_to_DAC(self, action):
        print('ERROR: conversion factor for height and voltage are not correct')
        action = self.convert_action_to_real(action)
        return [int(action[0]*self.convert_h_to_DAC ), 
                int(action[1]*self.convert_v_to_DAC ), 
                int(action[2]*self.convert_nm_to_DAC), 
                int(action[3]*self.convert_nm_to_DAC)]

    def set_manipulation_parameters(self, voltage_mV: int, z_approach_nm: float, duration_pulse_s=5):
        """
        Sets the shape of the voltage pulse for controlling the nanocar. Voltage and time parameters
        are set individually to generate the voltage pulse.
        
        Parameter
        ---------

        Functions
        ---------
        stm.setparam('name', 'value')
            Sets the parameter called name to the desired value.
        """
        # Sets the duration of the voltage pulse based on the number of datapoints:
        n_datapoints = 10000
        t_delay = duration_pulse_s/(2e-5*n_datapoints)

        # Set z approach
        self.stm.setp('VERTMAN.Z_OFFSET.NM', str(z_approach_nm))
        self.stm.setparam('Vertmandelay',str(t_delay))
        self.stm.setparam('Vertmangain','8')

        # Set voltage
        self.stm.setparam('Vpoint0.t','0')
        self.stm.setparam('Vpoint1.t',str(n_datapoints))
        self.stm.setparam('Vpoint2.t','0')
        self.stm.setparam('Vpoint3.t','0')
        self.stm.setparam('Vpoint4.t','0')
        self.stm.setparam('Vpoint5.t','0')
        self.stm.setparam('Vpoint6.t','0')
        self.stm.setparam('Vpoint7.t','0')

        self.stm.setparam('Vpoint0.V',str(voltage_mV))
        self.stm.setparam('Vpoint1.V',str(voltage_mV))
        self.stm.setparam('Vpoint2.V','0')
        self.stm.setparam('Vpoint3.V','0')
        self.stm.setparam('Vpoint4.V','0')
        self.stm.setparam('Vpoint5.V','0')
        self.stm.setparam('Vpoint6.V','0')
        self.stm.setparam('Vpoint7.V','0')

        self.stm.setparam('Zpoint0.t','0')
        self.stm.setparam('Zpoint1.t','0')
        self.stm.setparam('Zpoint2.t','0')
        self.stm.setparam('Zpoint3.t','0')
        self.stm.setparam('Zpoint4.t','0')
        self.stm.setparam('Zpoint5.t','0')
        self.stm.setparam('Zpoint6.t','0')
        self.stm.setparam('Zpoint7.t','0')

        self.stm.setparam('Zpoint0.z','0')
        self.stm.setparam('Zpoint1.z','0')
        self.stm.setparam('Zpoint2.z','0')
        self.stm.setparam('Zpoint3.z','0')
        self.stm.setparam('Zpoint4.z','0')
        self.stm.setparam('Zpoint5.z','0')
        self.stm.setparam('Zpoint6.z','0')
        self.stm.setparam('Zpoint7.z','0')

        self.stm.setp('UPDATE.VERTMAN','')


    def perform_vertical_manipulation(self, z_approach_nm: float, voltage_mV: int, x_position_nm: int, y_position_nm: int, current_limit: float, duration_pulse_s=5, speed_nm_s=20):
        """ 
        Takes a vertical manipulation spectrum at the current image point X,Y. Control is returned
        after the spectrum has been completely finished. The tip remains at the current lateral 
        position and the current signal is captured.

        Functions
        ---------
        stm.vertspectrum
            Takes a Vert.Spectrum at the current image point X,Y. Control is returned after the 
            spectrum has been completely finished. The tip remains at the current lateral position.

        stm.vertsave
            Saves the current vertspecdata.
        """
        # Set the position of the tip
        x_position_dac = x_position_nm*self.convert_nm_to_DAC
        y_position_dac = y_position_nm*self.convert_nm_to_DAC
        x_pixel = x_position_dac/self.deltaX + self.numX/2
        y_pixel = y_position_dac/self.deltaY

        # Set reference voltage
        self.stm.setparam('BiasVolt.[mV]', str(self.reference_voltage_mV))
        # Turn of feedback and set z-distance to reference value
        self.stm.setp('DSP.CHK.FBON','OFF')
        self.stm.setp('UPDATE.PARAMETER','')
        # Set induced max current to stop the vertical manipulation
        self.stm.setp('VERTMAN.CHK.LIMIT.CURRENT','ON') #'OFF')
        self.stm.setp('VERTMAN.LIMIT.CURRENT.MAX', str(current_limit))
        # Measures a vertspectrum at current position
        self.stm.setp('VERTMAN.LATSPEED.NM/SEC', str(speed_nm_s))
        # Adjust the z-distance to match the reference value before adding the z-approach selected by the agent
        current_z_position_nm = self.stm.getdacvalfb()*0.1
        delta_z_to_match_reference_nm = self.reference_z_position_nm-current_z_position_nm
        self.stm.setp('STMAFM.CMD.MOVE_TIP.FBZ.NM',str(delta_z_to_match_reference_nm))
        self.stm.setp('UPDATE.PARAMETER','')
        self.set_manipulation_parameters(voltage_mV=voltage_mV, z_approach_nm=z_approach_nm, duration_pulse_s=duration_pulse_s) 
        #self.stm.vertspectrum()
        self.stm.btn_vertspec(x_pixel,y_pixel)
        while (self.stm.getp('STMAFM.SCANSTATUS','') != 0):
            time.sleep(0.01)

        # Clear the z feedback distance and turn on the feedback
        self.stm.setp('STMAFM.CMD.MOVE_TIP.FBZ.NM','0.0') # Important to set to 0.0
        self.stm.setp('UPDATE.PARAMETER','')
        self.stm.setp('DSP.CHK.FBON','ON')

        vert_save_done = False
        while not vert_save_done:
            try:
                self.stm.savevertfilename = self.filename_per_timestep + '_V-pulse.vert'
                self.stm.vertsave()
                vert_save_done = True
            except Exception as ex:
                print('Error vertical manipulation: ' +  str(ex))
                vert_save_done = False
                continue

    def perform_lateral_manipulation(self, x_start_position_nm, y_start_position_nm, x_end_position_nm, y_end_position_nm, z_position_nm, voltage_mV, speed_nm_s=2, plot_data=False):
        """
        Takes a lateral manipulation spectrum between start and end point where steps defines the 
        number of measured points.

        Functions
        ---------
        latmanipxymove(Xstart, Ystart, Xend, Yend, steps, delay, preampgain, 
                       biasvoltage, currentset)
        1 | x_start          | integer | X start position in relative DAC units
        2 | y_start          | integer | Y start position in relative DAC units
        3 | x_end            | integer | X end position in relative DAC units
        4 | y_end            | integer | Y end position in relative DAC units
        5 | z_approach_nm   | float | Z approach form the imaging position in nm
        6 | voltage_mV      | integer | bias voltage in mV

        Returns
        -------
        data : list([steps])
            Contains the Z-topography between start and end.
        """

        self.lat_step_size_nm = 0.02

        # turn off feedback
        self.stm.setp('STMAFM.OLECOMBLOCKING','OFF')

        # --- set general parameters
        self.stm.setp('LATMAN.CURRENT.NAMP',str(self.reference_current_A*10e9))
        self.stm.setp('LATMAN.MANIP_SPEED.NM/SEC',str(speed_nm_s))
        self.stm.setp('LATMAN.STEPSIZE.NM',str(self.lat_step_size_nm))
        self.stm.setp('LATMAN.EXTENSION', '1')
        self.stm.setp('LATMAN.PREAMPGAIN.EXPONENT', '8')
        self.stm.setp('UPDATE.LATMAN','')
        # ---

        # --- move to start position of the lateral manipulation with imaging conditions
        self.stm.setp('STMAFM.CMD.MOVE_TIP.NEWPOS.NM',(str(x_start_position_nm),str(y_start_position_nm),str(self.SCAN_SPEED)))
        while (self.stm.getp('STMAFM.SCANSTATUS','') != 0):
            time.sleep(0.01)
        # ---

        # --- do lateral manipulation with const_height (no feedback) and apply z-approach
        self.stm.setp('DSP.CHK.FBON','OFF')
        self.stm.setp('UPDATE.PARAMETER','')
        # --- set bias voltage
        self.stm.setp('LATMAN.VOLTAGE.VOLT', str(float(voltage_mV/1000)))
        self.stm.setp('UPDATE.LATMAN','')
        # ---
        current_z_position_nm = self.stm.getdacvalfb()*0.1
        delta_z_to_match_reference_nm = self.reference_z_position_nm-current_z_position_nm
        self.stm.setp('STMAFM.CMD.MOVE_TIP.FBZ.NM',str(delta_z_to_match_reference_nm+z_position_nm))
        self.stm.setp('UPDATE.PARAMETER','')

        self.stm.setp('LATMAN.CMD.MANIPXYMOVE.NM',(str(x_start_position_nm),str(y_start_position_nm),str(x_end_position_nm),str(y_end_position_nm)))
        while (self.stm.getp('STMAFM.SCANSTATUS','') != 0):
            time.sleep(0.01)
        latdata=self.stm.getp('DATA.LATMAN',('15','2'))
        measure_current_at_const_height = np.ravel(latdata)
        # --- 

        # --- do lateral manipulation with const current: To find molecule position
        self.stm.setp('STMAFM.CMD.MOVE_TIP.FBZ.NM','0.0') # Important to set to 0.0
        self.stm.setp('UPDATE.PARAMETER','')
        self.stm.setp('DSP.CHK.FBON','ON')
        
        # do lat. manip. in reverse direction
        self.stm.setp('LATMAN.CMD.MANIPXYMOVE.NM',(str(x_end_position_nm),str(y_end_position_nm),str(x_start_position_nm),str(y_start_position_nm)))
        while (self.stm.getp('STMAFM.SCANSTATUS','') != 0):
            time.sleep(0.01)

        # --- move tip back to origin: retract tip and turn on feedbackloop
        self.stm.setp('STMAFM.CMD.MOVE_TIP.FBZ.NM','0.0')
        self.stm.setp('STMAFM.CMD.MOVE_TIP.NEWPOS.NM',('0.0','0.0','20')) # ('0.0','0.0','0.5')) 
        while (self.stm.getp('STMAFM.SCANSTATUS','') != 0):
            time.sleep(0.01)
        # ---
        
        latdata=self.stm.getp('DATA.LATMAN',('15','2')) # ('15','2'))
        latdata_15 = np.ravel(self.stm.getp('DATA.LATMAN',('15','2')))[::-1] # ('15','2'))
        self.latdata = -np.ravel(latdata)
        measure_height_at_const_current = self.latdata[::-1]


        # Save lateral manipulation data
        np.save(join(self.directory_for_stm_data, self.filename_per_timestep+'_lat_mani_I_at_const_h.npy'), measure_current_at_const_height)
        np.save(join(self.directory_for_stm_data, self.filename_per_timestep+'_lat_mani_h_at_consz_I.npy'), measure_height_at_const_current)

        # Determine rough molecule position after lateral manipulation
        number_datapoints = len(measure_height_at_const_current)
        topography_max_index = np.argmax(measure_height_at_const_current)
        distance_of_lat_manipulation = np.sqrt((x_end_position_nm-x_start_position_nm)**2+(y_end_position_nm-y_start_position_nm)**2)
        movement_of_moitey_nm = distance_of_lat_manipulation * topography_max_index / number_datapoints

        manipulation_angle = np.arctan2(y_end_position_nm-y_start_position_nm, x_end_position_nm-x_start_position_nm)

        self.rough_moiety_position_for_exact_search_nm = Point(x_start_position_nm+movement_of_moitey_nm*np.cos(manipulation_angle), 
                                                               y_start_position_nm+movement_of_moitey_nm*np.sin(manipulation_angle))

        lat_save_done = False
        while not lat_save_done:
            try:
                self.stm.savelatfilename = f'{datetime.now().strftime("%y-%m-%d__%H-%M-%S")}_Z-{z_position_nm}_V-{float(voltage_mV)}.LAT'
                self.stm.latsave()
                lat_save_done = True
            except Exception as ex:
                print('Error lateral manipulation: ' +  str(ex))
                lat_save_done = False
                continue

         # Plot height and current data in a (1x2) plot
        if plot_data:
            print(f'Number of datapoints {number_datapoints}')
            print(f'Topography max index {topography_max_index}')
            print(f'Distance of lateral manipulation {distance_of_lat_manipulation} nm')
            print(f'Movement of moiety {movement_of_moitey_nm} nm')
            print(f'x_start_position_nm {x_start_position_nm} | y_start_position_nm {y_start_position_nm}, x_end_position_nm {x_end_position_nm} | y_end_position_nm {y_end_position_nm}')
            print(f'x rough moiety position {self.rough_moiety_position_for_exact_search_nm.x} | y rough moiety position {self.rough_moiety_position_for_exact_search_nm.y}')
        
            fig, axs = plt.subplots(1, 2, figsize=(10, 5))
            axs[0].plot(measure_current_at_const_height)
            axs[0].set_title('Const. height manipulation')
            axs[0].set_xlabel('Step')
            axs[0].set_ylabel('Current [A]')
            axs[1].plot(measure_height_at_const_current)
            axs[1].set_title('Const. current manipulation')
            axs[1].set_xlabel('Step')
            axs[1].set_ylabel('Height [m]')
            plt.savefig(join(self.directory_for_stm_data, self.filename_per_timestep+'_lat_manip_const_h_and_I.png'))

        print('End - lat. manipulation')
        return measure_current_at_const_height, measure_height_at_const_current

    def _get_rough_lat_position_for_exact_search(self):
        return self.rough_moiety_position_for_exact_search_nm

    def get_current_spectrum(self):
        """
        Reads the current spectrum from the ADC channels of the STM/AFM program.

        stm.vertdata(channel,units)
            1 | channel | integer   | 0:Time in sec == 1:X == 2:Y == 3:Current_I
            2 | units   | integer   | 0:Default == 1:Volt == 2:DAC == 3:Ampere ==
                                      4:nm == 5:Hz

        Returns
        -------
        val_I : list([number of datapoints])
            Contains the current spectrum.
        """
        # Reads time signal in default units
        val_t = self.stm.vertdata(0,0)
        #self.update_parameters()

        # Reads current signal from channel (ADC0) in DAC units
        val_I = self.stm.vertdata(3,1) # 1= units in mV
        return val_t, val_I

    def is_idle(self):
        """
        Checks the status of the STM and returns true when idle.
        """

        status = self.stm.scanstatus
        self.logger.info("STM status: %i" % status)
        if status == 0:
            self.logger.info("Checking STM status: STM is idle")
        else:
            self.logger.info("Checking STM status: STM is busy")
        return status == 0

    def is_busy(self):
        """
        Checks the status of the STM and returns true when busy.
        """
        return not self.is_idle()

    def scan_topography_extension_for_testing(self, dipole_angle, action_x, action_y, mode):
        pass



#%%

def get_yaml_data(*args):
    """ Get input.yaml entry for a specific keyword.

        Parameters
        ----------------
        keyword | list
            Keyword for which the value is read from the input.yaml file.

        Return
        ----------------
        value_read | str, float, int
    """        
    with open(os.path.normpath(os.path.join(os.getcwd(),"input.yaml")), "r") as ymlfile:
        data = yaml.load(ymlfile, Loader=yaml.FullLoader)

    for val in args:
        if 'molecule_' in val or 'atom_' in val:
            moiety_type = val
        else:
            moiety_val = val

    if len(args) == 1:
        return data[args[0]]
    else:
        return data[moiety_type][moiety_val]
   

# #%%
# def main():
#     hardware = 'createc'
#     stm = STM(hardware, 'C/Repository/AMAN_SPM/stm_data')


#     current_moiety = 'molecule_FePc_4K'
#     save_val_table_after_n_learning_progressions=get_yaml_data(current_moiety, 'save_val_table_after_n_learning_progressions')
#     set_reference_voltage_mV=get_yaml_data(current_moiety,'set_reference_voltage_mV')
#     stop_after_episode=get_yaml_data('stop_after_episode')
#     current_moiety = 'atom_Ag_4K'
#     set_reference_current_A=get_yaml_data(current_moiety,'set_reference_current_A')


#     print(f'set_reference_voltage_mV {set_reference_voltage_mV}')
# #%%
# main()
# # %%
