保卫萝卜~上一篇说了使用CocoStudio制作主界面,这里来完善主界面动画及后续界面(主题界面ThemeUI、场景选择界面SelectMapUI),主要涉及的控件PageView。
  学习要写笔记,记录自己的步骤(练手作)。

一、主界面动画

     上一篇的主界面是静态的,这样很不好看。如果有玩过保卫萝卜就知道它的主界面是动的,云啊、叶子啊都在动,OK加几行代码来实现。
先看效果图:
  加入如下代码:
bool MenuScene :: init()
{
if ( !Layer :: init() )
{
return false ;
} cocos2d ::ui :: Widget* layout = cocostudio ::GUIReader :: getInstance()-> widgetFromJsonFile ("MainUi.json" );
this ->addChild ( layout);
Button * btnAdventure = ( Button*) layout ->getChildByName ( "btnAdventure");
Button * btnBossHome = ( Button*) layout ->getChildByName ( "btnBossHome");
Button * btnBoss = ( Button*) layout ->getChildByName ( "btnBoss");
btnAdventure ->addTouchEventListener ( this, toucheventselector (MenuScene :: btnClick));
btnBossHome ->addTouchEventListener ( this, toucheventselector (MenuScene :: btnClick));
btnBoss ->addTouchEventListener ( this, toucheventselector (MenuScene :: btnClick)); ImageView * cloud1 = ( ImageView*) layout ->getChildByName ( "cloud1");
ImageView * cloud2 = ( ImageView*) layout ->getChildByName ( "cloud2");
Vec2 size = Director ::getInstance ()-> getVisibleSize();
cloud1 ->runAction ( RepeatForever:: create (Sequence :: create( MoveBy ::create ( ,
Vec2 (size . x+ cloud1 ->getCustomSize (). width, )),Place :: create( Vec2 ( , cloud1-> getPositionY ())),nullptr )));
cloud2 ->runAction ( RepeatForever:: create (Sequence :: create( MoveBy ::create ( ,
Vec2 (size . x+ cloud2 ->getCustomSize (). width, )),Place :: create( Vec2 ( , cloud2-> getPositionY ())),nullptr )));
ImageView * bird = ( ImageView*) layout ->getChildByName ( "bird");
bird ->runAction ( RepeatForever:: create (
Sequence ::create (
MoveBy ::create ( , Vec2 ( , )),
MoveBy ::create ( , Vec2 ( ,- )), nullptr ))); ImageView * leaf2 = ( ImageView*) layout ->getChildByName ( "leaf2");
ImageView * leaf3 = ( ImageView*) layout ->getChildByName ( "leaf3");
RotateTo * rotateTo = RotateTo:: create (0.1f , );
RotateTo * rotateTo2 = RotateTo:: create (0.1f , );
leaf2 ->runAction ( RepeatForever:: create (
Sequence ::create (
Repeat ::create ( Sequence:: create (rotateTo , rotateTo2, nullptr ), ),
DelayTime ::create ( ), nullptr )));
leaf3 ->runAction ( RepeatForever:: create (
Sequence ::create (
DelayTime ::create ( ),
Repeat ::create ( Sequence:: create (rotateTo , rotateTo2, nullptr ), ),
DelayTime ::create ( ), nullptr ))); return true ;
}
  OK,加入动作表现之后界面利马生动起来。
 
二、主题选择
 
  1、场景UI制作
     菜单部分使用PageView控件,这个控件是翻页效果,先看效果:
 
     在CocoStudio中创建ThemeUI并使用PageView控件如:
 
 
  2、加入舞台
  这里有五个页面,分别是
  天际主题、森林主题、沙漠主题、海底主题、未开放的极地,
  每个页面下面的sign标签是通关数和总关卡数,这里是需要变动的。
  PageView控件可以使用手指滑动,也要有按钮的互动,因此有两个左右按钮,点击左上角的home会返回主界面。
  首先,把导出的json文件加至舞台,添加事件以及默认页:
cocos2d::ui::Widget* layout = cocostudio::GUIReader::getInstance()->widgetFromJsonFile("ThemeUI/ThemeUI.json");
this->addChild(layout); Button* btnMenu = (Button*)layout->getChildByName("btnMenu");
_btnLeft = (Button*)layout->getChildByName("btnLeft");
_btnRight = (Button*)layout->getChildByName("btnRight");
btnMenu->addTouchEventListener(this,toucheventselector(ThemeScene::btnClickMenu));
_btnLeft->addTouchEventListener(this,toucheventselector(ThemeScene::btnClickLeft));
_btnRight->addTouchEventListener(this,toucheventselector(ThemeScene::btnClickRight)); _pageView = (PageView*)layout->getChildByName("pageView");
_pageView->addEventListenerPageView(this,pagevieweventselector(ThemeScene::pageEvent));
int index = _pageView->getPages().size();
while (index-->)
{
Layout* pageIndex = (Layout*)_pageView->getPage(index);
pageIndex->setUserData((int*)index);
pageIndex->addTouchEventListener(this,toucheventselector(ThemeScene::layputEvent));
}
changePage( );
  这里有两个事件  1、PageView的翻动事件。2、点击PageView子页的事件
  这里有两个注意点
  1、翻动PageView要与按钮对应,首页左向按钮消失,尾页右向按钮消失
  2、点击按钮翻动PageView
