from PyQt5 import QtWidgets, QtCore from FlatCAMTool import FlatCAMTool from flatcamGUI.GUIElements import RadioSet, FCDoubleSpinner, EvalEntry, FCEntry, FCButton from FlatCAMObj import FlatCAMGerber, FlatCAMExcellon, FlatCAMGeometry from numpy import Inf from shapely.geometry import Point from shapely import affinity import logging import gettext import FlatCAMTranslation as fcTranslate import builtins fcTranslate.apply_language('strings') if '_' not in builtins.__dict__: _ = gettext.gettext log = logging.getLogger('base') class DblSidedTool(FlatCAMTool): toolName = _("2-Sided PCB") def __init__(self, app): FlatCAMTool.__init__(self, app) self.decimals = self.app.decimals # ## Title title_label = QtWidgets.QLabel("%s" % self.toolName) title_label.setStyleSheet(""" QLabel { font-size: 16px; font-weight: bold; } """) self.layout.addWidget(title_label) self.layout.addWidget(QtWidgets.QLabel("")) # ## Grid Layout grid_lay = QtWidgets.QGridLayout() grid_lay.setColumnStretch(0, 1) grid_lay.setColumnStretch(1, 0) self.layout.addLayout(grid_lay) # Objects to be mirrored self.m_objects_label = QtWidgets.QLabel("%s:" % _("Mirror Operation")) self.m_objects_label.setToolTip('%s.' % _("Objects to be mirrored")) grid_lay.addWidget(self.m_objects_label, 0, 0, 1, 2) # ## Gerber Object to mirror self.gerber_object_combo = QtWidgets.QComboBox() self.gerber_object_combo.setModel(self.app.collection) self.gerber_object_combo.setRootModelIndex(self.app.collection.index(0, 0, QtCore.QModelIndex())) self.gerber_object_combo.setCurrentIndex(1) self.botlay_label = QtWidgets.QLabel("%s:" % _("GERBER")) self.botlay_label.setToolTip('%s.' % _("Gerber to be mirrored")) self.mirror_gerber_button = QtWidgets.QPushButton(_("Mirror")) self.mirror_gerber_button.setToolTip( _("Mirrors (flips) the specified object around \n" "the specified axis. Does not create a new \n" "object, but modifies it.") ) self.mirror_gerber_button.setStyleSheet(""" QPushButton { font-weight: bold; } """) self.mirror_gerber_button.setMinimumWidth(60) grid_lay.addWidget(self.botlay_label, 1, 0) grid_lay.addWidget(self.gerber_object_combo, 2, 0) grid_lay.addWidget(self.mirror_gerber_button, 2, 1) # ## Excellon Object to mirror self.exc_object_combo = QtWidgets.QComboBox() self.exc_object_combo.setModel(self.app.collection) self.exc_object_combo.setRootModelIndex(self.app.collection.index(1, 0, QtCore.QModelIndex())) self.exc_object_combo.setCurrentIndex(1) self.excobj_label = QtWidgets.QLabel("%s:" % _("EXCELLON")) self.excobj_label.setToolTip(_("Excellon Object to be mirrored.")) self.mirror_exc_button = QtWidgets.QPushButton(_("Mirror")) self.mirror_exc_button.setToolTip( _("Mirrors (flips) the specified object around \n" "the specified axis. Does not create a new \n" "object, but modifies it.") ) self.mirror_exc_button.setStyleSheet(""" QPushButton { font-weight: bold; } """) self.mirror_exc_button.setMinimumWidth(60) grid_lay.addWidget(self.excobj_label, 3, 0) grid_lay.addWidget(self.exc_object_combo, 4, 0) grid_lay.addWidget(self.mirror_exc_button, 4, 1) # ## Geometry Object to mirror self.geo_object_combo = QtWidgets.QComboBox() self.geo_object_combo.setModel(self.app.collection) self.geo_object_combo.setRootModelIndex(self.app.collection.index(2, 0, QtCore.QModelIndex())) self.geo_object_combo.setCurrentIndex(1) self.geoobj_label = QtWidgets.QLabel("%s:" % _("GEOMETRY")) self.geoobj_label.setToolTip( _("Geometry Obj to be mirrored.") ) self.mirror_geo_button = QtWidgets.QPushButton(_("Mirror")) self.mirror_geo_button.setToolTip( _("Mirrors (flips) the specified object around \n" "the specified axis. Does not create a new \n" "object, but modifies it.") ) self.mirror_geo_button.setStyleSheet(""" QPushButton { font-weight: bold; } """) self.mirror_geo_button.setMinimumWidth(60) # grid_lay.addRow("Bottom Layer:", self.object_combo) grid_lay.addWidget(self.geoobj_label, 5, 0) grid_lay.addWidget(self.geo_object_combo, 6, 0) grid_lay.addWidget(self.mirror_geo_button, 6, 1) separator_line = QtWidgets.QFrame() separator_line.setFrameShape(QtWidgets.QFrame.HLine) separator_line.setFrameShadow(QtWidgets.QFrame.Sunken) grid_lay.addWidget(separator_line, 7, 0, 1, 2) self.layout.addWidget(QtWidgets.QLabel("")) # ## Grid Layout grid_lay1 = QtWidgets.QGridLayout() grid_lay1.setColumnStretch(0, 0) grid_lay1.setColumnStretch(1, 1) self.layout.addLayout(grid_lay1) # Objects to be mirrored self.param_label = QtWidgets.QLabel("%s:" % _("Mirror Parameters")) self.param_label.setToolTip('%s.' % _("Parameters for the mirror operation")) grid_lay1.addWidget(self.param_label, 0, 0, 1, 3) # ## Axis self.mirax_label = QtWidgets.QLabel(_("Axis:")) self.mirax_label.setToolTip(_("Mirror vertically (X) or horizontally (Y).")) self.mirror_axis = RadioSet([{'label': 'X', 'value': 'X'}, {'label': 'Y', 'value': 'Y'}]) grid_lay1.addWidget(self.mirax_label, 2, 0) grid_lay1.addWidget(self.mirror_axis, 2, 1, 1, 2) # ## Axis Location self.axloc_label = QtWidgets.QLabel('%s:' % _("Reference")) self.axloc_label.setToolTip( _("The coordinates used as reference for the mirror operation.\n" "Can be:\n" "- Point -> a set of coordinates (x,y) around which the object is mirrored\n" "- Box -> a set of coordinates (x, y) obtained from the center of the\n" "bounding box of another object selected below") ) self.axis_location = RadioSet([{'label': _('Point'), 'value': 'point'}, {'label': _('Box'), 'value': 'box'}]) grid_lay1.addWidget(self.axloc_label, 4, 0) grid_lay1.addWidget(self.axis_location, 4, 1, 1, 2) # ## Point/Box self.point_entry = EvalEntry() # Add a reference self.add_point_button = QtWidgets.QPushButton(_("Add")) self.add_point_button.setToolTip( _("Add the coordinates in format (x, y) through which the mirroring axis \n " "selected in 'MIRROR AXIS' pass.\n" "The (x, y) coordinates are captured by pressing SHIFT key\n" "and left mouse button click on canvas or you can enter the coords manually.") ) self.add_point_button.setStyleSheet(""" QPushButton { font-weight: bold; } """) self.add_point_button.setMinimumWidth(60) grid_lay1.addWidget(self.point_entry, 7, 0, 1, 2) grid_lay1.addWidget(self.add_point_button, 7, 2) # ## Grid Layout grid_lay2 = QtWidgets.QGridLayout() grid_lay2.setColumnStretch(0, 0) grid_lay2.setColumnStretch(1, 1) self.layout.addLayout(grid_lay2) self.box_type_label = QtWidgets.QLabel('%s:' % _("Object Type")) self.box_type_label.setToolTip( _("It can be of type: Gerber or Excellon or Geometry.\n" "The selection here decide the type of objects that will be\n" "in the Object combobox.") ) # Type of object used as BOX reference self.box_type_radio = RadioSet([{'label': _('Gerber'), 'value': 'grb'}, {'label': _('Excellon'), 'value': 'exc'}, {'label': _('Geometry'), 'value': 'geo'}]) self.box_type_label.hide() self.box_type_radio.hide() grid_lay2.addWidget(self.box_type_label, 0, 0, 1, 2) grid_lay2.addWidget(self.box_type_radio, 1, 0, 1, 2) self.box_object_label = QtWidgets.QLabel('%s:' % _("Object")) self.box_object_label.setToolTip( _("Object to be used as mirror reference.") ) # Object used as BOX reference self.box_combo = QtWidgets.QComboBox() self.box_combo.setModel(self.app.collection) self.box_combo.setRootModelIndex(self.app.collection.index(0, 0, QtCore.QModelIndex())) self.box_combo.setCurrentIndex(1) self.box_object_label.hide() self.box_combo.hide() grid_lay2.addWidget(self.box_object_label, 2, 0, 1, 2) grid_lay2.addWidget(self.box_combo, 3, 0, 1, 2) separator_line = QtWidgets.QFrame() separator_line.setFrameShape(QtWidgets.QFrame.HLine) separator_line.setFrameShadow(QtWidgets.QFrame.Sunken) self.layout.addWidget(separator_line) self.layout.addWidget(QtWidgets.QLabel("")) # ## Alignment holes self.alignment_label = QtWidgets.QLabel("%s:" % _('Alignment Excellon object')) self.alignment_label.setToolTip( _("Creates an Excellon Object containing the\n" "specified alignment holes and their mirror\n" "images.") ) self.layout.addWidget(self.alignment_label) # ## Alignment holes self.ah_label = QtWidgets.QLabel("%s:" % _('Alignment Drill Coordinates')) self.ah_label.setToolTip( _("Alignment holes (x1, y1), (x2, y2), ... " "on one side of the mirror axis. For each set of (x, y) coordinates\n" "entered here, a pair of drills will be created:\n\n" "- one drill at the coordinates from the field\n" "- one drill in mirror position over the axis selected above in the 'Mirror Axis'.") ) self.layout.addWidget(self.ah_label) grid_lay3 = QtWidgets.QGridLayout() self.layout.addLayout(grid_lay3) self.alignment_holes = EvalEntry() grid_lay3.addWidget(self.alignment_holes, 0, 0, 1, 2) self.add_drill_point_button = FCButton(_("Add")) self.add_drill_point_button.setToolTip( _("Add alignment drill holes coords in the format: (x1, y1), (x2, y2), ... \n" "on one side of the mirror axis.\n\n" "The coordinates set can be obtained:\n" "- press SHIFT key and left mouse clicking on canvas. Then click Add.\n" "- press SHIFT key and left mouse clicking on canvas. Then CTRL+V in the field.\n" "- press SHIFT key and left mouse clicking on canvas. Then RMB click in the field and click Paste.\n" "- by entering the coords manually in the format: (x1, y1), (x2, y2), ...") ) self.add_drill_point_button.setStyleSheet(""" QPushButton { font-weight: bold; } """) self.delete_drill_point_button = FCButton(_("Delete Last")) self.delete_drill_point_button.setToolTip( _("Delete the last coordinates tupple in the list.") ) drill_hlay = QtWidgets.QHBoxLayout() drill_hlay.addWidget(self.add_drill_point_button) drill_hlay.addWidget(self.delete_drill_point_button) grid_lay3.addLayout(drill_hlay, 1, 0, 1, 2) grid0 = QtWidgets.QGridLayout() self.layout.addLayout(grid0) grid0.setColumnStretch(0, 0) grid0.setColumnStretch(1, 1) # ## Drill diameter for alignment holes self.dt_label = QtWidgets.QLabel("%s:" % _('Alignment Drill Diameter')) self.dt_label.setToolTip( _("Diameter of the drill for the " "alignment holes.") ) grid0.addWidget(self.dt_label, 0, 0, 1, 2) # Drill diameter value self.drill_dia = FCDoubleSpinner() self.drill_dia.set_precision(self.decimals) self.drill_dia.set_range(0.0000, 9999.9999) self.drill_dia.setToolTip( _("Diameter of the drill for the " "alignment holes.") ) grid0.addWidget(self.drill_dia, 1, 0, 1, 2) # ## Buttons self.create_alignment_hole_button = QtWidgets.QPushButton(_("Create Excellon Object")) self.create_alignment_hole_button.setToolTip( _("Creates an Excellon Object containing the\n" "specified alignment holes and their mirror\n" "images.") ) self.create_alignment_hole_button.setStyleSheet(""" QPushButton { font-weight: bold; } """) self.layout.addWidget(self.create_alignment_hole_button) separator_line = QtWidgets.QFrame() separator_line.setFrameShape(QtWidgets.QFrame.HLine) separator_line.setFrameShadow(QtWidgets.QFrame.Sunken) self.layout.addWidget(separator_line) self.layout.addWidget(QtWidgets.QLabel('')) grid1 = QtWidgets.QGridLayout() self.layout.addLayout(grid1) grid1.setColumnStretch(0, 0) grid1.setColumnStretch(1, 1) # ## Title Bounds Values self.bv_label = QtWidgets.QLabel("%s:" % _('Bounds Values')) self.bv_label.setToolTip( _("Select on canvas the object(s)\n" "for which to calculate bounds values.") ) grid1.addWidget(self.bv_label, 0, 0, 1, 2) # Xmin value self.xmin_entry = FCDoubleSpinner() self.xmin_entry.set_precision(self.decimals) self.xmin_entry.set_range(-9999.9999, 9999.9999) self.xmin_label = QtWidgets.QLabel('%s:' % _("X min")) self.xmin_label.setToolTip( _("Minimum location.") ) self.xmin_entry.setReadOnly(True) grid1.addWidget(self.xmin_label, 1, 0) grid1.addWidget(self.xmin_entry, 1, 1) # Ymin value self.ymin_entry = FCDoubleSpinner() self.ymin_entry.set_precision(self.decimals) self.ymin_entry.set_range(-9999.9999, 9999.9999) self.ymin_label = QtWidgets.QLabel('%s:' % _("Y min")) self.ymin_label.setToolTip( _("Minimum location.") ) self.ymin_entry.setReadOnly(True) grid1.addWidget(self.ymin_label, 2, 0) grid1.addWidget(self.ymin_entry, 2, 1) # Xmax value self.xmax_entry = FCDoubleSpinner() self.xmax_entry.set_precision(self.decimals) self.xmax_entry.set_range(-9999.9999, 9999.9999) self.xmax_label = QtWidgets.QLabel('%s:' % _("X max")) self.xmax_label.setToolTip( _("Maximum location.") ) self.xmax_entry.setReadOnly(True) grid1.addWidget(self.xmax_label, 3, 0) grid1.addWidget(self.xmax_entry, 3, 1) # Ymax value self.ymax_entry = FCDoubleSpinner() self.ymax_entry.set_precision(self.decimals) self.ymax_entry.set_range(-9999.9999, 9999.9999) self.ymax_label = QtWidgets.QLabel('%s:' % _("Y max")) self.ymax_label.setToolTip( _("Maximum location.") ) self.ymax_entry.setReadOnly(True) grid1.addWidget(self.ymax_label, 4, 0) grid1.addWidget(self.ymax_entry, 4, 1) # Center point value self.center_entry = FCEntry() self.center_label = QtWidgets.QLabel('%s:' % _("Centroid")) self.center_label.setToolTip( _("The center point location for the rectangular\n" "bounding shape. Centroid. Format is (x, y).") ) self.center_entry.setReadOnly(True) grid1.addWidget(self.center_label, 5, 0) grid1.addWidget(self.center_entry, 5, 1) # Calculate Bounding box self.calculate_bb_button = QtWidgets.QPushButton(_("Calculate Bounds Values")) self.calculate_bb_button.setToolTip( _("Calculate the enveloping rectangular shape coordinates,\n" "for the selection of objects.\n" "The envelope shape is parallel with the X, Y axis.") ) self.calculate_bb_button.setStyleSheet(""" QPushButton { font-weight: bold; } """) self.layout.addWidget(self.calculate_bb_button) self.layout.addStretch() # ## Reset Tool self.reset_button = QtWidgets.QPushButton(_("Reset Tool")) self.reset_button.setToolTip( _("Will reset the tool parameters.") ) self.reset_button.setStyleSheet(""" QPushButton { font-weight: bold; } """) self.layout.addWidget(self.reset_button) # ## Signals self.mirror_gerber_button.clicked.connect(self.on_mirror_gerber) self.mirror_exc_button.clicked.connect(self.on_mirror_exc) self.mirror_geo_button.clicked.connect(self.on_mirror_geo) self.add_point_button.clicked.connect(self.on_point_add) self.add_drill_point_button.clicked.connect(self.on_drill_add) self.delete_drill_point_button.clicked.connect(self.on_drill_delete_last) self.box_type_radio.activated_custom.connect(self.on_combo_box_type) self.axis_location.group_toggle_fn = self.on_toggle_pointbox self.create_alignment_hole_button.clicked.connect(self.on_create_alignment_holes) self.calculate_bb_button.clicked.connect(self.on_bbox_coordinates) self.reset_button.clicked.connect(self.set_tool_ui) self.drill_values = "" def install(self, icon=None, separator=None, **kwargs): FlatCAMTool.install(self, icon, separator, shortcut='ALT+D', **kwargs) def run(self, toggle=True): self.app.report_usage("Tool2Sided()") if toggle: # if the splitter is hidden, display it, else hide it but only if the current widget is the same if self.app.ui.splitter.sizes()[0] == 0: self.app.ui.splitter.setSizes([1, 1]) else: try: if self.app.ui.tool_scroll_area.widget().objectName() == self.toolName: # if tab is populated with the tool but it does not have the focus, focus on it if not self.app.ui.notebook.currentWidget() is self.app.ui.tool_tab: # focus on Tool Tab self.app.ui.notebook.setCurrentWidget(self.app.ui.tool_tab) else: self.app.ui.splitter.setSizes([0, 1]) except AttributeError: pass else: if self.app.ui.splitter.sizes()[0] == 0: self.app.ui.splitter.setSizes([1, 1]) FlatCAMTool.run(self) self.set_tool_ui() self.app.ui.notebook.setTabText(2, _("2-Sided Tool")) def set_tool_ui(self): self.reset_fields() self.point_entry.set_value("") self.alignment_holes.set_value("") self.mirror_axis.set_value(self.app.defaults["tools_2sided_mirror_axis"]) self.axis_location.set_value(self.app.defaults["tools_2sided_axis_loc"]) self.drill_dia.set_value(self.app.defaults["tools_2sided_drilldia"]) self.xmin_entry.set_value(0.0) self.ymin_entry.set_value(0.0) self.xmax_entry.set_value(0.0) self.ymax_entry.set_value(0.0) self.center_entry.set_value('') def on_combo_box_type(self, val): obj_type = { 'grb': 0, 'exc': 1, 'geo': 2 }[val] self.box_combo.setRootModelIndex(self.app.collection.index(obj_type, 0, QtCore.QModelIndex())) self.box_combo.setCurrentIndex(0) def on_create_alignment_holes(self): axis = self.mirror_axis.get_value() mode = self.axis_location.get_value() if mode == "point": try: px, py = self.point_entry.get_value() except TypeError: self.app.inform.emit('[WARNING_NOTCL] %s' % _("'Point' reference is selected and 'Point' coordinates " "are missing. Add them and retry.")) return else: selection_index = self.box_combo.currentIndex() model_index = self.app.collection.index(selection_index, 0, self.gerber_object_combo.rootModelIndex()) try: bb_obj = model_index.internalPointer().obj except AttributeError: model_index = self.app.collection.index(selection_index, 0, self.exc_object_combo.rootModelIndex()) try: bb_obj = model_index.internalPointer().obj except AttributeError: model_index = self.app.collection.index(selection_index, 0, self.geo_object_combo.rootModelIndex()) try: bb_obj = model_index.internalPointer().obj except AttributeError: self.app.inform.emit( '[WARNING_NOTCL] %s' % _("There is no Box reference object loaded. Load one and retry.")) return xmin, ymin, xmax, ymax = bb_obj.bounds() px = 0.5 * (xmin + xmax) py = 0.5 * (ymin + ymax) xscale, yscale = {"X": (1.0, -1.0), "Y": (-1.0, 1.0)}[axis] dia = float(self.drill_dia.get_value()) if dia is '': self.app.inform.emit('[WARNING_NOTCL] %s' % _("No value or wrong format in Drill Dia entry. Add it and retry.")) return tools = dict() tools["1"] = dict() tools["1"]["C"] = dia tools["1"]['solid_geometry'] = list() # holes = self.alignment_holes.get_value() holes = eval('[{}]'.format(self.alignment_holes.text())) if not holes: self.app.inform.emit('[WARNING_NOTCL] %s' % _("There are no Alignment Drill Coordinates to use. " "Add them and retry.")) return drills = list() for hole in holes: point = Point(hole) point_mirror = affinity.scale(point, xscale, yscale, origin=(px, py)) drills.append({"point": point, "tool": "1"}) drills.append({"point": point_mirror, "tool": "1"}) tools["1"]['solid_geometry'].append(point) tools["1"]['solid_geometry'].append(point_mirror) def obj_init(obj_inst, app_inst): obj_inst.tools = tools obj_inst.drills = drills obj_inst.create_geometry() obj_inst.source_file = self.app.export_excellon(obj_name=obj_inst.options['name'], local_use=obj_inst, filename=None, use_thread=False) self.app.new_object("excellon", "Alignment Drills", obj_init) self.drill_values = '' self.app.inform.emit('[success] %s' % _("Excellon object with alignment drills created...")) def on_mirror_gerber(self): selection_index = self.gerber_object_combo.currentIndex() # fcobj = self.app.collection.object_list[selection_index] model_index = self.app.collection.index(selection_index, 0, self.gerber_object_combo.rootModelIndex()) try: fcobj = model_index.internalPointer().obj except Exception: self.app.inform.emit('[WARNING_NOTCL] %s' % _("There is no Gerber object loaded ...")) return if not isinstance(fcobj, FlatCAMGerber): self.app.inform.emit('[ERROR_NOTCL] %s' % _("Only Gerber, Excellon and Geometry objects can be mirrored.")) return axis = self.mirror_axis.get_value() mode = self.axis_location.get_value() if mode == "point": try: px, py = self.point_entry.get_value() except TypeError: self.app.inform.emit('[WARNING_NOTCL] %s' % _("'Point' coordinates missing. " "Using Origin (0, 0) as mirroring reference.")) px, py = (0, 0) else: selection_index_box = self.box_combo.currentIndex() model_index_box = self.app.collection.index(selection_index_box, 0, self.box_combo.rootModelIndex()) try: bb_obj = model_index_box.internalPointer().obj except Exception: self.app.inform.emit('[WARNING_NOTCL] %s' % _("There is no Box object loaded ...")) return xmin, ymin, xmax, ymax = bb_obj.bounds() px = 0.5 * (xmin + xmax) py = 0.5 * (ymin + ymax) fcobj.mirror(axis, [px, py]) self.app.object_changed.emit(fcobj) fcobj.plot() self.app.inform.emit('[success] Gerber %s %s...' % (str(fcobj.options['name']), _("was mirrored"))) def on_mirror_exc(self): selection_index = self.exc_object_combo.currentIndex() # fcobj = self.app.collection.object_list[selection_index] model_index = self.app.collection.index(selection_index, 0, self.exc_object_combo.rootModelIndex()) try: fcobj = model_index.internalPointer().obj except Exception: self.app.inform.emit('[WARNING_NOTCL] %s' % _("There is no Excellon object loaded ...")) return if not isinstance(fcobj, FlatCAMExcellon): self.app.inform.emit('[ERROR_NOTCL] %s' % _("Only Gerber, Excellon and Geometry objects can be mirrored.")) return axis = self.mirror_axis.get_value() mode = self.axis_location.get_value() if mode == "point": try: px, py = self.point_entry.get_value() except Exception as e: log.debug("DblSidedTool.on_mirror_geo() --> %s" % str(e)) self.app.inform.emit('[WARNING_NOTCL] %s' % _("There are no Point coordinates in the Point field. " "Add coords and try again ...")) return else: selection_index_box = self.box_combo.currentIndex() model_index_box = self.app.collection.index(selection_index_box, 0, self.box_combo.rootModelIndex()) try: bb_obj = model_index_box.internalPointer().obj except Exception as e: log.debug("DblSidedTool.on_mirror_geo() --> %s" % str(e)) self.app.inform.emit('[WARNING_NOTCL] %s' % _("There is no Box object loaded ...")) return xmin, ymin, xmax, ymax = bb_obj.bounds() px = 0.5 * (xmin + xmax) py = 0.5 * (ymin + ymax) fcobj.mirror(axis, [px, py]) self.app.object_changed.emit(fcobj) fcobj.plot() self.app.inform.emit('[success] Excellon %s %s...' % (str(fcobj.options['name']), _("was mirrored"))) def on_mirror_geo(self): selection_index = self.geo_object_combo.currentIndex() # fcobj = self.app.collection.object_list[selection_index] model_index = self.app.collection.index(selection_index, 0, self.geo_object_combo.rootModelIndex()) try: fcobj = model_index.internalPointer().obj except Exception: self.app.inform.emit('[WARNING_NOTCL] %s' % _("There is no Geometry object loaded ...")) return if not isinstance(fcobj, FlatCAMGeometry): self.app.inform.emit('[ERROR_NOTCL] %s' % _("Only Gerber, Excellon and Geometry objects can be mirrored.")) return axis = self.mirror_axis.get_value() mode = self.axis_location.get_value() if mode == "point": px, py = self.point_entry.get_value() else: selection_index_box = self.box_combo.currentIndex() model_index_box = self.app.collection.index(selection_index_box, 0, self.box_combo.rootModelIndex()) try: bb_obj = model_index_box.internalPointer().obj except Exception: self.app.inform.emit('[WARNING_NOTCL] %s' % _("There is no Box object loaded ...")) return xmin, ymin, xmax, ymax = bb_obj.bounds() px = 0.5 * (xmin + xmax) py = 0.5 * (ymin + ymax) fcobj.mirror(axis, [px, py]) self.app.object_changed.emit(fcobj) fcobj.plot() self.app.inform.emit('[success] Geometry %s %s...' % (str(fcobj.options['name']), _("was mirrored"))) def on_point_add(self): val = self.app.defaults["global_point_clipboard_format"] % (self.app.pos[0], self.app.pos[1]) self.point_entry.set_value(val) def on_drill_add(self): self.drill_values += (self.app.defaults["global_point_clipboard_format"] % (self.app.pos[0], self.app.pos[1])) + ',' self.alignment_holes.set_value(self.drill_values) def on_drill_delete_last(self): drill_values_without_last_tupple = self.drill_values.rpartition('(')[0] self.drill_values = drill_values_without_last_tupple self.alignment_holes.set_value(self.drill_values) def on_toggle_pointbox(self): if self.axis_location.get_value() == "point": self.point_entry.show() self.add_point_button.show() self.box_type_label.hide() self.box_type_radio.hide() self.box_object_label.hide() self.box_combo.hide() else: self.point_entry.hide() self.add_point_button.hide() self.box_type_label.show() self.box_type_radio.show() self.box_object_label.show() self.box_combo.show() def on_bbox_coordinates(self): xmin = Inf ymin = Inf xmax = -Inf ymax = -Inf obj_list = self.app.collection.get_selected() if not obj_list: self.app.inform.emit('[ERROR_NOTCL] %s' % _("Failed. No object(s) selected...")) return for obj in obj_list: try: gxmin, gymin, gxmax, gymax = obj.bounds() xmin = min([xmin, gxmin]) ymin = min([ymin, gymin]) xmax = max([xmax, gxmax]) ymax = max([ymax, gymax]) except Exception as e: log.warning("DEV WARNING: Tried to get bounds of empty geometry in DblSidedTool. %s" % str(e)) self.xmin_entry.set_value(xmin) self.ymin_entry.set_value(ymin) self.xmax_entry.set_value(xmax) self.ymax_entry.set_value(ymax) cx = '%.*f' % (self.decimals, (((xmax - xmin) / 2.0) + xmin)) cy = '%.*f' % (self.decimals, (((ymax - ymin) / 2.0) + ymin)) val_txt = '(%s, %s)' % (cx, cy) self.center_entry.set_value(val_txt) self.axis_location.set_value('point') self.point_entry.set_value(val_txt) self.app.delete_selection_shape() def reset_fields(self): self.gerber_object_combo.setRootModelIndex(self.app.collection.index(0, 0, QtCore.QModelIndex())) self.exc_object_combo.setRootModelIndex(self.app.collection.index(1, 0, QtCore.QModelIndex())) self.geo_object_combo.setRootModelIndex(self.app.collection.index(2, 0, QtCore.QModelIndex())) self.box_combo.setRootModelIndex(self.app.collection.index(0, 0, QtCore.QModelIndex())) self.gerber_object_combo.setCurrentIndex(0) self.exc_object_combo.setCurrentIndex(0) self.geo_object_combo.setCurrentIndex(0) self.box_combo.setCurrentIndex(0) self.box_type_radio.set_value('grb') self.drill_values = ""