Cocos2d-x 3.2:通过ClippingNode实现一个功能完善的跑马灯公告(1)

Cocos2d-x 3.2:通过ClippingNode实现一个功能完善的跑马灯公告(1)

本文转载至深入理解Cocos2d-x 3.x:一步一步通过ClippingNode实现一个功能完善的跑马灯公告(1)

 

这篇文章主要是通过一步一步实现一个功能完善的跑马灯公告来展示ClippingNode的用法并且最终深入ClippingNode的源码,了解其实现原理。

首先,先介绍一下ClippingNode,ClippingNode也叫裁剪节点,能将一些内容通过使用模板裁剪出来显示在界面上,可以实现一些很炫酷的效果。来看看今天要实现的效果

1.gif


1、ClippingNode类分析

先来看看ClippingNode的声明文件 看看其中的public方法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
class CC_DLL ClippingNode : public Node  
{  
public:  
    static ClippingNode* create();  
    static ClippingNode* create(Node *stencil);  
   
    Node* getStencil() const;  
    void setStencil(Node *stencil);  
   
    virtual bool hasContent() const;  
   
    GLfloat getAlphaThreshold() const;  
    void setAlphaThreshold(GLfloat alphaThreshold);  
       
    bool isInverted() const;  
    void setInverted(bool inverted);  
};

首先是create,这个方法是用于创建一个ClippingNode,这个就不多做赘述了,第二个create是创建一个带遮罩模板的裁剪节点。

接下来的getStencil和setStencil分别是获取和设置一个遮罩模板,裁剪物体方法就是通过这个遮罩模板的,遮罩模板只要是基于Node的对象都可以(非常重要)。

接下来的hasContent返回其是否有需要绘制的内容,如果没有绘制的内容则返回false,有则返回true。

getAlphaThreshold 和setAlphaThreshold分别是获取和设置一个像素的透明度值,取值范围从0-1,其中0表示都不绘制,1表示都绘制。0.5表示透明度在 0.5以上的都绘制,这个函数涉及到opengl的Alpha测试的相关概念,Alpha测试的作用通过一句话解释就是:所有像素的透明度值低于某个阀值 的统统抛弃,不绘制到屏幕上。

最后的isInverted和setInverted分别表示绘制的内容是模板内的还是模板外的,其效果如下:

2.jpg

3.jpg


2、简易跑马灯实现

上节简单介绍了一下ClippingNode的函数,这节就通过实现一个简易的跑马灯功能来直观的了解。首先介绍一下制作跑马灯的思路。

首先我们需要将跑马灯中的一部分超出的字裁剪掉,不让他显示在界面上。这就需要用到ClippingNode,现在先来做第一步。实现的代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
//设置模板  
auto stencil = Sprite::create();  
//设置显示区域大小  
stencil->setTextureRect(Rect(0, 0, 50, 30));  
   
//设置跑马灯文字  
auto text = Label::createWithSystemFont("-1dasdasdasd efadaewfevgds dfhrthrbgrg1-""", 24);  
//设置锚点  
text->setAnchorPoint(Vec2::ANCHOR_MIDDLE_LEFT);  
   
//创建裁剪节点  
auto clippingNode = ClippingNode::create(stencil);  
//设置节点位置  
clippingNode->setPosition(Vec2(700, 400));  
//显示模板内的内容  
clippingNode->setInverted(false);  
//添加显示内容  
clippingNode->addChild(text, 2);  
//加入到UI树  
addChild(clippingNode);

上述的每一句代码都有注释,就不再多解释了,这一步实现出来的效果如下图,但是跑马灯还不能动起来,待会我们就将跑马灯动起来。

4.jpg

现在我们就设计一个Action将跑马灯动起来,跑马灯一般需要先将文字左移,移动到文字看不见的时候再将文字移除或者隐藏,代码如下(为了简便,就直接设置隐藏了):

