第10节 实例-双击跑过去操作器

缘由

网友:KeepSmile 提出来能否做一个双击之后,跑过去的例子。大家如果在学习或工作中实现某个功能上需要我写一些例子和分析,可以在本文之后把功能描述清楚在评论区回复或在群里留言,我会帮忙写一些例子和解析。我的目标是用最简单的代码帮最大的忙 _

完整代码在本文最后的代码块和网盘中:

注意:务必使用浏览器打开:
链接:https://pan.baidu.com/s/13gwJLwo_LbRnN3Bl2NXXXw
提取码:xrf5

功能

场景:20x20米的格子。中间放着一个坐标轴。
启动场景:有一个5秒的开场动画,从(0.0, -18, 1.0)缓缓推到(0.0, -8, 1.0)。
操作:在场景格子上双击,双击后会在交点处加个球,然后视点会花两秒钟转过去,花三秒钟跑过去。在这个过程中你还可以双击别处,又会转往别处,跑往别处。如果当前朝向和点的球的朝向是一致的,也就是不需要转脑袋,则视点直接花5秒跑过去,不需要花2秒钟头。

实现关键点

操作器:使用osgGA::AnimationPathManipulator,这个操作器可以使用AnimationPath做为入参。因此最关键的就是如何生成AnimationPath
生成路径:路径的生成靠关键点,关键点有三个参数,一个是时间,一个是位置,一个是朝向。如下:

    if (fromRotate != rotate1)//代表不需要给时间转方向,当起点的方向和终点的方向是一致的时候,就不需要转方向
    {
        //给2秒钟转方向,起点不变,只有方向变
        animationPath->insert(2, osg::AnimationPath::ControlPoint(from, rotate1));
    }
    animationPath->insert(5, osg::AnimationPath::ControlPoint(to, rotate1));

其中5就是时间,也就是路径开始的时候第5S要跑到to, 朝向是rotate1,在两个点的间隔之间,会自动的位置和朝向都进行插值。

事件处理:在双击时,求交点,画球。我们专门写了个事件处理器MyEventHandler来做事件处理,然后获取当前视点和朝向做为起点,将交点做为终点,使用createAnimationByTwoPoints函数生成一个路径。设置到AnimationPathManipulator当中,注意将AnimationPathManipulator的时间设置为当前仿真时间的取负,也即:_apm->setTimeOffset(-ea.getTime());。这里估计是OSG的BUG,按说setTimeOffset设置成0就代表重新开始。但是源码里是仿真时间+时间偏移,因此动画要想重播,仿真时间就要被减掉。对应代码是:

                if (view->computeIntersections(ea, intersections))
                {
                    //有交点,先画个球
                    osg::Vec3 hitPoint = intersections.begin()->getWorldIntersectPoint();
                    _root->addChild(CreateSphere(hitPoint));

                    //操作器切换成animationPath操作器,创建animationPath直接跑过去
                    //获取当前视点信息
                    osg::Vec3 eye, center, up;
                    view->getCamera()->getViewMatrixAsLookAt(eye, center, up);

                    //得到当前视点的朝向
                    osg::Matrix viewMatrix = view->getCamera()->getInverseViewMatrix();
                    _apm->setAnimationPath(createAnimationByTwoPoints(eye, viewMatrix.getRotate(),hitPoint));
                    //减去当前仿真时间重头开始播,停个timeoffset设置的不太合理
                    //按说设置成0就应该是从头播了,但是其实现加了当前仿真时间,有点怪怪的
                    _apm->setTimeOffset(-ea.getTime());
                }

这样就实现了生成的路径始终以当前视点和朝向为起点,以双击地面的小球为终点。终点时的朝向是由起点和终点的连线决定,可以看看代码,弄不清楚的话建议看看上一篇第09节,里面讲的非常清晰。

#include 
#include 
#include 
#include 
#include 
#include 
#include 

