1. Support for ambient occlusion maps.
2. A fix for the incorrect handling of normals on all geometries. The optimizer usually fixed this bug so it probably was not noticed very often.
3. A new option flag on the reader. "StrictTransparency"
// Process transparent and transparency settings according to a strict interpretation of the spec
// See https://collada.org/public_forum/viewtopic.php?f=12&t=1210
otherwise
// Jump through various hoops to accomodate the multiplicity of different ways
// that various people have interpreted the specification
4. Handling of texures in the transparent channel. This is allowed provided that they are the same texture that is specified in the diffuse channel. Accomodating a different texture would require use of programmable pipeline functionality which I have tried to avoid.
5. Handling of <bind_vertex_input> elements.
""
184 lines
6.0 KiB
C++
184 lines
6.0 KiB
C++
/* -*-c++-*- OpenSceneGraph - Copyright (C) 1998-2006 Robert Osfield
|
|
*
|
|
* This application is open source and may be redistributed and/or modified
|
|
* freely and without restriction, both in commericial and non commericial
|
|
* applications, as long as this copyright notice is maintained.
|
|
*
|
|
* This application 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.
|
|
*
|
|
*/
|
|
|
|
#include <sstream>
|
|
#include <memory>
|
|
|
|
#include <osg/Notify>
|
|
#include <osg/NodeVisitor>
|
|
#include <osgDB/ReaderWriter>
|
|
#include <osgDB/FileNameUtils>
|
|
#include <osgDB/Registry>
|
|
|
|
#include <OpenThreads/ScopedLock>
|
|
|
|
#include "ReaderWriterDAE.h"
|
|
#include "daeReader.h"
|
|
#include "daeWriter.h"
|
|
|
|
#ifdef WIN32
|
|
#include "windows.h"
|
|
#endif
|
|
|
|
#define SERIALIZER() OpenThreads::ScopedLock<OpenThreads::ReentrantMutex> lock(_serializerMutex)
|
|
|
|
osgDB::ReaderWriter::ReadResult
|
|
ReaderWriterDAE::readNode(const std::string& fname,
|
|
const osgDB::ReaderWriter::Options* options) const
|
|
{
|
|
SERIALIZER();
|
|
|
|
bool bOwnDAE = false;
|
|
DAE* pDAE = NULL;
|
|
|
|
if ( options )
|
|
pDAE = (DAE*)options->getPluginData("DAE");
|
|
|
|
std::string ext( osgDB::getLowerCaseFileExtension(fname) );
|
|
if( ! acceptsExtension(ext) ) return ReadResult::FILE_NOT_HANDLED;
|
|
|
|
std::string fileName( osgDB::findDataFile( fname, options ) );
|
|
if( fileName.empty() ) return ReadResult::FILE_NOT_FOUND;
|
|
|
|
osg::notify(osg::INFO) << "ReaderWriterDAE( \"" << fileName << "\" )" << std::endl;
|
|
|
|
if (NULL == pDAE)
|
|
{
|
|
bOwnDAE = true;
|
|
pDAE = new DAE;
|
|
}
|
|
|
|
osgdae::daeReader daeReader(pDAE, options && options->getOptionString().find("StrictTransparency") != std::string::npos ) ;
|
|
|
|
// Convert file name to URI
|
|
std::string fileURI = ConvertFilePathToColladaCompatibleURI(fileName);
|
|
|
|
if ( ! daeReader.convert( fileURI ) )
|
|
{
|
|
osg::notify( osg::WARN ) << "Load failed in COLLADA DOM conversion" << std::endl;
|
|
return ReadResult::ERROR_IN_READING_FILE;
|
|
}
|
|
|
|
if ( options )
|
|
{
|
|
// Return the document URI
|
|
if (options->getPluginData("DAE-DocumentURI"))
|
|
*(std::string*)options->getPluginData("DAE-DocumentURI") = fileURI;
|
|
// Return some additional information about the document
|
|
if (options->getPluginData("DAE-AssetUnitName"))
|
|
*(std::string*)options->getPluginData("DAE-AssetUnitName") = daeReader.m_AssetUnitName;
|
|
if (options->getPluginData("DAE-AssetUnitMeter"))
|
|
*(float*)options->getPluginData("DAE-AssetUnitMeter") = daeReader.m_AssetUnitMeter;
|
|
if (options->getPluginData("DAE-AssetUp_axis"))
|
|
*(domUpAxisType*)options->getPluginData("DAE-AssetUp_axis") = daeReader.m_AssetUp_axis;
|
|
}
|
|
|
|
if (bOwnDAE)
|
|
delete pDAE;
|
|
|
|
osg::Node* rootNode( daeReader.getRootNode() );
|
|
return rootNode;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
osgDB::ReaderWriter::WriteResult
|
|
ReaderWriterDAE::writeNode( const osg::Node& node,
|
|
const std::string& fname, const osgDB::ReaderWriter::Options* options ) const
|
|
{
|
|
SERIALIZER();
|
|
|
|
bool bOwnDAE = false;
|
|
DAE* pDAE = NULL;
|
|
|
|
std::string ext( osgDB::getLowerCaseFileExtension(fname) );
|
|
if( ! acceptsExtension(ext) ) return WriteResult::FILE_NOT_HANDLED;
|
|
|
|
// Process options
|
|
bool usePolygon(false); // Use plugin option "polygon" to enable
|
|
bool GoogleMode(false); // Use plugin option "GoogleMode" to enable
|
|
bool writeExtras(true); // Use plugin option "NoExtras" to disable
|
|
if( options )
|
|
{
|
|
pDAE = (DAE*)options->getPluginData("DAE");
|
|
std::istringstream iss( options->getOptionString() );
|
|
std::string opt;
|
|
|
|
while( std::getline( iss, opt, ',' ) )
|
|
{
|
|
if( opt == "polygon") usePolygon = true;
|
|
else if (opt == "GoogleMode") GoogleMode = true;
|
|
else if (opt == "NoExtras") writeExtras = false;
|
|
else
|
|
{
|
|
osg::notify(osg::WARN)
|
|
<< std::endl << "COLLADA dae plugin: unrecognized option \"" << opt << std::endl
|
|
<< "comma-delimited options:" << std::endl << std::endl
|
|
<< "\tpolygon = use polygons instead of polylists for element" << std::endl
|
|
<< "\tGoogleMode = write files suitable for use by Google products" << std::endl
|
|
<< "example: osgviewer -O polygon bar.dae" << std::endl << std::endl;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (NULL == pDAE)
|
|
{
|
|
bOwnDAE = true;
|
|
pDAE = new DAE;
|
|
}
|
|
|
|
// Convert file name to URI
|
|
std::string fileURI = ConvertFilePathToColladaCompatibleURI(fname);
|
|
|
|
osg::NodeVisitor::TraversalMode traversalMode = writeExtras ? osg::NodeVisitor::TRAVERSE_ALL_CHILDREN : osg::NodeVisitor::TRAVERSE_ACTIVE_CHILDREN;
|
|
|
|
osgdae::daeWriter daeWriter(pDAE, fileURI, usePolygon, GoogleMode, traversalMode, writeExtras);
|
|
daeWriter.setRootNode( node );
|
|
const_cast<osg::Node*>(&node)->accept( daeWriter );
|
|
|
|
osgDB::ReaderWriter::WriteResult retVal( WriteResult::ERROR_IN_WRITING_FILE );
|
|
if ( daeWriter.isSuccess() )
|
|
{
|
|
if ( daeWriter.writeFile() )
|
|
retVal = WriteResult::FILE_SAVED;
|
|
}
|
|
|
|
if ( options )
|
|
{
|
|
if (!bOwnDAE)
|
|
{
|
|
// Return the document URI used so that users of an external DAE object
|
|
// can locate the correct database
|
|
if (options->getPluginData("DAE-DocumentURI"))
|
|
*(std::string*)options->getPluginData("DAE-DocumentURI") = fileURI;
|
|
}
|
|
}
|
|
|
|
if (bOwnDAE)
|
|
delete pDAE;
|
|
|
|
return retVal;
|
|
}
|
|
|
|
std::string ReaderWriterDAE::ConvertFilePathToColladaCompatibleURI(const std::string& FilePath)
|
|
{
|
|
return cdom::nativePathToUri(FilePath);
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// Add ourself to the Registry to instantiate the reader/writer.
|
|
|
|
REGISTER_OSGPLUGIN(dae, ReaderWriterDAE)
|
|
|
|
// vim: set sw=4 ts=8 et ic ai:
|