1
2
auto sequ = Sequence::create(MoveBy::create(5.0f, Vec2(-text->getContentSize().width, 0)), Hide::create(), nullptr);  
text->runAction(sequ);

现在跑马灯的样子就如同开篇展示的那样了,可是这样还不能直接使用,因为这只是一串代码,还需要对其进行一定的封装,然后提供一个非常简便的方法给别的类调用。


3、封装

现在我们从便捷性的角度考虑如何将跑马灯功能封装成一个函数供其他类调用。首先提取出函数的参数,分别是:显示区域,跑马灯文字,字体字号,跑马灯位置,跑马灯的父节点。下面是初步封装好的一套跑马灯函数的声明:

1
void showMarquee(Node* parent, const std::string& text, const std::string& font, float fontSize, const Rect& showRect, const Vec2& position);

看参数是不是有些略多,每次调用这个函数是不是非常的不方便,那么我们现在来看看究竟有那些参数是必须要传入的吧。每次调用跑马灯显示的文字都会改变,其他的参数在一个游戏中是不会改变的。那么就有必要做一个类来保证使用方法的便捷性了。

首先,我们简单的构建一下一个跑马灯类,如下

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
#include "cocos2d.h"  
   
USING_NS_CC;  
   
class Marquee : public Node  
{  
public:  
   
    CREATE_FUNC(Marquee);  
   
    bool init();  
   
    void show(const std::string& text);  
   
public:  
    const std::string& getFont() const return _font; }  
    void setFont(std::string& font) { _font = font; }  
    float getFontSize() const return _fontSize; }  
    void setFontSize(float fontSize) { _fontSize = fontSize; }  
   
public:  
    const Rect& getShowRect() const return _showRect; }  
    void setShowRect(Rect& showRect) { _showRect = showRect; }  
protected:  
    Marquee() :   
        _font(""),  
        _fontSize(24),  
        _showRect(Rect(0,0,200,30))  
    {};  
    ~Marquee() {};  
   
private:  
    std::string _font;  
    float _fontSize;  
   
    Rect _showRect;  
};

然后是最重要的init方法和show方法的实现

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
bool Marquee::init()  
{  
    //设置模板  
    auto stencil = Sprite::create();  
    //设置显示区域大小  
    stencil->setTextureRect(_showRect);  
   
    //设置跑马灯文字  
    _label = Label::createWithSystemFont("", _font, _fontSize);  
    //设置锚点  
    _label->setAnchorPoint(Vec2::ANCHOR_MIDDLE_LEFT);  
   
    _label->setAlignment(TextHAlignment::LEFT);  
   
    //创建裁剪节点  
    auto clippingNode = ClippingNode::create(stencil);  
    //显示模板内的内容  
    clippingNode->setInverted(false);  
    //添加显示内容  
    clippingNode->addChild(_label);  
    //加入到UI树  
    addChild(clippingNode);  
   
    return true;  
}  
   
void Marquee::show(const std::string& text)  
{  
    _label->setString(text);  
    _label->setPosition(Vec2(0, _label->getContentSize().height / 2));  
   
    auto sequ = Sequence::create(MoveBy::create(5.0f, Vec2(-_label->getContentSize().width, 0)), Hide::create(), nullptr);  
    _label->runAction(sequ);  
}

这样就可以通过以下的调用方法来调用跑马灯了

1
2
3
4
Marquee* m = Marquee::create();  
    m->show("----hhahahah veeeeee-----");  
    m->setPosition(Vec2(700, 300));  
    this->addChild(m);


4、完善

看 上去,此前的步骤我们已经完成了一个跑马灯的功能,实际上这个类距离真正能使用还差那么一点点,因为传入跑马灯的消息的传入时机是不确定的,可能这一条消 息还没有播放完成下一条就要开始播放了。这样就需要实现一个播放等待队列,将需要播放的消息加入播放队列,然后跑马灯自动判断是否需要显示。下面是改进后 的类声明文件以及实现文件。

.h:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
#include "cocos2d.h"  
   