//定义全局变量用着方便
osgGA::AnimationPathManipulator* _apm = new osgGA::AnimationPathManipulator;
osg::Group* _root = nullptr;
osg::Node* _base = nullptr;

osg::AnimationPath* createAnimationByTwoPoints(osg::Vec3 from, osg::Quat fromRotate, osg::Vec3 to)
{
    //防止视点贴地面太近
    if (from.z() < 1.0) from.z() = 1.0;
    if (to.z() < 1.0) to.z() = 1.0;

    osg::AnimationPath* animationPath = new osg::AnimationPath;
    animationPath->setLoopMode(osg::AnimationPath::NO_LOOPING);
    animationPath->insert(0, osg::AnimationPath::ControlPoint(from, fromRotate));
    osg::Quat rotate1(osg::inDegrees(90.0), osg::X_AXIS);

    osg::Vec3 tf = to - from;
    //经X旋转90度以后,视点已经朝向y轴正方向
    //当x,y都大于0在第一四象限,要顺时针旋转theta
    float theta = 0;

    if (tf.x() > 0.000001)//一四象限
    {
        rotate1 *= osg::Quat(std::atan(tf.y() / tf.x()) - osg::inDegrees(90.0), osg::Z_AXIS);
    }
    else if (tf.x() < -0.000001)//二三象限
    {
        rotate1 *= osg::Quat(osg::inDegrees(90.0) + std::atan(tf.y() / tf.x()), osg::Z_AXIS);
    }
    else //x=0
    {
        if (tf.y() > 0)
        {
            //啥也不用做,默认朝y轴正方向
        }
        else
        {
            //转180,朝y轴负方向
            rotate1 *= osg::Quat(osg::PI, osg::Z_AXIS);
        }
    }

    if (fromRotate != rotate1)//代表不需要给时间转方向,当起点的方向和终点的方向是一致的时候,就不需要转方向
    {
        //给2秒钟转方向,起点不变,只有方向变
        animationPath->insert(2, osg::AnimationPath::ControlPoint(from, rotate1));
    }

    animationPath->insert(5, osg::AnimationPath::ControlPoint(to, rotate1));
    return animationPath;
}

//画一个小球
osg::Geode* CreateSphere(osg::Vec3 center)
{
    osg::Geode* gnode = new osg::Geode;
    osg::ShapeDrawable* sd = new osg::ShapeDrawable(new osg::Sphere(center, 0.1));
    gnode->addDrawable(sd);
    return gnode;
}

class MyEventHandler : public osgGA::GUIEventHandler
{
public:
    MyEventHandler() {}

    virtual bool handle(const osgGA::GUIEventAdapter& ea, osgGA::GUIActionAdapter& aa)
    {
        //双击则加个球在击处,然后寻路过去
        if (ea.getEventType() == ea.DOUBLECLICK)
        {
            if (ea.getButton() == ea.LEFT_MOUSE_BUTTON)
            {
                //与地板求交
                osgUtil::LineSegmentIntersector::Intersections intersections;
                osgViewer::View* view = dynamic_cast(&aa);
                if (view->computeIntersections(ea, intersections))
                {
                    //有交点,先画个球
                    osg::Vec3 hitPoint = intersections.begin()->getWorldIntersectPoint();
                    _root->addChild(CreateSphere(hitPoint));

                    //操作器切换成animationPath操作器,创建animationPath直接跑过去
                    //获取当前视点信息
                    osg::Vec3 eye, center, up;
                    view->getCamera()->getViewMatrixAsLookAt(eye, center, up);

                    //得到当前视点的朝向
                    osg::Matrix viewMatrix = view->getCamera()->getInverseViewMatrix();
                    _apm->setAnimationPath(createAnimationByTwoPoints(eye, viewMatrix.getRotate(),hitPoint));
                    //减去当前仿真时间重头开始播,停个timeoffset设置的不太合理
                    //按说设置成0就应该是从头播了,但是其实现加了当前仿真时间,有点怪怪的
                    _apm->setTimeOffset(-ea.getTime());
                }
            }
        }

        return false;
    }
};

