/*************************************************************************** * Copyright (c) 2015 Stefan Tröger * * * * 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 # include # include # include # include # include # include # include #endif #include "ui_TaskPipeParameters.h" #include "ui_TaskPipeOrientation.h" #include "ui_TaskPipeScaling.h" #include #include "TaskPipeParameters.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "TaskSketchBasedParameters.h" #include "ReferenceSelection.h" #include "Utils.h" #include "TaskFeaturePick.h" Q_DECLARE_METATYPE(App::PropertyLinkSubList::SubSet); using namespace PartDesignGui; using namespace Gui; /* TRANSLATOR PartDesignGui::TaskPipeParameters */ //************************************************************************** //************************************************************************** // Task Parameter //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ TaskPipeParameters::TaskPipeParameters(ViewProviderPipe *PipeView, bool /*newObj*/, QWidget *parent) : TaskSketchBasedParameters(PipeView, parent, "PartDesign_AdditivePipe", tr("Pipe parameters")) , ui(new Ui_TaskPipeParameters) { // we need a separate container widget to add all controls to proxy = new QWidget(this); ui->setupUi(proxy); QMetaObject::connectSlotsByName(this); connect(ui->buttonProfileBase, SIGNAL(toggled(bool)), this, SLOT(onProfileButton(bool))); connect(ui->comboBoxTransition, SIGNAL(currentIndexChanged(int)), this, SLOT(onTransitionChanged(int))); connect(ui->buttonRefAdd, SIGNAL(toggled(bool)), this, SLOT(onButtonRefAdd(bool))); connect(ui->buttonRefRemove, SIGNAL(toggled(bool)), this, SLOT(onButtonRefRemove(bool))); connect(ui->buttonSpineBase, SIGNAL(toggled(bool)), this, SLOT(onBaseButton(bool))); // Create context menu QAction* remove = new QAction(tr("Remove"), this); remove->setShortcut(QKeySequence::Delete); remove->setShortcutContext(Qt::WidgetShortcut); #if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0) // display shortcut behind the context menu entry remove->setShortcutVisibleInContextMenu(true); #endif ui->listWidgetReferences->addAction(remove); connect(remove, SIGNAL(triggered()), this, SLOT(onDeleteEdge())); ui->listWidgetReferences->setContextMenuPolicy(Qt::ActionsContextMenu); this->groupLayout()->addWidget(proxy); PartDesign::Pipe* pipe = static_cast(PipeView->getObject()); Gui::Document* doc = PipeView->getDocument(); // make sure the user sees all important things and load the values // also save visibility state to reset it later when pipe is closed // first the spine if (pipe->Spine.getValue()) { auto* spineVP = doc->getViewProvider(pipe->Spine.getValue()); spineShow = spineVP->isShow(); spineVP->setVisible(true); ui->spineBaseEdit->setText(QString::fromUtf8(pipe->Spine.getValue()->Label.getValue())); } // the profile if (pipe->Profile.getValue()) { auto* profileVP = doc->getViewProvider(pipe->Profile.getValue()); profileShow = profileVP->isShow(); profileVP->setVisible(true); ui->profileBaseEdit->setText(make2DLabel(pipe->Profile.getValue(), pipe->Profile.getSubValues())); } // the auxiliary spine if (pipe->AuxillerySpine.getValue()) { auto* svp = doc->getViewProvider(pipe->AuxillerySpine.getValue()); auxSpineShow = svp->isShow(); svp->show(); } // the spine edges std::vector strings = pipe->Spine.getSubValues(); for (std::vector::const_iterator it = strings.begin(); it != strings.end(); ++it) { QString label = QString::fromStdString(*it); QListWidgetItem* item = new QListWidgetItem(); item->setText(label); item->setData(Qt::UserRole, QByteArray(label.toUtf8())); ui->listWidgetReferences->addItem(item); } if (!strings.empty()) { PipeView->makeTemporaryVisible(true); } ui->comboBoxTransition->setCurrentIndex(pipe->Transition.getValue()); updateUI(); } TaskPipeParameters::~TaskPipeParameters() { try { if (vp) { PartDesign::Pipe* pipe = static_cast(vp->getObject()); // setting visibility to true is needed when preselecting profile and path prior to invoking sweep Gui::cmdGuiObject(pipe, "Visibility = True"); static_cast(vp)->highlightReferences(ViewProviderPipe::Spine, false); static_cast(vp)->highlightReferences(ViewProviderPipe::Profile, false); } } catch (const Base::Exception& e) { // getDocument() may raise an exception e.ReportException(); } catch (const Py::Exception&) { Base::PyException e; // extract the Python error text e.ReportException(); } } void TaskPipeParameters::updateUI() { } void TaskPipeParameters::onSelectionChanged(const Gui::SelectionChanges& msg) { if (selectionMode == none) return; if (msg.Type == Gui::SelectionChanges::AddSelection) { if (referenceSelected(msg)) { if (selectionMode == refProfile) { App::Document* document = App::GetApplication().getDocument(msg.pDocName); App::DocumentObject* object = document ? document->getObject(msg.pObjectName) : nullptr; if (object) { QString label = make2DLabel(object, {msg.pSubName}); ui->profileBaseEdit->setText(label); } } else if (selectionMode == refAdd) { QString sub = QString::fromStdString(msg.pSubName); if (!sub.isEmpty()) { QListWidgetItem* item = new QListWidgetItem(); item->setText(sub); item->setData(Qt::UserRole, QByteArray(msg.pSubName)); ui->listWidgetReferences->addItem(item); } App::Document* document = App::GetApplication().getDocument(msg.pDocName); App::DocumentObject* object = document ? document->getObject(msg.pObjectName) : nullptr; if (object) { QString label = QString::fromUtf8(object->Label.getValue()); ui->spineBaseEdit->setText(label); } } else if (selectionMode == refRemove) { QString sub = QString::fromLatin1(msg.pSubName); if (!sub.isEmpty()) { removeFromListWidget(ui->listWidgetReferences, sub); } else { ui->spineBaseEdit->clear(); } } else if (selectionMode == refObjAdd) { ui->listWidgetReferences->clear(); App::Document* document = App::GetApplication().getDocument(msg.pDocName); App::DocumentObject* object = document ? document->getObject(msg.pObjectName) : nullptr; if (object) { QString label = QString::fromUtf8(object->Label.getValue()); ui->spineBaseEdit->setText(label); } } clearButtons(); recomputeFeature(); } clearButtons(); exitSelectionMode(); } } void TaskPipeParameters::onTransitionChanged(int idx) { static_cast(vp->getObject())->Transition.setValue(idx); recomputeFeature(); } void TaskPipeParameters::onButtonRefAdd(bool checked) { if (checked) { clearButtons(refAdd); //hideObject(); Gui::Selection().clearSelection(); selectionMode = refAdd; static_cast(vp)->highlightReferences(ViewProviderPipe::Spine, true); } else { Gui::Selection().clearSelection(); selectionMode = none; static_cast(vp)->highlightReferences(ViewProviderPipe::Spine, false); } } void TaskPipeParameters::onButtonRefRemove(bool checked) { if (checked) { clearButtons(refRemove); //hideObject(); Gui::Selection().clearSelection(); selectionMode = refRemove; static_cast(vp)->highlightReferences(ViewProviderPipe::Spine, true); } else { Gui::Selection().clearSelection(); selectionMode = none; static_cast(vp)->highlightReferences(ViewProviderPipe::Spine, false); } } void TaskPipeParameters::onBaseButton(bool checked) { if (checked) { clearButtons(refObjAdd); //hideObject(); Gui::Selection().clearSelection(); selectionMode = refObjAdd; static_cast(vp)->highlightReferences(ViewProviderPipe::Spine, true); } else { Gui::Selection().clearSelection(); selectionMode = none; static_cast(vp)->highlightReferences(ViewProviderPipe::Spine, false); } } void TaskPipeParameters::onProfileButton(bool checked) { if (checked) { PartDesign::Pipe* pipe = static_cast(vp->getObject()); Gui::Document* doc = vp->getDocument(); if (pipe->Profile.getValue()) { auto* pvp = doc->getViewProvider(pipe->Profile.getValue()); pvp->setVisible(true); } clearButtons(refProfile); //hideObject(); Gui::Selection().clearSelection(); selectionMode = refProfile; static_cast(vp)->highlightReferences(ViewProviderPipe::Profile, true); } else { Gui::Selection().clearSelection(); selectionMode = none; static_cast(vp)->highlightReferences(ViewProviderPipe::Profile, false); } } void TaskPipeParameters::onTangentChanged(bool checked) { static_cast(vp->getObject())->SpineTangent.setValue(checked); recomputeFeature(); } void TaskPipeParameters::removeFromListWidget(QListWidget* widget, QString itemstr) { QList items = widget->findItems(itemstr, Qt::MatchExactly); if (!items.empty()) { for (QList::const_iterator i = items.begin(); i != items.end(); i++) { QListWidgetItem* it = widget->takeItem(widget->row(*i)); delete it; } } } void TaskPipeParameters::onDeleteEdge() { // Delete the selected path edge int row = ui->listWidgetReferences->currentRow(); QListWidgetItem* item = ui->listWidgetReferences->takeItem(row); if (item) { QByteArray data = item->data(Qt::UserRole).toByteArray(); delete item; // search inside the list of spines PartDesign::Pipe* pipe = static_cast(vp->getObject()); std::vector refs = pipe->Spine.getSubValues(); std::string obj = data.constData(); std::vector::iterator f = std::find(refs.begin(), refs.end(), obj); // if something was found, delete it and update the spine list if (f != refs.end()) { refs.erase(f); pipe->Spine.setValue(pipe->Spine.getValue(), refs); clearButtons(); recomputeFeature(); } } } bool TaskPipeParameters::referenceSelected(const SelectionChanges& msg) const { if (msg.Type == Gui::SelectionChanges::AddSelection && selectionMode != none) { if (strcmp(msg.pDocName, vp->getObject()->getDocument()->getName()) != 0) return false; // not allowed to reference ourself const char* fname = vp->getObject()->getNameInDocument(); if (strcmp(msg.pObjectName, fname) == 0) return false; if (selectionMode == refProfile) { PartDesign::Pipe* pipe = static_cast(vp->getObject()); Gui::Document* doc = vp->getDocument(); static_cast(vp)->highlightReferences(ViewProviderPipe::Profile, false); bool success = true; App::DocumentObject* profile = pipe->getDocument()->getObject(msg.pObjectName); if (profile) { std::vector sections = pipe->Sections.getValues(); // cannot use the same object for profile and section if (std::find(sections.begin(), sections.end(), profile) != sections.end()) { success = false; } else { pipe->Profile.setValue(profile, {msg.pSubName}); } // hide the old or new profile again auto* pvp = doc->getViewProvider(pipe->Profile.getValue()); if (pvp) pvp->setVisible(false); } return success; } else if (selectionMode == refObjAdd || selectionMode == refAdd || selectionMode == refRemove) { //change the references std::string subName(msg.pSubName); std::vector refs = static_cast(vp->getObject())->Spine.getSubValues(); std::vector::iterator f = std::find(refs.begin(), refs.end(), subName); if (selectionMode == refObjAdd) { static_cast(vp)->highlightReferences(ViewProviderPipe::Spine, false); refs.clear(); } else if (selectionMode == refAdd) { if (f == refs.end()) refs.push_back(subName); else return false; // duplicate selection } else if (selectionMode == refRemove) { if (f != refs.end()) refs.erase(f); else return false; } static_cast(vp->getObject())->Spine.setValue( vp->getObject()->getDocument()->getObject(msg.pObjectName), refs); return true; } } return false; } void TaskPipeParameters::clearButtons(const selectionModes notThis) { // TODO: Clear buttons in the other pipe taskboxes as well if (notThis != refProfile) ui->buttonProfileBase->setChecked(false); if (notThis != refAdd) ui->buttonRefAdd->setChecked(false); if (notThis != refRemove) ui->buttonRefRemove->setChecked(false); if (notThis != refObjAdd) ui->buttonSpineBase->setChecked(false); } void TaskPipeParameters::exitSelectionMode() { selectionMode = none; Gui::Selection().clearSelection(); } void TaskPipeParameters::setVisibilityOfSpineAndProfile() { if (vp) { PartDesign::Pipe* pipe = static_cast(vp->getObject()); Gui::Document* doc = vp->getDocument(); // set visibility to the state when the pipe was opened for (auto obj : pipe->Sections.getValues()) { auto* sectionVP = doc->getViewProvider(obj); sectionVP->setVisible(profileShow); } if (pipe->Spine.getValue()) { auto* spineVP = doc->getViewProvider(pipe->Spine.getValue()); spineVP->setVisible(spineShow); spineShow = false; } if (pipe->Profile.getValue()) { auto* profileVP = doc->getViewProvider(pipe->Profile.getValue()); profileVP->setVisible(profileShow); profileShow = false; } if (pipe->AuxillerySpine.getValue()) { auto* svp = doc->getViewProvider(pipe->AuxillerySpine.getValue()); svp->setVisible(auxSpineShow); auxSpineShow = false; } } } bool TaskPipeParameters::accept() { //see what to do with external references //check the prerequisites for the selected objects //the user has to decide which option we should take if external references are used PartDesign::Pipe* pcPipe = static_cast(getPipeView()->getObject()); auto pcActiveBody = PartDesignGui::getBodyFor (pcPipe, false); if (!pcActiveBody) { QMessageBox::warning(this, tr("Input error"), tr("No active body")); return false; } //auto pcActivePart = PartDesignGui::getPartFor (pcActiveBody, false); std::vector copies; bool extReference = false; App::DocumentObject* spine = pcPipe->Spine.getValue(); App::DocumentObject* auxSpine = pcPipe->AuxillerySpine.getValue(); // If a spine isn't set but user entered a label then search for the appropriate document object QString label = ui->spineBaseEdit->text(); if (!spine && !label.isEmpty()) { QByteArray ba = label.toUtf8(); std::vector objs = pcPipe->getDocument()->findObjects(App::DocumentObject::getClassTypeId(), nullptr, ba.constData()); if (!objs.empty()) { pcPipe->Spine.setValue(objs.front()); spine = objs.front(); } } if (spine && !pcActiveBody->hasObject(spine) && !pcActiveBody->getOrigin()->hasObject(spine)) { extReference = true; } else if (auxSpine && !pcActiveBody->hasObject(auxSpine) && !pcActiveBody->getOrigin()->hasObject(auxSpine)) { extReference = true; } else { for (App::DocumentObject* obj : pcPipe->Sections.getValues()) { if (!pcActiveBody->hasObject(obj) && !pcActiveBody->getOrigin()->hasObject(obj)) { extReference = true; break; } } } if (extReference) { QDialog dia(Gui::getMainWindow()); Ui_DlgReference dlg; dlg.setupUi(&dia); dia.setModal(true); int result = dia.exec(); if (result == QDialog::DialogCode::Rejected) return false; if (!dlg.radioXRef->isChecked()) { if (!pcActiveBody->hasObject(spine) && !pcActiveBody->getOrigin()->hasObject(spine)) { pcPipe->Spine.setValue(PartDesignGui::TaskFeaturePick::makeCopy(spine, "", dlg.radioIndependent->isChecked()), pcPipe->Spine.getSubValues()); copies.push_back(pcPipe->Spine.getValue()); } else if (!pcActiveBody->hasObject(auxSpine) && !pcActiveBody->getOrigin()->hasObject(auxSpine)){ pcPipe->AuxillerySpine.setValue(PartDesignGui::TaskFeaturePick::makeCopy(auxSpine, "", dlg.radioIndependent->isChecked()), pcPipe->AuxillerySpine.getSubValues()); copies.push_back(pcPipe->AuxillerySpine.getValue()); } std::vector subSets; int index = 0; for (auto &subSet : pcPipe->Sections.getSubListValues()) { if (!pcActiveBody->hasObject(subSet.first) && !pcActiveBody->getOrigin()->hasObject(subSet.first)) { subSets.push_back( std::make_pair( PartDesignGui::TaskFeaturePick::makeCopy( subSet.first, "", dlg.radioIndependent->isChecked()), subSet.second)); copies.push_back(subSets.back().first); } else { subSets.push_back(subSet); } index++; } pcPipe->Sections.setSubListValues(subSets); } } try { setVisibilityOfSpineAndProfile(); App::DocumentObject* spine = pcPipe->Spine.getValue(); std::vector subNames = pcPipe->Spine.getSubValues(); App::PropertyLinkT propT(spine, subNames); Gui::cmdAppObjectArgs(pcPipe, "Spine = %s", propT.getPropertyPython()); Gui::cmdAppDocument(pcPipe, "recompute()"); if (!vp->getObject()->isValid()) throw Base::RuntimeError(vp->getObject()->getStatusString()); Gui::cmdGuiDocument(pcPipe, "resetEdit()"); Gui::Command::commitCommand(); //we need to add the copied features to the body after the command action, as otherwise FreeCAD crashes unexplainably for (auto obj : copies) { pcActiveBody->addObject(obj); } } catch (const Base::Exception& e) { QMessageBox::warning(this, tr("Input error"), QString::fromUtf8(e.what())); return false; } return true; } //************************************************************************** //************************************************************************** // Task Orientation //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ TaskPipeOrientation::TaskPipeOrientation(ViewProviderPipe* PipeView, bool /*newObj*/, QWidget* parent) : TaskSketchBasedParameters(PipeView, parent, "PartDesign_AdditivePipe", tr("Section orientation")), ui(new Ui_TaskPipeOrientation) { // we need a separate container widget to add all controls to proxy = new QWidget(this); ui->setupUi(proxy); QMetaObject::connectSlotsByName(this); connect(ui->comboBoxMode, SIGNAL(currentIndexChanged(int)), this, SLOT(onOrientationChanged(int))); connect(ui->buttonRefAdd, SIGNAL(toggled(bool)), this, SLOT(onButtonRefAdd(bool))); connect(ui->buttonRefRemove, SIGNAL(toggled(bool)), this, SLOT(onButtonRefRemove(bool))); connect(ui->buttonProfileBase, SIGNAL(toggled(bool)), this, SLOT(onBaseButton(bool))); connect(ui->buttonProfileClear, SIGNAL(clicked()), this, SLOT(onClearButton())); connect(ui->stackedWidget, SIGNAL(currentChanged(int)), this, SLOT(updateUI(int))); connect(ui->curvelinear, SIGNAL(toggled(bool)), this, SLOT(onCurvelinearChanged(bool))); connect(ui->doubleSpinBoxX, SIGNAL(valueChanged(double)), this, SLOT(onBinormalChanged(double))); connect(ui->doubleSpinBoxY, SIGNAL(valueChanged(double)), this, SLOT(onBinormalChanged(double))); connect(ui->doubleSpinBoxZ, SIGNAL(valueChanged(double)), this, SLOT(onBinormalChanged(double))); // Create context menu QAction* remove = new QAction(tr("Remove"), this); remove->setShortcut(QKeySequence::Delete); remove->setShortcutContext(Qt::WidgetShortcut); #if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0) // display shortcut behind the context menu entry remove->setShortcutVisibleInContextMenu(true); #endif ui->listWidgetReferences->addAction(remove); connect(remove, SIGNAL(triggered()), this, SLOT(onDeleteItem())); ui->listWidgetReferences->setContextMenuPolicy(Qt::ActionsContextMenu); this->groupLayout()->addWidget(proxy); PartDesign::Pipe* pipe = static_cast(PipeView->getObject()); //add initial values if (pipe->AuxillerySpine.getValue()) ui->profileBaseEdit->setText(QString::fromUtf8(pipe->AuxillerySpine.getValue()->Label.getValue())); std::vector strings = pipe->AuxillerySpine.getSubValues(); for (std::vector::const_iterator it = strings.begin(); it != strings.end(); ++it) { QString label = QString::fromStdString(*it); QListWidgetItem* item = new QListWidgetItem(); item->setText(label); item->setData(Qt::UserRole, QByteArray(label.toUtf8())); ui->listWidgetReferences->addItem(item); } ui->comboBoxMode->setCurrentIndex(pipe->Mode.getValue()); ui->curvelinear->setChecked(pipe->AuxilleryCurvelinear.getValue()); // should be called after panel has become visible QMetaObject::invokeMethod(this, "updateUI", Qt::QueuedConnection, QGenericReturnArgument(), Q_ARG(int,pipe->Mode.getValue())); } TaskPipeOrientation::~TaskPipeOrientation() { if (vp) { static_cast(vp)->highlightReferences(ViewProviderPipe::AuxiliarySpine, false); } } void TaskPipeOrientation::onOrientationChanged(int idx) { static_cast(vp->getObject())->Mode.setValue(idx); recomputeFeature(); } void TaskPipeOrientation::clearButtons(const selectionModes notThis) { // TODO: Clear buttons in the other pipe taskboxes as well if (notThis != refAdd) ui->buttonRefAdd->setChecked(false); if (notThis != refRemove) ui->buttonRefRemove->setChecked(false); if (notThis != refObjAdd) ui->buttonProfileBase->setChecked(false); } void TaskPipeOrientation::exitSelectionMode() { selectionMode = none; Gui::Selection().clearSelection(); } void TaskPipeOrientation::onButtonRefAdd(bool checked) { if (checked) { clearButtons(refAdd); Gui::Selection().clearSelection(); selectionMode = refAdd; static_cast(vp)->highlightReferences(ViewProviderPipe::AuxiliarySpine, true); } else { Gui::Selection().clearSelection(); selectionMode = none; static_cast(vp)->highlightReferences(ViewProviderPipe::AuxiliarySpine, false); } } void TaskPipeOrientation::onButtonRefRemove(bool checked) { if (checked) { clearButtons(refRemove); Gui::Selection().clearSelection(); selectionMode = refRemove; static_cast(vp)->highlightReferences(ViewProviderPipe::AuxiliarySpine, true); } else { Gui::Selection().clearSelection(); selectionMode = none; static_cast(vp)->highlightReferences(ViewProviderPipe::AuxiliarySpine, false); } } void TaskPipeOrientation::onBaseButton(bool checked) { if (checked) { clearButtons(refObjAdd); Gui::Selection().clearSelection(); selectionMode = refObjAdd; static_cast(vp)->highlightReferences(ViewProviderPipe::AuxiliarySpine, true); } else { Gui::Selection().clearSelection(); selectionMode = none; static_cast(vp)->highlightReferences(ViewProviderPipe::AuxiliarySpine, false); } } void TaskPipeOrientation::onClearButton() { static_cast(vp)->highlightReferences(ViewProviderPipe::AuxiliarySpine, false); ui->listWidgetReferences->clear(); ui->profileBaseEdit->clear(); static_cast(vp->getObject())->AuxillerySpine.setValue(nullptr); } void TaskPipeOrientation::onCurvelinearChanged(bool checked) { static_cast(vp->getObject())->AuxilleryCurvelinear.setValue(checked); recomputeFeature(); } void TaskPipeOrientation::onBinormalChanged(double) { Base::Vector3d vec(ui->doubleSpinBoxX->value(), ui->doubleSpinBoxY->value(), ui->doubleSpinBoxZ->value()); static_cast(vp->getObject())->Binormal.setValue(vec); recomputeFeature(); } void TaskPipeOrientation::onSelectionChanged(const SelectionChanges& msg) { if (selectionMode == none) return; if (msg.Type == Gui::SelectionChanges::AddSelection) { if (referenceSelected(msg)) { if (selectionMode == refAdd) { QString sub = QString::fromStdString(msg.pSubName); if (!sub.isEmpty()) { QListWidgetItem* item = new QListWidgetItem(); item->setText(sub); item->setData(Qt::UserRole, QByteArray(msg.pSubName)); ui->listWidgetReferences->addItem(item); } App::Document* document = App::GetApplication().getDocument(msg.pDocName); App::DocumentObject* object = document ? document->getObject(msg.pObjectName) : nullptr; if (object) { QString label = QString::fromUtf8(object->Label.getValue()); ui->profileBaseEdit->setText(label); } } else if (selectionMode == refRemove) { QString sub = QString::fromLatin1(msg.pSubName); if (!sub.isEmpty()) removeFromListWidget(ui->listWidgetReferences, sub); else { ui->profileBaseEdit->clear(); } } else if (selectionMode == refObjAdd) { ui->listWidgetReferences->clear(); App::Document* document = App::GetApplication().getDocument(msg.pDocName); App::DocumentObject* object = document ? document->getObject(msg.pObjectName) : nullptr; if (object) { QString label = QString::fromUtf8(object->Label.getValue()); ui->profileBaseEdit->setText(label); } } clearButtons(); static_cast(vp)->highlightReferences(ViewProviderPipe::AuxiliarySpine, false); recomputeFeature(); } clearButtons(); exitSelectionMode(); } } bool TaskPipeOrientation::referenceSelected(const SelectionChanges& msg) const { if (msg.Type == Gui::SelectionChanges::AddSelection && selectionMode != none) { if (strcmp(msg.pDocName, vp->getObject()->getDocument()->getName()) != 0) return false; // not allowed to reference ourself const char* fname = vp->getObject()->getNameInDocument(); if (strcmp(msg.pObjectName, fname) == 0) return false; //change the references std::string subName(msg.pSubName); std::vector refs = static_cast(vp->getObject())->AuxillerySpine.getSubValues(); std::vector::iterator f = std::find(refs.begin(), refs.end(), subName); if (selectionMode == refObjAdd) { refs.clear(); } else if (selectionMode == refAdd) { if (f == refs.end()) refs.push_back(subName); else return false; // duplicate selection } else if (selectionMode == refRemove) { if (f != refs.end()) refs.erase(f); else return false; } static_cast(vp->getObject())->AuxillerySpine.setValue (vp->getObject()->getDocument()->getObject(msg.pObjectName), refs); return true; } return false; } void TaskPipeOrientation::removeFromListWidget(QListWidget* widget, QString name) { QList items = widget->findItems(name, Qt::MatchExactly); if (!items.empty()) { for (QList::const_iterator i = items.begin(); i != items.end(); i++) { QListWidgetItem* it = widget->takeItem(widget->row(*i)); delete it; } } } void TaskPipeOrientation::onDeleteItem() { // Delete the selected spine int row = ui->listWidgetReferences->currentRow(); QListWidgetItem* item = ui->listWidgetReferences->takeItem(row); if (item) { QByteArray data = item->data(Qt::UserRole).toByteArray(); delete item; // search inside the list of spines PartDesign::Pipe* pipe = static_cast(vp->getObject()); std::vector refs = pipe->AuxillerySpine.getSubValues(); std::string obj = data.constData(); std::vector::iterator f = std::find(refs.begin(), refs.end(), obj); // if something was found, delete it and update the spine list if (f != refs.end()) { refs.erase(f); pipe->AuxillerySpine.setValue(pipe->AuxillerySpine.getValue(), refs); clearButtons(); recomputeFeature(); } } } void TaskPipeOrientation::updateUI(int idx) { //make sure we resize to the size of the current page for (int i=0; istackedWidget->count(); ++i) ui->stackedWidget->widget(i)->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Ignored); if (idx < ui->stackedWidget->count()) ui->stackedWidget->widget(idx)->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); } //************************************************************************** //************************************************************************** // Task Scaling //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ TaskPipeScaling::TaskPipeScaling(ViewProviderPipe* PipeView, bool /*newObj*/, QWidget* parent) : TaskSketchBasedParameters(PipeView, parent, "PartDesign_AdditivePipe", tr("Section transformation")), ui(new Ui_TaskPipeScaling) { // we need a separate container widget to add all controls to proxy = new QWidget(this); ui->setupUi(proxy); QMetaObject::connectSlotsByName(this); connect(ui->comboBoxScaling, SIGNAL(currentIndexChanged(int)), this, SLOT(onScalingChanged(int))); connect(ui->buttonRefAdd, SIGNAL(toggled(bool)), this, SLOT(onButtonRefAdd(bool))); connect(ui->buttonRefRemove, SIGNAL(toggled(bool)), this, SLOT(onButtonRefRemove(bool))); connect(ui->stackedWidget, SIGNAL(currentChanged(int)), this, SLOT(updateUI(int))); // Create context menu QAction* remove = new QAction(tr("Remove"), this); remove->setShortcut(QKeySequence::Delete); remove->setShortcutContext(Qt::WidgetShortcut); #if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0) // display shortcut behind the context menu entry remove->setShortcutVisibleInContextMenu(true); #endif ui->listWidgetReferences->addAction(remove); ui->listWidgetReferences->setContextMenuPolicy(Qt::ActionsContextMenu); connect(remove, SIGNAL(triggered()), this, SLOT(onDeleteSection())); this->groupLayout()->addWidget(proxy); PartDesign::Pipe* pipe = static_cast(PipeView->getObject()); for (auto &subSet : pipe->Sections.getSubListValues()) { Gui::Application::Instance->showViewProvider(subSet.first); QString label = make2DLabel(subSet.first, subSet.second); QListWidgetItem* item = new QListWidgetItem(); item->setText(label); item->setData(Qt::UserRole, QVariant::fromValue(subSet)); ui->listWidgetReferences->addItem(item); } ui->comboBoxScaling->setCurrentIndex(pipe->Transformation.getValue()); // should be called after panel has become visible QMetaObject::invokeMethod(this, "updateUI", Qt::QueuedConnection, QGenericReturnArgument(), Q_ARG(int,pipe->Transformation.getValue())); } TaskPipeScaling::~TaskPipeScaling() { if (vp) { static_cast(vp)->highlightReferences(ViewProviderPipe::Section, false); } } void TaskPipeScaling::clearButtons(const selectionModes notThis) { // TODO: Clear buttons in the other pipe taskboxes as well if (notThis != refRemove) ui->buttonRefRemove->setChecked(false); if (notThis != refAdd) ui->buttonRefAdd->setChecked(false); } void TaskPipeScaling::exitSelectionMode() { selectionMode = none; Gui::Selection().clearSelection(); } void TaskPipeScaling::onButtonRefAdd(bool checked) { if (checked) { clearButtons(refAdd); Gui::Selection().clearSelection(); selectionMode = refAdd; static_cast(vp)->highlightReferences(ViewProviderPipe::Section, true); } else { Gui::Selection().clearSelection(); selectionMode = none; static_cast(vp)->highlightReferences(ViewProviderPipe::Section, false); } } void TaskPipeScaling::onButtonRefRemove(bool checked) { if (checked) { clearButtons(refRemove); Gui::Selection().clearSelection(); selectionMode = refRemove; static_cast(vp)->highlightReferences(ViewProviderPipe::Section, true); } else { Gui::Selection().clearSelection(); selectionMode = none; static_cast(vp)->highlightReferences(ViewProviderPipe::Section, false); } } void TaskPipeScaling::onScalingChanged(int idx) { updateUI(idx); static_cast(vp->getObject())->Transformation.setValue(idx); } void TaskPipeScaling::onSelectionChanged(const SelectionChanges& msg) { if (selectionMode == none) return; if (msg.Type == Gui::SelectionChanges::AddSelection) { if (referenceSelected(msg)) { App::Document* document = App::GetApplication().getDocument(msg.pDocName); App::DocumentObject* object = document ? document->getObject(msg.pObjectName) : nullptr; if (object) { QString label = make2DLabel(object, {msg.pSubName}); if (selectionMode == refAdd) { QListWidgetItem* item = new QListWidgetItem(); item->setText(label); item->setData(Qt::UserRole, QVariant::fromValue(std::make_pair(object, std::vector(1, msg.pSubName)))); ui->listWidgetReferences->addItem(item); } else if (selectionMode == refRemove) { removeFromListWidget(ui->listWidgetReferences, label); } } clearButtons(); recomputeFeature(); } clearButtons(); exitSelectionMode(); } } bool TaskPipeScaling::referenceSelected(const SelectionChanges& msg) const { if ((msg.Type == Gui::SelectionChanges::AddSelection) && ( (selectionMode == refAdd) || (selectionMode == refRemove))) { if (strcmp(msg.pDocName, vp->getObject()->getDocument()->getName()) != 0) return false; // not allowed to reference ourself const char* fname = vp->getObject()->getNameInDocument(); if (strcmp(msg.pObjectName, fname) == 0) return false; //every selection needs to be a profile in itself, hence currently only full objects are //supported, not individual edges of a part //change the references PartDesign::Pipe* pipe = static_cast(vp->getObject()); std::vector refs = pipe->Sections.getValues(); App::DocumentObject* obj = vp->getObject()->getDocument()->getObject(msg.pObjectName); std::vector::iterator f = std::find(refs.begin(), refs.end(), obj); if (selectionMode == refAdd) { if (f == refs.end()) pipe->Sections.addValue(obj, {msg.pSubName}); else return false; // duplicate selection } else { if (f != refs.end()) // Removing just the object this way instead of `refs.erase` and // `setValues(ref)` cleanly ensures subnames are preserved. pipe->Sections.removeValue(obj); else return false; } static_cast(vp)->highlightReferences(ViewProviderPipe::Section, false); return true; } return false; } void TaskPipeScaling::removeFromListWidget(QListWidget* widget, QString name) { QList items = widget->findItems(name, Qt::MatchExactly); if (!items.empty()) { for (QList::const_iterator i = items.begin(); i != items.end(); i++) { QListWidgetItem* it = widget->takeItem(widget->row(*i)); delete it; } } } void TaskPipeScaling::onDeleteSection() { // Delete the selected profile int row = ui->listWidgetReferences->currentRow(); QListWidgetItem* item = ui->listWidgetReferences->takeItem(row); if (item) { QByteArray data(item->data(Qt::UserRole).value().first->getNameInDocument()); delete item; // search inside the list of sections PartDesign::Pipe* pipe = static_cast(vp->getObject()); std::vector refs = pipe->Sections.getValues(); App::DocumentObject* obj = pipe->getDocument()->getObject(data.constData()); std::vector::iterator f = std::find(refs.begin(), refs.end(), obj); // if something was found, delete it and update the section list if (f != refs.end()) { // Removing just the object this way instead of `refs.erase` and // `setValues(ref)` cleanly ensures subnames are preserved. pipe->Sections.removeValue(obj); clearButtons(); recomputeFeature(); } } } void TaskPipeScaling::updateUI(int idx) { //make sure we resize to the size of the current page for (int i=0; istackedWidget->count(); ++i) ui->stackedWidget->widget(i)->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Ignored); if (idx < ui->stackedWidget->count()) ui->stackedWidget->widget(idx)->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); } //************************************************************************** //************************************************************************** // TaskDialog //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ TaskDlgPipeParameters::TaskDlgPipeParameters(ViewProviderPipe *PipeView,bool newObj) : TaskDlgSketchBasedParameters(PipeView) { assert(PipeView); parameter = new TaskPipeParameters(PipeView,newObj); orientation = new TaskPipeOrientation(PipeView,newObj); scaling = new TaskPipeScaling(PipeView,newObj); Content.push_back(parameter); Content.push_back(orientation); Content.push_back(scaling); } TaskDlgPipeParameters::~TaskDlgPipeParameters() { } //==== calls from the TaskView =============================================================== bool TaskDlgPipeParameters::accept() { return parameter->accept(); } #include "moc_TaskPipeParameters.cpp"