USING_NS_CC;  
   
class Marquee : public Node  
{  
public:  
    enum class State  
    {  
        idle,  
        playing,  
    };  
   
public:  
   
    CREATE_FUNC(Marquee);  
   
    bool init();  
   
    void addMessage(const std::string& text);  
   
public:  
    const std::string& getFont() const return _font; }  
    void setFont(std::string& font) { _font = font; }  
    float getFontSize() const return _fontSize; }  
    void setFontSize(float fontSize) { _fontSize = fontSize; }  
   
public:  
    const Rect& getShowRect() const return _showRect; }  
    void setShowRect(Rect& showRect) { _showRect = showRect; }  
   
public:  
    const State& getState() const return _state; }  
   
protected:  
    Marquee() :   
        _font(""),  
        _fontSize(24),  
        _showRect(Rect(0,0,200,30)),  
        _state(State::idle)  
    {};  
    ~Marquee() {};  
    void show(const std::string& text);  
   
private:  
    State _state;  
   
private:  
    std::string _font;  
    float _fontSize;  
   
    Rect _showRect;  
   
private:  
    Label * _label;  
   
private:  
    std::queue<std::string> _texts;  
};

.cpp:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
#include <Marquee.h>  
   
bool Marquee::init()  
{  
    //设置模板  
    auto stencil = Sprite::create();  
    //设置显示区域大小  
    stencil->setTextureRect(_showRect);  
   
    //设置跑马灯文字  
    _label = Label::createWithSystemFont("", _font, _fontSize);  
    //设置锚点  
    _label->setAnchorPoint(Vec2::ANCHOR_MIDDLE_LEFT);  
   
    _label->setAlignment(TextHAlignment::LEFT);  
   
    //创建裁剪节点  
    auto clippingNode = ClippingNode::create(stencil);  
    //显示模板内的内容  
    clippingNode->setInverted(false);  
    //添加显示内容  
    clippingNode->addChild(_label);  
    //加入到UI树  
    addChild(clippingNode);  
    stencil->setColor(Color3B::BLACK);  
    addChild(stencil, -1);  
   
    return true;  
}  
   
void Marquee::show(const std::string& text)  
{  
    _state = State::playing;  
   
    _label->setString(text);  
    _label->setPosition(Vec2(0, 0));  
   
    auto sequ = Sequence::create(  
        Show::create(),   
        MoveBy::create(5.0f, Vec2(-(_label->getContentSize().width + _showRect.size.width / 2), 0)),  
        Hide::create() , DelayTime::create(1.0f),   
        CCCallFunc::create([&]()  
    {  
        if (_texts.size() == 0)  
        {  
            _state = State::idle;  
        }  
        else  
        {  
            show(_texts.front());  
            _texts.pop();  
        }  
    }), nullptr);  
    _label->runAction(sequ);  
}  
   
void Marquee::addMessage(const std::string& text)  
{  
    if (text.empty())  
    {  
        return;  
    }  
       
    if (_state == State::idle)  
    {  
        show(text);  
    }  
    else  
    {  
        _texts.push(text);  
    }  
}

此处将show方法隐藏,并且提供了addMessage方法,内部实现了一个有限状态机,根据状态来显示剩余的消息,其使用方法与此前相似:

1
2
3
4
m = Marquee::create();  
m->addMessage("----hhahahah veeeeee-----");  
m->setPosition(Vec2(700, 300));  
this->addChild(m);

 

转载于:https://www.cnblogs.com/dudu580231/p/4560310.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/400235.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【STC15库函数上手笔记】1、建立工程

目录新建工程添加文件到工程中main.c时钟STC实验箱4 IAP15W4K58S4 Keil uVision V5.29.0.0 PK51 Prof.Developers Kit Version:9.60.0.0 在STC官网发现了库函数&#xff0c;大受震撼&#xff1a; 宏晶科技官方网站 直接搜“库函数”&#xff0c;目前仅有STC15和STC8系列有库函…

