ToolProperties.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320
  1. # ########################################################## ##
  2. # FlatCAM: 2D Post-processing for Manufacturing #
  3. # http://flatcam.org #
  4. # File Author: Marius Adrian Stanciu (c) #
  5. # Date: 3/10/2019 #
  6. # MIT Licence #
  7. # ########################################################## ##
  8. from PyQt5 import QtGui, QtCore, QtWidgets
  9. from PyQt5.QtCore import Qt
  10. from FlatCAMTool import FlatCAMTool
  11. from FlatCAMObj import *
  12. import gettext
  13. import FlatCAMTranslation as fcTranslate
  14. import builtins
  15. fcTranslate.apply_language('strings')
  16. if '_' not in builtins.__dict__:
  17. _ = gettext.gettext
  18. class Properties(FlatCAMTool):
  19. toolName = _("Properties")
  20. area_finished = pyqtSignal(float, object)
  21. def __init__(self, app):
  22. FlatCAMTool.__init__(self, app)
  23. self.setSizePolicy(QtWidgets.QSizePolicy.Ignored, QtWidgets.QSizePolicy.Ignored)
  24. # this way I can hide/show the frame
  25. self.properties_frame = QtWidgets.QFrame()
  26. self.properties_frame.setContentsMargins(0, 0, 0, 0)
  27. self.layout.addWidget(self.properties_frame)
  28. self.properties_box = QtWidgets.QVBoxLayout()
  29. self.properties_box.setContentsMargins(0, 0, 0, 0)
  30. self.properties_frame.setLayout(self.properties_box)
  31. # ## Title
  32. title_label = QtWidgets.QLabel("%s" % self.toolName)
  33. title_label.setStyleSheet("""
  34. QLabel
  35. {
  36. font-size: 16px;
  37. font-weight: bold;
  38. }
  39. """)
  40. self.properties_box.addWidget(title_label)
  41. # self.layout.setMargin(0) # PyQt4
  42. self.properties_box.setContentsMargins(0, 0, 0, 0) # PyQt5
  43. self.vlay = QtWidgets.QVBoxLayout()
  44. self.properties_box.addLayout(self.vlay)
  45. self.treeWidget = QtWidgets.QTreeWidget()
  46. self.treeWidget.setColumnCount(2)
  47. self.treeWidget.setHeaderHidden(True)
  48. self.treeWidget.header().setSectionResizeMode(QtWidgets.QHeaderView.ResizeToContents)
  49. self.treeWidget.setSizePolicy(QtWidgets.QSizePolicy.Ignored, QtWidgets.QSizePolicy.Expanding)
  50. self.vlay.addWidget(self.treeWidget)
  51. self.vlay.setStretch(0, 0)
  52. self.area_finished.connect(self.show_area_chull)
  53. def run(self, toggle=True):
  54. self.app.report_usage("ToolProperties()")
  55. if self.app.tool_tab_locked is True:
  56. return
  57. if toggle:
  58. # if the splitter is hidden, display it, else hide it but only if the current widget is the same
  59. if self.app.ui.splitter.sizes()[0] == 0:
  60. self.app.ui.splitter.setSizes([1, 1])
  61. else:
  62. try:
  63. if self.app.ui.tool_scroll_area.widget().objectName() == self.toolName:
  64. # if tab is populated with the tool but it does not have the focus, focus on it
  65. if not self.app.ui.notebook.currentWidget() is self.app.ui.tool_tab:
  66. # focus on Tool Tab
  67. self.app.ui.notebook.setCurrentWidget(self.app.ui.tool_tab)
  68. else:
  69. self.app.ui.splitter.setSizes([0, 1])
  70. except AttributeError:
  71. pass
  72. else:
  73. if self.app.ui.splitter.sizes()[0] == 0:
  74. self.app.ui.splitter.setSizes([1, 1])
  75. FlatCAMTool.run(self)
  76. self.set_tool_ui()
  77. self.properties()
  78. def install(self, icon=None, separator=None, **kwargs):
  79. FlatCAMTool.install(self, icon, separator, shortcut='P', **kwargs)
  80. def set_tool_ui(self):
  81. # this reset the TreeWidget
  82. self.treeWidget.clear()
  83. self.properties_frame.show()
  84. def properties(self):
  85. obj_list = self.app.collection.get_selected()
  86. if not obj_list:
  87. self.app.inform.emit(_("[ERROR_NOTCL] Properties Tool was not displayed. No object selected."))
  88. self.app.ui.notebook.setTabText(2, _("Tools"))
  89. self.properties_frame.hide()
  90. self.app.ui.notebook.setCurrentWidget(self.app.ui.project_tab)
  91. return
  92. for obj in obj_list:
  93. self.addItems(obj)
  94. self.app.inform.emit(_("[success] Object Properties are displayed."))
  95. self.app.ui.notebook.setTabText(2, _("Properties Tool"))
  96. def addItems(self, obj):
  97. parent = self.treeWidget.invisibleRootItem()
  98. apertures = ''
  99. tools = ''
  100. font = QtGui.QFont()
  101. font.setBold(True)
  102. obj_type = self.addParent(parent, _('TYPE'), expanded=True, color=QtGui.QColor("#000000"), font=font)
  103. obj_name = self.addParent(parent, _('NAME'), expanded=True, color=QtGui.QColor("#000000"), font=font)
  104. dims = self.addParent(parent, _('Dimensions'), expanded=True, color=QtGui.QColor("#000000"), font=font)
  105. units = self.addParent(parent, _('Units'), expanded=True, color=QtGui.QColor("#000000"), font=font)
  106. options = self.addParent(parent, _('Options'), color=QtGui.QColor("#000000"), font=font)
  107. if obj.kind.lower() == 'gerber':
  108. apertures = self.addParent(parent, _('Apertures'), expanded=True, color=QtGui.QColor("#000000"), font=font)
  109. else:
  110. tools = self.addParent(parent, _('Tools'), expanded=True, color=QtGui.QColor("#000000"), font=font)
  111. separator = self.addParent(parent, '')
  112. self.addChild(obj_type, ['%s:' % _('Object Type'), ('%s' % (obj.kind.capitalize()))], True)
  113. try:
  114. self.addChild(obj_type,
  115. ['%s:' % _('Geo Type'),
  116. ('%s' % ({False: _("Single-Geo"), True: _("Multi-Geo")}[obj.multigeo]))],
  117. True)
  118. except Exception as e:
  119. log.debug("Properties.addItems() --> %s" % str(e))
  120. self.addChild(obj_name, [obj.options['name']])
  121. # calculate physical dimensions
  122. try:
  123. xmin, ymin, xmax, ymax = obj.bounds()
  124. except Exception as e:
  125. log.debug("PropertiesTool.addItems() --> %s" % str(e))
  126. return
  127. length = abs(xmax - xmin)
  128. width = abs(ymax - ymin)
  129. self.addChild(dims, ['%s:' % _('Length'), '%.4f %s' % (
  130. length, self.app.ui.general_defaults_form.general_app_group.units_radio.get_value().lower())], True)
  131. self.addChild(dims, ['%s:' % _('Width'), '%.4f %s' % (
  132. width, self.app.ui.general_defaults_form.general_app_group.units_radio.get_value().lower())], True)
  133. # calculate and add box area
  134. if self.app.ui.general_defaults_form.general_app_group.units_radio.get_value().lower() == 'mm':
  135. area = (length * width) / 100
  136. self.addChild(dims, ['%s:' % _('Box Area'), '%.4f %s' % (area, 'cm2')], True)
  137. else:
  138. area = length * width
  139. self.addChild(dims, ['%s:' % _('Box Area'), '%.4f %s' % (area, 'in2')], True)
  140. if not isinstance(obj, FlatCAMCNCjob):
  141. def job_thread():
  142. proc = self.app.proc_container.new(_("Calculating area ... Please wait."))
  143. # calculate and add convex hull area
  144. geo = obj.solid_geometry
  145. if geo:
  146. if isinstance(geo, MultiPolygon):
  147. env_obj = geo.convex_hull
  148. elif (isinstance(geo, MultiPolygon) and len(geo) == 1) or \
  149. (isinstance(geo, list) and len(geo) == 1) and isinstance(geo[0], Polygon):
  150. env_obj = cascaded_union(obj.solid_geometry)
  151. env_obj = env_obj.convex_hull
  152. else:
  153. env_obj = cascaded_union(obj.solid_geometry)
  154. env_obj = env_obj.convex_hull
  155. area_chull = env_obj.area
  156. else:
  157. try:
  158. area_chull = []
  159. for tool in obj.tools:
  160. area_el = cascaded_union(obj.tools[tool]['solid_geometry']).convex_hull
  161. area_chull.append(area_el.area)
  162. area_chull = max(area_chull)
  163. except Exception as e:
  164. area_chull = None
  165. log.debug("Properties.addItems() --> %s" % str(e))
  166. self.area_finished.emit(area_chull, dims)
  167. self.app.worker_task.emit({'fcn': job_thread, 'params': []})
  168. self.addChild(units,
  169. ['FlatCAM units:',
  170. {
  171. 'in': _('Inch'),
  172. 'mm': _('Metric')
  173. }
  174. [str(self.app.ui.general_defaults_form.general_app_group.units_radio.get_value().lower())]
  175. ],
  176. True
  177. )
  178. for option in obj.options:
  179. if option is 'name':
  180. continue
  181. self.addChild(options, [str(option), str(obj.options[option])], True)
  182. if obj.kind.lower() == 'gerber':
  183. temp_ap = dict()
  184. for ap in obj.apertures:
  185. temp_ap.clear()
  186. temp_ap = deepcopy(obj.apertures[ap])
  187. temp_ap.pop('geometry', None)
  188. solid_nr = 0
  189. follow_nr = 0
  190. clear_nr = 0
  191. if 'geometry' in obj.apertures[ap]:
  192. if obj.apertures[ap]['geometry']:
  193. font.setBold(True)
  194. for el in obj.apertures[ap]['geometry']:
  195. if 'solid' in el:
  196. solid_nr += 1
  197. if 'follow' in el:
  198. follow_nr += 1
  199. if 'clear' in el:
  200. clear_nr += 1
  201. else:
  202. font.setBold(False)
  203. temp_ap['Solid_Geo'] = '%s Polygons' % str(solid_nr)
  204. temp_ap['Follow_Geo'] = '%s LineStrings' % str(follow_nr)
  205. temp_ap['Clear_Geo'] = '%s Polygons' % str(clear_nr)
  206. apid = self.addParent(apertures, str(ap), expanded=False, color=QtGui.QColor("#000000"), font=font)
  207. for key in temp_ap:
  208. self.addChild(apid, [str(key), str(temp_ap[key])], True)
  209. elif obj.kind.lower() == 'excellon':
  210. for tool, value in obj.tools.items():
  211. self.addChild(tools, [str(tool), str(value['C'])], True)
  212. elif obj.kind.lower() == 'geometry':
  213. for tool, value in obj.tools.items():
  214. geo_tool = self.addParent(tools, str(tool), expanded=True, color=QtGui.QColor("#000000"), font=font)
  215. for k, v in value.items():
  216. if k == 'solid_geometry':
  217. printed_value = _('Present') if v else _('None')
  218. self.addChild(geo_tool, [str(k), printed_value], True)
  219. elif k == 'data':
  220. tool_data = self.addParent(geo_tool, str(k).capitalize(),
  221. color=QtGui.QColor("#000000"), font=font)
  222. for data_k, data_v in v.items():
  223. self.addChild(tool_data, [str(data_k), str(data_v)], True)
  224. else:
  225. self.addChild(geo_tool, [str(k), str(v)], True)
  226. elif obj.kind.lower() == 'cncjob':
  227. for tool, value in obj.cnc_tools.items():
  228. geo_tool = self.addParent(tools, str(tool), expanded=True, color=QtGui.QColor("#000000"), font=font)
  229. for k, v in value.items():
  230. if k == 'solid_geometry':
  231. printed_value = _('Present') if v else _('None')
  232. self.addChild(geo_tool, [str(k), printed_value], True)
  233. elif k == 'gcode':
  234. printed_value = _('Present') if v != '' else _('None')
  235. self.addChild(geo_tool, [str(k), printed_value], True)
  236. elif k == 'gcode_parsed':
  237. printed_value = _('Present') if v else _('None')
  238. self.addChild(geo_tool, [str(k), printed_value], True)
  239. elif k == 'data':
  240. tool_data = self.addParent(geo_tool, str(k).capitalize(),
  241. color=QtGui.QColor("#000000"), font=font)
  242. for data_k, data_v in v.items():
  243. self.addChild(tool_data, [str(data_k), str(data_v)], True)
  244. else:
  245. self.addChild(geo_tool, [str(k), str(v)], True)
  246. self.addChild(separator, [''])
  247. def addParent(self, parent, title, expanded=False, color=None, font=None):
  248. item = QtWidgets.QTreeWidgetItem(parent, [title])
  249. item.setChildIndicatorPolicy(QtWidgets.QTreeWidgetItem.ShowIndicator)
  250. item.setExpanded(expanded)
  251. if color is not None:
  252. # item.setTextColor(0, color) # PyQt4
  253. item.setForeground(0, QtGui.QBrush(color))
  254. if font is not None:
  255. item.setFont(0, font)
  256. return item
  257. def addChild(self, parent, title, column1=None):
  258. item = QtWidgets.QTreeWidgetItem(parent)
  259. item.setText(0, str(title[0]))
  260. if column1 is not None:
  261. item.setText(1, str(title[1]))
  262. def show_area_chull(self, area, location):
  263. if self.app.ui.general_defaults_form.general_app_group.units_radio.get_value().lower() == 'mm':
  264. area_chull = area / 100
  265. self.addChild(location, ['%s:' % _('Convex_Hull Area'), '%.4f %s' % (area_chull, 'cm2')], True)
  266. else:
  267. area_chull = area
  268. self.addChild(location, ['%s:' % _('Convex_Hull Area'), '%.4f %s' % (area_chull, 'in2')], True)
  269. # end of file