ToolCalculators.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370
  1. # ##########################################################
  2. # FlatCAM: 2D Post-processing for Manufacturing #
  3. # File Author: Marius Adrian Stanciu (c) #
  4. # Date: 3/10/2019 #
  5. # MIT Licence #
  6. # ##########################################################
  7. from PyQt5 import QtWidgets
  8. from FlatCAMTool import FlatCAMTool
  9. from flatcamGUI.GUIElements import FCSpinner, FCDoubleSpinner, FCEntry
  10. import math
  11. import gettext
  12. import FlatCAMTranslation as fcTranslate
  13. import builtins
  14. fcTranslate.apply_language('strings')
  15. if '_' not in builtins.__dict__:
  16. _ = gettext.gettext
  17. class ToolCalculator(FlatCAMTool):
  18. toolName = _("Calculators")
  19. v_shapeName = _("V-Shape Tool Calculator")
  20. unitsName = _("Units Calculator")
  21. eplateName = _("ElectroPlating Calculator")
  22. def __init__(self, app):
  23. FlatCAMTool.__init__(self, app)
  24. self.app = app
  25. self.decimals = self.app.decimals
  26. # ## Title
  27. title_label = QtWidgets.QLabel("%s" % self.toolName)
  28. title_label.setStyleSheet("""
  29. QLabel
  30. {
  31. font-size: 16px;
  32. font-weight: bold;
  33. }
  34. """)
  35. self.layout.addWidget(title_label)
  36. # #####################
  37. # ## Units Calculator #
  38. # #####################
  39. self.unists_spacer_label = QtWidgets.QLabel(" ")
  40. self.layout.addWidget(self.unists_spacer_label)
  41. # ## Title of the Units Calculator
  42. units_label = QtWidgets.QLabel("<font size=3><b>%s</b></font>" % self.unitsName)
  43. self.layout.addWidget(units_label)
  44. # Grid Layout
  45. grid_units_layout = QtWidgets.QGridLayout()
  46. self.layout.addLayout(grid_units_layout)
  47. inch_label = QtWidgets.QLabel(_("INCH"))
  48. mm_label = QtWidgets.QLabel(_("MM"))
  49. grid_units_layout.addWidget(mm_label, 0, 0)
  50. grid_units_layout.addWidget(inch_label, 0, 1)
  51. self.inch_entry = FCEntry()
  52. # self.inch_entry.setFixedWidth(70)
  53. # self.inch_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
  54. self.inch_entry.setToolTip(_("Here you enter the value to be converted from INCH to MM"))
  55. self.mm_entry = FCEntry()
  56. # self.mm_entry.setFixedWidth(130)
  57. # self.mm_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
  58. self.mm_entry.setToolTip(_("Here you enter the value to be converted from MM to INCH"))
  59. grid_units_layout.addWidget(self.mm_entry, 1, 0)
  60. grid_units_layout.addWidget(self.inch_entry, 1, 1)
  61. # ##############################
  62. # ## V-shape Tool Calculator ###
  63. # ##############################
  64. self.v_shape_spacer_label = QtWidgets.QLabel(" ")
  65. self.layout.addWidget(self.v_shape_spacer_label)
  66. # ## Title of the V-shape Tools Calculator
  67. v_shape_title_label = QtWidgets.QLabel("<font size=3><b>%s</b></font>" % self.v_shapeName)
  68. self.layout.addWidget(v_shape_title_label)
  69. # ## Form Layout
  70. form_layout = QtWidgets.QFormLayout()
  71. self.layout.addLayout(form_layout)
  72. self.tipDia_label = QtWidgets.QLabel('%s:' % _("Tip Diameter"))
  73. self.tipDia_entry = FCDoubleSpinner(callback=self.confirmation_message)
  74. self.tipDia_entry.set_precision(self.decimals)
  75. self.tipDia_entry.set_range(0.0, 9999.9999)
  76. self.tipDia_entry.setSingleStep(0.1)
  77. # self.tipDia_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
  78. self.tipDia_label.setToolTip(
  79. _("This is the tool tip diameter.\n"
  80. "It is specified by manufacturer.")
  81. )
  82. self.tipAngle_label = QtWidgets.QLabel('%s:' % _("Tip Angle"))
  83. self.tipAngle_entry = FCSpinner(callback=self.confirmation_message_int)
  84. self.tipAngle_entry.set_range(0,180)
  85. self.tipAngle_entry.set_step(5)
  86. # self.tipAngle_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
  87. self.tipAngle_label.setToolTip(_("This is the angle of the tip of the tool.\n"
  88. "It is specified by manufacturer."))
  89. self.cutDepth_label = QtWidgets.QLabel('%s:' % _("Cut Z"))
  90. self.cutDepth_entry = FCDoubleSpinner(callback=self.confirmation_message)
  91. self.cutDepth_entry.set_range(-9999.9999, 9999.9999)
  92. self.cutDepth_entry.set_precision(self.decimals)
  93. # self.cutDepth_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
  94. self.cutDepth_label.setToolTip(_("This is the depth to cut into the material.\n"
  95. "In the CNCJob is the CutZ parameter."))
  96. self.effectiveToolDia_label = QtWidgets.QLabel('%s:' % _("Tool Diameter"))
  97. self.effectiveToolDia_entry = FCDoubleSpinner(callback=self.confirmation_message)
  98. self.effectiveToolDia_entry.set_precision(self.decimals)
  99. # self.effectiveToolDia_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
  100. self.effectiveToolDia_label.setToolTip(_("This is the tool diameter to be entered into\n"
  101. "FlatCAM Gerber section.\n"
  102. "In the CNCJob section it is called >Tool dia<."))
  103. # self.effectiveToolDia_entry.setEnabled(False)
  104. form_layout.addRow(self.tipDia_label, self.tipDia_entry)
  105. form_layout.addRow(self.tipAngle_label, self.tipAngle_entry)
  106. form_layout.addRow(self.cutDepth_label, self.cutDepth_entry)
  107. form_layout.addRow(self.effectiveToolDia_label, self.effectiveToolDia_entry)
  108. # ## Buttons
  109. self.calculate_vshape_button = QtWidgets.QPushButton(_("Calculate"))
  110. # self.calculate_button.setFixedWidth(70)
  111. self.calculate_vshape_button.setToolTip(
  112. _("Calculate either the Cut Z or the effective tool diameter,\n "
  113. "depending on which is desired and which is known. ")
  114. )
  115. self.layout.addWidget(self.calculate_vshape_button)
  116. # ####################################
  117. # ## ElectroPlating Tool Calculator ##
  118. # ####################################
  119. self.plate_spacer_label = QtWidgets.QLabel(" ")
  120. self.layout.addWidget(self.plate_spacer_label)
  121. # ## Title of the ElectroPlating Tools Calculator
  122. plate_title_label = QtWidgets.QLabel("<font size=3><b>%s</b></font>" % self.eplateName)
  123. plate_title_label.setToolTip(
  124. _("This calculator is useful for those who plate the via/pad/drill holes,\n"
  125. "using a method like graphite ink or calcium hypophosphite ink or palladium chloride.")
  126. )
  127. self.layout.addWidget(plate_title_label)
  128. # ## Plate Form Layout
  129. plate_form_layout = QtWidgets.QFormLayout()
  130. self.layout.addLayout(plate_form_layout)
  131. self.pcblengthlabel = QtWidgets.QLabel('%s:' % _("Board Length"))
  132. self.pcblength_entry = FCDoubleSpinner(callback=self.confirmation_message)
  133. self.pcblength_entry.set_precision(self.decimals)
  134. self.pcblength_entry.set_range(0.0, 9999.9999)
  135. # self.pcblength_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
  136. self.pcblengthlabel.setToolTip(_('This is the board length. In centimeters.'))
  137. self.pcbwidthlabel = QtWidgets.QLabel('%s:' % _("Board Width"))
  138. self.pcbwidth_entry = FCDoubleSpinner(callback=self.confirmation_message)
  139. self.pcbwidth_entry.set_precision(self.decimals)
  140. self.pcbwidth_entry.set_range(0.0, 9999.9999)
  141. # self.pcbwidth_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
  142. self.pcbwidthlabel.setToolTip(_('This is the board width.In centimeters.'))
  143. self.cdensity_label = QtWidgets.QLabel('%s:' % _("Current Density"))
  144. self.cdensity_entry = FCDoubleSpinner(callback=self.confirmation_message)
  145. self.cdensity_entry.set_precision(self.decimals)
  146. self.cdensity_entry.set_range(0.0, 9999.9999)
  147. self.cdensity_entry.setSingleStep(0.1)
  148. # self.cdensity_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
  149. self.cdensity_label.setToolTip(_("Current density to pass through the board. \n"
  150. "In Amps per Square Feet ASF."))
  151. self.growth_label = QtWidgets.QLabel('%s:' % _("Copper Growth"))
  152. self.growth_entry = FCDoubleSpinner(callback=self.confirmation_message)
  153. self.growth_entry.set_precision(self.decimals)
  154. self.growth_entry.set_range(0.0, 9999.9999)
  155. self.growth_entry.setSingleStep(0.01)
  156. # self.growth_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
  157. self.growth_label.setToolTip(_("How thick the copper growth is intended to be.\n"
  158. "In microns."))
  159. # self.growth_entry.setEnabled(False)
  160. self.cvaluelabel = QtWidgets.QLabel('%s:' % _("Current Value"))
  161. self.cvalue_entry = FCDoubleSpinner(callback=self.confirmation_message)
  162. self.cvalue_entry.set_precision(self.decimals)
  163. self.cvalue_entry.set_range(0.0, 9999.9999)
  164. self.cvalue_entry.setSingleStep(0.1)
  165. # self.cvalue_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
  166. self.cvaluelabel.setToolTip(_('This is the current intensity value\n'
  167. 'to be set on the Power Supply. In Amps.'))
  168. self.cvalue_entry.setReadOnly(True)
  169. self.timelabel = QtWidgets.QLabel('%s:' % _("Time"))
  170. self.time_entry = FCDoubleSpinner(callback=self.confirmation_message)
  171. self.time_entry.set_precision(self.decimals)
  172. self.time_entry.set_range(0.0, 9999.9999)
  173. self.time_entry.setSingleStep(0.1)
  174. # self.time_entry.setAlignment(QtCore.Qt.AlignRight | QtCore.Qt.AlignVCenter)
  175. self.timelabel.setToolTip(_('This is the calculated time required for the procedure.\n'
  176. 'In minutes.'))
  177. self.time_entry.setReadOnly(True)
  178. plate_form_layout.addRow(self.pcblengthlabel, self.pcblength_entry)
  179. plate_form_layout.addRow(self.pcbwidthlabel, self.pcbwidth_entry)
  180. plate_form_layout.addRow(self.cdensity_label, self.cdensity_entry)
  181. plate_form_layout.addRow(self.growth_label, self.growth_entry)
  182. plate_form_layout.addRow(self.cvaluelabel, self.cvalue_entry)
  183. plate_form_layout.addRow(self.timelabel, self.time_entry)
  184. # ## Buttons
  185. self.calculate_plate_button = QtWidgets.QPushButton(_("Calculate"))
  186. # self.calculate_button.setFixedWidth(70)
  187. self.calculate_plate_button.setToolTip(
  188. _("Calculate the current intensity value and the procedure time,\n"
  189. "depending on the parameters above")
  190. )
  191. self.layout.addWidget(self.calculate_plate_button)
  192. self.layout.addStretch()
  193. # ## Reset Tool
  194. self.reset_button = QtWidgets.QPushButton(_("Reset Tool"))
  195. self.reset_button.setToolTip(
  196. _("Will reset the tool parameters.")
  197. )
  198. self.reset_button.setStyleSheet("""
  199. QPushButton
  200. {
  201. font-weight: bold;
  202. }
  203. """)
  204. self.layout.addWidget(self.reset_button)
  205. self.units = ''
  206. # ## Signals
  207. self.cutDepth_entry.valueChanged.connect(self.on_calculate_tool_dia)
  208. self.cutDepth_entry.returnPressed.connect(self.on_calculate_tool_dia)
  209. self.tipDia_entry.returnPressed.connect(self.on_calculate_tool_dia)
  210. self.tipAngle_entry.returnPressed.connect(self.on_calculate_tool_dia)
  211. self.calculate_vshape_button.clicked.connect(self.on_calculate_tool_dia)
  212. self.mm_entry.editingFinished.connect(self.on_calculate_inch_units)
  213. self.inch_entry.editingFinished.connect(self.on_calculate_mm_units)
  214. self.calculate_plate_button.clicked.connect(self.on_calculate_eplate)
  215. self.reset_button.clicked.connect(self.set_tool_ui)
  216. def run(self, toggle=True):
  217. self.app.defaults.report_usage("ToolCalculators()")
  218. if toggle:
  219. # if the splitter is hidden, display it, else hide it but only if the current widget is the same
  220. if self.app.ui.splitter.sizes()[0] == 0:
  221. self.app.ui.splitter.setSizes([1, 1])
  222. else:
  223. try:
  224. if self.app.ui.tool_scroll_area.widget().objectName() == self.toolName:
  225. # if tab is populated with the tool but it does not have the focus, focus on it
  226. if not self.app.ui.notebook.currentWidget() is self.app.ui.tool_tab:
  227. # focus on Tool Tab
  228. self.app.ui.notebook.setCurrentWidget(self.app.ui.tool_tab)
  229. else:
  230. self.app.ui.splitter.setSizes([0, 1])
  231. except AttributeError:
  232. pass
  233. else:
  234. if self.app.ui.splitter.sizes()[0] == 0:
  235. self.app.ui.splitter.setSizes([1, 1])
  236. FlatCAMTool.run(self)
  237. self.set_tool_ui()
  238. self.app.ui.notebook.setTabText(2, _("Calc. Tool"))
  239. def install(self, icon=None, separator=None, **kwargs):
  240. FlatCAMTool.install(self, icon, separator, shortcut='Alt+C', **kwargs)
  241. def set_tool_ui(self):
  242. self.units = self.app.defaults['units'].upper()
  243. # ## Initialize form
  244. self.mm_entry.set_value('%.*f' % (self.decimals, 0))
  245. self.inch_entry.set_value('%.*f' % (self.decimals, 0))
  246. length = self.app.defaults["tools_calc_electro_length"]
  247. width = self.app.defaults["tools_calc_electro_width"]
  248. density = self.app.defaults["tools_calc_electro_cdensity"]
  249. growth = self.app.defaults["tools_calc_electro_growth"]
  250. self.pcblength_entry.set_value(length)
  251. self.pcbwidth_entry.set_value(width)
  252. self.cdensity_entry.set_value(density)
  253. self.growth_entry.set_value(growth)
  254. self.cvalue_entry.set_value(0.00)
  255. self.time_entry.set_value(0.0)
  256. tip_dia = self.app.defaults["tools_calc_vshape_tip_dia"]
  257. tip_angle = self.app.defaults["tools_calc_vshape_tip_angle"]
  258. cut_z = self.app.defaults["tools_calc_vshape_cut_z"]
  259. self.tipDia_entry.set_value(tip_dia)
  260. self.tipAngle_entry.set_value(tip_angle)
  261. self.cutDepth_entry.set_value(cut_z)
  262. self.effectiveToolDia_entry.set_value('0.0000')
  263. def on_calculate_tool_dia(self):
  264. # Calculation:
  265. # Manufacturer gives total angle of the the tip but we need only half of it
  266. # tangent(half_tip_angle) = opposite side / adjacent = part_of _real_dia / depth_of_cut
  267. # effective_diameter = tip_diameter + part_of_real_dia_left_side + part_of_real_dia_right_side
  268. # tool is symmetrical therefore: part_of_real_dia_left_side = part_of_real_dia_right_side
  269. # effective_diameter = tip_diameter + (2 * part_of_real_dia_left_side)
  270. # effective diameter = tip_diameter + (2 * depth_of_cut * tangent(half_tip_angle))
  271. tip_diameter = float(self.tipDia_entry.get_value())
  272. half_tip_angle = float(self.tipAngle_entry.get_value()) / 2.0
  273. cut_depth = float(self.cutDepth_entry.get_value())
  274. cut_depth = -cut_depth if cut_depth < 0 else cut_depth
  275. tool_diameter = tip_diameter + (2 * cut_depth * math.tan(math.radians(half_tip_angle)))
  276. self.effectiveToolDia_entry.set_value("%.*f" % (self.decimals, tool_diameter))
  277. def on_calculate_inch_units(self):
  278. mm_val = float(self.mm_entry.get_value())
  279. self.inch_entry.set_value('%.*f' % (self.decimals, (mm_val / 25.4)))
  280. def on_calculate_mm_units(self):
  281. inch_val = float(self.inch_entry.get_value())
  282. self.mm_entry.set_value('%.*f' % (self.decimals, (inch_val * 25.4)))
  283. def on_calculate_eplate(self):
  284. length = float(self.pcblength_entry.get_value())
  285. width = float(self.pcbwidth_entry.get_value())
  286. density = float(self.cdensity_entry.get_value())
  287. copper = float(self.growth_entry.get_value())
  288. calculated_current = (length * width * density) * 0.0021527820833419
  289. calculated_time = copper * 2.142857142857143 * float(20 / density)
  290. self.cvalue_entry.set_value('%.2f' % calculated_current)
  291. self.time_entry.set_value('%.1f' % calculated_time)
  292. # end of file