ToolCopperThieving.py 68 KB

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