168 lines
7.0 KiB
C++
168 lines
7.0 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 "ViewProviderInventorObject.h"
|
|
#include <Gui/SoFCSelection.h>
|
|
#include <App/InventorObject.h>
|
|
#include <App/Document.h>
|
|
#include <Base/FileInfo.h>
|
|
#include <Base/Stream.h>
|
|
#include <sstream>
|
|
|
|
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);
|
|
SoGroup* 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(void) const
|
|
{
|
|
std::vector<std::string> StrList;
|
|
StrList.push_back("File+Buffer");
|
|
StrList.push_back("Buffer");
|
|
StrList.push_back("File");
|
|
return StrList;
|
|
}
|
|
|
|
void ViewProviderInventorObject::updateData(const App::Property* prop)
|
|
{
|
|
App::InventorObject* ivObj = static_cast<App::InventorObject*>(pcObject);
|
|
if (prop == &ivObj->Buffer) {
|
|
// read from buffer
|
|
SoInput in;
|
|
std::string buffer = ivObj->Buffer.getValue();
|
|
pcBuffer->removeAllChildren();
|
|
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;
|
|
pcFile->removeAllChildren();
|
|
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();
|
|
float q0 = (float)p.getRotation().getValue()[0];
|
|
float q1 = (float)p.getRotation().getValue()[1];
|
|
float q2 = (float)p.getRotation().getValue()[2];
|
|
float q3 = (float)p.getRotation().getValue()[3];
|
|
float px = (float)p.getPosition().x;
|
|
float py = (float)p.getPosition().y;
|
|
float 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())) {
|
|
SoGroup* group = static_cast<SoGroup*>(child);
|
|
for (int i=0; i<group->getNumChildren(); i++) {
|
|
SoNode* subchild = group->getChild(i);
|
|
adjustSelectionNodes(subchild, docname, objname);
|
|
}
|
|
}
|
|
}
|