9.3.4 变形体动画之对折硬纸

本文展示了如何使用osgAnimation库创建动态关键帧,并实现两个几何体之间的实时Morph变形。通过创建源和目标几何体,定义关键帧并创建动画,最后将动画应用到osg::Geode中,实现了可视化效果。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

#include <osgViewer/Viewer>
#include <osg/Node>
#include <osg/Geometry>
#include <osg/Geode>
#include <osg/Group>
#include <osg/AutoTransform>
#include <osg/ProxyNode>
#include <osgDB/ReadFile>
#include <osgDB/WriteFile>
#include <osgText/Text>
#include <osgUtil/Optimizer>
#include<osg/Camera>
#include<osgViewer/Viewer>
#include <osgManipulator/Dragger>
#include <osgManipulator/Selection>
#include <osgManipulator/Constraint>
#include <osgManipulator/TranslateAxisDragger>
#include <osgManipulator/CommandManager>
#include<osgFX/Scribe>
#include<osg/MatrixTransform>
#include<osgAnimation/Animation>
#include<osgAnimation/Bone>
#include<osgAnimation/Skeleton>
#include<osgAnimation/UpdateMatrixTransform>
#include<osgAnimation/BasicAnimationManager>
#include<osgAnimation/MorphGeometry>
#include<osgUtil/SmoothingVisitor>
#include <iostream>
#include <thread>
#include <future>
#include <string>
#include<map>
#include<vector>
#include<list>
using std::vector;
using std::string;
using std::list;
using namespace std;

osg::Geometry* createSourceGeometry()
{
	osg::ref_ptr<osg::Vec3Array> vertices = new osg::Vec3Array;
	vertices->push_back(osg::Vec3(0.0, 5.0, -2.5));
	vertices->push_back(osg::Vec3(0.0, 0.0, -2.5));
	vertices->push_back(osg::Vec3(2.5, 5.0, 0.0));
	vertices->push_back(osg::Vec3(2.5, 0.0, 0.0));
	vertices->push_back(osg::Vec3(5.0, 5.0, -2.5));
	vertices->push_back(osg::Vec3(5.0, 0.0, -2.5));
    
	osg::ref_ptr<osg::Geometry> geom = new osg::Geometry;
	geom->setVertexArray(vertices.get());
	geom->addPrimitiveSet(new osg::DrawArrays(GL_QUAD_STRIP, 0, 6));
	osgUtil::SmoothingVisitor smv;
	smv.smooth(*geom);

	return geom.release();
}

osg::Geometry* createTargetGeometry()
{
	osg::ref_ptr<osg::Vec3Array> vertices = new osg::Vec3Array;
	vertices->push_back(osg::Vec3(0.0, 5.0, 1.0));
	vertices->push_back(osg::Vec3(0.0, 0.0, 1.0));
	vertices->push_back(osg::Vec3(2.5, 5.0, -1.0));
	vertices->push_back(osg::Vec3(2.5, 0.0, -1.0));
	vertices->push_back(osg::Vec3(5.0, 5.0, 1.0));
	vertices->push_back(osg::Vec3(5.0, 0.0, 1.0));

	osg::ref_ptr<osg::Geometry> geom = new osg::Geometry;
	geom->setVertexArray(vertices.get());
	geom->addPrimitiveSet(new osg::DrawArrays(GL_QUAD_STRIP, 0, 6));
	osgUtil::SmoothingVisitor smv;
	smv.smooth(*geom);

	return geom.release();
}

void createMorphKeyframes(osgAnimation::FloatKeyframeContainer* ks)
{
	ks->push_back(osgAnimation::FloatKeyframe(0.0, 0.0));
	ks->push_back(osgAnimation::FloatKeyframe(2.0, 1.0));
}
int main()
{
	osg::ref_ptr<osgAnimation::FloatLinearChannel> channel = new osgAnimation::FloatLinearChannel;
	channel->setName("0");
	channel->setTargetName("MorphCallback");

	createMorphKeyframes(channel->getOrCreateSampler()->getOrCreateKeyframeContainer());
	osg::ref_ptr<osgAnimation::Animation> anim = new osgAnimation::Animation;
    anim->setPlaymode(osgAnimation::Animation::PPONG)
	anim->addChannel(channel.get());

	osg::ref_ptr<osgAnimation::BasicAnimationManager> manager = new osgAnimation::BasicAnimationManager;
	manager->registerAnimation(anim.get());

	osg::ref_ptr<osgAnimation::MorphGeometry> morph = new osgAnimation::MorphGeometry(*createSourceGeometry());
	morph->addMorphTarget(createTargetGeometry());

	osg::ref_ptr<osg::Geode> geode = new osg::Geode;
	geode->addDrawable(morph.get());
	geode->setUpdateCallback(new osgAnimation::UpdateMorph("MorphCallback"));
	osg::ref_ptr<osg::Group> root = new osg::Group;
	root->addChild(geode.get());
	root->setUpdateCallback(manager.get());

	manager->playAnimation(anim.get());

	osgViewer::Viewer viewer;
	viewer.setSceneData(root.get());
	return viewer.run();
}

评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值