| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622 |
- # ##########################################################
- # FlatCAM: 2D Post-processing for Manufacturing #
- # http://flatcam.org #
- # Author: Juan Pablo Caram (c) #
- # Date: 2/5/2014 #
- # MIT Licence #
- # ##########################################################
- # ##########################################################
- # File modified by: Marius Stanciu #
- # ##########################################################
- from shapely.geometry import Point, LineString
- from copy import deepcopy
- from flatcamParsers.ParseExcellon import Excellon
- from flatcamObjects.FlatCAMObj import *
- import itertools
- import gettext
- import FlatCAMTranslation as fcTranslate
- import builtins
- fcTranslate.apply_language('strings')
- if '_' not in builtins.__dict__:
- _ = gettext.gettext
- class ExcellonObject(FlatCAMObj, Excellon):
- """
- Represents Excellon/Drill code.
- """
- ui_type = ExcellonObjectUI
- optionChanged = QtCore.pyqtSignal(str)
- def __init__(self, name):
- self.decimals = self.app.decimals
- self.circle_steps = int(self.app.defaults["geometry_circle_steps"])
- Excellon.__init__(self, geo_steps_per_circle=self.circle_steps)
- FlatCAMObj.__init__(self, name)
- self.kind = "excellon"
- self.options.update({
- "plot": True,
- "solid": False,
- "operation": "drill",
- "milling_type": "drills",
- "milling_dia": 0.04,
- "cutz": -0.1,
- "multidepth": False,
- "depthperpass": 0.7,
- "travelz": 0.1,
- "feedrate": self.app.defaults["geometry_feedrate"],
- "feedrate_z": 5.0,
- "feedrate_rapid": 5.0,
- "tooldia": 0.1,
- "slot_tooldia": 0.1,
- "toolchange": False,
- "toolchangez": 1.0,
- "toolchangexy": "0.0, 0.0",
- "extracut": self.app.defaults["geometry_extracut"],
- "extracut_length": self.app.defaults["geometry_extracut_length"],
- "endz": 2.0,
- "endxy": '',
- "startz": None,
- "offset": 0.0,
- "spindlespeed": 0,
- "dwell": True,
- "dwelltime": 1000,
- "ppname_e": 'default',
- "ppname_g": self.app.defaults["geometry_ppname_g"],
- "z_pdepth": -0.02,
- "feedrate_probe": 3.0,
- "optimization_type": "B",
- })
- # TODO: Document this.
- self.tool_cbs = {}
- # dict that holds the object names and the option name
- # the key is the object name (defines in ObjectUI) for each UI element that is a parameter
- # particular for a tool and the value is the actual name of the option that the UI element is changing
- self.name2option = {}
- # default set of data to be added to each tool in self.tools as self.tools[tool]['data'] = self.default_data
- self.default_data = {}
- # fill in self.default_data values from self.options
- for opt_key, opt_val in self.app.options.items():
- if opt_key.find('excellon_') == 0:
- self.default_data[opt_key] = deepcopy(opt_val)
- for opt_key, opt_val in self.app.options.items():
- if opt_key.find('geometry_') == 0:
- self.default_data[opt_key] = deepcopy(opt_val)
- # variable to store the total amount of drills per job
- self.tot_drill_cnt = 0
- self.tool_row = 0
- # variable to store the total amount of slots per job
- self.tot_slot_cnt = 0
- self.tool_row_slots = 0
- # variable to store the distance travelled
- self.travel_distance = 0.0
- # store the source file here
- self.source_file = ""
- self.multigeo = False
- self.units_found = self.app.defaults['units']
- self.fill_color = self.app.defaults['excellon_plot_fill']
- self.outline_color = self.app.defaults['excellon_plot_line']
- self.alpha_level = 'bf'
- # Attributes to be included in serialization
- # Always append to it because it carries contents
- # from predecessors.
- self.ser_attrs += ['options', 'kind']
- def merge(self, exc_list, exc_final):
- """
- Merge Excellon objects found in exc_list parameter into exc_final object.
- Options are always copied from source .
- Tools are disregarded, what is taken in consideration is the unique drill diameters found as values in the
- exc_list tools dict's. In the reconstruction section for each unique tool diameter it will be created a
- tool_name to be used in the final Excellon object, exc_final.
- If only one object is in exc_list parameter then this function will copy that object in the exc_final
- :param exc_list: List or one object of ExcellonObject Objects to join.
- :param exc_final: Destination ExcellonObject object.
- :return: None
- """
- try:
- decimals_exc = self.decimals
- except AttributeError:
- decimals_exc = 4
- # flag to signal that we need to reorder the tools dictionary and drills and slots lists
- flag_order = False
- try:
- flattened_list = list(itertools.chain(*exc_list))
- except TypeError:
- flattened_list = exc_list
- # this dict will hold the unique tool diameters found in the exc_list objects as the dict keys and the dict
- # values will be list of Shapely Points; for drills
- custom_dict_drills = {}
- # this dict will hold the unique tool diameters found in the exc_list objects as the dict keys and the dict
- # values will be list of Shapely Points; for slots
- custom_dict_slots = {}
- for exc in flattened_list:
- # copy options of the current excellon obj to the final excellon obj
- for option in exc.options:
- if option != 'name':
- try:
- exc_final.options[option] = exc.options[option]
- except Exception:
- exc.app.log.warning("Failed to copy option.", option)
- for drill in exc.drills:
- exc_tool_dia = float('%.*f' % (decimals_exc, exc.tools[drill['tool']]['C']))
- if exc_tool_dia not in custom_dict_drills:
- custom_dict_drills[exc_tool_dia] = [drill['point']]
- else:
- custom_dict_drills[exc_tool_dia].append(drill['point'])
- for slot in exc.slots:
- exc_tool_dia = float('%.*f' % (decimals_exc, exc.tools[slot['tool']]['C']))
- if exc_tool_dia not in custom_dict_slots:
- custom_dict_slots[exc_tool_dia] = [[slot['start'], slot['stop']]]
- else:
- custom_dict_slots[exc_tool_dia].append([slot['start'], slot['stop']])
- # add the zeros and units to the exc_final object
- exc_final.zeros = exc.zeros
- exc_final.units = exc.units
- # ##########################################
- # Here we add data to the exc_final object #
- # ##########################################
- # variable to make tool_name for the tools
- current_tool = 0
- # The tools diameter are now the keys in the drill_dia dict and the values are the Shapely Points in case of
- # drills
- for tool_dia in custom_dict_drills:
- # we create a tool name for each key in the drill_dia dict (the key is a unique drill diameter)
- current_tool += 1
- tool_name = str(current_tool)
- spec = {"C": float(tool_dia)}
- exc_final.tools[tool_name] = spec
- # rebuild the drills list of dict's that belong to the exc_final object
- for point in custom_dict_drills[tool_dia]:
- exc_final.drills.append(
- {
- "point": point,
- "tool": str(current_tool)
- }
- )
- # The tools diameter are now the keys in the drill_dia dict and the values are a list ([start, stop])
- # of two Shapely Points in case of slots
- for tool_dia in custom_dict_slots:
- # we create a tool name for each key in the slot_dia dict (the key is a unique slot diameter)
- # but only if there are no drills
- if not exc_final.tools:
- current_tool += 1
- tool_name = str(current_tool)
- spec = {"C": float(tool_dia)}
- exc_final.tools[tool_name] = spec
- else:
- dia_list = []
- for v in exc_final.tools.values():
- dia_list.append(float(v["C"]))
- if tool_dia not in dia_list:
- flag_order = True
- current_tool = len(dia_list) + 1
- tool_name = str(current_tool)
- spec = {"C": float(tool_dia)}
- exc_final.tools[tool_name] = spec
- else:
- for k, v in exc_final.tools.items():
- if v["C"] == tool_dia:
- current_tool = int(k)
- break
- # rebuild the slots list of dict's that belong to the exc_final object
- for point in custom_dict_slots[tool_dia]:
- exc_final.slots.append(
- {
- "start": point[0],
- "stop": point[1],
- "tool": str(current_tool)
- }
- )
- # flag_order == True means that there was an slot diameter not in the tools and we also have drills
- # and the new tool was added to self.tools therefore we need to reorder the tools and drills and slots
- current_tool = 0
- if flag_order is True:
- dia_list = []
- temp_drills = []
- temp_slots = []
- temp_tools = {}
- for v in exc_final.tools.values():
- dia_list.append(float(v["C"]))
- dia_list.sort()
- for ordered_dia in dia_list:
- current_tool += 1
- tool_name_temp = str(current_tool)
- spec_temp = {"C": float(ordered_dia)}
- temp_tools[tool_name_temp] = spec_temp
- for drill in exc_final.drills:
- exc_tool_dia = float('%.*f' % (decimals_exc, exc_final.tools[drill['tool']]['C']))
- if exc_tool_dia == ordered_dia:
- temp_drills.append(
- {
- "point": drill["point"],
- "tool": str(current_tool)
- }
- )
- for slot in exc_final.slots:
- slot_tool_dia = float('%.*f' % (decimals_exc, exc_final.tools[slot['tool']]['C']))
- if slot_tool_dia == ordered_dia:
- temp_slots.append(
- {
- "start": slot["start"],
- "stop": slot["stop"],
- "tool": str(current_tool)
- }
- )
- # delete the exc_final tools, drills and slots
- exc_final.tools = {}
- exc_final.drills[:] = []
- exc_final.slots[:] = []
- # update the exc_final tools, drills and slots with the ordered values
- exc_final.tools = temp_tools
- exc_final.drills[:] = temp_drills
- exc_final.slots[:] = temp_slots
- # create the geometry for the exc_final object
- exc_final.create_geometry()
- def build_ui(self):
- FlatCAMObj.build_ui(self)
- self.units = self.app.defaults['units'].upper()
- for row in range(self.ui.tools_table.rowCount()):
- try:
- # if connected, disconnect the signal from the slot on item_changed as it creates issues
- offset_spin_widget = self.ui.tools_table.cellWidget(row, 4)
- offset_spin_widget.valueChanged.disconnect()
- except (TypeError, AttributeError):
- pass
- n = len(self.tools)
- # we have (n+2) rows because there are 'n' tools, each a row, plus the last 2 rows for totals.
- self.ui.tools_table.setRowCount(n + 2)
- self.tot_drill_cnt = 0
- self.tot_slot_cnt = 0
- self.tool_row = 0
- sort = []
- for k, v in list(self.tools.items()):
- sort.append((k, v.get('C')))
- sorted_tools = sorted(sort, key=lambda t1: t1[1])
- tools = [i[0] for i in sorted_tools]
- new_options = {}
- for opt in self.options:
- new_options[opt] = self.options[opt]
- for tool_no in tools:
- # add the data dictionary for each tool with the default values
- self.tools[tool_no]['data'] = deepcopy(new_options)
- # self.tools[tool_no]['data']["tooldia"] = self.tools[tool_no]["C"]
- # self.tools[tool_no]['data']["slot_tooldia"] = self.tools[tool_no]["C"]
- drill_cnt = 0 # variable to store the nr of drills per tool
- slot_cnt = 0 # variable to store the nr of slots per tool
- # Find no of drills for the current tool
- for drill in self.drills:
- if drill['tool'] == tool_no:
- drill_cnt += 1
- self.tot_drill_cnt += drill_cnt
- # Find no of slots for the current tool
- for slot in self.slots:
- if slot['tool'] == tool_no:
- slot_cnt += 1
- self.tot_slot_cnt += slot_cnt
- exc_id_item = QtWidgets.QTableWidgetItem('%d' % int(tool_no))
- exc_id_item.setFlags(QtCore.Qt.ItemIsSelectable | QtCore.Qt.ItemIsEnabled)
- dia_item = QtWidgets.QTableWidgetItem('%.*f' % (self.decimals, self.tools[tool_no]['C']))
- dia_item.setFlags(QtCore.Qt.ItemIsEnabled)
- drill_count_item = QtWidgets.QTableWidgetItem('%d' % drill_cnt)
- drill_count_item.setFlags(QtCore.Qt.ItemIsEnabled)
- # if the slot number is zero is better to not clutter the GUI with zero's so we print a space
- slot_count_str = '%d' % slot_cnt if slot_cnt > 0 else ''
- slot_count_item = QtWidgets.QTableWidgetItem(slot_count_str)
- slot_count_item.setFlags(QtCore.Qt.ItemIsEnabled)
- plot_item = FCCheckBox()
- plot_item.setLayoutDirection(QtCore.Qt.RightToLeft)
- if self.ui.plot_cb.isChecked():
- plot_item.setChecked(True)
- self.ui.tools_table.setItem(self.tool_row, 0, exc_id_item) # Tool name/id
- self.ui.tools_table.setItem(self.tool_row, 1, dia_item) # Diameter
- self.ui.tools_table.setItem(self.tool_row, 2, drill_count_item) # Number of drills per tool
- self.ui.tools_table.setItem(self.tool_row, 3, slot_count_item) # Number of drills per tool
- empty_plot_item = QtWidgets.QTableWidgetItem('')
- empty_plot_item.setFlags(~QtCore.Qt.ItemIsSelectable | QtCore.Qt.ItemIsEnabled)
- self.ui.tools_table.setItem(self.tool_row, 5, empty_plot_item)
- self.ui.tools_table.setCellWidget(self.tool_row, 5, plot_item)
- self.tool_row += 1
- # add a last row with the Total number of drills
- empty_1 = QtWidgets.QTableWidgetItem('')
- empty_1.setFlags(~QtCore.Qt.ItemIsSelectable | QtCore.Qt.ItemIsEnabled)
- empty_1_1 = QtWidgets.QTableWidgetItem('')
- empty_1_1.setFlags(~QtCore.Qt.ItemIsSelectable | QtCore.Qt.ItemIsEnabled)
- empty_1_2 = QtWidgets.QTableWidgetItem('')
- empty_1_2.setFlags(~QtCore.Qt.ItemIsSelectable | QtCore.Qt.ItemIsEnabled)
- empty_1_3 = QtWidgets.QTableWidgetItem('')
- empty_1_3.setFlags(~QtCore.Qt.ItemIsSelectable | QtCore.Qt.ItemIsEnabled)
- label_tot_drill_count = QtWidgets.QTableWidgetItem(_('Total Drills'))
- tot_drill_count = QtWidgets.QTableWidgetItem('%d' % self.tot_drill_cnt)
- label_tot_drill_count.setFlags(QtCore.Qt.ItemIsEnabled)
- tot_drill_count.setFlags(QtCore.Qt.ItemIsEnabled)
- self.ui.tools_table.setItem(self.tool_row, 0, empty_1)
- self.ui.tools_table.setItem(self.tool_row, 1, label_tot_drill_count)
- self.ui.tools_table.setItem(self.tool_row, 2, tot_drill_count) # Total number of drills
- self.ui.tools_table.setItem(self.tool_row, 3, empty_1_1)
- self.ui.tools_table.setItem(self.tool_row, 5, empty_1_3)
- font = QtGui.QFont()
- font.setBold(True)
- font.setWeight(75)
- for k in [1, 2]:
- self.ui.tools_table.item(self.tool_row, k).setForeground(QtGui.QColor(127, 0, 255))
- self.ui.tools_table.item(self.tool_row, k).setFont(font)
- self.tool_row += 1
- # add a last row with the Total number of slots
- empty_2 = QtWidgets.QTableWidgetItem('')
- empty_2.setFlags(~QtCore.Qt.ItemIsSelectable | QtCore.Qt.ItemIsEnabled)
- empty_2_1 = QtWidgets.QTableWidgetItem('')
- empty_2_1.setFlags(~QtCore.Qt.ItemIsSelectable | QtCore.Qt.ItemIsEnabled)
- empty_2_2 = QtWidgets.QTableWidgetItem('')
- empty_2_2.setFlags(~QtCore.Qt.ItemIsSelectable | QtCore.Qt.ItemIsEnabled)
- empty_2_3 = QtWidgets.QTableWidgetItem('')
- empty_2_3.setFlags(~QtCore.Qt.ItemIsSelectable | QtCore.Qt.ItemIsEnabled)
- label_tot_slot_count = QtWidgets.QTableWidgetItem(_('Total Slots'))
- tot_slot_count = QtWidgets.QTableWidgetItem('%d' % self.tot_slot_cnt)
- label_tot_slot_count.setFlags(QtCore.Qt.ItemIsEnabled)
- tot_slot_count.setFlags(QtCore.Qt.ItemIsEnabled)
- self.ui.tools_table.setItem(self.tool_row, 0, empty_2)
- self.ui.tools_table.setItem(self.tool_row, 1, label_tot_slot_count)
- self.ui.tools_table.setItem(self.tool_row, 2, empty_2_1)
- self.ui.tools_table.setItem(self.tool_row, 3, tot_slot_count) # Total number of slots
- self.ui.tools_table.setItem(self.tool_row, 5, empty_2_3)
- for kl in [1, 2, 3]:
- self.ui.tools_table.item(self.tool_row, kl).setFont(font)
- self.ui.tools_table.item(self.tool_row, kl).setForeground(QtGui.QColor(0, 70, 255))
- # sort the tool diameter column
- # self.ui.tools_table.sortItems(1)
- # all the tools are selected by default
- self.ui.tools_table.selectColumn(0)
- self.ui.tools_table.resizeColumnsToContents()
- self.ui.tools_table.resizeRowsToContents()
- vertical_header = self.ui.tools_table.verticalHeader()
- # vertical_header.setSectionResizeMode(QtWidgets.QHeaderView.ResizeToContents)
- vertical_header.hide()
- self.ui.tools_table.setVerticalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOff)
- horizontal_header = self.ui.tools_table.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.Stretch)
- horizontal_header.setSectionResizeMode(2, QtWidgets.QHeaderView.ResizeToContents)
- horizontal_header.setSectionResizeMode(3, QtWidgets.QHeaderView.ResizeToContents)
- horizontal_header.setSectionResizeMode(5, QtWidgets.QHeaderView.Fixed)
- horizontal_header.resizeSection(5, 17)
- self.ui.tools_table.setColumnWidth(5, 17)
- # horizontal_header.setStretchLastSection(True)
- # horizontal_header.setColumnWidth(2, QtWidgets.QHeaderView.ResizeToContents)
- # horizontal_header.setStretchLastSection(True)
- self.ui.tools_table.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOff)
- self.ui.tools_table.setSortingEnabled(False)
- self.ui.tools_table.setMinimumHeight(self.ui.tools_table.getHeight())
- self.ui.tools_table.setMaximumHeight(self.ui.tools_table.getHeight())
- if not self.drills:
- self.ui.tooldia_entry.hide()
- self.ui.generate_milling_button.hide()
- else:
- self.ui.tooldia_entry.show()
- self.ui.generate_milling_button.show()
- if not self.slots:
- self.ui.slot_tooldia_entry.hide()
- self.ui.generate_milling_slots_button.hide()
- else:
- self.ui.slot_tooldia_entry.show()
- self.ui.generate_milling_slots_button.show()
- # set the text on tool_data_label after loading the object
- sel_items = self.ui.tools_table.selectedItems()
- sel_rows = [it.row() for it in sel_items]
- if len(sel_rows) > 1:
- self.ui.tool_data_label.setText(
- "<b>%s: <font color='#0000FF'>%s</font></b>" % (_('Parameters for'), _("Multiple Tools"))
- )
- self.ui_connect()
- def set_ui(self, ui):
- """
- Configures the user interface for this object.
- Connects options to form fields.
- :param ui: User interface object.
- :type ui: ExcellonObjectUI
- :return: None
- """
- FlatCAMObj.set_ui(self, ui)
- log.debug("ExcellonObject.set_ui()")
- self.units = self.app.defaults['units'].upper()
- self.form_fields.update({
- "plot": self.ui.plot_cb,
- "solid": self.ui.solid_cb,
- "operation": self.ui.operation_radio,
- "milling_type": self.ui.milling_type_radio,
- "milling_dia": self.ui.mill_dia_entry,
- "cutz": self.ui.cutz_entry,
- "multidepth": self.ui.mpass_cb,
- "depthperpass": self.ui.maxdepth_entry,
- "travelz": self.ui.travelz_entry,
- "feedrate_z": self.ui.feedrate_z_entry,
- "feedrate": self.ui.xyfeedrate_entry,
- "feedrate_rapid": self.ui.feedrate_rapid_entry,
- "tooldia": self.ui.tooldia_entry,
- "slot_tooldia": self.ui.slot_tooldia_entry,
- "toolchange": self.ui.toolchange_cb,
- "toolchangez": self.ui.toolchangez_entry,
- "extracut": self.ui.extracut_cb,
- "extracut_length": self.ui.e_cut_entry,
- "spindlespeed": self.ui.spindlespeed_entry,
- "dwell": self.ui.dwell_cb,
- "dwelltime": self.ui.dwelltime_entry,
- "startz": self.ui.estartz_entry,
- "endz": self.ui.endz_entry,
- "endxy": self.ui.endxy_entry,
- "offset": self.ui.offset_entry,
- "ppname_e": self.ui.pp_excellon_name_cb,
- "ppname_g": self.ui.pp_geo_name_cb,
- "z_pdepth": self.ui.pdepth_entry,
- "feedrate_probe": self.ui.feedrate_probe_entry,
- # "gcode_type": self.ui.excellon_gcode_type_radio
- })
- self.name2option = {
- "e_operation": "operation",
- "e_milling_type": "milling_type",
- "e_milling_dia": "milling_dia",
- "e_cutz": "cutz",
- "e_multidepth": "multidepth",
- "e_depthperpass": "depthperpass",
- "e_travelz": "travelz",
- "e_feedratexy": "feedrate",
- "e_feedratez": "feedrate_z",
- "e_fr_rapid": "feedrate_rapid",
- "e_extracut": "extracut",
- "e_extracut_length": "extracut_length",
- "e_spindlespeed": "spindlespeed",
- "e_dwell": "dwell",
- "e_dwelltime": "dwelltime",
- "e_offset": "offset",
- }
- # populate Excellon preprocessor combobox list
- for name in list(self.app.preprocessors.keys()):
- # the HPGL preprocessor is only for Geometry not for Excellon job therefore don't add it
- if name == 'hpgl':
- continue
- self.ui.pp_excellon_name_cb.addItem(name)
- # populate Geometry (milling) preprocessor combobox list
- for name in list(self.app.preprocessors.keys()):
- self.ui.pp_geo_name_cb.addItem(name)
- # Fill form fields
- self.to_form()
- # update the changes in UI depending on the selected preprocessor in Preferences
- # after this moment all the changes in the Posprocessor combo will be handled by the activated signal of the
- # self.ui.pp_excellon_name_cb combobox
- self.on_pp_changed()
- # Show/Hide Advanced Options
- if self.app.defaults["global_app_level"] == 'b':
- self.ui.level.setText('<span style="color:green;"><b>%s</b></span>' % _('Basic'))
- self.ui.tools_table.setColumnHidden(4, True)
- self.ui.tools_table.setColumnHidden(5, True)
- self.ui.estartz_label.hide()
- self.ui.estartz_entry.hide()
- self.ui.feedrate_rapid_label.hide()
- self.ui.feedrate_rapid_entry.hide()
- self.ui.pdepth_label.hide()
- self.ui.pdepth_entry.hide()
- self.ui.feedrate_probe_label.hide()
- self.ui.feedrate_probe_entry.hide()
- else:
- self.ui.level.setText('<span style="color:red;"><b>%s</b></span>' % _('Advanced'))
- assert isinstance(self.ui, ExcellonObjectUI), \
- "Expected a ExcellonObjectUI, got %s" % type(self.ui)
- self.ui.plot_cb.stateChanged.connect(self.on_plot_cb_click)
- self.ui.solid_cb.stateChanged.connect(self.on_solid_cb_click)
- self.ui.generate_cnc_button.clicked.connect(self.on_create_cncjob_button_click)
- self.ui.generate_milling_button.clicked.connect(self.on_generate_milling_button_click)
- self.ui.generate_milling_slots_button.clicked.connect(self.on_generate_milling_slots_button_click)
- self.on_operation_type(val='drill')
- self.ui.operation_radio.activated_custom.connect(self.on_operation_type)
- self.ui.pp_excellon_name_cb.activated.connect(self.on_pp_changed)
- self.ui.apply_param_to_all.clicked.connect(self.on_apply_param_to_all_clicked)
- self.units_found = self.app.defaults['units']
- # ########################################
- # #######3 TEMP SETTINGS #################
- # ########################################
- self.ui.operation_radio.set_value("drill")
- self.ui.operation_radio.setEnabled(False)
- def ui_connect(self):
- # selective plotting
- for row in range(self.ui.tools_table.rowCount() - 2):
- self.ui.tools_table.cellWidget(row, 5).clicked.connect(self.on_plot_cb_click_table)
- self.ui.plot_cb.stateChanged.connect(self.on_plot_cb_click)
- # rows selected
- self.ui.tools_table.clicked.connect(self.on_row_selection_change)
- self.ui.tools_table.horizontalHeader().sectionClicked.connect(self.on_row_selection_change)
- # value changed in the particular parameters of a tool
- for key, option in self.name2option.items():
- current_widget = self.form_fields[option]
- if isinstance(current_widget, FCCheckBox):
- current_widget.stateChanged.connect(self.form_to_storage)
- if isinstance(current_widget, RadioSet):
- current_widget.activated_custom.connect(self.form_to_storage)
- elif isinstance(current_widget, FCDoubleSpinner) or isinstance(current_widget, FCSpinner):
- current_widget.returnPressed.connect(self.form_to_storage)
- def ui_disconnect(self):
- # selective plotting
- for row in range(self.ui.tools_table.rowCount()):
- try:
- self.ui.tools_table.cellWidget(row, 5).clicked.disconnect()
- except (TypeError, AttributeError):
- pass
- try:
- self.ui.plot_cb.stateChanged.disconnect()
- except (TypeError, AttributeError):
- pass
- # rows selected
- try:
- self.ui.tools_table.clicked.disconnect()
- except (TypeError, AttributeError):
- pass
- try:
- self.ui.tools_table.horizontalHeader().sectionClicked.disconnect()
- except (TypeError, AttributeError):
- pass
- # value changed in the particular parameters of a tool
- for key, option in self.name2option.items():
- current_widget = self.form_fields[option]
- if isinstance(current_widget, FCCheckBox):
- try:
- current_widget.stateChanged.disconnect(self.form_to_storage)
- except (TypeError, ValueError):
- pass
- if isinstance(current_widget, RadioSet):
- try:
- current_widget.activated_custom.disconnect(self.form_to_storage)
- except (TypeError, ValueError):
- pass
- elif isinstance(current_widget, FCDoubleSpinner) or isinstance(current_widget, FCSpinner):
- try:
- current_widget.returnPressed.disconnect(self.form_to_storage)
- except (TypeError, ValueError):
- pass
- def on_row_selection_change(self):
- self.ui_disconnect()
- sel_rows = []
- sel_items = self.ui.tools_table.selectedItems()
- for it in sel_items:
- sel_rows.append(it.row())
- if not sel_rows:
- self.ui.tool_data_label.setText(
- "<b>%s: <font color='#0000FF'>%s</font></b>" % (_('Parameters for'), _("No Tool Selected"))
- )
- self.ui.generate_cnc_button.setDisabled(True)
- self.ui.generate_milling_button.setDisabled(True)
- self.ui.generate_milling_slots_button.setDisabled(True)
- self.ui_connect()
- return
- else:
- self.ui.generate_cnc_button.setDisabled(False)
- self.ui.generate_milling_button.setDisabled(False)
- self.ui.generate_milling_slots_button.setDisabled(False)
- if len(sel_rows) == 1:
- # update the QLabel that shows for which Tool we have the parameters in the UI form
- tooluid = int(self.ui.tools_table.item(sel_rows[0], 0).text())
- self.ui.tool_data_label.setText(
- "<b>%s: <font color='#0000FF'>%s %d</font></b>" % (_('Parameters for'), _("Tool"), tooluid)
- )
- else:
- self.ui.tool_data_label.setText(
- "<b>%s: <font color='#0000FF'>%s</font></b>" % (_('Parameters for'), _("Multiple Tools"))
- )
- for c_row in sel_rows:
- # populate the form with the data from the tool associated with the row parameter
- try:
- item = self.ui.tools_table.item(c_row, 0)
- if type(item) is not None:
- tooluid = item.text()
- self.storage_to_form(self.tools[str(tooluid)]['data'])
- else:
- self.ui_connect()
- return
- except Exception as e:
- log.debug("Tool missing. Add a tool in Geo Tool Table. %s" % str(e))
- self.ui_connect()
- return
- self.ui_connect()
- 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 and form_key not in \
- ["toolchange", "toolchangez", "startz", "endz", "ppname_e", "ppname_g"]:
- try:
- self.form_fields[form_key].set_value(dict_storage[form_key])
- except Exception as e:
- log.debug("ExcellonObject.storage_to_form() --> %s" % str(e))
- pass
- def form_to_storage(self):
- if self.ui.tools_table.rowCount() == 0:
- # there is no tool in tool table so we can't save the GUI elements values to storage
- return
- self.ui_disconnect()
- widget_changed = self.sender()
- wdg_objname = widget_changed.objectName()
- option_changed = self.name2option[wdg_objname]
- # row = self.ui.tools_table.currentRow()
- rows = sorted(set(index.row() for index in self.ui.tools_table.selectedIndexes()))
- for row in rows:
- if row < 0:
- row = 0
- tooluid_item = int(self.ui.tools_table.item(row, 0).text())
- for tooluid_key, tooluid_val in self.tools.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.ui_connect()
- def on_operation_type(self, val):
- if val == 'mill':
- self.ui.mill_type_label.show()
- self.ui.milling_type_radio.show()
- self.ui.mill_dia_label.show()
- self.ui.mill_dia_entry.show()
- self.ui.frxylabel.show()
- self.ui.xyfeedrate_entry.show()
- self.ui.extracut_cb.show()
- self.ui.e_cut_entry.show()
- # if 'laser' not in self.ui.pp_excellon_name_cb.get_value().lower():
- # self.ui.mpass_cb.show()
- # self.ui.maxdepth_entry.show()
- else:
- self.ui.mill_type_label.hide()
- self.ui.milling_type_radio.hide()
- self.ui.mill_dia_label.hide()
- self.ui.mill_dia_entry.hide()
- # self.ui.mpass_cb.hide()
- # self.ui.maxdepth_entry.hide()
- self.ui.frxylabel.hide()
- self.ui.xyfeedrate_entry.hide()
- self.ui.extracut_cb.hide()
- self.ui.e_cut_entry.hide()
- def get_selected_tools_list(self):
- """
- Returns the keys to the self.tools dictionary corresponding
- to the selections on the tool list in the GUI.
- :return: List of tools.
- :rtype: list
- """
- return [str(x.text()) for x in self.ui.tools_table.selectedItems()]
- def get_selected_tools_table_items(self):
- """
- Returns a list of lists, each list in the list is made out of row elements
- :return: List of table_tools items.
- :rtype: list
- """
- table_tools_items = []
- for x in self.ui.tools_table.selectedItems():
- # from the columnCount we subtract a value of 1 which represent the last column (plot column)
- # which does not have text
- txt = ''
- elem = []
- for column in range(0, self.ui.tools_table.columnCount() - 1):
- try:
- txt = self.ui.tools_table.item(x.row(), column).text()
- except AttributeError:
- try:
- txt = self.ui.tools_table.cellWidget(x.row(), column).currentText()
- except AttributeError:
- pass
- elem.append(txt)
- table_tools_items.append(deepcopy(elem))
- # table_tools_items.append([self.ui.tools_table.item(x.row(), column).text()
- # for column in range(0, self.ui.tools_table.columnCount() - 1)])
- for item in table_tools_items:
- item[0] = str(item[0])
- return table_tools_items
- def export_excellon(self, whole, fract, e_zeros=None, form='dec', factor=1, slot_type='routing'):
- """
- Returns two values, first is a boolean , if 1 then the file has slots and second contain the Excellon code
- :return: has_slots and Excellon_code
- """
- excellon_code = ''
- # store here if the file has slots, return 1 if any slots, 0 if only drills
- has_slots = 0
- # drills processing
- try:
- if self.drills:
- length = whole + fract
- for tool in self.tools:
- excellon_code += 'T0%s\n' % str(tool) if int(tool) < 10 else 'T%s\n' % str(tool)
- for drill in self.drills:
- if form == 'dec' and tool == drill['tool']:
- drill_x = drill['point'].x * factor
- drill_y = drill['point'].y * factor
- excellon_code += "X{:.{dec}f}Y{:.{dec}f}\n".format(drill_x, drill_y, dec=fract)
- elif e_zeros == 'LZ' and tool == drill['tool']:
- drill_x = drill['point'].x * factor
- drill_y = drill['point'].y * factor
- exc_x_formatted = "{:.{dec}f}".format(drill_x, dec=fract)
- exc_y_formatted = "{:.{dec}f}".format(drill_y, dec=fract)
- # extract whole part and decimal part
- exc_x_formatted = exc_x_formatted.partition('.')
- exc_y_formatted = exc_y_formatted.partition('.')
- # left padd the 'whole' part with zeros
- x_whole = exc_x_formatted[0].rjust(whole, '0')
- y_whole = exc_y_formatted[0].rjust(whole, '0')
- # restore the coordinate padded in the left with 0 and added the decimal part
- # without the decinal dot
- exc_x_formatted = x_whole + exc_x_formatted[2]
- exc_y_formatted = y_whole + exc_y_formatted[2]
- excellon_code += "X{xform}Y{yform}\n".format(xform=exc_x_formatted,
- yform=exc_y_formatted)
- elif tool == drill['tool']:
- drill_x = drill['point'].x * factor
- drill_y = drill['point'].y * factor
- exc_x_formatted = "{:.{dec}f}".format(drill_x, dec=fract).replace('.', '')
- exc_y_formatted = "{:.{dec}f}".format(drill_y, dec=fract).replace('.', '')
- # pad with rear zeros
- exc_x_formatted.ljust(length, '0')
- exc_y_formatted.ljust(length, '0')
- excellon_code += "X{xform}Y{yform}\n".format(xform=exc_x_formatted,
- yform=exc_y_formatted)
- except Exception as e:
- log.debug(str(e))
- # slots processing
- try:
- if self.slots:
- has_slots = 1
- for tool in self.tools:
- excellon_code += 'G05\n'
- if int(tool) < 10:
- excellon_code += 'T0' + str(tool) + '\n'
- else:
- excellon_code += 'T' + str(tool) + '\n'
- for slot in self.slots:
- if form == 'dec' and tool == slot['tool']:
- start_slot_x = slot['start'].x * factor
- start_slot_y = slot['start'].y * factor
- stop_slot_x = slot['stop'].x * factor
- stop_slot_y = slot['stop'].y * factor
- if slot_type == 'routing':
- excellon_code += "G00X{:.{dec}f}Y{:.{dec}f}\nM15\n".format(start_slot_x,
- start_slot_y,
- dec=fract)
- excellon_code += "G01X{:.{dec}f}Y{:.{dec}f}\nM16\n".format(stop_slot_x,
- stop_slot_y,
- dec=fract)
- elif slot_type == 'drilling':
- excellon_code += "X{:.{dec}f}Y{:.{dec}f}G85X{:.{dec}f}Y{:.{dec}f}\nG05\n".format(
- start_slot_x, start_slot_y, stop_slot_x, stop_slot_y, dec=fract
- )
- elif e_zeros == 'LZ' and tool == slot['tool']:
- start_slot_x = slot['start'].x * factor
- start_slot_y = slot['start'].y * factor
- stop_slot_x = slot['stop'].x * factor
- stop_slot_y = slot['stop'].y * factor
- start_slot_x_formatted = "{:.{dec}f}".format(start_slot_x, dec=fract).replace('.', '')
- start_slot_y_formatted = "{:.{dec}f}".format(start_slot_y, dec=fract).replace('.', '')
- stop_slot_x_formatted = "{:.{dec}f}".format(stop_slot_x, dec=fract).replace('.', '')
- stop_slot_y_formatted = "{:.{dec}f}".format(stop_slot_y, dec=fract).replace('.', '')
- # extract whole part and decimal part
- start_slot_x_formatted = start_slot_x_formatted.partition('.')
- start_slot_y_formatted = start_slot_y_formatted.partition('.')
- stop_slot_x_formatted = stop_slot_x_formatted.partition('.')
- stop_slot_y_formatted = stop_slot_y_formatted.partition('.')
- # left padd the 'whole' part with zeros
- start_x_whole = start_slot_x_formatted[0].rjust(whole, '0')
- start_y_whole = start_slot_y_formatted[0].rjust(whole, '0')
- stop_x_whole = stop_slot_x_formatted[0].rjust(whole, '0')
- stop_y_whole = stop_slot_y_formatted[0].rjust(whole, '0')
- # restore the coordinate padded in the left with 0 and added the decimal part
- # without the decinal dot
- start_slot_x_formatted = start_x_whole + start_slot_x_formatted[2]
- start_slot_y_formatted = start_y_whole + start_slot_y_formatted[2]
- stop_slot_x_formatted = stop_x_whole + stop_slot_x_formatted[2]
- stop_slot_y_formatted = stop_y_whole + stop_slot_y_formatted[2]
- if slot_type == 'routing':
- excellon_code += "G00X{xstart}Y{ystart}\nM15\n".format(xstart=start_slot_x_formatted,
- ystart=start_slot_y_formatted)
- excellon_code += "G01X{xstop}Y{ystop}\nM16\n".format(xstop=stop_slot_x_formatted,
- ystop=stop_slot_y_formatted)
- elif slot_type == 'drilling':
- excellon_code += "{xstart}Y{ystart}G85X{xstop}Y{ystop}\nG05\n".format(
- xstart=start_slot_x_formatted, ystart=start_slot_y_formatted,
- xstop=stop_slot_x_formatted, ystop=stop_slot_y_formatted
- )
- elif tool == slot['tool']:
- start_slot_x = slot['start'].x * factor
- start_slot_y = slot['start'].y * factor
- stop_slot_x = slot['stop'].x * factor
- stop_slot_y = slot['stop'].y * factor
- length = whole + fract
- start_slot_x_formatted = "{:.{dec}f}".format(start_slot_x, dec=fract).replace('.', '')
- start_slot_y_formatted = "{:.{dec}f}".format(start_slot_y, dec=fract).replace('.', '')
- stop_slot_x_formatted = "{:.{dec}f}".format(stop_slot_x, dec=fract).replace('.', '')
- stop_slot_y_formatted = "{:.{dec}f}".format(stop_slot_y, dec=fract).replace('.', '')
- # pad with rear zeros
- start_slot_x_formatted.ljust(length, '0')
- start_slot_y_formatted.ljust(length, '0')
- stop_slot_x_formatted.ljust(length, '0')
- stop_slot_y_formatted.ljust(length, '0')
- if slot_type == 'routing':
- excellon_code += "G00X{xstart}Y{ystart}\nM15\n".format(xstart=start_slot_x_formatted,
- ystart=start_slot_y_formatted)
- excellon_code += "G01X{xstop}Y{ystop}\nM16\n".format(xstop=stop_slot_x_formatted,
- ystop=stop_slot_y_formatted)
- elif slot_type == 'drilling':
- excellon_code += "{xstart}Y{ystart}G85X{xstop}Y{ystop}\nG05\n".format(
- xstart=start_slot_x_formatted, ystart=start_slot_y_formatted,
- xstop=stop_slot_x_formatted, ystop=stop_slot_y_formatted
- )
- except Exception as e:
- log.debug(str(e))
- if not self.drills and not self.slots:
- log.debug("FlatCAMObj.ExcellonObject.export_excellon() --> Excellon Object is empty: no drills, no slots.")
- return 'fail'
- return has_slots, excellon_code
- def generate_milling_drills(self, tools=None, outname=None, tooldia=None, plot=False, use_thread=False):
- """
- Note: This method is a good template for generic operations as
- it takes it's options from parameters or otherwise from the
- object's options and returns a (success, msg) tuple as feedback
- for shell operations.
- :return: Success/failure condition tuple (bool, str).
- :rtype: tuple
- """
- # Get the tools from the list. These are keys
- # to self.tools
- if tools is None:
- tools = self.get_selected_tools_list()
- if outname is None:
- outname = self.options["name"] + "_mill"
- if tooldia is None:
- tooldia = float(self.options["tooldia"])
- # Sort tools by diameter. items() -> [('name', diameter), ...]
- # sorted_tools = sorted(list(self.tools.items()), key=lambda tl: tl[1]) # no longer works in Python3
- sort = []
- for k, v in self.tools.items():
- sort.append((k, v.get('C')))
- sorted_tools = sorted(sort, key=lambda t1: t1[1])
- if tools == "all":
- tools = [i[0] for i in sorted_tools] # List if ordered tool names.
- log.debug("Tools 'all' and sorted are: %s" % str(tools))
- if len(tools) == 0:
- self.app.inform.emit('[ERROR_NOTCL] %s' %
- _("Please select one or more tools from the list and try again."))
- return False, "Error: No tools."
- for tool in tools:
- if tooldia > self.tools[tool]["C"]:
- self.app.inform.emit(
- '[ERROR_NOTCL] %s %s: %s' % (
- _("Milling tool for DRILLS is larger than hole size. Cancelled."),
- _("Tool"),
- str(tool)
- )
- )
- return False, "Error: Milling tool is larger than hole."
- def geo_init(geo_obj, app_obj):
- assert geo_obj.kind == 'geometry', "Initializer expected a GeometryObject, got %s" % type(geo_obj)
- # ## Add properties to the object
- # get the tool_table items in a list of row items
- tool_table_items = self.get_selected_tools_table_items()
- # insert an information only element in the front
- tool_table_items.insert(0, [_("Tool_nr"), _("Diameter"), _("Drills_Nr"), _("Slots_Nr")])
- geo_obj.options['Tools_in_use'] = tool_table_items
- geo_obj.options['type'] = 'Excellon Geometry'
- geo_obj.options["cnctooldia"] = str(tooldia)
- geo_obj.solid_geometry = []
- # in case that the tool used has the same diameter with the hole, and since the maximum resolution
- # for FlatCAM is 6 decimals,
- # we add a tenth of the minimum value, meaning 0.0000001, which from our point of view is "almost zero"
- for hole in self.drills:
- if hole['tool'] in tools:
- buffer_value = self.tools[hole['tool']]["C"] / 2 - tooldia / 2
- if buffer_value == 0:
- geo_obj.solid_geometry.append(
- Point(hole['point']).buffer(0.0000001).exterior)
- else:
- geo_obj.solid_geometry.append(
- Point(hole['point']).buffer(buffer_value).exterior)
- if use_thread:
- def geo_thread(app_obj):
- app_obj.new_object("geometry", outname, geo_init, plot=plot)
- # Create a promise with the new name
- self.app.collection.promise(outname)
- # Send to worker
- self.app.worker_task.emit({'fcn': geo_thread, 'params': [self.app]})
- else:
- self.app.new_object("geometry", outname, geo_init, plot=plot)
- return True, ""
- def generate_milling_slots(self, tools=None, outname=None, tooldia=None, plot=True, use_thread=False):
- """
- Note: This method is a good template for generic operations as
- it takes it's options from parameters or otherwise from the
- object's options and returns a (success, msg) tuple as feedback
- for shell operations.
- :return: Success/failure condition tuple (bool, str).
- :rtype: tuple
- """
- # Get the tools from the list. These are keys
- # to self.tools
- if tools is None:
- tools = self.get_selected_tools_list()
- if outname is None:
- outname = self.options["name"] + "_mill"
- if tooldia is None:
- tooldia = float(self.options["slot_tooldia"])
- # Sort tools by diameter. items() -> [('name', diameter), ...]
- # sorted_tools = sorted(list(self.tools.items()), key=lambda tl: tl[1]) # no longer works in Python3
- sort = []
- for k, v in self.tools.items():
- sort.append((k, v.get('C')))
- sorted_tools = sorted(sort, key=lambda t1: t1[1])
- if tools == "all":
- tools = [i[0] for i in sorted_tools] # List if ordered tool names.
- log.debug("Tools 'all' and sorted are: %s" % str(tools))
- if len(tools) == 0:
- self.app.inform.emit('[ERROR_NOTCL] %s' %
- _("Please select one or more tools from the list and try again."))
- return False, "Error: No tools."
- for tool in tools:
- # I add the 0.0001 value to account for the rounding error in converting from IN to MM and reverse
- adj_toolstable_tooldia = float('%.*f' % (self.decimals, float(tooldia)))
- adj_file_tooldia = float('%.*f' % (self.decimals, float(self.tools[tool]["C"])))
- if adj_toolstable_tooldia > adj_file_tooldia + 0.0001:
- self.app.inform.emit('[ERROR_NOTCL] %s' %
- _("Milling tool for SLOTS is larger than hole size. Cancelled."))
- return False, "Error: Milling tool is larger than hole."
- def geo_init(geo_obj, app_obj):
- assert geo_obj.kind == 'geometry' "Initializer expected a GeometryObject, got %s" % type(geo_obj)
- # ## Add properties to the object
- # get the tool_table items in a list of row items
- tool_table_items = self.get_selected_tools_table_items()
- # insert an information only element in the front
- tool_table_items.insert(0, [_("Tool_nr"), _("Diameter"), _("Drills_Nr"), _("Slots_Nr")])
- geo_obj.options['Tools_in_use'] = tool_table_items
- geo_obj.options['type'] = 'Excellon Geometry'
- geo_obj.options["cnctooldia"] = str(tooldia)
- geo_obj.solid_geometry = []
- # in case that the tool used has the same diameter with the hole, and since the maximum resolution
- # for FlatCAM is 6 decimals,
- # we add a tenth of the minimum value, meaning 0.0000001, which from our point of view is "almost zero"
- for slot in self.slots:
- if slot['tool'] in tools:
- toolstable_tool = float('%.*f' % (self.decimals, float(tooldia)))
- file_tool = float('%.*f' % (self.decimals, float(self.tools[tool]["C"])))
- # I add the 0.0001 value to account for the rounding error in converting from IN to MM and reverse
- # for the file_tool (tooldia actually)
- buffer_value = float(file_tool / 2) - float(toolstable_tool / 2) + 0.0001
- if buffer_value == 0:
- start = slot['start']
- stop = slot['stop']
- lines_string = LineString([start, stop])
- poly = lines_string.buffer(0.0000001, int(self.geo_steps_per_circle)).exterior
- geo_obj.solid_geometry.append(poly)
- else:
- start = slot['start']
- stop = slot['stop']
- lines_string = LineString([start, stop])
- poly = lines_string.buffer(buffer_value, int(self.geo_steps_per_circle)).exterior
- geo_obj.solid_geometry.append(poly)
- if use_thread:
- def geo_thread(app_obj):
- app_obj.new_object("geometry", outname + '_slot', geo_init, plot=plot)
- # Create a promise with the new name
- self.app.collection.promise(outname)
- # Send to worker
- self.app.worker_task.emit({'fcn': geo_thread, 'params': [self.app]})
- else:
- self.app.new_object("geometry", outname + '_slot', geo_init, plot=plot)
- return True, ""
- def on_generate_milling_button_click(self, *args):
- self.app.report_usage("excellon_on_create_milling_drills button")
- self.read_form()
- self.generate_milling_drills(use_thread=False)
- def on_generate_milling_slots_button_click(self, *args):
- self.app.report_usage("excellon_on_create_milling_slots_button")
- self.read_form()
- self.generate_milling_slots(use_thread=False)
- def on_pp_changed(self):
- current_pp = self.ui.pp_excellon_name_cb.get_value()
- if "toolchange_probe" in current_pp.lower():
- self.ui.pdepth_entry.setVisible(True)
- self.ui.pdepth_label.show()
- self.ui.feedrate_probe_entry.setVisible(True)
- self.ui.feedrate_probe_label.show()
- else:
- self.ui.pdepth_entry.setVisible(False)
- self.ui.pdepth_label.hide()
- self.ui.feedrate_probe_entry.setVisible(False)
- self.ui.feedrate_probe_label.hide()
- if 'marlin' in current_pp.lower() or 'custom' in current_pp.lower():
- self.ui.feedrate_rapid_label.show()
- self.ui.feedrate_rapid_entry.show()
- else:
- self.ui.feedrate_rapid_label.hide()
- self.ui.feedrate_rapid_entry.hide()
- if 'laser' in current_pp.lower():
- self.ui.cutzlabel.hide()
- self.ui.cutz_entry.hide()
- try:
- self.ui.mpass_cb.hide()
- self.ui.maxdepth_entry.hide()
- except AttributeError:
- pass
- if 'marlin' in current_pp.lower():
- self.ui.travelzlabel.setText('%s:' % _("Focus Z"))
- self.ui.endz_label.show()
- self.ui.endz_entry.show()
- else:
- self.ui.travelzlabel.hide()
- self.ui.travelz_entry.hide()
- self.ui.endz_label.hide()
- self.ui.endz_entry.hide()
- try:
- self.ui.frzlabel.hide()
- self.ui.feedrate_z_entry.hide()
- except AttributeError:
- pass
- self.ui.dwell_cb.hide()
- self.ui.dwelltime_entry.hide()
- self.ui.spindle_label.setText('%s:' % _("Laser Power"))
- try:
- self.ui.tool_offset_label.hide()
- self.ui.offset_entry.hide()
- except AttributeError:
- pass
- else:
- self.ui.cutzlabel.show()
- self.ui.cutz_entry.show()
- try:
- self.ui.mpass_cb.show()
- self.ui.maxdepth_entry.show()
- except AttributeError:
- pass
- self.ui.travelzlabel.setText('%s:' % _('Travel Z'))
- self.ui.travelzlabel.show()
- self.ui.travelz_entry.show()
- self.ui.endz_label.show()
- self.ui.endz_entry.show()
- try:
- self.ui.frzlabel.show()
- self.ui.feedrate_z_entry.show()
- except AttributeError:
- pass
- self.ui.dwell_cb.show()
- self.ui.dwelltime_entry.show()
- self.ui.spindle_label.setText('%s:' % _('Spindle speed'))
- try:
- self.ui.tool_offset_lbl.show()
- self.ui.offset_entry.show()
- except AttributeError:
- pass
- def on_create_cncjob_button_click(self, *args):
- self.app.report_usage("excellon_on_create_cncjob_button")
- self.read_form()
- # Get the tools from the list
- tools = self.get_selected_tools_list()
- if len(tools) == 0:
- # if there is a single tool in the table (remember that the last 2 rows are for totals and do not count in
- # tool number) it means that there are 3 rows (1 tool and 2 totals).
- # in this case regardless of the selection status of that tool, use it.
- if self.ui.tools_table.rowCount() == 3:
- tools.append(self.ui.tools_table.item(0, 0).text())
- else:
- self.app.inform.emit('[ERROR_NOTCL] %s' %
- _("Please select one or more tools from the list and try again."))
- return
- xmin = self.options['xmin']
- ymin = self.options['ymin']
- xmax = self.options['xmax']
- ymax = self.options['ymax']
- job_name = self.options["name"] + "_cnc"
- pp_excellon_name = self.options["ppname_e"]
- # Object initialization function for app.new_object()
- def job_init(job_obj, app_obj):
- assert job_obj.kind == 'cncjob', "Initializer expected a CNCJobObject, got %s" % type(job_obj)
- # get the tool_table items in a list of row items
- tool_table_items = self.get_selected_tools_table_items()
- # insert an information only element in the front
- tool_table_items.insert(0, [_("Tool_nr"), _("Diameter"), _("Drills_Nr"), _("Slots_Nr")])
- # ## Add properties to the object
- job_obj.origin_kind = 'excellon'
- job_obj.options['Tools_in_use'] = tool_table_items
- job_obj.options['type'] = 'Excellon'
- job_obj.options['ppname_e'] = pp_excellon_name
- job_obj.multidepth = self.options["multidepth"]
- job_obj.z_depthpercut = self.options["depthperpass"]
- job_obj.z_move = float(self.options["travelz"])
- job_obj.feedrate = float(self.options["feedrate_z"])
- job_obj.z_feedrate = float(self.options["feedrate_z"])
- job_obj.feedrate_rapid = float(self.options["feedrate_rapid"])
- job_obj.spindlespeed = float(self.options["spindlespeed"]) if self.options["spindlespeed"] != 0 else None
- job_obj.spindledir = self.app.defaults['excellon_spindledir']
- job_obj.dwell = self.options["dwell"]
- job_obj.dwelltime = float(self.options["dwelltime"])
- job_obj.pp_excellon_name = pp_excellon_name
- job_obj.toolchange_xy_type = "excellon"
- job_obj.coords_decimals = int(self.app.defaults["cncjob_coords_decimals"])
- job_obj.fr_decimals = int(self.app.defaults["cncjob_fr_decimals"])
- job_obj.options['xmin'] = xmin
- job_obj.options['ymin'] = ymin
- job_obj.options['xmax'] = xmax
- job_obj.options['ymax'] = ymax
- job_obj.z_pdepth = float(self.options["z_pdepth"])
- job_obj.feedrate_probe = float(self.options["feedrate_probe"])
- job_obj.z_cut = float(self.options['cutz'])
- job_obj.toolchange = self.options["toolchange"]
- job_obj.xy_toolchange = self.app.defaults["excellon_toolchangexy"]
- job_obj.z_toolchange = float(self.options["toolchangez"])
- job_obj.startz = float(self.options["startz"]) if self.options["startz"] else None
- job_obj.endz = float(self.options["endz"])
- job_obj.xy_end = self.options["endxy"]
- job_obj.excellon_optimization_type = self.app.defaults["excellon_optimization_type"]
- tools_csv = ','.join(tools)
- ret_val = job_obj.generate_from_excellon_by_tool(self, tools_csv, use_ui=True)
- if ret_val == 'fail':
- return 'fail'
- job_obj.gcode_parse()
- job_obj.create_geometry()
- # To be run in separate thread
- def job_thread(app_obj):
- with self.app.proc_container.new(_("Generating CNC Code")):
- app_obj.new_object("cncjob", job_name, job_init)
- # Create promise for the new name.
- self.app.collection.promise(job_name)
- # Send to worker
- # self.app.worker.add_task(job_thread, [self.app])
- self.app.worker_task.emit({'fcn': job_thread, 'params': [self.app]})
- def convert_units(self, units):
- log.debug("FlatCAMObj.ExcellonObject.convert_units()")
- Excellon.convert_units(self, units)
- # factor = Excellon.convert_units(self, units)
- # self.options['drillz'] = float(self.options['drillz']) * factor
- # self.options['travelz'] = float(self.options['travelz']) * factor
- # self.options['feedrate'] = float(self.options['feedrate']) * factor
- # self.options['feedrate_rapid'] = float(self.options['feedrate_rapid']) * factor
- # self.options['toolchangez'] = float(self.options['toolchangez']) * factor
- #
- # if self.app.defaults["excellon_toolchangexy"] == '':
- # self.options['toolchangexy'] = "0.0, 0.0"
- # else:
- # coords_xy = [float(eval(coord)) for coord in self.app.defaults["excellon_toolchangexy"].split(",")]
- # if len(coords_xy) < 2:
- # self.app.inform.emit('[ERROR] %s' % _("The Toolchange X,Y field in Edit -> Preferences has to be "
- # "in the format (x, y) \n"
- # "but now there is only one value, not two. "))
- # return 'fail'
- # coords_xy[0] *= factor
- # coords_xy[1] *= factor
- # self.options['toolchangexy'] = "%f, %f" % (coords_xy[0], coords_xy[1])
- #
- # if self.options['startz'] is not None:
- # self.options['startz'] = float(self.options['startz']) * factor
- # self.options['endz'] = float(self.options['endz']) * factor
- def on_solid_cb_click(self, *args):
- if self.muted_ui:
- return
- self.read_form_item('solid')
- self.plot()
- def on_plot_cb_click(self, *args):
- if self.muted_ui:
- return
- self.plot()
- self.read_form_item('plot')
- self.ui_disconnect()
- cb_flag = self.ui.plot_cb.isChecked()
- for row in range(self.ui.tools_table.rowCount() - 2):
- table_cb = self.ui.tools_table.cellWidget(row, 5)
- if cb_flag:
- table_cb.setChecked(True)
- else:
- table_cb.setChecked(False)
- self.ui_connect()
- def on_plot_cb_click_table(self):
- # self.ui.cnc_tools_table.cellWidget(row, 2).widget().setCheckState(QtCore.Qt.Unchecked)
- self.ui_disconnect()
- # cw = self.sender()
- # cw_index = self.ui.tools_table.indexAt(cw.pos())
- # cw_row = cw_index.row()
- check_row = 0
- self.shapes.clear(update=True)
- for tool_key in self.tools:
- solid_geometry = self.tools[tool_key]['solid_geometry']
- # find the geo_tool_table row associated with the tool_key
- for row in range(self.ui.tools_table.rowCount()):
- tool_item = int(self.ui.tools_table.item(row, 0).text())
- if tool_item == int(tool_key):
- check_row = row
- break
- if self.ui.tools_table.cellWidget(check_row, 5).isChecked():
- self.options['plot'] = True
- # self.plot_element(element=solid_geometry, visible=True)
- # Plot excellon (All polygons?)
- if self.options["solid"]:
- for geo in solid_geometry:
- self.add_shape(shape=geo, color='#750000BF', face_color='#C40000BF',
- visible=self.options['plot'],
- layer=2)
- else:
- for geo in solid_geometry:
- self.add_shape(shape=geo.exterior, color='red', visible=self.options['plot'])
- for ints in geo.interiors:
- self.add_shape(shape=ints, color='green', visible=self.options['plot'])
- self.shapes.redraw()
- # make sure that the general plot is disabled if one of the row plot's are disabled and
- # if all the row plot's are enabled also enable the general plot checkbox
- cb_cnt = 0
- total_row = self.ui.tools_table.rowCount()
- for row in range(total_row - 2):
- if self.ui.tools_table.cellWidget(row, 5).isChecked():
- cb_cnt += 1
- else:
- cb_cnt -= 1
- if cb_cnt < total_row - 2:
- self.ui.plot_cb.setChecked(False)
- else:
- self.ui.plot_cb.setChecked(True)
- self.ui_connect()
- def plot(self, visible=None, kind=None):
- # Does all the required setup and returns False
- # if the 'ptint' option is set to False.
- if not FlatCAMObj.plot(self):
- return
- # try:
- # # Plot Excellon (All polygons?)
- # if self.options["solid"]:
- # for tool in self.tools:
- # for geo in self.tools[tool]['solid_geometry']:
- # self.add_shape(shape=geo, color='#750000BF', face_color='#C40000BF',
- # visible=self.options['plot'],
- # layer=2)
- # else:
- # for tool in self.tools:
- # for geo in self.tools[tool]['solid_geometry']:
- # self.add_shape(shape=geo.exterior, color='red', visible=self.options['plot'])
- # for ints in geo.interiors:
- # self.add_shape(shape=ints, color='orange', visible=self.options['plot'])
- #
- # self.shapes.redraw()
- # return
- # except (ObjectDeleted, AttributeError, KeyError):
- # self.shapes.clear(update=True)
- # this stays for compatibility reasons, in case we try to open old projects
- try:
- __ = iter(self.solid_geometry)
- except TypeError:
- self.solid_geometry = [self.solid_geometry]
- visible = visible if visible else self.options['plot']
- try:
- # Plot Excellon (All polygons?)
- if self.options["solid"]:
- for geo in self.solid_geometry:
- self.add_shape(shape=geo,
- color=self.outline_color,
- face_color=self.fill_color,
- visible=visible,
- layer=2)
- else:
- for geo in self.solid_geometry:
- self.add_shape(shape=geo.exterior, color='red', visible=visible)
- for ints in geo.interiors:
- self.add_shape(shape=ints, color='orange', visible=visible)
- self.shapes.redraw()
- except (ObjectDeleted, AttributeError):
- self.shapes.clear(update=True)
- def on_apply_param_to_all_clicked(self):
- if self.ui.tools_table.rowCount() == 0:
- # there is no tool in tool table so we can't save the GUI elements values to storage
- log.debug("ExcellonObject.on_apply_param_to_all_clicked() --> no tool in Tools Table, aborting.")
- return
- self.ui_disconnect()
- row = self.ui.tools_table.currentRow()
- if row < 0:
- row = 0
- tooluid_item = int(self.ui.tools_table.item(row, 0).text())
- temp_tool_data = {}
- for tooluid_key, tooluid_val in self.tools.items():
- if int(tooluid_key) == tooluid_item:
- # this will hold the 'data' key of the self.tools[tool] dictionary that corresponds to
- # the current row in the tool table
- temp_tool_data = tooluid_val['data']
- break
- for tooluid_key, tooluid_val in self.tools.items():
- tooluid_val['data'] = deepcopy(temp_tool_data)
- self.app.inform.emit('[success] %s' % _("Current Tool parameters were applied to all tools."))
- self.ui_connect()
|