void ThemeScene::btnClickLeft(Button* btn, TouchEventType eventType)
{
if( eventType == TOUCH_EVENT_ENDED )
{
auto nextPage = _pageView->getCurPageIndex() - ;
changePage( nextPage );
}
} void ThemeScene::btnClickRight(Button* btn, TouchEventType eventType)
{
if( eventType == TOUCH_EVENT_ENDED )
{
auto nextPage = _pageView->getCurPageIndex() + ;
changePage( nextPage );
}
} void ThemeScene::changePage( int index )
{
auto totalPage = _pageView->getPages().size();
_btnLeft->setVisible( index != );
_btnRight->setVisible( index != ( totalPage- ) ); if( index == _pageView->getCurPageIndex() ) return; if( index >= && index <= totalPage- )
{
_pageView->scrollToPage( index );
}
} void ThemeScene::pageEvent(Ref* page,PageViewEventType type)
{
log("EventType");
changePage( _pageView->getCurPageIndex() );
}
  这样不管是按钮还是触摸都可以正确的显示按钮状态。
 
  3、场景选择处理
  在初始化PageView子页的时候加入了标记数据UserData,这个数据表示选择的场景类型
enum MapType
{
Sky_Type,
Forest_Type,
Desert_Type,
Seabed_Type,
Max_Type
};
  这里的场景类型在后面的逻辑中是非常重要的数据,因此要找个地方去存储。
  选择主题事件触发
void ThemeScene::layputEvent(Layout* layout, TouchEventType eventType)
{
if( eventType == TOUCH_EVENT_ENDED )
{
int sign = (int)layout->getUserData();
log("layputEvent %d",(int)layout->getUserData());
if( sign >= && sign < MapType::Max_Type )
{
auto type = MapType((int)layout->getUserData());
SelectMapModel::getInstance()->setmapType(type);
Director::getInstance()->replaceScene(SelectMapUI::createScene());
}
else
{
log("not open");
}
}
}
  Ok这样就没有任何问题了。目前关于数据记录的方法还没有完全想好,所以子页的sign显示就是默认的。
三、地图选择
 
     1、场景UI制作

  场景类型MapType对应【1、2、3、6】
 
  这里也有一个PageView,这里面只做了一个pageItem,所有的地图子页都是一样的,可以用代码clone一下。当然每个场景的资源名字都是一样的,根据地图类型进行了区分。
  OK,看下效果吧:
 
 
   2、初始化地图
     首先,Json文件加入舞台,然后加入事件,初始化PageView。先拿到一个PageView子页,然后去初始化子页再clone添加至PageView
_pageView = (PageView*)layout->getChildByName("panelSky")->getChildByName("pageView");
Layout* item = (Layout*)_pageView->getChildByName("pageItem");
item->removeFromParent();
_pageView->removeAllPages();
int count = SelectMapModel::getInstance()->getMapCount(SelectMapModel::getInstance()->getmapType());
for (int idx = ; idx <= count; idx++)
{
createOnlyPage(_pageView,item,idx);
} void SelectMapUI::createOnlyPage(PageView* page,Layout* item, int idx)
{
std::string url = SelectMapModel::getInstance()->getMapResPath(SelectMapModel::getInstance()->getmapType());
ImageView* mapImg = (ImageView*)item->getChildByName("mapImg");
ImageView* towers = (ImageView*)item->getChildByName("towers"); mapImg->loadTexture(url+StringUtil::combination("ss_map%02d.png",idx));
towers->loadTexture(url+StringUtil::combination("ss_towers_%02d.png",idx)); page->addPage((Layout*)item->clone());
}
  其中自写StringUtil::combination是字符组合类,图片的路径都是通过MapType类决定的,根据选择不同的主题来决定地图数及图片路径。
  ”开始“按钮,根据选择的地图来决定战斗场景的初始化。
void SelectMapUI::btnClick(Button* btn, TouchEventType eventType)
{
if( eventType == TOUCH_EVENT_ENDED )
{
log("click btn is %s",btn->getName().c_str());
int sceneID = _pageView->getCurPageIndex() + ;
SceneMgr::getInstance()->replaceScene(sceneID);
}
}
四、最新进度
 
     目前,数据库、场景都已构架完毕,可以看下效果了。(地图太多了……时间不充裕目前只做了一个场景)

