cocos2d-x 3.8.1的骨骼动画加载方法addArmatureFileInfo仍然存在问题

前端之家收集整理的这篇文章主要介绍了cocos2d-x 3.8.1的骨骼动画加载方法addArmatureFileInfo仍然存在问题前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

提出如题所示的问题,心里非常别扭,但的确是事实。因此,Cocos Studio(我目前使用的是 2.3.2)在许多方面还有改进的地方,包括与之相对应的cocos2d-x中的代码操作部分。

  

问题

目前,我的试验结果发现,使用cocos2d-x 3.8.1中提供的如下方法:

ArmatureDataManager::getInstance()->addArmatureFileInfo(filename);

无法正常加载Cocos Studio 2.3.2导出的骨骼动画资源文件。例如如下代码无法正常通过项目构建

ArmatureDataManager::getInstance()->addArmatureFileInfo("DemoPlayer.csb");

令人遗憾的例子

尽管如此,但是cocos2d-x 3.8.1的cpp-tests实例中的确提供了使用addArmatureFileInfo方法加载.csb骨骼动画文件的例子!!

是的,因为.csb文件是二进制格式,目前还找不到其反编译工具,但是,从使用简单的工具分析,cpp-tests实例中提供的示例.csb骨骼动画文件的版本与Cocos Studio 2.3.2导出的骨骼动画资源文件.csb并不一致。

下面给出Notepad++观察到的结果图的对照(第1张是Cocos Studio 2.3.2导出骨骼动画文件DemoPlayer.csb查看结果,显然版本号是2.1.0.0,第2张是cpp-tests实例中提供的示例Cowboy.csb骨骼动画文件查看结果,显然版本号是1.0.1):

wKioL1ZLIqGBpW75AALR_GQJDyo639.jpg

wKiom1ZLIxmC3MHpAAQDiObULoI450.jpg

为了进一步分析上述问题,我还专门把cocos2d-x 3.8.1的cpp-tests实例中提供的使用addArmatureFileInfo方法加载其提供的相应.csb骨骼动画文件代码复制到一个简单示例工程中进行测试,的确OK。相关代码如下所示:

constchar*HelloWorld::m_binaryFilesNames[4]={"bear.csb","horse.csb","Cowboy.csb","ccc.csb"};
constchar*HelloWorld::m_armatureNames[4]={"bear","horse","Cowboy","Skeleton1"};
//......
//loadfrombinary
ArmatureDataManager::getInstance()->addArmatureFileInfo(m_binaryFilesNames[3]);
Armature*m_armature=Armature::create(m_armatureNames[3]);
m_armature->getAnimation()->playWithIndex(0);
m_armature->setScale(1.0f);
Sizesize=Director::getInstance()->getWinSize();
m_armature->setPosition(size.width/2,size.height/2);
addChild(m_armature);

对于数组中相应的前三个.csb文件(应该是老版本的STUdio导出的骨骼动画csb文件),运行上述代码非常顺利(当然,上述addArmatureFileInfo方法调用更早的ExportJson骨骼动画文件的情况也是能够顺利运行)。事实上,cpp-tests自然也已经在我的机器上顺序调试通过(我的环境是windows 7 64bits Visual Studio 2013)。但是,对于最后那个csb文件(使用当前新版本Cocos Studio 2.3.2导出的骨骼动画文件),则根本不行,执行中断停止在addArmatureFileInfo调用的下一行。

在经过部分的源码跟踪后,我尝试着使用碎图技术生成csb文件,尽量使之与cpp-tests提供的文件形式上一致,结果也根本通不过!

太遗憾了,我就是想使用Armature及相应的如下技术:

armature->getAnimation()->setMovementEventCallFunc(CC_CALLBACK_0(TestAnimationEvent::animationEvent,this,std::placeholders::_1,std::placeholders::_2,std::placeholders::_3));

但是,很遗憾,只能干瞪眼!没有Armature,我们根本无法使用setMovementEventCallFunc回调函数及其相应技术了。

遗憾的是,对于上述问题,官方网站上及DEMO中只字未提!

变通办法

对于我目前的程序中的上述要求,我只能尝试着其他的变通方法,因为我的要求也并不高。于是我尝试着使用帧事件方法解决了上述问题。

在此,我粘贴上我的示例游戏中的相关代码

第一部分如下:

//2loadtitleandmushroomanimation

Node*node2=CSLoader::createNode("SplashAnimationSkeleton.csb");

addChild(node2);

node2->setPosition(Vec2(VisibleRect::center().x,VisibleRect::center().y));

//在cocosstudio设计器中选择是否循环播放,对于在代码中动画的是否循环播放没有影响!!!//

ActionTimeline*action2=CSLoader::createTimeline("SplashAnimationSkeleton.csb");

node2->runAction(action2);

action2->gotoFrameAndPlay(0,false);

action2->setFrameEventCallFunc(CC_CALLBACK_1(SplashScene::onFrameEvent,this));

注意,上面的SplashAnimationSkeleton.csb是使用cocos studio 2.3.2创建的简单的骨骼动画文件

我原先设计的使用setMovementEventCallFunc方法结合Armature数据想实现的是目标是:当骨骼动画播放结束,触发另一个既定事件,并在这个事件中完成另外的动画播放任务。

对于上述目标,使用帧事件应该是可以的,只是稍微麻烦一些罢了。例如,需要在studio设计器中填写帧事件数据;但是,总算还可以实现。

另外一部分相关代码如下:

voidSplashScene::onFrameEvent(Frame*frame)
{
EventFrame*evnt=dynamic_cast<EventFrame*>(frame);
if(!evnt)
return;
std::stringstr=evnt->getEvent();
if(str=="lastFrame")
{
Node*butterfly_01=CSLoader::createNode("ButterflyArmature_01.csb");
addChild(butterfly_01,100);
butterfly_01->setPosition(Vec2(VisibleRect::right().x+100,0));
ActionTimeline*action2=CSLoader::createTimeline("ButterflyArmature_01.csb");
butterfly_01->runAction(action2);
action2->gotoFrameAndPlay(0,true);
Node*p1=_rootLayer->getChildByName("Mushroom_Point");
autoaction=Sequence::create(
MoveTo::create(2,p1->getPosition()),CallFunc::create(CC_CALLBACK_0(SplashScene::callback0,this)),nullptr);
butterfly_01->runAction(action);
}
}

大家看到,我在帧事件回调函数中进行判断,当动画播放到特定帧时(正是我以前要求的第一个动画播放结束时)触发另一个蝴蝶飞入动画的播放。

小结一下

通过学习与研究部分cocos2d-x 及cocos studio最新版本技术可以学习到更优秀的开发技术的同时,注定我要牺牲许多时间去“踏坑”,也许有得就有失吧。

最后,再提醒一下新手同学,示例工程中的代码部分与资源数据文件部分都有些不太明确的调用,当然,看起来官方是要尽量使用最新的c++代码来使用(或者说保护)早期studio导出的资源。但在同时,却露出了不少急于求成的“马脚”。

猜你在找的Cocos2d-x相关文章