Assembly: Introduce 'Exploded Views'
This commit is contained in:
committed by
Yorik van Havre
parent
6a834422e7
commit
687843ff41
@@ -29,6 +29,7 @@
|
||||
|
||||
#include "AssemblyObject.h"
|
||||
#include "JointGroup.h"
|
||||
#include "ViewGroup.h"
|
||||
|
||||
|
||||
namespace Assembly
|
||||
@@ -58,6 +59,7 @@ PyMOD_INIT_FUNC(AssemblyApp)
|
||||
|
||||
Assembly::AssemblyObject ::init();
|
||||
Assembly::JointGroup ::init();
|
||||
Assembly::ViewGroup ::init();
|
||||
|
||||
PyMOD_Return(mod);
|
||||
}
|
||||
|
||||
@@ -19,12 +19,15 @@ set(Assembly_LIBS
|
||||
|
||||
generate_from_xml(AssemblyObjectPy)
|
||||
generate_from_xml(JointGroupPy)
|
||||
generate_from_xml(ViewGroupPy)
|
||||
|
||||
SET(Python_SRCS
|
||||
AssemblyObjectPy.xml
|
||||
AssemblyObjectPyImp.cpp
|
||||
JointGroupPy.xml
|
||||
JointGroupPyImp.cpp
|
||||
ViewGroupPy.xml
|
||||
ViewGroupPyImp.cpp
|
||||
)
|
||||
SOURCE_GROUP("Python" FILES ${Python_SRCS})
|
||||
|
||||
@@ -41,6 +44,8 @@ SET(Assembly_SRCS
|
||||
AssemblyObject.h
|
||||
JointGroup.cpp
|
||||
JointGroup.h
|
||||
ViewGroup.cpp
|
||||
ViewGroup.h
|
||||
${Module_SRCS}
|
||||
${Python_SRCS}
|
||||
)
|
||||
|
||||
55
src/Mod/Assembly/App/ViewGroup.cpp
Normal file
55
src/Mod/Assembly/App/ViewGroup.cpp
Normal file
@@ -0,0 +1,55 @@
|
||||
// SPDX-License-Identifier: LGPL-2.1-or-later
|
||||
/****************************************************************************
|
||||
* *
|
||||
* Copyright (c) 2023 Ondsel <development@ondsel.com> *
|
||||
* *
|
||||
* This file is part of FreeCAD. *
|
||||
* *
|
||||
* FreeCAD is free software: you can redistribute it and/or modify it *
|
||||
* under the terms of the GNU Lesser General Public License as *
|
||||
* published by the Free Software Foundation, either version 2.1 of the *
|
||||
* License, or (at your option) any later version. *
|
||||
* *
|
||||
* FreeCAD is distributed in the hope that it will be useful, but *
|
||||
* WITHOUT ANY WARRANTY; without even the implied warranty of *
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU *
|
||||
* Lesser General Public License for more details. *
|
||||
* *
|
||||
* You should have received a copy of the GNU Lesser General Public *
|
||||
* License along with FreeCAD. If not, see *
|
||||
* <https://www.gnu.org/licenses/>. *
|
||||
* *
|
||||
***************************************************************************/
|
||||
|
||||
#include "PreCompiled.h"
|
||||
#ifndef _PreComp_
|
||||
#endif
|
||||
|
||||
#include <App/Application.h>
|
||||
#include <App/Document.h>
|
||||
#include <App/FeaturePythonPyImp.h>
|
||||
#include <App/PropertyPythonObject.h>
|
||||
#include <Base/Console.h>
|
||||
#include <Base/Tools.h>
|
||||
|
||||
#include "ViewGroup.h"
|
||||
#include "ViewGroupPy.h"
|
||||
|
||||
using namespace Assembly;
|
||||
|
||||
|
||||
PROPERTY_SOURCE(Assembly::ViewGroup, App::DocumentObjectGroup)
|
||||
|
||||
ViewGroup::ViewGroup()
|
||||
{}
|
||||
|
||||
ViewGroup::~ViewGroup() = default;
|
||||
|
||||
PyObject* ViewGroup::getPyObject()
|
||||
{
|
||||
if (PythonObject.is(Py::_None())) {
|
||||
// ref counter is set to 1
|
||||
PythonObject = Py::Object(new ViewGroupPy(this), true);
|
||||
}
|
||||
return Py::new_reference_to(PythonObject);
|
||||
}
|
||||
58
src/Mod/Assembly/App/ViewGroup.h
Normal file
58
src/Mod/Assembly/App/ViewGroup.h
Normal file
@@ -0,0 +1,58 @@
|
||||
// SPDX-License-Identifier: LGPL-2.1-or-later
|
||||
/****************************************************************************
|
||||
* *
|
||||
* Copyright (c) 2023 Ondsel <development@ondsel.com> *
|
||||
* *
|
||||
* This file is part of FreeCAD. *
|
||||
* *
|
||||
* FreeCAD is free software: you can redistribute it and/or modify it *
|
||||
* under the terms of the GNU Lesser General Public License as *
|
||||
* published by the Free Software Foundation, either version 2.1 of the *
|
||||
* License, or (at your option) any later version. *
|
||||
* *
|
||||
* FreeCAD is distributed in the hope that it will be useful, but *
|
||||
* WITHOUT ANY WARRANTY; without even the implied warranty of *
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU *
|
||||
* Lesser General Public License for more details. *
|
||||
* *
|
||||
* You should have received a copy of the GNU Lesser General Public *
|
||||
* License along with FreeCAD. If not, see *
|
||||
* <https://www.gnu.org/licenses/>. *
|
||||
* *
|
||||
***************************************************************************/
|
||||
|
||||
|
||||
#ifndef ASSEMBLY_ViewGroup_H
|
||||
#define ASSEMBLY_ViewGroup_H
|
||||
|
||||
#include <Mod/Assembly/AssemblyGlobal.h>
|
||||
|
||||
#include <App/DocumentObjectGroup.h>
|
||||
#include <App/PropertyLinks.h>
|
||||
|
||||
|
||||
namespace Assembly
|
||||
{
|
||||
|
||||
class AssemblyExport ViewGroup: public App::DocumentObjectGroup
|
||||
{
|
||||
PROPERTY_HEADER_WITH_OVERRIDE(Assembly::ViewGroup);
|
||||
|
||||
public:
|
||||
ViewGroup();
|
||||
~ViewGroup() override;
|
||||
|
||||
PyObject* getPyObject() override;
|
||||
|
||||
/// returns the type name of the ViewProvider
|
||||
const char* getViewProviderName() const override
|
||||
{
|
||||
return "AssemblyGui::ViewProviderViewGroup";
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
} // namespace Assembly
|
||||
|
||||
|
||||
#endif // ASSEMBLY_ViewGroup_H
|
||||
19
src/Mod/Assembly/App/ViewGroupPy.xml
Normal file
19
src/Mod/Assembly/App/ViewGroupPy.xml
Normal file
@@ -0,0 +1,19 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<GenerateModel xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="generateMetaModel_Module.xsd">
|
||||
<PythonExport
|
||||
Father="DocumentObjectGroupPy"
|
||||
Name="ViewGroupPy"
|
||||
Twin="ViewGroup"
|
||||
TwinPointer="ViewGroup"
|
||||
Include="Mod/Assembly/App/ViewGroup.h"
|
||||
Namespace="Assembly"
|
||||
FatherInclude="App/DocumentObjectGroupPy.h"
|
||||
FatherNamespace="App">
|
||||
<Documentation>
|
||||
<Author Licence="LGPL" Name="Ondsel" EMail="development@ondsel.com" />
|
||||
<UserDocu>This class is a group subclass for joints.</UserDocu>
|
||||
</Documentation>
|
||||
|
||||
<CustomAttributes />
|
||||
</PythonExport>
|
||||
</GenerateModel>
|
||||
46
src/Mod/Assembly/App/ViewGroupPyImp.cpp
Normal file
46
src/Mod/Assembly/App/ViewGroupPyImp.cpp
Normal file
@@ -0,0 +1,46 @@
|
||||
/***************************************************************************
|
||||
* Copyright (c) 2014 Jürgen Riegel <juergen.riegel@web.de> *
|
||||
* *
|
||||
* This file is part of the FreeCAD CAx development system. *
|
||||
* *
|
||||
* This library is free software; you can redistribute it and/or *
|
||||
* modify it under the terms of the GNU Library General Public *
|
||||
* License as published by the Free Software Foundation; either *
|
||||
* version 2 of the License, or (at your option) any later version. *
|
||||
* *
|
||||
* This library is distributed in the hope that it will be useful, *
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
||||
* GNU Library General Public License for more details. *
|
||||
* *
|
||||
* You should have received a copy of the GNU Library General Public *
|
||||
* License along with this library; see the file COPYING.LIB. If not, *
|
||||
* write to the Free Software Foundation, Inc., 59 Temple Place, *
|
||||
* Suite 330, Boston, MA 02111-1307, USA *
|
||||
* *
|
||||
***************************************************************************/
|
||||
|
||||
|
||||
#include "PreCompiled.h"
|
||||
|
||||
// inclusion of the generated files (generated out of ViewGroup.xml)
|
||||
#include "ViewGroupPy.h"
|
||||
#include "ViewGroupPy.cpp"
|
||||
|
||||
using namespace Assembly;
|
||||
|
||||
// returns a string which represents the object e.g. when printed in python
|
||||
std::string ViewGroupPy::representation() const
|
||||
{
|
||||
return {"<Exploded View Group>"};
|
||||
}
|
||||
|
||||
PyObject* ViewGroupPy::getCustomAttributes(const char* /*attr*/) const
|
||||
{
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
int ViewGroupPy::setCustomAttributes(const char* /*attr*/, PyObject* /*obj*/)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
@@ -10,6 +10,7 @@ set(Assembly_Scripts
|
||||
CommandInsertLink.py
|
||||
CommandSolveAssembly.py
|
||||
CommandCreateJoint.py
|
||||
CommandCreateView.py
|
||||
CommandExportASMT.py
|
||||
TestAssemblyWorkbench.py
|
||||
JointObject.py
|
||||
|
||||
866
src/Mod/Assembly/CommandCreateView.py
Normal file
866
src/Mod/Assembly/CommandCreateView.py
Normal file
@@ -0,0 +1,866 @@
|
||||
# SPDX-License-Identifier: LGPL-2.1-or-later
|
||||
# /**************************************************************************
|
||||
# *
|
||||
# Copyright (c) 2023 Ondsel <development@ondsel.com> *
|
||||
# *
|
||||
# This file is part of FreeCAD. *
|
||||
# *
|
||||
# FreeCAD is free software: you can redistribute it and/or modify it *
|
||||
# under the terms of the GNU Lesser General Public License as *
|
||||
# published by the Free Software Foundation, either version 2.1 of the *
|
||||
# License, or (at your option) any later version. *
|
||||
# *
|
||||
# FreeCAD is distributed in the hope that it will be useful, but *
|
||||
# WITHOUT ANY WARRANTY; without even the implied warranty of *
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU *
|
||||
# Lesser General Public License for more details. *
|
||||
# *
|
||||
# You should have received a copy of the GNU Lesser General Public *
|
||||
# License along with FreeCAD. If not, see *
|
||||
# <https://www.gnu.org/licenses/>. *
|
||||
# *
|
||||
# **************************************************************************/
|
||||
|
||||
import re
|
||||
import os
|
||||
import FreeCAD as App
|
||||
|
||||
from pivy import coin
|
||||
|
||||
from PySide.QtCore import QT_TRANSLATE_NOOP
|
||||
|
||||
if App.GuiUp:
|
||||
import FreeCADGui as Gui
|
||||
from PySide import QtCore, QtGui, QtWidgets
|
||||
from PySide.QtWidgets import QPushButton, QMenu
|
||||
|
||||
import UtilsAssembly
|
||||
import Preferences
|
||||
|
||||
# translate = App.Qt.translate
|
||||
|
||||
__title__ = "Assembly Command Create Exploded View"
|
||||
__author__ = "Ondsel"
|
||||
__url__ = "https://www.freecad.org"
|
||||
|
||||
|
||||
class CommandCreateView:
|
||||
def __init__(self):
|
||||
pass
|
||||
|
||||
def GetResources(self):
|
||||
return {
|
||||
"Pixmap": "Assembly_ExplodedView",
|
||||
"MenuText": QT_TRANSLATE_NOOP("Assembly_CreateView", "Create Exploded View"),
|
||||
"Accel": "V",
|
||||
"ToolTip": "<p>"
|
||||
+ QT_TRANSLATE_NOOP(
|
||||
"Assembly_CreateView",
|
||||
"Create an exploded view of the current assembly.",
|
||||
)
|
||||
+ "</p>",
|
||||
"CmdType": "ForEdit",
|
||||
}
|
||||
|
||||
def IsActive(self):
|
||||
return UtilsAssembly.isAssemblyCommandActive()
|
||||
|
||||
def Activated(self):
|
||||
assembly = UtilsAssembly.activeAssembly()
|
||||
if not assembly:
|
||||
return
|
||||
|
||||
self.panel = TaskAssemblyCreateView()
|
||||
Gui.Control.showDialog(self.panel)
|
||||
|
||||
|
||||
######### Exploded View Object ###########
|
||||
class ExplodedView:
|
||||
def __init__(self, expView):
|
||||
expView.addProperty(
|
||||
"App::PropertyLinkList", "Steps", "Exploded View", "Step objects of the exploded view."
|
||||
)
|
||||
expView.Proxy = self
|
||||
|
||||
self.stepsChangedCallback = None
|
||||
|
||||
def dumps(self):
|
||||
return None
|
||||
|
||||
def loads(self, state):
|
||||
return None
|
||||
|
||||
def getAssembly(self, viewObj):
|
||||
return viewObj.InList[0]
|
||||
|
||||
def onChanged(self, viewObj, prop):
|
||||
if prop == "Steps" and self.stepsChangedCallback is not None:
|
||||
self.stepsChangedCallback()
|
||||
|
||||
def setStepsChangedCallback(self, callback):
|
||||
self.stepsChangedCallback = callback
|
||||
|
||||
def execute(self, fp):
|
||||
"""Do something when doing a recomputation, this method is mandatory"""
|
||||
# App.Console.PrintMessage("Recompute Python Box feature\n")
|
||||
pass
|
||||
|
||||
|
||||
class ViewProviderExplodedView:
|
||||
def __init__(self, vobj):
|
||||
"""Set this object to the proxy object of the actual view provider"""
|
||||
vobj.Proxy = self
|
||||
|
||||
def attach(self, vobj):
|
||||
"""Setup the scene sub-graph of the view provider, this method is mandatory"""
|
||||
self.app_obj = vobj.Object
|
||||
|
||||
self.display_mode = coin.SoType.fromName("SoFCSelection").createInstance()
|
||||
|
||||
vobj.addDisplayMode(self.display_mode, "Wireframe")
|
||||
|
||||
def updateData(self, joint, prop):
|
||||
"""If a property of the handled feature has changed we have the chance to handle this here"""
|
||||
# joint is the handled feature, prop is the name of the property that has changed
|
||||
pass
|
||||
|
||||
def getDisplayModes(self, obj):
|
||||
"""Return a list of display modes."""
|
||||
return ["Wireframe"]
|
||||
|
||||
def getDefaultDisplayMode(self):
|
||||
"""Return the name of the default display mode. It must be defined in getDisplayModes."""
|
||||
return "Wireframe"
|
||||
|
||||
def onChanged(self, vp, prop):
|
||||
"""Here we can do something when a single property got changed"""
|
||||
# App.Console.PrintMessage("Change property: " + str(prop) + "\n")
|
||||
pass
|
||||
|
||||
def getIcon(self):
|
||||
return ":/icons/Assembly_ExplodedView.svg"
|
||||
|
||||
def dumps(self):
|
||||
"""When saving the document this object gets stored using Python's json module.\
|
||||
Since we have some un-serializable parts here -- the Coin stuff -- we must define this method\
|
||||
to return a tuple of all serializable objects or None."""
|
||||
return None
|
||||
|
||||
def loads(self, state):
|
||||
"""When restoring the serialized object from document we have the chance to set some internals here.\
|
||||
Since no data were serialized nothing needs to be done here."""
|
||||
return None
|
||||
|
||||
def claimChildren(self):
|
||||
return self.app_obj.Steps
|
||||
|
||||
def doubleClicked(self, vobj):
|
||||
task = Gui.Control.activeTaskDialog()
|
||||
if task:
|
||||
task.reject()
|
||||
|
||||
assembly = vobj.Object.InList[0]
|
||||
if UtilsAssembly.activeAssembly() != assembly:
|
||||
Gui.ActiveDocument.setEdit(assembly)
|
||||
|
||||
panel = TaskAssemblyCreateView(vobj.Object)
|
||||
Gui.Control.showDialog(panel)
|
||||
|
||||
|
||||
######### Exploded View Step #########
|
||||
ExplodedViewStepTypes = [
|
||||
"Normal",
|
||||
"Radial",
|
||||
]
|
||||
|
||||
|
||||
class ExplodedViewStep:
|
||||
def __init__(self, evStep, type_index=0):
|
||||
evStep.Proxy = self
|
||||
|
||||
# we cannot use "App::PropertyLinkList" for objs because they can be external
|
||||
evStep.addProperty(
|
||||
"App::PropertyStringList",
|
||||
"ObjNames",
|
||||
"Exploded Step",
|
||||
QT_TRANSLATE_NOOP("App::Property", "The object moved by the move"),
|
||||
)
|
||||
|
||||
evStep.addProperty(
|
||||
"App::PropertyLinkList",
|
||||
"Parts",
|
||||
"Exploded Step",
|
||||
QT_TRANSLATE_NOOP("App::Property", "The containing parts of objects moved by the move"),
|
||||
)
|
||||
|
||||
evStep.addProperty(
|
||||
"App::PropertyPlacement",
|
||||
"Placement",
|
||||
"Exploded Step",
|
||||
QT_TRANSLATE_NOOP(
|
||||
"App::Property",
|
||||
"This is the movement of the step. The end placement is the result of the start placement * this placement.",
|
||||
),
|
||||
)
|
||||
|
||||
evStep.addProperty(
|
||||
"App::PropertyEnumeration",
|
||||
"MoveType",
|
||||
"Exploded Step",
|
||||
QT_TRANSLATE_NOOP("App::Property", "The type of the move"),
|
||||
)
|
||||
evStep.MoveType = ExplodedViewStepTypes # sets the list
|
||||
evStep.MoveType = ExplodedViewStepTypes[type_index] # set the initial value
|
||||
|
||||
def dumps(self):
|
||||
return None
|
||||
|
||||
def loads(self, state):
|
||||
return None
|
||||
|
||||
def onChanged(self, joint, prop):
|
||||
"""Do something when a property has changed"""
|
||||
pass
|
||||
|
||||
def execute(self, fp):
|
||||
"""Do something when doing a recomputation, this method is mandatory"""
|
||||
# App.Console.PrintMessage("Recompute Python Box feature\n")
|
||||
pass
|
||||
|
||||
|
||||
class ViewProviderExplodedViewStep:
|
||||
def __init__(self, vobj):
|
||||
"""Set this object to the proxy object of the actual view provider"""
|
||||
vobj.Proxy = self
|
||||
|
||||
def attach(self, vobj):
|
||||
"""Setup the scene sub-graph of the view provider, this method is mandatory"""
|
||||
self.app_obj = vobj.Object
|
||||
|
||||
pref = Preferences.preferences()
|
||||
|
||||
self.line_thickness = pref.GetInt("StepLineThickness", 3)
|
||||
|
||||
param_step_line_color = pref.GetUnsigned("StepLineColor", 0xCC333300)
|
||||
self.so_color = coin.SoBaseColor()
|
||||
self.so_color.rgb.setValue(UtilsAssembly.color_from_unsigned(param_step_line_color))
|
||||
|
||||
self.draw_style = coin.SoDrawStyle()
|
||||
self.draw_style.style = coin.SoDrawStyle.LINES
|
||||
self.draw_style.lineWidth = self.line_thickness
|
||||
self.draw_style.linePattern = 0xF0F0 # Dashed line pattern
|
||||
|
||||
# Create a separator to hold all dashed lines
|
||||
self.lineSetGroup = coin.SoSeparator()
|
||||
|
||||
self.display_mode = coin.SoType.fromName("SoFCSelection").createInstance()
|
||||
self.display_mode.addChild(self.lineSetGroup) # Add the group to the display mode
|
||||
vobj.addDisplayMode(self.display_mode, "Wireframe")
|
||||
|
||||
if self.app_obj.MoveType == "Radial":
|
||||
assembly = UtilsAssembly.activeAssembly()
|
||||
self.assemblyCOM = UtilsAssembly.getCenterOfBoundingBox([assembly], [None])
|
||||
self.assemblyCOMSize = assembly.ViewObject.getBoundingBox().DiagonalLength
|
||||
|
||||
def updateData(self, stepObj, prop):
|
||||
"""If a property of the handled feature has changed we have the chance to handle this here"""
|
||||
# stepObj is the handled feature, prop is the name of the property that has changed
|
||||
if prop in ["Parts", "Placement"]:
|
||||
self.redrawLines(stepObj)
|
||||
|
||||
def redrawLines(self, stepObj):
|
||||
# Clear existing lines
|
||||
self.lineSetGroup.removeAllChildren()
|
||||
|
||||
if hasattr(stepObj, "Parts") and stepObj.Parts:
|
||||
if stepObj.MoveType == "Radial":
|
||||
distance = stepObj.Placement.Base.Length
|
||||
factor = 1 + 4 * distance / self.assemblyCOMSize
|
||||
|
||||
for objName, part in zip(stepObj.ObjNames, stepObj.Parts):
|
||||
if not objName:
|
||||
return
|
||||
|
||||
obj = UtilsAssembly.getObjectInPart(objName, part)
|
||||
|
||||
if not obj:
|
||||
return
|
||||
|
||||
plc2 = UtilsAssembly.getGlobalPlacement(obj, part)
|
||||
plc2.Base = UtilsAssembly.getCenterOfBoundingBox([obj], [part])
|
||||
endPoint = plc2.Base
|
||||
|
||||
if stepObj.MoveType == "Radial":
|
||||
startPoint = (endPoint - self.assemblyCOM) / factor + self.assemblyCOM
|
||||
|
||||
else:
|
||||
plc1 = stepObj.Placement.inverse() * plc2
|
||||
startPoint = plc1.Base
|
||||
|
||||
# Create the line
|
||||
line = coin.SoLineSet()
|
||||
line.numVertices.setValue(2)
|
||||
coords = coin.SoCoordinate3()
|
||||
coords.point.setValues(0, [startPoint, endPoint])
|
||||
|
||||
# Create separator for this line to apply the style
|
||||
line_sep = coin.SoSeparator()
|
||||
line_sep.addChild(self.draw_style)
|
||||
line_sep.addChild(self.so_color)
|
||||
line_sep.addChild(coords)
|
||||
line_sep.addChild(line)
|
||||
|
||||
# Add to the group
|
||||
self.lineSetGroup.addChild(line_sep)
|
||||
|
||||
def getDisplayModes(self, obj):
|
||||
"""Return a list of display modes."""
|
||||
modes = []
|
||||
modes.append("Wireframe")
|
||||
return modes
|
||||
|
||||
def getDefaultDisplayMode(self):
|
||||
"""Return the name of the default display mode. It must be defined in getDisplayModes."""
|
||||
return "Wireframe"
|
||||
|
||||
def onChanged(self, vp, prop):
|
||||
"""Here we can do something when a single property got changed"""
|
||||
# App.Console.PrintMessage("Change property: " + str(prop) + "\n")
|
||||
pass
|
||||
|
||||
def getIcon(self):
|
||||
return ":/icons/Assembly_ExplodedViewStep.svg"
|
||||
|
||||
def dumps(self):
|
||||
"""When saving the document this object gets stored using Python's json module.\
|
||||
Since we have some un-serializable parts here -- the Coin stuff -- we must define this method\
|
||||
to return a tuple of all serializable objects or None."""
|
||||
return None
|
||||
|
||||
def loads(self, state):
|
||||
"""When restoring the serialized object from document we have the chance to set some internals here.\
|
||||
Since no data were serialized nothing needs to be done here."""
|
||||
return None
|
||||
|
||||
|
||||
class ExplodedViewSelGate:
|
||||
def __init__(self, assembly, viewObj):
|
||||
self.assembly = assembly
|
||||
self.viewObj = viewObj
|
||||
|
||||
def allow(self, doc, obj, sub):
|
||||
if (obj.Name == self.assembly.Name and sub) or self.assembly.hasObject(obj, True):
|
||||
# Objects within the assembly.
|
||||
return True
|
||||
|
||||
if obj in self.viewObj.Steps:
|
||||
# Enable selection of steps object
|
||||
return True
|
||||
|
||||
return False
|
||||
|
||||
|
||||
######### Create Exploded View Task ###########
|
||||
class TaskAssemblyCreateView(QtCore.QObject):
|
||||
def __init__(self, viewObj=None):
|
||||
super().__init__()
|
||||
|
||||
view = Gui.activeDocument().activeView()
|
||||
|
||||
self.assembly = UtilsAssembly.activeAssembly()
|
||||
self.assembly.ViewObject.EnableMovement = False
|
||||
self.asmDragger = self.assembly.ViewObject.getDragger()
|
||||
self.cbFin = view.addDraggerCallback(
|
||||
self.asmDragger, "addFinishCallback", self.draggerFinished
|
||||
)
|
||||
self.cbMov = view.addDraggerCallback(
|
||||
self.asmDragger, "addMotionCallback", self.draggerMoved
|
||||
)
|
||||
|
||||
self.assemblyCOM = UtilsAssembly.getCenterOfBoundingBox([self.assembly], [None])
|
||||
self.assemblyCOMSize = self.assembly.ViewObject.getBoundingBox().DiagonalLength
|
||||
|
||||
# self.doc = App.ActiveDocument
|
||||
|
||||
Gui.Selection.clearSelection()
|
||||
|
||||
self.form = Gui.PySideUic.loadUi(":/panels/TaskAssemblyCreateView.ui")
|
||||
self.form.stepList.installEventFilter(self)
|
||||
self.form.stepList.itemClicked.connect(self.onItemClicked)
|
||||
|
||||
self.form.btnAlignDragger.setMenu(QMenu(self.form.btnAlignDragger))
|
||||
actionAlignTo = self.form.btnAlignDragger.menu().addAction("Align to...")
|
||||
actionAlignToCenter = self.form.btnAlignDragger.menu().addAction("Align to part center")
|
||||
actionAlignToOrigin = self.form.btnAlignDragger.menu().addAction("Align to part origin")
|
||||
|
||||
# Connect actions to the respective functions
|
||||
actionAlignTo.triggered.connect(self.onAlignTo)
|
||||
actionAlignToCenter.triggered.connect(self.onAlignToCenter)
|
||||
actionAlignToOrigin.triggered.connect(self.onAlignToPartOrigin)
|
||||
|
||||
self.form.btnAlignDragger.setVisible(False)
|
||||
self.form.btnRadialExplosion.clicked.connect(self.onRadialClicked)
|
||||
|
||||
pref = Preferences.preferences()
|
||||
self.form.CheckBox_PartsAsSingleSolid.setChecked(pref.GetBool("PartsAsSingleSolid", True))
|
||||
|
||||
self.saveAssemblyPartsPlacements(self.assembly)
|
||||
|
||||
if viewObj:
|
||||
App.setActiveTransaction("Edit Exploded View")
|
||||
self.viewObj = viewObj
|
||||
for step in self.viewObj.Steps:
|
||||
step.Visibility = True
|
||||
self.onStepsChanged()
|
||||
|
||||
else:
|
||||
App.setActiveTransaction("Create Exploded View")
|
||||
self.createExplodedViewObject()
|
||||
|
||||
Gui.Selection.addSelectionGate(
|
||||
ExplodedViewSelGate(self.assembly, self.viewObj), Gui.Selection.ResolveMode.NoResolve
|
||||
)
|
||||
Gui.Selection.addObserver(self, Gui.Selection.ResolveMode.NoResolve)
|
||||
|
||||
self.viewObj.Proxy.setStepsChangedCallback(self.onStepsChanged)
|
||||
self.callbackMove = view.addEventCallback("SoLocation2Event", self.moveMouse)
|
||||
self.callbackClick = view.addEventCallback("SoMouseButtonEvent", self.clickMouse)
|
||||
self.callbackKey = view.addEventCallback("SoKeyboardEvent", self.KeyboardEvent)
|
||||
|
||||
self.selectingFeature = False
|
||||
self.form.LabelAlignDragger.setVisible(False)
|
||||
self.preselection_dict = None
|
||||
|
||||
self.blockSetDragger = False
|
||||
self.blockDraggerMove = True
|
||||
self.currentStep = None
|
||||
|
||||
def accept(self):
|
||||
self.deactivate()
|
||||
self.restoreAssemblyPartsPlacements(self.assembly)
|
||||
for step in self.viewObj.Steps:
|
||||
step.Visibility = False
|
||||
App.closeActiveTransaction()
|
||||
return True
|
||||
|
||||
def reject(self):
|
||||
self.deactivate()
|
||||
App.closeActiveTransaction(True)
|
||||
return True
|
||||
|
||||
def deactivate(self):
|
||||
pref = Preferences.preferences()
|
||||
pref.SetBool("PartsAsSingleSolid", self.form.CheckBox_PartsAsSingleSolid.isChecked())
|
||||
|
||||
view = Gui.activeDocument().activeView()
|
||||
view.removeDraggerCallback(self.asmDragger, "addFinishCallback", self.cbFin)
|
||||
view.removeDraggerCallback(self.asmDragger, "addMotionCallback", self.cbMov)
|
||||
|
||||
self.assembly.ViewObject.DraggerVisibility = False
|
||||
self.assembly.ViewObject.EnableMovement = True
|
||||
|
||||
Gui.Selection.removeSelectionGate()
|
||||
Gui.Selection.removeObserver(self)
|
||||
Gui.Selection.clearSelection()
|
||||
|
||||
self.viewObj.Proxy.setStepsChangedCallback(None)
|
||||
view.removeEventCallback("SoLocation2Event", self.callbackMove)
|
||||
view.removeEventCallback("SoMouseButtonEvent", self.callbackClick)
|
||||
view.removeEventCallback("SoKeyboardEvent", self.callbackKey)
|
||||
|
||||
if Gui.Control.activeDialog():
|
||||
Gui.Control.closeDialog()
|
||||
|
||||
def saveAssemblyPartsPlacements(self, assembly):
|
||||
self.initialPlcDict = {}
|
||||
assemblyParts = UtilsAssembly.getMovablePartsWithin(assembly)
|
||||
for part in assemblyParts:
|
||||
self.initialPlcDict[part.Name] = part.Placement
|
||||
|
||||
def restoreAssemblyPartsPlacements(self, assembly):
|
||||
assemblyParts = UtilsAssembly.getMovablePartsWithin(assembly)
|
||||
for part in assemblyParts:
|
||||
if part.Name in self.initialPlcDict:
|
||||
part.Placement = self.initialPlcDict[part.Name]
|
||||
|
||||
def setDragger(self):
|
||||
if self.blockSetDragger:
|
||||
return
|
||||
|
||||
self.dismissCurrentStep()
|
||||
self.selectedObjs = []
|
||||
self.selectedParts = [] # containing parts
|
||||
self.selectedObjsInitPlc = []
|
||||
selection = Gui.Selection.getSelectionEx("*", 0)
|
||||
if not selection:
|
||||
self.enableDragger(False)
|
||||
return
|
||||
for sel in selection:
|
||||
# If you select 2 solids (bodies for example) within an assembly.
|
||||
# There'll be a single sel but 2 SubElementNames.
|
||||
|
||||
if not sel.SubElementNames:
|
||||
# no subnames, so its a root assembly itself that is selected.
|
||||
Gui.Selection.removeSelection(sel.Object)
|
||||
continue
|
||||
|
||||
for sub_name in sel.SubElementNames:
|
||||
# Only objects within the assembly.
|
||||
objs_names, element_name = UtilsAssembly.getObjsNamesAndElement(
|
||||
sel.ObjectName, sub_name
|
||||
)
|
||||
if self.assembly.Name not in objs_names:
|
||||
Gui.Selection.removeSelection(sel.Object, sub_name)
|
||||
continue
|
||||
|
||||
obj_name = sel.ObjectName
|
||||
full_obj_name = UtilsAssembly.getFullObjName(obj_name, sub_name)
|
||||
full_element_name = UtilsAssembly.getFullElementName(obj_name, sub_name)
|
||||
selected_object = UtilsAssembly.getObject(full_element_name)
|
||||
if selected_object is None:
|
||||
continue
|
||||
element_name = UtilsAssembly.getElementName(full_element_name)
|
||||
part = UtilsAssembly.getContainingPart(
|
||||
full_element_name, selected_object, self.assembly
|
||||
)
|
||||
|
||||
if selected_object == self.assembly or element_name != "":
|
||||
# do not accept selection of assembly itself or elements
|
||||
Gui.Selection.removeSelection(sel.Object, sub_name)
|
||||
continue
|
||||
|
||||
if self.form.CheckBox_PartsAsSingleSolid.isChecked():
|
||||
selected_object = part
|
||||
|
||||
if not selected_object in self.selectedObjs and hasattr(
|
||||
selected_object, "Placement"
|
||||
):
|
||||
self.selectedObjs.append(selected_object)
|
||||
self.selectedParts.append(part)
|
||||
self.selectedObjsInitPlc.append(App.Placement(selected_object.Placement))
|
||||
|
||||
if len(self.selectedObjs) != 0:
|
||||
self.enableDragger(True)
|
||||
self.onAlignToCenter()
|
||||
|
||||
else:
|
||||
self.enableDragger(False)
|
||||
|
||||
def enableDragger(self, val):
|
||||
self.assembly.ViewObject.DraggerVisibility = val
|
||||
self.form.btnAlignDragger.setVisible(val)
|
||||
|
||||
def onStepsChanged(self):
|
||||
# First reset positions
|
||||
self.restoreAssemblyPartsPlacements(self.assembly)
|
||||
|
||||
self.form.stepList.clear()
|
||||
|
||||
for step in self.viewObj.Steps:
|
||||
|
||||
if step.MoveType == "Radial":
|
||||
distance = step.Placement.Base.Length
|
||||
factor = 1 + 4 * distance / self.assemblyCOMSize
|
||||
|
||||
for objName, part in zip(step.ObjNames, step.Parts):
|
||||
obj = UtilsAssembly.getObjectInPart(objName, part)
|
||||
if not obj:
|
||||
continue
|
||||
|
||||
if step.MoveType == "Radial":
|
||||
init_vec = obj.Placement.Base - self.assemblyCOM
|
||||
obj.Placement.Base = self.assemblyCOM + init_vec * factor
|
||||
else:
|
||||
obj.Placement = step.Placement * obj.Placement
|
||||
|
||||
self.form.stepList.addItem(step.Name)
|
||||
step.ViewObject.Proxy.redrawLines(step)
|
||||
|
||||
def onItemClicked(self, item):
|
||||
Gui.Selection.clearSelection()
|
||||
Gui.Selection.addSelection(self.viewObj.Document.Name, item.text(), "")
|
||||
# we give back the focus to the item as addSelection gave the focus to the 3dview
|
||||
self.form.stepList.setCurrentItem(item)
|
||||
|
||||
def onRadialClicked(self):
|
||||
self.dismissCurrentStep()
|
||||
|
||||
# Add to selection all the movable parts
|
||||
partsAsSolid = self.form.CheckBox_PartsAsSingleSolid.isChecked()
|
||||
assemblyParts = UtilsAssembly.getMovablePartsWithin(self.assembly, partsAsSolid)
|
||||
self.blockSetDragger = True
|
||||
for part in assemblyParts:
|
||||
Gui.Selection.addSelection(part, "")
|
||||
self.blockSetDragger = False
|
||||
self.setDragger()
|
||||
|
||||
self.createExplodedStepObject(1) # 1 = type_index of "Radial"
|
||||
|
||||
def onAlignTo(self):
|
||||
self.alignMode = "Custom"
|
||||
self.selectingFeature = True
|
||||
# We use greedy selection to prevent that clicking again on the solid
|
||||
# clears selection before trying to select the whole assemly
|
||||
Gui.Selection.setSelectionStyle(Gui.Selection.SelectionStyle.GreedySelection)
|
||||
self.enableDragger(False)
|
||||
self.form.LabelAlignDragger.setVisible(True)
|
||||
|
||||
def endSelectionMode(self):
|
||||
self.selectingFeature = False
|
||||
self.enableDragger(True)
|
||||
Gui.Selection.setSelectionStyle(Gui.Selection.SelectionStyle.NormalSelection)
|
||||
self.form.LabelAlignDragger.setVisible(False)
|
||||
|
||||
def onAlignToCenter(self):
|
||||
self.alignMode = "Center"
|
||||
self.setDraggerObjectPlc()
|
||||
|
||||
def onAlignToPartOrigin(self):
|
||||
self.alignMode = "PartOrigin"
|
||||
self.setDraggerObjectPlc()
|
||||
|
||||
def findDraggerInitialPlc(self):
|
||||
if len(self.selectedObjs) == 0:
|
||||
return
|
||||
|
||||
if self.alignMode == "Custom":
|
||||
self.initialDraggerPlc = App.Placement(self.assembly.ViewObject.DraggerPlacement)
|
||||
else:
|
||||
plc = UtilsAssembly.getGlobalPlacement(self.selectedObjs[0], self.selectedParts[0])
|
||||
self.initialDraggerPlc = App.Placement(plc)
|
||||
if self.alignMode == "Center":
|
||||
self.initialDraggerPlc.Base = UtilsAssembly.getCenterOfBoundingBox(
|
||||
self.selectedObjs, self.selectedParts
|
||||
)
|
||||
|
||||
def setDraggerObjectPlc(self):
|
||||
self.findDraggerInitialPlc()
|
||||
|
||||
self.blockDraggerMove = True
|
||||
self.assembly.ViewObject.DraggerPlacement = self.initialDraggerPlc
|
||||
self.blockDraggerMove = False
|
||||
|
||||
def createExplodedViewObject(self):
|
||||
view_group = UtilsAssembly.getViewGroup(self.assembly)
|
||||
self.viewObj = view_group.newObject("App::FeaturePython", "Exploded View")
|
||||
|
||||
ExplodedView(self.viewObj)
|
||||
ViewProviderExplodedView(self.viewObj.ViewObject)
|
||||
|
||||
def createExplodedStepObject(self, moveType_index=0):
|
||||
self.currentStep = App.ActiveDocument.addObject("App::FeaturePython", "Move")
|
||||
ExplodedViewStep(self.currentStep, moveType_index)
|
||||
ViewProviderExplodedViewStep(self.currentStep.ViewObject)
|
||||
|
||||
# Note: self.viewObj.Steps.append(self.currentStep) does not work
|
||||
listOfSteps = self.viewObj.Steps
|
||||
listOfSteps.append(self.currentStep)
|
||||
self.viewObj.Steps = listOfSteps
|
||||
|
||||
objNames = []
|
||||
for obj in self.selectedObjs:
|
||||
objNames.append(obj.Name)
|
||||
|
||||
self.currentStep.Placement = App.Placement()
|
||||
self.currentStep.ObjNames = objNames
|
||||
self.currentStep.Parts = self.selectedParts
|
||||
|
||||
def dismissCurrentStep(self):
|
||||
if self.currentStep is None:
|
||||
return
|
||||
|
||||
for obj, init_plc in zip(self.selectedObjs, self.selectedObjsInitPlc):
|
||||
obj.Placement = init_plc
|
||||
|
||||
self.currentStep.Document.removeObject(self.currentStep.Name)
|
||||
self.currentStep = None
|
||||
|
||||
Gui.Selection.clearSelection()
|
||||
|
||||
def draggerMoved(self, event):
|
||||
if self.blockDraggerMove:
|
||||
return
|
||||
|
||||
if self.currentStep is None:
|
||||
self.createExplodedStepObject()
|
||||
|
||||
draggerPlc = self.assembly.ViewObject.DraggerPlacement
|
||||
movePlc = draggerPlc * self.initialDraggerPlc.inverse()
|
||||
|
||||
if self.currentStep.MoveType == "Radial":
|
||||
distance = movePlc.Base.Length
|
||||
factor = 1 + 4 * distance / self.assemblyCOMSize
|
||||
for obj, init_plc in zip(self.selectedObjs, self.selectedObjsInitPlc):
|
||||
init_vec = init_plc.Base - self.assemblyCOM
|
||||
obj.Placement.Base = self.assemblyCOM + init_vec * factor
|
||||
|
||||
else:
|
||||
for obj, init_plc in zip(self.selectedObjs, self.selectedObjsInitPlc):
|
||||
obj.Placement = movePlc * init_plc
|
||||
|
||||
# we update the step Placement after parts placement has updated.
|
||||
self.currentStep.Placement = movePlc
|
||||
|
||||
def draggerFinished(self, event):
|
||||
if self.currentStep.MoveType == "Radial":
|
||||
self.currentStep = None
|
||||
Gui.Selection.clearSelection()
|
||||
return
|
||||
|
||||
self.currentStep = None
|
||||
|
||||
# Reset the initial placements
|
||||
self.findDraggerInitialPlc()
|
||||
|
||||
for i, obj in enumerate(self.selectedObjs):
|
||||
self.selectedObjsInitPlc[i] = App.Placement(obj.Placement)
|
||||
|
||||
def moveMouse(self, info):
|
||||
if not self.selectingFeature:
|
||||
return
|
||||
|
||||
view = Gui.activeDocument().activeView()
|
||||
cursor_info = view.getObjectInfo(view.getCursorPos())
|
||||
|
||||
if not cursor_info or not self.preselection_dict:
|
||||
self.assembly.ViewObject.DraggerVisibility = False
|
||||
return
|
||||
|
||||
newPos = App.Vector(cursor_info["x"], cursor_info["y"], cursor_info["z"])
|
||||
self.preselection_dict["mouse_pos"] = newPos
|
||||
|
||||
if self.preselection_dict["element_name"] == "":
|
||||
self.preselection_dict["vertex_name"] = ""
|
||||
else:
|
||||
self.preselection_dict["vertex_name"] = UtilsAssembly.findElementClosestVertex(
|
||||
self.preselection_dict
|
||||
)
|
||||
|
||||
obj = self.preselection_dict["object"]
|
||||
part = self.preselection_dict["part"]
|
||||
plc = UtilsAssembly.findPlacement(
|
||||
obj,
|
||||
part,
|
||||
self.preselection_dict["element_name"],
|
||||
self.preselection_dict["vertex_name"],
|
||||
)
|
||||
global_plc = UtilsAssembly.getGlobalPlacement(obj, part)
|
||||
plc = global_plc * plc
|
||||
|
||||
self.blockDraggerMove = True
|
||||
self.assembly.ViewObject.DraggerPlacement = plc
|
||||
self.blockDraggerMove = False
|
||||
self.assembly.ViewObject.DraggerVisibility = True
|
||||
|
||||
def clickMouse(self, info):
|
||||
if info["Button"] == "BUTTON2" and info["State"] == "DOWN":
|
||||
if self.selectingFeature:
|
||||
self.endSelectionMode()
|
||||
|
||||
# 3D view keyboard handler
|
||||
def KeyboardEvent(self, info):
|
||||
if info["State"] == "UP" and info["Key"] == "ESCAPE":
|
||||
if self.currentStep is None:
|
||||
self.reject()
|
||||
else:
|
||||
if self.selectingFeature:
|
||||
self.endSelectionMode()
|
||||
else:
|
||||
self.dismissCurrentStep()
|
||||
|
||||
# Taskbox keyboard event handler
|
||||
def eventFilter(self, watched, event):
|
||||
if self.form is not None and watched == self.form.stepList:
|
||||
if event.type() == QtCore.QEvent.ShortcutOverride:
|
||||
if event.key() == QtCore.Qt.Key_Delete:
|
||||
event.accept()
|
||||
return True # Indicate that the event has been handled
|
||||
return False
|
||||
|
||||
elif event.type() == QtCore.QEvent.KeyPress:
|
||||
if event.key() == QtCore.Qt.Key_Delete:
|
||||
selected_indexes = self.form.stepList.selectedIndexes()
|
||||
sorted_indexes = sorted(selected_indexes, key=lambda x: x.row(), reverse=True)
|
||||
for index in sorted_indexes:
|
||||
row = index.row()
|
||||
if row < len(self.viewObj.Steps):
|
||||
step = self.viewObj.Steps[row]
|
||||
# First remove the link from the viewObj
|
||||
self.viewObj.Steps.remove(step)
|
||||
# Delete the object
|
||||
step.Document.removeObject(step.Name)
|
||||
|
||||
return True # Consume the event
|
||||
|
||||
return super().eventFilter(watched, event)
|
||||
|
||||
# selectionObserver stuff
|
||||
def addSelection(self, doc_name, obj_name, sub_name, mousePos):
|
||||
if self.selectingFeature:
|
||||
Gui.Selection.removeSelection(doc_name, obj_name, sub_name)
|
||||
return
|
||||
|
||||
else:
|
||||
full_element_name = UtilsAssembly.getFullElementName(obj_name, sub_name)
|
||||
selected_object = UtilsAssembly.getObject(full_element_name)
|
||||
if selected_object is None:
|
||||
return
|
||||
|
||||
element_name = UtilsAssembly.getElementName(full_element_name)
|
||||
part = UtilsAssembly.getContainingPart(
|
||||
full_element_name, selected_object, self.assembly
|
||||
)
|
||||
|
||||
if not self.form.CheckBox_PartsAsSingleSolid.isChecked():
|
||||
part = selected_object
|
||||
|
||||
if element_name != "":
|
||||
# When selecting, we do not want to select an element, but only the containing part.
|
||||
Gui.Selection.removeSelection(selected_object, element_name)
|
||||
if Gui.Selection.isSelected(part, ""):
|
||||
Gui.Selection.removeSelection(part, "")
|
||||
else:
|
||||
Gui.Selection.addSelection(part, "")
|
||||
else:
|
||||
self.setDragger()
|
||||
pass
|
||||
|
||||
def removeSelection(self, doc_name, obj_name, sub_name, mousePos=None):
|
||||
if self.selectingFeature:
|
||||
self.endSelectionMode()
|
||||
self.findDraggerInitialPlc()
|
||||
return
|
||||
|
||||
full_element_name = UtilsAssembly.getFullElementName(obj_name, sub_name)
|
||||
element_name = UtilsAssembly.getElementName(full_element_name)
|
||||
if element_name == "":
|
||||
self.setDragger()
|
||||
pass
|
||||
|
||||
def setPreselection(self, doc_name, obj_name, sub_name):
|
||||
if not self.selectingFeature or not sub_name:
|
||||
self.preselection_dict = None
|
||||
return
|
||||
|
||||
full_obj_name = UtilsAssembly.getFullObjName(obj_name, sub_name)
|
||||
full_element_name = UtilsAssembly.getFullElementName(obj_name, sub_name)
|
||||
selected_object = UtilsAssembly.getObject(full_element_name)
|
||||
element_name = UtilsAssembly.getElementName(full_element_name)
|
||||
part = UtilsAssembly.getContainingPart(full_element_name, selected_object, self.assembly)
|
||||
|
||||
self.preselection_dict = {
|
||||
"object": selected_object,
|
||||
"part": part,
|
||||
"sub_name": sub_name,
|
||||
"element_name": element_name,
|
||||
"full_element_name": full_element_name,
|
||||
"full_obj_name": full_obj_name,
|
||||
}
|
||||
|
||||
def clearSelection(self, doc_name):
|
||||
self.form.stepList.clearSelection()
|
||||
self.setDragger()
|
||||
|
||||
|
||||
if App.GuiUp:
|
||||
Gui.addCommand("Assembly_CreateView", CommandCreateView())
|
||||
@@ -28,6 +28,7 @@
|
||||
|
||||
#include "ViewProviderAssembly.h"
|
||||
#include "ViewProviderJointGroup.h"
|
||||
#include "ViewProviderViewGroup.h"
|
||||
|
||||
|
||||
namespace AssemblyGui
|
||||
@@ -48,6 +49,7 @@ PyMOD_INIT_FUNC(AssemblyGui)
|
||||
|
||||
AssemblyGui::ViewProviderAssembly ::init();
|
||||
AssemblyGui::ViewProviderJointGroup::init();
|
||||
AssemblyGui::ViewProviderViewGroup::init();
|
||||
|
||||
PyMOD_Return(mod);
|
||||
}
|
||||
|
||||
@@ -40,6 +40,8 @@ SET(AssemblyGui_SRCS_Module
|
||||
ViewProviderAssembly.h
|
||||
ViewProviderJointGroup.cpp
|
||||
ViewProviderJointGroup.h
|
||||
ViewProviderViewGroup.cpp
|
||||
ViewProviderViewGroup.h
|
||||
${Assembly_QRC_SRCS}
|
||||
)
|
||||
|
||||
|
||||
@@ -13,8 +13,11 @@
|
||||
<file>icons/Assembly_ExportASMT.svg</file>
|
||||
<file>icons/Assembly_SolveAssembly.svg</file>
|
||||
<file>icons/Assembly_JointGroup.svg</file>
|
||||
<file>icons/Assembly_ExplodedView.svg</file>
|
||||
<file>icons/Assembly_ExplodedViewGroup.svg</file>
|
||||
<file>panels/TaskAssemblyCreateJoint.ui</file>
|
||||
<file>panels/TaskAssemblyInsertLink.ui</file>
|
||||
<file>panels/TaskAssemblyCreateView.ui</file>
|
||||
<file>preferences/Assembly.ui</file>
|
||||
<file>icons/Assembly_CreateJointDistance.svg</file>
|
||||
<file>icons/AssemblyWorkbench.svg</file>
|
||||
|
||||
405
src/Mod/Assembly/Gui/Resources/icons/Assembly_ExplodedView.svg
Normal file
405
src/Mod/Assembly/Gui/Resources/icons/Assembly_ExplodedView.svg
Normal file
File diff suppressed because one or more lines are too long
|
After Width: | Height: | Size: 13 KiB |
File diff suppressed because one or more lines are too long
|
After Width: | Height: | Size: 24 KiB |
@@ -0,0 +1,73 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<ui version="4.0">
|
||||
<class>TaskAssemblyCreateView</class>
|
||||
<widget class="QWidget" name="TaskAssemblyCreateView">
|
||||
<property name="geometry">
|
||||
<rect>
|
||||
<x>0</x>
|
||||
<y>0</y>
|
||||
<width>376</width>
|
||||
<height>387</height>
|
||||
</rect>
|
||||
</property>
|
||||
<property name="windowTitle">
|
||||
<string>Create Exploded View</string>
|
||||
</property>
|
||||
<layout class="QGridLayout" name="gridLayout">
|
||||
<item row="0" column="0">
|
||||
<widget class="Gui::PrefCheckBox" name="CheckBox_PartsAsSingleSolid">
|
||||
<property name="toolTip">
|
||||
<string>If checked, Parts will be selected as a single solid.</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>Parts as single solid</string>
|
||||
</property>
|
||||
<property name="checked">
|
||||
<bool>true</bool>
|
||||
</property>
|
||||
<property name="prefEntry" stdset="0">
|
||||
<cstring>PartsAsSingleSolid</cstring>
|
||||
</property>
|
||||
<property name="prefPath" stdset="0">
|
||||
<cstring>Mod/Assembly</cstring>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="2" column="0">
|
||||
<widget class="QListWidget" name="stepList"/>
|
||||
</item>
|
||||
<item row="3" column="0">
|
||||
<widget class="QPushButton" name="btnAlignDragger">
|
||||
<property name="text">
|
||||
<string>Align dragger</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="4" column="0">
|
||||
<widget class="QLabel" name="LabelAlignDragger">
|
||||
<property name="text">
|
||||
<string>Aligning dragger:
|
||||
Select a feature.
|
||||
Press ESC to cancel.</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="5" column="0">
|
||||
<widget class="QPushButton" name="btnRadialExplosion">
|
||||
<property name="text">
|
||||
<string>Explode radially</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
<customwidgets>
|
||||
<customwidget>
|
||||
<class>Gui::PrefCheckBox</class>
|
||||
<extends>QCheckBox</extends>
|
||||
<header>Gui/PrefWidgets.h</header>
|
||||
</customwidget>
|
||||
</customwidgets>
|
||||
<resources/>
|
||||
<connections/>
|
||||
</ui>
|
||||
49
src/Mod/Assembly/Gui/ViewProviderViewGroup.cpp
Normal file
49
src/Mod/Assembly/Gui/ViewProviderViewGroup.cpp
Normal file
@@ -0,0 +1,49 @@
|
||||
// SPDX-License-Identifier: LGPL-2.1-or-later
|
||||
/****************************************************************************
|
||||
* *
|
||||
* Copyright (c) 2023 Ondsel <development@ondsel.com> *
|
||||
* *
|
||||
* This file is part of FreeCAD. *
|
||||
* *
|
||||
* FreeCAD is free software: you can redistribute it and/or modify it *
|
||||
* under the terms of the GNU Lesser General Public License as *
|
||||
* published by the Free Software Foundation, either version 2.1 of the *
|
||||
* License, or (at your option) any later version. *
|
||||
* *
|
||||
* FreeCAD is distributed in the hope that it will be useful, but *
|
||||
* WITHOUT ANY WARRANTY; without even the implied warranty of *
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU *
|
||||
* Lesser General Public License for more details. *
|
||||
* *
|
||||
* You should have received a copy of the GNU Lesser General Public *
|
||||
* License along with FreeCAD. If not, see *
|
||||
* <https://www.gnu.org/licenses/>. *
|
||||
* *
|
||||
***************************************************************************/
|
||||
|
||||
#include "PreCompiled.h"
|
||||
|
||||
#ifndef _PreComp_
|
||||
#endif
|
||||
|
||||
#include <App/Document.h>
|
||||
#include <App/DocumentObject.h>
|
||||
#include <Gui/Application.h>
|
||||
#include <Gui/BitmapFactory.h>
|
||||
|
||||
#include "ViewProviderViewGroup.h"
|
||||
|
||||
|
||||
using namespace AssemblyGui;
|
||||
|
||||
PROPERTY_SOURCE(AssemblyGui::ViewProviderViewGroup, Gui::ViewProviderDocumentObjectGroup)
|
||||
|
||||
ViewProviderViewGroup::ViewProviderViewGroup()
|
||||
{}
|
||||
|
||||
ViewProviderViewGroup::~ViewProviderViewGroup() = default;
|
||||
|
||||
QIcon ViewProviderViewGroup::getIcon() const
|
||||
{
|
||||
return Gui::BitmapFactory().pixmap("Assembly_ExplodedViewGroup.svg");
|
||||
}
|
||||
67
src/Mod/Assembly/Gui/ViewProviderViewGroup.h
Normal file
67
src/Mod/Assembly/Gui/ViewProviderViewGroup.h
Normal file
@@ -0,0 +1,67 @@
|
||||
// SPDX-License-Identifier: LGPL-2.1-or-later
|
||||
/****************************************************************************
|
||||
* *
|
||||
* Copyright (c) 2023 Ondsel <development@ondsel.com> *
|
||||
* *
|
||||
* This file is part of FreeCAD. *
|
||||
* *
|
||||
* FreeCAD is free software: you can redistribute it and/or modify it *
|
||||
* under the terms of the GNU Lesser General Public License as *
|
||||
* published by the Free Software Foundation, either version 2.1 of the *
|
||||
* License, or (at your option) any later version. *
|
||||
* *
|
||||
* FreeCAD is distributed in the hope that it will be useful, but *
|
||||
* WITHOUT ANY WARRANTY; without even the implied warranty of *
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU *
|
||||
* Lesser General Public License for more details. *
|
||||
* *
|
||||
* You should have received a copy of the GNU Lesser General Public *
|
||||
* License along with FreeCAD. If not, see *
|
||||
* <https://www.gnu.org/licenses/>. *
|
||||
* *
|
||||
***************************************************************************/
|
||||
|
||||
#ifndef ASSEMBLYGUI_VIEWPROVIDER_ViewProviderViewGroup_H
|
||||
#define ASSEMBLYGUI_VIEWPROVIDER_ViewProviderViewGroup_H
|
||||
|
||||
#include <Mod/Assembly/AssemblyGlobal.h>
|
||||
|
||||
#include <Gui/ViewProviderDocumentObjectGroup.h>
|
||||
|
||||
|
||||
namespace AssemblyGui
|
||||
{
|
||||
|
||||
class AssemblyGuiExport ViewProviderViewGroup: public Gui::ViewProviderDocumentObjectGroup
|
||||
{
|
||||
PROPERTY_HEADER_WITH_OVERRIDE(AssemblyGui::ViewProviderViewGroup);
|
||||
|
||||
public:
|
||||
ViewProviderViewGroup();
|
||||
~ViewProviderViewGroup() override;
|
||||
|
||||
/// deliver the icon shown in the tree view. Override from ViewProvider.h
|
||||
QIcon getIcon() const override;
|
||||
|
||||
// Prevent dragging of the joints and dropping things inside the joint group.
|
||||
bool canDragObjects() const override
|
||||
{
|
||||
return false;
|
||||
};
|
||||
bool canDropObjects() const override
|
||||
{
|
||||
return false;
|
||||
};
|
||||
bool canDragAndDropObject(App::DocumentObject*) const override
|
||||
{
|
||||
return false;
|
||||
};
|
||||
|
||||
// protected:
|
||||
/// get called by the container whenever a property has been changed
|
||||
// void onChanged(const App::Property* prop) override;
|
||||
};
|
||||
|
||||
} // namespace AssemblyGui
|
||||
|
||||
#endif // ASSEMBLYGUI_VIEWPROVIDER_ViewProviderViewGroup_H
|
||||
@@ -63,7 +63,7 @@ class AssemblyWorkbench(Workbench):
|
||||
# load the builtin modules
|
||||
from PySide import QtCore, QtGui
|
||||
from PySide.QtCore import QT_TRANSLATE_NOOP
|
||||
import CommandCreateAssembly, CommandInsertLink, CommandCreateJoint, CommandSolveAssembly, CommandExportASMT
|
||||
import CommandCreateAssembly, CommandInsertLink, CommandCreateJoint, CommandSolveAssembly, CommandExportASMT, CommandCreateView
|
||||
from Preferences import PreferencesPage
|
||||
|
||||
# from Preferences import preferences
|
||||
@@ -78,6 +78,7 @@ class AssemblyWorkbench(Workbench):
|
||||
"Assembly_CreateAssembly",
|
||||
"Assembly_InsertLink",
|
||||
"Assembly_SolveAssembly",
|
||||
"Assembly_CreateView",
|
||||
]
|
||||
|
||||
cmdListMenuOnly = [
|
||||
|
||||
@@ -603,6 +603,20 @@ def getJointGroup(assembly):
|
||||
return joint_group
|
||||
|
||||
|
||||
def getViewGroup(assembly):
|
||||
view_group = None
|
||||
|
||||
for obj in assembly.OutList:
|
||||
if obj.TypeId == "Assembly::ViewGroup":
|
||||
view_group = obj
|
||||
break
|
||||
|
||||
if not view_group:
|
||||
view_group = assembly.newObject("Assembly::ViewGroup", "Exploded Views")
|
||||
|
||||
return view_group
|
||||
|
||||
|
||||
def isAssemblyGrounded():
|
||||
assembly = activeAssembly()
|
||||
if not assembly:
|
||||
|
||||
Reference in New Issue
Block a user