initial support for importing animation tracks
parent
36e9571e9f
commit
710890d894
|
@ -0,0 +1,67 @@
|
||||||
|
/*
|
||||||
|
This file is part of Mitsuba, a physically based rendering system.
|
||||||
|
|
||||||
|
Copyright (c) 2007-2010 by Wenzel Jakob and others.
|
||||||
|
|
||||||
|
Mitsuba is free software; you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU General Public License Version 3
|
||||||
|
as published by the Free Software Foundation.
|
||||||
|
|
||||||
|
Mitsuba 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 General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#if !defined(__ANIMATION_TRACK_H)
|
||||||
|
#define __ANIMATION_TRACK_H
|
||||||
|
|
||||||
|
#include <mitsuba/mitsuba.h>
|
||||||
|
|
||||||
|
MTS_NAMESPACE_BEGIN
|
||||||
|
|
||||||
|
template <typename T> class AnimationTrack {
|
||||||
|
public:
|
||||||
|
enum EType {
|
||||||
|
EInvalid = 0,
|
||||||
|
ELocationX, ELocationY, ELocationZ, ELocationXYZ,
|
||||||
|
EScaleX, EScaleY, EScaleZ, EScaleXYZ,
|
||||||
|
ERotationX, ERotationY, ERotationZ, ERotationXYZ
|
||||||
|
};
|
||||||
|
|
||||||
|
AnimationTrack(EType type, size_t nKeyframes)
|
||||||
|
: m_type(type), m_times(nKeyframes), m_values(nKeyframes) { }
|
||||||
|
|
||||||
|
/// Return the type of this track
|
||||||
|
inline EType getType() const { return m_type; }
|
||||||
|
|
||||||
|
/// Return the number of keyframes
|
||||||
|
inline size_t getSize() const { return m_times.size(); }
|
||||||
|
|
||||||
|
/// Set the time value of a certain keyframe
|
||||||
|
inline void setTime(size_t idx, Float time) { m_times[idx] = time; }
|
||||||
|
|
||||||
|
/// Return the time value of a certain keyframe
|
||||||
|
inline Float getTime(size_t idx) const { return m_times[idx]; }
|
||||||
|
|
||||||
|
/// Set the value of a certain keyframe
|
||||||
|
inline void setValue(size_t idx, const T &value) { m_values[idx] = value; }
|
||||||
|
|
||||||
|
/// Return the value of a certain keyframe
|
||||||
|
inline const T &getValue(size_t idx) const { return m_values[idx]; }
|
||||||
|
|
||||||
|
/// Evaluate the animation track at an arbitrary time value
|
||||||
|
inline T lookup(Float time) const {
|
||||||
|
}
|
||||||
|
private:
|
||||||
|
EType m_type;
|
||||||
|
std::vector<Float> m_times;
|
||||||
|
std::vector<T> m_values;
|
||||||
|
};
|
||||||
|
|
||||||
|
MTS_NAMESPACE_END
|
||||||
|
|
||||||
|
#endif /* __ANIMATION_TRACK_H */
|
|
@ -20,6 +20,7 @@
|
||||||
#include <mitsuba/render/trimesh.h>
|
#include <mitsuba/render/trimesh.h>
|
||||||
#include <mitsuba/core/fresolver.h>
|
#include <mitsuba/core/fresolver.h>
|
||||||
#include <mitsuba/core/fstream.h>
|
#include <mitsuba/core/fstream.h>
|
||||||
|
#include <mitsuba/render/track.h>
|
||||||
#include <dom/domCOLLADA.h>
|
#include <dom/domCOLLADA.h>
|
||||||
#include <dae.h>
|
#include <dae.h>
|
||||||
#include <dae/daeErrorHandler.h>
|
#include <dae/daeErrorHandler.h>
|
||||||
|
@ -35,14 +36,21 @@
|
||||||
#include <GL/glu.h>
|
#include <GL/glu.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifndef WIN32
|
||||||
|
#define __stdcall
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "converter.h"
|
#include "converter.h"
|
||||||
|
|
||||||
|
typedef AnimationTrack<Float> FloatTrack;
|
||||||
typedef std::map<std::string, std::string> StringMap;
|
typedef std::map<std::string, std::string> StringMap;
|
||||||
typedef std::map<std::string, int> RefCountMap;
|
typedef std::map<std::string, int> RefCountMap;
|
||||||
|
typedef std::map<std::string, FloatTrack *> AnimationMap;
|
||||||
|
|
||||||
struct ColladaContext {
|
struct ColladaContext {
|
||||||
GeometryConverter *cvt;
|
GeometryConverter *cvt;
|
||||||
RefCountMap refCountMap;
|
RefCountMap refCountMap;
|
||||||
|
AnimationMap animations;
|
||||||
std::set<std::string> serializedGeometry;
|
std::set<std::string> serializedGeometry;
|
||||||
fs::path texturesDirectory;
|
fs::path texturesDirectory;
|
||||||
fs::path meshesDirectory;
|
fs::path meshesDirectory;
|
||||||
|
@ -153,6 +161,7 @@ VertexData *fetchVertexData(Transform transform,
|
||||||
for (size_t i=0; i<inputs.getCount(); ++i) {
|
for (size_t i=0; i<inputs.getCount(); ++i) {
|
||||||
int offsetInStream = (int) inputs[i]->getOffset(),
|
int offsetInStream = (int) inputs[i]->getOffset(),
|
||||||
offset = offsetInStream;
|
offset = offsetInStream;
|
||||||
|
|
||||||
daeURI &sourceRef = inputs[i]->getSource();
|
daeURI &sourceRef = inputs[i]->getSource();
|
||||||
sourceRef.resolveElement();
|
sourceRef.resolveElement();
|
||||||
domSource *source = daeSafeCast<domSource>(sourceRef.getElement());
|
domSource *source = daeSafeCast<domSource>(sourceRef.getElement());
|
||||||
|
@ -309,7 +318,6 @@ struct triangle_key_order : public std::binary_function<SimpleTriangle, SimpleTr
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
class CustomErrorHandler : public daeErrorHandler {
|
class CustomErrorHandler : public daeErrorHandler {
|
||||||
public:
|
public:
|
||||||
void handleError(daeString msg) {
|
void handleError(daeString msg) {
|
||||||
|
@ -1287,9 +1295,111 @@ void computeRefCounts(ColladaContext &ctx, domNode &node) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifndef WIN32
|
void loadAnimation(ColladaContext &ctx, domAnimation &anim) {
|
||||||
#define __stdcall
|
std::string identifier;
|
||||||
#endif
|
if (anim.getId() != NULL) {
|
||||||
|
identifier = anim.getId();
|
||||||
|
} else {
|
||||||
|
if (anim.getName() != NULL) {
|
||||||
|
identifier = anim.getName();
|
||||||
|
} else {
|
||||||
|
static int unnamedCtr = 0;
|
||||||
|
identifier = formatString("unnamedAnimation_%i", unnamedCtr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
SLog(EInfo, "Loading animation \"%s\" ..", identifier.c_str());
|
||||||
|
|
||||||
|
domChannel_Array &channels = anim.getChannel_array();
|
||||||
|
for (size_t i=0; i<channels.getCount(); ++i) {
|
||||||
|
domChannel *channel = channels[i];
|
||||||
|
std::vector<std::string> target = tokenize(channel->getTarget(), "./");
|
||||||
|
SAssertEx(target.size() == 3, "Encountered an unknown animation channel identifier!");
|
||||||
|
|
||||||
|
daeURI &sourceRef = channel->getSource();
|
||||||
|
sourceRef.resolveElement();
|
||||||
|
domSampler *sampler = daeSafeCast<domSampler>(sourceRef.getElement());
|
||||||
|
if (!sampler)
|
||||||
|
SLog(EError, "Referenced animation sampler not found!");
|
||||||
|
const domInputLocal_Array &inputs = sampler->getInput_array();
|
||||||
|
FloatTrack *track = NULL;
|
||||||
|
FloatTrack::EType trackType = FloatTrack::EInvalid;
|
||||||
|
boost::to_lower(target[1]);
|
||||||
|
boost::to_lower(target[2]);
|
||||||
|
if (target[1] == "location") {
|
||||||
|
if (target[2] == "x") {
|
||||||
|
trackType = FloatTrack::ELocationX;
|
||||||
|
} else if (target[2] == "y") {
|
||||||
|
trackType = FloatTrack::ELocationY;
|
||||||
|
} else if (target[3] == "z") {
|
||||||
|
trackType = FloatTrack::ELocationZ;
|
||||||
|
}
|
||||||
|
} else if (target[1] == "scale") {
|
||||||
|
if (target[2] == "x") {
|
||||||
|
trackType = FloatTrack::EScaleX;
|
||||||
|
} else if (target[2] == "y") {
|
||||||
|
trackType = FloatTrack::EScaleY;
|
||||||
|
} else if (target[3] == "z") {
|
||||||
|
trackType = FloatTrack::EScaleZ;
|
||||||
|
}
|
||||||
|
} else if (target[1] == "rotationx" && target[2] == "angle") {
|
||||||
|
trackType = FloatTrack::ERotationX;
|
||||||
|
} else if (target[1] == "rotationy" && target[2] == "angle") {
|
||||||
|
trackType = FloatTrack::ERotationY;
|
||||||
|
} else if (target[1] == "rotationz" && target[2] == "angle") {
|
||||||
|
trackType = FloatTrack::ERotationZ;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (trackType == FloatTrack::EInvalid) {
|
||||||
|
SLog(EWarn, "Skipping unsupported animation track of type %s.%s",
|
||||||
|
target[1].c_str(), target[2].c_str());
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (size_t j=0; j<inputs.getCount(); ++j) {
|
||||||
|
sourceRef = inputs[j]->getSource();
|
||||||
|
sourceRef.resolveElement();
|
||||||
|
domSource *source = daeSafeCast<domSource>(sourceRef.getElement());
|
||||||
|
if (!source)
|
||||||
|
SLog(EError, "Referenced animation source not found!");
|
||||||
|
std::string semantic = inputs[j]->getSemantic();
|
||||||
|
domSource::domTechnique_common *techniqueCommon = source->getTechnique_common();
|
||||||
|
if (!techniqueCommon)
|
||||||
|
SLog(EError, "Data source does not have a <technique_common> tag!");
|
||||||
|
domAccessor *accessor = techniqueCommon->getAccessor();
|
||||||
|
if (!accessor)
|
||||||
|
SLog(EError, "Data source does not have a <accessor> tag!");
|
||||||
|
unsigned int nParams = (unsigned int) accessor->getParam_array().getCount(),
|
||||||
|
stride = (unsigned int) accessor->getStride();
|
||||||
|
size_t size = (size_t) accessor->getCount();
|
||||||
|
|
||||||
|
if (stride != 1 || nParams != 1) {
|
||||||
|
/// Only single-valued tracks are supported for now
|
||||||
|
SLog(EError, "Encountered a multi-valued animation track.");
|
||||||
|
}
|
||||||
|
if (!track)
|
||||||
|
track = new FloatTrack(trackType, size);
|
||||||
|
else
|
||||||
|
SAssert(track->getSize() == size);
|
||||||
|
|
||||||
|
if (semantic == "INPUT") {
|
||||||
|
domListOfFloats &floatArray = source->getFloat_array()->getValue();
|
||||||
|
for (size_t i=0; i<size; ++i)
|
||||||
|
track->setTime(i, floatArray[i]);
|
||||||
|
} else if (semantic == "OUTPUT") {
|
||||||
|
domListOfFloats &floatArray = source->getFloat_array()->getValue();
|
||||||
|
for (size_t i=0; i<size; ++i)
|
||||||
|
track->setValue(i, floatArray[i]);
|
||||||
|
} else if (semantic == "INTERPOLATION") {
|
||||||
|
/// Ignored for now
|
||||||
|
} else {
|
||||||
|
SLog(EWarn, "Encountered an unsupported semantic: \"%s\"", semantic.c_str());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (track)
|
||||||
|
ctx.animations.insert(AnimationMap::value_type(target[0], track));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
GLvoid __stdcall tessBegin(GLenum type) {
|
GLvoid __stdcall tessBegin(GLenum type) {
|
||||||
SAssert(type == GL_TRIANGLES);
|
SAssert(type == GL_TRIANGLES);
|
||||||
|
@ -1338,15 +1448,12 @@ GLvoid __stdcall tessCombine(GLdouble coords[3], void *vertex_data[4],
|
||||||
*outData = result;
|
*outData = result;
|
||||||
}
|
}
|
||||||
|
|
||||||
GLvoid __stdcall tessEnd() {
|
GLvoid __stdcall tessEnd() { }
|
||||||
}
|
GLvoid __stdcall tessEdgeFlag(GLboolean) { }
|
||||||
|
|
||||||
GLvoid __stdcall tessError(GLenum error) {
|
GLvoid __stdcall tessError(GLenum error) {
|
||||||
SLog(EError, "The GLU tesselator generated an error: %s!", gluErrorString(error));
|
SLog(EError, "The GLU tesselator generated an error: %s!", gluErrorString(error));
|
||||||
}
|
}
|
||||||
|
|
||||||
GLvoid __stdcall tessEdgeFlag(GLboolean) {
|
|
||||||
}
|
|
||||||
|
|
||||||
void GeometryConverter::convertCollada(const fs::path &inputFile,
|
void GeometryConverter::convertCollada(const fs::path &inputFile,
|
||||||
std::ostream &os,
|
std::ostream &os,
|
||||||
|
@ -1404,12 +1511,23 @@ void GeometryConverter::convertCollada(const fs::path &inputFile,
|
||||||
for (size_t j=0; j<materials.getCount(); ++j)
|
for (size_t j=0; j<materials.getCount(); ++j)
|
||||||
loadMaterial(ctx, *materials.get(j));
|
loadMaterial(ctx, *materials.get(j));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
domLibrary_animations_Array &libraryAnimations = document->getLibrary_animations_array();
|
||||||
|
for (size_t i=0; i<libraryAnimations.getCount(); ++i) {
|
||||||
|
domAnimation_Array &animations = libraryAnimations[i]->getAnimation_array();
|
||||||
|
for (size_t j=0; j<animations.getCount(); ++j)
|
||||||
|
loadAnimation(ctx, *animations[j]);
|
||||||
|
}
|
||||||
|
|
||||||
for (size_t i=0; i<nodes.getCount(); ++i)
|
for (size_t i=0; i<nodes.getCount(); ++i)
|
||||||
computeRefCounts(ctx, *nodes[i]);
|
computeRefCounts(ctx, *nodes[i]);
|
||||||
|
|
||||||
for (size_t i=0; i<nodes.getCount(); ++i)
|
for (size_t i=0; i<nodes.getCount(); ++i)
|
||||||
loadNode(ctx, Transform(), *nodes[i], "");
|
loadNode(ctx, Transform(), *nodes[i], "");
|
||||||
|
|
||||||
|
for (AnimationMap::iterator it = ctx.animations.begin();
|
||||||
|
it != ctx.animations.end(); ++it)
|
||||||
|
delete it->second;
|
||||||
|
|
||||||
os << "</scene>" << endl;
|
os << "</scene>" << endl;
|
||||||
|
|
||||||
|
|
|
@ -61,7 +61,7 @@ class lamps(DataButtonsPanel, property_group_renderer, bpy.types.Panel):
|
||||||
else:
|
else:
|
||||||
layout.prop(lamp.mitsuba_lamp, "envmap_file", text="HDRI file")
|
layout.prop(lamp.mitsuba_lamp, "envmap_file", text="HDRI file")
|
||||||
layout.prop(lamp.mitsuba_lamp, "intensity", text="Intensity")
|
layout.prop(lamp.mitsuba_lamp, "intensity", text="Intensity")
|
||||||
layout.prop(lamp.mitsuba_lamp, "sampling_weight", text = "Sampling weight")
|
layout.prop(lamp.mitsuba_lamp, "samplingWeight", text = "Sampling weight")
|
||||||
|
|
||||||
# SPOT LAMP: Blender Properties
|
# SPOT LAMP: Blender Properties
|
||||||
if lamp.type == 'SPOT':
|
if lamp.type == 'SPOT':
|
||||||
|
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Loading…
Reference in New Issue