cocos2dx 3.6 如何写一个ting wings :动态纹理

首先,几年前曾经玩过一段时间游戏开发,当时没用什么游戏引擎,整个开发过程很繁琐,但流程很清晰。现在各种引擎,遍地开花,开发方便,效果绚丽,但封装过于复杂,好多时候,是知其然而不知其所以然。还是要踏踏实实的走啊。

最近刚好离职,也不着急找工作,索性把以前未完成的想法,实现一下。
参考文章:
how-to-create-dynamic-textures-with-ccrendertexture-in-cocos2d-2-x
how-to-create-a-game-like-tiny-wings-with-cocos2d-2-x-part-1
how-to-create-a-game-like-tiny-wings-with-cocos2d-2-x-part-2
(译)如何使用CCRenderTexture来创建动态纹理
(译)如何制作一个类似tiny wings的游戏:第一部分
(译)如何制作一个类似tiny wings的游戏:第二部分(完)
如何使用CCRenderTexture创建动态纹理 Cocos2d-x 2.1.4
如何制作一个类似Tiny Wings的游戏 Cocos2d-x 2.1.4
如何制作一个类似Tiny Wings的游戏(2) Cocos2d-x 2.1.4
Cocos2d-x3.2与OpenGL渲染总结(一)Cocos2d-x3.2的渲染流程

这个教程,原版是ios下得,在最后一个链接,作者进行了相应的移植,使用的时cocos2dx 2.1.4。我在学习得过程中,发现cocos2dx的v2和v3版本,api变化比较大,核心的渲染过程也有了明显的差异。

在测试创建动态纹理时,发现作者在2.1.4版本下得代码,在3.6版本下没有实现渐变,和条纹的效果,经过几天时间的研究,整理出一份可以在3.6版本下,实现动态纹理的代码。为方便大家互相学习,将代码附在下文。


iOS Simulator Screen Shot 2015年7月15日 9.22.19.png
//
//  TextureTestScene.h
//  MyTestGame
//
//  Created by shutup on 15-7-15.
//
//

#ifndef __MyTestGame__TextureTestScene__
#define __MyTestGame__TextureTestScene__

#include "cocos2d.h"
class TextureTestScene : public cocos2d::Layer
{
public:
    // there's no 'id' in cpp, so we recommend returning the class instance pointer
    static cocos2d::Scene* createScene();
    
    // Here's a difference. Method 'init' in cocos2d-x returns bool, instead of returning 'id' in cocos2d-iphone
    virtual bool init();
    // implement the "static create()" method manually
    CREATE_FUNC(TextureTestScene);
    virtual void onEnter();
    //单色渐变
    cocos2d::Sprite *spriteWithColor(cocos2d::Color4F bgColor, float textureWidth, float textureHeight);
    //渐变加条纹
    cocos2d::Sprite * spriteWithColor1AndColor2(cocos2d::Color4F c1, cocos2d::Color4F c2, float textureWidth, float textureHeight, int nStripes);
    
    //随机颜色
    cocos2d::Color4F randombrightColor();
    //生成背景
    void genBackground();
    //触屏事件
    virtual bool onTouchBegan(cocos2d::Touch *touch,cocos2d::Event * event);
    
    TextureTestScene();
    virtual ~TextureTestScene();
   
    //绘制条纹的回调
    void drawStripes(cocos2d::Color4F c2, float textureWidth, float textureHeight, int nStripes);
    //绘制渐变的回调
    void drawGridient(float textureWidth, float textureHeight);
    private :
    cocos2d::Sprite* _background;
    cocos2d::CustomCommand _customCommand;
};
#endif /* defined(__MyTestGame__TextureTestScene__) */

//
//  TextureTestScene.cpp
//  MyTestGame
//
//  Created by shutup on 15-7-15.
//
//

#include "TextureTestScene.h"
USING_NS_CC;
TextureTestScene::TextureTestScene() {
    // TODO Auto-generated constructor stub
    _background = nullptr;
}

TextureTestScene::~TextureTestScene() {
    // TODO Auto-generated destructor stub
}


bool TextureTestScene::init() {
    if ( !Layer::init() )
    {
        return false;
    }
    return true;
}

Scene* TextureTestScene::createScene() {
    auto scene = Scene::create();
    auto layer = TextureTestScene::create();
    scene->addChild(layer);
    return scene;
}

void TextureTestScene::onEnter()
{
    
    Layer::onEnter();
    genBackground();
    setTouchEnabled(true);
    //设置为单点触摸
    setTouchMode(Touch::DispatchMode::ONE_BY_ONE);
}
bool TextureTestScene::onTouchBegan(Touch *pTouches, Event *pEvent)
{
    this->genBackground();
    return true;
}
Color4F TextureTestScene::randombrightColor()
{
    while(true)
    {
        float requiredBrightness = 192;
        Color4B randomColor = Color4B(rand()%255,rand()%255,rand()%255,255);
        if(randomColor.r > requiredBrightness || randomColor.g > requiredBrightness || randomColor.b >requiredBrightness)
        {
            return Color4F(randomColor);
        }
    }
    return Color4F();
}

