Files
create/src/Gui/ViewProviderInventorObject.cpp
berniev ae53c9b0a4 Gui: Use auto and range-based for (#7481)
* On lines where the variable type is obvious from inspection, avoid repeating the type using auto. 
* When possible use a ranged for loop instead of begin() and end() iterators
2022-09-14 13:25:13 -05:00

167 lines
6.9 KiB
C++

/***************************************************************************
* Copyright (c) 2007 Werner Mayer <wmayer[at]users.sourceforge.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/SoDB.h>
# include <Inventor/SoInput.h>
# include <Inventor/nodes/SoSeparator.h>
# include <Inventor/nodes/SoTransform.h>
# include <QFile>
#endif
#include <App/Document.h>
#include <App/InventorObject.h>
#include "ViewProviderInventorObject.h"
#include "SoFCSelection.h"
using namespace Gui;
PROPERTY_SOURCE(Gui::ViewProviderInventorObject, Gui::ViewProviderDocumentObject)
ViewProviderInventorObject::ViewProviderInventorObject()
{
pcBuffer = new SoSeparator();
pcBuffer->ref();
pcFile = new SoSeparator();
pcFile->ref();
}
ViewProviderInventorObject::~ViewProviderInventorObject()
{
pcBuffer->unref();
pcFile->unref();
}
void ViewProviderInventorObject::attach(App::DocumentObject *pcObj)
{
ViewProviderDocumentObject::attach(pcObj);
auto pcFileBuf = new SoGroup();
pcFileBuf->addChild(pcBuffer);
pcFileBuf->addChild(pcFile);
addDisplayMaskMode(pcFileBuf, "FileBuffer");
addDisplayMaskMode(pcBuffer, "Buffer");
addDisplayMaskMode(pcFile, "File");
}
void ViewProviderInventorObject::setDisplayMode(const char* ModeName)
{
if (strcmp("File+Buffer",ModeName)==0)
setDisplayMaskMode("FileBuffer");
else if (strcmp("Buffer",ModeName)==0)
setDisplayMaskMode("Buffer");
else if (strcmp("File",ModeName)==0)
setDisplayMaskMode("File");
ViewProviderDocumentObject::setDisplayMode(ModeName);
}
std::vector<std::string> ViewProviderInventorObject::getDisplayModes() const
{
std::vector<std::string> StrList;
StrList.emplace_back("File+Buffer");
StrList.emplace_back("Buffer");
StrList.emplace_back("File");
return StrList;
}
void ViewProviderInventorObject::updateData(const App::Property* prop)
{
auto ivObj = static_cast<App::InventorObject*>(pcObject);
if (prop == &ivObj->Buffer) {
// read from buffer
SoInput in;
std::string buffer = ivObj->Buffer.getValue();
coinRemoveAllChildren(pcBuffer);
if (buffer.empty())
return;
in.setBuffer((void *)buffer.c_str(), buffer.size());
SoSeparator * node = SoDB::readAll(&in);
if (node) {
const char* doc = this->pcObject->getDocument()->getName();
const char* obj = this->pcObject->getNameInDocument();
adjustSelectionNodes(node, doc, obj);
pcBuffer->addChild(node);
}
}
else if (prop == &ivObj->FileName) {
// read also from file
const char* filename = ivObj->FileName.getValue();
QString fn = QString::fromUtf8(filename);
QFile file(fn);
SoInput in;
coinRemoveAllChildren(pcFile);
if (!fn.isEmpty() && file.open(QFile::ReadOnly)) {
QByteArray buffer = file.readAll();
in.setBuffer((void *)buffer.constData(), buffer.length());
SoSeparator * node = SoDB::readAll(&in);
if (node) {
const char* doc = this->pcObject->getDocument()->getName();
const char* obj = this->pcObject->getNameInDocument();
adjustSelectionNodes(node, doc, obj);
pcFile->addChild(node);
}
}
}
else if (prop->isDerivedFrom(App::PropertyPlacement::getClassTypeId()) &&
strcmp(prop->getName(), "Placement") == 0) {
// Note: If R is the rotation, c the rotation center and t the translation
// vector then Inventor applies the following transformation: R*(x-c)+c+t
// In FreeCAD a placement only has a rotation and a translation part but
// no rotation center. This means that the following equation must be ful-
// filled: R * (x-c) + c + t = R * x + t
// <==> R * x + t - R * c + c = R * x + t
// <==> (I-R) * c = 0 ==> c = 0
// This means that the center point must be the origin!
Base::Placement p = static_cast<const App::PropertyPlacement*>(prop)->getValue();
auto q0 = (float)p.getRotation().getValue()[0];
auto q1 = (float)p.getRotation().getValue()[1];
auto q2 = (float)p.getRotation().getValue()[2];
auto q3 = (float)p.getRotation().getValue()[3];
auto px = (float)p.getPosition().x;
auto py = (float)p.getPosition().y;
auto pz = (float)p.getPosition().z;
pcTransform->rotation.setValue(q0,q1,q2,q3);
pcTransform->translation.setValue(px,py,pz);
pcTransform->center.setValue(0.0f,0.0f,0.0f);
pcTransform->scaleFactor.setValue(1.0f,1.0f,1.0f);
}
}
void ViewProviderInventorObject::adjustSelectionNodes(SoNode* child, const char* docname,
const char* objname)
{
if (child->getTypeId().isDerivedFrom(SoFCSelection::getClassTypeId())) {
static_cast<SoFCSelection*>(child)->documentName = docname;
static_cast<SoFCSelection*>(child)->objectName = objname;
}
else if (child->getTypeId().isDerivedFrom(SoGroup::getClassTypeId())) {
auto group = static_cast<SoGroup*>(child);
for (int i=0; i<group->getNumChildren(); i++) {
SoNode* subchild = group->getChild(i);
adjustSelectionNodes(subchild, docname, objname);
}
}
}