| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470 |
- # ##########################################################
- # FlatCAM: 2D Post-processing for Manufacturing #
- # http://flatcam.org #
- # Author: Juan Pablo Caram (c) #
- # Date: 2/5/2014 #
- # MIT Licence #
- # ##########################################################
- # ##########################################################
- # File Modified (major mod): Marius Adrian Stanciu #
- # Date: 11/4/2019 #
- # ##########################################################
- from PyQt5 import QtGui, QtCore, QtWidgets
- from flatcamGUI.GUIElements import FCTable, FCEntry, FCButton, FCDoubleSpinner, FCComboBox, FCCheckBox, FCSpinner, \
- FCTree, RadioSet
- from camlib import to_dict
- import sys
- import webbrowser
- import json
- from copy import deepcopy
- from datetime import datetime
- import gettext
- import FlatCAMTranslation as fcTranslate
- import builtins
- fcTranslate.apply_language('strings')
- if '_' not in builtins.__dict__:
- _ = gettext.gettext
- class LoudDict(dict):
- """
- A Dictionary with a callback for
- item changes.
- """
- def __init__(self, *args, **kwargs):
- dict.__init__(self, *args, **kwargs)
- self.callback = lambda x: None
- def __setitem__(self, key, value):
- """
- Overridden __setitem__ method. Will emit 'changed(QString)'
- if the item was changed, with key as parameter.
- """
- if key in self and self.__getitem__(key) == value:
- return
- dict.__setitem__(self, key, value)
- self.callback(key)
- def update(self, *args, **kwargs):
- if len(args) > 1:
- raise TypeError("update expected at most 1 arguments, got %d" % len(args))
- other = dict(*args, **kwargs)
- for key in other:
- self[key] = other[key]
- def set_change_callback(self, callback):
- """
- Assigns a function as callback on item change. The callback
- will receive the key of the object that was changed.
- :param callback: Function to call on item change.
- :type callback: func
- :return: None
- """
- self.callback = callback
- class FCSignal:
- """
- Taken from here: https://blog.abstractfactory.io/dynamic-signals-in-pyqt/
- """
- def __init__(self):
- self.__subscribers = []
- def emit(self, *args, **kwargs):
- for subs in self.__subscribers:
- subs(*args, **kwargs)
- def connect(self, func):
- self.__subscribers.append(func)
- def disconnect(self, func):
- try:
- self.__subscribers.remove(func)
- except ValueError:
- print('Warning: function %s not removed '
- 'from signal %s' % (func, self))
- class BookmarkManager(QtWidgets.QWidget):
- mark_rows = QtCore.pyqtSignal()
- def __init__(self, app, storage, parent=None):
- super(BookmarkManager, self).__init__(parent)
- self.app = app
- assert isinstance(storage, dict), "Storage argument is not a dictionary"
- self.bm_dict = deepcopy(storage)
- # Icon and title
- # self.setWindowIcon(parent.app_icon)
- # self.setWindowTitle(_("Bookmark Manager"))
- # self.resize(600, 400)
- # title = QtWidgets.QLabel(
- # "<font size=8><B>FlatCAM</B></font><BR>"
- # )
- # title.setOpenExternalLinks(True)
- # layouts
- layout = QtWidgets.QVBoxLayout()
- self.setLayout(layout)
- table_hlay = QtWidgets.QHBoxLayout()
- layout.addLayout(table_hlay)
- self.table_widget = FCTable(drag_drop=True, protected_rows=[0, 1])
- self.table_widget.setSelectionBehavior(QtWidgets.QAbstractItemView.SelectRows)
- table_hlay.addWidget(self.table_widget)
- self.table_widget.setColumnCount(3)
- self.table_widget.setColumnWidth(0, 20)
- self.table_widget.setHorizontalHeaderLabels(
- [
- '#',
- _('Title'),
- _('Web Link')
- ]
- )
- self.table_widget.horizontalHeaderItem(0).setToolTip(
- _("Index.\n"
- "The rows in gray color will populate the Bookmarks menu.\n"
- "The number of gray colored rows is set in Preferences."))
- self.table_widget.horizontalHeaderItem(1).setToolTip(
- _("Description of the link that is set as an menu action.\n"
- "Try to keep it short because it is installed as a menu item."))
- self.table_widget.horizontalHeaderItem(2).setToolTip(
- _("Web Link. E.g: https://your_website.org "))
- # pal = QtGui.QPalette()
- # pal.setColor(QtGui.QPalette.Background, Qt.white)
- # New Bookmark
- new_vlay = QtWidgets.QVBoxLayout()
- layout.addLayout(new_vlay)
- new_title_lbl = QtWidgets.QLabel('<b>%s</b>' % _("New Bookmark"))
- new_vlay.addWidget(new_title_lbl)
- form0 = QtWidgets.QFormLayout()
- new_vlay.addLayout(form0)
- title_lbl = QtWidgets.QLabel('%s:' % _("Title"))
- self.title_entry = FCEntry()
- form0.addRow(title_lbl, self.title_entry)
- link_lbl = QtWidgets.QLabel('%s:' % _("Web Link"))
- self.link_entry = FCEntry()
- self.link_entry.set_value('http://')
- form0.addRow(link_lbl, self.link_entry)
- # Buttons Layout
- button_hlay = QtWidgets.QHBoxLayout()
- layout.addLayout(button_hlay)
- add_entry_btn = FCButton(_("Add Entry"))
- remove_entry_btn = FCButton(_("Remove Entry"))
- export_list_btn = FCButton(_("Export List"))
- import_list_btn = FCButton(_("Import List"))
- # closebtn = QtWidgets.QPushButton(_("Close"))
- # button_hlay.addStretch()
- button_hlay.addWidget(add_entry_btn)
- button_hlay.addWidget(remove_entry_btn)
- button_hlay.addWidget(export_list_btn)
- button_hlay.addWidget(import_list_btn)
- # button_hlay.addWidget(closebtn)
- # ##############################################################################
- # ######################## SIGNALS #############################################
- # ##############################################################################
- add_entry_btn.clicked.connect(self.on_add_entry)
- remove_entry_btn.clicked.connect(self.on_remove_entry)
- export_list_btn.clicked.connect(self.on_export_bookmarks)
- import_list_btn.clicked.connect(self.on_import_bookmarks)
- self.title_entry.returnPressed.connect(self.on_add_entry)
- self.link_entry.returnPressed.connect(self.on_add_entry)
- # closebtn.clicked.connect(self.accept)
- self.table_widget.drag_drop_sig.connect(self.mark_table_rows_for_actions)
- self.build_bm_ui()
- def build_bm_ui(self):
- self.table_widget.setRowCount(len(self.bm_dict))
- nr_crt = 0
- sorted_bookmarks = sorted(list(self.bm_dict.items()), key=lambda x: int(x[0]))
- for entry, bookmark in sorted_bookmarks:
- row = nr_crt
- nr_crt += 1
- title = bookmark[0]
- weblink = bookmark[1]
- id_item = QtWidgets.QTableWidgetItem('%d' % int(nr_crt))
- # id.setFlags(QtCore.Qt.ItemIsSelectable | QtCore.Qt.ItemIsEnabled)
- self.table_widget.setItem(row, 0, id_item) # Tool name/id
- title_item = QtWidgets.QTableWidgetItem(title)
- self.table_widget.setItem(row, 1, title_item)
- weblink_txt = QtWidgets.QTextBrowser()
- weblink_txt.setOpenExternalLinks(True)
- weblink_txt.setFrameStyle(QtWidgets.QFrame.NoFrame)
- weblink_txt.document().setDefaultStyleSheet("a{ text-decoration: none; }")
- weblink_txt.setHtml('<a href=%s>%s</a>' % (weblink, weblink))
- self.table_widget.setCellWidget(row, 2, weblink_txt)
- vertical_header = self.table_widget.verticalHeader()
- vertical_header.hide()
- horizontal_header = self.table_widget.horizontalHeader()
- horizontal_header.setMinimumSectionSize(10)
- horizontal_header.setDefaultSectionSize(70)
- horizontal_header.setSectionResizeMode(0, QtWidgets.QHeaderView.Fixed)
- horizontal_header.resizeSection(0, 20)
- horizontal_header.setSectionResizeMode(1, QtWidgets.QHeaderView.ResizeToContents)
- horizontal_header.setSectionResizeMode(2, QtWidgets.QHeaderView.Stretch)
- self.mark_table_rows_for_actions()
- self.app.defaults["global_bookmarks"].clear()
- for key, val in self.bm_dict.items():
- self.app.defaults["global_bookmarks"][key] = deepcopy(val)
- def on_add_entry(self, **kwargs):
- """
- Add a entry in the Bookmark Table and in the menu actions
- :return: None
- """
- if 'title' in kwargs:
- title = kwargs['title']
- else:
- title = self.title_entry.get_value()
- if title == '':
- self.app.inform.emit('[ERROR_NOTCL] %s' % _("Title entry is empty."))
- return 'fail'
- if 'link' is kwargs:
- link = kwargs['link']
- else:
- link = self.link_entry.get_value()
- if link == 'http://':
- self.app.inform.emit('[ERROR_NOTCL] %s' % _("Web link entry is empty."))
- return 'fail'
- # if 'http' not in link or 'https' not in link:
- # link = 'http://' + link
- for bookmark in self.bm_dict.values():
- if title == bookmark[0] or link == bookmark[1]:
- self.app.inform.emit('[ERROR_NOTCL] %s' % _("Either the Title or the Weblink already in the table."))
- return 'fail'
- # for some reason if the last char in the weblink is a slash it does not make the link clickable
- # so I remove it
- if link[-1] == '/':
- link = link[:-1]
- # add the new entry to storage
- new_entry = len(self.bm_dict) + 1
- self.bm_dict[str(new_entry)] = [title, link]
- # add the link to the menu but only if it is within the set limit
- bm_limit = int(self.app.defaults["global_bookmarks_limit"])
- if len(self.bm_dict) < bm_limit:
- act = QtWidgets.QAction(parent=self.app.ui.menuhelp_bookmarks)
- act.setText(title)
- act.setIcon(QtGui.QIcon(self.app.resource_location + '/link16.png'))
- act.triggered.connect(lambda: webbrowser.open(link))
- self.app.ui.menuhelp_bookmarks.insertAction(self.app.ui.menuhelp_bookmarks_manager, act)
- self.app.inform.emit('[success] %s' % _("Bookmark added."))
- # add the new entry to the bookmark manager table
- self.build_bm_ui()
- def on_remove_entry(self):
- """
- Remove an Entry in the Bookmark table and from the menu actions
- :return:
- """
- index_list = []
- for model_index in self.table_widget.selectionModel().selectedRows():
- index = QtCore.QPersistentModelIndex(model_index)
- index_list.append(index)
- title_to_remove = self.table_widget.item(model_index.row(), 1).text()
- if title_to_remove == 'FlatCAM' or title_to_remove == 'Backup Site':
- self.app.inform.emit('[WARNING_NOTCL] %s.' % _("This bookmark can not be removed"))
- self.build_bm_ui()
- return
- else:
- for k, bookmark in list(self.bm_dict.items()):
- if title_to_remove == bookmark[0]:
- # remove from the storage
- self.bm_dict.pop(k, None)
- for act in self.app.ui.menuhelp_bookmarks.actions():
- if act.text() == title_to_remove:
- # disconnect the signal
- try:
- act.triggered.disconnect()
- except TypeError:
- pass
- # remove the action from the menu
- self.app.ui.menuhelp_bookmarks.removeAction(act)
- # house keeping: it pays to have keys increased by one
- new_key = 0
- new_dict = {}
- for k, v in self.bm_dict.items():
- # we start with key 1 so we can use the len(self.bm_dict)
- # when adding bookmarks (keys in bm_dict)
- new_key += 1
- new_dict[str(new_key)] = v
- self.bm_dict = deepcopy(new_dict)
- new_dict.clear()
- self.app.inform.emit('[success] %s' % _("Bookmark removed."))
- # for index in index_list:
- # self.table_widget.model().removeRow(index.row())
- self.build_bm_ui()
- def on_export_bookmarks(self):
- self.app.report_usage("on_export_bookmarks")
- self.app.log.debug("on_export_bookmarks()")
- date = str(datetime.today()).rpartition('.')[0]
- date = ''.join(c for c in date if c not in ':-')
- date = date.replace(' ', '_')
- filter__ = "Text File (*.TXT);;All Files (*.*)"
- filename, _f = QtWidgets.QFileDialog.getSaveFileName(caption=_("Export FlatCAM Bookmarks"),
- directory='{l_save}/FlatCAM_{n}_{date}'.format(
- l_save=str(self.app.get_last_save_folder()),
- n=_("Bookmarks"),
- date=date),
- filter=filter__)
- filename = str(filename)
- if filename == "":
- self.app.inform.emit('[WARNING_NOTCL] %s' % _("FlatCAM bookmarks export cancelled."))
- return
- else:
- try:
- f = open(filename, 'w')
- f.close()
- except PermissionError:
- self.app.inform.emit('[WARNING] %s' %
- _("Permission denied, saving not possible.\n"
- "Most likely another app is holding the file open and not accessible."))
- return
- except IOError:
- self.app.log.debug('Creating a new bookmarks file ...')
- f = open(filename, 'w')
- f.close()
- except Exception:
- e = sys.exc_info()[0]
- self.app.log.error("Could not load defaults file.")
- self.app.log.error(str(e))
- self.app.inform.emit('[ERROR_NOTCL] %s' % _("Could not load bookmarks file."))
- return
- # Save Bookmarks to a file
- try:
- with open(filename, "w") as f:
- for title, link in self.bm_dict.items():
- line2write = str(title) + ':' + str(link) + '\n'
- f.write(line2write)
- except Exception:
- self.app.inform.emit('[ERROR_NOTCL] %s' % _("Failed to write bookmarks to file."))
- return
- self.app.inform.emit('[success] %s: %s' % (_("Exported bookmarks to"), filename))
- def on_import_bookmarks(self):
- self.app.log.debug("on_import_bookmarks()")
- filter_ = "Text File (*.txt);;All Files (*.*)"
- filename, _f = QtWidgets.QFileDialog.getOpenFileName(caption=_("Import FlatCAM Bookmarks"), filter=filter_)
- filename = str(filename)
- if filename == "":
- self.app.inform.emit('[WARNING_NOTCL] %s' % _("FlatCAM bookmarks import cancelled."))
- else:
- try:
- with open(filename) as f:
- bookmarks = f.readlines()
- except IOError:
- self.app.log.error("Could not load bookmarks file.")
- self.app.inform.emit('[ERROR_NOTCL] %s' % _("Could not load bookmarks file."))
- return
- for line in bookmarks:
- proc_line = line.replace(' ', '').partition(':')
- self.on_add_entry(title=proc_line[0], link=proc_line[2])
- self.app.inform.emit('[success] %s: %s' % (_("Imported Bookmarks from"), filename))
- def mark_table_rows_for_actions(self):
- for row in range(self.table_widget.rowCount()):
- item_to_paint = self.table_widget.item(row, 0)
- if row < self.app.defaults["global_bookmarks_limit"]:
- item_to_paint.setBackground(QtGui.QColor('gray'))
- # item_to_paint.setForeground(QtGui.QColor('black'))
- else:
- item_to_paint.setBackground(QtGui.QColor('white'))
- # item_to_paint.setForeground(QtGui.QColor('black'))
- def rebuild_actions(self):
- # rebuild the storage to reflect the order of the lines
- self.bm_dict.clear()
- for row in range(self.table_widget.rowCount()):
- title = self.table_widget.item(row, 1).text()
- wlink = self.table_widget.cellWidget(row, 2).toPlainText()
- entry = int(row) + 1
- self.bm_dict.update(
- {
- str(entry): [title, wlink]
- }
- )
- self.app.install_bookmarks(book_dict=self.bm_dict)
- # def accept(self):
- # self.rebuild_actions()
- # super().accept()
- def closeEvent(self, QCloseEvent):
- self.rebuild_actions()
- super().closeEvent(QCloseEvent)
- class ToolsDB(QtWidgets.QWidget):
- mark_tools_rows = QtCore.pyqtSignal()
- def __init__(self, app, callback_on_edited, callback_on_tool_request, parent=None):
- super(ToolsDB, self).__init__(parent)
- self.app = app
- self.decimals = 4
- self.callback_app = callback_on_edited
- self.on_tool_request = callback_on_tool_request
- self.offset_item_options = ["Path", "In", "Out", "Custom"]
- self.type_item_options = ["Iso", "Rough", "Finish"]
- self.tool_type_item_options = ["C1", "C2", "C3", "C4", "B", "V"]
- '''
- dict to hold all the tools in the Tools DB
- format:
- {
- tool_id: {
- 'name': 'new_tool'
- 'tooldia': self.app.defaults["geometry_cnctooldia"]
- 'offset': 'Path'
- 'offset_value': 0.0
- 'type': _('Rough'),
- 'tool_type': 'C1'
- 'data': dict()
- }
- }
- '''
- self.db_tool_dict = {}
- # layouts
- layout = QtWidgets.QVBoxLayout()
- self.setLayout(layout)
- table_hlay = QtWidgets.QHBoxLayout()
- layout.addLayout(table_hlay)
- self.table_widget = FCTable(drag_drop=True)
- self.table_widget.setSelectionBehavior(QtWidgets.QAbstractItemView.SelectRows)
- table_hlay.addWidget(self.table_widget)
- # set the number of columns and the headers tool tips
- self.configure_table()
- # pal = QtGui.QPalette()
- # pal.setColor(QtGui.QPalette.Background, Qt.white)
- # New Bookmark
- new_vlay = QtWidgets.QVBoxLayout()
- layout.addLayout(new_vlay)
- # new_tool_lbl = QtWidgets.QLabel('<b>%s</b>' % _("New Tool"))
- # new_vlay.addWidget(new_tool_lbl, alignment=QtCore.Qt.AlignBottom)
- self.buttons_frame = QtWidgets.QFrame()
- self.buttons_frame.setContentsMargins(0, 0, 0, 0)
- layout.addWidget(self.buttons_frame)
- self.buttons_box = QtWidgets.QHBoxLayout()
- self.buttons_box.setContentsMargins(0, 0, 0, 0)
- self.buttons_frame.setLayout(self.buttons_box)
- self.buttons_frame.show()
- add_entry_btn = FCButton(_("Add Geometry Tool in DB"))
- add_entry_btn.setToolTip(
- _("Add a new tool in the Tools Database.\n"
- "It will be used in the Geometry UI.\n"
- "You can edit it after it is added.")
- )
- self.buttons_box.addWidget(add_entry_btn)
- # add_fct_entry_btn = FCButton(_("Add Paint/NCC Tool in DB"))
- # add_fct_entry_btn.setToolTip(
- # _("Add a new tool in the Tools Database.\n"
- # "It will be used in the Paint/NCC Tools UI.\n"
- # "You can edit it after it is added.")
- # )
- # self.buttons_box.addWidget(add_fct_entry_btn)
- remove_entry_btn = FCButton(_("Delete Tool from DB"))
- remove_entry_btn.setToolTip(
- _("Remove a selection of tools in the Tools Database.")
- )
- self.buttons_box.addWidget(remove_entry_btn)
- export_db_btn = FCButton(_("Export DB"))
- export_db_btn.setToolTip(
- _("Save the Tools Database to a custom text file.")
- )
- self.buttons_box.addWidget(export_db_btn)
- import_db_btn = FCButton(_("Import DB"))
- import_db_btn.setToolTip(
- _("Load the Tools Database information's from a custom text file.")
- )
- self.buttons_box.addWidget(import_db_btn)
- self.add_tool_from_db = FCButton(_("Add Tool from Tools DB"))
- self.add_tool_from_db.setToolTip(
- _("Add a new tool in the Tools Table of the\n"
- "active Geometry object after selecting a tool\n"
- "in the Tools Database.")
- )
- self.add_tool_from_db.hide()
- self.cancel_tool_from_db = FCButton(_("Cancel"))
- self.cancel_tool_from_db.hide()
- hlay = QtWidgets.QHBoxLayout()
- layout.addLayout(hlay)
- hlay.addWidget(self.add_tool_from_db)
- hlay.addWidget(self.cancel_tool_from_db)
- hlay.addStretch()
- # ##############################################################################
- # ######################## SIGNALS #############################################
- # ##############################################################################
- add_entry_btn.clicked.connect(self.on_tool_add)
- remove_entry_btn.clicked.connect(self.on_tool_delete)
- export_db_btn.clicked.connect(self.on_export_tools_db_file)
- import_db_btn.clicked.connect(self.on_import_tools_db_file)
- # closebtn.clicked.connect(self.accept)
- self.add_tool_from_db.clicked.connect(self.on_tool_requested_from_app)
- self.cancel_tool_from_db.clicked.connect(self.on_cancel_tool)
- self.setup_db_ui()
- def configure_table(self):
- self.table_widget.setColumnCount(27)
- # self.table_widget.setColumnWidth(0, 20)
- self.table_widget.setHorizontalHeaderLabels(
- [
- '#',
- _("Tool Name"),
- _("Tool Dia"),
- _("Tool Offset"),
- _("Custom Offset"),
- _("Tool Type"),
- _("Tool Shape"),
- _("Cut Z"),
- _("MultiDepth"),
- _("DPP"),
- _("V-Dia"),
- _("V-Angle"),
- _("Travel Z"),
- _("FR"),
- _("FR Z"),
- _("FR Rapids"),
- _("Spindle Speed"),
- _("Dwell"),
- _("Dwelltime"),
- _("Preprocessor"),
- _("ExtraCut"),
- _("E-Cut Length"),
- _("Toolchange"),
- _("Toolchange XY"),
- _("Toolchange Z"),
- _("Start Z"),
- _("End Z"),
- ]
- )
- self.table_widget.horizontalHeaderItem(0).setToolTip(
- _("Tool Index."))
- self.table_widget.horizontalHeaderItem(1).setToolTip(
- _("Tool name.\n"
- "This is not used in the app, it's function\n"
- "is to serve as a note for the user."))
- self.table_widget.horizontalHeaderItem(2).setToolTip(
- _("Tool Diameter."))
- self.table_widget.horizontalHeaderItem(3).setToolTip(
- _("Tool Offset.\n"
- "Can be of a few types:\n"
- "Path = zero offset\n"
- "In = offset inside by half of tool diameter\n"
- "Out = offset outside by half of tool diameter\n"
- "Custom = custom offset using the Custom Offset value"))
- self.table_widget.horizontalHeaderItem(4).setToolTip(
- _("Custom Offset.\n"
- "A value to be used as offset from the current path."))
- self.table_widget.horizontalHeaderItem(5).setToolTip(
- _("Tool Type.\n"
- "Can be:\n"
- "Iso = isolation cut\n"
- "Rough = rough cut, low feedrate, multiple passes\n"
- "Finish = finishing cut, high feedrate"))
- self.table_widget.horizontalHeaderItem(6).setToolTip(
- _("Tool Shape. \n"
- "Can be:\n"
- "C1 ... C4 = circular tool with x flutes\n"
- "B = ball tip milling tool\n"
- "V = v-shape milling tool"))
- self.table_widget.horizontalHeaderItem(7).setToolTip(
- _("Cutting Depth.\n"
- "The depth at which to cut into material."))
- self.table_widget.horizontalHeaderItem(8).setToolTip(
- _("Multi Depth.\n"
- "Selecting this will allow cutting in multiple passes,\n"
- "each pass adding a DPP parameter depth."))
- self.table_widget.horizontalHeaderItem(9).setToolTip(
- _("DPP. Depth per Pass.\n"
- "The value used to cut into material on each pass."))
- self.table_widget.horizontalHeaderItem(10).setToolTip(
- _("V-Dia.\n"
- "Diameter of the tip for V-Shape Tools."))
- self.table_widget.horizontalHeaderItem(11).setToolTip(
- _("V-Agle.\n"
- "Angle at the tip for the V-Shape Tools."))
- self.table_widget.horizontalHeaderItem(12).setToolTip(
- _("Clearance Height.\n"
- "Height at which the milling bit will travel between cuts,\n"
- "above the surface of the material, avoiding all fixtures."))
- self.table_widget.horizontalHeaderItem(13).setToolTip(
- _("FR. Feedrate\n"
- "The speed on XY plane used while cutting into material."))
- self.table_widget.horizontalHeaderItem(14).setToolTip(
- _("FR Z. Feedrate Z\n"
- "The speed on Z plane."))
- self.table_widget.horizontalHeaderItem(15).setToolTip(
- _("FR Rapids. Feedrate Rapids\n"
- "Speed used while moving as fast as possible.\n"
- "This is used only by some devices that can't use\n"
- "the G0 g-code command. Mostly 3D printers."))
- self.table_widget.horizontalHeaderItem(16).setToolTip(
- _("Spindle Speed.\n"
- "If it's left empty it will not be used.\n"
- "The speed of the spindle in RPM."))
- self.table_widget.horizontalHeaderItem(17).setToolTip(
- _("Dwell.\n"
- "Check this if a delay is needed to allow\n"
- "the spindle motor to reach it's set speed."))
- self.table_widget.horizontalHeaderItem(18).setToolTip(
- _("Dwell Time.\n"
- "A delay used to allow the motor spindle reach it's set speed."))
- self.table_widget.horizontalHeaderItem(19).setToolTip(
- _("Preprocessor.\n"
- "A selection of files that will alter the generated G-code\n"
- "to fit for a number of use cases."))
- self.table_widget.horizontalHeaderItem(20).setToolTip(
- _("Extra Cut.\n"
- "If checked, after a isolation is finished an extra cut\n"
- "will be added where the start and end of isolation meet\n"
- "such as that this point is covered by this extra cut to\n"
- "ensure a complete isolation."))
- self.table_widget.horizontalHeaderItem(21).setToolTip(
- _("Extra Cut length.\n"
- "If checked, after a isolation is finished an extra cut\n"
- "will be added where the start and end of isolation meet\n"
- "such as that this point is covered by this extra cut to\n"
- "ensure a complete isolation. This is the length of\n"
- "the extra cut."))
- self.table_widget.horizontalHeaderItem(22).setToolTip(
- _("Toolchange.\n"
- "It will create a toolchange event.\n"
- "The kind of toolchange is determined by\n"
- "the preprocessor file."))
- self.table_widget.horizontalHeaderItem(23).setToolTip(
- _("Toolchange XY.\n"
- "A set of coordinates in the format (x, y).\n"
- "Will determine the cartesian position of the point\n"
- "where the tool change event take place."))
- self.table_widget.horizontalHeaderItem(24).setToolTip(
- _("Toolchange Z.\n"
- "The position on Z plane where the tool change event take place."))
- self.table_widget.horizontalHeaderItem(25).setToolTip(
- _("Start Z.\n"
- "If it's left empty it will not be used.\n"
- "A position on Z plane to move immediately after job start."))
- self.table_widget.horizontalHeaderItem(26).setToolTip(
- _("End Z.\n"
- "A position on Z plane to move immediately after job stop."))
- def setup_db_ui(self):
- filename = self.app.data_path + '/geo_tools_db.FlatDB'
- # load the database tools from the file
- try:
- with open(filename) as f:
- tools = f.read()
- except IOError:
- self.app.log.error("Could not load tools DB file.")
- self.app.inform.emit('[ERROR] %s' % _("Could not load Tools DB file."))
- return
- try:
- self.db_tool_dict = json.loads(tools)
- except Exception:
- e = sys.exc_info()[0]
- self.app.log.error(str(e))
- self.app.inform.emit('[ERROR] %s' % _("Failed to parse Tools DB file."))
- return
- self.app.inform.emit('[success] %s: %s' % (_("Loaded FlatCAM Tools DB from"), filename))
- self.build_db_ui()
- self.table_widget.setupContextMenu()
- self.table_widget.addContextMenu(
- _("Add to DB"), self.on_tool_add, icon=QtGui.QIcon(self.app.resource_location + "/plus16.png"))
- self.table_widget.addContextMenu(
- _("Copy from DB"), self.on_tool_copy, icon=QtGui.QIcon(self.app.resource_location + "/copy16.png"))
- self.table_widget.addContextMenu(
- _("Delete from DB"), self.on_tool_delete, icon=QtGui.QIcon(self.app.resource_location + "/delete32.png"))
- def build_db_ui(self):
- self.ui_disconnect()
- self.table_widget.setRowCount(len(self.db_tool_dict))
- nr_crt = 0
- for toolid, dict_val in self.db_tool_dict.items():
- row = nr_crt
- nr_crt += 1
- t_name = dict_val['name']
- try:
- self.add_tool_table_line(row, name=t_name, widget=self.table_widget, tooldict=dict_val)
- except Exception as e:
- self.app.log.debug("ToolDB.build_db_ui.add_tool_table_line() --> %s" % str(e))
- vertical_header = self.table_widget.verticalHeader()
- vertical_header.hide()
- horizontal_header = self.table_widget.horizontalHeader()
- horizontal_header.setMinimumSectionSize(10)
- horizontal_header.setDefaultSectionSize(70)
- self.table_widget.setSizeAdjustPolicy(QtWidgets.QAbstractScrollArea.AdjustToContents)
- for x in range(27):
- self.table_widget.resizeColumnToContents(x)
- horizontal_header.setSectionResizeMode(0, QtWidgets.QHeaderView.Fixed)
- # horizontal_header.setSectionResizeMode(1, QtWidgets.QHeaderView.Stretch)
- # horizontal_header.setSectionResizeMode(13, QtWidgets.QHeaderView.Fixed)
- horizontal_header.resizeSection(0, 20)
- # horizontal_header.setSectionResizeMode(1, QtWidgets.QHeaderView.ResizeToContents)
- # horizontal_header.setSectionResizeMode(2, QtWidgets.QHeaderView.Stretch)
- self.ui_connect()
- def add_tool_table_line(self, row, name, widget, tooldict):
- data = tooldict['data']
- nr_crt = row + 1
- id_item = QtWidgets.QTableWidgetItem('%d' % int(nr_crt))
- # id_item.setFlags(QtCore.Qt.ItemIsSelectable | QtCore.Qt.ItemIsEnabled)
- flags = id_item.flags() & ~QtCore.Qt.ItemIsEditable
- id_item.setFlags(flags)
- widget.setItem(row, 0, id_item) # Tool name/id
- tool_name_item = QtWidgets.QTableWidgetItem(name)
- widget.setItem(row, 1, tool_name_item)
- dia_item = FCDoubleSpinner()
- dia_item.set_precision(self.decimals)
- dia_item.setSingleStep(0.1)
- dia_item.set_range(0.0, 9999.9999)
- dia_item.set_value(float(tooldict['tooldia']))
- widget.setCellWidget(row, 2, dia_item)
- tool_offset_item = FCComboBox()
- for item in self.offset_item_options:
- tool_offset_item.addItem(item)
- tool_offset_item.set_value(tooldict['offset'])
- widget.setCellWidget(row, 3, tool_offset_item)
- c_offset_item = FCDoubleSpinner()
- c_offset_item.set_precision(self.decimals)
- c_offset_item.setSingleStep(0.1)
- c_offset_item.set_range(-9999.9999, 9999.9999)
- c_offset_item.set_value(float(tooldict['offset_value']))
- widget.setCellWidget(row, 4, c_offset_item)
- tt_item = FCComboBox()
- for item in self.type_item_options:
- tt_item.addItem(item)
- tt_item.set_value(tooldict['type'])
- widget.setCellWidget(row, 5, tt_item)
- tshape_item = FCComboBox()
- for item in self.tool_type_item_options:
- tshape_item.addItem(item)
- tshape_item.set_value(tooldict['tool_type'])
- widget.setCellWidget(row, 6, tshape_item)
- cutz_item = FCDoubleSpinner()
- cutz_item.set_precision(self.decimals)
- cutz_item.setSingleStep(0.1)
- if self.app.defaults['global_machinist_setting']:
- cutz_item.set_range(-9999.9999, 9999.9999)
- else:
- cutz_item.set_range(-9999.9999, -0.0000)
- cutz_item.set_value(float(data['cutz']))
- widget.setCellWidget(row, 7, cutz_item)
- multidepth_item = FCCheckBox()
- multidepth_item.set_value(data['multidepth'])
- widget.setCellWidget(row, 8, multidepth_item)
- # to make the checkbox centered but it can no longer have it's value accessed - needs a fix using findchild()
- # multidepth_item = QtWidgets.QWidget()
- # cb = FCCheckBox()
- # cb.set_value(data['multidepth'])
- # qhboxlayout = QtWidgets.QHBoxLayout(multidepth_item)
- # qhboxlayout.addWidget(cb)
- # qhboxlayout.setAlignment(QtCore.Qt.AlignCenter)
- # qhboxlayout.setContentsMargins(0, 0, 0, 0)
- # widget.setCellWidget(row, 8, multidepth_item)
- depth_per_pass_item = FCDoubleSpinner()
- depth_per_pass_item.set_precision(self.decimals)
- depth_per_pass_item.setSingleStep(0.1)
- depth_per_pass_item.set_range(0.0, 9999.9999)
- depth_per_pass_item.set_value(float(data['depthperpass']))
- widget.setCellWidget(row, 9, depth_per_pass_item)
- vtip_dia_item = FCDoubleSpinner()
- vtip_dia_item.set_precision(self.decimals)
- vtip_dia_item.setSingleStep(0.1)
- vtip_dia_item.set_range(0.0, 9999.9999)
- vtip_dia_item.set_value(float(data['vtipdia']))
- widget.setCellWidget(row, 10, vtip_dia_item)
- vtip_angle_item = FCDoubleSpinner()
- vtip_angle_item.set_precision(self.decimals)
- vtip_angle_item.setSingleStep(0.1)
- vtip_angle_item.set_range(-360.0, 360.0)
- vtip_angle_item.set_value(float(data['vtipangle']))
- widget.setCellWidget(row, 11, vtip_angle_item)
- travelz_item = FCDoubleSpinner()
- travelz_item.set_precision(self.decimals)
- travelz_item.setSingleStep(0.1)
- if self.app.defaults['global_machinist_setting']:
- travelz_item.set_range(-9999.9999, 9999.9999)
- else:
- travelz_item.set_range(0.0000, 9999.9999)
- travelz_item.set_value(float(data['travelz']))
- widget.setCellWidget(row, 12, travelz_item)
- fr_item = FCDoubleSpinner()
- fr_item.set_precision(self.decimals)
- fr_item.set_range(0.0, 9999.9999)
- fr_item.set_value(float(data['feedrate']))
- widget.setCellWidget(row, 13, fr_item)
- frz_item = FCDoubleSpinner()
- frz_item.set_precision(self.decimals)
- frz_item.set_range(0.0, 9999.9999)
- frz_item.set_value(float(data['feedrate_z']))
- widget.setCellWidget(row, 14, frz_item)
- frrapids_item = FCDoubleSpinner()
- frrapids_item.set_precision(self.decimals)
- frrapids_item.set_range(0.0, 9999.9999)
- frrapids_item.set_value(float(data['feedrate_rapid']))
- widget.setCellWidget(row, 15, frrapids_item)
- spindlespeed_item = FCSpinner()
- spindlespeed_item.set_range(0, 1000000)
- spindlespeed_item.set_value(int(data['spindlespeed']))
- spindlespeed_item.set_step(100)
- widget.setCellWidget(row, 16, spindlespeed_item)
- dwell_item = FCCheckBox()
- dwell_item.set_value(data['dwell'])
- widget.setCellWidget(row, 17, dwell_item)
- dwelltime_item = FCDoubleSpinner()
- dwelltime_item.set_precision(self.decimals)
- dwelltime_item.set_range(0.0000, 9999.9999)
- dwelltime_item.set_value(float(data['dwelltime']))
- widget.setCellWidget(row, 18, dwelltime_item)
- pp_item = FCComboBox()
- for item in self.app.preprocessors:
- pp_item.addItem(item)
- pp_item.set_value(data['ppname_g'])
- widget.setCellWidget(row, 19, pp_item)
- ecut_item = FCCheckBox()
- ecut_item.set_value(data['extracut'])
- widget.setCellWidget(row, 20, ecut_item)
- ecut_length_item = FCDoubleSpinner()
- ecut_length_item.set_precision(self.decimals)
- ecut_length_item.set_range(0.0000, 9999.9999)
- ecut_length_item.set_value(data['extracut_length'])
- widget.setCellWidget(row, 21, ecut_length_item)
- toolchange_item = FCCheckBox()
- toolchange_item.set_value(data['toolchange'])
- widget.setCellWidget(row, 22, toolchange_item)
- toolchangexy_item = QtWidgets.QTableWidgetItem(str(data['toolchangexy']) if data['toolchangexy'] else '')
- widget.setItem(row, 23, toolchangexy_item)
- toolchangez_item = FCDoubleSpinner()
- toolchangez_item.set_precision(self.decimals)
- toolchangez_item.setSingleStep(0.1)
- if self.app.defaults['global_machinist_setting']:
- toolchangez_item.set_range(-9999.9999, 9999.9999)
- else:
- toolchangez_item.set_range(0.0000, 9999.9999)
- toolchangez_item.set_value(float(data['toolchangez']))
- widget.setCellWidget(row, 24, toolchangez_item)
- startz_item = QtWidgets.QTableWidgetItem(str(data['startz']) if data['startz'] else '')
- widget.setItem(row, 25, startz_item)
- endz_item = FCDoubleSpinner()
- endz_item.set_precision(self.decimals)
- endz_item.setSingleStep(0.1)
- if self.app.defaults['global_machinist_setting']:
- endz_item.set_range(-9999.9999, 9999.9999)
- else:
- endz_item.set_range(0.0000, 9999.9999)
- endz_item.set_value(float(data['endz']))
- widget.setCellWidget(row, 26, endz_item)
- def on_tool_add(self):
- """
- Add a tool in the DB Tool Table
- :return: None
- """
- default_data = {}
- default_data.update({
- "cutz": float(self.app.defaults["geometry_cutz"]),
- "multidepth": self.app.defaults["geometry_multidepth"],
- "depthperpass": float(self.app.defaults["geometry_depthperpass"]),
- "vtipdia": float(self.app.defaults["geometry_vtipdia"]),
- "vtipangle": float(self.app.defaults["geometry_vtipangle"]),
- "travelz": float(self.app.defaults["geometry_travelz"]),
- "feedrate": float(self.app.defaults["geometry_feedrate"]),
- "feedrate_z": float(self.app.defaults["geometry_feedrate_z"]),
- "feedrate_rapid": float(self.app.defaults["geometry_feedrate_rapid"]),
- "spindlespeed": self.app.defaults["geometry_spindlespeed"],
- "dwell": self.app.defaults["geometry_dwell"],
- "dwelltime": float(self.app.defaults["geometry_dwelltime"]),
- "ppname_g": self.app.defaults["geometry_ppname_g"],
- "extracut": self.app.defaults["geometry_extracut"],
- "extracut_length": float(self.app.defaults["geometry_extracut_length"]),
- "toolchange": self.app.defaults["geometry_toolchange"],
- "toolchangexy": self.app.defaults["geometry_toolchangexy"],
- "toolchangez": float(self.app.defaults["geometry_toolchangez"]),
- "startz": self.app.defaults["geometry_startz"],
- "endz": float(self.app.defaults["geometry_endz"])
- })
- dict_elem = {}
- dict_elem['name'] = 'new_tool'
- if type(self.app.defaults["geometry_cnctooldia"]) == float:
- dict_elem['tooldia'] = self.app.defaults["geometry_cnctooldia"]
- else:
- try:
- tools_string = self.app.defaults["geometry_cnctooldia"].split(",")
- tools_diameters = [eval(a) for a in tools_string if a != '']
- dict_elem['tooldia'] = tools_diameters[0] if tools_diameters else 0.0
- except Exception as e:
- self.app.log.debug("ToolDB.on_tool_add() --> %s" % str(e))
- return
- dict_elem['offset'] = 'Path'
- dict_elem['offset_value'] = 0.0
- dict_elem['type'] = 'Rough'
- dict_elem['tool_type'] = 'C1'
- dict_elem['data'] = default_data
- new_toolid = len(self.db_tool_dict) + 1
- self.db_tool_dict[new_toolid] = deepcopy(dict_elem)
- # add the new entry to the Tools DB table
- self.build_db_ui()
- self.callback_on_edited()
- self.app.inform.emit('[success] %s' % _("Tool added to DB."))
- def on_tool_copy(self):
- """
- Copy a selection of Tools in the Tools DB table
- :return:
- """
- new_tool_id = self.table_widget.rowCount() + 1
- for model_index in self.table_widget.selectionModel().selectedRows():
- # index = QtCore.QPersistentModelIndex(model_index)
- old_tool_id = self.table_widget.item(model_index.row(), 0).text()
- new_tool_id += 1
- for toolid, dict_val in list(self.db_tool_dict.items()):
- if int(old_tool_id) == int(toolid):
- self.db_tool_dict.update({
- new_tool_id: deepcopy(dict_val)
- })
- self.build_db_ui()
- self.callback_on_edited()
- self.app.inform.emit('[success] %s' % _("Tool copied from Tools DB."))
- def on_tool_delete(self):
- """
- Delete a selection of Tools in the Tools DB table
- :return:
- """
- for model_index in self.table_widget.selectionModel().selectedRows():
- # index = QtCore.QPersistentModelIndex(model_index)
- toolname_to_remove = self.table_widget.item(model_index.row(), 0).text()
- for toolid, dict_val in list(self.db_tool_dict.items()):
- if int(toolname_to_remove) == int(toolid):
- # remove from the storage
- self.db_tool_dict.pop(toolid, None)
- self.build_db_ui()
- self.callback_on_edited()
- self.app.inform.emit('[success] %s' % _("Tool removed from Tools DB."))
- def on_export_tools_db_file(self):
- self.app.report_usage("on_export_tools_db_file")
- self.app.log.debug("on_export_tools_db_file()")
- date = str(datetime.today()).rpartition('.')[0]
- date = ''.join(c for c in date if c not in ':-')
- date = date.replace(' ', '_')
- filter__ = "Text File (*.TXT);;All Files (*.*)"
- filename, _f = QtWidgets.QFileDialog.getSaveFileName(caption=_("Export Tools Database"),
- directory='{l_save}/FlatCAM_{n}_{date}'.format(
- l_save=str(self.app.get_last_save_folder()),
- n=_("Tools_Database"),
- date=date),
- filter=filter__)
- filename = str(filename)
- if filename == "":
- self.app.inform.emit('[WARNING_NOTCL] %s' % _("FlatCAM Tools DB export cancelled."))
- return
- else:
- try:
- f = open(filename, 'w')
- f.close()
- except PermissionError:
- self.app.inform.emit('[WARNING] %s' %
- _("Permission denied, saving not possible.\n"
- "Most likely another app is holding the file open and not accessible."))
- return
- except IOError:
- self.app.log.debug('Creating a new Tools DB file ...')
- f = open(filename, 'w')
- f.close()
- except Exception:
- e = sys.exc_info()[0]
- self.app.log.error("Could not load Tools DB file.")
- self.app.log.error(str(e))
- self.app.inform.emit('[ERROR_NOTCL] %s' % _("Could not load Tools DB file."))
- return
- # Save update options
- try:
- # Save Tools DB in a file
- try:
- with open(filename, "w") as f:
- json.dump(self.db_tool_dict, f, default=to_dict, indent=2)
- except Exception as e:
- self.app.log.debug("App.on_save_tools_db() --> %s" % str(e))
- self.inform.emit('[ERROR_NOTCL] %s' % _("Failed to write Tools DB to file."))
- return
- except Exception:
- self.app.inform.emit('[ERROR_NOTCL] %s' % _("Failed to write Tools DB to file."))
- return
- self.app.inform.emit('[success] %s: %s' % (_("Exported Tools DB to"), filename))
- def on_import_tools_db_file(self):
- self.app.report_usage("on_import_tools_db_file")
- self.app.log.debug("on_import_tools_db_file()")
- filter__ = "Text File (*.TXT);;All Files (*.*)"
- filename, _f = QtWidgets.QFileDialog.getOpenFileName(caption=_("Import FlatCAM Tools DB"), filter=filter__)
- if filename == "":
- self.app.inform.emit('[WARNING_NOTCL] %s' % _("FlatCAM Tools DB import cancelled."))
- else:
- try:
- with open(filename) as f:
- tools_in_db = f.read()
- except IOError:
- self.app.log.error("Could not load Tools DB file.")
- self.app.inform.emit('[ERROR_NOTCL] %s' % _("Could not load Tools DB file."))
- return
- try:
- self.db_tool_dict = json.loads(tools_in_db)
- except Exception:
- e = sys.exc_info()[0]
- self.app.log.error(str(e))
- self.app.inform.emit('[ERROR] %s' % _("Failed to parse Tools DB file."))
- return
- self.app.inform.emit('[success] %s: %s' % (_("Loaded FlatCAM Tools DB from"), filename))
- self.build_db_ui()
- self.callback_on_edited()
- def on_save_tools_db(self, silent=False):
- self.app.log.debug("ToolsDB.on_save_button() --> Saving Tools Database to file.")
- filename = self.app.data_path + "/geo_tools_db.FlatDB"
- # Preferences save, update the color of the Tools DB Tab text
- for idx in range(self.app.ui.plot_tab_area.count()):
- if self.app.ui.plot_tab_area.tabText(idx) == _("Tools Database"):
- self.app.ui.plot_tab_area.tabBar.setTabTextColor(idx, QtGui.QColor('black'))
- # Save Tools DB in a file
- try:
- f = open(filename, "w")
- json.dump(self.db_tool_dict, f, default=to_dict, indent=2)
- f.close()
- except Exception as e:
- self.app.log.debug("ToolsDB.on_save_tools_db() --> %s" % str(e))
- self.app.inform.emit('[ERROR_NOTCL] %s' % _("Failed to write Tools DB to file."))
- return
- if not silent:
- self.app.inform.emit('[success] %s' % _("Saved Tools DB."))
- def ui_connect(self):
- try:
- try:
- self.table_widget.itemChanged.disconnect(self.callback_on_edited)
- except (TypeError, AttributeError):
- pass
- self.table_widget.itemChanged.connect(self.callback_on_edited)
- except AttributeError:
- pass
- for row in range(self.table_widget.rowCount()):
- for col in range(self.table_widget.columnCount()):
- # ComboBox
- try:
- try:
- self.table_widget.cellWidget(row, col).currentIndexChanged.disconnect(self.callback_on_edited)
- except (TypeError, AttributeError):
- pass
- self.table_widget.cellWidget(row, col).currentIndexChanged.connect(self.callback_on_edited)
- except AttributeError:
- pass
- # CheckBox
- try:
- try:
- self.table_widget.cellWidget(row, col).toggled.disconnect(self.callback_on_edited)
- except (TypeError, AttributeError):
- pass
- self.table_widget.cellWidget(row, col).toggled.connect(self.callback_on_edited)
- except AttributeError:
- pass
- # SpinBox, DoubleSpinBox
- try:
- try:
- self.table_widget.cellWidget(row, col).valueChanged.disconnect(self.callback_on_edited)
- except (TypeError, AttributeError):
- pass
- self.table_widget.cellWidget(row, col).valueChanged.connect(self.callback_on_edited)
- except AttributeError:
- pass
- def ui_disconnect(self):
- try:
- self.table_widget.itemChanged.disconnect(self.callback_on_edited)
- except (TypeError, AttributeError):
- pass
- for row in range(self.table_widget.rowCount()):
- for col in range(self.table_widget.columnCount()):
- # ComboBox
- try:
- self.table_widget.cellWidget(row, col).currentIndexChanged.disconnect(self.callback_on_edited)
- except (TypeError, AttributeError):
- pass
- # CheckBox
- try:
- self.table_widget.cellWidget(row, col).toggled.disconnect(self.callback_on_edited)
- except (TypeError, AttributeError):
- pass
- # SpinBox, DoubleSpinBox
- try:
- self.table_widget.cellWidget(row, col).valueChanged.disconnect(self.callback_on_edited)
- except (TypeError, AttributeError):
- pass
- def callback_on_edited(self):
- # update the dictionary storage self.db_tool_dict
- self.db_tool_dict.clear()
- dict_elem = {}
- default_data = {}
- for row in range(self.table_widget.rowCount()):
- new_toolid = row + 1
- for col in range(self.table_widget.columnCount()):
- column_header_text = self.table_widget.horizontalHeaderItem(col).text()
- if column_header_text == _('Tool Name'):
- dict_elem['name'] = self.table_widget.item(row, col).text()
- elif column_header_text == _('Tool Dia'):
- dict_elem['tooldia'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('Tool Offset'):
- dict_elem['offset'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('Custom Offset'):
- dict_elem['offset_value'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('Tool Type'):
- dict_elem['type'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('Tool Shape'):
- dict_elem['tool_type'] = self.table_widget.cellWidget(row, col).get_value()
- else:
- if column_header_text == _('Cut Z'):
- default_data['cutz'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('MultiDepth'):
- default_data['multidepth'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('DPP'):
- default_data['depthperpass'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('V-Dia'):
- default_data['vtipdia'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('V-Angle'):
- default_data['vtipangle'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('Travel Z'):
- default_data['travelz'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('FR'):
- default_data['feedrate'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('FR Z'):
- default_data['feedrate_z'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('FR Rapids'):
- default_data['feedrate_rapid'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('Spindle Speed'):
- default_data['spindlespeed'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('Dwell'):
- default_data['dwell'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('Dwelltime'):
- default_data['dwelltime'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('Preprocessor'):
- default_data['ppname_g'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('ExtraCut'):
- default_data['extracut'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _("E-Cut Length"):
- default_data['extracut_length'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('Toolchange'):
- default_data['toolchange'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('Toolchange XY'):
- default_data['toolchangexy'] = self.table_widget.item(row, col).text()
- elif column_header_text == _('Toolchange Z'):
- default_data['toolchangez'] = self.table_widget.cellWidget(row, col).get_value()
- elif column_header_text == _('Start Z'):
- default_data['startz'] = float(self.table_widget.item(row, col).text()) \
- if self.table_widget.item(row, col).text() is not '' else None
- elif column_header_text == _('End Z'):
- default_data['endz'] = self.table_widget.cellWidget(row, col).get_value()
- dict_elem['data'] = default_data
- self.db_tool_dict.update(
- {
- new_toolid: deepcopy(dict_elem)
- }
- )
- self.callback_app()
- def on_tool_requested_from_app(self):
- if not self.table_widget.selectionModel().selectedRows():
- self.app.inform.emit('[WARNING_NOTCL] %s...' % _("No Tool/row selected in the Tools Database table"))
- return
- model_index_list = self.table_widget.selectionModel().selectedRows()
- for model_index in model_index_list:
- selected_row = model_index.row()
- tool_uid = selected_row + 1
- for key in self.db_tool_dict.keys():
- if str(key) == str(tool_uid):
- selected_tool = self.db_tool_dict[key]
- self.on_tool_request(tool=selected_tool)
- def on_cancel_tool(self):
- for idx in range(self.app.ui.plot_tab_area.count()):
- if self.app.ui.plot_tab_area.tabText(idx) == _("Tools Database"):
- wdg = self.app.ui.plot_tab_area.widget(idx)
- wdg.deleteLater()
- self.app.ui.plot_tab_area.removeTab(idx)
- self.app.inform.emit('%s' % _("Cancelled adding tool from DB."))
- def resize_new_tool_table_widget(self, min_size, max_size):
- """
- Resize the table widget responsible for adding new tool in the Tool Database
- :param min_size: passed by rangeChanged signal or the self.new_tool_table_widget.horizontalScrollBar()
- :param max_size: passed by rangeChanged signal or the self.new_tool_table_widget.horizontalScrollBar()
- :return:
- """
- t_height = self.t_height
- if max_size > min_size:
- t_height = self.t_height + self.new_tool_table_widget.verticalScrollBar().height()
- self.new_tool_table_widget.setMaximumHeight(t_height)
- def closeEvent(self, QCloseEvent):
- super().closeEvent(QCloseEvent)
- class ToolsDB2(QtWidgets.QWidget):
- mark_tools_rows = QtCore.pyqtSignal()
- def __init__(self, app, callback_on_edited, callback_on_tool_request, parent=None):
- super(ToolsDB2, self).__init__(parent)
- self.app = app
- self.decimals = self.app.decimals
- self.callback_app = callback_on_edited
- self.on_tool_request = callback_on_tool_request
- self.offset_item_options = ["Path", "In", "Out", "Custom"]
- self.type_item_options = ["Iso", "Rough", "Finish"]
- self.tool_type_item_options = ["C1", "C2", "C3", "C4", "B", "V"]
- '''
- dict to hold all the tools in the Tools DB
- format:
- {
- tool_id: {
- 'name': 'new_tool'
- 'tooldia': self.app.defaults["geometry_cnctooldia"]
- 'offset': 'Path'
- 'offset_value': 0.0
- 'type': _('Rough'),
- 'tool_type': 'C1'
- 'data': dict()
- }
- }
- '''
- self.db_tool_dict = {}
- # layouts
- grid_layout = QtWidgets.QGridLayout()
- grid_layout.setColumnStretch(0, 0)
- grid_layout.setColumnStretch(1, 1)
- self.setLayout(grid_layout)
- tree_layout = QtWidgets.QVBoxLayout()
- grid_layout.addLayout(tree_layout, 0, 0)
- self.tree_widget = FCTree(columns=2, header_hidden=False, protected_column=[0])
- self.tree_widget.setHeaderLabels(["ID", "Tool Name"])
- self.tree_widget.setIndentation(0)
- self.tree_widget.setContextMenuPolicy(QtCore.Qt.CustomContextMenu)
- self.tree_widget.setSelectionMode(QtWidgets.QAbstractItemView.ExtendedSelection)
- # set alternating colors
- # self.tree_widget.setAlternatingRowColors(True)
- # p = QtGui.QPalette()
- # p.setColor(QtGui.QPalette.AlternateBase, QtGui.QColor(226, 237, 253) )
- # self.tree_widget.setPalette(p)
- self.tree_widget.setSizePolicy(QtWidgets.QSizePolicy.MinimumExpanding, QtWidgets.QSizePolicy.MinimumExpanding)
- tree_layout.addWidget(self.tree_widget)
- param_hlay = QtWidgets.QHBoxLayout()
- param_area = QtWidgets.QScrollArea()
- param_widget = QtWidgets.QWidget()
- param_widget.setLayout(param_hlay)
- param_area.setWidget(param_widget)
- param_area.setWidgetResizable(True)
- grid_layout.addWidget(param_area, 0, 1)
- # ###########################################################################
- # ############## The UI form ################################################
- # ###########################################################################
- self.basic_box = QtWidgets.QGroupBox()
- self.basic_box.setStyleSheet("""
- QGroupBox
- {
- font-size: 16px;
- font-weight: bold;
- }
- """)
- self.basic_vlay = QtWidgets.QVBoxLayout()
- self.basic_box.setTitle(_("Basic Parameters"))
- self.basic_box.setMinimumWidth(250)
- self.advanced_box = QtWidgets.QGroupBox()
- self.advanced_box.setStyleSheet("""
- QGroupBox
- {
- font-size: 16px;
- font-weight: bold;
- }
- """)
- self.advanced_vlay = QtWidgets.QVBoxLayout()
- self.advanced_box.setTitle(_("Advanced Parameters"))
- self.advanced_box.setMinimumWidth(250)
- self.basic_box.setLayout(self.basic_vlay)
- self.advanced_box.setLayout(self.advanced_vlay)
- param_hlay.addWidget(self.basic_box)
- param_hlay.addWidget(self.advanced_box)
- param_hlay.addStretch()
- # ###########################################################################
- # ############### BASIC UI form #############################################
- # ###########################################################################
- self.grid0 = QtWidgets.QGridLayout()
- self.basic_vlay.addLayout(self.grid0)
- self.grid0.setColumnStretch(0, 0)
- self.grid0.setColumnStretch(1, 1)
- self.basic_vlay.addStretch()
- # Tool Name
- self.name_label = QtWidgets.QLabel('%s:' % _('Tool Name'))
- self.name_label.setToolTip(
- _("Tool name.\n"
- "This is not used in the app, it's function\n"
- "is to serve as a note for the user."))
- self.name_entry = FCEntry()
- self.name_entry.setObjectName('gdb_name')
- self.grid0.addWidget(self.name_label, 0, 0)
- self.grid0.addWidget(self.name_entry, 0, 1)
- # Tool Dia
- self.dia_label = QtWidgets.QLabel('%s:' % _('Tool Dia'))
- self.dia_label.setToolTip(
- _("Tool Diameter."))
- self.dia_entry = FCDoubleSpinner()
- self.dia_entry.set_range(-9999.9999, 9999.9999)
- self.dia_entry.set_precision(self.decimals)
- self.dia_entry.setObjectName('gdb_dia')
- self.grid0.addWidget(self.dia_label, 1, 0)
- self.grid0.addWidget(self.dia_entry, 1, 1)
- # Tool Shape
- self.shape_label = QtWidgets.QLabel('%s:' % _('Tool Shape'))
- self.shape_label.setToolTip(
- _("Tool Shape. \n"
- "Can be:\n"
- "C1 ... C4 = circular tool with x flutes\n"
- "B = ball tip milling tool\n"
- "V = v-shape milling tool"))
- self.shape_combo = FCComboBox()
- self.shape_combo.addItems(["C1", "C2", "C3", "C4", "B", "V"])
- self.shape_combo.setObjectName('gdb_shape')
- self.grid0.addWidget(self.shape_label, 2, 0)
- self.grid0.addWidget(self.shape_combo, 2, 1)
- # Cut Z
- self.cutz_label = QtWidgets.QLabel('%s:' % _("Cut Z"))
- self.cutz_label.setToolTip(
- _("Cutting Depth.\n"
- "The depth at which to cut into material."))
- self.cutz_entry = FCDoubleSpinner()
- self.cutz_entry.set_range(-9999.9999, 9999.9999)
- self.cutz_entry.set_precision(self.decimals)
- self.cutz_entry.setObjectName('gdb_cutz')
- self.grid0.addWidget(self.cutz_label, 4, 0)
- self.grid0.addWidget(self.cutz_entry, 4, 1)
- # Multi Depth
- self.multidepth_label = QtWidgets.QLabel('%s:' % _("MultiDepth"))
- self.multidepth_label.setToolTip(
- _("Multi Depth.\n"
- "Selecting this will allow cutting in multiple passes,\n"
- "each pass adding a DPP parameter depth."))
- self.multidepth_cb = FCCheckBox()
- self.multidepth_cb.setObjectName('gdb_multidepth')
- self.grid0.addWidget(self.multidepth_label, 5, 0)
- self.grid0.addWidget(self.multidepth_cb, 5, 1)
- # Depth Per Pass
- self.dpp_label = QtWidgets.QLabel('%s:' % _("DPP"))
- self.dpp_label.setToolTip(
- _("DPP. Depth per Pass.\n"
- "The value used to cut into material on each pass."))
- self.multidepth_entry = FCDoubleSpinner()
- self.multidepth_entry.set_range(-9999.9999, 9999.9999)
- self.multidepth_entry.set_precision(self.decimals)
- self.multidepth_entry.setObjectName('gdb_multidepth_entry')
- self.grid0.addWidget(self.dpp_label, 7, 0)
- self.grid0.addWidget(self.multidepth_entry, 7, 1)
- # Travel Z
- self.travelz_label = QtWidgets.QLabel('%s:' % _("Travel Z"))
- self.travelz_label.setToolTip(
- _("Clearance Height.\n"
- "Height at which the milling bit will travel between cuts,\n"
- "above the surface of the material, avoiding all fixtures."))
- self.travelz_entry = FCDoubleSpinner()
- self.travelz_entry.set_range(-9999.9999, 9999.9999)
- self.travelz_entry.set_precision(self.decimals)
- self.travelz_entry.setObjectName('gdb_travel')
- self.grid0.addWidget(self.travelz_label, 9, 0)
- self.grid0.addWidget(self.travelz_entry, 9, 1)
- # Feedrate X-Y
- self.frxy_label = QtWidgets.QLabel('%s:' % _("Feedrate X-Y"))
- self.frxy_label.setToolTip(
- _("Feedrate X-Y. Feedrate\n"
- "The speed on XY plane used while cutting into material."))
- self.frxy_entry = FCDoubleSpinner()
- self.frxy_entry.set_range(-9999.9999, 9999.9999)
- self.frxy_entry.set_precision(self.decimals)
- self.frxy_entry.setObjectName('gdb_frxy')
- self.grid0.addWidget(self.frxy_label, 12, 0)
- self.grid0.addWidget(self.frxy_entry, 12, 1)
- # Feedrate Z
- self.frz_label = QtWidgets.QLabel('%s:' % _("Feedrate Z"))
- self.frz_label.setToolTip(
- _("Feedrate Z\n"
- "The speed on Z plane."))
- self.frz_entry = FCDoubleSpinner()
- self.frz_entry.set_range(-9999.9999, 9999.9999)
- self.frz_entry.set_precision(self.decimals)
- self.frz_entry.setObjectName('gdb_frz')
- self.grid0.addWidget(self.frz_label, 14, 0)
- self.grid0.addWidget(self.frz_entry, 14, 1)
- # Spindle Spped
- self.spindle_label = QtWidgets.QLabel('%s:' % _("Spindle Speed"))
- self.spindle_label.setToolTip(
- _("Spindle Speed.\n"
- "If it's left empty it will not be used.\n"
- "The speed of the spindle in RPM."))
- self.spindle_entry = FCDoubleSpinner()
- self.spindle_entry.set_range(-9999.9999, 9999.9999)
- self.spindle_entry.set_precision(self.decimals)
- self.frz_entry.setObjectName('gdb_spindle')
- self.grid0.addWidget(self.spindle_label, 15, 0)
- self.grid0.addWidget(self.spindle_entry, 15, 1)
- # Dwell
- self.dwell_label = QtWidgets.QLabel('%s:' % _("Dwell"))
- self.dwell_label.setToolTip(
- _("Dwell.\n"
- "Check this if a delay is needed to allow\n"
- "the spindle motor to reach it's set speed."))
- self.dwell_cb = FCCheckBox()
- self.dwell_cb.setObjectName('gdb_dwell')
- self.grid0.addWidget(self.dwell_label, 16, 0)
- self.grid0.addWidget(self.dwell_cb, 16, 1)
- # Dwell Time
- self.dwelltime_label = QtWidgets.QLabel('%s:' % _("Dwelltime"))
- self.dwelltime_label.setToolTip(
- _("Dwell Time.\n"
- "A delay used to allow the motor spindle reach it's set speed."))
- self.dwelltime_entry = FCDoubleSpinner()
- self.dwelltime_entry.set_range(0.0000, 9999.9999)
- self.dwelltime_entry.set_precision(self.decimals)
- self.dwelltime_entry.setObjectName('gdb_dwelltime')
- self.grid0.addWidget(self.dwelltime_label, 17, 0)
- self.grid0.addWidget(self.dwelltime_entry, 17, 1)
- # ###########################################################################
- # ############### ADVANCED UI form ##########################################
- # ###########################################################################
- self.grid1 = QtWidgets.QGridLayout()
- self.advanced_vlay.addLayout(self.grid1)
- self.grid1.setColumnStretch(0, 0)
- self.grid1.setColumnStretch(1, 1)
- self.advanced_vlay.addStretch()
- # Tool Type
- self.type_label = QtWidgets.QLabel('%s:' % _("Tool Type"))
- self.type_label.setToolTip(
- _("Tool Type.\n"
- "Can be:\n"
- "Iso = isolation cut\n"
- "Rough = rough cut, low feedrate, multiple passes\n"
- "Finish = finishing cut, high feedrate"))
- self.type_combo = FCComboBox()
- self.type_combo.addItems(["Iso", "Rough", "Finish"])
- self.type_combo.setObjectName('gdb_type')
- self.grid1.addWidget(self.type_label, 0, 0)
- self.grid1.addWidget(self.type_combo, 0, 1)
- # Tool Offset
- self.tooloffset_label = QtWidgets.QLabel('%s:' % _('Tool Offset'))
- self.tooloffset_label.setToolTip(
- _("Tool Offset.\n"
- "Can be of a few types:\n"
- "Path = zero offset\n"
- "In = offset inside by half of tool diameter\n"
- "Out = offset outside by half of tool diameter\n"
- "Custom = custom offset using the Custom Offset value"))
- self.tooloffset_combo = FCComboBox()
- self.tooloffset_combo.addItems(["Path", "In", "Out", "Custom"])
- self.tooloffset_combo.setObjectName('gdb_tool_offset')
- self.grid1.addWidget(self.tooloffset_label, 2, 0)
- self.grid1.addWidget(self.tooloffset_combo, 2, 1)
- # Custom Offset
- self.custom_offset_label = QtWidgets.QLabel('%s:' % _("Custom Offset"))
- self.custom_offset_label.setToolTip(
- _("Custom Offset.\n"
- "A value to be used as offset from the current path."))
- self.custom_offset_entry = FCDoubleSpinner()
- self.custom_offset_entry.set_range(-9999.9999, 9999.9999)
- self.custom_offset_entry.set_precision(self.decimals)
- self.custom_offset_entry.setObjectName('gdb_custom_offset')
- self.grid1.addWidget(self.custom_offset_label, 5, 0)
- self.grid1.addWidget(self.custom_offset_entry, 5, 1)
- # V-Dia
- self.vdia_label = QtWidgets.QLabel('%s:' % _("V-Dia"))
- self.vdia_label.setToolTip(
- _("V-Dia.\n"
- "Diameter of the tip for V-Shape Tools."))
- self.vdia_entry = FCDoubleSpinner()
- self.vdia_entry.set_range(0.0000, 9999.9999)
- self.vdia_entry.set_precision(self.decimals)
- self.vdia_entry.setObjectName('gdb_vdia')
- self.grid1.addWidget(self.vdia_label, 7, 0)
- self.grid1.addWidget(self.vdia_entry, 7, 1)
- # V-Angle
- self.vangle_label = QtWidgets.QLabel('%s:' % _("V-Angle"))
- self.vangle_label.setToolTip(
- _("V-Agle.\n"
- "Angle at the tip for the V-Shape Tools."))
- self.vangle_entry = FCDoubleSpinner()
- self.vangle_entry.set_range(-360.0, 360.0)
- self.vangle_entry.set_precision(self.decimals)
- self.vangle_entry.setObjectName('gdb_vangle')
- self.grid1.addWidget(self.vangle_label, 8, 0)
- self.grid1.addWidget(self.vangle_entry, 8, 1)
- # Feedrate Rapids
- self.frapids_label = QtWidgets.QLabel('%s:' % _("FR Rapids"))
- self.frapids_label.setToolTip(
- _("FR Rapids. Feedrate Rapids\n"
- "Speed used while moving as fast as possible.\n"
- "This is used only by some devices that can't use\n"
- "the G0 g-code command. Mostly 3D printers."))
- self.frapids_entry = FCDoubleSpinner()
- self.frapids_entry.set_range(0.0000, 9999.9999)
- self.frapids_entry.set_precision(self.decimals)
- self.frapids_entry.setObjectName('gdb_frapids')
- self.grid1.addWidget(self.frapids_label, 10, 0)
- self.grid1.addWidget(self.frapids_entry, 10, 1)
- # Extra Cut
- self.ecut_label = QtWidgets.QLabel('%s:' % _("ExtraCut"))
- self.ecut_label.setToolTip(
- _("Extra Cut.\n"
- "If checked, after a isolation is finished an extra cut\n"
- "will be added where the start and end of isolation meet\n"
- "such as that this point is covered by this extra cut to\n"
- "ensure a complete isolation."))
- self.ecut_cb = FCCheckBox()
- self.ecut_cb.setObjectName('gdb_ecut')
- self.grid1.addWidget(self.ecut_label, 12, 0)
- self.grid1.addWidget(self.ecut_cb, 12, 1)
- # Extra Cut Length
- self.ecut_length_label = QtWidgets.QLabel('%s:' % _("E-Cut Length"))
- self.ecut_length_label.setToolTip(
- _("Extra Cut length.\n"
- "If checked, after a isolation is finished an extra cut\n"
- "will be added where the start and end of isolation meet\n"
- "such as that this point is covered by this extra cut to\n"
- "ensure a complete isolation. This is the length of\n"
- "the extra cut."))
- self.ecut_length_entry = FCDoubleSpinner()
- self.ecut_length_entry.set_range(0.0000, 9999.9999)
- self.ecut_length_entry.set_precision(self.decimals)
- self.ecut_length_entry.setObjectName('gdb_ecut_length')
- self.grid1.addWidget(self.ecut_length_label, 13, 0)
- self.grid1.addWidget(self.ecut_length_entry, 13, 1)
- # ####################################################################
- # ####################################################################
- # GUI for the lower part of the window
- # ####################################################################
- # ####################################################################
- new_vlay = QtWidgets.QVBoxLayout()
- grid_layout.addLayout(new_vlay, 1, 0, 1, 2)
- self.buttons_frame = QtWidgets.QFrame()
- self.buttons_frame.setContentsMargins(0, 0, 0, 0)
- new_vlay.addWidget(self.buttons_frame)
- self.buttons_box = QtWidgets.QHBoxLayout()
- self.buttons_box.setContentsMargins(0, 0, 0, 0)
- self.buttons_frame.setLayout(self.buttons_box)
- self.buttons_frame.show()
- add_entry_btn = FCButton(_("Add Tool in DB"))
- add_entry_btn.setToolTip(
- _("Add a new tool in the Tools Database.\n"
- "It will be used in the Geometry UI.\n"
- "You can edit it after it is added.")
- )
- self.buttons_box.addWidget(add_entry_btn)
- # add_fct_entry_btn = FCButton(_("Add Paint/NCC Tool in DB"))
- # add_fct_entry_btn.setToolTip(
- # _("Add a new tool in the Tools Database.\n"
- # "It will be used in the Paint/NCC Tools UI.\n"
- # "You can edit it after it is added.")
- # )
- # self.buttons_box.addWidget(add_fct_entry_btn)
- remove_entry_btn = FCButton(_("Delete Tool from DB"))
- remove_entry_btn.setToolTip(
- _("Remove a selection of tools in the Tools Database.")
- )
- self.buttons_box.addWidget(remove_entry_btn)
- export_db_btn = FCButton(_("Export DB"))
- export_db_btn.setToolTip(
- _("Save the Tools Database to a custom text file.")
- )
- self.buttons_box.addWidget(export_db_btn)
- import_db_btn = FCButton(_("Import DB"))
- import_db_btn.setToolTip(
- _("Load the Tools Database information's from a custom text file.")
- )
- self.buttons_box.addWidget(import_db_btn)
- self.add_tool_from_db = FCButton(_("Add Tool from Tools DB"))
- self.add_tool_from_db.setToolTip(
- _("Add a new tool in the Tools Table of the\n"
- "active Geometry object after selecting a tool\n"
- "in the Tools Database.")
- )
- self.add_tool_from_db.hide()
- self.cancel_tool_from_db = FCButton(_("Cancel"))
- self.cancel_tool_from_db.hide()
- hlay = QtWidgets.QHBoxLayout()
- tree_layout.addLayout(hlay)
- hlay.addWidget(self.add_tool_from_db)
- hlay.addWidget(self.cancel_tool_from_db)
- hlay.addStretch()
- # ##############################################################################
- # ##############################################################################
- # ########## SETUP THE DICTIONARIES THAT HOLD THE WIDGETS #####################
- # ##############################################################################
- # ##############################################################################
- self.form_fields = {
- # Basic
- "name": self.name_entry,
- "tooldia": self.dia_entry,
- "tool_type": self.shape_combo,
- "cutz": self.cutz_entry,
- "multidepth": self.multidepth_cb,
- "depthperpass": self.multidepth_entry,
- "travelz": self.travelz_entry,
- "feedrate": self.frxy_entry,
- "feedrate_z": self.frz_entry,
- "spindlespeed": self.spindle_entry,
- "dwell": self.dwell_cb,
- "dwelltime": self.dwelltime_entry,
- # Advanced
- "type": self.type_combo,
- "offset": self.tooloffset_combo,
- "offset_value": self.custom_offset_entry,
- "vtipdia": self.vdia_entry,
- "vtipangle": self.vangle_entry,
- "feedrate_rapid": self.frapids_entry,
- "extracut": self.ecut_cb,
- "extracut_length": self.ecut_length_entry
- }
- self.name2option = {
- # Basic
- "gdb_name": "name",
- "gdb_dia": "tooldia",
- "gdb_shape": "tool_type",
- "gdb_cutz": "cutz",
- "gdb_multidepth": "multidepth",
- "gdb_multidepth_entry": "depthperpass",
- "gdb_travel": "travelz",
- "gdb_frxy": "feedrate",
- "gdb_frz": "feedrate_z",
- "gdb_spindle": "spindlespeed",
- "gdb_dwell": "dwell",
- "gdb_dwelltime": "dwelltime",
- # Advanced
- "gdb_type": "type",
- "gdb_tool_offset": "offset",
- "gdb_custom_offset": "offset_value",
- "gdb_vdia": "vtipdia",
- "gdb_vangle": "vtipangle",
- "gdb_frapids": "feedrate_rapid",
- "gdb_ecut": "extracut",
- "gdb_ecut_length": "extracut_length"
- }
- self.current_toolid = None
- # ##############################################################################
- # ######################## SIGNALS #############################################
- # ##############################################################################
- add_entry_btn.clicked.connect(self.on_tool_add)
- remove_entry_btn.clicked.connect(self.on_tool_delete)
- export_db_btn.clicked.connect(self.on_export_tools_db_file)
- import_db_btn.clicked.connect(self.on_import_tools_db_file)
- # closebtn.clicked.connect(self.accept)
- self.add_tool_from_db.clicked.connect(self.on_tool_requested_from_app)
- self.cancel_tool_from_db.clicked.connect(self.on_cancel_tool)
- # self.tree_widget.selectionModel().selectionChanged.connect(self.on_list_selection_change)
- self.tree_widget.currentItemChanged.connect(self.on_list_selection_change)
- self.tree_widget.itemChanged.connect(self.on_list_item_edited)
- self.tree_widget.customContextMenuRequested.connect(self.on_menu_request)
- self.setup_db_ui()
- def on_menu_request(self, pos):
- menu = QtWidgets.QMenu()
- add_tool = menu.addAction(QtGui.QIcon(self.app.resource_location + '/plus16.png'), _("Add to DB"))
- add_tool.triggered.connect(self.on_tool_add)
- copy_tool = menu.addAction(QtGui.QIcon(self.app.resource_location + '/copy16.png'), _("Copy from DB"))
- copy_tool.triggered.connect(self.on_tool_copy)
- delete_tool = menu.addAction(QtGui.QIcon(self.app.resource_location + '/delete32.png'), _("Delete from DB"))
- delete_tool.triggered.connect(self.on_tool_delete)
- # tree_item = self.tree_widget.itemAt(pos)
- menu.exec(self.tree_widget.viewport().mapToGlobal(pos))
- def on_list_selection_change(self, current, previous):
- # for idx in current.indexes():
- # print(idx.data())
- # print(current.text(0))
- self.current_toolid = int(current.text(0))
- self.storage_to_form(self.db_tool_dict[current.text(0)])
- def on_list_item_edited(self, item, column):
- if column == 0:
- return
- self.name_entry.set_value(item.text(1))
- def storage_to_form(self, dict_storage):
- for form_key in self.form_fields:
- for storage_key in dict_storage:
- if form_key == storage_key:
- try:
- self.form_fields[form_key].set_value(dict_storage[form_key])
- except Exception as e:
- print(str(e))
- if storage_key == 'data':
- for data_key in dict_storage[storage_key]:
- if form_key == data_key:
- try:
- self.form_fields[form_key].set_value(dict_storage['data'][data_key])
- except Exception as e:
- print(str(e))
- def form_to_storage(self, tool):
- self.blockSignals(True)
- widget_changed = self.sender()
- wdg_objname = widget_changed.objectName()
- option_changed = self.name2option[wdg_objname]
- tooluid_item = int(tool)
- for tooluid_key, tooluid_val in self.db_tool_dict.items():
- if int(tooluid_key) == tooluid_item:
- new_option_value = self.form_fields[option_changed].get_value()
- if option_changed in tooluid_val:
- tooluid_val[option_changed] = new_option_value
- if option_changed in tooluid_val['data']:
- tooluid_val['data'][option_changed] = new_option_value
- self.blockSignals(False)
- def setup_db_ui(self):
- filename = self.app.data_path + '/geo_tools_db.FlatDB'
- # load the database tools from the file
- try:
- with open(filename) as f:
- tools = f.read()
- except IOError:
- self.app.log.error("Could not load tools DB file.")
- self.app.inform.emit('[ERROR] %s' % _("Could not load Tools DB file."))
- return
- try:
- self.db_tool_dict = json.loads(tools)
- except Exception:
- e = sys.exc_info()[0]
- self.app.log.error(str(e))
- self.app.inform.emit('[ERROR] %s' % _("Failed to parse Tools DB file."))
- return
- self.app.inform.emit('[success] %s: %s' % (_("Loaded FlatCAM Tools DB from"), filename))
- self.build_db_ui()
- def build_db_ui(self):
- self.ui_disconnect()
- nr_crt = 0
- parent = self.tree_widget
- self.tree_widget.blockSignals(True)
- self.tree_widget.clear()
- self.tree_widget.blockSignals(False)
- for toolid, dict_val in self.db_tool_dict.items():
- row = nr_crt
- nr_crt += 1
- t_name = dict_val['name']
- try:
- # self.add_tool_table_line(row, name=t_name, tooldict=dict_val)
- self.tree_widget.blockSignals(True)
- try:
- self.tree_widget.addParentEditable(parent=parent, title=[str(row+1), t_name], editable=True)
- except Exception as e:
- print('FlatCAMCoomn.ToolDB2.build_db_ui() -> ', str(e))
- self.tree_widget.blockSignals(False)
- except Exception as e:
- self.app.log.debug("ToolDB.build_db_ui.add_tool_table_line() --> %s" % str(e))
- if self.current_toolid is None or self.current_toolid < 1:
- if self.db_tool_dict:
- self.storage_to_form(self.db_tool_dict['1'])
- else:
- self.storage_to_form(self.db_tool_dict[str(self.current_toolid)])
- self.ui_connect()
- def add_tool_table_line(self, row, name, tooldict):
- data = tooldict['data']
- def on_tool_add(self):
- """
- Add a tool in the DB Tool Table
- :return: None
- """
- default_data = {}
- default_data.update({
- "cutz": float(self.app.defaults["geometry_cutz"]),
- "multidepth": self.app.defaults["geometry_multidepth"],
- "depthperpass": float(self.app.defaults["geometry_depthperpass"]),
- "vtipdia": float(self.app.defaults["geometry_vtipdia"]),
- "vtipangle": float(self.app.defaults["geometry_vtipangle"]),
- "travelz": float(self.app.defaults["geometry_travelz"]),
- "feedrate": float(self.app.defaults["geometry_feedrate"]),
- "feedrate_z": float(self.app.defaults["geometry_feedrate_z"]),
- "feedrate_rapid": float(self.app.defaults["geometry_feedrate_rapid"]),
- "spindlespeed": self.app.defaults["geometry_spindlespeed"],
- "dwell": self.app.defaults["geometry_dwell"],
- "dwelltime": float(self.app.defaults["geometry_dwelltime"]),
- "ppname_g": self.app.defaults["geometry_ppname_g"],
- "extracut": self.app.defaults["geometry_extracut"],
- "extracut_length": float(self.app.defaults["geometry_extracut_length"]),
- "toolchange": self.app.defaults["geometry_toolchange"],
- "toolchangexy": self.app.defaults["geometry_toolchangexy"],
- "toolchangez": float(self.app.defaults["geometry_toolchangez"]),
- "startz": self.app.defaults["geometry_startz"],
- "endz": float(self.app.defaults["geometry_endz"])
- })
- dict_elem = {}
- dict_elem['name'] = 'new_tool'
- if type(self.app.defaults["geometry_cnctooldia"]) == float:
- dict_elem['tooldia'] = self.app.defaults["geometry_cnctooldia"]
- else:
- try:
- tools_string = self.app.defaults["geometry_cnctooldia"].split(",")
- tools_diameters = [eval(a) for a in tools_string if a != '']
- dict_elem['tooldia'] = tools_diameters[0] if tools_diameters else 0.0
- except Exception as e:
- self.app.log.debug("ToolDB.on_tool_add() --> %s" % str(e))
- return
- dict_elem['offset'] = 'Path'
- dict_elem['offset_value'] = 0.0
- dict_elem['type'] = 'Rough'
- dict_elem['tool_type'] = 'C1'
- dict_elem['data'] = default_data
- new_toolid = len(self.db_tool_dict) + 1
- self.db_tool_dict[str(new_toolid)] = deepcopy(dict_elem)
- # add the new entry to the Tools DB table
- self.update_storage()
- self.build_db_ui()
- self.app.inform.emit('[success] %s' % _("Tool added to DB."))
- def on_tool_copy(self):
- """
- Copy a selection of Tools in the Tools DB table
- :return:
- """
- new_tool_id = len(self.db_tool_dict)
- for item in self.tree_widget.selectedItems():
- old_tool_id = item.data(0, QtCore.Qt.DisplayRole)
- for toolid, dict_val in list(self.db_tool_dict.items()):
- if int(old_tool_id) == int(toolid):
- new_tool_id += 1
- new_key = str(new_tool_id)
- self.db_tool_dict.update({
- new_key: deepcopy(dict_val)
- })
- self.current_toolid = new_tool_id
- self.update_storage()
- self.build_db_ui()
- self.app.inform.emit('[success] %s' % _("Tool copied from Tools DB."))
- def on_tool_delete(self):
- """
- Delete a selection of Tools in the Tools DB table
- :return:
- """
- for item in self.tree_widget.selectedItems():
- toolname_to_remove = item.data(0, QtCore.Qt.DisplayRole)
- for toolid, dict_val in list(self.db_tool_dict.items()):
- if int(toolname_to_remove) == int(toolid):
- # remove from the storage
- self.db_tool_dict.pop(toolid, None)
- self.current_toolid -= 1
- self.update_storage()
- self.build_db_ui()
- self.app.inform.emit('[success] %s' % _("Tool removed from Tools DB."))
- def on_export_tools_db_file(self):
- self.app.report_usage("on_export_tools_db_file")
- self.app.log.debug("on_export_tools_db_file()")
- date = str(datetime.today()).rpartition('.')[0]
- date = ''.join(c for c in date if c not in ':-')
- date = date.replace(' ', '_')
- filter__ = "Text File (*.TXT);;All Files (*.*)"
- filename, _f = QtWidgets.QFileDialog.getSaveFileName(caption=_("Export Tools Database"),
- directory='{l_save}/FlatCAM_{n}_{date}'.format(
- l_save=str(self.app.get_last_save_folder()),
- n=_("Tools_Database"),
- date=date),
- filter=filter__)
- filename = str(filename)
- if filename == "":
- self.app.inform.emit('[WARNING_NOTCL] %s' % _("FlatCAM Tools DB export cancelled."))
- return
- else:
- try:
- f = open(filename, 'w')
- f.close()
- except PermissionError:
- self.app.inform.emit('[WARNING] %s' %
- _("Permission denied, saving not possible.\n"
- "Most likely another app is holding the file open and not accessible."))
- return
- except IOError:
- self.app.log.debug('Creating a new Tools DB file ...')
- f = open(filename, 'w')
- f.close()
- except Exception:
- e = sys.exc_info()[0]
- self.app.log.error("Could not load Tools DB file.")
- self.app.log.error(str(e))
- self.app.inform.emit('[ERROR_NOTCL] %s' % _("Could not load Tools DB file."))
- return
- # Save update options
- try:
- # Save Tools DB in a file
- try:
- with open(filename, "w") as f:
- json.dump(self.db_tool_dict, f, default=to_dict, indent=2)
- except Exception as e:
- self.app.log.debug("App.on_save_tools_db() --> %s" % str(e))
- self.inform.emit('[ERROR_NOTCL] %s' % _("Failed to write Tools DB to file."))
- return
- except Exception:
- self.app.inform.emit('[ERROR_NOTCL] %s' % _("Failed to write Tools DB to file."))
- return
- self.app.inform.emit('[success] %s: %s' % (_("Exported Tools DB to"), filename))
- def on_import_tools_db_file(self):
- self.app.report_usage("on_import_tools_db_file")
- self.app.log.debug("on_import_tools_db_file()")
- filter__ = "Text File (*.TXT);;All Files (*.*)"
- filename, _f = QtWidgets.QFileDialog.getOpenFileName(caption=_("Import FlatCAM Tools DB"), filter=filter__)
- if filename == "":
- self.app.inform.emit('[WARNING_NOTCL] %s' % _("FlatCAM Tools DB import cancelled."))
- else:
- try:
- with open(filename) as f:
- tools_in_db = f.read()
- except IOError:
- self.app.log.error("Could not load Tools DB file.")
- self.app.inform.emit('[ERROR_NOTCL] %s' % _("Could not load Tools DB file."))
- return
- try:
- self.db_tool_dict = json.loads(tools_in_db)
- except Exception:
- e = sys.exc_info()[0]
- self.app.log.error(str(e))
- self.app.inform.emit('[ERROR] %s' % _("Failed to parse Tools DB file."))
- return
- self.app.inform.emit('[success] %s: %s' % (_("Loaded FlatCAM Tools DB from"), filename))
- self.build_db_ui()
- self.update_storage()
- def on_save_tools_db(self, silent=False):
- self.app.log.debug("ToolsDB.on_save_button() --> Saving Tools Database to file.")
- filename = self.app.data_path + "/geo_tools_db.FlatDB"
- # Preferences save, update the color of the Tools DB Tab text
- for idx in range(self.app.ui.plot_tab_area.count()):
- if self.app.ui.plot_tab_area.tabText(idx) == _("Tools Database"):
- self.app.ui.plot_tab_area.tabBar.setTabTextColor(idx, QtGui.QColor('black'))
- # Save Tools DB in a file
- try:
- f = open(filename, "w")
- json.dump(self.db_tool_dict, f, default=to_dict, indent=2)
- f.close()
- except Exception as e:
- self.app.log.debug("ToolsDB.on_save_tools_db() --> %s" % str(e))
- self.app.inform.emit('[ERROR_NOTCL] %s' % _("Failed to write Tools DB to file."))
- return
- if not silent:
- self.app.inform.emit('[success] %s' % _("Saved Tools DB."))
- def ui_connect(self):
- # make sure that we don't make multiple connections to the widgets
- self.ui_disconnect()
- self.name_entry.editingFinished.connect(self.update_tree_name)
- for key in self.form_fields:
- wdg = self.form_fields[key]
- # FCEntry
- if isinstance(wdg, FCEntry):
- wdg.textChanged.connect(self.update_storage)
- # ComboBox
- if isinstance(wdg, FCComboBox):
- wdg.currentIndexChanged.connect(self.update_storage)
- # CheckBox
- if isinstance(wdg, FCCheckBox):
- wdg.toggled.connect(self.update_storage)
- # SpinBox, DoubleSpinBox
- if isinstance(wdg, FCSpinner) or isinstance(wdg, FCDoubleSpinner):
- wdg.valueChanged.connect(self.update_storage)
- def ui_disconnect(self):
- try:
- self.name_entry.editingFinished.disconnect(self.update_tree_name)
- except (TypeError, AttributeError):
- pass
- for key in self.form_fields:
- wdg = self.form_fields[key]
- # FCEntry
- if isinstance(wdg, FCEntry):
- try:
- wdg.textChanged.disconnect(self.update_storage)
- except (TypeError, AttributeError):
- pass
- # ComboBox
- if isinstance(wdg, FCComboBox):
- try:
- wdg.currentIndexChanged.disconnect(self.update_storage)
- except (TypeError, AttributeError):
- pass
- # CheckBox
- if isinstance(wdg, FCCheckBox):
- try:
- wdg.toggled.disconnect(self.update_storage)
- except (TypeError, AttributeError):
- pass
- # SpinBox, DoubleSpinBox
- if isinstance(wdg, FCSpinner) or isinstance(wdg, FCDoubleSpinner):
- try:
- wdg.valueChanged.disconnect(self.update_storage)
- except (TypeError, AttributeError):
- pass
- def update_tree_name(self):
- val = self.name_entry.get_value()
- item = self.tree_widget.currentItem()
- # I'm setting the value for the second column (designated by 1) because first column holds the ID
- # and second column holds the Name (this behavior is set in the build_ui method)
- item.setData(1, QtCore.Qt.DisplayRole, val)
- def update_storage(self):
- tool_id = str(self.current_toolid)
- wdg = self.sender()
- if wdg is None:
- return
- wdg_name = wdg.objectName()
- if wdg_name == "gdb_name":
- self.db_tool_dict[tool_id]['name'] = wdg.get_value()
- elif wdg_name == "gdb_dia":
- self.db_tool_dict[tool_id]['tooldia'] = wdg.get_value()
- elif wdg_name == "gdb_tool_offset":
- self.db_tool_dict[tool_id]['offset'] = wdg.get_value()
- elif wdg_name == "gdb_custom_offset":
- self.db_tool_dict[tool_id]['offset_value'] = wdg.get_value()
- elif wdg_name == "gdb_type":
- self.db_tool_dict[tool_id]['type'] = wdg.get_value()
- elif wdg_name == "gdb_shape":
- self.db_tool_dict[tool_id]['tool_type'] = wdg.get_value()
- else:
- if wdg_name == "gdb_cutz":
- self.db_tool_dict[tool_id]['data']['cutz'] = wdg.get_value()
- elif wdg_name == "gdb_multidepth":
- self.db_tool_dict[tool_id]['data']['multidepth'] = wdg.get_value()
- elif wdg_name == "gdb_multidepth_entry":
- self.db_tool_dict[tool_id]['data']['depthperpass'] = wdg.get_value()
- elif wdg_name == "gdb_travel":
- self.db_tool_dict[tool_id]['data']['travelz'] = wdg.get_value()
- elif wdg_name == "gdb_frxy":
- self.db_tool_dict[tool_id]['data']['feedrate'] = wdg.get_value()
- elif wdg_name == "gdb_frz":
- self.db_tool_dict[tool_id]['data']['feedrate_z'] = wdg.get_value()
- elif wdg_name == "gdb_spindle":
- self.db_tool_dict[tool_id]['data']['spindlespeed'] = wdg.get_value()
- elif wdg_name == "gdb_dwell":
- self.db_tool_dict[tool_id]['data']['dwell'] = wdg.get_value()
- elif wdg_name == "gdb_dwelltime":
- self.db_tool_dict[tool_id]['data']['dwelltime'] = wdg.get_value()
- elif wdg_name == "gdb_vdia":
- self.db_tool_dict[tool_id]['data']['vtipdia'] = wdg.get_value()
- elif wdg_name == "gdb_vangle":
- self.db_tool_dict[tool_id]['data']['vtipangle'] = wdg.get_value()
- elif wdg_name == "gdb_frapids":
- self.db_tool_dict[tool_id]['data']['feedrate_rapid'] = wdg.get_value()
- elif wdg_name == "gdb_ecut":
- self.db_tool_dict[tool_id]['data']['extracut'] = wdg.get_value()
- elif wdg_name == "gdb_ecut_length":
- self.db_tool_dict[tool_id]['data']['extracut_length'] = wdg.get_value()
- self.callback_app()
- def on_tool_requested_from_app(self):
- if not self.table_widget.selectionModel().selectedRows():
- self.app.inform.emit('[WARNING_NOTCL] %s...' % _("No Tool/row selected in the Tools Database table"))
- return
- model_index_list = self.table_widget.selectionModel().selectedRows()
- for model_index in model_index_list:
- selected_row = model_index.row()
- tool_uid = selected_row + 1
- for key in self.db_tool_dict.keys():
- if str(key) == str(tool_uid):
- selected_tool = self.db_tool_dict[key]
- self.on_tool_request(tool=selected_tool)
- def on_cancel_tool(self):
- for idx in range(self.app.ui.plot_tab_area.count()):
- if self.app.ui.plot_tab_area.tabText(idx) == _("Tools Database"):
- wdg = self.app.ui.plot_tab_area.widget(idx)
- wdg.deleteLater()
- self.app.ui.plot_tab_area.removeTab(idx)
- self.app.inform.emit('%s' % _("Cancelled adding tool from DB."))
- def resize_new_tool_table_widget(self, min_size, max_size):
- """
- Resize the table widget responsible for adding new tool in the Tool Database
- :param min_size: passed by rangeChanged signal or the self.new_tool_table_widget.horizontalScrollBar()
- :param max_size: passed by rangeChanged signal or the self.new_tool_table_widget.horizontalScrollBar()
- :return:
- """
- t_height = self.t_height
- if max_size > min_size:
- t_height = self.t_height + self.new_tool_table_widget.verticalScrollBar().height()
- self.new_tool_table_widget.setMaximumHeight(t_height)
- def closeEvent(self, QCloseEvent):
- super().closeEvent(QCloseEvent)
- def color_variant(hex_color, bright_factor=1):
- """
- Takes a color in HEX format #FF00FF and produces a lighter or darker variant
- :param hex_color: color to change
- :param bright_factor: factor to change the color brightness [0 ... 1]
- :return: modified color
- """
- if len(hex_color) != 7:
- print("Color is %s, but needs to be in #FF00FF format. Returning original color." % hex_color)
- return hex_color
- if bright_factor > 1.0:
- bright_factor = 1.0
- if bright_factor < 0.0:
- bright_factor = 0.0
- rgb_hex = [hex_color[x:x + 2] for x in [1, 3, 5]]
- new_rgb = []
- for hex_value in rgb_hex:
- # adjust each color channel and turn it into a INT suitable as argument for hex()
- mod_color = round(int(hex_value, 16) * bright_factor)
- # make sure that each color channel has two digits without the 0x prefix
- mod_color_hex = str(hex(mod_color)[2:]).zfill(2)
- new_rgb.append(mod_color_hex)
- return "#" + "".join([i for i in new_rgb])
|