def __init__(self, process): super().__init__() self.mount(process) self.resize(640, 480) self.title = "Table Control" self.add_x_button = KeypadButton(text="+X", clicked=self.on_add_x) self.sub_x_button = KeypadButton(text="-X", clicked=self.on_sub_x) self.add_y_button = KeypadButton(text="+Y", clicked=self.on_add_y) self.sub_y_button = KeypadButton(text="-Y", clicked=self.on_sub_y) self.add_z_button = KeypadButton(text="+Z", clicked=self.on_add_z) self.sub_z_button = KeypadButton(text="-Z", clicked=self.on_sub_z) self.control_buttons = (self.add_x_button, self.sub_x_button, self.add_y_button, self.sub_y_button, self.add_z_button, self.sub_z_button) self.probecard_light_button = ToggleButton( text="PC Light", tool_tip="Toggle probe card light", checkable=True, checked=False, enabled=False, toggled=self.on_probecard_light_clicked) self.microscope_light_button = ToggleButton( text="Mic Light", tool_tip="Toggle microscope light", checkable=True, checked=False, enabled=False, toggled=self.on_microscope_light_clicked) self.box_light_button = ToggleButton(text="Box Light", tool_tip="Toggle box light", checkable=True, checked=False, enabled=False, toggled=self.on_box_light_clicked) # Create movement radio buttons self.step_width_buttons = ui.Column() for item in self.load_table_step_sizes(): step_size = item.get('step_size') * comet.ureg('um') step_color = item.get('step_color') step_size_label = format_metric(step_size.to('m').m, 'm', decimals=1) button = ui.RadioButton( text=step_size_label, tool_tip=f"Move in {step_size_label} steps.", stylesheet=f"QRadioButton:enabled{{color:{step_color};}}", checked=len(self.step_width_buttons) == 0, toggled=self.on_step_toggled) button.movement_width = step_size.to('mm').m button.movement_color = step_color self.step_width_buttons.append(button) self.control_layout = ui.Column(ui.Row(ui.Row( ui.Column(KeypadSpacer(), self.sub_y_button, KeypadSpacer()), ui.Column( self.sub_x_button, KeypadSpacer(), self.add_x_button, ), ui.Column( KeypadSpacer(), self.add_y_button, KeypadSpacer(), ), KeypadSpacer(), ui.Column(self.add_z_button, KeypadSpacer(), self.sub_z_button)), ui.Spacer(), stretch=(0, 1)), ui.Spacer(), stretch=(0, 1)) self.positions_widget = TablePositionsWidget( position_picked=self.on_position_picked, absolute_move=self.on_absolute_move) self.contacts_widget = TableContactsWidget( position_picked=self.on_position_picked, absolute_move=self.on_absolute_move) self.pos_x_label = PositionLabel() self.pos_y_label = PositionLabel() self.pos_z_label = PositionLabel() self.cal_x_label = CalibrationLabel("cal") self.cal_y_label = CalibrationLabel("cal") self.cal_z_label = CalibrationLabel("cal") self.rm_x_label = CalibrationLabel("rm") self.rm_y_label = CalibrationLabel("rm") self.rm_z_label = CalibrationLabel("rm") self.z_limit_label = PositionLabel() self.laser_label = SwitchLabel() self.calibrate_button = ui.Button(text="Calibrate", clicked=self.on_calibrate) self.update_interval_number = ui.Number( value=self.settings.get("table_control_update_interval") or 1.0, minimum=.5, maximum=10.0, decimals=2, step=0.25, suffix="s", changed=self.on_update_interval_changed) self.progress_bar = ui.ProgressBar(visible=False) self.message_label = ui.Label() self.stop_button = ui.Button(text="&Stop", default=False, auto_default=False, enabled=False, clicked=self.on_stop) self.close_button = ui.Button(text="&Close", default=False, auto_default=False, clicked=self.on_close) self.layout = ui.Column( ui.Row( ui.Column( ui.Row( ui.GroupBox( title="Control", layout=ui.Column( ui.Spacer(horizontal=False), self.control_layout, ui.Spacer(horizontal=False) ) ), ui.GroupBox( title="Step Width", layout=self.step_width_buttons ), ui.GroupBox( title="Lights", layout=ui.Column( self.probecard_light_button, self.microscope_light_button, self.box_light_button, ui.Spacer() ) ), stretch=(0, 1) ), ui.Tabs( ui.Tab( title="Move", layout=self.positions_widget ), ui.Tab( title="Contacts", layout=self.contacts_widget ), ui.Tab( title="Calibrate", layout=ui.Column( ui.GroupBox( title="Table Calibration", layout=ui.Column( ui.Row( self.calibrate_button, ui.Label("Calibrate table by moving into cal/rm switches\nof every axis in" \ " a safe manner to protect the probe card."), stretch=(0, 1) ) ) ), ui.Spacer(), stretch=(0, 1) ) ), ui.Tab( title="Options", layout=ui.Column( ui.GroupBox( title="Update Interval", layout=ui.Row( self.update_interval_number, ui.Spacer(), stretch=(0, 1) ) ), ui.Spacer(), stretch=(0, 1) ) ) ), stretch=(0, 1) ), ui.Column( ui.GroupBox( title="Position", layout=ui.Row( ui.Column( ui.Label("X"), ui.Label("Y"), ui.Label("Z") ), ui.Column( self.pos_x_label, self.pos_y_label, self.pos_z_label ) ) ), ui.GroupBox( title="Calibration", layout=ui.Row( ui.Column( ui.Label("X"), ui.Label("Y"), ui.Label("Z") ), ui.Column( self.cal_x_label, self.cal_y_label, self.cal_z_label ), ui.Column( self.rm_x_label, self.rm_y_label, self.rm_z_label ), stretch=(1, 1) ) ), ui.GroupBox( title="Limits", layout=ui.Row( ui.Column( ui.Label("Z") ), ui.Column( self.z_limit_label ) ) ), ui.GroupBox( title="Safety", layout=ui.Row( ui.Label( text="Laser Sensor" ), self.laser_label ) ), ui.Spacer() ), stretch=(1, 0) ), ui.Row( self.progress_bar, self.message_label, ui.Spacer(), self.stop_button, self.close_button ), stretch=(1, 0) ) self.close_event = self.on_close_event self.reset_position() self.reset_caldone() self.update_limits() self.reset_safety() self.update_control_buttons()
def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) self.title = "Bias + IV Ramp" self.register_vsource() self.register_hvsource() self.register_environment() self.plot = ui.Plot(height=300, legend="right") self.plot.add_axis("x", align="bottom", text="Voltage [V]") self.plot.add_axis("y", align="right", text="Current [uA]") self.plot.add_series("vsrc", "x", "y", text="V Source", color="blue") self.plot.add_series("xfit", "x", "y", text="Fit", color="magenta") self.data_tabs.insert(0, ui.Tab(title="IV Curve", layout=self.plot)) self.voltage_start = ui.Number(decimals=3, suffix="V") self.voltage_stop = ui.Number(decimals=3, suffix="V") self.voltage_step = ui.Number(minimum=0, maximum=200, decimals=3, suffix="V") self.waiting_time = ui.Number(minimum=0, decimals=2, suffix="s") self.bias_voltage = ui.Number(decimals=3, suffix="V") self.bias_mode = ui.ComboBox(["constant", "offset"]) self.hvsrc_current_compliance = ui.Metric(minimum=0, decimals=3, prefixes='mun', unit="A") self.vsrc_current_compliance = ui.Metric(minimum=0, decimals=3, prefixes='mun', unit="A") self.bind("voltage_start", self.voltage_start, 0, unit="V") self.bind("voltage_stop", self.voltage_stop, 0, unit="V") self.bind("voltage_step", self.voltage_step, 0, unit="V") self.bind("waiting_time", self.waiting_time, 1, unit="s") self.bind("bias_voltage", self.bias_voltage, 0, unit="V") self.bind("bias_mode", self.bias_mode, "constant") self.bind("hvsrc_current_compliance", self.hvsrc_current_compliance, 0, unit="A") self.bind("vsrc_current_compliance", self.vsrc_current_compliance, 0, unit="A") self.general_tab.layout = ui.Row( ui.GroupBox(title="HV Source Ramp", layout=ui.Column(ui.Label(text="Start"), self.voltage_start, ui.Label(text="Stop"), self.voltage_stop, ui.Label(text="Step"), self.voltage_step, ui.Label(text="Waiting Time"), self.waiting_time, ui.Spacer())), ui.GroupBox(title="V Source Bias", layout=ui.Column(ui.Label(text="Bias Voltage"), self.bias_voltage, ui.Label(text="Bias Compliance"), self.vsrc_current_compliance, ui.Label(text="Bias Mode"), self.bias_mode, ui.Spacer())), ui.GroupBox(title="HV Source", layout=ui.Column(ui.Label(text="Compliance"), self.hvsrc_current_compliance, ui.Spacer())), stretch=(1, 1, 1)) ampere = comet.ureg('A') volt = comet.ureg('V') self.series_transform['vsrc'] = lambda x, y: ((x * ampere).to('uA').m, (y * volt).to('V').m) self.series_transform['xfit'] = self.series_transform.get('vsrc')
def register_lcr(self): def change_lcr_open_correction_mode(mode): self.lcr_open_correction_channel.enabled = mode == "multi" self.lcr_integration_time = ui.ComboBox(["short", "medium", "long"]) self.lcr_averaging_rate = ui.Number(minimum=1, maximum=256, decimals=0) self.lcr_auto_level_control = ui.CheckBox(text="Auto Level Control") self.lcr_soft_filter = ui.CheckBox(text="Filter STD/mean < 0.005") self.lcr_open_correction_mode = ui.ComboBox( ["single", "multi"], changed=change_lcr_open_correction_mode) self.lcr_open_correction_channel = ui.Number(minimum=0, maximum=127, decimals=0) change_lcr_open_correction_mode(self.lcr_open_correction_mode.current) self.bind("lcr_integration_time", self.lcr_integration_time, "medium") self.bind("lcr_averaging_rate", self.lcr_averaging_rate, 1) self.bind("lcr_auto_level_control", self.lcr_auto_level_control, True) self.bind("lcr_soft_filter", self.lcr_soft_filter, True) self.bind("lcr_open_correction_mode", self.lcr_open_correction_mode, "single") self.bind("lcr_open_correction_channel", self.lcr_open_correction_channel, 0) self.status_lcr_voltage = ui.Text(value=NO_VALUE, readonly=True) self.status_lcr_current = ui.Text(value=NO_VALUE, readonly=True) self.status_lcr_output = ui.Text(value=NO_VALUE, readonly=True) self.bind("status_lcr_voltage", self.status_lcr_voltage, NO_VALUE) self.bind("status_lcr_current", self.status_lcr_current, NO_VALUE) self.bind("status_lcr_output", self.status_lcr_output, NO_VALUE) self.status_panel.append( ui.GroupBox(title="LCR Status", layout=ui.Row( ui.Column(ui.Label("Voltage"), self.status_lcr_voltage), ui.Column(ui.Label("Current"), self.status_lcr_current), ui.Column(ui.Label("Output"), self.status_lcr_output)))) self.control_tabs.append( ui.Tab(title="LCR", layout=ui.Row( ui.GroupBox(title="Open Correction", layout=ui.Column( ui.Label(text="Method"), self.lcr_open_correction_mode, ui.Label(text="Channel"), self.lcr_open_correction_channel, ui.Spacer())), ui.GroupBox(title="Options", layout=ui.Column( ui.Label(text="Integration Time"), self.lcr_integration_time, ui.Label(text="Averaging Rate"), self.lcr_averaging_rate, self.lcr_auto_level_control, self.lcr_soft_filter, ui.Spacer())), ui.Spacer(), stretch=(1, 1, 1)))) def handler(state): if 'lcr_voltage' in state: value = state.get('lcr_voltage') self.status_lcr_voltage.value = format_metric(value, "V") if 'lcr_current' in state: value = state.get('lcr_current') self.status_lcr_current.value = format_metric(value, "A") if 'lcr_output' in state: value = state.get('lcr_output') self.status_lcr_output.value = format_switch(value, default=NO_VALUE) self.state_handlers.append(handler)
def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) self.title = "CV Ramp (HV Source)" self.register_vsource() self.register_lcr() self.register_environment() self.plot = ui.Plot(height=300, legend="right") self.plot.add_axis("x", align="bottom", text="Voltage [V] (abs)") self.plot.add_axis("y", align="right", text="Capacitance [pF]") self.plot.add_series("lcr", "x", "y", text="LCR Cp", color="blue") self.data_tabs.insert(0, ui.Tab(title="CV Curve", layout=self.plot)) self.plot2 = ui.Plot(height=300, legend="right") self.plot2.add_axis("x", align="bottom", text="Voltage [V] (abs)") self.plot2.add_axis("y", align="right", text="1/Capacitance² [1/F²]") self.plot2.axes.get("y").qt.setLabelFormat("%G") self.plot2.add_series("lcr2", "x", "y", text="LCR Cp", color="blue") self.data_tabs.insert(1, ui.Tab(title="1/C² Curve", layout=self.plot2)) self.voltage_start = ui.Number(decimals=3, suffix="V") self.voltage_stop = ui.Number(decimals=3, suffix="V") self.voltage_step = ui.Number(minimum=0, maximum=200, decimals=3, suffix="V") self.waiting_time = ui.Number(minimum=0, decimals=2, suffix="s") self.hvsrc_current_compliance = ui.Number(decimals=3, suffix="uA") self.lcr_frequency = ui.Number(value=1, minimum=0.020, maximum=20e3, decimals=3, suffix="kHz") self.lcr_amplitude = ui.Number(minimum=0, decimals=3, suffix="mV") self.bind("bias_voltage_start", self.voltage_start, 0, unit="V") self.bind("bias_voltage_stop", self.voltage_stop, 100, unit="V") self.bind("bias_voltage_step", self.voltage_step, 1, unit="V") self.bind("waiting_time", self.waiting_time, 1, unit="s") self.bind("hvsrc_current_compliance", self.hvsrc_current_compliance, 0, unit="uA") self.bind("lcr_frequency", self.lcr_frequency, 1.0, unit="kHz") self.bind("lcr_amplitude", self.lcr_amplitude, 250, unit="mV") self.general_tab.layout = ui.Row( ui.GroupBox(title="HV Source Ramp", layout=ui.Column(ui.Label(text="Start"), self.voltage_start, ui.Label(text="Stop"), self.voltage_stop, ui.Label(text="Step"), self.voltage_step, ui.Label(text="Waiting Time"), self.waiting_time, ui.Spacer())), ui.GroupBox(title="HV Source", layout=ui.Column(ui.Label(text="Compliance"), self.hvsrc_current_compliance, ui.Spacer())), ui.GroupBox(title="LCR", layout=ui.Column(ui.Label(text="AC Frequency"), self.lcr_frequency, ui.Label(text="AC Amplitude"), self.lcr_amplitude, ui.Spacer())), stretch=(1, 1, 1)) fahrad = comet.ureg('F') volt = comet.ureg('V') self.series_transform['lcr'] = lambda x, y: ((x * volt).to('V').m, (y * fahrad).to('pF').m)
def register_vsource(self): self.hvsrc_sense_mode = ui.ComboBox(["local", "remote"]) self.hvsrc_route_terminal = ui.ComboBox(["front", "rear"]) def toggle_hvsrc_filter(enabled): self.hvsrc_filter_count.enabled = enabled self.hvsrc_filter_count_label.enabled = enabled self.hvsrc_filter_type.enabled = enabled self.hvsrc_filter_type_label.enabled = enabled self.hvsrc_filter_enable = ui.CheckBox(text="Enable", changed=toggle_hvsrc_filter) self.hvsrc_filter_count = ui.Number(minimum=0, maximum=100, decimals=0) self.hvsrc_filter_count_label = ui.Label(text="Count") self.hvsrc_filter_type = ui.ComboBox(["repeat", "moving"]) self.hvsrc_filter_type_label = ui.Label(text="Type") def toggle_hvsrc_source_voltage_autorange(enabled): self.hvsrc_source_voltage_range.enabled = not enabled self.hvsrc_source_voltage_autorange_enable = ui.CheckBox( text="Autorange", changed=toggle_hvsrc_source_voltage_autorange) self.hvsrc_source_voltage_range = ui.Number(minimum=-1100, maximum=1100, decimals=1, suffix="V") toggle_hvsrc_filter(False) toggle_hvsrc_source_voltage_autorange(False) self.bind("hvsrc_sense_mode", self.hvsrc_sense_mode, "local") self.bind("hvsrc_route_terminal", self.hvsrc_route_terminal, "rear") self.bind("hvsrc_filter_enable", self.hvsrc_filter_enable, False) self.bind("hvsrc_filter_count", self.hvsrc_filter_count, 10) self.bind("hvsrc_filter_type", self.hvsrc_filter_type, "repeat") self.bind("hvsrc_source_voltage_autorange_enable", self.hvsrc_source_voltage_autorange_enable, True) self.bind("hvsrc_source_voltage_range", self.hvsrc_source_voltage_range, 20, unit="V") self.status_hvsrc_voltage = ui.Text(value=NO_VALUE, readonly=True) self.status_hvsrc_current = ui.Text(value=NO_VALUE, readonly=True) self.status_hvsrc_output = ui.Text(value=NO_VALUE, readonly=True) self.bind("status_hvsrc_voltage", self.status_hvsrc_voltage, NO_VALUE) self.bind("status_hvsrc_current", self.status_hvsrc_current, NO_VALUE) self.bind("status_hvsrc_output", self.status_hvsrc_output, NO_VALUE) self.status_panel.append( ui.GroupBox(title="HV Source Status", layout=ui.Column( ui.Row( ui.Column(ui.Label("Voltage"), self.status_hvsrc_voltage), ui.Column(ui.Label("Current"), self.status_hvsrc_current), ui.Column(ui.Label("Output"), self.status_hvsrc_output))))) self.control_tabs.append( ui.Tab( title="HV Source", layout=ui.Row( ui.GroupBox(title="Filter", layout=ui.Column(self.hvsrc_filter_enable, self.hvsrc_filter_count_label, self.hvsrc_filter_count, self.hvsrc_filter_type_label, self.hvsrc_filter_type, ui.Spacer())), ui.GroupBox(title="Source Voltage Range", layout=ui.Column( self.hvsrc_source_voltage_autorange_enable, ui.Label(text="Range"), self.hvsrc_source_voltage_range, ui.Spacer())), ui.GroupBox(title="Options", layout=ui.Column( ui.Label(text="Sense Mode"), self.hvsrc_sense_mode, ui.Label(text="Route Terminal"), self.hvsrc_route_terminal, ui.Spacer())), stretch=(1, 1, 1)))) def handler(state): if 'hvsrc_voltage' in state: value = state.get('hvsrc_voltage') self.status_hvsrc_voltage.value = format_metric(value, "V") if 'hvsrc_current' in state: value = state.get('hvsrc_current') self.status_hvsrc_current.value = format_metric(value, "A") if 'hvsrc_output' in state: value = state.get('hvsrc_output') self.status_hvsrc_output.value = format_switch( value, default=NO_VALUE) self.state_handlers.append(handler)
def register_electrometer(self): def toggle_elm_filter(enabled): self.elm_filter_count.enabled = enabled self.elm_filter_count_label.enabled = enabled self.elm_filter_type.enabled = enabled self.elm_filter_type_label.enabled = enabled self.elm_filter_enable = ui.CheckBox(text="Enable", changed=toggle_elm_filter) self.elm_filter_count = ui.Number(minimum=0, maximum=100, decimals=0) self.elm_filter_count_label = ui.Label(text="Count") self.elm_filter_type = ui.ComboBox(["repeat", "moving"]) self.elm_filter_type_label = ui.Label(text="Type") self.elm_zero_correction = ui.CheckBox(text="Zero Correction") self.elm_integration_rate = ui.Number(minimum=0, maximum=100.0, decimals=2, suffix="Hz") def toggle_elm_current_autorange(enabled): self.elm_current_range.enabled = not enabled self.elm_current_autorange_minimum.enabled = enabled self.elm_current_autorange_maximum.enabled = enabled self.elm_current_range = ui.Metric(minimum=0, decimals=3, prefixes='munp', unit="A") self.elm_current_autorange_enable = ui.CheckBox( text="Enable", changed=toggle_elm_current_autorange) self.elm_current_autorange_minimum = ui.Metric(minimum=0, decimals=3, prefixes='munp', unit="A") self.elm_current_autorange_maximum = ui.Metric(minimum=0, decimals=3, prefixes='munp', unit="A") toggle_elm_filter(False) toggle_elm_current_autorange(False) self.bind("elm_filter_enable", self.elm_filter_enable, False) self.bind("elm_filter_count", self.elm_filter_count, 10) self.bind("elm_filter_type", self.elm_filter_type, "repeat") self.bind("elm_zero_correction", self.elm_zero_correction, False) self.bind("elm_integration_rate", self.elm_integration_rate, 50.0) self.bind("elm_current_range", self.elm_current_range, 20e-12, unit="A") self.bind("elm_current_autorange_enable", self.elm_current_autorange_enable, False) self.bind("elm_current_autorange_minimum", self.elm_current_autorange_minimum, 2.0E-11, unit="A") self.bind("elm_current_autorange_maximum", self.elm_current_autorange_maximum, 2.0E-2, unit="A") self.status_elm_current = ui.Text(value=NO_VALUE, readonly=True) self.bind("status_elm_current", self.status_elm_current, NO_VALUE) self.status_panel.append( ui.GroupBox(title="Electrometer Status", layout=ui.Column( ui.Row(ui.Column(ui.Label("Current"), self.status_elm_current), ui.Spacer(), stretch=(1, 2))))) self.control_tabs.append( ui.Tab( title="Electrometer", layout=ui.Row( ui.GroupBox(title="Filter", layout=ui.Column(self.elm_filter_enable, self.elm_filter_count_label, self.elm_filter_count, self.elm_filter_type_label, self.elm_filter_type, ui.Spacer())), ui.Column( ui.GroupBox(title="Range", layout=ui.Column( ui.Label(text="Current Range"), self.elm_current_range, )), ui.GroupBox(title="Auto Range", layout=ui.Column( self.elm_current_autorange_enable, ui.Label(text="Minimum Current"), self.elm_current_autorange_minimum, ui.Label(text="Maximum Current"), self.elm_current_autorange_maximum, ui.Spacer()))), ui.GroupBox(title="Options", layout=ui.Column( self.elm_zero_correction, ui.Label(text="Integration Rate"), self.elm_integration_rate, ui.Spacer())), ui.Spacer(), stretch=(1, 1, 1)))) def handler(state): if 'elm_current' in state: value = state.get('elm_current') self.status_elm_current.value = format_metric(value, "A") self.state_handlers.append(handler)
from comet import ui app = comet.Application("comet-example") # Register resources app.resources.add("INSTR", comet.Resource("ASRL2::INSTR", visa_library="@sim")) # Load stored settings (optional) app.resources.load_settings() def on_click(): """Read IDN from IEC60488 compatible resource.""" try: with app.resources.get("INSTR") as instr: idn_text.value = comet.IEC60488(instr).identification except comet.ResourceError as exc: ui.show_exception(exc) idn_text = ui.Text(readonly=True) app.layout = ui.Column( ui.Row( idn_text, ui.Button("Reload", clicked=on_click) ), ui.Spacer(), stretch=(0, 1) ) app.run()
def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) self.title = "Frequency Scan" self.register_vsource() self.register_lcr() self.register_environment() self.plot = ui.Plot(height=300, legend="right") self.plot.add_axis("x", align="bottom", text="Voltage [V] (abs)") self.plot.add_axis("y", align="right", text="Capacitance [pF]") self.plot.add_series("lcr", "x", "y", text="LCR", color="blue") self.data_tabs.insert(0, ui.Tab(title="CV Curve", layout=self.plot)) self.bias_voltage = ui.Number(decimals=3, suffix="V") self.hvsrc_current_compliance = ui.Number(decimals=3, suffix="uA") self.lcr_frequency_start = ui.Number(minimum=0, decimals=3, suffix="Hz") self.lcr_frequency_stop = ui.Number(minimum=0, decimals=3, suffix="MHz") self.lcr_frequency_steps = ui.Number(minimum=1, maximum=1000, decimals=0) self.lcr_amplitude = ui.Number(minimum=0, decimals=3, suffix="mV") self.bind("bias_voltage", self.bias_voltage, 0, unit="V") self.bind("hvsrc_current_compliance", self.hvsrc_current_compliance, 0, unit="uA") self.bind("lcr_frequency_start", self.lcr_frequency_start, 0, unit="Hz") self.bind("lcr_frequency_stop", self.lcr_frequency_stop, 0, unit="MHz") self.bind("lcr_frequency_steps", self.lcr_frequency_steps, 1) self.bind("lcr_amplitude", self.lcr_amplitude, 0, unit="mV") self.general_tab.layout = ui.Row( ui.GroupBox(title="HV Source", layout=ui.Column(ui.Label(text="Bias Voltage"), self.bias_voltage, ui.Label(text="Current Compliance"), self.hvsrc_current_compliance, ui.Spacer())), ui.GroupBox(title="LCR", layout=ui.Column( ui.Label(text="AC Frequency Start"), self.lcr_frequency_start, ui.Label(text="AC Frequency Stop"), self.lcr_frequency_stop, ui.Label(text="AC Frequency Steps (log10)"), self.lcr_frequency_steps, ui.Label(text="AC Amplitude"), self.lcr_amplitude, ui.Spacer())), ui.Spacer(), stretch=(1, 1, 1)) fahrad = comet.ureg('F') volt = comet.ureg('V') self.series_transform['lcr'] = lambda x, y: ((x * volt).to('V').m, (y * fahrad).to('pF').m) self.series_transform['xfit'] = self.series_transform.get('lcr')
def __init__(self, process, lcr_process): super().__init__() self.lcr_process = lcr_process self.lcr_process.finished = self.on_lcr_finished self.lcr_process.failed = self.on_lcr_failed self.lcr_process.reading = self.on_lcr_reading self.mount(process) self.resize(640, 480) self.title = "Table Control" self.add_x_button = KeypadButton(text="+X", clicked=self.on_add_x) self.sub_x_button = KeypadButton(text="-X", clicked=self.on_sub_x) self.add_y_button = KeypadButton(text="+Y", clicked=self.on_add_y) self.sub_y_button = KeypadButton(text="-Y", clicked=self.on_sub_y) self.add_z_button = KeypadButton(text="+Z", clicked=self.on_add_z) self.sub_z_button = KeypadButton(text="-Z", clicked=self.on_sub_z) self.step_up_button = KeypadButton( text="↑⇵", tool_tip= "Step up, move single step up then double step down and double step up (experimental).", clicked=self.on_step_up) self.control_buttons = (self.add_x_button, self.sub_x_button, self.add_y_button, self.sub_y_button, self.add_z_button, self.sub_z_button, self.step_up_button) self._lcr_prim_text = ui.Text(readonly=True) self._lcr_sec_text = ui.Text(readonly=True) self._lcr_chart = LCRChart() self._lcr_groupbox = ui.GroupBox( title="Contact Quality (LCR)", checkable=True, checked=False, toggled=self.on_lcr_toggled, layout=ui.Column(ui.Row(ui.Label("Cp"), self._lcr_prim_text), ui.Row(ui.Label("Rp"), self._lcr_sec_text), self._lcr_chart)) self.probecard_light_button = ToggleButton( text="PC Light", tool_tip="Toggle probe card light", checkable=True, checked=False, enabled=False, toggled=self.on_probecard_light_clicked) self.microscope_light_button = ToggleButton( text="Mic Light", tool_tip="Toggle microscope light", checkable=True, checked=False, enabled=False, toggled=self.on_microscope_light_clicked) self.box_light_button = ToggleButton(text="Box Light", tool_tip="Toggle box light", checkable=True, checked=False, enabled=False, toggled=self.on_box_light_clicked) # Create movement radio buttons self.step_width_buttons = ui.Column() for item in self.load_table_step_sizes(): step_size = item.get('step_size') * comet.ureg('um') step_color = item.get('step_color') step_size_label = format_metric(step_size.to('m').m, 'm', decimals=1) button = ui.RadioButton( text=step_size_label, tool_tip=f"Move in {step_size_label} steps.", stylesheet=f"QRadioButton:enabled{{color:{step_color};}}", checked=len(self.step_width_buttons) == 0, toggled=self.on_step_toggled) button.movement_width = step_size.to('mm').m button.movement_color = step_color self.step_width_buttons.append(button) self.control_layout = ui.Column(ui.Row(ui.Row( ui.Column(KeypadSpacer(), self.sub_y_button, KeypadSpacer()), ui.Column( self.sub_x_button, KeypadSpacer(), self.add_x_button, ), ui.Column( KeypadSpacer(), self.add_y_button, KeypadSpacer(), ), KeypadSpacer(), ui.Column(ui.Row(self.add_z_button, self.step_up_button), KeypadSpacer(), self.sub_z_button)), ui.Spacer(), stretch=(0, 1)), ui.Spacer(), stretch=(0, 1)) self.positions_widget = TablePositionsWidget( enabled=False, position_picked=self.on_position_picked, absolute_move=self.on_absolute_move) self.contacts_widget = TableContactsWidget( enabled=False, position_picked=self.on_position_picked, absolute_move=self.on_absolute_move) self._position_widget = PositionWidget() self._calibration_widget = CalibrationWidget() self.z_limit_label = PositionLabel() self.x_hard_limit_label = PositionLabel() self.y_hard_limit_label = PositionLabel() self.z_hard_limit_label = PositionLabel() self.laser_label = SwitchLabel() self._calibrate_button = ui.Button(text="Calibrate", clicked=self.on_calibrate) self._update_interval_number = ui.Number( value=settings.table_control_update_interval, minimum=.5, maximum=10.0, decimals=2, step=0.25, suffix="s", changed=self.on_update_interval_changed) self._interval_groupbox = ui.GroupBox(title="Update Interval", layout=ui.Row( self._update_interval_number, ui.Spacer(), stretch=(0, 1))) self._dodge_height_number = ui.Number( tool_tip="Doge height in microns.", minimum=0, maximum=10000, decimals=0, step=1, suffix="um") self._dodge_groupbox = ui.GroupBox( title="X/Y Dodge", tool_tip="Enables -/+ Z dodge for XY movements.", checkable=True, layout=ui.Row(ui.Label("Height"), self._dodge_height_number, ui.Spacer(), stretch=(0, 0, 1))) self._lcr_reset_on_move_checkbox = ui.CheckBox( text="Reset graph on X/Y move") self._contact_quality_groupbox = ui.GroupBox( title="Contact Quality (LCR)", layout=ui.Row(self._lcr_reset_on_move_checkbox, ui.Spacer(), stretch=(0, 1))) self._step_up_delay_number = ui.Number(minimum=0, maximum=1000, decimals=0, step=25, suffix="ms") self._step_up_multiply_number = ui.Number(minimum=1, maximum=10, decimals=0, suffix="x") self._step_up_groubox = ui.GroupBox( title="Step Up (↑⇵)", layout=ui.Row( ui.Column(ui.Label("Delay"), ui.Label("Multiplicator (⇵)")), ui.Column(self._step_up_delay_number, self._step_up_multiply_number), ui.Spacer())) self._lcr_update_interval_number = ui.Number(minimum=0, maximum=1000, decimals=0, step=25, suffix="ms") self._lcr_matrix_channels_text = ui.Text() self._lcr_options_groupbox = ui.GroupBox( title="Contact Quality (LCR)", layout=ui.Row( ui.Column( ui.Label("Reading Interval"), ui.Label("Matrix Channels"), ), ui.Column( ui.Row(self._lcr_update_interval_number, ui.Spacer()), self._lcr_matrix_channels_text))) self.progress_bar = ui.ProgressBar(visible=False) self.message_label = ui.Label() self.stop_button = ui.Button(text="&Stop", default=False, auto_default=False, enabled=False, clicked=self.on_stop) self.close_button = ui.Button(text="&Close", default=False, auto_default=False, clicked=self.on_close) self.layout = ui.Column( ui.Row( ui.Column( ui.Row( ui.GroupBox( title="Control", layout=ui.Column( ui.Spacer(horizontal=False), self.control_layout, ui.Spacer(horizontal=False) ) ), ui.GroupBox( title="Step Width", layout=self.step_width_buttons ), self._lcr_groupbox, ui.Column( ui.GroupBox( title="Lights", layout=ui.Column( self.probecard_light_button, self.microscope_light_button, self.box_light_button, ui.Spacer() ) ) ), stretch=(0, 0, 1, 0) ), ui.Tabs( ui.Tab( title="Move", layout=self.positions_widget ), ui.Tab( title="Contacts", layout=self.contacts_widget ), ui.Tab( title="Calibrate", layout=ui.Column( ui.GroupBox( title="Table Calibration", layout=ui.Column( ui.Row( self._calibrate_button, ui.Label("Calibrate table by moving into cal/rm switches\nof every axis in" \ " a safe manner to protect the probe card."), stretch=(0, 1) ) ) ), ui.Spacer(), stretch=(0, 1) ) ), ui.Tab( title="Options", layout=ui.Column( ui.Row( self._interval_groupbox, self._dodge_groupbox, self._contact_quality_groupbox, stretch=(0, 0, 1) ), self._step_up_groubox, self._lcr_options_groupbox, ui.Spacer(), stretch=(0, 0, 0, 1) ) ) ), stretch=(0, 1) ), ui.Column( self._position_widget, self._calibration_widget, ui.GroupBox( title="Soft Limits", layout=ui.Row( ui.Column( ui.Label("Z") ), ui.Column( self.z_limit_label ) ) ), ui.GroupBox( title="Hard Limits", layout=ui.Row( ui.Column( ui.Label("X"), ui.Label("Y"), ui.Label("Z") ), ui.Column( self.x_hard_limit_label, self.y_hard_limit_label, self.z_hard_limit_label ) ) ), ui.GroupBox( title="Safety", layout=ui.Row( ui.Label( text="Laser Sensor" ), self.laser_label ) ), ui.Spacer() ), stretch=(1, 0) ), ui.Row( self.progress_bar, self.message_label, ui.Spacer(), self.stop_button, self.close_button ), stretch=(1, 0) ) self.close_event = self.on_close_event self.reset_position() self.reset_caldone() self.update_limits() self.reset_safety() self.update_control_buttons()
def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) self.title = "IV Ramp Elm" self.register_vsource() self.register_electrometer() self.register_environment() self.plot = ui.Plot(height=300, legend="right") self.plot.add_axis("x", align="bottom", text="Voltage [V] (abs)") self.plot.add_axis("y", align="right", text="Current [uA]") self.plot.add_series("hvsrc", "x", "y", text="HV Source", color="red") self.plot.add_series("elm", "x", "y", text="Electrometer", color="blue") self.plot.add_series("xfit", "x", "y", text="Fit", color="magenta") self.data_tabs.insert(0, ui.Tab(title="IV Curve", layout=self.plot)) self.voltage_start = ui.Number(decimals=3, suffix="V") self.voltage_stop = ui.Number(decimals=3, suffix="V") self.voltage_step = ui.Number(minimum=0, maximum=200, decimals=3, suffix="V") self.waiting_time = ui.Number(minimum=0, decimals=2, suffix="s") self.hvsrc_current_compliance = ui.Metric(minimum=0, decimals=3, prefixes='mun', unit="A") self.hvsrc_accept_compliance = ui.CheckBox("Accept Compliance") self.bind("voltage_start", self.voltage_start, 0, unit="V") self.bind("voltage_stop", self.voltage_stop, 100, unit="V") self.bind("voltage_step", self.voltage_step, 1, unit="V") self.bind("waiting_time", self.waiting_time, 1, unit="s") self.bind("hvsrc_current_compliance", self.hvsrc_current_compliance, 0, unit="A") self.bind("hvsrc_accept_compliance", self.hvsrc_accept_compliance, False) # Instruments status self.general_tab.layout = ui.Row( ui.GroupBox(title="HV Source Ramp", layout=ui.Column(ui.Label(text="Start"), self.voltage_start, ui.Label(text="Stop"), self.voltage_stop, ui.Label(text="Step"), self.voltage_step, ui.Label(text="Waiting Time"), self.waiting_time, ui.Spacer())), ui.GroupBox(title="HV Source", layout=ui.Column(ui.Label(text="Compliance"), self.hvsrc_current_compliance, self.hvsrc_accept_compliance, ui.Spacer())), ui.Spacer(), stretch=(1, 1, 1)) ampere = comet.ureg('A') volt = comet.ureg('V') self.series_transform['elm'] = lambda x, y: ((x * volt).to('V').m, (y * ampere).to('uA').m) self.series_transform['hvsrc'] = self.series_transform.get('elm') self.series_transform['xfit'] = self.series_transform.get('elm')
def __init__(self, *, tree_selected, tree_double_clicked, start_all, start, stop, reset_sequence_state, edit_sequence): super().__init__() self.current_path = user_home() self.title = "Sequence" self.tree_double_clicked = tree_double_clicked self._sequence_tree = SequenceTree( selected=tree_selected, double_clicked=self.tree_double_clicked) self._sequence_tree.minimum_width = 360 self._start_all_action = ui.Action(text="&All Samples", triggered=start_all) self._start_sample_action = ui.Action(text="&Sample", triggered=start) self._start_sample_action.qt.setEnabled(False) self._start_contact_action = ui.Action(text="&Contact", triggered=start) self._start_contact_action.qt.setEnabled(False) self._start_measurement_action = ui.Action(text="&Measurement", triggered=start) self._start_measurement_action.qt.setEnabled(False) self._start_menu = ui.Menu() self._start_menu.append(self._start_all_action) self._start_menu.append(self._start_sample_action) self._start_menu.append(self._start_contact_action) self._start_menu.append(self._start_measurement_action) self._start_button = ui.Button( text="Start", tool_tip="Start measurement sequence.", stylesheet="QPushButton:enabled{color:green;font-weight:bold;}") self._start_button.qt.setMenu(self._start_menu.qt) self._stop_button = ui.Button( text="Stop", tool_tip="Stop measurement sequence.", enabled=False, clicked=stop, stylesheet="QPushButton:enabled{color:red;font-weight:bold;}") self._reset_button = ui.Button( text="Reset", tool_tip="Reset measurement sequence state.", clicked=reset_sequence_state) self._edit_button = ui.Button( text="Edit", tool_tip="Quick edit properties of sequence items.", clicked=edit_sequence) self._reload_config_button = ui.ToolButton( icon=make_path('assets', 'icons', 'reload.svg'), tool_tip="Reload sequence configurations from file.", clicked=self.on_reload_config_clicked) self._add_sample_button = ui.ToolButton( icon=make_path('assets', 'icons', 'add.svg'), tool_tip="Add new sample sequence.", clicked=self.on_add_sample_clicked) self._remove_sample_button = ui.ToolButton( icon=make_path('assets', 'icons', 'delete.svg'), tool_tip="Remove current sample sequence.", clicked=self.on_remove_sample_clicked) self._open_button = ui.ToolButton( icon=make_path('assets', 'icons', 'document_open.svg'), tool_tip="Open sequence tree from file.", clicked=self.on_open_clicked) self._save_button = ui.ToolButton( icon=make_path('assets', 'icons', 'document_save.svg'), tool_tip="Save sequence tree to file.", clicked=self.on_save_clicked) self.layout = ui.Column( self._sequence_tree, ui.Row(self._start_button, self._stop_button, self._reset_button, self._edit_button, self._reload_config_button, self._add_sample_button, self._remove_sample_button, self._open_button, self._save_button))
def __init__(self, message_changed=None, progress_changed=None, **kwargs): super().__init__() # Callbacks self.message_changed = message_changed self.progress_changed = progress_changed # Layout self.temporary_z_limit_label = ui.Label( text="Temporary Probecard Z-Limit applied. " "Revert after finishing current measurements.", stylesheet= "QLabel{color: black; background-color: yellow; padding: 4px; border-radius: 4px;}", visible=False) self.sequence_widget = SequenceWidget( tree_selected=self.on_tree_selected, tree_double_clicked=self.on_tree_double_clicked, start_all=self.on_start_all, start=self.on_start, stop=self.on_stop, reset_sequence_state=self.on_reset_sequence_state, edit_sequence=self.on_edit_sequence) self.sequence_tree = self.sequence_widget._sequence_tree self.start_sample_action = self.sequence_widget._start_sample_action self.start_contact_action = self.sequence_widget._start_contact_action self.start_measurement_action = self.sequence_widget._start_measurement_action # Environment Controls self.environment_control_widget = EnvironmentControlWidget( toggled=self.on_environment_groupbox_toggled, laser_sensor_toggled=self.on_laser_sensor_toggled, box_light_toggled=self.on_box_light_toggled, microscope_light_toggled=self.on_microscope_light_toggled, microscope_camera_toggled=self.on_microscope_camera_toggled, microscope_control_toggled=self.on_microscope_control_toggled, probecard_light_toggled=self.on_probecard_light_toggled, probecard_camera_toggled=self.on_probecard_camera_toggled, pid_control_toggled=self.on_pid_control_toggled) # Table controls self.table_control_widget = TableControlWidget( toggled=self.on_table_groupbox_toggled, joystick_toggled=self.on_table_joystick_toggled, control_clicked=self.on_table_control_clicked) # Operator self.operator_widget = OperatorWidget() self.operator_widget.load_settings() self.operator_groupbox = ui.GroupBox(title="Operator", layout=self.operator_widget) # Working directory self.output_widget = WorkingDirectoryWidget() self.output_groupbox = ui.GroupBox(title="Working Directory", layout=self.output_widget) # Controls self.control_widget = ui.Column(self.sequence_widget, self.table_control_widget, self.environment_control_widget, ui.Row(self.operator_groupbox, self.output_groupbox, stretch=(3, 7)), stretch=(1, 0, 0)) # Tabs self.measurement_tab = MeasurementTab(restore=self.on_measure_restore) self.environment_tab = EnvironmentTab() self.status_tab = StatusTab(reload=self.on_status_start) self.summary_tab = SummaryTab() self.panels = self.measurement_tab.panels self.panels.sample_changed = self.on_sample_changed self.log_widget = LogWidget() self.log_widget.add_logger(logging.getLogger()) self.logging_tab = ui.Tab(title="Logs", layout=self.log_widget) # Tabs self.tab_widget = ui.Tabs(self.measurement_tab, self.environment_tab, self.status_tab, self.logging_tab, self.summary_tab) # Layout self.splitter = ui.Splitter() self.splitter.append(self.control_widget) self.splitter.append(self.tab_widget) self.splitter.stretch = 4, 9 self.splitter.collapsible = False self.append(self.temporary_z_limit_label) self.append(self.splitter) self.stretch = 0, 1 # Setup process callbacks self.environ_process = self.processes.get("environ") self.environ_process.pc_data_updated = self.on_pc_data_updated self.status_process = self.processes.get("status") self.status_process.finished = self.on_status_finished self.table_process = self.processes.get("table") self.table_process.joystick_changed = self.on_table_joystick_changed self.table_process.position_changed = self.on_table_position_changed self.table_process.caldone_changed = self.on_table_calibration_changed self.measure_process = self.processes.get("measure") self.measure_process.finished = self.on_finished self.measure_process.measurement_state = self.on_measurement_state self.measure_process.measurement_reset = self.on_measurement_reset self.measure_process.save_to_image = self.on_save_to_image self.contact_quality_process = self.processes.get("contact_quality") # Experimental # Install timer to update environment controls self.environment_timer = Timer(timeout=self.sync_environment_controls) self.environment_timer.start(self.environment_poll_interval) self.close_event = self.on_stop
def __init__(self, laser_sensor_toggled=None, box_light_toggled=None, microscope_light_toggled=None, microscope_camera_toggled=None, microscope_control_toggled=None, probecard_light_toggled=None, probecard_camera_toggled=None, pid_control_toggled=None, **kwargs): super().__init__(**kwargs) self.title = "Environment Box" self.checkable = True self._laser_sensor_button = ToggleButton( text="Laser", tool_tip="Toggle laser", checkable=True, checked=False, toggled=self.on_laser_sensor_toggled) self._box_light_button = ToggleButton( text="Box Light", tool_tip="Toggle box light", checkable=True, checked=False, toggled=self.on_box_light_toggled) self._microscope_light_button = ToggleButton( text="Mic Light", tool_tip="Toggle microscope light", checkable=True, checked=False, toggled=self.on_microscope_light_toggled) self._microscope_camera_button = ToggleButton( text="Mic Cam", tool_tip="Toggle microscope camera power", checkable=True, checked=False, toggled=self.on_microscope_camera_toggled) self._microscope_control_button = ToggleButton( text="Mic Ctrl", tool_tip="Toggle microscope control", checkable=True, checked=False, toggled=self.on_microscope_control_toggled) self._probecard_light_button = ToggleButton( text="PC Light", tool_tip="Toggle probe card light", checkable=True, checked=False, toggled=self.on_probecard_light_toggled) self._probecard_camera_button = ToggleButton( text="PC Cam", tool_tip="Toggle probe card camera power", checkable=True, checked=False, toggled=self.on_probecard_camera_toggled) self._pid_control_button = ToggleButton( text="PID Control", tool_tip="Toggle PID control", checkable=True, checked=False, toggled=self.on_pid_control_toggled) self.layout = ui.Row(ui.Column( self._laser_sensor_button, self._microscope_camera_button, ), ui.Column( self._box_light_button, self._probecard_camera_button, ), ui.Column( self._microscope_light_button, self._microscope_control_button, ), ui.Column(self._probecard_light_button, self._pid_control_button), stretch=(1, 1, 1, 1)) # Callbacks self.laser_sensor_toggled = laser_sensor_toggled self.box_light_toggled = box_light_toggled self.microscope_light_toggled = microscope_light_toggled self.microscope_camera_toggled = microscope_camera_toggled self.microscope_control_toggled = microscope_control_toggled self.probecard_light_toggled = probecard_light_toggled self.probecard_camera_toggled = probecard_camera_toggled self.pid_control_toggled = pid_control_toggled