hdu 6086 Rikka with String(AC自动机+状压dp)

题目链接&#xff1a;hdu 6086 Rikka with String 题意&#xff1a; 给你n个只含01的串&#xff0c;和一个长度L,现在让你构造出满足s[i]≠s[|s|−i1] for all i∈[1,|s|] &#xff0c;长度为2L&#xff0c;并且包含给出的n个串&#xff0c;问能有多少种这样的串。 题解&#x…

【STC15库函数上手笔记】2、GPIO

目录硬知识IO口初始化函数测试main.c实验现象STC实验箱4 IAP15W4K58S4 Keil uVision V5.29.0.0 PK51 Prof.Developers Kit Version:9.60.0.0 硬知识 摘自《STC库函数使用参考》 IO口初始化函数 GPIO_Inilize GPIO_InitTypeDef的定义见于文件"GPIO.H"。 typede…

Json-转自菜鸟教程

1. python中为什么用json有什么作用&#xff1f;&#xff1f;不是python用json&#xff0c;json是类似xml的一种通用格式&#xff0c;在很多地方都可以用。json相比xml&#xff0c;数据量更小&#xff0c;而且可以很方便的和解释型语言的结构相互转换。 2. 常用的两种Json函数&…

centos下编译安装curl拓展

---恢复内容开始--- 新的php环境没有curl拓展&#xff0c;现在用编译方式增加拓展。 一、安装curl 当前curl最新版本为7.32&#xff0c;下载地址为http://curl.haxx.se/download/curl-7.32.0.tar.gz 使用wget方式下载到相关目录 wget http://curl.haxx.se/download/curl-7.32.0…

【STC15库函数上手笔记】3、外部中断

目录硬知识外中断初始化函数测试main.cExti.cSTC实验箱4 IAP15W4K58S4 Keil uVision V5.29.0.0 PK51 Prof.Developers Kit Version:9.60.0.0 硬知识 摘自《STC库函数使用参考》 外中断初始化函数 Ext_Inilize EXTI_InitTypeDef的定义见于文件"Exti.H"。 typed…

Mac pycharm flask 用内网ip 运行 web

1. 菜单 run -> run 2. 选择&#xff0c;你要运行的 py 程序 3. 设置 Additional options: 为 --host0.0.0.0 &#xff0c;把 FLASK_DEBUG 的勾&#xff0c;去掉&#xff0c;如下图 4. 在 py 程序中 &#xff08;运行的主程序&#xff09;&#xff0c; 修改以下代码 …

【STC15库函数上手笔记】4、USART串口

目录硬知识串口初始化函数串口1写缓冲函数串口2写缓冲函数串口1写数据块函数串口2写数据块函数模拟串口字节发送函数模拟串口写数据块函数测试硬件串口模拟串口soft_uart.csoft_uart.hmain.cSTC实验箱4 IAP15W4K58S4 Keil uVision V5.29.0.0 PK51 Prof.Developers Kit Versio…

CSS--使用伪选择器制作箭头图标

// 使用Transform的属性&#xff0c;组合translate&#xff08;位移&#xff09;和rotate&#xff08;旋转&#xff09;&#xff0c;将生成的小矩形组合成各种箭头样式&#xff1b; HTML 1 <section class"main">2 <header class"title">图…

Linux多命令协作:管道及重定向

认识Linux中管道和重定向 如下图。 实践&#xff1a;重定向>的使用 转载于:https://www.cnblogs.com/mcgrady/p/7339770.html

【STC15库函数上手笔记】5、定时器

目录硬知识定时器初始化函数测试基本功能测试main.c中断服务函数实验现象模拟PWM信号main.c实验现象STC实验箱4 IAP15W4K58S4 Keil uVision V5.29.0.0 PK51 Prof.Developers Kit Version:9.60.0.0 硬知识 摘自《STC库函数使用参考》 定时器初始化函数 Timer_Inilize TIM_…

