Files
create/src/Mod/Fem/Gui/ViewProviderAnalysis.cpp
2025-06-30 10:35:42 -05:00

355 lines
12 KiB
C++

/***************************************************************************
* Copyright (c) 2013 Jürgen Riegel <FreeCAD@juergen-riegel.net> *
* *
* 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"
#ifndef _PreComp_
#include <Inventor/nodes/SoSeparator.h>
#include <QAction>
#include <QApplication>
#include <QMenu>
#include <QMessageBox>
#include <QTextStream>
#endif
#include <App/Document.h>
#include <App/MaterialObject.h>
#include <App/TextDocument.h>
#include <Gui/ActionFunction.h>
#include <Gui/Command.h>
#include <Gui/Control.h>
#include <Gui/Document.h>
#include <Gui/MainWindow.h>
#include <Gui/Selection/Selection.h>
#include <Gui/Selection/SelectionObject.h>
#include <Gui/Workbench.h>
#include <Gui/WorkbenchManager.h>
#include <Mod/Fem/App/FemAnalysis.h>
#include <Mod/Fem/App/FemConstraint.h>
#include <Mod/Fem/App/FemMeshObject.h>
#include <Mod/Fem/App/FemResultObject.h>
#include <Mod/Fem/App/FemSetObject.h>
#include <Mod/Fem/App/FemSolverObject.h>
#ifdef FC_USE_VTK
#include <Mod/Fem/App/FemPostObject.h>
#endif
#include "TaskDlgAnalysis.h"
#include "ViewProviderAnalysis.h"
using namespace FemGui;
ViewProviderFemHighlighter::ViewProviderFemHighlighter()
{
annotate = new SoSeparator();
annotate->ref();
}
ViewProviderFemHighlighter::~ViewProviderFemHighlighter()
{
annotate->unref();
}
void ViewProviderFemHighlighter::attach(ViewProviderFemAnalysis* view)
{
SoGroup* root = view->getRoot();
root->addChild(annotate);
}
void ViewProviderFemHighlighter::highlightView(Gui::ViewProviderDocumentObject* view)
{
annotate->removeAllChildren();
if (view) {
annotate->addChild(view->getRoot());
}
}
void ViewProviderFemHighlighter::removeView(Gui::ViewProviderDocumentObject* view)
{
if (view) {
annotate->removeChild(view->getRoot());
}
}
// ----------------------------------------------------------------------------
/* TRANSLATOR FemGui::ViewProviderFemAnalysis */
PROPERTY_SOURCE(FemGui::ViewProviderFemAnalysis, Gui::ViewProviderDocumentObjectGroup)
ViewProviderFemAnalysis::ViewProviderFemAnalysis()
{
setToggleVisibility(ToggleVisibilityMode::NoToggleVisibility);
sPixmap = "FEM_Analysis";
}
ViewProviderFemAnalysis::~ViewProviderFemAnalysis() = default;
void ViewProviderFemAnalysis::attach(App::DocumentObject* obj)
{
Gui::ViewProviderDocumentObjectGroup::attach(obj);
extension.attach(this);
}
void ViewProviderFemAnalysis::highlightView(Gui::ViewProviderDocumentObject* view)
{
extension.highlightView(view);
}
void ViewProviderFemAnalysis::removeView(Gui::ViewProviderDocumentObject* view)
{
extension.removeView(view);
}
bool ViewProviderFemAnalysis::doubleClicked()
{
Gui::Command::assureWorkbench("FemWorkbench");
Gui::Command::addModule(Gui::Command::Gui, "FemGui");
Gui::Command::doCommand(Gui::Command::Gui,
"FemGui.setActiveAnalysis(App.activeDocument().%s)",
this->getObject()->getNameInDocument());
// After activation of the analysis the allowed FEM toolbar buttons should become active.
// To achieve this we must clear the object selection to trigger the selection observer.
Gui::Command::doCommand(Gui::Command::Gui, "Gui.Selection.clearSelection()");
// indicate the activated analysis by selecting it
// especially useful for files with 2 or more analyses but also
// necessary for the workflow with new files to add a solver as next object
std::vector<App::DocumentObject*> selVector {};
selVector.push_back(this->getObject());
auto* docName = this->getObject()->getDocument()->getName();
Gui::Selection().setSelection(docName, selVector);
return true;
}
std::vector<App::DocumentObject*> ViewProviderFemAnalysis::claimChildren() const
{
return Gui::ViewProviderDocumentObjectGroup::claimChildren();
}
void ViewProviderFemAnalysis::hide()
{
Gui::ViewProviderDocumentObjectGroup::hide();
}
void ViewProviderFemAnalysis::show()
{
Gui::ViewProviderDocumentObjectGroup::show();
}
void ViewProviderFemAnalysis::setupContextMenu(QMenu* menu, QObject*, const char*)
{
Gui::ActionFunction* func = new Gui::ActionFunction(menu);
QAction* act = menu->addAction(tr("Activate Analysis"));
func->trigger(act, [this]() {
this->doubleClicked();
});
}
bool ViewProviderFemAnalysis::setEdit(int ModNum)
{
if (ModNum == ViewProvider::Default) {
// When double-clicking on the item for this pad the object
// unsets and sets its edit mode without closing the task panel
// Gui::TaskView::TaskDialog *dlg = Gui::Control().activeDialog();
// TaskDlgAnalysis *anaDlg = qobject_cast<TaskDlgAnalysis *>(dlg);
// if (padDlg && anaDlg->getPadView() != this)
// padDlg = 0; // another pad left open its task panel
// if (dlg && !padDlg) {
// QMessageBox msgBox;
// msgBox.setText(QObject::tr("A dialog is already open in the task panel"));
// msgBox.setInformativeText(QObject::tr("Do you want to close this dialog?"));
// msgBox.setStandardButtons(QMessageBox::Yes | QMessageBox::No);
// msgBox.setDefaultButton(QMessageBox::Yes);
// int ret = msgBox.exec();
// if (ret == QMessageBox::Yes)
// Gui::Control().closeDialog();
// else
// return false;
// }
// start the edit dialog
// if (padDlg)
// Gui::Control().showDialog(padDlg);
// else
// Fem::FemAnalysis* pcAna = this->getObject<Fem::FemAnalysis>();
// Gui::Control().showDialog(new TaskDlgAnalysis(pcAna));
// return true;
return false;
}
else {
return Gui::ViewProviderDocumentObjectGroup::setEdit(ModNum);
}
}
void ViewProviderFemAnalysis::unsetEdit(int ModNum)
{
if (ModNum == ViewProvider::Default) {
// when pressing ESC make sure to close the dialog
Gui::Control().closeDialog();
}
else {
Gui::ViewProviderDocumentObjectGroup::unsetEdit(ModNum);
}
}
bool ViewProviderFemAnalysis::canDragObjects() const
{
return true;
}
bool ViewProviderFemAnalysis::canDragObject(App::DocumentObject* obj) const
{
if (!obj) {
return false;
}
// clang-format off: keep line breaks for readability
if (obj->isDerivedFrom<Fem::FemMeshObject>()
|| obj->isDerivedFrom<Fem::FemSolverObject>()
|| obj->isDerivedFrom<Fem::FemResultObject>()
|| obj->isDerivedFrom<Fem::Constraint>()
|| obj->isDerivedFrom<Fem::FemSetObject>()
|| obj->isDerivedFrom(Base::Type::fromName("Fem::FeaturePython"))
|| obj->isDerivedFrom<App::MaterialObject>()
|| obj->isDerivedFrom<App::TextDocument>()) {
return true;
}
// clang-format on
#ifdef FC_USE_VTK
else if (obj->isDerivedFrom<Fem::FemPostObject>()) {
return true;
}
#endif
return false;
}
void ViewProviderFemAnalysis::dragObject(App::DocumentObject* obj)
{
ViewProviderDocumentObjectGroup::dragObject(obj);
}
bool ViewProviderFemAnalysis::canDropObjects() const
{
return true;
}
bool ViewProviderFemAnalysis::canDropObject(App::DocumentObject* obj) const
{
return canDragObject(obj);
}
void ViewProviderFemAnalysis::dropObject(App::DocumentObject* obj)
{
ViewProviderDocumentObjectGroup::dropObject(obj);
}
bool ViewProviderFemAnalysis::onDelete(const std::vector<std::string>&)
{
// warn the user if the object has unselected children
auto objs = claimChildren();
return checkSelectedChildren(objs, this->getDocument(), "analysis");
}
bool ViewProviderFemAnalysis::checkSelectedChildren(const std::vector<App::DocumentObject*> objs,
Gui::Document* docGui,
std::string objectName)
{
// warn the user if the object has unselected children
if (!objs.empty()) {
// check if all children are in the selection
bool found = false;
auto selectionList = Gui::Selection().getSelectionEx(docGui->getDocument()->getName());
for (auto child : objs) {
found = false;
for (Gui::SelectionObject selection : selectionList) {
if (std::string(child->getNameInDocument())
== std::string(selection.getFeatName())) {
found = true;
break;
}
}
if (!found) {
break;
}
}
if (found) { // all children are selected too
return true;
}
// generate dialog
QString bodyMessage;
QTextStream bodyMessageStream(&bodyMessage);
bodyMessageStream << qApp->translate("Std_Delete",
("The " + objectName
+ " is not empty, therefore the\nfollowing "
"referencing objects might be lost:")
.c_str());
bodyMessageStream << '\n';
for (auto ObjIterator : objs) {
bodyMessageStream << '\n' << QString::fromUtf8(ObjIterator->Label.getValue());
}
bodyMessageStream << "\n\n" << QObject::tr("Are you sure you want to continue?");
// show and evaluate the dialog
int DialogResult =
QMessageBox::warning(Gui::getMainWindow(),
qApp->translate("Std_Delete", "Object dependencies"),
bodyMessage,
QMessageBox::Yes,
QMessageBox::No);
if (DialogResult == QMessageBox::Yes) {
return true;
}
else {
return false;
}
}
else {
return true;
}
}
bool ViewProviderFemAnalysis::canDelete(App::DocumentObject* obj) const
{
// deletions of objects from a FemAnalysis don't necessarily destroy anything
// thus we can pass this action
// we can warn the user if necessary in the object's ViewProvider in the onDelete() function
Q_UNUSED(obj)
return true;
}
// Python feature -----------------------------------------------------------------------
namespace Gui
{
/// @cond DOXERR
PROPERTY_SOURCE_TEMPLATE(FemGui::ViewProviderFemAnalysisPython, FemGui::ViewProviderFemAnalysis)
/// @endcond
// explicit template instantiation
template class FemGuiExport ViewProviderFeaturePythonT<ViewProviderFemAnalysis>;
} // namespace Gui