osg::Node* createBase(const osg::Vec3 center, float radius)
{
    osg::Group* root = new osg::Group;
    _root = root;

    int numTilesX = 10;
    int numTilesY = 10;

    float width = 2 * radius;
    float height = 2 * radius;

    osg::Vec3 v000(center - osg::Vec3(width * 0.5f, height * 0.5f, 0.0f));
    osg::Vec3 dx(osg::Vec3(width / ((float)numTilesX), 0.0, 0.0f));
    osg::Vec3 dy(osg::Vec3(0.0f, height / ((float)numTilesY), 0.0f));

    // fill in vertices for grid, note numTilesX+1 * numTilesY+1...
    osg::Vec3Array* coords = new osg::Vec3Array;
    int iy;
    for (iy = 0; iy <= numTilesY; ++iy)
    {
        for (int ix = 0; ix <= numTilesX; ++ix)
        {
            coords->push_back(v000 + dx * (float)ix + dy * (float)iy);
        }
    }

    //Just two colours - black and white.
    osg::Vec4Array* colors = new osg::Vec4Array;
    colors->push_back(osg::Vec4(1.0f, 1.0f, 1.0f, 1.0f)); // white
    colors->push_back(osg::Vec4(0.0f, 0.0f, 0.0f, 1.0f)); // black

    osg::ref_ptr whitePrimitives = new osg::DrawElementsUShort(GL_QUADS);
    osg::ref_ptr blackPrimitives = new osg::DrawElementsUShort(GL_QUADS);

    int numIndicesPerRow = numTilesX + 1;
    for (iy = 0; iy < numTilesY; ++iy)
    {
        for (int ix = 0; ix < numTilesX; ++ix)
        {
            osg::DrawElementsUShort* primitives = ((iy + ix) % 2 == 0) ? whitePrimitives.get() : blackPrimitives.get();
            primitives->push_back(ix + (iy + 1) * numIndicesPerRow);
            primitives->push_back(ix + iy * numIndicesPerRow);
            primitives->push_back((ix + 1) + iy * numIndicesPerRow);
            primitives->push_back((ix + 1) + (iy + 1) * numIndicesPerRow);
        }
    }

    // set up a single normal
    osg::Vec3Array* normals = new osg::Vec3Array;
    normals->push_back(osg::Vec3(0.0f, 0.0f, 1.0f));

    osg::Geometry* geom = new osg::Geometry;
    geom->setVertexArray(coords);

    geom->setColorArray(colors, osg::Array::BIND_PER_PRIMITIVE_SET);

    geom->setNormalArray(normals, osg::Array::BIND_OVERALL);

    geom->addPrimitiveSet(whitePrimitives.get());
    geom->addPrimitiveSet(blackPrimitives.get());

    osg::Geode* geode = new osg::Geode;
    _base = geode;

    geode->addDrawable(geom);
    geode->getOrCreateStateSet()->setMode(GL_LIGHTING, osg::StateAttribute::OFF);
    root->addChild(geode);

    root->addChild(osgDB::readNodeFile("axes.osgt"));

    return root;
}

int main()
{
    osgViewer::Viewer viewer;

    viewer.addEventHandler(new MyEventHandler());

    osg::Quat rotate0(osg::inDegrees(90.0), osg::X_AXIS);
    _apm->setAnimationPath(createAnimationByTwoPoints(osg::Vec3(0.0, -18, 1.0), rotate0, osg::Vec3(0.0, -8, 1.0)));

    viewer.setCameraManipulator(_apm);
    viewer.setSceneData(createBase(osg::Vec3(0.0, 0.0, 0.0), 20.0));

    return viewer.run();
}

你可能感兴趣的:(第10节 实例-双击跑过去操作器)