/**************************************************************************** Copyright (c) 2014 Chukong Technologies Inc. http://www.cocos2d-x.org Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. ****************************************************************************/ #include "ShaderTest2.h" #include "ShaderTest.h" #include "../testResource.h" #include "cocos2d.h" namespace ShaderTest2 { static std::function createFunctions[] = { CL(AttribShaderTest), CL(NormalSpriteTest), CL(GreyScaleSpriteTest), CL(BlurSpriteTest), CL(NoiseSpriteTest), CL(EdgeDetectionSpriteTest), CL(BloomSpriteTest), CL(CelShadingSpriteTest), CL(LensFlareSpriteTest), CL(OutlineShadingSpriteTest), CL(UniformShaderTest), CL(UniformAttribShaderTest) }; static unsigned int TEST_CASE_COUNT = sizeof(ShaderTest2::createFunctions) / sizeof(ShaderTest2::createFunctions[0]); static int sceneIdx=-1; Layer* createTest(int index) { auto layer = (createFunctions[index])();; return layer; } Layer* nextAction(); Layer* backAction(); Layer* restartAction(); Layer* nextAction() { sceneIdx++; sceneIdx = sceneIdx % TEST_CASE_COUNT; return createTest(sceneIdx); } Layer* backAction() { sceneIdx--; if( sceneIdx < 0 ) sceneIdx = TEST_CASE_COUNT -1; return createTest(sceneIdx); } Layer* restartAction() { return createTest(sceneIdx); } } ShaderTestDemo2::ShaderTestDemo2() { } void ShaderTestDemo2::backCallback(Ref* sender) { auto s = ShaderTestScene2::create(); s->addChild( ShaderTest2::backAction() ); Director::getInstance()->replaceScene(s); } void ShaderTestDemo2::nextCallback(Ref* sender) { auto s = ShaderTestScene2::create(); s->addChild( ShaderTest2::nextAction() ); Director::getInstance()->replaceScene(s); } void ShaderTestDemo2::restartCallback(Ref* sender) { auto s = ShaderTestScene2::create(); s->addChild(ShaderTest2::restartAction()); Director::getInstance()->replaceScene(s); } void ShaderTestScene2::runThisTest() { auto layer = ShaderTest2::nextAction(); addChild(layer); Director::getInstance()->replaceScene(this); } template class ShaderSpriteCreator { public: static spriteType* createSprite(const std::string& filename) { spriteType* ret = spriteType::create(); ret->setTexture(filename); ret->initShader(); ret->setBackgroundNotification(); return ret; } }; class ShaderSprite : public Sprite { public: ShaderSprite(); ~ShaderSprite(); virtual void initShader(); void setBackgroundNotification(); virtual void draw(Renderer *renderer, const Matrix &transform, bool transformUpdated) override; void listenBackToForeground(Ref *obj); protected: virtual void setCustomUniforms() = 0; std::string _fragSourceFile; std::string _vertSourceFile; CustomCommand _renderCommand; void onDraw(const Matrix &transform, bool transformUpdated); }; ShaderSprite::ShaderSprite() :_vertSourceFile("") { } ShaderSprite::~ShaderSprite() { } void ShaderSprite::setBackgroundNotification() { #if CC_ENABLE_CACHE_TEXTURE_DATA auto listener = EventListenerCustom::create(EVENT_COME_TO_FOREGROUND, [this](EventCustom* event){ this->setShaderProgram(nullptr); this->initShader(); }); _eventDispatcher->addEventListenerWithSceneGraphPriority(listener, this); #endif } void ShaderSprite::initShader() { auto fileUtiles = FileUtils::getInstance(); auto fragmentFilePath = fileUtiles->fullPathForFilename(_fragSourceFile); auto fragSource = fileUtiles->getStringFromFile(fragmentFilePath); std::string vertSource; if (_vertSourceFile.empty()) { vertSource = ccPositionTextureColor_vert; }else{ std::string vertexFilePath = fileUtiles->fullPathForFilename(_vertSourceFile); vertSource = fileUtiles->getStringFromFile(vertexFilePath); } auto program = GLProgram::createWithByteArrays(vertSource.c_str(), fragSource.c_str()); setGLProgram(program); auto glprogramState = getGLProgramState(); setCustomUniforms(); #define kQuadSize sizeof(_quad.bl) size_t offset = (size_t)&_quad; // vertex int diff = offsetof( V3F_C4B_T2F, vertices); glprogramState->setVertexAttribPointer("a_position", 3, GL_FLOAT, GL_FALSE, kQuadSize, (void*) (offset + diff)); // texCoods diff = offsetof( V3F_C4B_T2F, texCoords); glprogramState->setVertexAttribPointer("a_texCoord", 2, GL_FLOAT, GL_FALSE, kQuadSize, (void*) (offset + diff)); // color diff = offsetof( V3F_C4B_T2F, colors); glprogramState->setVertexAttribPointer("a_color", 4, GL_UNSIGNED_BYTE, GL_TRUE, kQuadSize, (void*) (offset + diff)); } void ShaderSprite::draw(Renderer *renderer, const Matrix &transform, bool transformUpdated) { _renderCommand.init(_globalZOrder); _renderCommand.func = CC_CALLBACK_0(ShaderSprite::onDraw, this, transform, transformUpdated); renderer->addCommand(&_renderCommand); } void ShaderSprite::onDraw(const Matrix &transform, bool transformUpdated) { GL::blendFunc(_blendFunc.src, _blendFunc.dst); GL::bindTexture2D(_texture->getName()); getGLProgramState()->apply(transform); glDrawArrays(GL_TRIANGLE_STRIP, 0, 4); CC_INCREMENT_GL_DRAWN_BATCHES_AND_VERTICES(1, 4); } class NormalSprite : public ShaderSprite, public ShaderSpriteCreator { public: CREATE_FUNC(NormalSprite); NormalSprite(); protected: virtual void setCustomUniforms() override; }; NormalSprite::NormalSprite() { _fragSourceFile = "Shaders/example_normal.fsh"; } void NormalSprite::setCustomUniforms() { } class GreyScaleSprite : public ShaderSprite, public ShaderSpriteCreator { public: CREATE_FUNC(GreyScaleSprite); GreyScaleSprite(); protected: virtual void setCustomUniforms() override; }; GreyScaleSprite::GreyScaleSprite() { _fragSourceFile = "Shaders/example_greyScale.fsh"; } void GreyScaleSprite::setCustomUniforms() { } class BlurSprite : public ShaderSprite, public ShaderSpriteCreator { public: CREATE_FUNC(BlurSprite); BlurSprite(); void setBlurSize(float f); protected: virtual void setCustomUniforms() override; int _blurRadius; Vector2 _pixelSize; int _samplingRadius; float _scale; float _cons; float _weightSum; }; BlurSprite::BlurSprite() { _fragSourceFile = "Shaders/example_Blur.fsh"; } void BlurSprite::setCustomUniforms() { auto s = getTexture()->getContentSizeInPixels(); _blurRadius = 0; _pixelSize = Vector2(1/s.width, 1/s.height); _samplingRadius = 0; setBlurSize(3.0f); auto programState = getGLProgramState(); programState->getUniformValue("onePixelSize")->setValue(_pixelSize); programState->getUniformValue("gaussianCoefficient")->setValue(Vector4(_samplingRadius, _scale, _cons, _weightSum)); } void BlurSprite::setBlurSize(float f) { if(_blurRadius == (int)f) return; _blurRadius = (int)f; _samplingRadius = _blurRadius; if (_samplingRadius > 10) { _samplingRadius = 10; } if (_blurRadius > 0) { float sigma = _blurRadius / 2.0f; _scale = -0.5f / (sigma * sigma); _cons = -1.0f * _scale / 3.141592f; _weightSum = -_cons; float weight; int squareX; for(int dx = 0; dx <= _samplingRadius; ++dx) { squareX = dx * dx; weight = _cons * exp(squareX * _scale); _weightSum += 2.0 * weight; for (int dy = 1; dy <= _samplingRadius; ++dy) { weight = _cons * exp((squareX + dy * dy) * _scale); _weightSum += 4.0 * weight; } } } log("_blurRadius:%d",_blurRadius); } class NoiseSprite : public ShaderSprite, public ShaderSpriteCreator { public: CREATE_FUNC(NoiseSprite); NoiseSprite(); protected: virtual void setCustomUniforms() override; GLfloat _resolution[2]; }; NoiseSprite::NoiseSprite() { _fragSourceFile = "Shaders/example_Noisy.fsh"; } void NoiseSprite::setCustomUniforms() { _resolution[0] = getTexture()->getContentSizeInPixels().width; _resolution[1] = getTexture()->getContentSizeInPixels().height; getGLProgramState()->getUniformValue("resolution")->setValue(Vector2(_resolution[0],_resolution[1])); } class EdgeDetectionSprite : public ShaderSprite, public ShaderSpriteCreator { public: CREATE_FUNC(EdgeDetectionSprite); EdgeDetectionSprite(); protected: virtual void setCustomUniforms() override; GLfloat _resolution[2]; }; EdgeDetectionSprite::EdgeDetectionSprite() { _fragSourceFile = "Shaders/example_edgeDetection.fsh"; } void EdgeDetectionSprite::setCustomUniforms() { auto programState = getGLProgramState(); _resolution[0] = getContentSize().width; _resolution[1] = getContentSize().height; programState->getUniformValue("resolution")->setValue(Vector2(_resolution[0], _resolution[1])); } class BloomSprite : public ShaderSprite, public ShaderSpriteCreator { public: CREATE_FUNC(BloomSprite); BloomSprite(); protected: virtual void setCustomUniforms() override; GLfloat _resolution[2]; }; BloomSprite::BloomSprite() { _fragSourceFile = "Shaders/example_bloom.fsh"; } void BloomSprite::setCustomUniforms() { } class CelShadingSprite : public ShaderSprite, public ShaderSpriteCreator { public: CREATE_FUNC(CelShadingSprite); CelShadingSprite(); protected: virtual void setCustomUniforms() override; GLfloat _resolution[2]; }; CelShadingSprite::CelShadingSprite() { _fragSourceFile = "Shaders/example_celShading.fsh"; } void CelShadingSprite::setCustomUniforms() { auto programState = getGLProgramState(); _resolution[0] = getContentSize().width; _resolution[1] = getContentSize().height; programState->getUniformValue("resolution")->setValue(Vector2(_resolution[0], _resolution[1])); } class LensFlareSprite : public ShaderSprite, public ShaderSpriteCreator { public: CREATE_FUNC(LensFlareSprite); LensFlareSprite(); protected: virtual void setCustomUniforms() override; GLfloat _resolution[2]; GLfloat _textureResolution[2]; }; LensFlareSprite::LensFlareSprite() { _fragSourceFile = "Shaders/example_lensFlare.fsh"; } void LensFlareSprite::setCustomUniforms() { auto programState = getGLProgramState(); _textureResolution[0] = getTexture()->getContentSizeInPixels().width; _textureResolution[1] = getTexture()->getContentSizeInPixels().height; _resolution[0] = getContentSize().width; _resolution[1] = getContentSize().height; programState->getUniformValue("resolution")->setValue(Vector2(_resolution[0], _resolution[1])); programState->getUniformValue("textureResolution")->setValue(Vector2(_textureResolution[0], _textureResolution[1])); } NormalSpriteTest::NormalSpriteTest() { if (ShaderTestDemo2::init()) { auto s = Director::getInstance()->getWinSize(); NormalSprite* sprite = NormalSprite::createSprite("Images/powered.png"); sprite->setPosition(Vector2(s.width/2, s.height/2)); addChild(sprite); } } GreyScaleSpriteTest::GreyScaleSpriteTest() { if (ShaderTestDemo2::init()) { auto s = Director::getInstance()->getWinSize(); GreyScaleSprite* sprite = GreyScaleSprite::createSprite("Images/powered.png"); sprite->setPosition(Vector2(s.width * 0.75, s.height/2)); auto sprite2 = Sprite::create("Images/powered.png"); sprite2->setPosition(Vector2(s.width * 0.25, s.height/2)); addChild(sprite); addChild(sprite2); } } BlurSpriteTest::BlurSpriteTest() { if (ShaderTestDemo2::init()) { auto s = Director::getInstance()->getWinSize(); BlurSprite* sprite = BlurSprite::createSprite("Images/powered.png"); sprite->setPosition(Vector2(s.width * 0.75, s.height/2)); auto sprite2 = Sprite::create("Images/powered.png"); sprite2->setPosition(Vector2(s.width * 0.25, s.height/2)); addChild(sprite); addChild(sprite2); } } NoiseSpriteTest::NoiseSpriteTest() { if (ShaderTestDemo2::init()) { auto s = Director::getInstance()->getWinSize(); NoiseSprite* sprite = NoiseSprite::createSprite("Images/powered.png"); sprite->setPosition(Vector2(s.width * 0.75, s.height/2)); auto sprite2 = Sprite::create("Images/powered.png"); sprite2->setPosition(Vector2(s.width * 0.25, s.height/2)); addChild(sprite); addChild(sprite2); } } EdgeDetectionSpriteTest::EdgeDetectionSpriteTest() { if (ShaderTestDemo2::init()) { auto s = Director::getInstance()->getWinSize(); EdgeDetectionSprite* sprite = EdgeDetectionSprite::createSprite("Images/powered.png"); sprite->setPosition(Vector2(s.width * 0.75, s.height/2)); auto sprite2 = Sprite::create("Images/powered.png"); sprite2->setPosition(Vector2(s.width * 0.25, s.height/2)); addChild(sprite); addChild(sprite2); } } BloomSpriteTest::BloomSpriteTest() { if (ShaderTestDemo2::init()) { auto s = Director::getInstance()->getWinSize(); BloomSprite* sprite = BloomSprite::createSprite("Images/stone.png"); sprite->setPosition(Vector2(s.width * 0.75, s.height/2)); auto sprite2 = Sprite::create("Images/stone.png"); sprite2->setPosition(Vector2(s.width * 0.25, s.height/2)); addChild(sprite); addChild(sprite2); } } CelShadingSpriteTest::CelShadingSpriteTest() { if (ShaderTestDemo2::init()) { auto s = Director::getInstance()->getWinSize(); CelShadingSprite* sprite = CelShadingSprite::createSprite("Images/stone.png"); sprite->setPosition(Vector2(s.width * 0.75, s.height/2)); auto sprite2 = Sprite::create("Images/stone.png"); sprite2->setPosition(Vector2(s.width * 0.25, s.height/2)); addChild(sprite); addChild(sprite2); } } LensFlareSpriteTest::LensFlareSpriteTest() { if (ShaderTestDemo2::init()) { auto s = Director::getInstance()->getWinSize(); LensFlareSprite* sprite = LensFlareSprite::createSprite("Images/noise.png"); Rect rect = Rect::ZERO; rect.size = Size(480,320); sprite->setPosition(Vector2(s.width * 0.5, s.height/2)); addChild(sprite); } } class OutlineSprite : public ShaderSprite, public ShaderSpriteCreator { public: CREATE_FUNC(OutlineSprite); OutlineSprite(); protected: virtual void setCustomUniforms() override; }; OutlineSprite::OutlineSprite() { _fragSourceFile = "Shaders/example_outline.fsh"; _vertSourceFile = "Shaders/example_outline.vsh"; _blendFunc = BlendFunc::ALPHA_PREMULTIPLIED; } void OutlineSprite::setCustomUniforms() { Vector3 color(1.0, 0.2, 0.3); GLfloat radius = 0.01; GLfloat threshold = 1.75; auto programState = getGLProgramState(); programState->getUniformValue("u_outlineColor")->setValue(color); programState->getUniformValue("u_radius")->setValue(radius); programState->getUniformValue("u_threshold")->setValue(threshold); } OutlineShadingSpriteTest::OutlineShadingSpriteTest() { if (ShaderTestDemo2::init()) { auto s = Director::getInstance()->getWinSize(); OutlineSprite* sprite = OutlineSprite::createSprite("Images/grossini_dance_10.png"); sprite->setPosition(Vector2(s.width * 0.75, s.height/2)); auto sprite2 = Sprite::create("Images/grossini_dance_10.png"); sprite2->setPosition(Vector2(s.width * 0.25, s.height/2)); addChild(sprite); addChild(sprite2); } } class UniformSprite : public Sprite { public: UniformSprite(); ~UniformSprite(); CREATE_FUNC(UniformSprite); virtual void initShader(); // void setBackgroundNotification(); virtual void draw(Renderer *renderer, const Matrix &transform, bool transformUpdated) override; // void listenBackToForeground(Ref *obj); protected: virtual void setCustomUniforms(); std::string _fragSourceFile; std::string _vertSourceFile; CustomCommand _renderCommand; void onDraw(const Matrix &transform, bool transformUpdated); }; UniformSprite::UniformSprite() { _vertSourceFile = "Shaders/example_Heart.vsh"; _fragSourceFile = "Shaders/example_Heart.fsh"; } UniformSprite::~UniformSprite() { } void UniformSprite::initShader() { auto shader = GLProgram::createWithFilenames(_vertSourceFile, _fragSourceFile); this->setGLProgram(shader); } void UniformSprite::draw(Renderer *renderer, const Matrix &transform, bool transformUpdated) { _renderCommand.init(_globalZOrder); _renderCommand.func = CC_CALLBACK_0(UniformSprite::onDraw, this, transform, transformUpdated); renderer->addCommand(&_renderCommand); } void UniformSprite::setCustomUniforms() { auto glprogramstate = getGLProgramState(); glprogramstate->getUniformValue("center")->setValue(Vector2(480,320)); glprogramstate->getUniformValue("resolution")->setValue(Vector2(256,256)); } void UniformSprite::onDraw(const Matrix &transform, bool transformUpdated) { GL::blendFunc(_blendFunc.src, _blendFunc.dst); GL::bindTexture2D(_texture->getName()); float w = 256, h = 256; GLfloat vertices[12] = {0,0, w,0, w,h, 0,0, 0,h, w,h}; auto glprogramstate = getGLProgramState(); glprogramstate->setVertexAttribPointer("a_position", 2, GL_FLOAT, GL_FALSE, 0, vertices); glprogramstate->apply(transform); // Draw Call Test glDrawArrays(GL_TRIANGLES, 0, 6); CC_INCREMENT_GL_DRAWN_BATCHES_AND_VERTICES(1,6); CHECK_GL_ERROR_DEBUG(); } UniformShaderTest::UniformShaderTest() { if (ShaderTestDemo2::init()) { auto s = Director::getInstance()->getWinSize(); UniformSprite* sprite = UniformSprite::create(); setContentSize(Size(256, 256)); setAnchorPoint(Vector2(0.5f, 0.5f)); sprite->initShader(); sprite->setPosition(64,64); addChild(sprite); } } class AttribSprite : public Sprite { public: AttribSprite(); ~AttribSprite(); CREATE_FUNC(AttribSprite); virtual void initShader(); // void setBackgroundNotification(); virtual void draw(Renderer *renderer, const Matrix &transform, bool transformUpdated) override; // void listenBackToForeground(Ref *obj); protected: virtual void setCustomUniforms(); std::string _fragSourceFile; std::string _vertSourceFile; CustomCommand _renderCommand; void onDraw(const Matrix &transform, bool transformUpdated); }; AttribSprite::AttribSprite() { _fragSourceFile = "Shaders/example_normal.fsh"; _vertSourceFile = "Shaders/example_normal.vsh"; } AttribSprite::~AttribSprite() { } void AttribSprite::initShader() { auto glProgram = GLProgram::createWithFilenames(_vertSourceFile, _fragSourceFile); setGLProgram(glProgram); auto glProgramState = getGLProgramState(); #define kQuadSize sizeof(_quad.bl) size_t offset = (size_t)&_quad; // vertex int diff = offsetof( V3F_C4B_T2F, vertices); glProgramState->setVertexAttribPointer("a_position", 3, GL_FLOAT, GL_FALSE, kQuadSize, (void*) (offset + diff)); // texCoods diff = offsetof( V3F_C4B_T2F, texCoords); glProgramState->setVertexAttribPointer("a_texCoord", 2, GL_FLOAT, GL_FALSE, kQuadSize, (void*) (offset + diff)); // color diff = offsetof( V3F_C4B_T2F, colors); glProgramState->setVertexAttribPointer("a_color", 4, GL_UNSIGNED_BYTE, GL_TRUE, kQuadSize, (void*) (offset + diff)); //program->getUniformValue("u_diffuseColor")->setValue(Vector4(1,1,1,1)); } void AttribSprite::draw(Renderer *renderer, const Matrix &transform, bool transformUpdated) { _renderCommand.init(_globalZOrder); _renderCommand.func = CC_CALLBACK_0(AttribSprite::onDraw, this, transform, transformUpdated); renderer->addCommand(&_renderCommand); } void AttribSprite::setCustomUniforms() { } void AttribSprite::onDraw(const Matrix &transform, bool transformUpdated) { GL::blendFunc(_blendFunc.src, _blendFunc.dst); GL::bindTexture2D(_texture->getName()); getGLProgramState()->apply(transform); glDrawArrays(GL_TRIANGLE_STRIP, 0, 4); CC_INCREMENT_GL_DRAWN_BATCHES_AND_VERTICES(1, 4); CHECK_GL_ERROR_DEBUG(); } AttribShaderTest::AttribShaderTest() { if (ShaderTestDemo2::init()) { auto s = Director::getInstance()->getWinSize(); AttribSprite* sprite = AttribSprite::create(); sprite->setTexture("Images/powered.png"); sprite->initShader(); sprite->setPosition(Vector2(s.width/2, s.height/2)); addChild(sprite); } } UniformAttribShaderTest::UniformAttribShaderTest() { }