void TextureTestScene::genBackground()
{
    if (_background)
    {
        _background->removeFromParentAndCleanup(true);
    }
    //渐变纹理
//    Color4F bgColor = this->randombrightColor();
//    _background = this->spriteWithColor(bgColor, 512, 512);
    
    //条纹纹理
    Color4F bgColor = this->randombrightColor();
    Color4F color2 = this->randombrightColor();
    int nStripes =((rand() % 4) + 1) * 2;
    _background = this->spriteWithColor1AndColor2(bgColor, color2, 512, 512, nStripes);
    
    
    Size winSize = Director::getInstance()->getVisibleSize();
    _background->setPosition(Vec2(winSize.width / 2, winSize.height / 2));
    this->addChild(_background, -1);
}

Sprite* TextureTestScene::spriteWithColor(Color4F bgColor,float textureWidth,float textureHeight) {
    
    RenderTexture* rt = RenderTexture::create(textureWidth,textureHeight);
    rt->beginWithClear(bgColor.r,bgColor.g,bgColor.b,bgColor.a);
    //draw the gradient
    
    _customCommand.init(_globalZOrder);
    _customCommand.func = CC_CALLBACK_0(TextureTestScene::drawGridient, this,textureWidth,textureHeight);
    auto renderer = Director::getInstance()->getRenderer();
    renderer->addCommand(&_customCommand);
    
    //do something more
//    draw the noise
    Sprite* noise = Sprite::create("Noise.png");
    BlendFunc blendFunc = {GL_DST_COLOR,GL_ZERO};
    noise->setBlendFunc(blendFunc);
    noise->setPosition(Vec2(textureWidth/2,textureHeight/2));
    noise->visit();
    
    rt->end();
    return Sprite::createWithTexture(rt->getSprite()->getTexture());
}
Sprite * TextureTestScene::spriteWithColor1AndColor2(Color4F c1, Color4F c2, float textureWidth, float textureHeight, int nStripes)
{
    // 1: Create new RenderTexture
    RenderTexture *rt = RenderTexture::create(textureWidth, textureHeight);
    
    // 2: Call CCRenderTexture:begin
    rt->beginWithClear(c1.r, c1.g, c1.b, c1.a);
    
    // 3: Draw into the texture
    
    _customCommand.init(_globalZOrder);
    _customCommand.func = CC_CALLBACK_0(TextureTestScene::drawStripes, this,c2,textureWidth,textureHeight,nStripes);
    auto renderer = Director::getInstance()->getRenderer();
    renderer->addCommand(&_customCommand);
    // Layer 4: Noise
    Sprite *noise = Sprite::create("Noise.png");
    BlendFunc blendFunc = {GL_DST_COLOR, GL_ZERO};
    noise->setBlendFunc(blendFunc);
    noise->setPosition(Vec2(textureWidth / 2, textureHeight / 2));
    noise->visit();
    
    // 4: Call CCRenderTexture:end
    rt->end();
    
    // 5: Create a new Sprite from the texture
    return Sprite::createWithTexture(rt->getSprite()->getTexture());
}

