Wed, 20 Jan 2021 11:37:03 +0100
reimplemented lasercutter changes
# FILE MODIFIED BY NEOSOFT - MALTE DI DONATO # Embed Lasercut controls # This file is part of the Printrun suite. # # Printrun is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # Printrun is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with Printrun. If not, see <http://www.gnu.org/licenses/>. import wx from .xybuttons import XYButtons, XYButtonsMini from .zbuttons import ZButtons, ZButtonsMini from .graph import Graph from .widgets import TempGauge from wx.lib.agw.floatspin import FloatSpin from .utils import make_button, make_custom_button class XYZControlsSizer(wx.GridBagSizer): def __init__(self, root, parentpanel = None): super(XYZControlsSizer, self).__init__() if not parentpanel: parentpanel = root.panel root.xyb = XYButtons(parentpanel, root.moveXY, root.homeButtonClicked, root.spacebarAction, root.bgcolor, zcallback=root.moveZ) root.xyb.SetToolTip(_('[J]og controls. (Shift)+TAB ESC Shift/Ctrl+(arrows PgUp/PgDn)')) self.Add(root.xyb, pos = (0, 1), flag = wx.ALIGN_CENTER) root.zb = ZButtons(parentpanel, root.moveZ, root.bgcolor) self.Add(root.zb, pos = (0, 2), flag = wx.ALIGN_CENTER) def add_extra_controls(self, root, parentpanel, extra_buttons = None, mini_mode = False): standalone_mode = extra_buttons is not None base_line = 1 if standalone_mode else 2 if standalone_mode: gauges_base_line = base_line + 10 elif mini_mode and root.display_graph: gauges_base_line = base_line + 7 else: gauges_base_line = base_line + 6 tempdisp_line = gauges_base_line + (2 if root.display_gauges else 0) if mini_mode and root.display_graph: e_base_line = base_line + 3 else: e_base_line = base_line + 2 lasercut_base_line = 11 pos_mapping = { "htemp_label": (base_line + 0, 0), "htemp_off": (base_line + 0, 2), "htemp_val": (base_line + 0, 3), "htemp_set": (base_line + 0, 4), "btemp_label": (base_line + 1, 0), "btemp_off": (base_line + 1, 2), "btemp_val": (base_line + 1, 3), "btemp_set": (base_line + 1, 4), "ebuttons": (e_base_line + 0, 0), "esettings": (e_base_line + 1, 0), "speedcontrol": (e_base_line + 2, 0), "flowcontrol": (e_base_line + 3, 0), "htemp_gauge": (gauges_base_line + 0, 0), "btemp_gauge": (gauges_base_line + 1, 0), "tempdisp": (tempdisp_line, 0), "extrude": (3, 0), "reverse": (3, 2), "lasercut_optionsbtn": (lasercut_base_line, 0), "lasercut_printbtn": (lasercut_base_line, 2), "lasercut_material_thickness": (lasercut_base_line+1, 4), "lasercut_material_thickness_label": (lasercut_base_line+1, 0), "lasercut_pass_count": (lasercut_base_line+2, 4), "lasercut_pass_count_label": (lasercut_base_line+2, 0), "lasercut_pass_zdiff": (lasercut_base_line+3, 4), "lasercut_pass_zdiff_label": (lasercut_base_line+3, 0), } span_mapping = { "htemp_label": (1, 2), "htemp_off": (1, 1), "htemp_val": (1, 1), "htemp_set": (1, 1 if root.display_graph else 2), "btemp_label": (1, 2), "btemp_off": (1, 1), "btemp_val": (1, 1), "btemp_set": (1, 1 if root.display_graph else 2), "ebuttons": (1, 5 if root.display_graph else 6), "esettings": (1, 5 if root.display_graph else 6), "speedcontrol": (1, 5 if root.display_graph else 6), "flowcontrol": (1, 5 if root.display_graph else 6), "htemp_gauge": (1, 5 if mini_mode else 6), "btemp_gauge": (1, 5 if mini_mode else 6), "tempdisp": (1, 5 if mini_mode else 6), "extrude": (1, 2), "reverse": (1, 3), } if standalone_mode: pos_mapping["tempgraph"] = (base_line + 6, 0) span_mapping["tempgraph"] = (3, 2) elif mini_mode: pos_mapping["tempgraph"] = (base_line + 2, 0) span_mapping["tempgraph"] = (1, 5) else: pos_mapping["tempgraph"] = (base_line + 0, 5) span_mapping["tempgraph"] = (5, 1) if mini_mode: pos_mapping["etool_label"] = (0, 0) pos_mapping["etool_val"] = (0, 1) pos_mapping["edist_label"] = (0, 2) pos_mapping["edist_val"] = (0, 3) pos_mapping["edist_unit"] = (0, 4) else: pos_mapping["edist_label"] = (0, 0) pos_mapping["edist_val"] = (1, 0) pos_mapping["edist_unit"] = (1, 1) pos_mapping["efeed_label"] = (0, 2) pos_mapping["efeed_val"] = (1, 2) pos_mapping["efeed_unit"] = (1, 3) def add(name, widget, *args, **kwargs): kwargs["pos"] = pos_mapping[name] if name in span_mapping: kwargs["span"] = span_mapping[name] if "container" in kwargs: container = kwargs["container"] del kwargs["container"] else: container = self container.Add(widget, *args, **kwargs) # Lasercutter quick controls # root.lc_optionsbtn = make_button(parentpanel, _("Lasercutter options"), lambda e: PronterOptions(root, "Laser"), _("Open Lasercutter options"), style = wx.BU_EXACTFIT) root.printerControls.append(root.lc_optionsbtn) add("lasercut_optionsbtn", root.lc_optionsbtn, flag = wx.EXPAND) root.lc_printbtn = make_button(parentpanel, _("Start cutting"), root.on_lc_printfile, _("Start printjob with lasercutting features"), style = wx.BU_EXACTFIT) root.printerControls.append(root.lc_printbtn) add("lasercut_printbtn", root.lc_printbtn, flag = wx.EXPAND) root.lc_pass_count = speed_spin = FloatSpin(parentpanel, -1, value = root.settings.lc_pass_count, min_val = 1, max_val = 10, digits = 0, style = wx.ALIGN_LEFT, size = (80, -1)) add("lasercut_pass_count", root.lc_pass_count) add("lasercut_pass_count_label", wx.StaticText(parentpanel, -1, _("Number of cutting passes:")), flag = wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_LEFT) root.lc_pass_zdiff = speed_spin = FloatSpin(parentpanel, -1, increment = 0.1, value = root.settings.lc_pass_zdiff, min_val = -2, max_val = 2, digits = 1, style = wx.ALIGN_LEFT, size = (80, -1)) add("lasercut_pass_zdiff", root.lc_pass_zdiff) add("lasercut_pass_zdiff_label", wx.StaticText(parentpanel, -1, _("Z movement after each cut:")), flag = wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_LEFT) root.lc_material_thickness = speed_spin = FloatSpin(parentpanel, -1, increment = 0.1, value = root.settings.lc_material_thickness, min_val = 0, max_val = 75, digits = 1, style = wx.ALIGN_LEFT, size = (80, -1)) add("lasercut_material_thickness", root.lc_material_thickness) add("lasercut_material_thickness_label", wx.StaticText(parentpanel, -1, _("Material Thickness:")), flag = wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_LEFT) # Hotend & bed temperatures # # Hotend temp add("htemp_label", wx.StaticText(parentpanel, -1, _("Heat:")), flag = wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_RIGHT) root.settoff = make_button(parentpanel, _("Off"), lambda e: root.do_settemp("off"), _("Switch Hotend Off"), size = (38, -1), style = wx.BU_EXACTFIT) root.printerControls.append(root.settoff) add("htemp_off", root.settoff) root.htemp = wx.ComboBox(parentpanel, style = wx.CB_DROPDOWN, size = (115, -1)) root.htemp.SetToolTip(wx.ToolTip(_("Select Temperature for [H]otend"))) root.htemp.Bind(wx.EVT_COMBOBOX, root.htemp_change) add("htemp_val", root.htemp) root.settbtn = make_button(parentpanel, _("Set"), root.do_settemp, _("Switch Hotend On"), size = (38, -1), style = wx.BU_EXACTFIT) root.printerControls.append(root.settbtn) add("htemp_set", root.settbtn, flag = wx.EXPAND) # Bed temp add("btemp_label", wx.StaticText(parentpanel, -1, _("Bed:")), flag = wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_RIGHT) root.setboff = make_button(parentpanel, _("Off"), lambda e: root.do_bedtemp("off"), _("Switch Heated Bed Off"), size = (38, -1), style = wx.BU_EXACTFIT) root.printerControls.append(root.setboff) add("btemp_off", root.setboff) root.btemp = wx.ComboBox(parentpanel, style = wx.CB_DROPDOWN, size = (115, -1)) root.btemp.SetToolTip(wx.ToolTip(_("Select Temperature for Heated [B]ed"))) root.btemp.Bind(wx.EVT_COMBOBOX, root.btemp_change) add("btemp_val", root.btemp) root.setbbtn = make_button(parentpanel, _("Set"), root.do_bedtemp, _("Switch Heated Bed On"), size = (38, -1), style = wx.BU_EXACTFIT) root.printerControls.append(root.setbbtn) add("btemp_set", root.setbbtn, flag = wx.EXPAND) def set_labeled(temp, choices, widget): choices = [(float(p[1]), p[0]) for p in choices.items()] if not next((1 for p in choices if p[0] == temp), False): choices.append((temp, 'user')) choices = sorted(choices) widget.Items = ['%s (%s)'%tl for tl in choices] widget.Selection = next((i for i, tl in enumerate(choices) if tl[0] == temp), -1) set_labeled(root.settings.last_bed_temperature, root.bedtemps, root.btemp) set_labeled(root.settings.last_temperature, root.temps, root.htemp) # Speed control # speedpanel = root.newPanel(parentpanel) speedsizer = wx.BoxSizer(wx.HORIZONTAL) speedsizer.Add(wx.StaticText(speedpanel, -1, _("Print speed:")), flag = wx.ALIGN_CENTER_VERTICAL) root.speed_slider = wx.Slider(speedpanel, -1, 100, 1, 300) speedsizer.Add(root.speed_slider, 1, flag = wx.EXPAND) root.speed_spin = wx.SpinCtrlDouble(speedpanel, -1, initial = 100, min = 1, max = 300, style = wx.ALIGN_LEFT, size = (115, -1)) root.speed_spin.SetDigits(0) speedsizer.Add(root.speed_spin, 0, flag = wx.ALIGN_CENTER_VERTICAL) root.speed_label = wx.StaticText(speedpanel, -1, _("%")) speedsizer.Add(root.speed_label, flag = wx.ALIGN_CENTER_VERTICAL) def speedslider_set(event): root.do_setspeed() root.speed_setbtn.SetBackgroundColour(wx.NullColour) root.speed_setbtn = make_button(speedpanel, _("Set"), speedslider_set, _("Set print speed factor"), size = (38, -1), style = wx.BU_EXACTFIT) root.printerControls.append(root.speed_setbtn) speedsizer.Add(root.speed_setbtn, flag = wx.ALIGN_CENTER) speedpanel.SetSizer(speedsizer) add("speedcontrol", speedpanel, flag = wx.EXPAND) def speedslider_spin(event): value = root.speed_spin.GetValue() root.speed_setbtn.SetBackgroundColour("red") root.speed_slider.SetValue(value) root.speed_spin.Bind(wx.EVT_SPINCTRLDOUBLE, speedslider_spin) def speedslider_scroll(event): value = root.speed_slider.GetValue() root.speed_setbtn.SetBackgroundColour("red") root.speed_spin.SetValue(value) root.speed_slider.Bind(wx.EVT_SCROLL, speedslider_scroll) # Flow control # flowpanel = root.newPanel(parentpanel) flowsizer = wx.BoxSizer(wx.HORIZONTAL) flowsizer.Add(wx.StaticText(flowpanel, -1, _("Print flow:")), flag = wx.ALIGN_CENTER_VERTICAL) root.flow_slider = wx.Slider(flowpanel, -1, 100, 1, 300) flowsizer.Add(root.flow_slider, 1, flag = wx.EXPAND) root.flow_spin = wx.SpinCtrlDouble(flowpanel, -1, initial = 100, min = 1, max = 300, style = wx.ALIGN_LEFT, size = (115, -1)) flowsizer.Add(root.flow_spin, 0, flag = wx.ALIGN_CENTER_VERTICAL) root.flow_label = wx.StaticText(flowpanel, -1, _("%")) flowsizer.Add(root.flow_label, flag = wx.ALIGN_CENTER_VERTICAL) def flowslider_set(event): root.do_setflow() root.flow_setbtn.SetBackgroundColour(wx.NullColour) root.flow_setbtn = make_button(flowpanel, _("Set"), flowslider_set, _("Set print flow factor"), size = (38, -1), style = wx.BU_EXACTFIT) root.printerControls.append(root.flow_setbtn) flowsizer.Add(root.flow_setbtn, flag = wx.ALIGN_CENTER) flowpanel.SetSizer(flowsizer) add("flowcontrol", flowpanel, flag = wx.EXPAND) def flowslider_spin(event): value = root.flow_spin.GetValue() root.flow_setbtn.SetBackgroundColour("red") root.flow_slider.SetValue(value) root.flow_spin.Bind(wx.EVT_SPINCTRLDOUBLE, flowslider_spin) def flowslider_scroll(event): value = root.flow_slider.GetValue() root.flow_setbtn.SetBackgroundColour("red") root.flow_spin.SetValue(value) root.flow_slider.Bind(wx.EVT_SCROLL, flowslider_scroll) # Temperature gauges # if root.display_gauges: root.hottgauge = TempGauge(parentpanel, size = (-1, 24), title = _("Heater:"), maxval = 300, bgcolor = root.bgcolor) root.hottgauge.SetTarget(root.settings.last_temperature) # root.hsetpoint = root.settings.last_temperature add("htemp_gauge", root.hottgauge, flag = wx.EXPAND) root.bedtgauge = TempGauge(parentpanel, size = (-1, 24), title = _("Bed:"), maxval = 150, bgcolor = root.bgcolor) root.bedtgauge.SetTarget(root.settings.last_bed_temperature) # root.bsetpoint = root.settings.last_bed_temperature add("btemp_gauge", root.bedtgauge, flag = wx.EXPAND) def scroll_gauge(rot, cmd, setpoint): if rot: temp = setpoint + (1 if rot > 0 else -1) cmd(str(max(0, temp))) def hotend_handler(e): scroll_gauge(e.WheelRotation, root.do_settemp, root.hsetpoint) def bed_handler(e): scroll_gauge(e.WheelRotation, root.do_bedtemp, root.bsetpoint) root.hottgauge.Bind(wx.EVT_MOUSEWHEEL, hotend_handler) root.bedtgauge.Bind(wx.EVT_MOUSEWHEEL, bed_handler) def updateGauge(e, gauge): gauge.SetTarget(float(e.String.split()[0])) root.htemp.Bind(wx.EVT_TEXT, lambda e: updateGauge(e, root.hottgauge)) root.btemp.Bind(wx.EVT_TEXT, lambda e: updateGauge(e, root.bedtgauge)) # Temperature (M105) feedback display # root.tempdisp = wx.StaticText(parentpanel, -1, "", style = wx.ST_NO_AUTORESIZE) def on_tempdisp_size(evt): root.tempdisp.Wrap(root.tempdisp.GetSize().width) root.tempdisp.Bind(wx.EVT_SIZE, on_tempdisp_size) def tempdisp_setlabel(label): wx.StaticText.SetLabel(root.tempdisp, label) root.tempdisp.Wrap(root.tempdisp.GetSize().width) root.tempdisp.SetSize((-1, root.tempdisp.GetBestSize().height)) root.tempdisp.SetLabel = tempdisp_setlabel add("tempdisp", root.tempdisp, flag = wx.EXPAND) # Temperature graph # if root.display_graph: root.graph = Graph(parentpanel, wx.ID_ANY, root) add("tempgraph", root.graph, flag = wx.EXPAND | wx.ALL, border = 5) root.graph.Bind(wx.EVT_LEFT_DOWN, root.graph.show_graph_window) # Extrusion controls # # Extrusion settings esettingspanel = root.newPanel(parentpanel) esettingssizer = wx.GridBagSizer() esettingssizer.SetEmptyCellSize((0, 0)) root.edist = wx.SpinCtrlDouble(esettingspanel, -1, initial = root.settings.last_extrusion, min = 0, max = 1000, size = (135, -1)) root.edist.SetDigits(1) root.edist.Bind(wx.EVT_SPINCTRLDOUBLE, root.setfeeds) root.edist.SetBackgroundColour((225, 200, 200)) root.edist.SetForegroundColour("black") root.edist.Bind(wx.EVT_TEXT, root.setfeeds) add("edist_label", wx.StaticText(esettingspanel, -1, _("Length:")), container = esettingssizer, flag = wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_LEFT | wx.RIGHT | wx.LEFT, border = 5) add("edist_val", root.edist, container = esettingssizer, flag = wx.ALIGN_CENTER | wx.RIGHT, border = 5) unit_label = _("mm") if mini_mode else _("mm @") add("edist_unit", wx.StaticText(esettingspanel, -1, unit_label), container = esettingssizer, flag = wx.ALIGN_CENTER | wx.RIGHT, border = 5) root.edist.SetToolTip(wx.ToolTip(_("Amount to Extrude or Retract (mm)"))) if not mini_mode: root.efeedc = wx.SpinCtrlDouble(esettingspanel, -1, initial = root.settings.e_feedrate, min = 0, max = 50000, size = (145, -1)) root.efeedc.SetDigits(1) root.efeedc.Bind(wx.EVT_SPINCTRLDOUBLE, root.setfeeds) root.efeedc.SetToolTip(wx.ToolTip(_("Extrude / Retract speed (mm/min)"))) root.efeedc.SetBackgroundColour((225, 200, 200)) root.efeedc.SetForegroundColour("black") root.efeedc.Bind(wx.EVT_TEXT, root.setfeeds) add("efeed_val", root.efeedc, container = esettingssizer, flag = wx.ALIGN_CENTER | wx.RIGHT, border = 5) add("efeed_label", wx.StaticText(esettingspanel, -1, _("Speed:")), container = esettingssizer, flag = wx.ALIGN_LEFT) add("efeed_unit", wx.StaticText(esettingspanel, -1, _("mm/\nmin")), container = esettingssizer, flag = wx.ALIGN_CENTER) else: root.efeedc = None esettingspanel.SetSizer(esettingssizer) add("esettings", esettingspanel, flag = wx.ALIGN_LEFT) if not standalone_mode: ebuttonspanel = root.newPanel(parentpanel) ebuttonssizer = wx.BoxSizer(wx.HORIZONTAL) if root.settings.extruders > 1: etool_sel_panel = esettingspanel if mini_mode else ebuttonspanel etool_label = wx.StaticText(etool_sel_panel, -1, _("Tool:")) if root.settings.extruders == 2: root.extrudersel = wx.Button(etool_sel_panel, -1, "0", style = wx.BU_EXACTFIT) root.extrudersel.SetToolTip(wx.ToolTip(_("Click to switch current extruder"))) def extrudersel_cb(event): if root.extrudersel.GetLabel() == "1": new = "0" else: new = "1" root.extrudersel.SetLabel(new) root.tool_change(event) root.extrudersel.Bind(wx.EVT_BUTTON, extrudersel_cb) root.extrudersel.GetValue = root.extrudersel.GetLabel root.extrudersel.SetValue = root.extrudersel.SetLabel else: choices = [str(i) for i in range(0, root.settings.extruders)] root.extrudersel = wx.ComboBox(etool_sel_panel, -1, choices = choices, style = wx.CB_DROPDOWN | wx.CB_READONLY, size = (50, -1)) root.extrudersel.SetToolTip(wx.ToolTip(_("Select current extruder"))) root.extrudersel.SetValue(choices[0]) root.extrudersel.Bind(wx.EVT_COMBOBOX, root.tool_change) root.printerControls.append(root.extrudersel) if mini_mode: add("etool_label", etool_label, container = esettingssizer, flag = wx.ALIGN_CENTER) add("etool_val", root.extrudersel, container = esettingssizer) else: ebuttonssizer.Add(etool_label, flag = wx.ALIGN_CENTER) ebuttonssizer.Add(root.extrudersel) for key in ["extrude", "reverse"]: desc = root.cpbuttons[key] btn = make_custom_button(root, ebuttonspanel, desc, style = wx.BU_EXACTFIT) ebuttonssizer.Add(btn, 1, flag = wx.EXPAND) ebuttonspanel.SetSizer(ebuttonssizer) add("ebuttons", ebuttonspanel, flag = wx.EXPAND) else: for key, btn in extra_buttons.items(): add(key, btn, flag = wx.EXPAND) class ControlsSizer(wx.GridBagSizer): def __init__(self, root, parentpanel = None, standalone_mode = False, mini_mode = False): super(ControlsSizer, self).__init__() if not parentpanel: parentpanel = root.panel if mini_mode: self.make_mini(root, parentpanel) else: self.make_standard(root, parentpanel, standalone_mode) def make_standard(self, root, parentpanel, standalone_mode): lltspanel = root.newPanel(parentpanel) llts = wx.BoxSizer(wx.HORIZONTAL) lltspanel.SetSizer(llts) self.Add(lltspanel, pos = (0, 0), span = (1, 6)) xyzpanel = root.newPanel(parentpanel) self.xyzsizer = XYZControlsSizer(root, xyzpanel) xyzpanel.SetSizer(self.xyzsizer) self.Add(xyzpanel, pos = (1, 0), span = (1, 6), flag = wx.ALIGN_CENTER) self.extra_buttons = {} pos_mapping = {"extrude": (4, 0), "reverse": (4, 2), } span_mapping = {"extrude": (1, 2), "reverse": (1, 3), } for key, desc in root.cpbuttons.items(): if not standalone_mode and key in ["extrude", "reverse"]: continue panel = lltspanel if key == "motorsoff" else parentpanel btn = make_custom_button(root, panel, desc) if key == "motorsoff": llts.Add(btn) elif not standalone_mode: self.Add(btn, pos = pos_mapping[key], span = span_mapping[key], flag = wx.EXPAND) else: self.extra_buttons[key] = btn root.xyfeedc = wx.SpinCtrl(lltspanel, -1, str(root.settings.xy_feedrate), min = 0, max = 50000, size = (130, -1)) root.xyfeedc.SetToolTip(wx.ToolTip(_("Set Maximum Speed for X & Y axes (mm/min)"))) llts.Add(wx.StaticText(lltspanel, -1, _("XY:")), flag = wx.ALIGN_CENTER_VERTICAL) llts.Add(root.xyfeedc) llts.Add(wx.StaticText(lltspanel, -1, _("mm/min Z:")), flag = wx.ALIGN_CENTER_VERTICAL) root.zfeedc = wx.SpinCtrl(lltspanel, -1, str(root.settings.z_feedrate), min = 0, max = 50000, size = (130, -1)) root.zfeedc.SetToolTip(wx.ToolTip(_("Set Maximum Speed for Z axis (mm/min)"))) llts.Add(root.zfeedc,) root.xyfeedc.Bind(wx.EVT_SPINCTRL, root.setfeeds) root.zfeedc.Bind(wx.EVT_SPINCTRL, root.setfeeds) root.xyfeedc.Bind(wx.EVT_TEXT, root.setfeeds) root.zfeedc.Bind(wx.EVT_TEXT, root.setfeeds) root.zfeedc.SetBackgroundColour((180, 255, 180)) root.zfeedc.SetForegroundColour("black") if not standalone_mode: add_extra_controls(self, root, parentpanel, None) def make_mini(self, root, parentpanel): root.xyb = XYButtonsMini(parentpanel, root.moveXY, root.homeButtonClicked, root.spacebarAction, root.bgcolor, zcallback = root.moveZ) self.Add(root.xyb, pos = (1, 0), span = (1, 4), flag = wx.ALIGN_CENTER) root.zb = ZButtonsMini(parentpanel, root.moveZ, root.bgcolor) self.Add(root.zb, pos = (0, 4), span = (2, 1), flag = wx.ALIGN_CENTER) wx.CallAfter(root.xyb.SetFocus) pos_mapping = {"motorsoff": (0, 0), } span_mapping = {"motorsoff": (1, 4), } btn = make_custom_button(root, parentpanel, root.cpbuttons["motorsoff"]) self.Add(btn, pos = pos_mapping["motorsoff"], span = span_mapping["motorsoff"], flag = wx.EXPAND) add_extra_controls(self, root, parentpanel, None, True)