ToolCopperThieving.py 72 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648
  1. # ##########################################################
  2. # FlatCAM: 2D Post-processing for Manufacturing #
  3. # File Author: Marius Adrian Stanciu (c) #
  4. # Date: 10/25/2019 #
  5. # MIT Licence #
  6. # ##########################################################
  7. from PyQt5 import QtWidgets, QtCore, QtGui
  8. from camlib import grace
  9. from appTool import AppTool
  10. from appGUI.GUIElements import FCDoubleSpinner, RadioSet, FCEntry, FCComboBox
  11. import shapely.geometry.base as base
  12. from shapely.ops import unary_union
  13. from shapely.geometry import Polygon, MultiPolygon, Point, LineString
  14. from shapely.geometry import box as box
  15. import shapely.affinity as affinity
  16. import logging
  17. from copy import deepcopy
  18. import numpy as np
  19. from collections import Iterable
  20. import gettext
  21. import appTranslation as fcTranslate
  22. import builtins
  23. fcTranslate.apply_language('strings')
  24. if '_' not in builtins.__dict__:
  25. _ = gettext.gettext
  26. log = logging.getLogger('base')
  27. class ToolCopperThieving(AppTool):
  28. work_finished = QtCore.pyqtSignal()
  29. def __init__(self, app):
  30. AppTool.__init__(self, app)
  31. self.app = app
  32. self.canvas = self.app.plotcanvas
  33. self.decimals = self.app.decimals
  34. self.units = self.app.defaults['units']
  35. # #############################################################################
  36. # ######################### Tool GUI ##########################################
  37. # #############################################################################
  38. self.ui = ThievingUI(layout=self.layout, app=self.app)
  39. self.toolName = self.ui.toolName
  40. # Objects involved in Copper thieving
  41. self.grb_object = None
  42. self.ref_obj = None
  43. self.sel_rect = []
  44. self.sm_object = None
  45. # store the flattened geometry here:
  46. self.flat_geometry = []
  47. # Events ID
  48. self.mr = None
  49. self.mm = None
  50. # Mouse cursor positions
  51. self.mouse_is_dragging = False
  52. self.cursor_pos = (0, 0)
  53. self.first_click = False
  54. self.handlers_connected = False
  55. # Tool properties
  56. self.clearance_val = None
  57. self.margin_val = None
  58. self.geo_steps_per_circle = 128
  59. # Thieving geometry storage
  60. self.new_solid_geometry = []
  61. # Robber bar geometry storage
  62. self.robber_geo = None
  63. self.robber_line = None
  64. self.rb_thickness = None
  65. # SIGNALS
  66. self.ui.ref_combo_type.currentIndexChanged.connect(self.on_ref_combo_type_change)
  67. self.ui.reference_radio.group_toggle_fn = self.on_toggle_reference
  68. self.ui.fill_type_radio.activated_custom.connect(self.on_thieving_type)
  69. self.ui.fill_button.clicked.connect(self.on_add_copper_thieving_click)
  70. self.ui.rb_button.clicked.connect(self.on_add_robber_bar_click)
  71. self.ui.ppm_button.clicked.connect(self.on_add_ppm_click)
  72. self.ui.reset_button.clicked.connect(self.set_tool_ui)
  73. self.work_finished.connect(self.on_new_pattern_plating_object)
  74. def run(self, toggle=True):
  75. self.app.defaults.report_usage("ToolCopperThieving()")
  76. if toggle:
  77. # if the splitter is hidden, display it, else hide it but only if the current widget is the same
  78. if self.app.ui.splitter.sizes()[0] == 0:
  79. self.app.ui.splitter.setSizes([1, 1])
  80. else:
  81. try:
  82. if self.app.ui.tool_scroll_area.widget().objectName() == self.toolName:
  83. # if tab is populated with the tool but it does not have the focus, focus on it
  84. if not self.app.ui.notebook.currentWidget() is self.app.ui.tool_tab:
  85. # focus on Tool Tab
  86. self.app.ui.notebook.setCurrentWidget(self.app.ui.tool_tab)
  87. else:
  88. self.app.ui.splitter.setSizes([0, 1])
  89. except AttributeError:
  90. pass
  91. else:
  92. if self.app.ui.splitter.sizes()[0] == 0:
  93. self.app.ui.splitter.setSizes([1, 1])
  94. AppTool.run(self)
  95. self.set_tool_ui()
  96. self.app.ui.notebook.setTabText(2, _("Copper Thieving Tool"))
  97. def install(self, icon=None, separator=None, **kwargs):
  98. AppTool.install(self, icon, separator, shortcut='Alt+J', **kwargs)
  99. def set_tool_ui(self):
  100. self.units = self.app.defaults['units']
  101. self.geo_steps_per_circle = int(self.app.defaults["tools_copper_thieving_circle_steps"])
  102. self.ui.clearance_entry.set_value(float(self.app.defaults["tools_copper_thieving_clearance"]))
  103. self.ui.margin_entry.set_value(float(self.app.defaults["tools_copper_thieving_margin"]))
  104. self.ui.reference_radio.set_value(self.app.defaults["tools_copper_thieving_reference"])
  105. self.ui.bbox_type_radio.set_value(self.app.defaults["tools_copper_thieving_box_type"])
  106. self.ui.fill_type_radio.set_value(self.app.defaults["tools_copper_thieving_fill_type"])
  107. self.ui.dot_dia_entry.set_value(self.app.defaults["tools_copper_thieving_dots_dia"])
  108. self.ui.dot_spacing_entry.set_value(self.app.defaults["tools_copper_thieving_dots_spacing"])
  109. self.ui.square_size_entry.set_value(self.app.defaults["tools_copper_thieving_squares_size"])
  110. self.ui.squares_spacing_entry.set_value(self.app.defaults["tools_copper_thieving_squares_spacing"])
  111. self.ui.line_size_entry.set_value(self.app.defaults["tools_copper_thieving_lines_size"])
  112. self.ui.lines_spacing_entry.set_value(self.app.defaults["tools_copper_thieving_lines_spacing"])
  113. self.ui.rb_margin_entry.set_value(self.app.defaults["tools_copper_thieving_rb_margin"])
  114. self.ui.rb_thickness_entry.set_value(self.app.defaults["tools_copper_thieving_rb_thickness"])
  115. self.ui.clearance_ppm_entry.set_value(self.app.defaults["tools_copper_thieving_mask_clearance"])
  116. # INIT SECTION
  117. self.handlers_connected = False
  118. self.robber_geo = None
  119. self.robber_line = None
  120. self.new_solid_geometry = None
  121. def on_ref_combo_type_change(self):
  122. obj_type = self.ui.ref_combo_type.currentIndex()
  123. self.ui.ref_combo.setRootModelIndex(self.app.collection.index(obj_type, 0, QtCore.QModelIndex()))
  124. self.ui.ref_combo.setCurrentIndex(0)
  125. self.ui.ref_combo.obj_type = {
  126. _("Gerber"): "Gerber", _("Excellon"): "Excellon", _("Geometry"): "Geometry"
  127. }[self.ui.ref_combo_type.get_value()]
  128. def on_toggle_reference(self):
  129. if self.ui.reference_radio.get_value() == "itself" or self.ui.reference_radio.get_value() == "area":
  130. self.ui.ref_combo.hide()
  131. self.ui.ref_combo_label.hide()
  132. self.ui.ref_combo_type.hide()
  133. self.ui.ref_combo_type_label.hide()
  134. else:
  135. self.ui.ref_combo.show()
  136. self.ui.ref_combo_label.show()
  137. self.ui.ref_combo_type.show()
  138. self.ui.ref_combo_type_label.show()
  139. if self.ui.reference_radio.get_value() == "itself":
  140. self.ui.bbox_type_label.show()
  141. self.ui.bbox_type_radio.show()
  142. else:
  143. if self.ui.fill_type_radio.get_value() == 'line':
  144. self.ui.reference_radio.set_value('itself')
  145. self.app.inform.emit('[WARNING_NOTCL] %s' % _("Lines Grid works only for 'itself' reference ..."))
  146. return
  147. self.ui.bbox_type_label.hide()
  148. self.ui.bbox_type_radio.hide()
  149. def on_thieving_type(self, choice):
  150. if choice == 'solid':
  151. self.ui.dots_frame.hide()
  152. self.ui.squares_frame.hide()
  153. self.ui.lines_frame.hide()
  154. self.app.inform.emit(_("Solid fill selected."))
  155. elif choice == 'dot':
  156. self.ui.dots_frame.show()
  157. self.ui.squares_frame.hide()
  158. self.ui.lines_frame.hide()
  159. self.app.inform.emit(_("Dots grid fill selected."))
  160. elif choice == 'square':
  161. self.ui.dots_frame.hide()
  162. self.ui.squares_frame.show()
  163. self.ui.lines_frame.hide()
  164. self.app.inform.emit(_("Squares grid fill selected."))
  165. else:
  166. if self.ui.reference_radio.get_value() != 'itself':
  167. self.ui.reference_radio.set_value('itself')
  168. self.app.inform.emit('[WARNING_NOTCL] %s' % _("Lines Grid works only for 'itself' reference ..."))
  169. self.ui.dots_frame.hide()
  170. self.ui.squares_frame.hide()
  171. self.ui.lines_frame.show()
  172. def on_add_robber_bar_click(self):
  173. rb_margin = self.ui.rb_margin_entry.get_value()
  174. self.rb_thickness = self.ui.rb_thickness_entry.get_value()
  175. # get the Gerber object on which the Robber bar will be inserted
  176. selection_index = self.ui.grb_object_combo.currentIndex()
  177. model_index = self.app.collection.index(selection_index, 0, self.ui.grb_object_combo.rootModelIndex())
  178. try:
  179. self.grb_object = model_index.internalPointer().obj
  180. except Exception as e:
  181. log.debug("ToolCopperThieving.on_add_robber_bar_click() --> %s" % str(e))
  182. self.app.inform.emit('[WARNING_NOTCL] %s' % _("There is no Gerber object loaded ..."))
  183. return
  184. try:
  185. outline_pol = self.grb_object.solid_geometry.envelope
  186. except (TypeError, AttributeError):
  187. outline_pol = MultiPolygon(self.grb_object.solid_geometry).envelope
  188. rb_distance = rb_margin + (self.rb_thickness / 2.0)
  189. self.robber_line = outline_pol.buffer(rb_distance).exterior
  190. self.robber_geo = self.robber_line.buffer(self.rb_thickness / 2.0)
  191. self.app.proc_container.update_view_text(' %s' % _("Append geometry"))
  192. new_apertures = deepcopy(self.grb_object.apertures)
  193. aperture_found = None
  194. for ap_id, ap_val in self.grb_object.apertures.items():
  195. if ap_val['type'] == 'C' and ap_val['size'] == self.rb_thickness:
  196. aperture_found = ap_id
  197. break
  198. if aperture_found:
  199. geo_elem = {'solid': self.robber_geo, 'follow': self.robber_line}
  200. new_apertures[aperture_found]['geometry'].append(deepcopy(geo_elem))
  201. else:
  202. ap_keys = list(new_apertures.keys())
  203. if ap_keys:
  204. new_apid = str(int(max(ap_keys)) + 1)
  205. else:
  206. new_apid = '10'
  207. new_apertures[new_apid] = {
  208. 'type': 'C',
  209. 'size': deepcopy(self.rb_thickness),
  210. 'geometry': []
  211. }
  212. geo_elem = {'solid': self.robber_geo, 'follow': self.robber_line}
  213. new_apertures[new_apid]['geometry'].append(deepcopy(geo_elem))
  214. geo_obj = deepcopy(self.grb_object.solid_geometry)
  215. if isinstance(geo_obj, MultiPolygon):
  216. s_list = []
  217. for pol in geo_obj.geoms:
  218. s_list.append(pol)
  219. s_list.append(deepcopy(self.robber_geo))
  220. geo_obj = MultiPolygon(s_list)
  221. elif isinstance(geo_obj, list):
  222. geo_obj.append(deepcopy(self.robber_geo))
  223. elif isinstance(geo_obj, Polygon):
  224. geo_obj = MultiPolygon([geo_obj, deepcopy(self.robber_geo)])
  225. outname = '%s_%s' % (str(self.grb_object.options['name']), 'robber')
  226. def initialize(grb_obj, app_obj):
  227. grb_obj.options = {}
  228. for opt in self.grb_object.options:
  229. if opt != 'name':
  230. grb_obj.options[opt] = deepcopy(self.grb_object.options[opt])
  231. grb_obj.options['name'] = outname
  232. grb_obj.multitool = False
  233. grb_obj.multigeo = False
  234. grb_obj.follow = deepcopy(self.grb_object.follow)
  235. grb_obj.apertures = new_apertures
  236. grb_obj.solid_geometry = unary_union(geo_obj)
  237. grb_obj.follow_geometry = deepcopy(self.grb_object.follow_geometry) + [deepcopy(self.robber_line)]
  238. app_obj.proc_container.update_view_text(' %s' % _("Append source file"))
  239. # update the source file with the new geometry:
  240. grb_obj.source_file = app_obj.f_handlers.export_gerber(obj_name=outname, filename=None, local_use=grb_obj,
  241. use_thread=False)
  242. ret_val = self.app.app_obj.new_object('gerber', outname, initialize, plot=True)
  243. self.app.proc_container.update_view_text(' %s' % '')
  244. if ret_val == 'fail':
  245. self.app.call_source = "app"
  246. self.app.inform.emit('[ERROR_NOTCL] %s' % _("Failed."))
  247. return
  248. self.on_exit()
  249. self.app.inform.emit('[success] %s' % _("Copper Thieving Tool done."))
  250. def on_add_copper_thieving_click(self):
  251. self.app.call_source = "copper_thieving_tool"
  252. self.clearance_val = self.ui.clearance_entry.get_value()
  253. self.margin_val = self.ui.margin_entry.get_value()
  254. reference_method = self.ui.reference_radio.get_value()
  255. # get the Gerber object on which the Copper thieving will be inserted
  256. selection_index = self.ui.grb_object_combo.currentIndex()
  257. model_index = self.app.collection.index(selection_index, 0, self.ui.grb_object_combo.rootModelIndex())
  258. try:
  259. self.grb_object = model_index.internalPointer().obj
  260. except Exception as e:
  261. log.debug("ToolCopperThieving.on_add_copper_thieving_click() --> %s" % str(e))
  262. self.app.inform.emit('[WARNING_NOTCL] %s' % _("There is no Gerber object loaded ..."))
  263. return
  264. if reference_method == 'itself':
  265. bound_obj_name = self.ui.grb_object_combo.currentText()
  266. # Get reference object.
  267. try:
  268. self.ref_obj = self.app.collection.get_by_name(bound_obj_name)
  269. except Exception as e:
  270. self.app.inform.emit('[ERROR_NOTCL] %s: %s' % (_("Could not retrieve object"), str(e)))
  271. return "Could not retrieve object: %s" % self.obj_name
  272. self.on_copper_thieving(
  273. thieving_obj=self.grb_object,
  274. c_val=self.clearance_val,
  275. margin=self.margin_val
  276. )
  277. elif reference_method == 'area':
  278. self.app.inform.emit('[WARNING_NOTCL] %s' % _("Click the start point of the area."))
  279. self.connect_event_handlers()
  280. elif reference_method == 'box':
  281. bound_obj_name = self.ui.ref_combo.currentText()
  282. # Get reference object.
  283. try:
  284. self.ref_obj = self.app.collection.get_by_name(bound_obj_name)
  285. except Exception:
  286. self.app.inform.emit('[ERROR_NOTCL] %s: %s' % (_("Could not retrieve object"), bound_obj_name))
  287. return
  288. self.on_copper_thieving(
  289. thieving_obj=self.grb_object,
  290. ref_obj=self.ref_obj,
  291. c_val=self.clearance_val,
  292. margin=self.margin_val
  293. )
  294. # To be called after clicking on the plot.
  295. def on_mouse_release(self, event):
  296. if self.app.is_legacy is False:
  297. event_pos = event.pos
  298. # event_is_dragging = event.is_dragging
  299. right_button = 2
  300. else:
  301. event_pos = (event.xdata, event.ydata)
  302. # event_is_dragging = self.app.plotcanvas.is_dragging
  303. right_button = 3
  304. event_pos = self.app.plotcanvas.translate_coords(event_pos)
  305. # do clear area only for left mouse clicks
  306. if event.button == 1:
  307. if self.first_click is False:
  308. self.first_click = True
  309. self.app.inform.emit('[WARNING_NOTCL] %s' % _("Click the end point of the filling area."))
  310. self.cursor_pos = self.app.plotcanvas.translate_coords(event_pos)
  311. if self.app.grid_status() is True:
  312. self.cursor_pos = self.app.geo_editor.snap(event_pos[0], event_pos[1])
  313. else:
  314. self.app.inform.emit(_("Zone added. Click to start adding next zone or right click to finish."))
  315. self.app.delete_selection_shape()
  316. if self.app.grid_status() is True:
  317. curr_pos = self.app.geo_editor.snap(event_pos[0], event_pos[1])
  318. else:
  319. curr_pos = (event_pos[0], event_pos[1])
  320. x0, y0 = self.cursor_pos[0], self.cursor_pos[1]
  321. x1, y1 = curr_pos[0], curr_pos[1]
  322. pt1 = (x0, y0)
  323. pt2 = (x1, y0)
  324. pt3 = (x1, y1)
  325. pt4 = (x0, y1)
  326. new_rectangle = Polygon([pt1, pt2, pt3, pt4])
  327. self.sel_rect.append(new_rectangle)
  328. # add a temporary shape on canvas
  329. self.draw_tool_selection_shape(old_coords=(x0, y0), coords=(x1, y1))
  330. self.first_click = False
  331. return
  332. elif event.button == right_button and self.mouse_is_dragging is False:
  333. self.first_click = False
  334. self.delete_tool_selection_shape()
  335. self.disconnect_event_handlers()
  336. if len(self.sel_rect) == 0:
  337. return
  338. self.sel_rect = unary_union(self.sel_rect)
  339. if not isinstance(self.sel_rect, Iterable):
  340. self.sel_rect = [self.sel_rect]
  341. self.on_copper_thieving(
  342. thieving_obj=self.grb_object,
  343. ref_obj=self.sel_rect,
  344. c_val=self.clearance_val,
  345. margin=self.margin_val
  346. )
  347. # called on mouse move
  348. def on_mouse_move(self, event):
  349. if self.app.is_legacy is False:
  350. event_pos = event.pos
  351. event_is_dragging = event.is_dragging
  352. # right_button = 2
  353. else:
  354. event_pos = (event.xdata, event.ydata)
  355. event_is_dragging = self.app.plotcanvas.is_dragging
  356. # right_button = 3
  357. curr_pos = self.app.plotcanvas.translate_coords(event_pos)
  358. # detect mouse dragging motion
  359. if event_is_dragging is True:
  360. self.mouse_is_dragging = True
  361. else:
  362. self.mouse_is_dragging = False
  363. # update the cursor position
  364. if self.app.grid_status() is True:
  365. # Update cursor
  366. curr_pos = self.app.geo_editor.snap(curr_pos[0], curr_pos[1])
  367. self.app.app_cursor.set_data(np.asarray([(curr_pos[0], curr_pos[1])]),
  368. symbol='++', edge_color=self.app.cursor_color_3D,
  369. edge_width=self.app.defaults["global_cursor_width"],
  370. size=self.app.defaults["global_cursor_size"])
  371. if self.cursor_pos is None:
  372. self.cursor_pos = (0, 0)
  373. self.app.dx = curr_pos[0] - float(self.cursor_pos[0])
  374. self.app.dy = curr_pos[1] - float(self.cursor_pos[1])
  375. # # update the positions on status bar
  376. self.app.ui.position_label.setText("&nbsp;<b>X</b>: %.4f&nbsp;&nbsp; "
  377. "<b>Y</b>: %.4f&nbsp;" % (curr_pos[0], curr_pos[1]))
  378. self.app.ui.rel_position_label.setText("<b>Dx</b>: %.4f&nbsp;&nbsp; <b>Dy</b>: "
  379. "%.4f&nbsp;&nbsp;&nbsp;&nbsp;" % (self.app.dx, self.app.dy))
  380. units = self.app.defaults["units"].lower()
  381. self.app.plotcanvas.text_hud.text = \
  382. 'Dx:\t{:<.4f} [{:s}]\nDy:\t{:<.4f} [{:s}]\n\nX: \t{:<.4f} [{:s}]\nY: \t{:<.4f} [{:s}]'.format(
  383. self.app.dx, units, self.app.dy, units, curr_pos[0], units, curr_pos[1], units)
  384. # draw the utility geometry
  385. if self.first_click:
  386. self.app.delete_selection_shape()
  387. self.app.draw_moving_selection_shape(old_coords=(self.cursor_pos[0], self.cursor_pos[1]),
  388. coords=(curr_pos[0], curr_pos[1]))
  389. def on_copper_thieving(self, thieving_obj, ref_obj=None, c_val=None, margin=None, run_threaded=True):
  390. """
  391. :param thieving_obj:
  392. :param ref_obj:
  393. :param c_val:
  394. :param margin:
  395. :param run_threaded:
  396. :return:
  397. """
  398. if run_threaded:
  399. self.app.proc_container.new('%s ...' % _("Thieving"))
  400. else:
  401. QtWidgets.QApplication.processEvents()
  402. self.app.proc_container.view.set_busy('%s ...' % _("Thieving"))
  403. # #####################################################################
  404. # ####### Read the parameters #########################################
  405. # #####################################################################
  406. log.debug("Copper Thieving Tool started. Reading parameters.")
  407. self.app.inform.emit(_("Copper Thieving Tool started. Reading parameters."))
  408. ref_selected = self.ui.reference_radio.get_value()
  409. if c_val is None:
  410. c_val = float(self.app.defaults["tools_copper_thieving_clearance"])
  411. if margin is None:
  412. margin = float(self.app.defaults["tools_copper_thieving_margin"])
  413. fill_type = self.ui.fill_type_radio.get_value()
  414. dot_dia = self.ui.dot_dia_entry.get_value()
  415. dot_spacing = self.ui.dot_spacing_entry.get_value()
  416. square_size = self.ui.square_size_entry.get_value()
  417. square_spacing = self.ui.squares_spacing_entry.get_value()
  418. line_size = self.ui.line_size_entry.get_value()
  419. line_spacing = self.ui.lines_spacing_entry.get_value()
  420. # make sure that the source object solid geometry is an Iterable
  421. if not isinstance(self.grb_object.solid_geometry, Iterable):
  422. self.grb_object.solid_geometry = [self.grb_object.solid_geometry]
  423. def job_thread_thieving(tool_obj):
  424. # #########################################################################################################
  425. # Prepare isolation polygon. This will create the clearance over the Gerber features
  426. # #########################################################################################################
  427. log.debug("Copper Thieving Tool. Preparing isolation polygons.")
  428. tool_obj.app.inform.emit(_("Copper Thieving Tool. Preparing isolation polygons."))
  429. # variables to display the percentage of work done
  430. try:
  431. geo_len = len(tool_obj.grb_object.solid_geometry)
  432. except TypeError:
  433. geo_len = 1
  434. old_disp_number = 0
  435. pol_nr = 0
  436. clearance_geometry = []
  437. try:
  438. for pol in tool_obj.grb_object.solid_geometry:
  439. if tool_obj.app.abort_flag:
  440. # graceful abort requested by the user
  441. raise grace
  442. clearance_geometry.append(
  443. pol.buffer(c_val, int(int(tool_obj.geo_steps_per_circle) / 4))
  444. )
  445. pol_nr += 1
  446. disp_number = int(np.interp(pol_nr, [0, geo_len], [0, 100]))
  447. if old_disp_number < disp_number <= 100:
  448. msg = ' %s ... %d%%' % (_("Thieving"), int(disp_number))
  449. tool_obj.app.proc_container.update_view_text(msg)
  450. old_disp_number = disp_number
  451. except TypeError:
  452. # taking care of the case when the self.solid_geometry is just a single Polygon, not a list or a
  453. # MultiPolygon (not an iterable)
  454. clearance_geometry.append(
  455. tool_obj.grb_object.solid_geometry.buffer(c_val, int(int(tool_obj.geo_steps_per_circle) / 4))
  456. )
  457. tool_obj.app.proc_container.update_view_text(' %s ...' % _("Buffering"))
  458. clearance_geometry = unary_union(clearance_geometry)
  459. # #########################################################################################################
  460. # Prepare the area to fill with copper.
  461. # #########################################################################################################
  462. log.debug("Copper Thieving Tool. Preparing areas to fill with copper.")
  463. tool_obj.app.inform.emit(_("Copper Thieving Tool. Preparing areas to fill with copper."))
  464. try:
  465. if ref_obj is None or ref_obj == 'itself':
  466. working_obj = thieving_obj
  467. else:
  468. working_obj = ref_obj
  469. except Exception as e:
  470. log.debug("ToolCopperThieving.on_copper_thieving() --> %s" % str(e))
  471. return 'fail'
  472. tool_obj.app.proc_container.update_view_text(' %s' % _("Working..."))
  473. # #########################################################################################################
  474. # generate the bounding box geometry
  475. # #########################################################################################################
  476. if ref_selected == 'itself':
  477. geo_n = deepcopy(working_obj.solid_geometry)
  478. try:
  479. if tool_obj.ui.bbox_type_radio.get_value() == 'min':
  480. if isinstance(geo_n, MultiPolygon):
  481. env_obj = geo_n.convex_hull
  482. elif (isinstance(geo_n, MultiPolygon) and len(geo_n) == 1) or \
  483. (isinstance(geo_n, list) and len(geo_n) == 1) and isinstance(geo_n[0], Polygon):
  484. env_obj = unary_union(geo_n)
  485. else:
  486. env_obj = unary_union(geo_n)
  487. env_obj = env_obj.convex_hull
  488. bounding_box = env_obj.buffer(distance=margin, join_style=base.JOIN_STYLE.mitre)
  489. else:
  490. if isinstance(geo_n, Polygon):
  491. bounding_box = geo_n.buffer(distance=margin, join_style=base.JOIN_STYLE.mitre)
  492. elif isinstance(geo_n, list):
  493. geo_n = MultiPolygon(geo_n)
  494. x0, y0, x1, y1 = geo_n.bounds
  495. geo = box(x0, y0, x1, y1)
  496. bounding_box = geo.buffer(distance=margin, join_style=base.JOIN_STYLE.mitre)
  497. elif isinstance(geo_n, MultiPolygon):
  498. x0, y0, x1, y1 = geo_n.bounds
  499. geo = box(x0, y0, x1, y1)
  500. bounding_box = geo.buffer(distance=margin, join_style=base.JOIN_STYLE.mitre)
  501. else:
  502. tool_obj.app.inform.emit(
  503. '[ERROR_NOTCL] %s: %s' % (_("Geometry not supported for bounding box"), type(geo_n))
  504. )
  505. return 'fail'
  506. except Exception as e:
  507. log.debug("ToolCopperFIll.on_copper_thieving() 'itself' --> %s" % str(e))
  508. tool_obj.app.inform.emit('[ERROR_NOTCL] %s' % _("No object available."))
  509. return 'fail'
  510. elif ref_selected == 'area':
  511. geo_buff_list = []
  512. try:
  513. for poly in working_obj:
  514. if tool_obj.app.abort_flag:
  515. # graceful abort requested by the user
  516. raise grace
  517. geo_buff_list.append(poly.buffer(distance=margin, join_style=base.JOIN_STYLE.mitre))
  518. except TypeError:
  519. geo_buff_list.append(working_obj.buffer(distance=margin, join_style=base.JOIN_STYLE.mitre))
  520. bounding_box = MultiPolygon(geo_buff_list)
  521. else: # ref_selected == 'box'
  522. geo_n = working_obj.solid_geometry
  523. if working_obj.kind == 'geometry':
  524. try:
  525. __ = iter(geo_n)
  526. except Exception as e:
  527. log.debug("ToolCopperFIll.on_copper_thieving() 'box' --> %s" % str(e))
  528. geo_n = [geo_n]
  529. geo_buff_list = []
  530. for poly in geo_n:
  531. if tool_obj.app.abort_flag:
  532. # graceful abort requested by the user
  533. raise grace
  534. geo_buff_list.append(poly.buffer(distance=margin, join_style=base.JOIN_STYLE.mitre))
  535. bounding_box = unary_union(geo_buff_list)
  536. elif working_obj.kind == 'gerber':
  537. geo_n = unary_union(geo_n).convex_hull
  538. bounding_box = unary_union(thieving_obj.solid_geometry).convex_hull.intersection(geo_n)
  539. bounding_box = bounding_box.buffer(distance=margin, join_style=base.JOIN_STYLE.mitre)
  540. else:
  541. tool_obj.app.inform.emit('[ERROR_NOTCL] %s' % _("The reference object type is not supported."))
  542. return 'fail'
  543. log.debug("Copper Thieving Tool. Finished creating areas to fill with copper.")
  544. tool_obj.app.inform.emit(_("Copper Thieving Tool. Appending new geometry and buffering."))
  545. # #########################################################################################################
  546. # ########## Generate filling geometry.
  547. # #########################################################################################################
  548. tool_obj.new_solid_geometry = bounding_box.difference(clearance_geometry)
  549. # determine the bounding box polygon for the entire Gerber object to which we add copper thieving
  550. # if isinstance(geo_n, list):
  551. # env_obj = unary_union(geo_n).buffer(distance=margin, join_style=base.JOIN_STYLE.mitre)
  552. # else:
  553. # env_obj = geo_n.buffer(distance=margin, join_style=base.JOIN_STYLE.mitre)
  554. #
  555. # x0, y0, x1, y1 = env_obj.bounds
  556. # bounding_box = box(x0, y0, x1, y1)
  557. tool_obj.app.proc_container.update_view_text(' %s' % _("Create geometry"))
  558. # #########################################################################################################
  559. # apply the 'margin' to the bounding box geometry
  560. # #########################################################################################################
  561. try:
  562. bounding_box = thieving_obj.solid_geometry.envelope.buffer(
  563. distance=margin,
  564. join_style=base.JOIN_STYLE.mitre
  565. )
  566. except AttributeError:
  567. bounding_box = MultiPolygon(thieving_obj.solid_geometry).envelope.buffer(
  568. distance=margin,
  569. join_style=base.JOIN_STYLE.mitre
  570. )
  571. x0, y0, x1, y1 = bounding_box.bounds
  572. # #########################################################################################################
  573. # add Thieving geometry
  574. # #########################################################################################################
  575. if fill_type == 'dot' or fill_type == 'square':
  576. # build the MultiPolygon of dots/squares that will fill the entire bounding box
  577. thieving_list = []
  578. if fill_type == 'dot':
  579. radius = dot_dia / 2.0
  580. new_x = x0 + radius
  581. new_y = y0 + radius
  582. while new_x <= x1 - radius:
  583. while new_y <= y1 - radius:
  584. dot_geo = Point((new_x, new_y)).buffer(radius, resolution=64)
  585. thieving_list.append(dot_geo)
  586. new_y += dot_dia + dot_spacing
  587. new_x += dot_dia + dot_spacing
  588. new_y = y0 + radius
  589. else:
  590. h_size = square_size / 2.0
  591. new_x = x0 + h_size
  592. new_y = y0 + h_size
  593. while new_x <= x1 - h_size:
  594. while new_y <= y1 - h_size:
  595. a, b, c, d = (Point((new_x, new_y)).buffer(h_size)).bounds
  596. square_geo = box(a, b, c, d)
  597. thieving_list.append(square_geo)
  598. new_y += square_size + square_spacing
  599. new_x += square_size + square_spacing
  600. new_y = y0 + h_size
  601. thieving_box_geo = MultiPolygon(thieving_list)
  602. dx = bounding_box.centroid.x - thieving_box_geo.centroid.x
  603. dy = bounding_box.centroid.y - thieving_box_geo.centroid.y
  604. thieving_box_geo = affinity.translate(thieving_box_geo, xoff=dx, yoff=dy)
  605. try:
  606. _it = iter(tool_obj.new_solid_geometry)
  607. except TypeError:
  608. tool_obj.new_solid_geometry = [tool_obj.new_solid_geometry]
  609. try:
  610. _it = iter(thieving_box_geo)
  611. except TypeError:
  612. thieving_box_geo = [thieving_box_geo]
  613. thieving_geo = []
  614. for dot_geo in thieving_box_geo:
  615. for geo_t in tool_obj.new_solid_geometry:
  616. if dot_geo.within(geo_t):
  617. thieving_geo.append(dot_geo)
  618. tool_obj.new_solid_geometry = thieving_geo
  619. if fill_type == 'line':
  620. half_thick_line = line_size / 2.0
  621. # create a thick polygon-line that surrounds the copper features
  622. outline_geometry = []
  623. try:
  624. for pol in tool_obj.grb_object.solid_geometry:
  625. if tool_obj.app.abort_flag:
  626. # graceful abort requested by the user
  627. raise grace
  628. outline_geometry.append(
  629. pol.buffer(c_val+half_thick_line, int(int(tool_obj.geo_steps_per_circle) / 4))
  630. )
  631. pol_nr += 1
  632. disp_number = int(np.interp(pol_nr, [0, geo_len], [0, 100]))
  633. if old_disp_number < disp_number <= 100:
  634. msg = ' %s ... %d%%' % (_("Buffering"), int(disp_number))
  635. tool_obj.app.proc_container.update_view_text(msg)
  636. old_disp_number = disp_number
  637. except TypeError:
  638. # taking care of the case when the self.solid_geometry is just a single Polygon, not a list or a
  639. # MultiPolygon (not an iterable)
  640. outline_geometry.append(
  641. tool_obj.grb_object.solid_geometry.buffer(
  642. c_val+half_thick_line,
  643. int(int(tool_obj.geo_steps_per_circle) / 4)
  644. )
  645. )
  646. tool_obj.app.proc_container.update_view_text(' %s' % _("Buffering"))
  647. outline_geometry = unary_union(outline_geometry)
  648. outline_line = []
  649. try:
  650. for geo_o in outline_geometry:
  651. outline_line.append(
  652. geo_o.exterior.buffer(
  653. half_thick_line, resolution=int(int(tool_obj.geo_steps_per_circle) / 4)
  654. )
  655. )
  656. except TypeError:
  657. outline_line.append(
  658. outline_geometry.exterior.buffer(
  659. half_thick_line, resolution=int(int(tool_obj.geo_steps_per_circle) / 4)
  660. )
  661. )
  662. outline_geometry = unary_union(outline_line)
  663. # create a polygon-line that surrounds in the inside the bounding box polygon of the target Gerber
  664. box_outline_geo = box(x0, y0, x1, y1).buffer(-half_thick_line)
  665. box_outline_geo_exterior = box_outline_geo.exterior
  666. box_outline_geometry = box_outline_geo_exterior.buffer(
  667. half_thick_line,
  668. resolution=int(int(tool_obj.geo_steps_per_circle) / 4)
  669. )
  670. bx0, by0, bx1, by1 = box_outline_geo.bounds
  671. thieving_lines_geo = []
  672. new_x = bx0
  673. new_y = by0
  674. while new_x <= x1 - half_thick_line:
  675. line_geo = LineString([(new_x, by0), (new_x, by1)]).buffer(
  676. half_thick_line,
  677. resolution=int(int(tool_obj.geo_steps_per_circle) / 4)
  678. )
  679. thieving_lines_geo.append(line_geo)
  680. new_x += line_size + line_spacing
  681. while new_y <= y1 - half_thick_line:
  682. line_geo = LineString([(bx0, new_y), (bx1, new_y)]).buffer(
  683. half_thick_line,
  684. resolution=int(int(tool_obj.geo_steps_per_circle) / 4)
  685. )
  686. thieving_lines_geo.append(line_geo)
  687. new_y += line_size + line_spacing
  688. # merge everything together
  689. diff_lines_geo = []
  690. for line_poly in thieving_lines_geo:
  691. rest_line = line_poly.difference(clearance_geometry)
  692. diff_lines_geo.append(rest_line)
  693. tool_obj.flatten([outline_geometry, box_outline_geometry, diff_lines_geo])
  694. tool_obj.new_solid_geometry = tool_obj.flat_geometry
  695. tool_obj.app.proc_container.update_view_text(' %s' % _("Append geometry"))
  696. geo_list = deepcopy(tool_obj.grb_object.solid_geometry)
  697. if isinstance(tool_obj.grb_object.solid_geometry, MultiPolygon):
  698. geo_list = list(geo_list.geoms)
  699. new_apertures = deepcopy(tool_obj.grb_object.apertures)
  700. if '0' not in new_apertures:
  701. new_apertures['0'] = {
  702. 'type': 'REG',
  703. 'size': 0.0,
  704. 'geometry': []
  705. }
  706. try:
  707. for poly in tool_obj.new_solid_geometry:
  708. # append to the new solid geometry
  709. geo_list.append(poly)
  710. # append into the '0' aperture
  711. geo_elem = {'solid': poly, 'follow': poly.exterior}
  712. new_apertures['0']['geometry'].append(deepcopy(geo_elem))
  713. except TypeError:
  714. # append to the new solid geometry
  715. geo_list.append(tool_obj.new_solid_geometry)
  716. # append into the '0' aperture
  717. geo_elem = {'solid': tool_obj.new_solid_geometry, 'follow': tool_obj.new_solid_geometry.exterior}
  718. new_apertures['0']['geometry'].append(deepcopy(geo_elem))
  719. new_solid_geo = MultiPolygon(geo_list).buffer(0.0000001).buffer(-0.0000001)
  720. outname = '%s_%s' % (str(self.grb_object.options['name']), 'thief')
  721. def initialize(grb_obj, app_obj):
  722. grb_obj.options = {}
  723. for opt in self.grb_object.options:
  724. if opt != 'name':
  725. grb_obj.options[opt] = deepcopy(self.grb_object.options[opt])
  726. grb_obj.options['name'] = outname
  727. grb_obj.multitool = False
  728. grb_obj.multigeo = False
  729. grb_obj.follow = deepcopy(self.grb_object.follow)
  730. grb_obj.apertures = new_apertures
  731. grb_obj.solid_geometry = deepcopy(new_solid_geo)
  732. grb_obj.follow_geometry = deepcopy(self.grb_object.follow_geometry)
  733. app_obj.proc_container.update_view_text(' %s' % _("Append source file"))
  734. grb_obj.source_file = app_obj.f_handlers.export_gerber(obj_name=outname, filename=None,
  735. local_use=grb_obj,
  736. use_thread=False)
  737. ret_val = self.app.app_obj.new_object('gerber', outname, initialize, plot=True)
  738. tool_obj.app.proc_container.update_view_text(' %s' % '')
  739. if ret_val == 'fail':
  740. self.app.call_source = "app"
  741. self.app.inform.emit('[ERROR_NOTCL] %s' % _("Failed."))
  742. return
  743. tool_obj.on_exit()
  744. tool_obj.app.inform.emit('[success] %s' % _("Copper Thieving Tool done."))
  745. if run_threaded:
  746. self.app.worker_task.emit({'fcn': job_thread_thieving, 'params': [self]})
  747. else:
  748. job_thread_thieving(self)
  749. def on_add_ppm_click(self):
  750. run_threaded = True
  751. if run_threaded:
  752. self.app.proc_container.new('%s ...' % _("P-Plating Mask"))
  753. else:
  754. QtWidgets.QApplication.processEvents()
  755. self.app.proc_container.view.set_busy('%s ...' % _("P-Plating Mask"))
  756. if run_threaded:
  757. self.app.worker_task.emit({'fcn': self.on_new_pattern_plating_object, 'params': []})
  758. else:
  759. self.on_new_pattern_plating_object()
  760. def on_new_pattern_plating_object(self):
  761. ppm_clearance = self.ui.clearance_ppm_entry.get_value()
  762. rb_thickness = self.rb_thickness
  763. # get the Gerber object on which the Copper thieving will be inserted
  764. selection_index = self.ui.sm_object_combo.currentIndex()
  765. model_index = self.app.collection.index(selection_index, 0, self.ui.sm_object_combo.rootModelIndex())
  766. try:
  767. self.sm_object = model_index.internalPointer().obj
  768. except Exception as e:
  769. log.debug("ToolCopperThieving.on_add_ppm_click() --> %s" % str(e))
  770. self.app.inform.emit('[WARNING_NOTCL] %s' % _("There is no Gerber object loaded ..."))
  771. return
  772. self.app.proc_container.update_view_text(' %s' % _("Append PP-M geometry"))
  773. geo_list = self.sm_object.solid_geometry
  774. if isinstance(self.sm_object.solid_geometry, MultiPolygon):
  775. geo_list = list(self.sm_object.solid_geometry.geoms)
  776. # if the clearance is negative apply it to the original soldermask too
  777. if ppm_clearance < 0:
  778. temp_geo_list = []
  779. for geo in geo_list:
  780. temp_geo_list.append(geo.buffer(ppm_clearance))
  781. geo_list = temp_geo_list
  782. plated_area = 0.0
  783. for geo in geo_list:
  784. plated_area += geo.area
  785. if self.new_solid_geometry:
  786. for geo in self.new_solid_geometry:
  787. plated_area += geo.area
  788. if self.robber_geo:
  789. plated_area += self.robber_geo.area
  790. self.ui.plated_area_entry.set_value(plated_area)
  791. thieving_solid_geo = deepcopy(self.new_solid_geometry)
  792. robber_solid_geo = deepcopy(self.robber_geo)
  793. robber_line = deepcopy(self.robber_line)
  794. def obj_init(grb_obj, app_obj):
  795. grb_obj.options = {}
  796. for opt in self.sm_object.options:
  797. if opt != 'name':
  798. grb_obj.options[opt] = deepcopy(self.sm_object.options[opt])
  799. grb_obj.options['name'] = outname
  800. grb_obj.multitool = False
  801. grb_obj.source_file = []
  802. grb_obj.multigeo = False
  803. grb_obj.follow = False
  804. grb_obj.apertures = {}
  805. grb_obj.solid_geometry = []
  806. # if we have copper thieving geometry, add it
  807. if thieving_solid_geo:
  808. if '0' not in grb_obj.apertures:
  809. grb_obj.apertures['0'] = {
  810. 'type': 'REG',
  811. 'size': 0.0,
  812. 'geometry': []
  813. }
  814. try:
  815. for poly in thieving_solid_geo:
  816. poly_b = poly.buffer(ppm_clearance)
  817. # append to the new solid geometry
  818. geo_list.append(poly_b)
  819. # append into the '0' aperture
  820. geo_elem = {
  821. 'solid': poly_b,
  822. 'follow': poly_b.exterior
  823. }
  824. grb_obj.apertures['0']['geometry'].append(deepcopy(geo_elem))
  825. except TypeError:
  826. # append to the new solid geometry
  827. assert isinstance(thieving_solid_geo, Polygon)
  828. geo_list.append(thieving_solid_geo.buffer(ppm_clearance))
  829. # append into the '0' aperture
  830. geo_elem = {
  831. 'solid': thieving_solid_geo.buffer(ppm_clearance),
  832. 'follow': thieving_solid_geo.buffer(ppm_clearance).exterior
  833. }
  834. grb_obj.apertures['0']['geometry'].append(deepcopy(geo_elem))
  835. # if we have robber bar geometry, add it
  836. if robber_solid_geo:
  837. aperture_found = None
  838. for ap_id, ap_val in grb_obj.apertures.items():
  839. if ap_val['type'] == 'C' and ap_val['size'] == app_obj.rb_thickness + ppm_clearance:
  840. aperture_found = ap_id
  841. break
  842. if aperture_found:
  843. geo_elem = {'solid': robber_solid_geo, 'follow': robber_line}
  844. grb_obj.apertures[aperture_found]['geometry'].append(deepcopy(geo_elem))
  845. else:
  846. ap_keys = list(grb_obj.apertures.keys())
  847. max_apid = int(max(ap_keys))
  848. if ap_keys and max_apid != 0:
  849. new_apid = str(max_apid + 1)
  850. else:
  851. new_apid = '10'
  852. grb_obj.apertures[new_apid] = {
  853. 'type': 'C',
  854. 'size': rb_thickness + ppm_clearance,
  855. 'geometry': []
  856. }
  857. geo_elem = {
  858. 'solid': robber_solid_geo.buffer(ppm_clearance),
  859. 'follow': Polygon(robber_line).buffer(ppm_clearance / 2.0).exterior
  860. }
  861. grb_obj.apertures[new_apid]['geometry'].append(deepcopy(geo_elem))
  862. geo_list.append(robber_solid_geo.buffer(ppm_clearance))
  863. grb_obj.solid_geometry = MultiPolygon(geo_list).buffer(0.0000001).buffer(-0.0000001)
  864. app_obj.proc_container.update_view_text(' %s' % _("Append source file"))
  865. # update the source file with the new geometry:
  866. grb_obj.source_file = app_obj.f_handlers.export_gerber(obj_name=outname, filename=None, local_use=grb_obj,
  867. use_thread=False)
  868. app_obj.proc_container.update_view_text(' %s' % '')
  869. # Object name
  870. obj_name, separatpr, obj_extension = self.sm_object.options['name'].rpartition('.')
  871. outname = '%s_%s.%s' % (obj_name, 'plating_mask', obj_extension)
  872. ret_val = self.app.app_obj.new_object('gerber', outname, obj_init, autoselected=False)
  873. if ret_val == 'fail':
  874. self.app.call_source = "app"
  875. self.app.inform.emit('[ERROR_NOTCL] %s' % _("Failed."))
  876. return
  877. # Register recent file
  878. self.app.file_opened.emit("gerber", outname)
  879. self.on_exit()
  880. self.app.inform.emit('[success] %s' % _("Generating Pattern Plating Mask done."))
  881. def replot(self, obj, run_thread=True):
  882. def worker_task():
  883. with self.app.proc_container.new('%s...' % _("Plotting")):
  884. obj.plot()
  885. self.app.app_obj.object_plotted.emit(obj)
  886. if run_thread:
  887. self.app.worker_task.emit({'fcn': worker_task, 'params': []})
  888. else:
  889. worker_task()
  890. def on_exit(self, obj=None):
  891. # plot the objects
  892. if obj:
  893. try:
  894. for ob in obj:
  895. self.replot(obj=ob)
  896. except (AttributeError, TypeError):
  897. self.replot(obj=obj)
  898. except Exception:
  899. return
  900. # reset the variables
  901. self.sel_rect = []
  902. # Events ID
  903. self.mr = None
  904. self.mm = None
  905. # Mouse cursor positions
  906. self.mouse_is_dragging = False
  907. self.cursor_pos = (0, 0)
  908. self.first_click = False
  909. # if True it means we exited from tool in the middle of area adding therefore disconnect the events
  910. if self.handlers_connected is True:
  911. self.app.delete_selection_shape()
  912. self.disconnect_event_handlers()
  913. self.app.call_source = "app"
  914. self.app.inform.emit('[success] %s' % _("Copper Thieving Tool exit."))
  915. def connect_event_handlers(self):
  916. if self.handlers_connected is False:
  917. if self.app.is_legacy is False:
  918. self.app.plotcanvas.graph_event_disconnect('mouse_press', self.app.on_mouse_click_over_plot)
  919. self.app.plotcanvas.graph_event_disconnect('mouse_move', self.app.on_mouse_move_over_plot)
  920. self.app.plotcanvas.graph_event_disconnect('mouse_release', self.app.on_mouse_click_release_over_plot)
  921. else:
  922. self.app.plotcanvas.graph_event_disconnect(self.app.mp)
  923. self.app.plotcanvas.graph_event_disconnect(self.app.mm)
  924. self.app.plotcanvas.graph_event_disconnect(self.app.mr)
  925. self.mr = self.app.plotcanvas.graph_event_connect('mouse_release', self.on_mouse_release)
  926. self.mm = self.app.plotcanvas.graph_event_connect('mouse_move', self.on_mouse_move)
  927. self.handlers_connected = True
  928. def disconnect_event_handlers(self):
  929. if self.handlers_connected is True:
  930. if self.app.is_legacy is False:
  931. self.app.plotcanvas.graph_event_disconnect('mouse_release', self.on_mouse_release)
  932. self.app.plotcanvas.graph_event_disconnect('mouse_move', self.on_mouse_move)
  933. else:
  934. self.app.plotcanvas.graph_event_disconnect(self.mr)
  935. self.app.plotcanvas.graph_event_disconnect(self.mm)
  936. self.app.mp = self.app.plotcanvas.graph_event_connect('mouse_press',
  937. self.app.on_mouse_click_over_plot)
  938. self.app.mm = self.app.plotcanvas.graph_event_connect('mouse_move',
  939. self.app.on_mouse_move_over_plot)
  940. self.app.mr = self.app.plotcanvas.graph_event_connect('mouse_release',
  941. self.app.on_mouse_click_release_over_plot)
  942. self.handlers_connected = False
  943. def flatten(self, geometry):
  944. """
  945. Creates a list of non-iterable linear geometry objects.
  946. :param geometry: Shapely type or list or list of list of such.
  947. Results are placed in self.flat_geometry
  948. """
  949. # ## If iterable, expand recursively.
  950. try:
  951. for geo in geometry:
  952. if geo is not None:
  953. self.flatten(geometry=geo)
  954. # ## Not iterable, do the actual indexing and add.
  955. except TypeError:
  956. self.flat_geometry.append(geometry)
  957. return self.flat_geometry
  958. class ThievingUI:
  959. toolName = _("Copper Thieving Tool")
  960. def __init__(self, layout, app):
  961. self.app = app
  962. self.decimals = self.app.decimals
  963. self.units = self.app.defaults['units']
  964. self.layout = layout
  965. # ## Title
  966. title_label = QtWidgets.QLabel("%s" % self.toolName)
  967. title_label.setStyleSheet("""
  968. QLabel
  969. {
  970. font-size: 16px;
  971. font-weight: bold;
  972. }
  973. """)
  974. self.layout.addWidget(title_label)
  975. self.layout.addWidget(QtWidgets.QLabel(""))
  976. # ## Grid Layout
  977. i_grid_lay = QtWidgets.QGridLayout()
  978. self.layout.addLayout(i_grid_lay)
  979. i_grid_lay.setColumnStretch(0, 0)
  980. i_grid_lay.setColumnStretch(1, 1)
  981. self.grb_object_combo = FCComboBox()
  982. self.grb_object_combo.setModel(self.app.collection)
  983. self.grb_object_combo.setRootModelIndex(self.app.collection.index(0, 0, QtCore.QModelIndex()))
  984. self.grb_object_combo.is_last = True
  985. self.grb_object_combo.obj_type = 'Gerber'
  986. self.grbobj_label = QtWidgets.QLabel("<b>%s:</b>" % _("GERBER"))
  987. self.grbobj_label.setToolTip(
  988. _("Gerber Object to which will be added a copper thieving.")
  989. )
  990. i_grid_lay.addWidget(self.grbobj_label, 0, 0)
  991. i_grid_lay.addWidget(self.grb_object_combo, 1, 0, 1, 2)
  992. separator_line = QtWidgets.QFrame()
  993. separator_line.setFrameShape(QtWidgets.QFrame.HLine)
  994. separator_line.setFrameShadow(QtWidgets.QFrame.Sunken)
  995. i_grid_lay.addWidget(separator_line, 2, 0, 1, 2)
  996. # ## Grid Layout
  997. grid_lay = QtWidgets.QGridLayout()
  998. self.layout.addLayout(grid_lay)
  999. grid_lay.setColumnStretch(0, 0)
  1000. grid_lay.setColumnStretch(1, 1)
  1001. self.copper_fill_label = QtWidgets.QLabel('<b>%s</b>' % _('Parameters'))
  1002. self.copper_fill_label.setToolTip(
  1003. _("Parameters used for this tool.")
  1004. )
  1005. grid_lay.addWidget(self.copper_fill_label, 0, 0, 1, 2)
  1006. # CLEARANCE #
  1007. self.clearance_label = QtWidgets.QLabel('%s:' % _("Clearance"))
  1008. self.clearance_label.setToolTip(
  1009. _("This set the distance between the copper thieving components\n"
  1010. "(the polygon fill may be split in multiple polygons)\n"
  1011. "and the copper traces in the Gerber file.")
  1012. )
  1013. self.clearance_entry = FCDoubleSpinner(callback=self.confirmation_message)
  1014. self.clearance_entry.set_range(0.00001, 9999.9999)
  1015. self.clearance_entry.set_precision(self.decimals)
  1016. self.clearance_entry.setSingleStep(0.1)
  1017. grid_lay.addWidget(self.clearance_label, 1, 0)
  1018. grid_lay.addWidget(self.clearance_entry, 1, 1)
  1019. # MARGIN #
  1020. self.margin_label = QtWidgets.QLabel('%s:' % _("Margin"))
  1021. self.margin_label.setToolTip(
  1022. _("Bounding box margin.")
  1023. )
  1024. self.margin_entry = FCDoubleSpinner(callback=self.confirmation_message)
  1025. self.margin_entry.set_range(0.0, 9999.9999)
  1026. self.margin_entry.set_precision(self.decimals)
  1027. self.margin_entry.setSingleStep(0.1)
  1028. grid_lay.addWidget(self.margin_label, 2, 0)
  1029. grid_lay.addWidget(self.margin_entry, 2, 1)
  1030. # Reference #
  1031. self.reference_radio = RadioSet([
  1032. {'label': _('Itself'), 'value': 'itself'},
  1033. {"label": _("Area Selection"), "value": "area"},
  1034. {'label': _("Reference Object"), 'value': 'box'}
  1035. ], orientation='vertical', stretch=False)
  1036. self.reference_label = QtWidgets.QLabel(_("Reference:"))
  1037. self.reference_label.setToolTip(
  1038. _("- 'Itself' - the copper thieving extent is based on the object extent.\n"
  1039. "- 'Area Selection' - left mouse click to start selection of the area to be filled.\n"
  1040. "- 'Reference Object' - will do copper thieving within the area specified by another object.")
  1041. )
  1042. grid_lay.addWidget(self.reference_label, 3, 0)
  1043. grid_lay.addWidget(self.reference_radio, 3, 1)
  1044. self.ref_combo_type_label = QtWidgets.QLabel('%s:' % _("Ref. Type"))
  1045. self.ref_combo_type_label.setToolTip(
  1046. _("The type of FlatCAM object to be used as copper thieving reference.\n"
  1047. "It can be Gerber, Excellon or Geometry.")
  1048. )
  1049. self.ref_combo_type = FCComboBox()
  1050. self.ref_combo_type.addItems([_("Gerber"), _("Excellon"), _("Geometry")])
  1051. grid_lay.addWidget(self.ref_combo_type_label, 4, 0)
  1052. grid_lay.addWidget(self.ref_combo_type, 4, 1)
  1053. self.ref_combo_label = QtWidgets.QLabel('%s:' % _("Ref. Object"))
  1054. self.ref_combo_label.setToolTip(
  1055. _("The FlatCAM object to be used as non copper clearing reference.")
  1056. )
  1057. self.ref_combo = FCComboBox()
  1058. self.ref_combo.setModel(self.app.collection)
  1059. self.ref_combo.setRootModelIndex(self.app.collection.index(0, 0, QtCore.QModelIndex()))
  1060. self.ref_combo.is_last = True
  1061. self.ref_combo.obj_type = {
  1062. _("Gerber"): "Gerber", _("Excellon"): "Excellon", _("Geometry"): "Geometry"
  1063. }[self.ref_combo_type.get_value()]
  1064. grid_lay.addWidget(self.ref_combo_label, 5, 0)
  1065. grid_lay.addWidget(self.ref_combo, 5, 1)
  1066. self.ref_combo.hide()
  1067. self.ref_combo_label.hide()
  1068. self.ref_combo_type.hide()
  1069. self.ref_combo_type_label.hide()
  1070. # Bounding Box Type #
  1071. self.bbox_type_radio = RadioSet([
  1072. {'label': _('Rectangular'), 'value': 'rect'},
  1073. {"label": _("Minimal"), "value": "min"}
  1074. ], stretch=False)
  1075. self.bbox_type_label = QtWidgets.QLabel(_("Box Type:"))
  1076. self.bbox_type_label.setToolTip(
  1077. _("- 'Rectangular' - the bounding box will be of rectangular shape.\n"
  1078. "- 'Minimal' - the bounding box will be the convex hull shape.")
  1079. )
  1080. grid_lay.addWidget(self.bbox_type_label, 6, 0)
  1081. grid_lay.addWidget(self.bbox_type_radio, 6, 1)
  1082. self.bbox_type_label.hide()
  1083. self.bbox_type_radio.hide()
  1084. separator_line = QtWidgets.QFrame()
  1085. separator_line.setFrameShape(QtWidgets.QFrame.HLine)
  1086. separator_line.setFrameShadow(QtWidgets.QFrame.Sunken)
  1087. grid_lay.addWidget(separator_line, 7, 0, 1, 2)
  1088. # Fill Type
  1089. self.fill_type_radio = RadioSet([
  1090. {'label': _('Solid'), 'value': 'solid'},
  1091. {"label": _("Dots Grid"), "value": "dot"},
  1092. {"label": _("Squares Grid"), "value": "square"},
  1093. {"label": _("Lines Grid"), "value": "line"}
  1094. ], orientation='vertical', stretch=False)
  1095. self.fill_type_label = QtWidgets.QLabel(_("Fill Type:"))
  1096. self.fill_type_label.setToolTip(
  1097. _("- 'Solid' - copper thieving will be a solid polygon.\n"
  1098. "- 'Dots Grid' - the empty area will be filled with a pattern of dots.\n"
  1099. "- 'Squares Grid' - the empty area will be filled with a pattern of squares.\n"
  1100. "- 'Lines Grid' - the empty area will be filled with a pattern of lines.")
  1101. )
  1102. grid_lay.addWidget(self.fill_type_label, 8, 0)
  1103. grid_lay.addWidget(self.fill_type_radio, 8, 1)
  1104. # DOTS FRAME
  1105. self.dots_frame = QtWidgets.QFrame()
  1106. self.dots_frame.setContentsMargins(0, 0, 0, 0)
  1107. self.layout.addWidget(self.dots_frame)
  1108. dots_grid = QtWidgets.QGridLayout()
  1109. dots_grid.setColumnStretch(0, 0)
  1110. dots_grid.setColumnStretch(1, 1)
  1111. dots_grid.setContentsMargins(0, 0, 0, 0)
  1112. self.dots_frame.setLayout(dots_grid)
  1113. self.dots_frame.hide()
  1114. self.dots_label = QtWidgets.QLabel('<b>%s</b>:' % _("Dots Grid Parameters"))
  1115. dots_grid.addWidget(self.dots_label, 0, 0, 1, 2)
  1116. # Dot diameter #
  1117. self.dotdia_label = QtWidgets.QLabel('%s:' % _("Dia"))
  1118. self.dotdia_label.setToolTip(
  1119. _("Dot diameter in Dots Grid.")
  1120. )
  1121. self.dot_dia_entry = FCDoubleSpinner(callback=self.confirmation_message)
  1122. self.dot_dia_entry.set_range(0.0, 9999.9999)
  1123. self.dot_dia_entry.set_precision(self.decimals)
  1124. self.dot_dia_entry.setSingleStep(0.1)
  1125. dots_grid.addWidget(self.dotdia_label, 1, 0)
  1126. dots_grid.addWidget(self.dot_dia_entry, 1, 1)
  1127. # Dot spacing #
  1128. self.dotspacing_label = QtWidgets.QLabel('%s:' % _("Spacing"))
  1129. self.dotspacing_label.setToolTip(
  1130. _("Distance between each two dots in Dots Grid.")
  1131. )
  1132. self.dot_spacing_entry = FCDoubleSpinner(callback=self.confirmation_message)
  1133. self.dot_spacing_entry.set_range(0.0, 9999.9999)
  1134. self.dot_spacing_entry.set_precision(self.decimals)
  1135. self.dot_spacing_entry.setSingleStep(0.1)
  1136. dots_grid.addWidget(self.dotspacing_label, 2, 0)
  1137. dots_grid.addWidget(self.dot_spacing_entry, 2, 1)
  1138. # SQUARES FRAME
  1139. self.squares_frame = QtWidgets.QFrame()
  1140. self.squares_frame.setContentsMargins(0, 0, 0, 0)
  1141. self.layout.addWidget(self.squares_frame)
  1142. squares_grid = QtWidgets.QGridLayout()
  1143. squares_grid.setColumnStretch(0, 0)
  1144. squares_grid.setColumnStretch(1, 1)
  1145. squares_grid.setContentsMargins(0, 0, 0, 0)
  1146. self.squares_frame.setLayout(squares_grid)
  1147. self.squares_frame.hide()
  1148. self.squares_label = QtWidgets.QLabel('<b>%s</b>:' % _("Squares Grid Parameters"))
  1149. squares_grid.addWidget(self.squares_label, 0, 0, 1, 2)
  1150. # Square Size #
  1151. self.square_size_label = QtWidgets.QLabel('%s:' % _("Size"))
  1152. self.square_size_label.setToolTip(
  1153. _("Square side size in Squares Grid.")
  1154. )
  1155. self.square_size_entry = FCDoubleSpinner(callback=self.confirmation_message)
  1156. self.square_size_entry.set_range(0.0, 9999.9999)
  1157. self.square_size_entry.set_precision(self.decimals)
  1158. self.square_size_entry.setSingleStep(0.1)
  1159. squares_grid.addWidget(self.square_size_label, 1, 0)
  1160. squares_grid.addWidget(self.square_size_entry, 1, 1)
  1161. # Squares spacing #
  1162. self.squares_spacing_label = QtWidgets.QLabel('%s:' % _("Spacing"))
  1163. self.squares_spacing_label.setToolTip(
  1164. _("Distance between each two squares in Squares Grid.")
  1165. )
  1166. self.squares_spacing_entry = FCDoubleSpinner(callback=self.confirmation_message)
  1167. self.squares_spacing_entry.set_range(0.0, 9999.9999)
  1168. self.squares_spacing_entry.set_precision(self.decimals)
  1169. self.squares_spacing_entry.setSingleStep(0.1)
  1170. squares_grid.addWidget(self.squares_spacing_label, 2, 0)
  1171. squares_grid.addWidget(self.squares_spacing_entry, 2, 1)
  1172. # LINES FRAME
  1173. self.lines_frame = QtWidgets.QFrame()
  1174. self.lines_frame.setContentsMargins(0, 0, 0, 0)
  1175. self.layout.addWidget(self.lines_frame)
  1176. lines_grid = QtWidgets.QGridLayout()
  1177. lines_grid.setColumnStretch(0, 0)
  1178. lines_grid.setColumnStretch(1, 1)
  1179. lines_grid.setContentsMargins(0, 0, 0, 0)
  1180. self.lines_frame.setLayout(lines_grid)
  1181. self.lines_frame.hide()
  1182. self.lines_label = QtWidgets.QLabel('<b>%s</b>:' % _("Lines Grid Parameters"))
  1183. lines_grid.addWidget(self.lines_label, 0, 0, 1, 2)
  1184. # Square Size #
  1185. self.line_size_label = QtWidgets.QLabel('%s:' % _("Size"))
  1186. self.line_size_label.setToolTip(
  1187. _("Line thickness size in Lines Grid.")
  1188. )
  1189. self.line_size_entry = FCDoubleSpinner(callback=self.confirmation_message)
  1190. self.line_size_entry.set_range(0.0, 9999.9999)
  1191. self.line_size_entry.set_precision(self.decimals)
  1192. self.line_size_entry.setSingleStep(0.1)
  1193. lines_grid.addWidget(self.line_size_label, 1, 0)
  1194. lines_grid.addWidget(self.line_size_entry, 1, 1)
  1195. # Lines spacing #
  1196. self.lines_spacing_label = QtWidgets.QLabel('%s:' % _("Spacing"))
  1197. self.lines_spacing_label.setToolTip(
  1198. _("Distance between each two lines in Lines Grid.")
  1199. )
  1200. self.lines_spacing_entry = FCDoubleSpinner(callback=self.confirmation_message)
  1201. self.lines_spacing_entry.set_range(0.0, 9999.9999)
  1202. self.lines_spacing_entry.set_precision(self.decimals)
  1203. self.lines_spacing_entry.setSingleStep(0.1)
  1204. lines_grid.addWidget(self.lines_spacing_label, 2, 0)
  1205. lines_grid.addWidget(self.lines_spacing_entry, 2, 1)
  1206. # ## Insert Copper Thieving
  1207. self.fill_button = QtWidgets.QPushButton(_("Insert Copper thieving"))
  1208. self.fill_button.setIcon(QtGui.QIcon(self.app.resource_location + '/copperfill32.png'))
  1209. self.fill_button.setToolTip(
  1210. _("Will add a polygon (may be split in multiple parts)\n"
  1211. "that will surround the actual Gerber traces at a certain distance.")
  1212. )
  1213. self.fill_button.setStyleSheet("""
  1214. QPushButton
  1215. {
  1216. font-weight: bold;
  1217. }
  1218. """)
  1219. self.layout.addWidget(self.fill_button)
  1220. # ## Grid Layout
  1221. grid_lay_1 = QtWidgets.QGridLayout()
  1222. self.layout.addLayout(grid_lay_1)
  1223. grid_lay_1.setColumnStretch(0, 0)
  1224. grid_lay_1.setColumnStretch(1, 1)
  1225. grid_lay_1.setColumnStretch(2, 0)
  1226. separator_line_1 = QtWidgets.QFrame()
  1227. separator_line_1.setFrameShape(QtWidgets.QFrame.HLine)
  1228. separator_line_1.setFrameShadow(QtWidgets.QFrame.Sunken)
  1229. grid_lay_1.addWidget(separator_line_1, 0, 0, 1, 3)
  1230. grid_lay_1.addWidget(QtWidgets.QLabel(''))
  1231. self.robber_bar_label = QtWidgets.QLabel('<b>%s</b>' % _('Robber Bar Parameters'))
  1232. self.robber_bar_label.setToolTip(
  1233. _("Parameters used for the robber bar.\n"
  1234. "Robber bar = copper border to help in pattern hole plating.")
  1235. )
  1236. grid_lay_1.addWidget(self.robber_bar_label, 1, 0, 1, 3)
  1237. # ROBBER BAR MARGIN #
  1238. self.rb_margin_label = QtWidgets.QLabel('%s:' % _("Margin"))
  1239. self.rb_margin_label.setToolTip(
  1240. _("Bounding box margin for robber bar.")
  1241. )
  1242. self.rb_margin_entry = FCDoubleSpinner(callback=self.confirmation_message)
  1243. self.rb_margin_entry.set_range(-9999.9999, 9999.9999)
  1244. self.rb_margin_entry.set_precision(self.decimals)
  1245. self.rb_margin_entry.setSingleStep(0.1)
  1246. grid_lay_1.addWidget(self.rb_margin_label, 2, 0)
  1247. grid_lay_1.addWidget(self.rb_margin_entry, 2, 1, 1, 2)
  1248. # THICKNESS #
  1249. self.rb_thickness_label = QtWidgets.QLabel('%s:' % _("Thickness"))
  1250. self.rb_thickness_label.setToolTip(
  1251. _("The robber bar thickness.")
  1252. )
  1253. self.rb_thickness_entry = FCDoubleSpinner(callback=self.confirmation_message)
  1254. self.rb_thickness_entry.set_range(0.0000, 9999.9999)
  1255. self.rb_thickness_entry.set_precision(self.decimals)
  1256. self.rb_thickness_entry.setSingleStep(0.1)
  1257. grid_lay_1.addWidget(self.rb_thickness_label, 3, 0)
  1258. grid_lay_1.addWidget(self.rb_thickness_entry, 3, 1, 1, 2)
  1259. # ## Insert Robber Bar
  1260. self.rb_button = QtWidgets.QPushButton(_("Insert Robber Bar"))
  1261. self.rb_button.setIcon(QtGui.QIcon(self.app.resource_location + '/robber32.png'))
  1262. self.rb_button.setToolTip(
  1263. _("Will add a polygon with a defined thickness\n"
  1264. "that will surround the actual Gerber object\n"
  1265. "at a certain distance.\n"
  1266. "Required when doing holes pattern plating.")
  1267. )
  1268. self.rb_button.setStyleSheet("""
  1269. QPushButton
  1270. {
  1271. font-weight: bold;
  1272. }
  1273. """)
  1274. grid_lay_1.addWidget(self.rb_button, 4, 0, 1, 3)
  1275. separator_line_2 = QtWidgets.QFrame()
  1276. separator_line_2.setFrameShape(QtWidgets.QFrame.HLine)
  1277. separator_line_2.setFrameShadow(QtWidgets.QFrame.Sunken)
  1278. grid_lay_1.addWidget(separator_line_2, 5, 0, 1, 3)
  1279. self.patern_mask_label = QtWidgets.QLabel('<b>%s</b>' % _('Pattern Plating Mask'))
  1280. self.patern_mask_label.setToolTip(
  1281. _("Generate a mask for pattern plating.")
  1282. )
  1283. grid_lay_1.addWidget(self.patern_mask_label, 6, 0, 1, 3)
  1284. self.sm_obj_label = QtWidgets.QLabel("%s:" % _("Select Soldermask object"))
  1285. self.sm_obj_label.setToolTip(
  1286. _("Gerber Object with the soldermask.\n"
  1287. "It will be used as a base for\n"
  1288. "the pattern plating mask.")
  1289. )
  1290. self.sm_object_combo = FCComboBox()
  1291. self.sm_object_combo.setModel(self.app.collection)
  1292. self.sm_object_combo.setRootModelIndex(self.app.collection.index(0, 0, QtCore.QModelIndex()))
  1293. self.sm_object_combo.is_last = True
  1294. self.sm_object_combo.obj_type = 'Gerber'
  1295. grid_lay_1.addWidget(self.sm_obj_label, 7, 0, 1, 3)
  1296. grid_lay_1.addWidget(self.sm_object_combo, 8, 0, 1, 3)
  1297. # Openings CLEARANCE #
  1298. self.clearance_ppm_label = QtWidgets.QLabel('%s:' % _("Clearance"))
  1299. self.clearance_ppm_label.setToolTip(
  1300. _("The distance between the possible copper thieving elements\n"
  1301. "and/or robber bar and the actual openings in the mask.")
  1302. )
  1303. self.clearance_ppm_entry = FCDoubleSpinner(callback=self.confirmation_message)
  1304. self.clearance_ppm_entry.set_range(-9999.9999, 9999.9999)
  1305. self.clearance_ppm_entry.set_precision(self.decimals)
  1306. self.clearance_ppm_entry.setSingleStep(0.1)
  1307. grid_lay_1.addWidget(self.clearance_ppm_label, 9, 0)
  1308. grid_lay_1.addWidget(self.clearance_ppm_entry, 9, 1, 1, 2)
  1309. # Plated area
  1310. self.plated_area_label = QtWidgets.QLabel('%s:' % _("Plated area"))
  1311. self.plated_area_label.setToolTip(
  1312. _("The area to be plated by pattern plating.\n"
  1313. "Basically is made from the openings in the plating mask.\n\n"
  1314. "<<WARNING>> - the calculated area is actually a bit larger\n"
  1315. "due of the fact that the soldermask openings are by design\n"
  1316. "a bit larger than the copper pads, and this area is\n"
  1317. "calculated from the soldermask openings.")
  1318. )
  1319. self.plated_area_entry = FCEntry()
  1320. self.plated_area_entry.setDisabled(True)
  1321. if self.units.upper() == 'MM':
  1322. self.units_area_label = QtWidgets.QLabel('%s<sup>2</sup>' % _("mm"))
  1323. else:
  1324. self.units_area_label = QtWidgets.QLabel('%s<sup>2</sup>' % _("in"))
  1325. grid_lay_1.addWidget(self.plated_area_label, 10, 0)
  1326. grid_lay_1.addWidget(self.plated_area_entry, 10, 1)
  1327. grid_lay_1.addWidget(self.units_area_label, 10, 2)
  1328. # ## Pattern Plating Mask
  1329. self.ppm_button = QtWidgets.QPushButton(_("Generate pattern plating mask"))
  1330. self.ppm_button.setIcon(QtGui.QIcon(self.app.resource_location + '/pattern32.png'))
  1331. self.ppm_button.setToolTip(
  1332. _("Will add to the soldermask gerber geometry\n"
  1333. "the geometries of the copper thieving and/or\n"
  1334. "the robber bar if those were generated.")
  1335. )
  1336. self.ppm_button.setStyleSheet("""
  1337. QPushButton
  1338. {
  1339. font-weight: bold;
  1340. }
  1341. """)
  1342. grid_lay_1.addWidget(self.ppm_button, 11, 0, 1, 3)
  1343. self.layout.addStretch()
  1344. # ## Reset Tool
  1345. self.reset_button = QtWidgets.QPushButton(_("Reset Tool"))
  1346. self.reset_button.setIcon(QtGui.QIcon(self.app.resource_location + '/reset32.png'))
  1347. self.reset_button.setToolTip(
  1348. _("Will reset the tool parameters.")
  1349. )
  1350. self.reset_button.setStyleSheet("""
  1351. QPushButton
  1352. {
  1353. font-weight: bold;
  1354. }
  1355. """)
  1356. self.layout.addWidget(self.reset_button)
  1357. # #################################### FINSIHED GUI ###########################
  1358. # #############################################################################
  1359. def confirmation_message(self, accepted, minval, maxval):
  1360. if accepted is False:
  1361. self.app.inform[str, bool].emit('[WARNING_NOTCL] %s: [%.*f, %.*f]' % (_("Edited value is out of range"),
  1362. self.decimals,
  1363. minval,
  1364. self.decimals,
  1365. maxval), False)
  1366. else:
  1367. self.app.inform[str, bool].emit('[success] %s' % _("Edited value is within limits."), False)
  1368. def confirmation_message_int(self, accepted, minval, maxval):
  1369. if accepted is False:
  1370. self.app.inform[str, bool].emit('[WARNING_NOTCL] %s: [%d, %d]' %
  1371. (_("Edited value is out of range"), minval, maxval), False)
  1372. else:
  1373. self.app.inform[str, bool].emit('[success] %s' % _("Edited value is within limits."), False)