void TextureTestScene::drawStripes(Color4F c2, float textureWidth, float textureHeight, int nStripes)
{
    this->setGLProgram(GLProgramCache::getInstance()->getGLProgram(GLProgram::SHADER_NAME_POSITION_COLOR));
    CC_NODE_DRAW_SETUP();
    
    {
        // Layer 1: Stripes
        Point *vertices = new Point[nStripes * 6];
        Color4F *colors = new Color4F[nStripes * 6];
        
        int nVertices = 0;
        float x1 = -textureHeight;
        float x2;
        float y1 = textureHeight;
        float y2 = 0;
        float dx = textureWidth / nStripes * 2;
        float stripeWidth = dx / 2;
        for (int i = 0; i < nStripes; ++i)
        {
            x2  = x1 + textureHeight;
            
            vertices[nVertices] = Point(x1, y1);
            colors[nVertices++] = Color4F(c2.r, c2.g, c2.b, c2.a);
            
            vertices[nVertices] = Point(x1 + stripeWidth, y1);
            colors[nVertices++] = Color4F(c2.r, c2.g, c2.b, c2.a);
            
            vertices[nVertices] = Point(x2, y2);
            colors[nVertices++] = Color4F(c2.r, c2.g, c2.b, c2.a);
            
            vertices[nVertices] = vertices[nVertices - 2];
            colors[nVertices++] = Color4F(c2.r, c2.g, c2.b, c2.a);
            
            vertices[nVertices] = vertices[nVertices - 2];
            colors[nVertices++] = Color4F(c2.r, c2.g, c2.b, c2.a);
            
            vertices[nVertices] = Point(x2 + stripeWidth, y2);
            colors[nVertices++] = Color4F(c2.r, c2.g, c2.b, c2.a);
            x1 += dx;
        }
        
//      glEnableVertexAttribArray(GLProgram::VERTEX_ATTRIB_POSITION|GLProgram::VERTEX_ATTRIB_COLOR);
        GL::enableVertexAttribs(GL::VERTEX_ATTRIB_FLAG_POSITION | GL::VERTEX_ATTRIB_FLAG_COLOR);
        
        glVertexAttribPointer(GLProgram::VERTEX_ATTRIB_POSITION, 2, GL_FLOAT, GL_FALSE, 0, vertices);
        glVertexAttribPointer(GLProgram::VERTEX_ATTRIB_COLOR, 4, GL_FLOAT, GL_FALSE, 0, colors);
        glBlendFunc(CC_BLEND_SRC, CC_BLEND_DST);
        glDrawArrays(GL_TRIANGLES, 0, (GLsizei)nVertices);
        //通知cocos2d-x 的renderer,让它在合适的时候调用这些OpenGL命令
//        CC_INCREMENT_GL_DRAWN_BATCHES_AND_VERTICES(1, 3);
        //如果出错了,可以使用这个函数来获取出错信息
        CHECK_GL_ERROR_DEBUG();
        CC_SAFE_DELETE_ARRAY(vertices);
        CC_SAFE_DELETE_ARRAY(colors);
        
    }
    
    {float gradientAlpha = 0.7f;
        Point vertices[4];
        Color4F colors[4];
        int nVertices =0;
        vertices[nVertices] = Point(0,0);
        colors[nVertices++] = Color4F(0,0,0,0);
        vertices[nVertices] = Point(textureWidth,0);
        colors[nVertices++] = Color4F(0,0,0,0);
        vertices[nVertices] = Point(0,textureHeight);
        colors[nVertices++] = Color4F(0,0,0,gradientAlpha);
        vertices[nVertices] = Point(textureWidth,textureHeight);
        colors[nVertices++] = Color4F(0,0,0,gradientAlpha);
        //这句有点问题
        //  glEnableVertexAttribArray(GLProgram::VERTEX_ATTRIB_POSITION|GLProgram::VERTEX_ATTRIB_COLOR);
        GL::enableVertexAttribs(GL::VERTEX_ATTRIB_FLAG_POSITION | GL::VERTEX_ATTRIB_FLAG_COLOR);
        glVertexAttribPointer(GLProgram::VERTEX_ATTRIB_POSITION,2,GL_FLOAT,GL_FALSE,0,vertices);
        glVertexAttribPointer(GLProgram::VERTEX_ATTRIB_COLOR,4,GL_FLOAT,GL_FALSE,0,colors);
        glBlendFunc(CC_BLEND_SRC,CC_BLEND_DST);
        glDrawArrays(GL_TRIANGLE_STRIP,0,(GLsizei)nVertices);
        CHECK_GL_ERROR_DEBUG();
    }
}

void TextureTestScene::drawGridient(float textureWidth, float textureHeight)
{
    this->setGLProgram(GLProgramCache::getInstance()->getGLProgram(GLProgram::SHADER_NAME_POSITION_COLOR));
    CC_NODE_DRAW_SETUP();
    
    {
        float gradientAlpha = 0.7f;
        Point vertices[4];
        Color4F colors[4];
        int nVertices =0;
        vertices[nVertices] = Point(0,0);
        colors[nVertices++] = Color4F(0,0,0,0);
        vertices[nVertices] = Point(textureWidth,0);
        colors[nVertices++] = Color4F(0,0,0,0);
        vertices[nVertices] = Point(0,textureHeight);
        colors[nVertices++] = Color4F(0,0,0,gradientAlpha);
        vertices[nVertices] = Point(textureWidth,textureHeight);
        colors[nVertices++] = Color4F(0,0,0,gradientAlpha);
        //这句有点问题
        //  glEnableVertexAttribArray(GLProgram::VERTEX_ATTRIB_POSITION|GLProgram::VERTEX_ATTRIB_COLOR);
        GL::enableVertexAttribs(GL::VERTEX_ATTRIB_FLAG_POSITION | GL::VERTEX_ATTRIB_FLAG_COLOR);
        glVertexAttribPointer(GLProgram::VERTEX_ATTRIB_POSITION,2,GL_FLOAT,GL_FALSE,0,vertices);
        glVertexAttribPointer(GLProgram::VERTEX_ATTRIB_COLOR,4,GL_FLOAT,GL_FALSE,0,colors);
        glBlendFunc(CC_BLEND_SRC,CC_BLEND_DST);
        glDrawArrays(GL_TRIANGLE_STRIP,0,(GLsizei)nVertices);
        CHECK_GL_ERROR_DEBUG();
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351

推荐阅读更多精彩内容