2019-10-13 15:13:39 +00:00
|
|
|
# ##########################################################
|
2019-03-10 13:22:16 +00:00
|
|
|
# FlatCAM: 2D Post-processing for Manufacturing #
|
|
|
|
# File Author: Marius Adrian Stanciu (c) #
|
|
|
|
# Date: 3/10/2019 #
|
|
|
|
# MIT Licence #
|
2019-10-13 15:13:39 +00:00
|
|
|
# ##########################################################
|
2019-03-10 13:22:16 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
from PyQt5 import QtWidgets, QtGui, QtCore
|
2020-06-03 17:35:59 +00:00
|
|
|
from appTool import AppTool
|
|
|
|
from appGUI.GUIElements import FCDoubleSpinner, FCCheckBox, FCButton, OptionalInputSection, FCEntry, FCComboBox, \
|
2020-06-02 23:41:25 +00:00
|
|
|
NumericalEvalTupleEntry
|
|
|
|
|
|
|
|
import numpy as np
|
2019-03-07 16:04:11 +00:00
|
|
|
|
2019-03-07 23:32:18 +00:00
|
|
|
import gettext
|
2020-06-03 17:35:59 +00:00
|
|
|
import appTranslation as fcTranslate
|
2019-07-16 13:22:20 +00:00
|
|
|
import builtins
|
2019-03-10 15:12:58 +00:00
|
|
|
|
2019-03-13 23:09:06 +00:00
|
|
|
fcTranslate.apply_language('strings')
|
2019-03-10 15:12:58 +00:00
|
|
|
if '_' not in builtins.__dict__:
|
|
|
|
_ = gettext.gettext
|
2019-01-03 19:25:08 +00:00
|
|
|
|
|
|
|
|
2020-06-02 15:29:45 +00:00
|
|
|
class ToolTransform(AppTool):
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-03-10 12:34:13 +00:00
|
|
|
toolName = _("Object Transform")
|
|
|
|
rotateName = _("Rotate")
|
|
|
|
skewName = _("Skew/Shear")
|
|
|
|
scaleName = _("Scale")
|
|
|
|
flipName = _("Mirror (Flip)")
|
|
|
|
offsetName = _("Offset")
|
2019-12-23 20:59:01 +00:00
|
|
|
bufferName = _("Buffer")
|
2019-01-03 19:25:08 +00:00
|
|
|
|
|
|
|
def __init__(self, app):
|
2020-06-02 15:29:45 +00:00
|
|
|
AppTool.__init__(self, app)
|
2019-12-05 13:18:54 +00:00
|
|
|
self.decimals = self.app.decimals
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-05-30 18:05:12 +00:00
|
|
|
# ## Title
|
2019-02-18 01:45:34 +00:00
|
|
|
title_label = QtWidgets.QLabel("%s" % self.toolName)
|
|
|
|
title_label.setStyleSheet("""
|
|
|
|
QLabel
|
|
|
|
{
|
|
|
|
font-size: 16px;
|
|
|
|
font-weight: bold;
|
|
|
|
}
|
|
|
|
""")
|
2020-04-22 20:00:54 +00:00
|
|
|
self.layout.addWidget(title_label)
|
|
|
|
self.layout.addWidget(QtWidgets.QLabel(''))
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-10-05 19:01:02 +00:00
|
|
|
# ## Layout
|
|
|
|
grid0 = QtWidgets.QGridLayout()
|
2020-04-22 20:00:54 +00:00
|
|
|
self.layout.addLayout(grid0)
|
2019-10-05 19:01:02 +00:00
|
|
|
grid0.setColumnStretch(0, 0)
|
|
|
|
grid0.setColumnStretch(1, 1)
|
|
|
|
grid0.setColumnStretch(2, 0)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-10-05 19:01:02 +00:00
|
|
|
grid0.addWidget(QtWidgets.QLabel(''))
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
# Reference
|
|
|
|
ref_label = QtWidgets.QLabel('%s:' % _("Reference"))
|
|
|
|
ref_label.setToolTip(
|
|
|
|
_("The reference point for Rotate, Skew, Scale, Mirror.\n"
|
|
|
|
"Can be:\n"
|
|
|
|
"- Origin -> it is the 0, 0 point\n"
|
|
|
|
"- Selection -> the center of the bounding box of the selected objects\n"
|
|
|
|
"- Point -> a custom point defined by X,Y coordinates\n"
|
|
|
|
"- Object -> the center of the bounding box of a specific object")
|
|
|
|
)
|
|
|
|
self.ref_combo = FCComboBox()
|
|
|
|
self.ref_items = [_("Origin"), _("Selection"), _("Point"), _("Object")]
|
|
|
|
self.ref_combo.addItems(self.ref_items)
|
|
|
|
|
|
|
|
grid0.addWidget(ref_label, 0, 0)
|
|
|
|
grid0.addWidget(self.ref_combo, 0, 1, 1, 2)
|
|
|
|
|
|
|
|
self.point_label = QtWidgets.QLabel('%s:' % _("Value"))
|
|
|
|
self.point_label.setToolTip(
|
|
|
|
_("A point of reference in format X,Y.")
|
|
|
|
)
|
|
|
|
self.point_entry = NumericalEvalTupleEntry()
|
|
|
|
|
|
|
|
grid0.addWidget(self.point_label, 1, 0)
|
|
|
|
grid0.addWidget(self.point_entry, 1, 1, 1, 2)
|
|
|
|
|
|
|
|
self.point_button = FCButton(_("Add"))
|
|
|
|
self.point_button.setToolTip(
|
|
|
|
_("Add point coordinates from clipboard.")
|
|
|
|
)
|
|
|
|
grid0.addWidget(self.point_button, 2, 0, 1, 3)
|
|
|
|
|
|
|
|
# Type of object to be used as reference
|
|
|
|
self.type_object_label = QtWidgets.QLabel('%s:' % _("Type"))
|
|
|
|
self.type_object_label.setToolTip(
|
|
|
|
_("The type of object used as reference.")
|
|
|
|
)
|
|
|
|
|
|
|
|
self.type_obj_combo = FCComboBox()
|
|
|
|
self.type_obj_combo.addItem(_("Gerber"))
|
|
|
|
self.type_obj_combo.addItem(_("Excellon"))
|
|
|
|
self.type_obj_combo.addItem(_("Geometry"))
|
|
|
|
|
|
|
|
self.type_obj_combo.setItemIcon(0, QtGui.QIcon(self.app.resource_location + "/flatcam_icon16.png"))
|
|
|
|
self.type_obj_combo.setItemIcon(1, QtGui.QIcon(self.app.resource_location + "/drill16.png"))
|
|
|
|
self.type_obj_combo.setItemIcon(2, QtGui.QIcon(self.app.resource_location + "/geometry16.png"))
|
|
|
|
|
|
|
|
grid0.addWidget(self.type_object_label, 3, 0)
|
|
|
|
grid0.addWidget(self.type_obj_combo, 3, 1, 1, 2)
|
|
|
|
|
|
|
|
# Object to be used as reference
|
|
|
|
self.object_combo = FCComboBox()
|
|
|
|
self.object_combo.setModel(self.app.collection)
|
|
|
|
self.object_combo.setRootModelIndex(self.app.collection.index(0, 0, QtCore.QModelIndex()))
|
|
|
|
self.object_combo.is_last = True
|
|
|
|
|
|
|
|
self.object_combo.setToolTip(
|
|
|
|
_("The object used as reference.\n"
|
|
|
|
"The used point is the center of it's bounding box.")
|
|
|
|
)
|
|
|
|
grid0.addWidget(self.object_combo, 4, 0, 1, 3)
|
|
|
|
|
|
|
|
separator_line = QtWidgets.QFrame()
|
|
|
|
separator_line.setFrameShape(QtWidgets.QFrame.HLine)
|
|
|
|
separator_line.setFrameShadow(QtWidgets.QFrame.Sunken)
|
|
|
|
grid0.addWidget(separator_line, 5, 0, 1, 3)
|
|
|
|
|
2019-05-30 18:05:12 +00:00
|
|
|
# ## Rotate Title
|
2019-01-03 19:25:08 +00:00
|
|
|
rotate_title_label = QtWidgets.QLabel("<font size=3><b>%s</b></font>" % self.rotateName)
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(rotate_title_label, 6, 0, 1, 3)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-08-19 14:24:56 +00:00
|
|
|
self.rotate_label = QtWidgets.QLabel('%s:' % _("Angle"))
|
2019-01-03 19:25:08 +00:00
|
|
|
self.rotate_label.setToolTip(
|
2019-03-10 12:34:13 +00:00
|
|
|
_("Angle for Rotation action, in degrees.\n"
|
2019-07-16 13:22:20 +00:00
|
|
|
"Float number between -360 and 359.\n"
|
|
|
|
"Positive numbers for CW motion.\n"
|
|
|
|
"Negative numbers for CCW motion.")
|
2019-01-03 19:25:08 +00:00
|
|
|
)
|
|
|
|
|
2020-02-17 02:43:01 +00:00
|
|
|
self.rotate_entry = FCDoubleSpinner(callback=self.confirmation_message)
|
2019-10-05 19:01:02 +00:00
|
|
|
self.rotate_entry.set_precision(self.decimals)
|
|
|
|
self.rotate_entry.setSingleStep(45)
|
|
|
|
self.rotate_entry.setWrapping(True)
|
|
|
|
self.rotate_entry.set_range(-360, 360)
|
|
|
|
|
|
|
|
# self.rotate_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
self.rotate_button = FCButton(_("Rotate"))
|
2019-01-03 19:25:08 +00:00
|
|
|
self.rotate_button.setToolTip(
|
2019-03-10 12:34:13 +00:00
|
|
|
_("Rotate the selected object(s).\n"
|
2019-07-16 13:22:20 +00:00
|
|
|
"The point of reference is the middle of\n"
|
|
|
|
"the bounding box for all selected objects.")
|
2019-01-03 19:25:08 +00:00
|
|
|
)
|
2019-08-07 11:29:59 +00:00
|
|
|
self.rotate_button.setMinimumWidth(90)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(self.rotate_label, 7, 0)
|
|
|
|
grid0.addWidget(self.rotate_entry, 7, 1)
|
|
|
|
grid0.addWidget(self.rotate_button, 7, 2)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-12-29 23:18:56 +00:00
|
|
|
separator_line = QtWidgets.QFrame()
|
|
|
|
separator_line.setFrameShape(QtWidgets.QFrame.HLine)
|
|
|
|
separator_line.setFrameShadow(QtWidgets.QFrame.Sunken)
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(separator_line, 8, 0, 1, 3)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-05-30 18:05:12 +00:00
|
|
|
# ## Skew Title
|
2019-01-03 19:25:08 +00:00
|
|
|
skew_title_label = QtWidgets.QLabel("<font size=3><b>%s</b></font>" % self.skewName)
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(skew_title_label, 9, 0, 1, 2)
|
|
|
|
|
|
|
|
self.skew_link_cb = FCCheckBox()
|
|
|
|
self.skew_link_cb.setText(_("Link"))
|
|
|
|
self.skew_link_cb.setToolTip(
|
2020-06-03 18:08:06 +00:00
|
|
|
_("Link the Y entry to X entry and copy its content.")
|
2020-06-02 23:41:25 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
grid0.addWidget(self.skew_link_cb, 9, 2)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-10-05 19:01:02 +00:00
|
|
|
self.skewx_label = QtWidgets.QLabel('%s:' % _("X angle"))
|
2019-01-03 19:25:08 +00:00
|
|
|
self.skewx_label.setToolTip(
|
2019-03-10 12:34:13 +00:00
|
|
|
_("Angle for Skew action, in degrees.\n"
|
2019-10-05 19:01:02 +00:00
|
|
|
"Float number between -360 and 360.")
|
2019-01-03 19:25:08 +00:00
|
|
|
)
|
2020-02-17 02:43:01 +00:00
|
|
|
self.skewx_entry = FCDoubleSpinner(callback=self.confirmation_message)
|
2019-10-05 19:01:02 +00:00
|
|
|
# self.skewx_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
|
|
|
|
self.skewx_entry.set_precision(self.decimals)
|
|
|
|
self.skewx_entry.set_range(-360, 360)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
self.skewx_button = FCButton(_("Skew X"))
|
2019-01-03 19:25:08 +00:00
|
|
|
self.skewx_button.setToolTip(
|
2019-03-10 12:34:13 +00:00
|
|
|
_("Skew/shear the selected object(s).\n"
|
2019-07-16 13:22:20 +00:00
|
|
|
"The point of reference is the middle of\n"
|
|
|
|
"the bounding box for all selected objects."))
|
2019-08-07 11:29:59 +00:00
|
|
|
self.skewx_button.setMinimumWidth(90)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(self.skewx_label, 10, 0)
|
|
|
|
grid0.addWidget(self.skewx_entry, 10, 1)
|
|
|
|
grid0.addWidget(self.skewx_button, 10, 2)
|
2019-10-05 19:01:02 +00:00
|
|
|
|
|
|
|
self.skewy_label = QtWidgets.QLabel('%s:' % _("Y angle"))
|
2019-01-03 19:25:08 +00:00
|
|
|
self.skewy_label.setToolTip(
|
2019-03-10 12:34:13 +00:00
|
|
|
_("Angle for Skew action, in degrees.\n"
|
2019-10-05 19:01:02 +00:00
|
|
|
"Float number between -360 and 360.")
|
2019-01-03 19:25:08 +00:00
|
|
|
)
|
2020-02-17 02:43:01 +00:00
|
|
|
self.skewy_entry = FCDoubleSpinner(callback=self.confirmation_message)
|
2019-10-05 19:01:02 +00:00
|
|
|
# self.skewy_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
|
|
|
|
self.skewy_entry.set_precision(self.decimals)
|
|
|
|
self.skewy_entry.set_range(-360, 360)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
self.skewy_button = FCButton(_("Skew Y"))
|
2019-01-03 19:25:08 +00:00
|
|
|
self.skewy_button.setToolTip(
|
2019-03-10 12:34:13 +00:00
|
|
|
_("Skew/shear the selected object(s).\n"
|
2019-07-16 13:22:20 +00:00
|
|
|
"The point of reference is the middle of\n"
|
|
|
|
"the bounding box for all selected objects."))
|
2019-08-07 11:29:59 +00:00
|
|
|
self.skewy_button.setMinimumWidth(90)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(self.skewy_label, 12, 0)
|
|
|
|
grid0.addWidget(self.skewy_entry, 12, 1)
|
|
|
|
grid0.addWidget(self.skewy_button, 12, 2)
|
|
|
|
|
|
|
|
self.ois_sk = OptionalInputSection(self.skew_link_cb, [self.skewy_label, self.skewy_entry, self.skewy_button],
|
|
|
|
logic=False)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-12-29 23:18:56 +00:00
|
|
|
separator_line = QtWidgets.QFrame()
|
|
|
|
separator_line.setFrameShape(QtWidgets.QFrame.HLine)
|
|
|
|
separator_line.setFrameShadow(QtWidgets.QFrame.Sunken)
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(separator_line, 14, 0, 1, 3)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-05-30 18:05:12 +00:00
|
|
|
# ## Scale Title
|
2019-01-03 19:25:08 +00:00
|
|
|
scale_title_label = QtWidgets.QLabel("<font size=3><b>%s</b></font>" % self.scaleName)
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(scale_title_label, 15, 0, 1, 2)
|
|
|
|
|
|
|
|
self.scale_link_cb = FCCheckBox()
|
|
|
|
self.scale_link_cb.setText(_("Link"))
|
|
|
|
self.scale_link_cb.setToolTip(
|
2020-06-03 18:08:06 +00:00
|
|
|
_("Link the Y entry to X entry and copy its content.")
|
2020-06-02 23:41:25 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
grid0.addWidget(self.scale_link_cb, 15, 2)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-10-05 19:01:02 +00:00
|
|
|
self.scalex_label = QtWidgets.QLabel('%s:' % _("X factor"))
|
2019-01-03 19:25:08 +00:00
|
|
|
self.scalex_label.setToolTip(
|
2019-08-19 14:24:56 +00:00
|
|
|
_("Factor for scaling on X axis.")
|
2019-01-03 19:25:08 +00:00
|
|
|
)
|
2020-02-17 02:43:01 +00:00
|
|
|
self.scalex_entry = FCDoubleSpinner(callback=self.confirmation_message)
|
2019-10-05 19:01:02 +00:00
|
|
|
# self.scalex_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
|
|
|
|
self.scalex_entry.set_precision(self.decimals)
|
|
|
|
self.scalex_entry.setMinimum(-1e6)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
self.scalex_button = FCButton(_("Scale X"))
|
2019-01-03 19:25:08 +00:00
|
|
|
self.scalex_button.setToolTip(
|
2019-03-10 12:34:13 +00:00
|
|
|
_("Scale the selected object(s).\n"
|
2019-07-16 13:22:20 +00:00
|
|
|
"The point of reference depends on \n"
|
|
|
|
"the Scale reference checkbox state."))
|
2019-08-07 11:29:59 +00:00
|
|
|
self.scalex_button.setMinimumWidth(90)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(self.scalex_label, 17, 0)
|
|
|
|
grid0.addWidget(self.scalex_entry, 17, 1)
|
|
|
|
grid0.addWidget(self.scalex_button, 17, 2)
|
2019-10-05 19:01:02 +00:00
|
|
|
|
|
|
|
self.scaley_label = QtWidgets.QLabel('%s:' % _("Y factor"))
|
2019-01-03 19:25:08 +00:00
|
|
|
self.scaley_label.setToolTip(
|
2019-08-19 14:24:56 +00:00
|
|
|
_("Factor for scaling on Y axis.")
|
2019-01-03 19:25:08 +00:00
|
|
|
)
|
2020-02-17 02:43:01 +00:00
|
|
|
self.scaley_entry = FCDoubleSpinner(callback=self.confirmation_message)
|
2019-10-05 19:01:02 +00:00
|
|
|
# self.scaley_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
|
|
|
|
self.scaley_entry.set_precision(self.decimals)
|
|
|
|
self.scaley_entry.setMinimum(-1e6)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
self.scaley_button = FCButton(_("Scale Y"))
|
2019-01-03 19:25:08 +00:00
|
|
|
self.scaley_button.setToolTip(
|
2019-03-10 12:34:13 +00:00
|
|
|
_("Scale the selected object(s).\n"
|
2019-07-16 13:22:20 +00:00
|
|
|
"The point of reference depends on \n"
|
|
|
|
"the Scale reference checkbox state."))
|
2019-08-07 11:29:59 +00:00
|
|
|
self.scaley_button.setMinimumWidth(90)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(self.scaley_label, 19, 0)
|
|
|
|
grid0.addWidget(self.scaley_entry, 19, 1)
|
|
|
|
grid0.addWidget(self.scaley_button, 19, 2)
|
2019-10-05 19:01:02 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
self.ois_s = OptionalInputSection(self.scale_link_cb,
|
|
|
|
[
|
|
|
|
self.scaley_label,
|
|
|
|
self.scaley_entry,
|
|
|
|
self.scaley_button
|
|
|
|
], logic=False)
|
|
|
|
|
|
|
|
separator_line = QtWidgets.QFrame()
|
|
|
|
separator_line.setFrameShape(QtWidgets.QFrame.HLine)
|
|
|
|
separator_line.setFrameShadow(QtWidgets.QFrame.Sunken)
|
|
|
|
grid0.addWidget(separator_line, 21, 0, 1, 3)
|
|
|
|
|
|
|
|
# ## Flip Title
|
|
|
|
flip_title_label = QtWidgets.QLabel("<font size=3><b>%s</b></font>" % self.flipName)
|
|
|
|
grid0.addWidget(flip_title_label, 23, 0, 1, 3)
|
|
|
|
|
|
|
|
self.flipx_button = FCButton(_("Flip on X"))
|
|
|
|
self.flipx_button.setToolTip(
|
|
|
|
_("Flip the selected object(s) over the X axis.")
|
2019-08-19 14:24:56 +00:00
|
|
|
)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
self.flipy_button = FCButton(_("Flip on Y"))
|
|
|
|
self.flipy_button.setToolTip(
|
|
|
|
_("Flip the selected object(s) over the X axis.")
|
|
|
|
)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
hlay0 = QtWidgets.QHBoxLayout()
|
|
|
|
grid0.addLayout(hlay0, 25, 0, 1, 3)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
hlay0.addWidget(self.flipx_button)
|
|
|
|
hlay0.addWidget(self.flipy_button)
|
2019-12-29 23:18:56 +00:00
|
|
|
|
|
|
|
separator_line = QtWidgets.QFrame()
|
|
|
|
separator_line.setFrameShape(QtWidgets.QFrame.HLine)
|
|
|
|
separator_line.setFrameShadow(QtWidgets.QFrame.Sunken)
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(separator_line, 27, 0, 1, 3)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-05-30 18:05:12 +00:00
|
|
|
# ## Offset Title
|
2019-01-03 19:25:08 +00:00
|
|
|
offset_title_label = QtWidgets.QLabel("<font size=3><b>%s</b></font>" % self.offsetName)
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(offset_title_label, 29, 0, 1, 3)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-10-05 19:01:02 +00:00
|
|
|
self.offx_label = QtWidgets.QLabel('%s:' % _("X val"))
|
2019-01-03 19:25:08 +00:00
|
|
|
self.offx_label.setToolTip(
|
2019-08-19 14:24:56 +00:00
|
|
|
_("Distance to offset on X axis. In current units.")
|
2019-01-03 19:25:08 +00:00
|
|
|
)
|
2020-02-17 02:43:01 +00:00
|
|
|
self.offx_entry = FCDoubleSpinner(callback=self.confirmation_message)
|
2019-10-05 19:01:02 +00:00
|
|
|
# self.offx_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
|
|
|
|
self.offx_entry.set_precision(self.decimals)
|
|
|
|
self.offx_entry.setMinimum(-1e6)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
self.offx_button = FCButton(_("Offset X"))
|
2019-01-03 19:25:08 +00:00
|
|
|
self.offx_button.setToolTip(
|
2019-03-10 12:34:13 +00:00
|
|
|
_("Offset the selected object(s).\n"
|
2019-07-16 13:22:20 +00:00
|
|
|
"The point of reference is the middle of\n"
|
|
|
|
"the bounding box for all selected objects.\n"))
|
2019-08-07 11:29:59 +00:00
|
|
|
self.offx_button.setMinimumWidth(90)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(self.offx_label, 31, 0)
|
|
|
|
grid0.addWidget(self.offx_entry, 31, 1)
|
|
|
|
grid0.addWidget(self.offx_button, 31, 2)
|
2019-10-05 19:01:02 +00:00
|
|
|
|
|
|
|
self.offy_label = QtWidgets.QLabel('%s:' % _("Y val"))
|
2019-01-03 19:25:08 +00:00
|
|
|
self.offy_label.setToolTip(
|
2019-08-19 14:24:56 +00:00
|
|
|
_("Distance to offset on Y axis. In current units.")
|
2019-01-03 19:25:08 +00:00
|
|
|
)
|
2020-02-17 02:43:01 +00:00
|
|
|
self.offy_entry = FCDoubleSpinner(callback=self.confirmation_message)
|
2019-10-05 19:01:02 +00:00
|
|
|
# self.offy_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
|
|
|
|
self.offy_entry.set_precision(self.decimals)
|
|
|
|
self.offy_entry.setMinimum(-1e6)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
self.offy_button = FCButton(_("Offset Y"))
|
2019-01-03 19:25:08 +00:00
|
|
|
self.offy_button.setToolTip(
|
2019-03-10 12:34:13 +00:00
|
|
|
_("Offset the selected object(s).\n"
|
2019-07-16 13:22:20 +00:00
|
|
|
"The point of reference is the middle of\n"
|
|
|
|
"the bounding box for all selected objects.\n"))
|
2019-08-07 11:29:59 +00:00
|
|
|
self.offy_button.setMinimumWidth(90)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(self.offy_label, 32, 0)
|
|
|
|
grid0.addWidget(self.offy_entry, 32, 1)
|
|
|
|
grid0.addWidget(self.offy_button, 32, 2)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-12-29 23:18:56 +00:00
|
|
|
separator_line = QtWidgets.QFrame()
|
|
|
|
separator_line.setFrameShape(QtWidgets.QFrame.HLine)
|
|
|
|
separator_line.setFrameShadow(QtWidgets.QFrame.Sunken)
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(separator_line, 34, 0, 1, 3)
|
2019-10-05 19:01:02 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
# ## Buffer Title
|
|
|
|
buffer_title_label = QtWidgets.QLabel("<font size=3><b>%s</b></font>" % self.bufferName)
|
|
|
|
grid0.addWidget(buffer_title_label, 35, 0, 1, 2)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
self.buffer_rounded_cb = FCCheckBox('%s' % _("Rounded"))
|
|
|
|
self.buffer_rounded_cb.setToolTip(
|
|
|
|
_("If checked then the buffer will surround the buffered shape,\n"
|
|
|
|
"every corner will be rounded.\n"
|
|
|
|
"If not checked then the buffer will follow the exact geometry\n"
|
|
|
|
"of the buffered shape.")
|
2019-01-03 19:25:08 +00:00
|
|
|
)
|
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(self.buffer_rounded_cb, 35, 2)
|
2019-12-23 20:59:01 +00:00
|
|
|
|
|
|
|
self.buffer_label = QtWidgets.QLabel('%s:' % _("Distance"))
|
|
|
|
self.buffer_label.setToolTip(
|
|
|
|
_("A positive value will create the effect of dilation,\n"
|
|
|
|
"while a negative value will create the effect of erosion.\n"
|
|
|
|
"Each geometry element of the object will be increased\n"
|
|
|
|
"or decreased with the 'distance'.")
|
|
|
|
)
|
|
|
|
|
2020-02-17 02:43:01 +00:00
|
|
|
self.buffer_entry = FCDoubleSpinner(callback=self.confirmation_message)
|
2019-12-23 20:59:01 +00:00
|
|
|
self.buffer_entry.set_precision(self.decimals)
|
|
|
|
self.buffer_entry.setSingleStep(0.1)
|
|
|
|
self.buffer_entry.setWrapping(True)
|
|
|
|
self.buffer_entry.set_range(-9999.9999, 9999.9999)
|
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
self.buffer_button = FCButton(_("Buffer D"))
|
2019-12-23 20:59:01 +00:00
|
|
|
self.buffer_button.setToolTip(
|
|
|
|
_("Create the buffer effect on each geometry,\n"
|
2019-12-29 23:18:56 +00:00
|
|
|
"element from the selected object, using the distance.")
|
2019-12-23 20:59:01 +00:00
|
|
|
)
|
|
|
|
self.buffer_button.setMinimumWidth(90)
|
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(self.buffer_label, 37, 0)
|
|
|
|
grid0.addWidget(self.buffer_entry, 37, 1)
|
|
|
|
grid0.addWidget(self.buffer_button, 37, 2)
|
2019-12-23 20:59:01 +00:00
|
|
|
|
2019-12-30 20:04:39 +00:00
|
|
|
self.buffer_factor_label = QtWidgets.QLabel('%s:' % _("Value"))
|
2019-12-29 23:18:56 +00:00
|
|
|
self.buffer_factor_label.setToolTip(
|
|
|
|
_("A positive value will create the effect of dilation,\n"
|
|
|
|
"while a negative value will create the effect of erosion.\n"
|
|
|
|
"Each geometry element of the object will be increased\n"
|
2019-12-30 20:04:39 +00:00
|
|
|
"or decreased to fit the 'Value'. Value is a percentage\n"
|
|
|
|
"of the initial dimension.")
|
2019-12-29 23:18:56 +00:00
|
|
|
)
|
|
|
|
|
2020-02-17 02:56:57 +00:00
|
|
|
self.buffer_factor_entry = FCDoubleSpinner(callback=self.confirmation_message, suffix='%')
|
2019-12-29 23:18:56 +00:00
|
|
|
self.buffer_factor_entry.set_range(-100.0000, 1000.0000)
|
|
|
|
self.buffer_factor_entry.set_precision(self.decimals)
|
|
|
|
self.buffer_factor_entry.setWrapping(True)
|
|
|
|
self.buffer_factor_entry.setSingleStep(1)
|
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
self.buffer_factor_button = FCButton(_("Buffer F"))
|
2019-12-29 23:18:56 +00:00
|
|
|
self.buffer_factor_button.setToolTip(
|
|
|
|
_("Create the buffer effect on each geometry,\n"
|
|
|
|
"element from the selected object, using the factor.")
|
|
|
|
)
|
|
|
|
self.buffer_factor_button.setMinimumWidth(90)
|
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(self.buffer_factor_label, 38, 0)
|
|
|
|
grid0.addWidget(self.buffer_factor_entry, 38, 1)
|
|
|
|
grid0.addWidget(self.buffer_factor_button, 38, 2)
|
2019-12-29 23:18:56 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
grid0.addWidget(QtWidgets.QLabel(''), 42, 0, 1, 3)
|
2019-12-23 20:59:01 +00:00
|
|
|
|
2020-04-22 20:00:54 +00:00
|
|
|
self.layout.addStretch()
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-02-13 19:22:21 +00:00
|
|
|
# ## Reset Tool
|
|
|
|
self.reset_button = QtWidgets.QPushButton(_("Reset Tool"))
|
2020-07-15 18:11:47 +00:00
|
|
|
self.reset_button.setIcon(QtGui.QIcon(self.app.resource_location + '/reset32.png'))
|
2020-02-13 19:22:21 +00:00
|
|
|
self.reset_button.setToolTip(
|
|
|
|
_("Will reset the tool parameters.")
|
|
|
|
)
|
|
|
|
self.reset_button.setStyleSheet("""
|
|
|
|
QPushButton
|
|
|
|
{
|
|
|
|
font-weight: bold;
|
|
|
|
}
|
|
|
|
""")
|
2020-04-22 20:00:54 +00:00
|
|
|
self.layout.addWidget(self.reset_button)
|
2020-02-13 19:22:21 +00:00
|
|
|
|
2019-05-30 18:05:12 +00:00
|
|
|
# ## Signals
|
2020-06-02 23:41:25 +00:00
|
|
|
self.ref_combo.currentIndexChanged.connect(self.on_reference_changed)
|
|
|
|
self.type_obj_combo.currentIndexChanged.connect(self.on_type_obj_index_changed)
|
|
|
|
self.point_button.clicked.connect(self.on_add_coords)
|
|
|
|
|
2019-01-03 19:25:08 +00:00
|
|
|
self.rotate_button.clicked.connect(self.on_rotate)
|
2020-06-02 23:41:25 +00:00
|
|
|
|
2019-01-03 19:25:08 +00:00
|
|
|
self.skewx_button.clicked.connect(self.on_skewx)
|
|
|
|
self.skewy_button.clicked.connect(self.on_skewy)
|
2020-06-02 23:41:25 +00:00
|
|
|
|
2019-01-03 19:25:08 +00:00
|
|
|
self.scalex_button.clicked.connect(self.on_scalex)
|
|
|
|
self.scaley_button.clicked.connect(self.on_scaley)
|
2020-06-02 23:41:25 +00:00
|
|
|
|
2019-01-03 19:25:08 +00:00
|
|
|
self.offx_button.clicked.connect(self.on_offx)
|
|
|
|
self.offy_button.clicked.connect(self.on_offy)
|
2020-06-02 23:41:25 +00:00
|
|
|
|
2019-01-03 19:25:08 +00:00
|
|
|
self.flipx_button.clicked.connect(self.on_flipx)
|
|
|
|
self.flipy_button.clicked.connect(self.on_flipy)
|
2020-06-02 23:41:25 +00:00
|
|
|
|
2019-12-29 23:18:56 +00:00
|
|
|
self.buffer_button.clicked.connect(self.on_buffer_by_distance)
|
|
|
|
self.buffer_factor_button.clicked.connect(self.on_buffer_by_factor)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-02-13 19:22:21 +00:00
|
|
|
self.reset_button.clicked.connect(self.set_tool_ui)
|
|
|
|
|
2019-03-10 15:12:58 +00:00
|
|
|
def run(self, toggle=True):
|
2020-04-29 01:46:52 +00:00
|
|
|
self.app.defaults.report_usage("ToolTransform()")
|
2019-02-03 21:08:09 +00:00
|
|
|
|
2019-03-11 20:58:27 +00:00
|
|
|
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:
|
2019-03-11 21:04:38 +00:00
|
|
|
try:
|
|
|
|
if self.app.ui.tool_scroll_area.widget().objectName() == self.toolName:
|
2019-08-23 16:30:33 +00:00
|
|
|
# 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])
|
2019-03-11 21:04:38 +00:00
|
|
|
except AttributeError:
|
|
|
|
pass
|
2019-03-13 23:09:06 +00:00
|
|
|
else:
|
|
|
|
if self.app.ui.splitter.sizes()[0] == 0:
|
|
|
|
self.app.ui.splitter.setSizes([1, 1])
|
2019-03-11 20:58:27 +00:00
|
|
|
|
2020-06-02 15:29:45 +00:00
|
|
|
AppTool.run(self)
|
2019-02-27 15:32:52 +00:00
|
|
|
self.set_tool_ui()
|
2019-02-12 02:00:11 +00:00
|
|
|
|
2019-03-10 12:34:13 +00:00
|
|
|
self.app.ui.notebook.setTabText(2, _("Transform Tool"))
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-02-02 16:26:01 +00:00
|
|
|
def install(self, icon=None, separator=None, **kwargs):
|
2020-06-02 15:29:45 +00:00
|
|
|
AppTool.install(self, icon, separator, shortcut='Alt+T', **kwargs)
|
2019-02-02 16:26:01 +00:00
|
|
|
|
|
|
|
def set_tool_ui(self):
|
2020-02-17 12:27:45 +00:00
|
|
|
|
2019-05-30 18:05:12 +00:00
|
|
|
# ## Initialize form
|
2020-06-02 23:41:25 +00:00
|
|
|
self.ref_combo.set_value(self.app.defaults["tools_transform_reference"])
|
|
|
|
self.type_obj_combo.set_value(self.app.defaults["tools_transform_ref_object"])
|
|
|
|
self.point_entry.set_value(self.app.defaults["tools_transform_ref_point"])
|
|
|
|
self.rotate_entry.set_value(self.app.defaults["tools_transform_rotate"])
|
|
|
|
|
|
|
|
self.skewx_entry.set_value(self.app.defaults["tools_transform_skew_x"])
|
|
|
|
self.skewy_entry.set_value(self.app.defaults["tools_transform_skew_y"])
|
|
|
|
self.skew_link_cb.set_value(self.app.defaults["tools_transform_skew_link"])
|
|
|
|
|
|
|
|
self.scalex_entry.set_value(self.app.defaults["tools_transform_scale_x"])
|
|
|
|
self.scaley_entry.set_value(self.app.defaults["tools_transform_scale_y"])
|
|
|
|
self.scale_link_cb.set_value(self.app.defaults["tools_transform_scale_link"])
|
|
|
|
|
|
|
|
self.offx_entry.set_value(self.app.defaults["tools_transform_offset_x"])
|
|
|
|
self.offy_entry.set_value(self.app.defaults["tools_transform_offset_y"])
|
|
|
|
|
|
|
|
self.buffer_entry.set_value(self.app.defaults["tools_transform_buffer_dis"])
|
|
|
|
self.buffer_factor_entry.set_value(self.app.defaults["tools_transform_buffer_factor"])
|
|
|
|
self.buffer_rounded_cb.set_value(self.app.defaults["tools_transform_buffer_corner"])
|
|
|
|
|
|
|
|
# initial state is hidden
|
|
|
|
self.point_label.hide()
|
|
|
|
self.point_entry.hide()
|
|
|
|
self.point_button.hide()
|
|
|
|
|
|
|
|
self.type_object_label.hide()
|
|
|
|
self.type_obj_combo.hide()
|
|
|
|
self.object_combo.hide()
|
|
|
|
|
|
|
|
def on_type_obj_index_changed(self, index):
|
|
|
|
self.object_combo.setRootModelIndex(self.app.collection.index(index, 0, QtCore.QModelIndex()))
|
|
|
|
self.object_combo.setCurrentIndex(0)
|
|
|
|
self.object_combo.obj_type = {
|
|
|
|
_("Gerber"): "Gerber", _("Excellon"): "Excellon", _("Geometry"): "Geometry"
|
|
|
|
}[self.type_obj_combo.get_value()]
|
|
|
|
|
|
|
|
def on_reference_changed(self, index):
|
|
|
|
if index == 0 or index == 1: # "Origin" or "Selection" reference
|
|
|
|
self.point_label.hide()
|
|
|
|
self.point_entry.hide()
|
|
|
|
self.point_button.hide()
|
|
|
|
|
|
|
|
self.type_object_label.hide()
|
|
|
|
self.type_obj_combo.hide()
|
|
|
|
self.object_combo.hide()
|
|
|
|
elif index == 2: # "Point" reference
|
|
|
|
self.point_label.show()
|
|
|
|
self.point_entry.show()
|
|
|
|
self.point_button.show()
|
|
|
|
|
|
|
|
self.type_object_label.hide()
|
|
|
|
self.type_obj_combo.hide()
|
|
|
|
self.object_combo.hide()
|
|
|
|
else: # "Object" reference
|
|
|
|
self.point_label.hide()
|
|
|
|
self.point_entry.hide()
|
|
|
|
self.point_button.hide()
|
|
|
|
|
|
|
|
self.type_object_label.show()
|
|
|
|
self.type_obj_combo.show()
|
|
|
|
self.object_combo.show()
|
|
|
|
|
|
|
|
def on_calculate_reference(self):
|
|
|
|
ref_val = self.ref_combo.currentIndex()
|
|
|
|
|
|
|
|
if ref_val == 0: # "Origin" reference
|
|
|
|
return 0, 0
|
|
|
|
elif ref_val == 1: # "Selection" reference
|
|
|
|
sel_list = self.app.collection.get_selected()
|
|
|
|
if sel_list:
|
|
|
|
xmin, ymin, xmax, ymax = self.alt_bounds(obj_list=sel_list)
|
|
|
|
px = (xmax + xmin) * 0.5
|
|
|
|
py = (ymax + ymin) * 0.5
|
|
|
|
return px, py
|
|
|
|
else:
|
|
|
|
self.app.inform.emit('[ERROR_NOTCL] %s' % _("No object selected."))
|
|
|
|
return "fail"
|
|
|
|
elif ref_val == 2: # "Point" reference
|
|
|
|
point_val = self.point_entry.get_value()
|
|
|
|
try:
|
|
|
|
px, py = eval('{}'.format(point_val))
|
|
|
|
return px, py
|
|
|
|
except Exception:
|
|
|
|
self.app.inform.emit('[WARNING_NOTCL] %s' % _("Incorrect format for Point value. Needs format X,Y"))
|
|
|
|
return "fail"
|
|
|
|
else: # "Object" reference
|
|
|
|
obj_name = self.object_combo.get_value()
|
|
|
|
ref_obj = self.app.collection.get_by_name(obj_name)
|
|
|
|
xmin, ymin, xmax, ymax = ref_obj.bounds()
|
|
|
|
px = (xmax + xmin) * 0.5
|
|
|
|
py = (ymax + ymin) * 0.5
|
|
|
|
return px, py
|
|
|
|
|
|
|
|
def on_add_coords(self):
|
|
|
|
val = self.app.clipboard.text()
|
|
|
|
self.point_entry.set_value(val)
|
2019-12-23 20:59:01 +00:00
|
|
|
|
2019-01-03 19:25:08 +00:00
|
|
|
def on_rotate(self):
|
2019-10-05 19:01:02 +00:00
|
|
|
value = float(self.rotate_entry.get_value())
|
|
|
|
if value == 0:
|
2020-06-02 23:41:25 +00:00
|
|
|
self.app.inform.emit('[WARNING_NOTCL] %s' % _("Rotate transformation can not be done for a value of 0."))
|
|
|
|
return
|
|
|
|
point = self.on_calculate_reference()
|
|
|
|
if point == 'fail':
|
|
|
|
return
|
|
|
|
self.app.worker_task.emit({'fcn': self.on_rotate_action, 'params': [value, point]})
|
2019-01-03 19:25:08 +00:00
|
|
|
|
|
|
|
def on_flipx(self):
|
|
|
|
axis = 'Y'
|
2020-06-02 23:41:25 +00:00
|
|
|
point = self.on_calculate_reference()
|
|
|
|
if point == 'fail':
|
|
|
|
return
|
|
|
|
self.app.worker_task.emit({'fcn': self.on_flip, 'params': [axis, point]})
|
2019-01-03 19:25:08 +00:00
|
|
|
|
|
|
|
def on_flipy(self):
|
|
|
|
axis = 'X'
|
2020-06-02 23:41:25 +00:00
|
|
|
point = self.on_calculate_reference()
|
|
|
|
if point == 'fail':
|
|
|
|
return
|
|
|
|
self.app.worker_task.emit({'fcn': self.on_flip, 'params': [axis, point]})
|
2019-10-05 19:01:02 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
def on_skewx(self):
|
|
|
|
xvalue = float(self.skewx_entry.get_value())
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
if xvalue == 0:
|
|
|
|
return
|
|
|
|
|
|
|
|
if self.skew_link_cb.get_value():
|
|
|
|
yvalue = xvalue
|
|
|
|
else:
|
|
|
|
yvalue = 0
|
2019-01-03 19:25:08 +00:00
|
|
|
|
|
|
|
axis = 'X'
|
2020-06-02 23:41:25 +00:00
|
|
|
point = self.on_calculate_reference()
|
|
|
|
if point == 'fail':
|
|
|
|
return
|
2019-10-05 19:01:02 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
self.app.worker_task.emit({'fcn': self.on_skew, 'params': [axis, xvalue, yvalue, point]})
|
2019-01-03 19:25:08 +00:00
|
|
|
|
|
|
|
def on_skewy(self):
|
2020-06-02 23:41:25 +00:00
|
|
|
xvalue = 0
|
|
|
|
yvalue = float(self.skewy_entry.get_value())
|
|
|
|
|
|
|
|
if yvalue == 0:
|
|
|
|
return
|
|
|
|
|
2019-01-03 19:25:08 +00:00
|
|
|
axis = 'Y'
|
2020-06-02 23:41:25 +00:00
|
|
|
point = self.on_calculate_reference()
|
|
|
|
if point == 'fail':
|
|
|
|
return
|
2019-10-05 19:01:02 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
self.app.worker_task.emit({'fcn': self.on_skew, 'params': [axis, xvalue, yvalue, point]})
|
2019-01-03 19:25:08 +00:00
|
|
|
|
|
|
|
def on_scalex(self):
|
2019-10-05 19:01:02 +00:00
|
|
|
xvalue = float(self.scalex_entry.get_value())
|
|
|
|
|
|
|
|
if xvalue == 0 or xvalue == 1:
|
|
|
|
self.app.inform.emit('[WARNING_NOTCL] %s' %
|
|
|
|
_("Scale transformation can not be done for a factor of 0 or 1."))
|
|
|
|
return
|
|
|
|
|
2019-01-03 19:25:08 +00:00
|
|
|
if self.scale_link_cb.get_value():
|
|
|
|
yvalue = xvalue
|
|
|
|
else:
|
|
|
|
yvalue = 1
|
|
|
|
|
|
|
|
axis = 'X'
|
2020-06-02 23:41:25 +00:00
|
|
|
point = self.on_calculate_reference()
|
|
|
|
if point == 'fail':
|
|
|
|
return
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
self.app.worker_task.emit({'fcn': self.on_scale, 'params': [axis, xvalue, yvalue, point]})
|
2019-01-03 19:25:08 +00:00
|
|
|
|
|
|
|
def on_scaley(self):
|
|
|
|
xvalue = 1
|
2019-10-05 19:01:02 +00:00
|
|
|
yvalue = float(self.scaley_entry.get_value())
|
|
|
|
|
|
|
|
if yvalue == 0 or yvalue == 1:
|
|
|
|
self.app.inform.emit('[WARNING_NOTCL] %s' %
|
|
|
|
_("Scale transformation can not be done for a factor of 0 or 1."))
|
|
|
|
return
|
2019-01-03 19:25:08 +00:00
|
|
|
|
|
|
|
axis = 'Y'
|
2020-06-02 23:41:25 +00:00
|
|
|
point = self.on_calculate_reference()
|
|
|
|
if point == 'fail':
|
|
|
|
return
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
self.app.worker_task.emit({'fcn': self.on_scale, 'params': [axis, xvalue, yvalue, point]})
|
2019-01-03 19:25:08 +00:00
|
|
|
|
|
|
|
def on_offx(self):
|
2019-10-05 19:01:02 +00:00
|
|
|
value = float(self.offx_entry.get_value())
|
|
|
|
if value == 0:
|
2019-12-23 20:59:01 +00:00
|
|
|
self.app.inform.emit('[WARNING_NOTCL] %s' % _("Offset transformation can not be done for a value of 0."))
|
2019-10-05 19:01:02 +00:00
|
|
|
return
|
2019-01-03 19:25:08 +00:00
|
|
|
axis = 'X'
|
2019-10-05 19:01:02 +00:00
|
|
|
|
|
|
|
self.app.worker_task.emit({'fcn': self.on_offset, 'params': [axis, value]})
|
2019-01-03 19:25:08 +00:00
|
|
|
|
|
|
|
def on_offy(self):
|
2019-10-05 19:01:02 +00:00
|
|
|
value = float(self.offy_entry.get_value())
|
|
|
|
if value == 0:
|
2019-12-23 20:59:01 +00:00
|
|
|
self.app.inform.emit('[WARNING_NOTCL] %s' % _("Offset transformation can not be done for a value of 0."))
|
2019-10-05 19:01:02 +00:00
|
|
|
return
|
2019-01-03 19:25:08 +00:00
|
|
|
axis = 'Y'
|
2019-10-05 19:01:02 +00:00
|
|
|
|
|
|
|
self.app.worker_task.emit({'fcn': self.on_offset, 'params': [axis, value]})
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-12-29 23:18:56 +00:00
|
|
|
def on_buffer_by_distance(self):
|
2019-12-23 20:59:01 +00:00
|
|
|
value = self.buffer_entry.get_value()
|
|
|
|
join = 1 if self.buffer_rounded_cb.get_value() else 2
|
|
|
|
|
|
|
|
self.app.worker_task.emit({'fcn': self.on_buffer_action, 'params': [value, join]})
|
|
|
|
|
2019-12-29 23:18:56 +00:00
|
|
|
def on_buffer_by_factor(self):
|
2020-06-03 17:35:59 +00:00
|
|
|
value = 1 + self.buffer_factor_entry.get_value() / 100.0
|
2019-12-29 23:18:56 +00:00
|
|
|
join = 1 if self.buffer_rounded_cb.get_value() else 2
|
|
|
|
|
|
|
|
# tell the buffer method to use the factor
|
|
|
|
factor = True
|
|
|
|
|
|
|
|
self.app.worker_task.emit({'fcn': self.on_buffer_action, 'params': [value, join, factor]})
|
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
def on_rotate_action(self, num, point):
|
2019-01-03 19:25:08 +00:00
|
|
|
obj_list = self.app.collection.get_selected()
|
|
|
|
|
|
|
|
if not obj_list:
|
2019-12-29 23:18:56 +00:00
|
|
|
self.app.inform.emit('[WARNING_NOTCL] %s' % _("No object selected. Please Select an object to rotate!"))
|
2019-01-03 19:25:08 +00:00
|
|
|
return
|
|
|
|
else:
|
2019-03-10 12:34:13 +00:00
|
|
|
with self.app.proc_container.new(_("Appying Rotate")):
|
2019-01-03 19:25:08 +00:00
|
|
|
try:
|
2020-06-02 23:41:25 +00:00
|
|
|
px, py = point
|
2019-01-03 19:25:08 +00:00
|
|
|
for sel_obj in obj_list:
|
2020-04-27 07:03:22 +00:00
|
|
|
if sel_obj.kind == 'cncjob':
|
2019-03-10 12:34:13 +00:00
|
|
|
self.app.inform.emit(_("CNCJob objects can't be rotated."))
|
2019-01-03 19:25:08 +00:00
|
|
|
else:
|
|
|
|
sel_obj.rotate(-num, point=(px, py))
|
2020-06-02 15:29:45 +00:00
|
|
|
self.app.app_obj.object_changed.emit(sel_obj)
|
2019-01-03 19:25:08 +00:00
|
|
|
|
|
|
|
# add information to the object that it was changed and how much
|
|
|
|
sel_obj.options['rotate'] = num
|
2019-05-10 23:57:06 +00:00
|
|
|
sel_obj.plot()
|
2019-09-06 18:02:08 +00:00
|
|
|
self.app.inform.emit('[success] %s...' % _('Rotate done'))
|
2019-01-03 19:25:08 +00:00
|
|
|
except Exception as e:
|
2019-09-06 18:02:08 +00:00
|
|
|
self.app.inform.emit('[ERROR_NOTCL] %s %s, %s.' %
|
|
|
|
(_("Due of"), str(e), _("action was not executed.")))
|
2019-01-03 19:25:08 +00:00
|
|
|
return
|
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
def on_flip(self, axis, point):
|
2019-01-03 19:25:08 +00:00
|
|
|
obj_list = self.app.collection.get_selected()
|
|
|
|
|
|
|
|
if not obj_list:
|
2020-06-02 23:41:25 +00:00
|
|
|
self.app.inform.emit('[WARNING_NOTCL] %s!' % _("No object selected. Please Select an object to flip"))
|
2019-01-03 19:25:08 +00:00
|
|
|
return
|
|
|
|
else:
|
2019-03-10 12:34:13 +00:00
|
|
|
with self.app.proc_container.new(_("Applying Flip")):
|
2019-01-03 19:25:08 +00:00
|
|
|
try:
|
2020-06-02 23:41:25 +00:00
|
|
|
px, py = point
|
2019-01-03 19:25:08 +00:00
|
|
|
|
|
|
|
# execute mirroring
|
2019-05-10 23:57:06 +00:00
|
|
|
for sel_obj in obj_list:
|
2020-04-27 07:03:22 +00:00
|
|
|
if sel_obj.kind == 'cncjob':
|
2019-03-10 12:34:13 +00:00
|
|
|
self.app.inform.emit(_("CNCJob objects can't be mirrored/flipped."))
|
2019-01-03 19:25:08 +00:00
|
|
|
else:
|
2020-04-01 15:45:20 +00:00
|
|
|
if axis == 'X':
|
2019-05-10 23:57:06 +00:00
|
|
|
sel_obj.mirror('X', (px, py))
|
2019-01-03 19:25:08 +00:00
|
|
|
# add information to the object that it was changed and how much
|
|
|
|
# the axis is reversed because of the reference
|
2019-05-10 23:57:06 +00:00
|
|
|
if 'mirror_y' in sel_obj.options:
|
|
|
|
sel_obj.options['mirror_y'] = not sel_obj.options['mirror_y']
|
2019-01-03 19:25:08 +00:00
|
|
|
else:
|
2019-05-10 23:57:06 +00:00
|
|
|
sel_obj.options['mirror_y'] = True
|
2020-06-02 23:41:25 +00:00
|
|
|
self.app.inform.emit('[success] %s...' % _('Flip on the Y axis done'))
|
2020-04-01 15:45:20 +00:00
|
|
|
elif axis == 'Y':
|
2019-05-10 23:57:06 +00:00
|
|
|
sel_obj.mirror('Y', (px, py))
|
2019-01-03 19:25:08 +00:00
|
|
|
# add information to the object that it was changed and how much
|
|
|
|
# the axis is reversed because of the reference
|
2019-05-10 23:57:06 +00:00
|
|
|
if 'mirror_x' in sel_obj.options:
|
|
|
|
sel_obj.options['mirror_x'] = not sel_obj.options['mirror_x']
|
2019-01-03 19:25:08 +00:00
|
|
|
else:
|
2019-05-10 23:57:06 +00:00
|
|
|
sel_obj.options['mirror_x'] = True
|
2020-01-02 23:41:03 +00:00
|
|
|
self.app.inform.emit('[success] %s...' % _('Flip on the X axis done'))
|
2020-06-02 15:29:45 +00:00
|
|
|
self.app.app_obj.object_changed.emit(sel_obj)
|
2019-05-10 23:57:06 +00:00
|
|
|
sel_obj.plot()
|
2019-01-03 19:25:08 +00:00
|
|
|
except Exception as e:
|
2019-09-06 18:02:08 +00:00
|
|
|
self.app.inform.emit('[ERROR_NOTCL] %s %s, %s.' %
|
|
|
|
(_("Due of"), str(e), _("action was not executed.")))
|
2019-01-03 19:25:08 +00:00
|
|
|
return
|
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
def on_skew(self, axis, xvalue, yvalue, point):
|
2019-01-03 19:25:08 +00:00
|
|
|
obj_list = self.app.collection.get_selected()
|
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
if xvalue in [90, 180] or yvalue in [90, 180] or xvalue == yvalue == 0:
|
2019-10-05 19:01:02 +00:00
|
|
|
self.app.inform.emit('[WARNING_NOTCL] %s' %
|
|
|
|
_("Skew transformation can not be done for 0, 90 and 180 degrees."))
|
|
|
|
return
|
|
|
|
|
2019-01-03 19:25:08 +00:00
|
|
|
if not obj_list:
|
2019-09-06 18:02:08 +00:00
|
|
|
self.app.inform.emit('[WARNING_NOTCL] %s' %
|
|
|
|
_("No object selected. Please Select an object to shear/skew!"))
|
2019-01-03 19:25:08 +00:00
|
|
|
return
|
|
|
|
else:
|
2019-03-10 12:34:13 +00:00
|
|
|
with self.app.proc_container.new(_("Applying Skew")):
|
2019-01-03 19:25:08 +00:00
|
|
|
try:
|
2020-06-02 23:41:25 +00:00
|
|
|
px, py = point
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-05-10 23:57:06 +00:00
|
|
|
for sel_obj in obj_list:
|
2020-04-27 07:03:22 +00:00
|
|
|
if sel_obj.kind == 'cncjob':
|
2019-03-10 12:34:13 +00:00
|
|
|
self.app.inform.emit(_("CNCJob objects can't be skewed."))
|
2019-01-03 19:25:08 +00:00
|
|
|
else:
|
2020-06-02 23:41:25 +00:00
|
|
|
sel_obj.skew(xvalue, yvalue, point=(px, py))
|
|
|
|
# add information to the object that it was changed and how much
|
|
|
|
sel_obj.options['skew_x'] = xvalue
|
|
|
|
sel_obj.options['skew_y'] = yvalue
|
2020-06-02 15:29:45 +00:00
|
|
|
self.app.app_obj.object_changed.emit(sel_obj)
|
2019-05-10 23:57:06 +00:00
|
|
|
sel_obj.plot()
|
2020-01-02 23:41:03 +00:00
|
|
|
self.app.inform.emit('[success] %s %s %s...' % (_('Skew on the'), str(axis), _("axis done")))
|
2019-01-03 19:25:08 +00:00
|
|
|
except Exception as e:
|
2019-09-06 18:02:08 +00:00
|
|
|
self.app.inform.emit('[ERROR_NOTCL] %s %s, %s.' %
|
|
|
|
(_("Due of"), str(e), _("action was not executed.")))
|
2019-01-03 19:25:08 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
def on_scale(self, axis, xfactor, yfactor, point=None):
|
|
|
|
obj_list = self.app.collection.get_selected()
|
|
|
|
|
|
|
|
if not obj_list:
|
2020-04-27 07:03:22 +00:00
|
|
|
self.app.inform.emit('[WARNING_NOTCL] %s' % _("No object selected. Please Select an object to scale!"))
|
2019-01-03 19:25:08 +00:00
|
|
|
return
|
|
|
|
else:
|
2019-03-10 12:34:13 +00:00
|
|
|
with self.app.proc_container.new(_("Applying Scale")):
|
2019-01-03 19:25:08 +00:00
|
|
|
try:
|
2020-06-02 23:41:25 +00:00
|
|
|
px, py = point
|
2019-01-03 19:25:08 +00:00
|
|
|
|
2019-05-10 23:57:06 +00:00
|
|
|
for sel_obj in obj_list:
|
2020-04-27 07:03:22 +00:00
|
|
|
if sel_obj.kind == 'cncjob':
|
2019-03-10 12:34:13 +00:00
|
|
|
self.app.inform.emit(_("CNCJob objects can't be scaled."))
|
2019-01-03 19:25:08 +00:00
|
|
|
else:
|
2019-05-10 23:57:06 +00:00
|
|
|
sel_obj.scale(xfactor, yfactor, point=(px, py))
|
2019-01-03 19:25:08 +00:00
|
|
|
# add information to the object that it was changed and how much
|
2019-05-10 23:57:06 +00:00
|
|
|
sel_obj.options['scale_x'] = xfactor
|
|
|
|
sel_obj.options['scale_y'] = yfactor
|
2020-06-02 15:29:45 +00:00
|
|
|
self.app.app_obj.object_changed.emit(sel_obj)
|
2019-05-10 23:57:06 +00:00
|
|
|
sel_obj.plot()
|
|
|
|
|
2020-04-27 07:03:22 +00:00
|
|
|
self.app.inform.emit('[success] %s %s %s...' % (_('Scale on the'), str(axis), _('axis done')))
|
2019-01-03 19:25:08 +00:00
|
|
|
except Exception as e:
|
2019-09-06 18:02:08 +00:00
|
|
|
self.app.inform.emit('[ERROR_NOTCL] %s %s, %s.' %
|
|
|
|
(_("Due of"), str(e), _("action was not executed.")))
|
2019-01-03 19:25:08 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
def on_offset(self, axis, num):
|
|
|
|
obj_list = self.app.collection.get_selected()
|
|
|
|
|
|
|
|
if not obj_list:
|
2020-04-27 07:03:22 +00:00
|
|
|
self.app.inform.emit('[WARNING_NOTCL] %s' % _("No object selected. Please Select an object to offset!"))
|
2019-01-03 19:25:08 +00:00
|
|
|
return
|
|
|
|
else:
|
2019-03-10 12:34:13 +00:00
|
|
|
with self.app.proc_container.new(_("Applying Offset")):
|
2019-01-03 19:25:08 +00:00
|
|
|
try:
|
2019-05-10 23:57:06 +00:00
|
|
|
for sel_obj in obj_list:
|
2020-04-27 07:03:22 +00:00
|
|
|
if sel_obj.kind == 'cncjob':
|
2019-09-06 18:02:08 +00:00
|
|
|
self.app.inform.emit(_("CNCJob objects can't be offset."))
|
2019-01-03 19:25:08 +00:00
|
|
|
else:
|
2020-04-01 15:45:20 +00:00
|
|
|
if axis == 'X':
|
2019-05-10 23:57:06 +00:00
|
|
|
sel_obj.offset((num, 0))
|
2019-01-03 19:25:08 +00:00
|
|
|
# add information to the object that it was changed and how much
|
2019-05-10 23:57:06 +00:00
|
|
|
sel_obj.options['offset_x'] = num
|
2020-04-01 15:45:20 +00:00
|
|
|
elif axis == 'Y':
|
2019-05-10 23:57:06 +00:00
|
|
|
sel_obj.offset((0, num))
|
2019-01-03 19:25:08 +00:00
|
|
|
# add information to the object that it was changed and how much
|
2019-05-10 23:57:06 +00:00
|
|
|
sel_obj.options['offset_y'] = num
|
2020-06-02 15:29:45 +00:00
|
|
|
self.app.app_obj.object_changed.emit(sel_obj)
|
2019-05-10 23:57:06 +00:00
|
|
|
sel_obj.plot()
|
|
|
|
|
2020-04-27 07:03:22 +00:00
|
|
|
self.app.inform.emit('[success] %s %s %s...' % (_('Offset on the'), str(axis), _('axis done')))
|
2019-01-03 19:25:08 +00:00
|
|
|
except Exception as e:
|
2020-06-03 17:35:59 +00:00
|
|
|
self.app.inform.emit('[ERROR_NOTCL] %s: %s.' %
|
|
|
|
(_("Action was not executed, due of"), str(e)))
|
2019-01-03 19:25:08 +00:00
|
|
|
return
|
|
|
|
|
2019-12-29 23:18:56 +00:00
|
|
|
def on_buffer_action(self, value, join, factor=None):
|
2019-12-23 20:59:01 +00:00
|
|
|
obj_list = self.app.collection.get_selected()
|
|
|
|
|
|
|
|
if not obj_list:
|
|
|
|
self.app.inform.emit('[WARNING_NOTCL] %s' % _("No object selected. Please Select an object to buffer!"))
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
with self.app.proc_container.new(_("Applying Buffer")):
|
|
|
|
try:
|
|
|
|
for sel_obj in obj_list:
|
2020-04-27 07:03:22 +00:00
|
|
|
if sel_obj.kind == 'cncjob':
|
2019-12-23 20:59:01 +00:00
|
|
|
self.app.inform.emit(_("CNCJob objects can't be buffered."))
|
|
|
|
elif sel_obj.kind.lower() == 'gerber':
|
2019-12-29 23:18:56 +00:00
|
|
|
sel_obj.buffer(value, join, factor)
|
2019-12-23 20:59:01 +00:00
|
|
|
sel_obj.source_file = self.app.export_gerber(obj_name=sel_obj.options['name'],
|
|
|
|
filename=None, local_use=sel_obj,
|
|
|
|
use_thread=False)
|
|
|
|
elif sel_obj.kind.lower() == 'excellon':
|
2019-12-29 23:18:56 +00:00
|
|
|
sel_obj.buffer(value, join, factor)
|
2019-12-23 20:59:01 +00:00
|
|
|
sel_obj.source_file = self.app.export_excellon(obj_name=sel_obj.options['name'],
|
|
|
|
filename=None, local_use=sel_obj,
|
|
|
|
use_thread=False)
|
|
|
|
elif sel_obj.kind.lower() == 'geometry':
|
2019-12-29 23:18:56 +00:00
|
|
|
sel_obj.buffer(value, join, factor)
|
2019-12-23 20:59:01 +00:00
|
|
|
|
2020-06-02 15:29:45 +00:00
|
|
|
self.app.app_obj.object_changed.emit(sel_obj)
|
2019-12-23 20:59:01 +00:00
|
|
|
sel_obj.plot()
|
|
|
|
|
|
|
|
self.app.inform.emit('[success] %s...' % _('Buffer done'))
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
self.app.log.debug("ToolTransform.on_buffer_action() --> %s" % str(e))
|
2020-06-03 17:35:59 +00:00
|
|
|
self.app.inform.emit('[ERROR_NOTCL] %s: %s.' %
|
|
|
|
(_("Action was not executed, due of"), str(e)))
|
2019-12-23 20:59:01 +00:00
|
|
|
return
|
|
|
|
|
2020-06-02 23:41:25 +00:00
|
|
|
@staticmethod
|
|
|
|
def alt_bounds(obj_list):
|
|
|
|
"""
|
|
|
|
Returns coordinates of rectangular bounds
|
|
|
|
of an object with geometry: (xmin, ymin, xmax, ymax).
|
|
|
|
"""
|
|
|
|
|
|
|
|
def bounds_rec(lst):
|
|
|
|
minx = np.Inf
|
|
|
|
miny = np.Inf
|
|
|
|
maxx = -np.Inf
|
|
|
|
maxy = -np.Inf
|
|
|
|
|
|
|
|
try:
|
|
|
|
for obj in lst:
|
|
|
|
if obj.kind != 'cncjob':
|
|
|
|
minx_, miny_, maxx_, maxy_ = bounds_rec(obj)
|
|
|
|
minx = min(minx, minx_)
|
|
|
|
miny = min(miny, miny_)
|
|
|
|
maxx = max(maxx, maxx_)
|
|
|
|
maxy = max(maxy, maxy_)
|
|
|
|
return minx, miny, maxx, maxy
|
|
|
|
except TypeError:
|
|
|
|
# it's an object, return it's bounds
|
|
|
|
return lst.bounds()
|
|
|
|
|
|
|
|
return bounds_rec(obj_list)
|
|
|
|
|
2019-03-07 15:37:38 +00:00
|
|
|
# end of file
|