AppDatabase.py 108 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586
  1. from PyQt5 import QtGui, QtCore, QtWidgets
  2. from AppGUI.GUIElements import FCTable, FCEntry, FCButton, FCDoubleSpinner, FCComboBox, FCCheckBox, FCSpinner, \
  3. FCTree, RadioSet, FCFileSaveDialog
  4. from camlib import to_dict
  5. import sys
  6. import json
  7. from copy import deepcopy
  8. from datetime import datetime
  9. import gettext
  10. import AppTranslation as fcTranslate
  11. import builtins
  12. fcTranslate.apply_language('strings')
  13. if '_' not in builtins.__dict__:
  14. _ = gettext.gettext
  15. class ToolsDB(QtWidgets.QWidget):
  16. mark_tools_rows = QtCore.pyqtSignal()
  17. def __init__(self, app, callback_on_edited, callback_on_tool_request, parent=None):
  18. super(ToolsDB, self).__init__(parent)
  19. self.app = app
  20. self.decimals = 4
  21. self.callback_app = callback_on_edited
  22. self.on_tool_request = callback_on_tool_request
  23. self.offset_item_options = ["Path", "In", "Out", "Custom"]
  24. self.type_item_options = ["Iso", "Rough", "Finish"]
  25. self.tool_type_item_options = ["C1", "C2", "C3", "C4", "B", "V"]
  26. '''
  27. dict to hold all the tools in the Tools DB
  28. format:
  29. {
  30. tool_id: {
  31. 'name': 'new_tool'
  32. 'tooldia': self.app.defaults["geometry_cnctooldia"]
  33. 'offset': 'Path'
  34. 'offset_value': 0.0
  35. 'type': _('Rough'),
  36. 'tool_type': 'C1'
  37. 'data': dict()
  38. }
  39. }
  40. '''
  41. self.db_tool_dict = {}
  42. # layouts
  43. layout = QtWidgets.QVBoxLayout()
  44. self.setLayout(layout)
  45. table_hlay = QtWidgets.QHBoxLayout()
  46. layout.addLayout(table_hlay)
  47. self.table_widget = FCTable(drag_drop=True)
  48. self.table_widget.setSelectionBehavior(QtWidgets.QAbstractItemView.SelectRows)
  49. table_hlay.addWidget(self.table_widget)
  50. # set the number of columns and the headers tool tips
  51. self.configure_table()
  52. # pal = QtGui.QPalette()
  53. # pal.setColor(QtGui.QPalette.Background, Qt.white)
  54. # New Bookmark
  55. new_vlay = QtWidgets.QVBoxLayout()
  56. layout.addLayout(new_vlay)
  57. # new_tool_lbl = QtWidgets.QLabel('<b>%s</b>' % _("New Tool"))
  58. # new_vlay.addWidget(new_tool_lbl, alignment=QtCore.Qt.AlignBottom)
  59. self.buttons_frame = QtWidgets.QFrame()
  60. self.buttons_frame.setContentsMargins(0, 0, 0, 0)
  61. layout.addWidget(self.buttons_frame)
  62. self.buttons_box = QtWidgets.QHBoxLayout()
  63. self.buttons_box.setContentsMargins(0, 0, 0, 0)
  64. self.buttons_frame.setLayout(self.buttons_box)
  65. self.buttons_frame.show()
  66. add_entry_btn = FCButton(_("Add Geometry Tool in DB"))
  67. add_entry_btn.setToolTip(
  68. _("Add a new tool in the Tools Database.\n"
  69. "It will be used in the Geometry UI.\n"
  70. "You can edit it after it is added.")
  71. )
  72. self.buttons_box.addWidget(add_entry_btn)
  73. # add_fct_entry_btn = FCButton(_("Add Paint/NCC Tool in DB"))
  74. # add_fct_entry_btn.setToolTip(
  75. # _("Add a new tool in the Tools Database.\n"
  76. # "It will be used in the Paint/NCC Tools UI.\n"
  77. # "You can edit it after it is added.")
  78. # )
  79. # self.buttons_box.addWidget(add_fct_entry_btn)
  80. remove_entry_btn = FCButton(_("Delete Tool from DB"))
  81. remove_entry_btn.setToolTip(
  82. _("Remove a selection of tools in the Tools Database.")
  83. )
  84. self.buttons_box.addWidget(remove_entry_btn)
  85. export_db_btn = FCButton(_("Export DB"))
  86. export_db_btn.setToolTip(
  87. _("Save the Tools Database to a custom text file.")
  88. )
  89. self.buttons_box.addWidget(export_db_btn)
  90. import_db_btn = FCButton(_("Import DB"))
  91. import_db_btn.setToolTip(
  92. _("Load the Tools Database information's from a custom text file.")
  93. )
  94. self.buttons_box.addWidget(import_db_btn)
  95. self.add_tool_from_db = FCButton(_("Add Tool from Tools DB"))
  96. self.add_tool_from_db.setToolTip(
  97. _("Add a new tool in the Tools Table of the\n"
  98. "active Geometry object after selecting a tool\n"
  99. "in the Tools Database.")
  100. )
  101. self.add_tool_from_db.hide()
  102. self.cancel_tool_from_db = FCButton(_("Cancel"))
  103. self.cancel_tool_from_db.hide()
  104. hlay = QtWidgets.QHBoxLayout()
  105. layout.addLayout(hlay)
  106. hlay.addWidget(self.add_tool_from_db)
  107. hlay.addWidget(self.cancel_tool_from_db)
  108. hlay.addStretch()
  109. # ##############################################################################
  110. # ######################## SIGNALS #############################################
  111. # ##############################################################################
  112. add_entry_btn.clicked.connect(self.on_tool_add)
  113. remove_entry_btn.clicked.connect(self.on_tool_delete)
  114. export_db_btn.clicked.connect(self.on_export_tools_db_file)
  115. import_db_btn.clicked.connect(self.on_import_tools_db_file)
  116. # closebtn.clicked.connect(self.accept)
  117. self.add_tool_from_db.clicked.connect(self.on_tool_requested_from_app)
  118. self.cancel_tool_from_db.clicked.connect(self.on_cancel_tool)
  119. self.setup_db_ui()
  120. def configure_table(self):
  121. self.table_widget.setColumnCount(27)
  122. # self.table_widget.setColumnWidth(0, 20)
  123. self.table_widget.setHorizontalHeaderLabels(
  124. [
  125. '#',
  126. _("Tool Name"),
  127. _("Tool Dia"),
  128. _("Tool Offset"),
  129. _("Custom Offset"),
  130. _("Tool Type"),
  131. _("Tool Shape"),
  132. _("Cut Z"),
  133. _("MultiDepth"),
  134. _("DPP"),
  135. _("V-Dia"),
  136. _("V-Angle"),
  137. _("Travel Z"),
  138. _("FR"),
  139. _("FR Z"),
  140. _("FR Rapids"),
  141. _("Spindle Speed"),
  142. _("Dwell"),
  143. _("Dwelltime"),
  144. _("Preprocessor"),
  145. _("ExtraCut"),
  146. _("E-Cut Length"),
  147. _("Toolchange"),
  148. _("Toolchange XY"),
  149. _("Toolchange Z"),
  150. _("Start Z"),
  151. _("End Z"),
  152. ]
  153. )
  154. self.table_widget.horizontalHeaderItem(0).setToolTip(
  155. _("Tool Index."))
  156. self.table_widget.horizontalHeaderItem(1).setToolTip(
  157. _("Tool name.\n"
  158. "This is not used in the app, it's function\n"
  159. "is to serve as a note for the user."))
  160. self.table_widget.horizontalHeaderItem(2).setToolTip(
  161. _("Tool Diameter."))
  162. self.table_widget.horizontalHeaderItem(3).setToolTip(
  163. _("Tool Offset.\n"
  164. "Can be of a few types:\n"
  165. "Path = zero offset\n"
  166. "In = offset inside by half of tool diameter\n"
  167. "Out = offset outside by half of tool diameter\n"
  168. "Custom = custom offset using the Custom Offset value"))
  169. self.table_widget.horizontalHeaderItem(4).setToolTip(
  170. _("Custom Offset.\n"
  171. "A value to be used as offset from the current path."))
  172. self.table_widget.horizontalHeaderItem(5).setToolTip(
  173. _("Tool Type.\n"
  174. "Can be:\n"
  175. "Iso = isolation cut\n"
  176. "Rough = rough cut, low feedrate, multiple passes\n"
  177. "Finish = finishing cut, high feedrate"))
  178. self.table_widget.horizontalHeaderItem(6).setToolTip(
  179. _("Tool Shape. \n"
  180. "Can be:\n"
  181. "C1 ... C4 = circular tool with x flutes\n"
  182. "B = ball tip milling tool\n"
  183. "V = v-shape milling tool"))
  184. self.table_widget.horizontalHeaderItem(7).setToolTip(
  185. _("Cutting Depth.\n"
  186. "The depth at which to cut into material."))
  187. self.table_widget.horizontalHeaderItem(8).setToolTip(
  188. _("Multi Depth.\n"
  189. "Selecting this will allow cutting in multiple passes,\n"
  190. "each pass adding a DPP parameter depth."))
  191. self.table_widget.horizontalHeaderItem(9).setToolTip(
  192. _("DPP. Depth per Pass.\n"
  193. "The value used to cut into material on each pass."))
  194. self.table_widget.horizontalHeaderItem(10).setToolTip(
  195. _("V-Dia.\n"
  196. "Diameter of the tip for V-Shape Tools."))
  197. self.table_widget.horizontalHeaderItem(11).setToolTip(
  198. _("V-Agle.\n"
  199. "Angle at the tip for the V-Shape Tools."))
  200. self.table_widget.horizontalHeaderItem(12).setToolTip(
  201. _("Clearance Height.\n"
  202. "Height at which the milling bit will travel between cuts,\n"
  203. "above the surface of the material, avoiding all fixtures."))
  204. self.table_widget.horizontalHeaderItem(13).setToolTip(
  205. _("FR. Feedrate\n"
  206. "The speed on XY plane used while cutting into material."))
  207. self.table_widget.horizontalHeaderItem(14).setToolTip(
  208. _("FR Z. Feedrate Z\n"
  209. "The speed on Z plane."))
  210. self.table_widget.horizontalHeaderItem(15).setToolTip(
  211. _("FR Rapids. Feedrate Rapids\n"
  212. "Speed used while moving as fast as possible.\n"
  213. "This is used only by some devices that can't use\n"
  214. "the G0 g-code command. Mostly 3D printers."))
  215. self.table_widget.horizontalHeaderItem(16).setToolTip(
  216. _("Spindle Speed.\n"
  217. "If it's left empty it will not be used.\n"
  218. "The speed of the spindle in RPM."))
  219. self.table_widget.horizontalHeaderItem(17).setToolTip(
  220. _("Dwell.\n"
  221. "Check this if a delay is needed to allow\n"
  222. "the spindle motor to reach it's set speed."))
  223. self.table_widget.horizontalHeaderItem(18).setToolTip(
  224. _("Dwell Time.\n"
  225. "A delay used to allow the motor spindle reach it's set speed."))
  226. self.table_widget.horizontalHeaderItem(19).setToolTip(
  227. _("Preprocessor.\n"
  228. "A selection of files that will alter the generated G-code\n"
  229. "to fit for a number of use cases."))
  230. self.table_widget.horizontalHeaderItem(20).setToolTip(
  231. _("Extra Cut.\n"
  232. "If checked, after a isolation is finished an extra cut\n"
  233. "will be added where the start and end of isolation meet\n"
  234. "such as that this point is covered by this extra cut to\n"
  235. "ensure a complete isolation."))
  236. self.table_widget.horizontalHeaderItem(21).setToolTip(
  237. _("Extra Cut length.\n"
  238. "If checked, after a isolation is finished an extra cut\n"
  239. "will be added where the start and end of isolation meet\n"
  240. "such as that this point is covered by this extra cut to\n"
  241. "ensure a complete isolation. This is the length of\n"
  242. "the extra cut."))
  243. self.table_widget.horizontalHeaderItem(22).setToolTip(
  244. _("Toolchange.\n"
  245. "It will create a toolchange event.\n"
  246. "The kind of toolchange is determined by\n"
  247. "the preprocessor file."))
  248. self.table_widget.horizontalHeaderItem(23).setToolTip(
  249. _("Toolchange XY.\n"
  250. "A set of coordinates in the format (x, y).\n"
  251. "Will determine the cartesian position of the point\n"
  252. "where the tool change event take place."))
  253. self.table_widget.horizontalHeaderItem(24).setToolTip(
  254. _("Toolchange Z.\n"
  255. "The position on Z plane where the tool change event take place."))
  256. self.table_widget.horizontalHeaderItem(25).setToolTip(
  257. _("Start Z.\n"
  258. "If it's left empty it will not be used.\n"
  259. "A position on Z plane to move immediately after job start."))
  260. self.table_widget.horizontalHeaderItem(26).setToolTip(
  261. _("End Z.\n"
  262. "A position on Z plane to move immediately after job stop."))
  263. def setup_db_ui(self):
  264. filename = self.app.data_path + '/geo_tools_db.FlatDB'
  265. # load the database tools from the file
  266. try:
  267. with open(filename) as f:
  268. tools = f.read()
  269. except IOError:
  270. self.app.log.error("Could not load tools DB file.")
  271. self.app.inform.emit('[ERROR] %s' % _("Could not load Tools DB file."))
  272. return
  273. try:
  274. self.db_tool_dict = json.loads(tools)
  275. except Exception:
  276. e = sys.exc_info()[0]
  277. self.app.log.error(str(e))
  278. self.app.inform.emit('[ERROR] %s' % _("Failed to parse Tools DB file."))
  279. return
  280. self.app.inform.emit('[success] %s: %s' % (_("Loaded FlatCAM Tools DB from"), filename))
  281. self.build_db_ui()
  282. self.table_widget.setupContextMenu()
  283. self.table_widget.addContextMenu(
  284. _("Add to DB"), self.on_tool_add, icon=QtGui.QIcon(self.app.resource_location + "/plus16.png"))
  285. self.table_widget.addContextMenu(
  286. _("Copy from DB"), self.on_tool_copy, icon=QtGui.QIcon(self.app.resource_location + "/copy16.png"))
  287. self.table_widget.addContextMenu(
  288. _("Delete from DB"), self.on_tool_delete, icon=QtGui.QIcon(self.app.resource_location + "/delete32.png"))
  289. def build_db_ui(self):
  290. self.ui_disconnect()
  291. self.table_widget.setRowCount(len(self.db_tool_dict))
  292. nr_crt = 0
  293. for toolid, dict_val in self.db_tool_dict.items():
  294. row = nr_crt
  295. nr_crt += 1
  296. t_name = dict_val['name']
  297. try:
  298. self.add_tool_table_line(row, name=t_name, widget=self.table_widget, tooldict=dict_val)
  299. except Exception as e:
  300. self.app.log.debug("ToolDB.build_db_ui.add_tool_table_line() --> %s" % str(e))
  301. vertical_header = self.table_widget.verticalHeader()
  302. vertical_header.hide()
  303. horizontal_header = self.table_widget.horizontalHeader()
  304. horizontal_header.setMinimumSectionSize(10)
  305. horizontal_header.setDefaultSectionSize(70)
  306. self.table_widget.setSizeAdjustPolicy(QtWidgets.QAbstractScrollArea.AdjustToContents)
  307. for x in range(27):
  308. self.table_widget.resizeColumnToContents(x)
  309. horizontal_header.setSectionResizeMode(0, QtWidgets.QHeaderView.Fixed)
  310. # horizontal_header.setSectionResizeMode(1, QtWidgets.QHeaderView.Stretch)
  311. # horizontal_header.setSectionResizeMode(13, QtWidgets.QHeaderView.Fixed)
  312. horizontal_header.resizeSection(0, 20)
  313. # horizontal_header.setSectionResizeMode(1, QtWidgets.QHeaderView.ResizeToContents)
  314. # horizontal_header.setSectionResizeMode(2, QtWidgets.QHeaderView.Stretch)
  315. self.ui_connect()
  316. def add_tool_table_line(self, row, name, widget, tooldict):
  317. data = tooldict['data']
  318. nr_crt = row + 1
  319. id_item = QtWidgets.QTableWidgetItem('%d' % int(nr_crt))
  320. # id_item.setFlags(QtCore.Qt.ItemIsSelectable | QtCore.Qt.ItemIsEnabled)
  321. flags = id_item.flags() & ~QtCore.Qt.ItemIsEditable
  322. id_item.setFlags(flags)
  323. widget.setItem(row, 0, id_item) # Tool name/id
  324. tool_name_item = QtWidgets.QTableWidgetItem(name)
  325. widget.setItem(row, 1, tool_name_item)
  326. dia_item = FCDoubleSpinner()
  327. dia_item.set_precision(self.decimals)
  328. dia_item.setSingleStep(0.1)
  329. dia_item.set_range(0.0, 9999.9999)
  330. dia_item.set_value(float(tooldict['tooldia']))
  331. widget.setCellWidget(row, 2, dia_item)
  332. tool_offset_item = FCComboBox()
  333. for item in self.offset_item_options:
  334. tool_offset_item.addItem(item)
  335. tool_offset_item.set_value(tooldict['offset'])
  336. widget.setCellWidget(row, 3, tool_offset_item)
  337. c_offset_item = FCDoubleSpinner()
  338. c_offset_item.set_precision(self.decimals)
  339. c_offset_item.setSingleStep(0.1)
  340. c_offset_item.set_range(-9999.9999, 9999.9999)
  341. c_offset_item.set_value(float(tooldict['offset_value']))
  342. widget.setCellWidget(row, 4, c_offset_item)
  343. tt_item = FCComboBox()
  344. for item in self.type_item_options:
  345. tt_item.addItem(item)
  346. tt_item.set_value(tooldict['type'])
  347. widget.setCellWidget(row, 5, tt_item)
  348. tshape_item = FCComboBox()
  349. for item in self.tool_type_item_options:
  350. tshape_item.addItem(item)
  351. tshape_item.set_value(tooldict['tool_type'])
  352. widget.setCellWidget(row, 6, tshape_item)
  353. cutz_item = FCDoubleSpinner()
  354. cutz_item.set_precision(self.decimals)
  355. cutz_item.setSingleStep(0.1)
  356. if self.app.defaults['global_machinist_setting']:
  357. cutz_item.set_range(-9999.9999, 9999.9999)
  358. else:
  359. cutz_item.set_range(-9999.9999, -0.0000)
  360. cutz_item.set_value(float(data['cutz']))
  361. widget.setCellWidget(row, 7, cutz_item)
  362. multidepth_item = FCCheckBox()
  363. multidepth_item.set_value(data['multidepth'])
  364. widget.setCellWidget(row, 8, multidepth_item)
  365. # to make the checkbox centered but it can no longer have it's value accessed - needs a fix using findchild()
  366. # multidepth_item = QtWidgets.QWidget()
  367. # cb = FCCheckBox()
  368. # cb.set_value(data['multidepth'])
  369. # qhboxlayout = QtWidgets.QHBoxLayout(multidepth_item)
  370. # qhboxlayout.addWidget(cb)
  371. # qhboxlayout.setAlignment(QtCore.Qt.AlignCenter)
  372. # qhboxlayout.setContentsMargins(0, 0, 0, 0)
  373. # widget.setCellWidget(row, 8, multidepth_item)
  374. depth_per_pass_item = FCDoubleSpinner()
  375. depth_per_pass_item.set_precision(self.decimals)
  376. depth_per_pass_item.setSingleStep(0.1)
  377. depth_per_pass_item.set_range(0.0, 9999.9999)
  378. depth_per_pass_item.set_value(float(data['depthperpass']))
  379. widget.setCellWidget(row, 9, depth_per_pass_item)
  380. vtip_dia_item = FCDoubleSpinner()
  381. vtip_dia_item.set_precision(self.decimals)
  382. vtip_dia_item.setSingleStep(0.1)
  383. vtip_dia_item.set_range(0.0, 9999.9999)
  384. vtip_dia_item.set_value(float(data['vtipdia']))
  385. widget.setCellWidget(row, 10, vtip_dia_item)
  386. vtip_angle_item = FCDoubleSpinner()
  387. vtip_angle_item.set_precision(self.decimals)
  388. vtip_angle_item.setSingleStep(0.1)
  389. vtip_angle_item.set_range(-360.0, 360.0)
  390. vtip_angle_item.set_value(float(data['vtipangle']))
  391. widget.setCellWidget(row, 11, vtip_angle_item)
  392. travelz_item = FCDoubleSpinner()
  393. travelz_item.set_precision(self.decimals)
  394. travelz_item.setSingleStep(0.1)
  395. if self.app.defaults['global_machinist_setting']:
  396. travelz_item.set_range(-9999.9999, 9999.9999)
  397. else:
  398. travelz_item.set_range(0.0000, 9999.9999)
  399. travelz_item.set_value(float(data['travelz']))
  400. widget.setCellWidget(row, 12, travelz_item)
  401. fr_item = FCDoubleSpinner()
  402. fr_item.set_precision(self.decimals)
  403. fr_item.set_range(0.0, 9999.9999)
  404. fr_item.set_value(float(data['feedrate']))
  405. widget.setCellWidget(row, 13, fr_item)
  406. frz_item = FCDoubleSpinner()
  407. frz_item.set_precision(self.decimals)
  408. frz_item.set_range(0.0, 9999.9999)
  409. frz_item.set_value(float(data['feedrate_z']))
  410. widget.setCellWidget(row, 14, frz_item)
  411. frrapids_item = FCDoubleSpinner()
  412. frrapids_item.set_precision(self.decimals)
  413. frrapids_item.set_range(0.0, 9999.9999)
  414. frrapids_item.set_value(float(data['feedrate_rapid']))
  415. widget.setCellWidget(row, 15, frrapids_item)
  416. spindlespeed_item = FCSpinner()
  417. spindlespeed_item.set_range(0, 1000000)
  418. spindlespeed_item.set_value(int(data['spindlespeed']))
  419. spindlespeed_item.set_step(100)
  420. widget.setCellWidget(row, 16, spindlespeed_item)
  421. dwell_item = FCCheckBox()
  422. dwell_item.set_value(data['dwell'])
  423. widget.setCellWidget(row, 17, dwell_item)
  424. dwelltime_item = FCDoubleSpinner()
  425. dwelltime_item.set_precision(self.decimals)
  426. dwelltime_item.set_range(0.0000, 9999.9999)
  427. dwelltime_item.set_value(float(data['dwelltime']))
  428. widget.setCellWidget(row, 18, dwelltime_item)
  429. pp_item = FCComboBox()
  430. for item in self.app.preprocessors:
  431. pp_item.addItem(item)
  432. pp_item.set_value(data['ppname_g'])
  433. widget.setCellWidget(row, 19, pp_item)
  434. ecut_item = FCCheckBox()
  435. ecut_item.set_value(data['extracut'])
  436. widget.setCellWidget(row, 20, ecut_item)
  437. ecut_length_item = FCDoubleSpinner()
  438. ecut_length_item.set_precision(self.decimals)
  439. ecut_length_item.set_range(0.0000, 9999.9999)
  440. ecut_length_item.set_value(data['extracut_length'])
  441. widget.setCellWidget(row, 21, ecut_length_item)
  442. toolchange_item = FCCheckBox()
  443. toolchange_item.set_value(data['toolchange'])
  444. widget.setCellWidget(row, 22, toolchange_item)
  445. toolchangexy_item = QtWidgets.QTableWidgetItem(str(data['toolchangexy']) if data['toolchangexy'] else '')
  446. widget.setItem(row, 23, toolchangexy_item)
  447. toolchangez_item = FCDoubleSpinner()
  448. toolchangez_item.set_precision(self.decimals)
  449. toolchangez_item.setSingleStep(0.1)
  450. if self.app.defaults['global_machinist_setting']:
  451. toolchangez_item.set_range(-9999.9999, 9999.9999)
  452. else:
  453. toolchangez_item.set_range(0.0000, 9999.9999)
  454. toolchangez_item.set_value(float(data['toolchangez']))
  455. widget.setCellWidget(row, 24, toolchangez_item)
  456. startz_item = QtWidgets.QTableWidgetItem(str(data['startz']) if data['startz'] else '')
  457. widget.setItem(row, 25, startz_item)
  458. endz_item = FCDoubleSpinner()
  459. endz_item.set_precision(self.decimals)
  460. endz_item.setSingleStep(0.1)
  461. if self.app.defaults['global_machinist_setting']:
  462. endz_item.set_range(-9999.9999, 9999.9999)
  463. else:
  464. endz_item.set_range(0.0000, 9999.9999)
  465. endz_item.set_value(float(data['endz']))
  466. widget.setCellWidget(row, 26, endz_item)
  467. def on_tool_add(self):
  468. """
  469. Add a tool in the DB Tool Table
  470. :return: None
  471. """
  472. default_data = {}
  473. default_data.update({
  474. "cutz": float(self.app.defaults["geometry_cutz"]),
  475. "multidepth": self.app.defaults["geometry_multidepth"],
  476. "depthperpass": float(self.app.defaults["geometry_depthperpass"]),
  477. "vtipdia": float(self.app.defaults["geometry_vtipdia"]),
  478. "vtipangle": float(self.app.defaults["geometry_vtipangle"]),
  479. "travelz": float(self.app.defaults["geometry_travelz"]),
  480. "feedrate": float(self.app.defaults["geometry_feedrate"]),
  481. "feedrate_z": float(self.app.defaults["geometry_feedrate_z"]),
  482. "feedrate_rapid": float(self.app.defaults["geometry_feedrate_rapid"]),
  483. "spindlespeed": self.app.defaults["geometry_spindlespeed"],
  484. "dwell": self.app.defaults["geometry_dwell"],
  485. "dwelltime": float(self.app.defaults["geometry_dwelltime"]),
  486. "ppname_g": self.app.defaults["geometry_ppname_g"],
  487. "extracut": self.app.defaults["geometry_extracut"],
  488. "extracut_length": float(self.app.defaults["geometry_extracut_length"]),
  489. "toolchange": self.app.defaults["geometry_toolchange"],
  490. "toolchangexy": self.app.defaults["geometry_toolchangexy"],
  491. "toolchangez": float(self.app.defaults["geometry_toolchangez"]),
  492. "startz": self.app.defaults["geometry_startz"],
  493. "endz": float(self.app.defaults["geometry_endz"])
  494. })
  495. dict_elem = {}
  496. dict_elem['name'] = 'new_tool'
  497. if type(self.app.defaults["geometry_cnctooldia"]) == float:
  498. dict_elem['tooldia'] = self.app.defaults["geometry_cnctooldia"]
  499. else:
  500. try:
  501. tools_string = self.app.defaults["geometry_cnctooldia"].split(",")
  502. tools_diameters = [eval(a) for a in tools_string if a != '']
  503. dict_elem['tooldia'] = tools_diameters[0] if tools_diameters else 0.0
  504. except Exception as e:
  505. self.app.log.debug("ToolDB.on_tool_add() --> %s" % str(e))
  506. return
  507. dict_elem['offset'] = 'Path'
  508. dict_elem['offset_value'] = 0.0
  509. dict_elem['type'] = 'Rough'
  510. dict_elem['tool_type'] = 'C1'
  511. dict_elem['data'] = default_data
  512. new_toolid = len(self.db_tool_dict) + 1
  513. self.db_tool_dict[new_toolid] = deepcopy(dict_elem)
  514. # add the new entry to the Tools DB table
  515. self.build_db_ui()
  516. self.callback_on_edited()
  517. self.app.inform.emit('[success] %s' % _("Tool added to DB."))
  518. def on_tool_copy(self):
  519. """
  520. Copy a selection of Tools in the Tools DB table
  521. :return:
  522. """
  523. new_tool_id = self.table_widget.rowCount() + 1
  524. for model_index in self.table_widget.selectionModel().selectedRows():
  525. # index = QtCore.QPersistentModelIndex(model_index)
  526. old_tool_id = self.table_widget.item(model_index.row(), 0).text()
  527. new_tool_id += 1
  528. for toolid, dict_val in list(self.db_tool_dict.items()):
  529. if int(old_tool_id) == int(toolid):
  530. self.db_tool_dict.update({
  531. new_tool_id: deepcopy(dict_val)
  532. })
  533. self.build_db_ui()
  534. self.callback_on_edited()
  535. self.app.inform.emit('[success] %s' % _("Tool copied from Tools DB."))
  536. def on_tool_delete(self):
  537. """
  538. Delete a selection of Tools in the Tools DB table
  539. :return:
  540. """
  541. for model_index in self.table_widget.selectionModel().selectedRows():
  542. # index = QtCore.QPersistentModelIndex(model_index)
  543. toolname_to_remove = self.table_widget.item(model_index.row(), 0).text()
  544. for toolid, dict_val in list(self.db_tool_dict.items()):
  545. if int(toolname_to_remove) == int(toolid):
  546. # remove from the storage
  547. self.db_tool_dict.pop(toolid, None)
  548. self.build_db_ui()
  549. self.callback_on_edited()
  550. self.app.inform.emit('[success] %s' % _("Tool removed from Tools DB."))
  551. def on_export_tools_db_file(self):
  552. self.app.defaults.report_usage("on_export_tools_db_file")
  553. self.app.log.debug("on_export_tools_db_file()")
  554. date = str(datetime.today()).rpartition('.')[0]
  555. date = ''.join(c for c in date if c not in ':-')
  556. date = date.replace(' ', '_')
  557. filter__ = "Text File (*.TXT);;All Files (*.*)"
  558. filename, _f = FCFileSaveDialog.get_saved_filename(caption=_("Export Tools Database"),
  559. directory='{l_save}/FlatCAM_{n}_{date}'.format(
  560. l_save=str(self.app.get_last_save_folder()),
  561. n=_("Tools_Database"),
  562. date=date),
  563. ext_filter=filter__)
  564. filename = str(filename)
  565. if filename == "":
  566. self.app.inform.emit('[WARNING_NOTCL] %s' % _("Cancelled."))
  567. return
  568. else:
  569. try:
  570. f = open(filename, 'w')
  571. f.close()
  572. except PermissionError:
  573. self.app.inform.emit('[WARNING] %s' %
  574. _("Permission denied, saving not possible.\n"
  575. "Most likely another app is holding the file open and not accessible."))
  576. return
  577. except IOError:
  578. self.app.log.debug('Creating a new Tools DB file ...')
  579. f = open(filename, 'w')
  580. f.close()
  581. except Exception:
  582. e = sys.exc_info()[0]
  583. self.app.log.error("Could not load Tools DB file.")
  584. self.app.log.error(str(e))
  585. self.app.inform.emit('[ERROR_NOTCL] %s' % _("Could not load Tools DB file."))
  586. return
  587. # Save update options
  588. try:
  589. # Save Tools DB in a file
  590. try:
  591. with open(filename, "w") as f:
  592. json.dump(self.db_tool_dict, f, default=to_dict, indent=2)
  593. except Exception as e:
  594. self.app.log.debug("App.on_save_tools_db() --> %s" % str(e))
  595. self.app.inform.emit('[ERROR_NOTCL] %s' % _("Failed to write Tools DB to file."))
  596. return
  597. except Exception:
  598. self.app.inform.emit('[ERROR_NOTCL] %s' % _("Failed to write Tools DB to file."))
  599. return
  600. self.app.inform.emit('[success] %s: %s' % (_("Exported Tools DB to"), filename))
  601. def on_import_tools_db_file(self):
  602. self.app.defaults.report_usage("on_import_tools_db_file")
  603. self.app.log.debug("on_import_tools_db_file()")
  604. filter__ = "Text File (*.TXT);;All Files (*.*)"
  605. filename, _f = QtWidgets.QFileDialog.getOpenFileName(caption=_("Import FlatCAM Tools DB"), filter=filter__)
  606. if filename == "":
  607. self.app.inform.emit('[WARNING_NOTCL] %s' % _("Cancelled."))
  608. else:
  609. try:
  610. with open(filename) as f:
  611. tools_in_db = f.read()
  612. except IOError:
  613. self.app.log.error("Could not load Tools DB file.")
  614. self.app.inform.emit('[ERROR_NOTCL] %s' % _("Could not load Tools DB file."))
  615. return
  616. try:
  617. self.db_tool_dict = json.loads(tools_in_db)
  618. except Exception:
  619. e = sys.exc_info()[0]
  620. self.app.log.error(str(e))
  621. self.app.inform.emit('[ERROR] %s' % _("Failed to parse Tools DB file."))
  622. return
  623. self.app.inform.emit('[success] %s: %s' % (_("Loaded FlatCAM Tools DB from"), filename))
  624. self.build_db_ui()
  625. self.callback_on_edited()
  626. def on_save_tools_db(self, silent=False):
  627. self.app.log.debug("ToolsDB.on_save_button() --> Saving Tools Database to file.")
  628. filename = self.app.data_path + "/geo_tools_db.FlatDB"
  629. # Preferences save, update the color of the Tools DB Tab text
  630. for idx in range(self.app.ui.plot_tab_area.count()):
  631. if self.app.ui.plot_tab_area.tabText(idx) == _("Tools Database"):
  632. self.app.ui.plot_tab_area.tabBar.setTabTextColor(idx, QtGui.QColor('black'))
  633. # Save Tools DB in a file
  634. try:
  635. f = open(filename, "w")
  636. json.dump(self.db_tool_dict, f, default=to_dict, indent=2)
  637. f.close()
  638. except Exception as e:
  639. self.app.log.debug("ToolsDB.on_save_tools_db() --> %s" % str(e))
  640. self.app.inform.emit('[ERROR_NOTCL] %s' % _("Failed to write Tools DB to file."))
  641. return
  642. if not silent:
  643. self.app.inform.emit('[success] %s' % _("Saved Tools DB."))
  644. def ui_connect(self):
  645. try:
  646. try:
  647. self.table_widget.itemChanged.disconnect(self.callback_on_edited)
  648. except (TypeError, AttributeError):
  649. pass
  650. self.table_widget.itemChanged.connect(self.callback_on_edited)
  651. except AttributeError:
  652. pass
  653. for row in range(self.table_widget.rowCount()):
  654. for col in range(self.table_widget.columnCount()):
  655. # ComboBox
  656. try:
  657. try:
  658. self.table_widget.cellWidget(row, col).currentIndexChanged.disconnect(self.callback_on_edited)
  659. except (TypeError, AttributeError):
  660. pass
  661. self.table_widget.cellWidget(row, col).currentIndexChanged.connect(self.callback_on_edited)
  662. except AttributeError:
  663. pass
  664. # CheckBox
  665. try:
  666. try:
  667. self.table_widget.cellWidget(row, col).toggled.disconnect(self.callback_on_edited)
  668. except (TypeError, AttributeError):
  669. pass
  670. self.table_widget.cellWidget(row, col).toggled.connect(self.callback_on_edited)
  671. except AttributeError:
  672. pass
  673. # SpinBox, DoubleSpinBox
  674. try:
  675. try:
  676. self.table_widget.cellWidget(row, col).valueChanged.disconnect(self.callback_on_edited)
  677. except (TypeError, AttributeError):
  678. pass
  679. self.table_widget.cellWidget(row, col).valueChanged.connect(self.callback_on_edited)
  680. except AttributeError:
  681. pass
  682. def ui_disconnect(self):
  683. try:
  684. self.table_widget.itemChanged.disconnect(self.callback_on_edited)
  685. except (TypeError, AttributeError):
  686. pass
  687. for row in range(self.table_widget.rowCount()):
  688. for col in range(self.table_widget.columnCount()):
  689. # ComboBox
  690. try:
  691. self.table_widget.cellWidget(row, col).currentIndexChanged.disconnect(self.callback_on_edited)
  692. except (TypeError, AttributeError):
  693. pass
  694. # CheckBox
  695. try:
  696. self.table_widget.cellWidget(row, col).toggled.disconnect(self.callback_on_edited)
  697. except (TypeError, AttributeError):
  698. pass
  699. # SpinBox, DoubleSpinBox
  700. try:
  701. self.table_widget.cellWidget(row, col).valueChanged.disconnect(self.callback_on_edited)
  702. except (TypeError, AttributeError):
  703. pass
  704. def callback_on_edited(self):
  705. # update the dictionary storage self.db_tool_dict
  706. self.db_tool_dict.clear()
  707. dict_elem = {}
  708. default_data = {}
  709. for row in range(self.table_widget.rowCount()):
  710. new_toolid = row + 1
  711. for col in range(self.table_widget.columnCount()):
  712. column_header_text = self.table_widget.horizontalHeaderItem(col).text()
  713. if column_header_text == _('Tool Name'):
  714. dict_elem['name'] = self.table_widget.item(row, col).text()
  715. elif column_header_text == _('Tool Dia'):
  716. dict_elem['tooldia'] = self.table_widget.cellWidget(row, col).get_value()
  717. elif column_header_text == _('Tool Offset'):
  718. dict_elem['offset'] = self.table_widget.cellWidget(row, col).get_value()
  719. elif column_header_text == _('Custom Offset'):
  720. dict_elem['offset_value'] = self.table_widget.cellWidget(row, col).get_value()
  721. elif column_header_text == _('Tool Type'):
  722. dict_elem['type'] = self.table_widget.cellWidget(row, col).get_value()
  723. elif column_header_text == _('Tool Shape'):
  724. dict_elem['tool_type'] = self.table_widget.cellWidget(row, col).get_value()
  725. else:
  726. if column_header_text == _('Cut Z'):
  727. default_data['cutz'] = self.table_widget.cellWidget(row, col).get_value()
  728. elif column_header_text == _('MultiDepth'):
  729. default_data['multidepth'] = self.table_widget.cellWidget(row, col).get_value()
  730. elif column_header_text == _('DPP'):
  731. default_data['depthperpass'] = self.table_widget.cellWidget(row, col).get_value()
  732. elif column_header_text == _('V-Dia'):
  733. default_data['vtipdia'] = self.table_widget.cellWidget(row, col).get_value()
  734. elif column_header_text == _('V-Angle'):
  735. default_data['vtipangle'] = self.table_widget.cellWidget(row, col).get_value()
  736. elif column_header_text == _('Travel Z'):
  737. default_data['travelz'] = self.table_widget.cellWidget(row, col).get_value()
  738. elif column_header_text == _('FR'):
  739. default_data['feedrate'] = self.table_widget.cellWidget(row, col).get_value()
  740. elif column_header_text == _('FR Z'):
  741. default_data['feedrate_z'] = self.table_widget.cellWidget(row, col).get_value()
  742. elif column_header_text == _('FR Rapids'):
  743. default_data['feedrate_rapid'] = self.table_widget.cellWidget(row, col).get_value()
  744. elif column_header_text == _('Spindle Speed'):
  745. default_data['spindlespeed'] = self.table_widget.cellWidget(row, col).get_value()
  746. elif column_header_text == _('Dwell'):
  747. default_data['dwell'] = self.table_widget.cellWidget(row, col).get_value()
  748. elif column_header_text == _('Dwelltime'):
  749. default_data['dwelltime'] = self.table_widget.cellWidget(row, col).get_value()
  750. elif column_header_text == _('Preprocessor'):
  751. default_data['ppname_g'] = self.table_widget.cellWidget(row, col).get_value()
  752. elif column_header_text == _('ExtraCut'):
  753. default_data['extracut'] = self.table_widget.cellWidget(row, col).get_value()
  754. elif column_header_text == _("E-Cut Length"):
  755. default_data['extracut_length'] = self.table_widget.cellWidget(row, col).get_value()
  756. elif column_header_text == _('Toolchange'):
  757. default_data['toolchange'] = self.table_widget.cellWidget(row, col).get_value()
  758. elif column_header_text == _('Toolchange XY'):
  759. default_data['toolchangexy'] = self.table_widget.item(row, col).text()
  760. elif column_header_text == _('Toolchange Z'):
  761. default_data['toolchangez'] = self.table_widget.cellWidget(row, col).get_value()
  762. elif column_header_text == _('Start Z'):
  763. default_data['startz'] = float(self.table_widget.item(row, col).text()) \
  764. if self.table_widget.item(row, col).text() != '' else None
  765. elif column_header_text == _('End Z'):
  766. default_data['endz'] = self.table_widget.cellWidget(row, col).get_value()
  767. dict_elem['data'] = default_data
  768. self.db_tool_dict.update(
  769. {
  770. new_toolid: deepcopy(dict_elem)
  771. }
  772. )
  773. self.callback_app()
  774. def on_tool_requested_from_app(self):
  775. if not self.table_widget.selectionModel().selectedRows():
  776. self.app.inform.emit('[WARNING_NOTCL] %s...' % _("No Tool/row selected in the Tools Database table"))
  777. return
  778. model_index_list = self.table_widget.selectionModel().selectedRows()
  779. for model_index in model_index_list:
  780. selected_row = model_index.row()
  781. tool_uid = selected_row + 1
  782. for key in self.db_tool_dict.keys():
  783. if str(key) == str(tool_uid):
  784. selected_tool = self.db_tool_dict[key]
  785. self.on_tool_request(tool=selected_tool)
  786. def on_cancel_tool(self):
  787. for idx in range(self.app.ui.plot_tab_area.count()):
  788. if self.app.ui.plot_tab_area.tabText(idx) == _("Tools Database"):
  789. wdg = self.app.ui.plot_tab_area.widget(idx)
  790. wdg.deleteLater()
  791. self.app.ui.plot_tab_area.removeTab(idx)
  792. self.app.inform.emit('%s' % _("Cancelled adding tool from DB."))
  793. # def resize_new_tool_table_widget(self, min_size, max_size):
  794. # """
  795. # Resize the table widget responsible for adding new tool in the Tool Database
  796. #
  797. # :param min_size: passed by rangeChanged signal or the self.new_tool_table_widget.horizontalScrollBar()
  798. # :param max_size: passed by rangeChanged signal or the self.new_tool_table_widget.horizontalScrollBar()
  799. # :return:
  800. # """
  801. # t_height = self.t_height
  802. # if max_size > min_size:
  803. # t_height = self.t_height + self.new_tool_table_widget.verticalScrollBar().height()
  804. #
  805. # self.new_tool_table_widget.setMaximumHeight(t_height)
  806. def closeEvent(self, QCloseEvent):
  807. super().closeEvent(QCloseEvent)
  808. class ToolsDB2(QtWidgets.QWidget):
  809. mark_tools_rows = QtCore.pyqtSignal()
  810. def __init__(self, app, callback_on_edited, callback_on_tool_request, parent=None):
  811. super(ToolsDB2, self).__init__(parent)
  812. self.app = app
  813. self.decimals = self.app.decimals
  814. self.callback_app = callback_on_edited
  815. self.on_tool_request = callback_on_tool_request
  816. self.offset_item_options = ["Path", "In", "Out", "Custom"]
  817. self.type_item_options = ["Iso", "Rough", "Finish"]
  818. self.tool_type_item_options = ["C1", "C2", "C3", "C4", "B", "V"]
  819. '''
  820. dict to hold all the tools in the Tools DB
  821. format:
  822. {
  823. tool_id: {
  824. 'name': 'new_tool'
  825. 'tooldia': self.app.defaults["geometry_cnctooldia"]
  826. 'offset': 'Path'
  827. 'offset_value': 0.0
  828. 'type': _('Rough'),
  829. 'tool_type': 'C1'
  830. 'data': dict()
  831. }
  832. }
  833. '''
  834. self.db_tool_dict = {}
  835. # layouts
  836. grid_layout = QtWidgets.QGridLayout()
  837. grid_layout.setColumnStretch(0, 0)
  838. grid_layout.setColumnStretch(1, 1)
  839. self.setLayout(grid_layout)
  840. tree_layout = QtWidgets.QVBoxLayout()
  841. grid_layout.addLayout(tree_layout, 0, 0)
  842. self.tree_widget = FCTree(columns=2, header_hidden=False, protected_column=[0])
  843. self.tree_widget.setHeaderLabels(["ID", "Tool Name"])
  844. self.tree_widget.setIndentation(0)
  845. self.tree_widget.setContextMenuPolicy(QtCore.Qt.CustomContextMenu)
  846. self.tree_widget.setSelectionMode(QtWidgets.QAbstractItemView.ExtendedSelection)
  847. # set alternating colors
  848. # self.tree_widget.setAlternatingRowColors(True)
  849. # p = QtGui.QPalette()
  850. # p.setColor(QtGui.QPalette.AlternateBase, QtGui.QColor(226, 237, 253) )
  851. # self.tree_widget.setPalette(p)
  852. self.tree_widget.setSizePolicy(QtWidgets.QSizePolicy.MinimumExpanding, QtWidgets.QSizePolicy.MinimumExpanding)
  853. tree_layout.addWidget(self.tree_widget)
  854. param_hlay = QtWidgets.QHBoxLayout()
  855. param_area = QtWidgets.QScrollArea()
  856. param_widget = QtWidgets.QWidget()
  857. param_widget.setLayout(param_hlay)
  858. param_area.setWidget(param_widget)
  859. param_area.setWidgetResizable(True)
  860. grid_layout.addWidget(param_area, 0, 1)
  861. # ###########################################################################
  862. # ############## The UI form ################################################
  863. # ###########################################################################
  864. self.basic_box = QtWidgets.QGroupBox()
  865. self.basic_box.setStyleSheet("""
  866. QGroupBox
  867. {
  868. font-size: 16px;
  869. font-weight: bold;
  870. }
  871. """)
  872. self.basic_vlay = QtWidgets.QVBoxLayout()
  873. self.basic_box.setTitle(_("Basic Geo Parameters"))
  874. self.basic_box.setFixedWidth(250)
  875. self.advanced_box = QtWidgets.QGroupBox()
  876. self.advanced_box.setStyleSheet("""
  877. QGroupBox
  878. {
  879. font-size: 16px;
  880. font-weight: bold;
  881. }
  882. """)
  883. self.advanced_vlay = QtWidgets.QVBoxLayout()
  884. self.advanced_box.setTitle(_("Advanced Geo Parameters"))
  885. self.advanced_box.setFixedWidth(250)
  886. # NCC TOOL BOX
  887. self.ncc_box = QtWidgets.QGroupBox()
  888. self.ncc_box.setStyleSheet("""
  889. QGroupBox
  890. {
  891. font-size: 16px;
  892. font-weight: bold;
  893. }
  894. """)
  895. self.ncc_vlay = QtWidgets.QVBoxLayout()
  896. self.ncc_box.setTitle(_("NCC Parameters"))
  897. self.ncc_box.setFixedWidth(250)
  898. # PAINT TOOL BOX
  899. self.paint_box = QtWidgets.QGroupBox()
  900. self.paint_box.setStyleSheet("""
  901. QGroupBox
  902. {
  903. font-size: 16px;
  904. font-weight: bold;
  905. }
  906. """)
  907. self.paint_vlay = QtWidgets.QVBoxLayout()
  908. self.paint_box.setTitle(_("Paint Parameters"))
  909. self.paint_box.setFixedWidth(250)
  910. # ISOLATION TOOL BOX
  911. self.iso_box = QtWidgets.QGroupBox()
  912. self.iso_box.setStyleSheet("""
  913. QGroupBox
  914. {
  915. font-size: 16px;
  916. font-weight: bold;
  917. }
  918. """)
  919. self.iso_vlay = QtWidgets.QVBoxLayout()
  920. self.iso_box.setTitle(_("Isolation Parameters"))
  921. self.iso_box.setFixedWidth(250)
  922. self.basic_box.setLayout(self.basic_vlay)
  923. self.advanced_box.setLayout(self.advanced_vlay)
  924. self.ncc_box.setLayout(self.ncc_vlay)
  925. self.paint_box.setLayout(self.paint_vlay)
  926. self.iso_box.setLayout(self.iso_vlay)
  927. geo_vlay = QtWidgets.QVBoxLayout()
  928. geo_vlay.addWidget(self.basic_box)
  929. geo_vlay.addWidget(self.advanced_box)
  930. geo_vlay.addStretch()
  931. tools_vlay = QtWidgets.QVBoxLayout()
  932. tools_vlay.addWidget(self.ncc_box)
  933. tools_vlay.addWidget(self.paint_box)
  934. tools_vlay.addWidget(self.iso_box)
  935. tools_vlay.addStretch()
  936. param_hlay.addLayout(geo_vlay)
  937. param_hlay.addLayout(tools_vlay)
  938. param_hlay.addStretch()
  939. # ###########################################################################
  940. # ############### BASIC UI form #############################################
  941. # ###########################################################################
  942. self.grid0 = QtWidgets.QGridLayout()
  943. self.basic_vlay.addLayout(self.grid0)
  944. self.grid0.setColumnStretch(0, 0)
  945. self.grid0.setColumnStretch(1, 1)
  946. self.basic_vlay.addStretch()
  947. # Tool Name
  948. self.name_label = QtWidgets.QLabel('<span style="color:red;"><b>%s:</b></span>' % _('Tool Name'))
  949. self.name_label.setToolTip(
  950. _("Tool name.\n"
  951. "This is not used in the app, it's function\n"
  952. "is to serve as a note for the user."))
  953. self.name_entry = FCEntry()
  954. self.name_entry.setObjectName('gdb_name')
  955. self.grid0.addWidget(self.name_label, 0, 0)
  956. self.grid0.addWidget(self.name_entry, 0, 1)
  957. # Tool Dia
  958. self.dia_label = QtWidgets.QLabel('%s:' % _('Tool Dia'))
  959. self.dia_label.setToolTip(
  960. _("Tool Diameter."))
  961. self.dia_entry = FCDoubleSpinner()
  962. self.dia_entry.set_range(-9999.9999, 9999.9999)
  963. self.dia_entry.set_precision(self.decimals)
  964. self.dia_entry.setObjectName('gdb_dia')
  965. self.grid0.addWidget(self.dia_label, 1, 0)
  966. self.grid0.addWidget(self.dia_entry, 1, 1)
  967. # Tool Shape
  968. self.shape_label = QtWidgets.QLabel('%s:' % _('Tool Shape'))
  969. self.shape_label.setToolTip(
  970. _("Tool Shape. \n"
  971. "Can be:\n"
  972. "C1 ... C4 = circular tool with x flutes\n"
  973. "B = ball tip milling tool\n"
  974. "V = v-shape milling tool"))
  975. self.shape_combo = FCComboBox()
  976. self.shape_combo.addItems(["C1", "C2", "C3", "C4", "B", "V"])
  977. self.shape_combo.setObjectName('gdb_shape')
  978. self.grid0.addWidget(self.shape_label, 2, 0)
  979. self.grid0.addWidget(self.shape_combo, 2, 1)
  980. # Cut Z
  981. self.cutz_label = QtWidgets.QLabel('%s:' % _("Cut Z"))
  982. self.cutz_label.setToolTip(
  983. _("Cutting Depth.\n"
  984. "The depth at which to cut into material."))
  985. self.cutz_entry = FCDoubleSpinner()
  986. self.cutz_entry.set_range(-9999.9999, 9999.9999)
  987. self.cutz_entry.set_precision(self.decimals)
  988. self.cutz_entry.setObjectName('gdb_cutz')
  989. self.grid0.addWidget(self.cutz_label, 4, 0)
  990. self.grid0.addWidget(self.cutz_entry, 4, 1)
  991. # Multi Depth
  992. self.multidepth_label = QtWidgets.QLabel('%s:' % _("MultiDepth"))
  993. self.multidepth_label.setToolTip(
  994. _("Multi Depth.\n"
  995. "Selecting this will allow cutting in multiple passes,\n"
  996. "each pass adding a DPP parameter depth."))
  997. self.multidepth_cb = FCCheckBox()
  998. self.multidepth_cb.setObjectName('gdb_multidepth')
  999. self.grid0.addWidget(self.multidepth_label, 5, 0)
  1000. self.grid0.addWidget(self.multidepth_cb, 5, 1)
  1001. # Depth Per Pass
  1002. self.dpp_label = QtWidgets.QLabel('%s:' % _("DPP"))
  1003. self.dpp_label.setToolTip(
  1004. _("DPP. Depth per Pass.\n"
  1005. "The value used to cut into material on each pass."))
  1006. self.multidepth_entry = FCDoubleSpinner()
  1007. self.multidepth_entry.set_range(-9999.9999, 9999.9999)
  1008. self.multidepth_entry.set_precision(self.decimals)
  1009. self.multidepth_entry.setObjectName('gdb_multidepth_entry')
  1010. self.grid0.addWidget(self.dpp_label, 7, 0)
  1011. self.grid0.addWidget(self.multidepth_entry, 7, 1)
  1012. # Travel Z
  1013. self.travelz_label = QtWidgets.QLabel('%s:' % _("Travel Z"))
  1014. self.travelz_label.setToolTip(
  1015. _("Clearance Height.\n"
  1016. "Height at which the milling bit will travel between cuts,\n"
  1017. "above the surface of the material, avoiding all fixtures."))
  1018. self.travelz_entry = FCDoubleSpinner()
  1019. self.travelz_entry.set_range(-9999.9999, 9999.9999)
  1020. self.travelz_entry.set_precision(self.decimals)
  1021. self.travelz_entry.setObjectName('gdb_travel')
  1022. self.grid0.addWidget(self.travelz_label, 9, 0)
  1023. self.grid0.addWidget(self.travelz_entry, 9, 1)
  1024. # Feedrate X-Y
  1025. self.frxy_label = QtWidgets.QLabel('%s:' % _("Feedrate X-Y"))
  1026. self.frxy_label.setToolTip(
  1027. _("Feedrate X-Y. Feedrate\n"
  1028. "The speed on XY plane used while cutting into material."))
  1029. self.frxy_entry = FCDoubleSpinner()
  1030. self.frxy_entry.set_range(-999999.9999, 999999.9999)
  1031. self.frxy_entry.set_precision(self.decimals)
  1032. self.frxy_entry.setObjectName('gdb_frxy')
  1033. self.grid0.addWidget(self.frxy_label, 12, 0)
  1034. self.grid0.addWidget(self.frxy_entry, 12, 1)
  1035. # Feedrate Z
  1036. self.frz_label = QtWidgets.QLabel('%s:' % _("Feedrate Z"))
  1037. self.frz_label.setToolTip(
  1038. _("Feedrate Z\n"
  1039. "The speed on Z plane."))
  1040. self.frz_entry = FCDoubleSpinner()
  1041. self.frz_entry.set_range(-999999.9999, 999999.9999)
  1042. self.frz_entry.set_precision(self.decimals)
  1043. self.frz_entry.setObjectName('gdb_frz')
  1044. self.grid0.addWidget(self.frz_label, 14, 0)
  1045. self.grid0.addWidget(self.frz_entry, 14, 1)
  1046. # Spindle Spped
  1047. self.spindle_label = QtWidgets.QLabel('%s:' % _("Spindle Speed"))
  1048. self.spindle_label.setToolTip(
  1049. _("Spindle Speed.\n"
  1050. "If it's left empty it will not be used.\n"
  1051. "The speed of the spindle in RPM."))
  1052. self.spindle_entry = FCDoubleSpinner()
  1053. self.spindle_entry.set_range(-999999.9999, 999999.9999)
  1054. self.spindle_entry.set_precision(self.decimals)
  1055. self.spindle_entry.setObjectName('gdb_spindle')
  1056. self.grid0.addWidget(self.spindle_label, 15, 0)
  1057. self.grid0.addWidget(self.spindle_entry, 15, 1)
  1058. # Dwell
  1059. self.dwell_label = QtWidgets.QLabel('%s:' % _("Dwell"))
  1060. self.dwell_label.setToolTip(
  1061. _("Dwell.\n"
  1062. "Check this if a delay is needed to allow\n"
  1063. "the spindle motor to reach it's set speed."))
  1064. self.dwell_cb = FCCheckBox()
  1065. self.dwell_cb.setObjectName('gdb_dwell')
  1066. self.grid0.addWidget(self.dwell_label, 16, 0)
  1067. self.grid0.addWidget(self.dwell_cb, 16, 1)
  1068. # Dwell Time
  1069. self.dwelltime_label = QtWidgets.QLabel('%s:' % _("Dwelltime"))
  1070. self.dwelltime_label.setToolTip(
  1071. _("Dwell Time.\n"
  1072. "A delay used to allow the motor spindle reach it's set speed."))
  1073. self.dwelltime_entry = FCDoubleSpinner()
  1074. self.dwelltime_entry.set_range(0.0000, 9999.9999)
  1075. self.dwelltime_entry.set_precision(self.decimals)
  1076. self.dwelltime_entry.setObjectName('gdb_dwelltime')
  1077. self.grid0.addWidget(self.dwelltime_label, 17, 0)
  1078. self.grid0.addWidget(self.dwelltime_entry, 17, 1)
  1079. # ###########################################################################
  1080. # ############### ADVANCED UI form ##########################################
  1081. # ###########################################################################
  1082. self.grid1 = QtWidgets.QGridLayout()
  1083. self.advanced_vlay.addLayout(self.grid1)
  1084. self.grid1.setColumnStretch(0, 0)
  1085. self.grid1.setColumnStretch(1, 1)
  1086. self.advanced_vlay.addStretch()
  1087. # Tool Type
  1088. self.type_label = QtWidgets.QLabel('%s:' % _("Tool Type"))
  1089. self.type_label.setToolTip(
  1090. _("Tool Type.\n"
  1091. "Can be:\n"
  1092. "Iso = isolation cut\n"
  1093. "Rough = rough cut, low feedrate, multiple passes\n"
  1094. "Finish = finishing cut, high feedrate"))
  1095. self.type_combo = FCComboBox()
  1096. self.type_combo.addItems(["Iso", "Rough", "Finish"])
  1097. self.type_combo.setObjectName('gdb_type')
  1098. self.grid1.addWidget(self.type_label, 0, 0)
  1099. self.grid1.addWidget(self.type_combo, 0, 1)
  1100. # Tool Offset
  1101. self.tooloffset_label = QtWidgets.QLabel('%s:' % _('Tool Offset'))
  1102. self.tooloffset_label.setToolTip(
  1103. _("Tool Offset.\n"
  1104. "Can be of a few types:\n"
  1105. "Path = zero offset\n"
  1106. "In = offset inside by half of tool diameter\n"
  1107. "Out = offset outside by half of tool diameter\n"
  1108. "Custom = custom offset using the Custom Offset value"))
  1109. self.tooloffset_combo = FCComboBox()
  1110. self.tooloffset_combo.addItems(["Path", "In", "Out", "Custom"])
  1111. self.tooloffset_combo.setObjectName('gdb_tool_offset')
  1112. self.grid1.addWidget(self.tooloffset_label, 2, 0)
  1113. self.grid1.addWidget(self.tooloffset_combo, 2, 1)
  1114. # Custom Offset
  1115. self.custom_offset_label = QtWidgets.QLabel('%s:' % _("Custom Offset"))
  1116. self.custom_offset_label.setToolTip(
  1117. _("Custom Offset.\n"
  1118. "A value to be used as offset from the current path."))
  1119. self.custom_offset_entry = FCDoubleSpinner()
  1120. self.custom_offset_entry.set_range(-9999.9999, 9999.9999)
  1121. self.custom_offset_entry.set_precision(self.decimals)
  1122. self.custom_offset_entry.setObjectName('gdb_custom_offset')
  1123. self.grid1.addWidget(self.custom_offset_label, 5, 0)
  1124. self.grid1.addWidget(self.custom_offset_entry, 5, 1)
  1125. # V-Dia
  1126. self.vdia_label = QtWidgets.QLabel('%s:' % _("V-Dia"))
  1127. self.vdia_label.setToolTip(
  1128. _("V-Dia.\n"
  1129. "Diameter of the tip for V-Shape Tools."))
  1130. self.vdia_entry = FCDoubleSpinner()
  1131. self.vdia_entry.set_range(0.0000, 9999.9999)
  1132. self.vdia_entry.set_precision(self.decimals)
  1133. self.vdia_entry.setObjectName('gdb_vdia')
  1134. self.grid1.addWidget(self.vdia_label, 7, 0)
  1135. self.grid1.addWidget(self.vdia_entry, 7, 1)
  1136. # V-Angle
  1137. self.vangle_label = QtWidgets.QLabel('%s:' % _("V-Angle"))
  1138. self.vangle_label.setToolTip(
  1139. _("V-Agle.\n"
  1140. "Angle at the tip for the V-Shape Tools."))
  1141. self.vangle_entry = FCDoubleSpinner()
  1142. self.vangle_entry.set_range(-360.0, 360.0)
  1143. self.vangle_entry.set_precision(self.decimals)
  1144. self.vangle_entry.setObjectName('gdb_vangle')
  1145. self.grid1.addWidget(self.vangle_label, 8, 0)
  1146. self.grid1.addWidget(self.vangle_entry, 8, 1)
  1147. # Feedrate Rapids
  1148. self.frapids_label = QtWidgets.QLabel('%s:' % _("FR Rapids"))
  1149. self.frapids_label.setToolTip(
  1150. _("FR Rapids. Feedrate Rapids\n"
  1151. "Speed used while moving as fast as possible.\n"
  1152. "This is used only by some devices that can't use\n"
  1153. "the G0 g-code command. Mostly 3D printers."))
  1154. self.frapids_entry = FCDoubleSpinner()
  1155. self.frapids_entry.set_range(0.0000, 9999.9999)
  1156. self.frapids_entry.set_precision(self.decimals)
  1157. self.frapids_entry.setObjectName('gdb_frapids')
  1158. self.grid1.addWidget(self.frapids_label, 10, 0)
  1159. self.grid1.addWidget(self.frapids_entry, 10, 1)
  1160. # Extra Cut
  1161. self.ecut_label = QtWidgets.QLabel('%s:' % _("ExtraCut"))
  1162. self.ecut_label.setToolTip(
  1163. _("Extra Cut.\n"
  1164. "If checked, after a isolation is finished an extra cut\n"
  1165. "will be added where the start and end of isolation meet\n"
  1166. "such as that this point is covered by this extra cut to\n"
  1167. "ensure a complete isolation."))
  1168. self.ecut_cb = FCCheckBox()
  1169. self.ecut_cb.setObjectName('gdb_ecut')
  1170. self.grid1.addWidget(self.ecut_label, 12, 0)
  1171. self.grid1.addWidget(self.ecut_cb, 12, 1)
  1172. # Extra Cut Length
  1173. self.ecut_length_label = QtWidgets.QLabel('%s:' % _("E-Cut Length"))
  1174. self.ecut_length_label.setToolTip(
  1175. _("Extra Cut length.\n"
  1176. "If checked, after a isolation is finished an extra cut\n"
  1177. "will be added where the start and end of isolation meet\n"
  1178. "such as that this point is covered by this extra cut to\n"
  1179. "ensure a complete isolation. This is the length of\n"
  1180. "the extra cut."))
  1181. self.ecut_length_entry = FCDoubleSpinner()
  1182. self.ecut_length_entry.set_range(0.0000, 9999.9999)
  1183. self.ecut_length_entry.set_precision(self.decimals)
  1184. self.ecut_length_entry.setObjectName('gdb_ecut_length')
  1185. self.grid1.addWidget(self.ecut_length_label, 13, 0)
  1186. self.grid1.addWidget(self.ecut_length_entry, 13, 1)
  1187. # ###########################################################################
  1188. # ############### NCC UI form ###############################################
  1189. # ###########################################################################
  1190. self.grid2 = QtWidgets.QGridLayout()
  1191. self.ncc_vlay.addLayout(self.grid2)
  1192. self.grid2.setColumnStretch(0, 0)
  1193. self.grid2.setColumnStretch(1, 1)
  1194. self.ncc_vlay.addStretch()
  1195. # Operation
  1196. op_label = QtWidgets.QLabel('%s:' % _('Operation'))
  1197. op_label.setToolTip(
  1198. _("The 'Operation' can be:\n"
  1199. "- Isolation -> will ensure that the non-copper clearing is always complete.\n"
  1200. "If it's not successful then the non-copper clearing will fail, too.\n"
  1201. "- Clear -> the regular non-copper clearing.")
  1202. )
  1203. self.op_radio = RadioSet([
  1204. {"label": _("Clear"), "value": "clear"},
  1205. {"label": _("Isolation"), "value": "iso"}
  1206. ], orientation='horizontal', stretch=False)
  1207. self.op_radio.setObjectName("gdb_n_operation")
  1208. self.grid2.addWidget(op_label, 13, 0)
  1209. self.grid2.addWidget(self.op_radio, 13, 1)
  1210. # Milling Type Radio Button
  1211. self.milling_type_label = QtWidgets.QLabel('%s:' % _('Milling Type'))
  1212. self.milling_type_label.setToolTip(
  1213. _("Milling type when the selected tool is of type: 'iso_op':\n"
  1214. "- climb / best for precision milling and to reduce tool usage\n"
  1215. "- conventional / useful when there is no backlash compensation")
  1216. )
  1217. self.milling_type_radio = RadioSet([{'label': _('Climb'), 'value': 'cl'},
  1218. {'label': _('Conventional'), 'value': 'cv'}])
  1219. self.milling_type_radio.setToolTip(
  1220. _("Milling type when the selected tool is of type: 'iso_op':\n"
  1221. "- climb / best for precision milling and to reduce tool usage\n"
  1222. "- conventional / useful when there is no backlash compensation")
  1223. )
  1224. self.milling_type_radio.setObjectName("gdb_n_milling_type")
  1225. self.grid2.addWidget(self.milling_type_label, 14, 0)
  1226. self.grid2.addWidget(self.milling_type_radio, 14, 1)
  1227. # Overlap Entry
  1228. nccoverlabel = QtWidgets.QLabel('%s:' % _('Overlap'))
  1229. nccoverlabel.setToolTip(
  1230. _("How much (percentage) of the tool width to overlap each tool pass.\n"
  1231. "Adjust the value starting with lower values\n"
  1232. "and increasing it if areas that should be cleared are still \n"
  1233. "not cleared.\n"
  1234. "Lower values = faster processing, faster execution on CNC.\n"
  1235. "Higher values = slow processing and slow execution on CNC\n"
  1236. "due of too many paths.")
  1237. )
  1238. self.ncc_overlap_entry = FCDoubleSpinner(suffix='%')
  1239. self.ncc_overlap_entry.set_precision(self.decimals)
  1240. self.ncc_overlap_entry.setWrapping(True)
  1241. self.ncc_overlap_entry.setRange(0.000, 99.9999)
  1242. self.ncc_overlap_entry.setSingleStep(0.1)
  1243. self.ncc_overlap_entry.setObjectName("gdb_n_overlap")
  1244. self.grid2.addWidget(nccoverlabel, 15, 0)
  1245. self.grid2.addWidget(self.ncc_overlap_entry, 15, 1)
  1246. # Margin
  1247. nccmarginlabel = QtWidgets.QLabel('%s:' % _('Margin'))
  1248. nccmarginlabel.setToolTip(
  1249. _("Bounding box margin.")
  1250. )
  1251. self.ncc_margin_entry = FCDoubleSpinner()
  1252. self.ncc_margin_entry.set_precision(self.decimals)
  1253. self.ncc_margin_entry.set_range(-9999.9999, 9999.9999)
  1254. self.ncc_margin_entry.setObjectName("gdb_n_margin")
  1255. self.grid2.addWidget(nccmarginlabel, 16, 0)
  1256. self.grid2.addWidget(self.ncc_margin_entry, 16, 1)
  1257. # Method
  1258. methodlabel = QtWidgets.QLabel('%s:' % _('Method'))
  1259. methodlabel.setToolTip(
  1260. _("Algorithm for copper clearing:\n"
  1261. "- Standard: Fixed step inwards.\n"
  1262. "- Seed-based: Outwards from seed.\n"
  1263. "- Line-based: Parallel lines.")
  1264. )
  1265. self.ncc_method_combo = FCComboBox()
  1266. self.ncc_method_combo.addItems(
  1267. [_("Standard"), _("Seed"), _("Lines"), _("Combo")]
  1268. )
  1269. self.ncc_method_combo.setObjectName("gdb_n_method")
  1270. self.grid2.addWidget(methodlabel, 17, 0)
  1271. self.grid2.addWidget(self.ncc_method_combo, 17, 1)
  1272. # Connect lines
  1273. self.ncc_connect_cb = FCCheckBox('%s' % _("Connect"))
  1274. self.ncc_connect_cb.setObjectName("gdb_n_connect")
  1275. self.ncc_connect_cb.setToolTip(
  1276. _("Draw lines between resulting\n"
  1277. "segments to minimize tool lifts.")
  1278. )
  1279. self.grid2.addWidget(self.ncc_connect_cb, 18, 0)
  1280. # Contour
  1281. self.ncc_contour_cb = FCCheckBox('%s' % _("Contour"))
  1282. self.ncc_contour_cb.setObjectName("gdb_n_contour")
  1283. self.ncc_contour_cb.setToolTip(
  1284. _("Cut around the perimeter of the polygon\n"
  1285. "to trim rough edges.")
  1286. )
  1287. self.grid2.addWidget(self.ncc_contour_cb, 18, 1)
  1288. # ## NCC Offset choice
  1289. self.ncc_choice_offset_cb = FCCheckBox('%s' % _("Offset"))
  1290. self.ncc_choice_offset_cb.setObjectName("gdb_n_offset")
  1291. self.ncc_choice_offset_cb.setToolTip(
  1292. _("If used, it will add an offset to the copper features.\n"
  1293. "The copper clearing will finish to a distance\n"
  1294. "from the copper features.\n"
  1295. "The value can be between 0 and 10 FlatCAM units.")
  1296. )
  1297. self.grid2.addWidget(self.ncc_choice_offset_cb, 19, 0)
  1298. # ## NCC Offset Entry
  1299. self.ncc_offset_spinner = FCDoubleSpinner()
  1300. self.ncc_offset_spinner.set_range(0.00, 10.00)
  1301. self.ncc_offset_spinner.set_precision(4)
  1302. self.ncc_offset_spinner.setWrapping(True)
  1303. self.ncc_offset_spinner.setObjectName("gdb_n_offset_value")
  1304. units = self.app.defaults['units'].upper()
  1305. if units == 'MM':
  1306. self.ncc_offset_spinner.setSingleStep(0.1)
  1307. else:
  1308. self.ncc_offset_spinner.setSingleStep(0.01)
  1309. self.grid2.addWidget(self.ncc_offset_spinner, 19, 1)
  1310. # ###########################################################################
  1311. # ############### Paint UI form #############################################
  1312. # ###########################################################################
  1313. self.grid3 = QtWidgets.QGridLayout()
  1314. self.paint_vlay.addLayout(self.grid3)
  1315. self.grid3.setColumnStretch(0, 0)
  1316. self.grid3.setColumnStretch(1, 1)
  1317. self.paint_vlay.addStretch()
  1318. # Overlap
  1319. ovlabel = QtWidgets.QLabel('%s:' % _('Overlap'))
  1320. ovlabel.setToolTip(
  1321. _("How much (percentage) of the tool width to overlap each tool pass.\n"
  1322. "Adjust the value starting with lower values\n"
  1323. "and increasing it if areas that should be painted are still \n"
  1324. "not painted.\n"
  1325. "Lower values = faster processing, faster execution on CNC.\n"
  1326. "Higher values = slow processing and slow execution on CNC\n"
  1327. "due of too many paths.")
  1328. )
  1329. self.paintoverlap_entry = FCDoubleSpinner(suffix='%')
  1330. self.paintoverlap_entry.set_precision(3)
  1331. self.paintoverlap_entry.setWrapping(True)
  1332. self.paintoverlap_entry.setRange(0.0000, 99.9999)
  1333. self.paintoverlap_entry.setSingleStep(0.1)
  1334. self.paintoverlap_entry.setObjectName('gdb_p_overlap')
  1335. self.grid3.addWidget(ovlabel, 1, 0)
  1336. self.grid3.addWidget(self.paintoverlap_entry, 1, 1)
  1337. # Margin
  1338. marginlabel = QtWidgets.QLabel('%s:' % _('Margin'))
  1339. marginlabel.setToolTip(
  1340. _("Distance by which to avoid\n"
  1341. "the edges of the polygon to\n"
  1342. "be painted.")
  1343. )
  1344. self.paintmargin_entry = FCDoubleSpinner()
  1345. self.paintmargin_entry.set_precision(self.decimals)
  1346. self.paintmargin_entry.set_range(-9999.9999, 9999.9999)
  1347. self.paintmargin_entry.setObjectName('gdb_p_margin')
  1348. self.grid3.addWidget(marginlabel, 2, 0)
  1349. self.grid3.addWidget(self.paintmargin_entry, 2, 1)
  1350. # Method
  1351. methodlabel = QtWidgets.QLabel('%s:' % _('Method'))
  1352. methodlabel.setToolTip(
  1353. _("Algorithm for painting:\n"
  1354. "- Standard: Fixed step inwards.\n"
  1355. "- Seed-based: Outwards from seed.\n"
  1356. "- Line-based: Parallel lines.\n"
  1357. "- Laser-lines: Active only for Gerber objects.\n"
  1358. "Will create lines that follow the traces.\n"
  1359. "- Combo: In case of failure a new method will be picked from the above\n"
  1360. "in the order specified.")
  1361. )
  1362. self.paintmethod_combo = FCComboBox()
  1363. self.paintmethod_combo.addItems(
  1364. [_("Standard"), _("Seed"), _("Lines"), _("Laser_lines"), _("Combo")]
  1365. )
  1366. idx = self.paintmethod_combo.findText(_("Laser_lines"))
  1367. self.paintmethod_combo.model().item(idx).setEnabled(False)
  1368. self.paintmethod_combo.setObjectName('gdb_p_method')
  1369. self.grid3.addWidget(methodlabel, 7, 0)
  1370. self.grid3.addWidget(self.paintmethod_combo, 7, 1)
  1371. # Connect lines
  1372. self.pathconnect_cb = FCCheckBox('%s' % _("Connect"))
  1373. self.pathconnect_cb.setObjectName('gdb_p_connect')
  1374. self.pathconnect_cb.setToolTip(
  1375. _("Draw lines between resulting\n"
  1376. "segments to minimize tool lifts.")
  1377. )
  1378. self.paintcontour_cb = FCCheckBox('%s' % _("Contour"))
  1379. self.paintcontour_cb.setObjectName('gdb_p_contour')
  1380. self.paintcontour_cb.setToolTip(
  1381. _("Cut around the perimeter of the polygon\n"
  1382. "to trim rough edges.")
  1383. )
  1384. self.grid3.addWidget(self.pathconnect_cb, 10, 0)
  1385. self.grid3.addWidget(self.paintcontour_cb, 10, 1)
  1386. # ###########################################################################
  1387. # ############### Paint UI form #############################################
  1388. # ###########################################################################
  1389. self.grid4 = QtWidgets.QGridLayout()
  1390. self.iso_vlay.addLayout(self.grid4)
  1391. self.grid4.setColumnStretch(0, 0)
  1392. self.grid4.setColumnStretch(1, 1)
  1393. self.iso_vlay.addStretch()
  1394. # Passes
  1395. passlabel = QtWidgets.QLabel('%s:' % _('Passes'))
  1396. passlabel.setToolTip(
  1397. _("Width of the isolation gap in\n"
  1398. "number (integer) of tool widths.")
  1399. )
  1400. self.passes_entry = FCSpinner()
  1401. self.passes_entry.set_range(1, 999)
  1402. self.passes_entry.setObjectName("gdb_i_passes")
  1403. self.grid4.addWidget(passlabel, 0, 0)
  1404. self.grid4.addWidget(self.passes_entry, 0, 1)
  1405. # Overlap Entry
  1406. overlabel = QtWidgets.QLabel('%s:' % _('Overlap'))
  1407. overlabel.setToolTip(
  1408. _("How much (percentage) of the tool width to overlap each tool pass.")
  1409. )
  1410. self.iso_overlap_entry = FCDoubleSpinner(suffix='%')
  1411. self.iso_overlap_entry.set_precision(self.decimals)
  1412. self.iso_overlap_entry.setWrapping(True)
  1413. self.iso_overlap_entry.set_range(0.0000, 99.9999)
  1414. self.iso_overlap_entry.setSingleStep(0.1)
  1415. self.iso_overlap_entry.setObjectName("gdb_i_overlap")
  1416. self.grid4.addWidget(overlabel, 2, 0)
  1417. self.grid4.addWidget(self.iso_overlap_entry, 2, 1)
  1418. # Milling Type Radio Button
  1419. self.milling_type_label = QtWidgets.QLabel('%s:' % _('Milling Type'))
  1420. self.milling_type_label.setToolTip(
  1421. _("Milling type when the selected tool is of type: 'iso_op':\n"
  1422. "- climb / best for precision milling and to reduce tool usage\n"
  1423. "- conventional / useful when there is no backlash compensation")
  1424. )
  1425. self.milling_type_radio = RadioSet([{'label': _('Climb'), 'value': 'cl'},
  1426. {'label': _('Conventional'), 'value': 'cv'}])
  1427. self.milling_type_radio.setToolTip(
  1428. _("Milling type when the selected tool is of type: 'iso_op':\n"
  1429. "- climb / best for precision milling and to reduce tool usage\n"
  1430. "- conventional / useful when there is no backlash compensation")
  1431. )
  1432. self.milling_type_radio.setObjectName("gdb_i_milling_type")
  1433. self.grid4.addWidget(self.milling_type_label, 4, 0)
  1434. self.grid4.addWidget(self.milling_type_radio, 4, 1)
  1435. # Follow
  1436. self.follow_label = QtWidgets.QLabel('%s:' % _('Follow'))
  1437. self.follow_label.setToolTip(
  1438. _("Generate a 'Follow' geometry.\n"
  1439. "This means that it will cut through\n"
  1440. "the middle of the trace.")
  1441. )
  1442. self.follow_cb = FCCheckBox()
  1443. self.follow_cb.setToolTip(_("Generate a 'Follow' geometry.\n"
  1444. "This means that it will cut through\n"
  1445. "the middle of the trace."))
  1446. self.follow_cb.setObjectName("gdb_i_follow")
  1447. self.grid4.addWidget(self.follow_label, 6, 0)
  1448. self.grid4.addWidget(self.follow_cb, 6, 1)
  1449. # Isolation Type
  1450. self.iso_type_label = QtWidgets.QLabel('%s:' % _('Isolation Type'))
  1451. self.iso_type_label.setToolTip(
  1452. _("Choose how the isolation will be executed:\n"
  1453. "- 'Full' -> complete isolation of polygons\n"
  1454. "- 'Ext' -> will isolate only on the outside\n"
  1455. "- 'Int' -> will isolate only on the inside\n"
  1456. "'Exterior' isolation is almost always possible\n"
  1457. "(with the right tool) but 'Interior'\n"
  1458. "isolation can be done only when there is an opening\n"
  1459. "inside of the polygon (e.g polygon is a 'doughnut' shape).")
  1460. )
  1461. self.iso_type_radio = RadioSet([{'label': _('Full'), 'value': 'full'},
  1462. {'label': _('Ext'), 'value': 'ext'},
  1463. {'label': _('Int'), 'value': 'int'}])
  1464. self.iso_type_radio.setObjectName("gdb_i_iso_type")
  1465. self.grid4.addWidget(self.iso_type_label, 8, 0)
  1466. self.grid4.addWidget(self.iso_type_radio, 8, 1)
  1467. # ####################################################################
  1468. # ####################################################################
  1469. # GUI for the lower part of the window
  1470. # ####################################################################
  1471. # ####################################################################
  1472. new_vlay = QtWidgets.QVBoxLayout()
  1473. grid_layout.addLayout(new_vlay, 1, 0, 1, 2)
  1474. self.buttons_frame = QtWidgets.QFrame()
  1475. self.buttons_frame.setContentsMargins(0, 0, 0, 0)
  1476. new_vlay.addWidget(self.buttons_frame)
  1477. self.buttons_box = QtWidgets.QHBoxLayout()
  1478. self.buttons_box.setContentsMargins(0, 0, 0, 0)
  1479. self.buttons_frame.setLayout(self.buttons_box)
  1480. self.buttons_frame.show()
  1481. add_entry_btn = FCButton(_("Add Tool in DB"))
  1482. add_entry_btn.setToolTip(
  1483. _("Add a new tool in the Tools Database.\n"
  1484. "It will be used in the Geometry UI.\n"
  1485. "You can edit it after it is added.")
  1486. )
  1487. self.buttons_box.addWidget(add_entry_btn)
  1488. # add_fct_entry_btn = FCButton(_("Add Paint/NCC Tool in DB"))
  1489. # add_fct_entry_btn.setToolTip(
  1490. # _("Add a new tool in the Tools Database.\n"
  1491. # "It will be used in the Paint/NCC Tools UI.\n"
  1492. # "You can edit it after it is added.")
  1493. # )
  1494. # self.buttons_box.addWidget(add_fct_entry_btn)
  1495. remove_entry_btn = FCButton(_("Delete Tool from DB"))
  1496. remove_entry_btn.setToolTip(
  1497. _("Remove a selection of tools in the Tools Database.")
  1498. )
  1499. self.buttons_box.addWidget(remove_entry_btn)
  1500. export_db_btn = FCButton(_("Export DB"))
  1501. export_db_btn.setToolTip(
  1502. _("Save the Tools Database to a custom text file.")
  1503. )
  1504. self.buttons_box.addWidget(export_db_btn)
  1505. import_db_btn = FCButton(_("Import DB"))
  1506. import_db_btn.setToolTip(
  1507. _("Load the Tools Database information's from a custom text file.")
  1508. )
  1509. self.buttons_box.addWidget(import_db_btn)
  1510. self.save_db_btn = FCButton(_("Save DB"))
  1511. self.save_db_btn.setToolTip(
  1512. _("Save the Tools Database information's.")
  1513. )
  1514. self.buttons_box.addWidget(self.save_db_btn)
  1515. self.add_tool_from_db = FCButton(_("Add Tool from Tools DB"))
  1516. self.add_tool_from_db.setToolTip(
  1517. _("Add a new tool in the Tools Table of the\n"
  1518. "active Geometry object after selecting a tool\n"
  1519. "in the Tools Database.")
  1520. )
  1521. self.add_tool_from_db.hide()
  1522. self.cancel_tool_from_db = FCButton(_("Cancel"))
  1523. self.cancel_tool_from_db.hide()
  1524. hlay = QtWidgets.QHBoxLayout()
  1525. tree_layout.addLayout(hlay)
  1526. hlay.addWidget(self.add_tool_from_db)
  1527. hlay.addWidget(self.cancel_tool_from_db)
  1528. hlay.addStretch()
  1529. # ##############################################################################
  1530. # ##############################################################################
  1531. # ########## SETUP THE DICTIONARIES THAT HOLD THE WIDGETS #####################
  1532. # ##############################################################################
  1533. # ##############################################################################
  1534. self.form_fields = {
  1535. # Basic
  1536. "name": self.name_entry,
  1537. "tooldia": self.dia_entry,
  1538. "tool_type": self.shape_combo,
  1539. "cutz": self.cutz_entry,
  1540. "multidepth": self.multidepth_cb,
  1541. "depthperpass": self.multidepth_entry,
  1542. "travelz": self.travelz_entry,
  1543. "feedrate": self.frxy_entry,
  1544. "feedrate_z": self.frz_entry,
  1545. "spindlespeed": self.spindle_entry,
  1546. "dwell": self.dwell_cb,
  1547. "dwelltime": self.dwelltime_entry,
  1548. # Advanced
  1549. "type": self.type_combo,
  1550. "offset": self.tooloffset_combo,
  1551. "offset_value": self.custom_offset_entry,
  1552. "vtipdia": self.vdia_entry,
  1553. "vtipangle": self.vangle_entry,
  1554. "feedrate_rapid": self.frapids_entry,
  1555. "extracut": self.ecut_cb,
  1556. "extracut_length": self.ecut_length_entry,
  1557. # NCC
  1558. "tools_nccoperation": self.op_radio,
  1559. "tools_nccmilling_type": self.milling_type_radio,
  1560. "tools_nccoverlap": self.ncc_overlap_entry,
  1561. "tools_nccmargin": self.ncc_margin_entry,
  1562. "tools_nccmethod": self.ncc_method_combo,
  1563. "tools_nccconnect": self.ncc_connect_cb,
  1564. "tools_ncccontour": self.ncc_contour_cb,
  1565. "tools_ncc_offset_choice": self.ncc_choice_offset_cb,
  1566. "tools_ncc_offset_value": self.ncc_offset_spinner,
  1567. # Paint
  1568. "tools_paintoverlap": self.paintoverlap_entry,
  1569. "tools_paintmargin": self.paintmargin_entry,
  1570. "tools_paintmethod": self.paintmethod_combo,
  1571. "tools_pathconnect": self.pathconnect_cb,
  1572. "tools_paintcontour": self.paintcontour_cb,
  1573. # Isolation
  1574. "tools_iso_passes": self.passes_entry,
  1575. "tools_iso_overlap": self.iso_overlap_entry,
  1576. "tools_iso_milling_type": self.milling_type_radio,
  1577. "tools_iso_follow": self.follow_cb,
  1578. "tools_iso_isotype": self.iso_type_radio
  1579. }
  1580. self.name2option = {
  1581. # Basic
  1582. "gdb_name": "name",
  1583. "gdb_dia": "tooldia",
  1584. "gdb_shape": "tool_type",
  1585. "gdb_cutz": "cutz",
  1586. "gdb_multidepth": "multidepth",
  1587. "gdb_multidepth_entry": "depthperpass",
  1588. "gdb_travel": "travelz",
  1589. "gdb_frxy": "feedrate",
  1590. "gdb_frz": "feedrate_z",
  1591. "gdb_spindle": "spindlespeed",
  1592. "gdb_dwell": "dwell",
  1593. "gdb_dwelltime": "dwelltime",
  1594. # Advanced
  1595. "gdb_type": "type",
  1596. "gdb_tool_offset": "offset",
  1597. "gdb_custom_offset": "offset_value",
  1598. "gdb_vdia": "vtipdia",
  1599. "gdb_vangle": "vtipangle",
  1600. "gdb_frapids": "feedrate_rapid",
  1601. "gdb_ecut": "extracut",
  1602. "gdb_ecut_length": "extracut_length",
  1603. # NCC
  1604. "gdb_n_operation": "tools_nccoperation",
  1605. "gdb_n_overlap": "tools_nccoverlap",
  1606. "gdb_n_margin": "tools_nccmargin",
  1607. "gdb_n_method": "tools_nccmethod",
  1608. "gdb_n_connect": "tools_nccconnect",
  1609. "gdb_n_contour": "tools_ncccontour",
  1610. "gdb_n_offset": "tools_ncc_offset_choice",
  1611. "gdb_n_offset_value": "tools_ncc_offset_value",
  1612. "gdb_n_milling_type": "tools_nccmilling_type",
  1613. # Paint
  1614. 'gdb_p_overlap': "tools_paintoverlap",
  1615. 'gdb_p_margin': "tools_paintmargin",
  1616. 'gdb_p_method': "tools_paintmethod",
  1617. 'gdb_p_connect': "tools_pathconnect",
  1618. 'gdb_p_contour': "tools_paintcontour",
  1619. # Isolation
  1620. "gdb_i_passes": "tools_iso_passes",
  1621. "gdb_i_overlap": "tools_iso_overlap",
  1622. "gdb_i_milling_type": "tools_iso_milling_type",
  1623. "gdb_i_follow": "tools_iso_follow",
  1624. "gdb_i_iso_type": "tools_iso_isotype"
  1625. }
  1626. self.current_toolid = None
  1627. # variable to show if double clicking and item will trigger adding a tool from DB
  1628. self.ok_to_add = False
  1629. # ##############################################################################
  1630. # ######################## SIGNALS #############################################
  1631. # ##############################################################################
  1632. add_entry_btn.clicked.connect(self.on_tool_add)
  1633. remove_entry_btn.clicked.connect(self.on_tool_delete)
  1634. export_db_btn.clicked.connect(self.on_export_tools_db_file)
  1635. import_db_btn.clicked.connect(self.on_import_tools_db_file)
  1636. self.save_db_btn.clicked.connect(self.on_save_db_btn_click)
  1637. # closebtn.clicked.connect(self.accept)
  1638. self.add_tool_from_db.clicked.connect(self.on_tool_requested_from_app)
  1639. self.cancel_tool_from_db.clicked.connect(self.on_cancel_tool)
  1640. # self.tree_widget.selectionModel().selectionChanged.connect(self.on_list_selection_change)
  1641. self.tree_widget.currentItemChanged.connect(self.on_list_selection_change)
  1642. self.tree_widget.itemChanged.connect(self.on_list_item_edited)
  1643. self.tree_widget.customContextMenuRequested.connect(self.on_menu_request)
  1644. self.tree_widget.itemDoubleClicked.connect(self.on_item_double_clicked)
  1645. self.setup_db_ui()
  1646. def on_menu_request(self, pos):
  1647. menu = QtWidgets.QMenu()
  1648. add_tool = menu.addAction(QtGui.QIcon(self.app.resource_location + '/plus16.png'), _("Add to DB"))
  1649. add_tool.triggered.connect(self.on_tool_add)
  1650. copy_tool = menu.addAction(QtGui.QIcon(self.app.resource_location + '/copy16.png'), _("Copy from DB"))
  1651. copy_tool.triggered.connect(self.on_tool_copy)
  1652. delete_tool = menu.addAction(QtGui.QIcon(self.app.resource_location + '/delete32.png'), _("Delete from DB"))
  1653. delete_tool.triggered.connect(self.on_tool_delete)
  1654. # tree_item = self.tree_widget.itemAt(pos)
  1655. menu.exec(self.tree_widget.viewport().mapToGlobal(pos))
  1656. def on_item_double_clicked(self, item, column):
  1657. if column == 0 and self.ok_to_add is True:
  1658. self.ok_to_add = False
  1659. self.on_tool_requested_from_app()
  1660. def on_list_selection_change(self, current, previous):
  1661. # for idx in current.indexes():
  1662. # print(idx.data())
  1663. # print(current.text(0))
  1664. self.current_toolid = int(current.text(0))
  1665. self.storage_to_form(self.db_tool_dict[current.text(0)])
  1666. def on_list_item_edited(self, item, column):
  1667. if column == 0:
  1668. return
  1669. self.name_entry.set_value(item.text(1))
  1670. def storage_to_form(self, dict_storage):
  1671. for form_key in self.form_fields:
  1672. for storage_key in dict_storage:
  1673. if form_key == storage_key:
  1674. try:
  1675. self.form_fields[form_key].set_value(dict_storage[form_key])
  1676. except Exception as e:
  1677. print(str(e))
  1678. if storage_key == 'data':
  1679. for data_key in dict_storage[storage_key]:
  1680. if form_key == data_key:
  1681. try:
  1682. self.form_fields[form_key].set_value(dict_storage['data'][data_key])
  1683. except Exception as e:
  1684. print(str(e))
  1685. def form_to_storage(self, tool):
  1686. self.blockSignals(True)
  1687. widget_changed = self.sender()
  1688. wdg_objname = widget_changed.objectName()
  1689. option_changed = self.name2option[wdg_objname]
  1690. tooluid_item = int(tool)
  1691. for tooluid_key, tooluid_val in self.db_tool_dict.items():
  1692. if int(tooluid_key) == tooluid_item:
  1693. new_option_value = self.form_fields[option_changed].get_value()
  1694. if option_changed in tooluid_val:
  1695. tooluid_val[option_changed] = new_option_value
  1696. if option_changed in tooluid_val['data']:
  1697. tooluid_val['data'][option_changed] = new_option_value
  1698. self.blockSignals(False)
  1699. def setup_db_ui(self):
  1700. filename = self.app.data_path + '/geo_tools_db.FlatDB'
  1701. # load the database tools from the file
  1702. try:
  1703. with open(filename) as f:
  1704. tools = f.read()
  1705. except IOError:
  1706. self.app.log.error("Could not load tools DB file.")
  1707. self.app.inform.emit('[ERROR] %s' % _("Could not load Tools DB file."))
  1708. return
  1709. try:
  1710. self.db_tool_dict = json.loads(tools)
  1711. except Exception:
  1712. e = sys.exc_info()[0]
  1713. self.app.log.error(str(e))
  1714. self.app.inform.emit('[ERROR] %s' % _("Failed to parse Tools DB file."))
  1715. return
  1716. self.app.inform.emit('[success] %s: %s' % (_("Loaded FlatCAM Tools DB from"), filename))
  1717. self.build_db_ui()
  1718. def build_db_ui(self):
  1719. self.ui_disconnect()
  1720. nr_crt = 0
  1721. parent = self.tree_widget
  1722. self.tree_widget.blockSignals(True)
  1723. self.tree_widget.clear()
  1724. self.tree_widget.blockSignals(False)
  1725. for toolid, dict_val in self.db_tool_dict.items():
  1726. row = nr_crt
  1727. nr_crt += 1
  1728. t_name = dict_val['name']
  1729. try:
  1730. # self.add_tool_table_line(row, name=t_name, tooldict=dict_val)
  1731. self.tree_widget.blockSignals(True)
  1732. try:
  1733. self.tree_widget.addParentEditable(parent=parent, title=[str(row+1), t_name], editable=True)
  1734. except Exception as e:
  1735. print('FlatCAMCoomn.ToolDB2.build_db_ui() -> ', str(e))
  1736. self.tree_widget.blockSignals(False)
  1737. except Exception as e:
  1738. self.app.log.debug("ToolDB.build_db_ui.add_tool_table_line() --> %s" % str(e))
  1739. if self.current_toolid is None or self.current_toolid < 1:
  1740. if self.db_tool_dict:
  1741. self.storage_to_form(self.db_tool_dict['1'])
  1742. # Enable AppGUI
  1743. self.basic_box.setEnabled(True)
  1744. self.advanced_box.setEnabled(True)
  1745. self.ncc_box.setEnabled(True)
  1746. self.paint_box.setEnabled(True)
  1747. self.iso_box.setEnabled(True)
  1748. self.tree_widget.setCurrentItem(self.tree_widget.topLevelItem(0))
  1749. # self.tree_widget.setFocus()
  1750. else:
  1751. # Disable AppGUI
  1752. self.basic_box.setEnabled(False)
  1753. self.advanced_box.setEnabled(False)
  1754. self.ncc_box.setEnabled(False)
  1755. self.paint_box.setEnabled(False)
  1756. self.iso_box.setEnabled(False)
  1757. else:
  1758. self.storage_to_form(self.db_tool_dict[str(self.current_toolid)])
  1759. self.ui_connect()
  1760. def on_tool_add(self):
  1761. """
  1762. Add a tool in the DB Tool Table
  1763. :return: None
  1764. """
  1765. default_data = {}
  1766. default_data.update({
  1767. "plot": True,
  1768. "cutz": float(self.app.defaults["geometry_cutz"]),
  1769. "multidepth": self.app.defaults["geometry_multidepth"],
  1770. "depthperpass": float(self.app.defaults["geometry_depthperpass"]),
  1771. "vtipdia": float(self.app.defaults["geometry_vtipdia"]),
  1772. "vtipangle": float(self.app.defaults["geometry_vtipangle"]),
  1773. "travelz": float(self.app.defaults["geometry_travelz"]),
  1774. "feedrate": float(self.app.defaults["geometry_feedrate"]),
  1775. "feedrate_z": float(self.app.defaults["geometry_feedrate_z"]),
  1776. "feedrate_rapid": float(self.app.defaults["geometry_feedrate_rapid"]),
  1777. "spindlespeed": self.app.defaults["geometry_spindlespeed"],
  1778. "dwell": self.app.defaults["geometry_dwell"],
  1779. "dwelltime": float(self.app.defaults["geometry_dwelltime"]),
  1780. "ppname_g": self.app.defaults["geometry_ppname_g"],
  1781. "extracut": self.app.defaults["geometry_extracut"],
  1782. "extracut_length": float(self.app.defaults["geometry_extracut_length"]),
  1783. "toolchange": self.app.defaults["geometry_toolchange"],
  1784. "toolchangexy": self.app.defaults["geometry_toolchangexy"],
  1785. "toolchangez": float(self.app.defaults["geometry_toolchangez"]),
  1786. "startz": self.app.defaults["geometry_startz"],
  1787. "endz": float(self.app.defaults["geometry_endz"]),
  1788. # NCC
  1789. "tools_nccoperation": self.app.defaults["tools_nccoperation"],
  1790. "tools_nccmilling_type": self.app.defaults["tools_nccmilling_type"],
  1791. "tools_nccoverlap": float(self.app.defaults["tools_nccoverlap"]),
  1792. "tools_nccmargin": float(self.app.defaults["tools_nccmargin"]),
  1793. "tools_nccmethod": self.app.defaults["tools_nccmethod"],
  1794. "tools_nccconnect": self.app.defaults["tools_nccconnect"],
  1795. "tools_ncccontour": self.app.defaults["tools_ncccontour"],
  1796. "tools_ncc_offset_choice": self.app.defaults["tools_ncc_offset_choice"],
  1797. "tools_ncc_offset_value": float(self.app.defaults["tools_ncc_offset_value"]),
  1798. # Paint
  1799. "tools_paintoverlap": float(self.app.defaults["tools_paintoverlap"]),
  1800. "tools_paintmargin": float(self.app.defaults["tools_paintmargin"]),
  1801. "tools_paintmethod": self.app.defaults["tools_paintmethod"],
  1802. "tools_pathconnect": self.app.defaults["tools_pathconnect"],
  1803. "tools_paintcontour": self.app.defaults["tools_paintcontour"],
  1804. # Isolation
  1805. "tools_iso_passes": int(self.app.defaults["tools_iso_passes"]),
  1806. "tools_iso_overlap": float(self.app.defaults["tools_iso_overlap"]),
  1807. "tools_iso_milling_type": self.app.defaults["tools_iso_milling_type"],
  1808. "tools_iso_follow": self.app.defaults["tools_iso_follow"],
  1809. "tools_iso_isotype": self.app.defaults["tools_iso_isotype"],
  1810. })
  1811. dict_elem = {}
  1812. dict_elem['name'] = 'new_tool'
  1813. if type(self.app.defaults["geometry_cnctooldia"]) == float:
  1814. dict_elem['tooldia'] = self.app.defaults["geometry_cnctooldia"]
  1815. else:
  1816. try:
  1817. tools_string = self.app.defaults["geometry_cnctooldia"].split(",")
  1818. tools_diameters = [eval(a) for a in tools_string if a != '']
  1819. dict_elem['tooldia'] = tools_diameters[0] if tools_diameters else 0.0
  1820. except Exception as e:
  1821. self.app.log.debug("ToolDB.on_tool_add() --> %s" % str(e))
  1822. return
  1823. dict_elem['offset'] = 'Path'
  1824. dict_elem['offset_value'] = 0.0
  1825. dict_elem['type'] = 'Rough'
  1826. dict_elem['tool_type'] = 'C1'
  1827. dict_elem['data'] = default_data
  1828. new_toolid = len(self.db_tool_dict) + 1
  1829. self.db_tool_dict[str(new_toolid)] = deepcopy(dict_elem)
  1830. # add the new entry to the Tools DB table
  1831. self.update_storage()
  1832. self.build_db_ui()
  1833. # select the last Tree item just added
  1834. nr_items = self.tree_widget.topLevelItemCount()
  1835. if nr_items:
  1836. last_item = self.tree_widget.topLevelItem(nr_items - 1)
  1837. self.tree_widget.setCurrentItem(last_item)
  1838. last_item.setSelected(True)
  1839. self.app.inform.emit('[success] %s' % _("Tool added to DB."))
  1840. def on_tool_copy(self):
  1841. """
  1842. Copy a selection of Tools in the Tools DB table
  1843. :return:
  1844. """
  1845. new_tool_id = len(self.db_tool_dict)
  1846. for item in self.tree_widget.selectedItems():
  1847. old_tool_id = item.data(0, QtCore.Qt.DisplayRole)
  1848. for toolid, dict_val in list(self.db_tool_dict.items()):
  1849. if int(old_tool_id) == int(toolid):
  1850. new_tool_id += 1
  1851. new_key = str(new_tool_id)
  1852. self.db_tool_dict.update({
  1853. new_key: deepcopy(dict_val)
  1854. })
  1855. self.current_toolid = new_tool_id
  1856. self.update_storage()
  1857. self.build_db_ui()
  1858. # select the last Tree item just added
  1859. nr_items = self.tree_widget.topLevelItemCount()
  1860. if nr_items:
  1861. last_item = self.tree_widget.topLevelItem(nr_items - 1)
  1862. self.tree_widget.setCurrentItem(last_item)
  1863. last_item.setSelected(True)
  1864. self.callback_app()
  1865. self.app.inform.emit('[success] %s' % _("Tool copied from Tools DB."))
  1866. def on_tool_delete(self):
  1867. """
  1868. Delete a selection of Tools in the Tools DB table
  1869. :return:
  1870. """
  1871. for item in self.tree_widget.selectedItems():
  1872. toolname_to_remove = item.data(0, QtCore.Qt.DisplayRole)
  1873. for toolid, dict_val in list(self.db_tool_dict.items()):
  1874. if int(toolname_to_remove) == int(toolid):
  1875. # remove from the storage
  1876. self.db_tool_dict.pop(toolid, None)
  1877. self.current_toolid -= 1
  1878. self.update_storage()
  1879. self.build_db_ui()
  1880. # select the first Tree item
  1881. nr_items = self.tree_widget.topLevelItemCount()
  1882. if nr_items:
  1883. first_item = self.tree_widget.topLevelItem(0)
  1884. self.tree_widget.setCurrentItem(first_item)
  1885. first_item.setSelected(True)
  1886. self.app.inform.emit('[success] %s' % _("Tool removed from Tools DB."))
  1887. def on_export_tools_db_file(self):
  1888. self.app.defaults.report_usage("on_export_tools_db_file")
  1889. self.app.log.debug("on_export_tools_db_file()")
  1890. date = str(datetime.today()).rpartition('.')[0]
  1891. date = ''.join(c for c in date if c not in ':-')
  1892. date = date.replace(' ', '_')
  1893. filter__ = "Text File (*.TXT);;All Files (*.*)"
  1894. filename, _f = FCFileSaveDialog.get_saved_filename(caption=_("Export Tools Database"),
  1895. directory='{l_save}/FlatCAM_{n}_{date}'.format(
  1896. l_save=str(self.app.get_last_save_folder()),
  1897. n=_("Tools_Database"),
  1898. date=date),
  1899. ext_filter=filter__)
  1900. filename = str(filename)
  1901. if filename == "":
  1902. self.app.inform.emit('[WARNING_NOTCL] %s' % _("Cancelled."))
  1903. return
  1904. else:
  1905. try:
  1906. f = open(filename, 'w')
  1907. f.close()
  1908. except PermissionError:
  1909. self.app.inform.emit('[WARNING] %s' %
  1910. _("Permission denied, saving not possible.\n"
  1911. "Most likely another app is holding the file open and not accessible."))
  1912. return
  1913. except IOError:
  1914. self.app.log.debug('Creating a new Tools DB file ...')
  1915. f = open(filename, 'w')
  1916. f.close()
  1917. except Exception:
  1918. e = sys.exc_info()[0]
  1919. self.app.log.error("Could not load Tools DB file.")
  1920. self.app.log.error(str(e))
  1921. self.app.inform.emit('[ERROR_NOTCL] %s' % _("Could not load Tools DB file."))
  1922. return
  1923. # Save update options
  1924. try:
  1925. # Save Tools DB in a file
  1926. try:
  1927. with open(filename, "w") as f:
  1928. json.dump(self.db_tool_dict, f, default=to_dict, indent=2)
  1929. except Exception as e:
  1930. self.app.log.debug("App.on_save_tools_db() --> %s" % str(e))
  1931. self.app.inform.emit('[ERROR_NOTCL] %s' % _("Failed to write Tools DB to file."))
  1932. return
  1933. except Exception:
  1934. self.app.inform.emit('[ERROR_NOTCL] %s' % _("Failed to write Tools DB to file."))
  1935. return
  1936. self.app.inform.emit('[success] %s: %s' % (_("Exported Tools DB to"), filename))
  1937. def on_import_tools_db_file(self):
  1938. self.app.defaults.report_usage("on_import_tools_db_file")
  1939. self.app.log.debug("on_import_tools_db_file()")
  1940. filter__ = "Text File (*.TXT);;All Files (*.*)"
  1941. filename, _f = QtWidgets.QFileDialog.getOpenFileName(caption=_("Import FlatCAM Tools DB"), filter=filter__)
  1942. if filename == "":
  1943. self.app.inform.emit('[WARNING_NOTCL] %s' % _("Cancelled."))
  1944. else:
  1945. try:
  1946. with open(filename) as f:
  1947. tools_in_db = f.read()
  1948. except IOError:
  1949. self.app.log.error("Could not load Tools DB file.")
  1950. self.app.inform.emit('[ERROR_NOTCL] %s' % _("Could not load Tools DB file."))
  1951. return
  1952. try:
  1953. self.db_tool_dict = json.loads(tools_in_db)
  1954. except Exception:
  1955. e = sys.exc_info()[0]
  1956. self.app.log.error(str(e))
  1957. self.app.inform.emit('[ERROR] %s' % _("Failed to parse Tools DB file."))
  1958. return
  1959. self.app.inform.emit('[success] %s: %s' % (_("Loaded FlatCAM Tools DB from"), filename))
  1960. self.build_db_ui()
  1961. self.update_storage()
  1962. def on_save_tools_db(self, silent=False):
  1963. self.app.log.debug("ToolsDB.on_save_button() --> Saving Tools Database to file.")
  1964. filename = self.app.data_path + "/geo_tools_db.FlatDB"
  1965. # Preferences save, update the color of the Tools DB Tab text
  1966. for idx in range(self.app.ui.plot_tab_area.count()):
  1967. if self.app.ui.plot_tab_area.tabText(idx) == _("Tools Database"):
  1968. self.app.ui.plot_tab_area.tabBar.setTabTextColor(idx, QtGui.QColor('black'))
  1969. self.save_db_btn.setStyleSheet("")
  1970. # Save Tools DB in a file
  1971. try:
  1972. f = open(filename, "w")
  1973. json.dump(self.db_tool_dict, f, default=to_dict, indent=2)
  1974. f.close()
  1975. except Exception as e:
  1976. self.app.log.debug("ToolsDB.on_save_tools_db() --> %s" % str(e))
  1977. self.app.inform.emit('[ERROR_NOTCL] %s' % _("Failed to write Tools DB to file."))
  1978. return
  1979. if not silent:
  1980. self.app.inform.emit('[success] %s' % _("Saved Tools DB."))
  1981. def on_save_db_btn_click(self):
  1982. self.app.tools_db_changed_flag = False
  1983. self.on_save_tools_db()
  1984. def ui_connect(self):
  1985. # make sure that we don't make multiple connections to the widgets
  1986. self.ui_disconnect()
  1987. self.name_entry.editingFinished.connect(self.update_tree_name)
  1988. for key in self.form_fields:
  1989. wdg = self.form_fields[key]
  1990. # FCEntry
  1991. if isinstance(wdg, FCEntry):
  1992. wdg.textChanged.connect(self.update_storage)
  1993. # ComboBox
  1994. if isinstance(wdg, FCComboBox):
  1995. wdg.currentIndexChanged.connect(self.update_storage)
  1996. # CheckBox
  1997. if isinstance(wdg, FCCheckBox):
  1998. wdg.toggled.connect(self.update_storage)
  1999. # FCRadio
  2000. if isinstance(wdg, RadioSet):
  2001. wdg.activated_custom.connect(self.update_storage)
  2002. # SpinBox, DoubleSpinBox
  2003. if isinstance(wdg, FCSpinner) or isinstance(wdg, FCDoubleSpinner):
  2004. wdg.valueChanged.connect(self.update_storage)
  2005. def ui_disconnect(self):
  2006. try:
  2007. self.name_entry.editingFinished.disconnect(self.update_tree_name)
  2008. except (TypeError, AttributeError):
  2009. pass
  2010. for key in self.form_fields:
  2011. wdg = self.form_fields[key]
  2012. # FCEntry
  2013. if isinstance(wdg, FCEntry):
  2014. try:
  2015. wdg.textChanged.disconnect(self.update_storage)
  2016. except (TypeError, AttributeError):
  2017. pass
  2018. # ComboBox
  2019. if isinstance(wdg, FCComboBox):
  2020. try:
  2021. wdg.currentIndexChanged.disconnect(self.update_storage)
  2022. except (TypeError, AttributeError):
  2023. pass
  2024. # CheckBox
  2025. if isinstance(wdg, FCCheckBox):
  2026. try:
  2027. wdg.toggled.disconnect(self.update_storage)
  2028. except (TypeError, AttributeError):
  2029. pass
  2030. # FCRadio
  2031. if isinstance(wdg, RadioSet):
  2032. try:
  2033. wdg.activated_custom.disconnect(self.update_storage)
  2034. except (TypeError, AttributeError):
  2035. pass
  2036. # SpinBox, DoubleSpinBox
  2037. if isinstance(wdg, FCSpinner) or isinstance(wdg, FCDoubleSpinner):
  2038. try:
  2039. wdg.valueChanged.disconnect(self.update_storage)
  2040. except (TypeError, AttributeError):
  2041. pass
  2042. def update_tree_name(self):
  2043. val = self.name_entry.get_value()
  2044. item = self.tree_widget.currentItem()
  2045. if item is None:
  2046. return
  2047. # I'm setting the value for the second column (designated by 1) because first column holds the ID
  2048. # and second column holds the Name (this behavior is set in the build_ui method)
  2049. item.setData(1, QtCore.Qt.DisplayRole, val)
  2050. def update_storage(self):
  2051. """
  2052. Update the dictionary that is the storage of the tools 'database'
  2053. :return:
  2054. """
  2055. tool_id = str(self.current_toolid)
  2056. try:
  2057. wdg = self.sender()
  2058. assert isinstance(wdg, QtWidgets.QWidget) or isinstance(wdg, QtWidgets.QAction), \
  2059. "Expected a QWidget got %s" % type(wdg)
  2060. if wdg is None:
  2061. return
  2062. wdg_name = wdg.objectName()
  2063. val = wdg.get_value()
  2064. except AttributeError:
  2065. return
  2066. if wdg_name == "gdb_name":
  2067. self.db_tool_dict[tool_id]['name'] = val
  2068. elif wdg_name == "gdb_dia":
  2069. self.db_tool_dict[tool_id]['tooldia'] = val
  2070. elif wdg_name == "gdb_tool_offset":
  2071. self.db_tool_dict[tool_id]['offset'] = val
  2072. elif wdg_name == "gdb_custom_offset":
  2073. self.db_tool_dict[tool_id]['offset_value'] = val
  2074. elif wdg_name == "gdb_type":
  2075. self.db_tool_dict[tool_id]['type'] = val
  2076. elif wdg_name == "gdb_shape":
  2077. self.db_tool_dict[tool_id]['tool_type'] = val
  2078. else:
  2079. if wdg_name == "gdb_cutz":
  2080. self.db_tool_dict[tool_id]['data']['cutz'] = val
  2081. elif wdg_name == "gdb_multidepth":
  2082. self.db_tool_dict[tool_id]['data']['multidepth'] = val
  2083. elif wdg_name == "gdb_multidepth_entry":
  2084. self.db_tool_dict[tool_id]['data']['depthperpass'] = val
  2085. elif wdg_name == "gdb_travel":
  2086. self.db_tool_dict[tool_id]['data']['travelz'] = val
  2087. elif wdg_name == "gdb_frxy":
  2088. self.db_tool_dict[tool_id]['data']['feedrate'] = val
  2089. elif wdg_name == "gdb_frz":
  2090. self.db_tool_dict[tool_id]['data']['feedrate_z'] = val
  2091. elif wdg_name == "gdb_spindle":
  2092. self.db_tool_dict[tool_id]['data']['spindlespeed'] = val
  2093. elif wdg_name == "gdb_dwell":
  2094. self.db_tool_dict[tool_id]['data']['dwell'] = val
  2095. elif wdg_name == "gdb_dwelltime":
  2096. self.db_tool_dict[tool_id]['data']['dwelltime'] = val
  2097. elif wdg_name == "gdb_vdia":
  2098. self.db_tool_dict[tool_id]['data']['vtipdia'] = val
  2099. elif wdg_name == "gdb_vangle":
  2100. self.db_tool_dict[tool_id]['data']['vtipangle'] = val
  2101. elif wdg_name == "gdb_frapids":
  2102. self.db_tool_dict[tool_id]['data']['feedrate_rapid'] = val
  2103. elif wdg_name == "gdb_ecut":
  2104. self.db_tool_dict[tool_id]['data']['extracut'] = val
  2105. elif wdg_name == "gdb_ecut_length":
  2106. self.db_tool_dict[tool_id]['data']['extracut_length'] = val
  2107. # NCC Tool
  2108. elif wdg_name == "gdb_n_operation":
  2109. self.db_tool_dict[tool_id]['data']['tools_nccoperation'] = val
  2110. elif wdg_name == "gdb_n_overlap":
  2111. self.db_tool_dict[tool_id]['data']['tools_nccoverlap'] = val
  2112. elif wdg_name == "gdb_n_margin":
  2113. self.db_tool_dict[tool_id]['data']['tools_nccmargin'] = val
  2114. elif wdg_name == "gdb_n_method":
  2115. self.db_tool_dict[tool_id]['data']['tools_nccmethod'] = val
  2116. elif wdg_name == "gdb_n_connect":
  2117. self.db_tool_dict[tool_id]['data']['tools_nccconnect'] = val
  2118. elif wdg_name == "gdb_n_contour":
  2119. self.db_tool_dict[tool_id]['data']['tools_ncccontour'] = val
  2120. elif wdg_name == "gdb_n_offset":
  2121. self.db_tool_dict[tool_id]['data']['tools_ncc_offset_choice'] = val
  2122. elif wdg_name == "gdb_n_offset_value":
  2123. self.db_tool_dict[tool_id]['data']['tools_ncc_offset_value'] = val
  2124. elif wdg_name == "gdb_n_milling_type":
  2125. self.db_tool_dict[tool_id]['data']['tools_nccmilling_type'] = val
  2126. # Paint Tool
  2127. elif wdg_name == "gdb_p_overlap":
  2128. self.db_tool_dict[tool_id]['data']['tools_paintoverlap'] = val
  2129. elif wdg_name == "gdb_p_margin":
  2130. self.db_tool_dict[tool_id]['data']['tools_paintmargin'] = val
  2131. elif wdg_name == "gdb_p_method":
  2132. self.db_tool_dict[tool_id]['data']['tools_paintmethod'] = val
  2133. elif wdg_name == "gdb_p_connect":
  2134. self.db_tool_dict[tool_id]['data']['tools_pathconnect'] = val
  2135. elif wdg_name == "gdb_p_contour":
  2136. self.db_tool_dict[tool_id]['data']['tools_paintcontour'] = val
  2137. # Isolation Tool
  2138. elif wdg_name == "gdb_i_passes":
  2139. self.db_tool_dict[tool_id]['data']['tools_iso_passes'] = val
  2140. elif wdg_name == "gdb_i_overlap":
  2141. self.db_tool_dict[tool_id]['data']['tools_iso_overlap'] = val
  2142. elif wdg_name == "gdb_i_milling_type":
  2143. self.db_tool_dict[tool_id]['data']['tools_iso_milling_type'] = val
  2144. elif wdg_name == "gdb_i_follow":
  2145. self.db_tool_dict[tool_id]['data']['tools_iso_follow'] = val
  2146. elif wdg_name == "gdb_i_iso_type":
  2147. self.db_tool_dict[tool_id]['data']['tools_iso_isotype'] = val
  2148. self.callback_app()
  2149. def on_tool_requested_from_app(self):
  2150. if not self.tree_widget.selectedItems():
  2151. self.app.inform.emit('[WARNING_NOTCL] %s...' % _("No Tool/row selected in the Tools Database table"))
  2152. return
  2153. for item in self.tree_widget.selectedItems():
  2154. tool_uid = item.data(0, QtCore.Qt.DisplayRole)
  2155. for key in self.db_tool_dict.keys():
  2156. if str(key) == str(tool_uid):
  2157. selected_tool = self.db_tool_dict[key]
  2158. self.on_tool_request(tool=selected_tool)
  2159. def on_cancel_tool(self):
  2160. for idx in range(self.app.ui.plot_tab_area.count()):
  2161. if self.app.ui.plot_tab_area.tabText(idx) == _("Tools Database"):
  2162. wdg = self.app.ui.plot_tab_area.widget(idx)
  2163. wdg.deleteLater()
  2164. self.app.ui.plot_tab_area.removeTab(idx)
  2165. self.app.inform.emit('%s' % _("Cancelled adding tool from DB."))
  2166. # def resize_new_tool_table_widget(self, min_size, max_size):
  2167. # """
  2168. # Resize the table widget responsible for adding new tool in the Tool Database
  2169. #
  2170. # :param min_size: passed by rangeChanged signal or the self.new_tool_table_widget.horizontalScrollBar()
  2171. # :param max_size: passed by rangeChanged signal or the self.new_tool_table_widget.horizontalScrollBar()
  2172. # :return:
  2173. # """
  2174. # t_height = self.t_height
  2175. # if max_size > min_size:
  2176. # t_height = self.t_height + self.new_tool_table_widget.verticalScrollBar().height()
  2177. #
  2178. # self.new_tool_table_widget.setMaximumHeight(t_height)
  2179. def closeEvent(self, QCloseEvent):
  2180. super().closeEvent(QCloseEvent)