| | |
| |
|
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| |
|
| | 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") |
| |
|