提出如题所示的问题,心里非常别扭,但的确是事实。因此,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):

 

 

 

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

const  char* HelloWorld::m_binaryFilesNames[4] = { "bear.csb", "horse.csb","Cowboy.csb","ccc.csb"}; const  char* HelloWorld::m_armatureNames[4] = { "bear", "horse","Cowboy","Skeleton1"}; //......  // load from binary  ArmatureDataManager::getInstance()->addArmatureFileInfo(m_binaryFilesNames[3]);  Armature *m_armature = Armature::create(m_armatureNames[3]);  m_armature->getAnimation()->playWithIndex(0);  m_armature->setScale(1.0f);  Size size = 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中只字未提!

 

变通办法

 

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

 

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

第一部分如下:

 //2load title and mushroom animation  Node* node2 = CSLoader::createNode("SplashAnimationSkeleton.csb");  addChild(node2);  node2->setPosition(Vec2(VisibleRect::center().x, VisibleRect::center().y));  //在cocos studio设计器中选择是否循环播放,对于在代码中动画的是否循环播放没有影响!!!//  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设计器中填写帧事件数据;但是,总算还可以实现。

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

void SplashScene::onFrameEvent(Frame* frame) {  EventFrame* evnt = dynamic_cast<EventFrame*>(frame);  if (!evnt)   return;  std::string str = 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");   auto  action = 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导出的资源。但在同时,却露出了不少急于求成的“马脚”。