Linux内存卡(SD卡、TF卡)作为Swap交换空间

目录平台&#xff1a;华硕 Thinker Edge R 瑞芯微 RK3399Pro 固件版本&#xff1a;Tinker_Edge_R-Debian-Stretch-V1.0.4-20200615 编译ORB-SLAM3时内存不足报错&#xff0c;专门买来一个32G的内存卡&#xff0c;设成swap试试。 插上内存卡&#xff0c;使用如下命令查看&#…

mysqldumper

介绍MySQL自身的mysqldump工具支持单线程工作&#xff0c;依次一个个导出多个表&#xff0c;没有一个并行的机&#xff0c;这就使得它无法迅速的备份数据。mydumper作为一个实用工具&#xff0c;能够良好支持多线程工作&#xff0c;可以并行的多线程的从表中读入数据并同时写到…

【STC15库函数上手笔记】6、ADC

目录硬知识ADC初始化函数ADC电源控制函数ADC查询转换函数测试main.c实验现象STC实验箱4 IAP15W4K58S4 Keil uVision V5.29.0.0 PK51 Prof.Developers Kit Version:9.60.0.0 硬知识 摘自《STC库函数使用参考》 ADC初始化函数 ADC_Inilize ADC_InitTypeDef的定义见于文件&q…

【RK3399Pro学习笔记】十七、Debian安装ORB-SLAM3和单目demo的运行

目录安装OpenCV 3注意事项安装Glew安装Pangolin安装boost安装Eigen 3安装ORB_SLAM3试用平台&#xff1a;华硕 Thinker Edge R 瑞芯微 RK3399Pro 固件版本&#xff1a;Tinker_Edge_R-Debian-Stretch-V1.0.4-20200615 参考资料&#xff1a; 【入门必看】视觉惯性SLAM“灭霸”&am…

【0.96寸 OLED屏实现1500Fps的帧率】STM32 软件、硬件SPI、I2C驱动总结

目录SPI版OLED SPI 端口定义七针OLED引脚定义六针OLED引脚定义软件SPI硬件SPI启用DMA帧率测试I2C 版软件I2C硬件I2CDMASTM32F103VET6 STM32 Cube IDE SPI版 OLED SPI 端口定义 本节引自STM32驱动0.96寸OLED液晶屏(12864液晶屏) —— 小牧同学 两种屏幕的引脚数不一样&#x…

网络工程师第五站-有线、无线同网段混合组网(多FAT案例)

本案例中用到的设备&#xff1a;路由器&#xff1a;cisco 2851交换机&#xff1a;cisco WS-C3750G-12S &#xff08;HX&#xff09;&#xff0c;接入&#xff1a;WS-C2960-48TC-LAP &#xff1a; 华为AP3010dn-AGN3750G 作为该网络核心交换机HX&#xff0c;前面的接口&#xff…

Jstat使用样例

jstat -gc pid //以字节形式展现 jstat -gcutil pid //以百分比形式展现 jstat -class pid //类加载和卸载状况 jstat -compilier pid //查看Jit编译的方法数及失败情况 jstat -printcompilation 4672 //打印最近编译的一个方法 转载于:https://www.cnblogs.com/ironroot/p…

便宜的手机图传遥控模块

目录测试程序FHDFPV.cFHDFPV.h应用效果STM32F103RC STM32 Cube IDE 大一时买的&#xff0c;当时是50元左右&#xff0c;很便宜&#xff0c;安卓上位机为FHDFPV 我买的型号波特率约为18464&#xff0c; 实测帧头为102&#xff0c;帧尾为153&#xff0c;前四个数为左右遥感对应…

使用pycharm配置flask项目,并使用git进行版本控制

https://blog.csdn.net/s1025491598/article/details/86609255 遇到的坑 (Mac pycharm 2018) 1. 在pycharm Terminal 命令窗口下&#xff0c;运行 git init mac下查看git安装目录&#xff0c;后解决