Cocos2d-x 3.2 学习笔记(十四)保卫萝卜之界面UI的更多相关文章

  1. python3.4学习笔记(十四) 网络爬虫实例代码,抓取新浪爱彩双色球开奖数据实例

    python3.4学习笔记(十四) 网络爬虫实例代码,抓取新浪爱彩双色球开奖数据实例 新浪爱彩双色球开奖数据URL:http://zst.aicai.com/ssq/openInfo/ 最终输出结果格 ...

  2. (C/C++学习笔记) 十四. 动态分配

    十四. 动态分配 ● C语言实现动态数组 C语言实现动态数组,克服静态数组大小固定的缺陷 C语言中,数组长度必须在创建数组时指定,并且只能是一个常数,不能是变量.一旦定义了一个数组,系统将为它分配一个 ...

  3. SharpGL学习笔记(十四) 材质:十二个材质球

    材质颜色 OpenGL用材料对光的红.绿.蓝三原色的反射率来近似定义材料的颜色.象光源一样,材料颜色也分成环境.漫反射和镜面反射成分,它们决定了材料对环境光.漫反射光和镜面反射光的反射程度.在进行光照 ...

  4. 【转】angular学习笔记(十四)-$watch(1)

    本篇主要介绍$watch的基本概念: $watch是所有控制器的$scope中内置的方法: $scope.$watch(watchObj,watchCallback,ifDeep) watchObj: ...

  5. angular学习笔记(十四)-$watch(1)

    本篇主要介绍$watch的基本概念: $watch是所有控制器的$scope中内置的方法: $scope.$watch(watchObj,watchCallback,ifDeep) watchObj: ...

  6. Java学习笔记十四:如何定义Java中的类以及使用对象的属性

    如何定义Java中的类以及使用对象的属性 一:类的重要性: 所有Java程序都以类class为组织单元: 二:什么是类: 类是模子,确定对象将会拥有的特征(属性)和行为(方法): 三:类的组成: 属性 ...

  7. MYSQL进阶学习笔记十四:MySQL 应用程序优化!(视频序号:进阶_32)

    知识点十五:MySQL 的应用程序优化(32) 一.访问数据库采用连接池 把连接当做对象或设备,统一放在‘连接池’里.凡是需要访问数据库的地方都从连接池里取连接 二.采用缓存减少对于MySQL的访问: ...

  8. Swift学习笔记十四:构造(Initialization)

         类和结构体在实例创建时,必须为全部存储型属性设置合适的初始值. 存储型属性的值不能处于一个未知的状态.     你能够在构造器中为存储型属性赋初值,也能够在定义属性时为其设置默认值.下面章节 ...

  9. JavaScript权威设计--Window对象之Iframe(简要学习笔记十四)

    1.Window对象属性的文档元素(id) 如果在HTML文档中用id属性来为元素命名,并且如果Window对象没有此名字的属性,Window对象会赋予一个属性,它的名字是id属性的值,而他们的值指向 ...

  10. Oracle学习笔记十四 内置程序包

    扩展数据库的功能 为 PL/SQL 提供对 SQL 功能的访问 用户 SYS 拥有所有程序包 是公有同义词 可以由任何用户访问 一些内置程序包 程序包名称 说明 STANDARD和DBMS_STAND ...

随机推荐

  1. android IOC框架学习记录

    一.框架如下几种: 1.Roboguice   2.Spring for Android   3.afinal   4.xUtils   二.Roboguice说明 项目地址:https://gith ...

  2. iOS开发--Block

    iOS开发--Block 1.什么是Block,block 的作用 ui开发和网络常见功能实现回调,按钮的事件处理方法是回调方法以及网络下载后的回调处理 (1)按钮 target-action   一 ...

  3. 我的c漏洞

    传入指针参数 #include<stdio.h> #include<stdlib.h> void READ(int *a) { scanf("%d",a); ...

  4. Redis 系列 (一) StackExchange.Redis的使用

    一.DLL安装 用NuGet搜索StackExchange.Redis,然后下载就可以. ConnectionMultiplexer对象是StackExchange.Redis最中枢的对象.这个类的实 ...

  5. MongoDB索引的使用

    Table of Contents 1. 基本索引 2. 联合索引 3. 索引类型 4. 索引管理 1 基本索引 在数据库开发中索引是非常重要的,对于检索速度,执行效率有很大的影响.本 文主要描述了M ...

  6. ASP.NET 开发必备知识点(2):那些年追过的ASP.NET权限管理

    一.前言 在前一篇文章已经为大家介绍了OWIN和Katana,有了对他们的了解之后,才能更好地去学习Asp.net Identity,因为Asp.net Identity的实现集成了Owin.其实在A ...

  7. Linux1:Linux概述

    为什么服务器尤其大型服务器都使用Linux系统 服务器尤其是大型服务器一般都使用Linux系统,有以下几点原因: 1.成本低,Linux操作系统是免费的 2.安全性好,Linux采取了许多的安全措施, ...

  8. 细说angular Form addControl方法

    在本篇博文中,我们将接触angular的验证.angular的验证是由form 指令和ngModel协调完成的.今天博主在这里想要说的是在验证在的一种特殊情况,当验证控件没有没有name属性这是不会被 ...

  9. 使用Guava提供的transform批量转换

    实际开发了,为了快速查询,我们会把日期以Long类型的方式存储到数据库中,比如20000000000000L,但显示的时候,要完整的日期,即yyyy-MM-dd的格式显示. 这个时候,我们就可以使用C ...

  10. ECSHOP农行支付接口开发(含手机端)

    对于ECSHOP来说,支付是以接口的形式存在的.于是: 1:首先添加接口文件 includes\modules\payment下,增加abcbank.php,代码如下: <?php /** * ...