| |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| import FreeCAD |
| import Path |
| import math |
| import Path.Base.Gui.Util as PathGuiUtil |
| import PathScripts.PathUtils as PathUtils |
| import Path.Dressup.Utils as PathDressup |
| from PySide.QtCore import QT_TRANSLATE_NOOP |
|
|
| if False: |
| Path.Log.setLevel(Path.Log.Level.DEBUG, Path.Log.thisModule()) |
| Path.Log.trackModule(Path.Log.thisModule()) |
| else: |
| Path.Log.setLevel(Path.Log.Level.INFO, Path.Log.thisModule()) |
|
|
|
|
| if FreeCAD.GuiUp: |
| import FreeCADGui |
|
|
| __doc__ = """Axis remapping Dressup object and FreeCAD command. This dressup remaps one axis of motion to another. |
| For example, you can re-map the Y axis to A to control a 4th axis rotary.""" |
|
|
|
|
| translate = FreeCAD.Qt.translate |
|
|
|
|
| class ObjectDressup: |
| def __init__(self, obj): |
| maplist = ["X->A", "Y->A", "X->B", "Y->B", "X->C", "Y->C"] |
| obj.addProperty( |
| "App::PropertyLink", |
| "Base", |
| "Path", |
| QT_TRANSLATE_NOOP("App::Property", "The base path to modify"), |
| ) |
| obj.addProperty( |
| "App::PropertyEnumeration", |
| "AxisMap", |
| "Path", |
| QT_TRANSLATE_NOOP("App::Property", "The input mapping axis"), |
| ) |
| obj.addProperty( |
| "App::PropertyDistance", |
| "Radius", |
| "Path", |
| QT_TRANSLATE_NOOP("App::Property", "The radius of the wrapped axis"), |
| ) |
| obj.AxisMap = maplist |
| obj.AxisMap = "Y->A" |
| obj.Proxy = self |
|
|
| def dumps(self): |
| return None |
|
|
| def loads(self, state): |
| return None |
|
|
| def _linear2angular(self, radius, length): |
| """returns an angular distance in degrees to achieve a linear move of a given length""" |
| circum = 2 * math.pi * float(radius) |
| return 360 * (float(length) / circum) |
|
|
| def _stripArcs(self, path, d): |
| """converts all G2/G3 commands into G1 commands""" |
| newcommandlist = [] |
| currLocation = {"X": 0, "Y": 0, "Z": 0, "F": 0} |
|
|
| for p in path: |
| if p.Name in Path.Geom.CmdMoveArc: |
| curVec = FreeCAD.Vector(currLocation["X"], currLocation["Y"], currLocation["Z"]) |
| arcwire = Path.Geom.edgeForCmd(p, curVec) |
| pointlist = arcwire.discretize(Deflection=d) |
| for point in pointlist: |
| newcommand = Path.Command("G1", {"X": point.x, "Y": point.y, "Z": point.z}) |
| newcommandlist.append(newcommand) |
| currLocation.update(newcommand.Parameters) |
| else: |
| newcommandlist.append(p) |
| currLocation.update(p.Parameters) |
|
|
| return newcommandlist |
|
|
| def execute(self, obj): |
|
|
| inAxis = obj.AxisMap[0] |
| outAxis = obj.AxisMap[3] |
| d = 0.1 |
|
|
| if obj.Base: |
| if obj.Base.isDerivedFrom("Path::Feature"): |
| if obj.Base.Path: |
| if obj.Base.Path.Commands: |
| pp = PathUtils.getPathWithPlacement(obj.Base).Commands |
| if len([i for i in pp if i.Name in Path.Geom.CmdMoveArc]) == 0: |
| pathlist = pp |
| else: |
| pathlist = self._stripArcs(pp, d) |
|
|
| newcommandlist = [] |
| currLocation = {"X": 0, "Y": 0, "Z": 0, "F": 0} |
|
|
| for c in pathlist: |
| newparams = dict(c.Parameters) |
| remapvar = newparams.pop(inAxis, None) |
| if remapvar is not None: |
| newparams[outAxis] = self._linear2angular(obj.Radius, remapvar) |
| locdiff = dict(set(newparams.items()) - set(currLocation.items())) |
| if ( |
| len(locdiff) == 1 and outAxis in locdiff |
| ): |
| if "F" in c.Parameters: |
| feed = c.Parameters["F"] |
| else: |
| feed = currLocation["F"] |
| newparams.update({"F": self._linear2angular(obj.Radius, feed)}) |
| newcommand = Path.Command(c.Name, newparams) |
| newcommandlist.append(newcommand) |
| currLocation.update(newparams) |
| else: |
| newcommandlist.append(c) |
| currLocation.update(c.Parameters) |
|
|
| path = Path.Path(newcommandlist) |
| path.Center = self.center(obj) |
| obj.Path = path |
|
|
| def onChanged(self, obj, prop): |
| if "Restore" not in obj.State and prop == "Radius": |
| job = PathUtils.findParentJob(obj) |
| if job: |
| job.Proxy.setCenterOfRotation(self.center(obj)) |
| if prop == "Path" and obj.ViewObject: |
| obj.ViewObject.signalChangeIcon() |
|
|
| def center(self, obj): |
| return FreeCAD.Vector(0, 0, 0 - obj.Radius.Value) |
|
|
|
|
| class TaskPanel: |
| def __init__(self, obj): |
| self.obj = obj |
| self.form = FreeCADGui.PySideUic.loadUi(":/panels/AxisMapEdit.ui") |
| self.radius = PathGuiUtil.QuantitySpinBox(self.form.radius, obj, "Radius") |
| FreeCAD.ActiveDocument.openTransaction("Edit Dragknife Dress-up") |
|
|
| def reject(self): |
| FreeCAD.ActiveDocument.abortTransaction() |
| FreeCADGui.Control.closeDialog() |
| FreeCAD.ActiveDocument.recompute() |
|
|
| def accept(self): |
| self.getFields() |
| FreeCAD.ActiveDocument.commitTransaction() |
| FreeCADGui.ActiveDocument.resetEdit() |
| FreeCADGui.Control.closeDialog() |
| FreeCAD.ActiveDocument.recompute() |
|
|
| def getFields(self): |
| self.radius.updateProperty() |
| self.obj.AxisMap = self.form.axisMapInput.currentText() |
| self.obj.Proxy.execute(self.obj) |
|
|
| def updateUI(self): |
| self.radius.updateWidget() |
| self.form.axisMapInput.setCurrentText(self.obj.AxisMap) |
| self.updateModel() |
|
|
| def updateModel(self): |
| self.getFields() |
| FreeCAD.ActiveDocument.recompute() |
|
|
| def setFields(self): |
| self.updateUI() |
|
|
| def open(self): |
| pass |
|
|
| def setupUi(self): |
| self.setFields() |
| self.form.radius.valueChanged.connect(self.updateModel) |
| self.form.axisMapInput.currentIndexChanged.connect(self.updateModel) |
|
|
|
|
| class ViewProviderDressup: |
| def __init__(self, vobj): |
| self.obj = vobj.Object |
|
|
| def attach(self, vobj): |
| self.obj = vobj.Object |
| if self.obj and self.obj.Base: |
| for i in self.obj.Base.InList: |
| if hasattr(i, "Group"): |
| group = i.Group |
| for g in group: |
| if g.Name == self.obj.Base.Name: |
| group.remove(g) |
| i.Group = group |
| |
| return |
|
|
| def unsetEdit(self, vobj, mode=0): |
| return False |
|
|
| def setEdit(self, vobj, mode=0): |
| FreeCADGui.Control.closeDialog() |
| panel = TaskPanel(vobj.Object) |
| FreeCADGui.Control.showDialog(panel) |
| panel.setupUi() |
| return True |
|
|
| def claimChildren(self): |
| return [self.obj.Base] |
|
|
| def dumps(self): |
| return None |
|
|
| def loads(self, state): |
| return None |
|
|
| def onDelete(self, arg1=None, arg2=None): |
| """this makes sure that the base operation is added back to the project and visible""" |
| if arg1.Object and arg1.Object.Base: |
| FreeCADGui.ActiveDocument.getObject(arg1.Object.Base.Name).Visibility = True |
| job = PathUtils.findParentJob(arg1.Object) |
| if job: |
| job.Proxy.addOperation(arg1.Object.Base, arg1.Object) |
| arg1.Object.Base = None |
| return True |
|
|
| def getIcon(self): |
| if getattr(PathDressup.baseOp(self.obj), "Active", True): |
| return ":/icons/CAM_Dressup.svg" |
| else: |
| return ":/icons/CAM_OpActive.svg" |
|
|
|
|
| class CommandPathDressup: |
| def GetResources(self): |
| return { |
| "Pixmap": "CAM_Dressup", |
| "MenuText": QT_TRANSLATE_NOOP("CAM_DressupAxisMap", "Axis Map"), |
| "Accel": "", |
| "ToolTip": QT_TRANSLATE_NOOP("CAM_DressupAxisMap", "Remaps one axis to another"), |
| } |
|
|
| def IsActive(self): |
| if FreeCAD.ActiveDocument is not None: |
| for o in FreeCAD.ActiveDocument.Objects: |
| if o.Name[:3] == "Job": |
| return True |
| return False |
|
|
| def Activated(self): |
|
|
| |
| selection = FreeCADGui.Selection.getSelection() |
| if len(selection) != 1: |
| FreeCAD.Console.PrintError(translate("CAM_Dressup", "Select one toolpath object\n")) |
| return |
| if not selection[0].isDerivedFrom("Path::Feature"): |
| FreeCAD.Console.PrintError( |
| translate("CAM_Dressup", "The selected object is not a toolpath\n") |
| ) |
| return |
| if selection[0].isDerivedFrom("Path::FeatureCompoundPython"): |
| FreeCAD.Console.PrintError(translate("CAM_Dressup", "Select a toolpath object")) |
| return |
|
|
| |
| FreeCAD.ActiveDocument.openTransaction("Create Dress-up") |
| FreeCADGui.addModule("Path.Dressup.Gui.AxisMap") |
| FreeCADGui.addModule("PathScripts.PathUtils") |
| FreeCADGui.doCommand( |
| 'obj = FreeCAD.ActiveDocument.addObject("Path::FeaturePython", "AxisMapDressup")' |
| ) |
| FreeCADGui.doCommand("Path.Dressup.Gui.AxisMap.ObjectDressup(obj)") |
| FreeCADGui.doCommand("base = FreeCAD.ActiveDocument." + selection[0].Name) |
| FreeCADGui.doCommand("job = PathScripts.PathUtils.findParentJob(base)") |
| FreeCADGui.doCommand("obj.Base = base") |
| FreeCADGui.doCommand("obj.Radius = 45") |
| FreeCADGui.doCommand("job.Proxy.addOperation(obj, base)") |
| FreeCADGui.doCommand( |
| "obj.ViewObject.Proxy = Path.Dressup.Gui.AxisMap.ViewProviderDressup(obj.ViewObject)" |
| ) |
| FreeCADGui.doCommand("Gui.ActiveDocument.getObject(base.Name).Visibility = False") |
| FreeCADGui.doCommand("obj.ViewObject.Document.setEdit(obj.ViewObject, 0)") |
| |
| FreeCAD.ActiveDocument.recompute() |
|
|
|
|
| if FreeCAD.GuiUp: |
| |
| FreeCADGui.addCommand("CAM_DressupAxisMap", CommandPathDressup()) |
|
|
| FreeCAD.Console.PrintLog("Loading PathDressup… done\n") |
|
|