Cocos2d-x使用CCGLProgram和Shader文件实现精灵置灰

文章目录
  1. 1. 1. 实现方法
    1. 1.1. 方法和属性声明
  2. 2. 2. shader字符串
  3. 3. 3. 核心代码
  4. 4. 4. lua使用示例

1. 实现方法

CCGraySprite 继承自 CCSprite 覆盖initWithTexturedraw方法实现置灰。

方法和属性声明
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
class CCGraySprite : public CCSprite{
public:
CCGraySprite();
virtual ~CCGraySprite();
static CCGraySprite* create(const char* pszFileName);
bool initWithTexture(CCTexture2D* pTexture, const CCRect& tRect);
static CCGraySprite* createWithSprite(CCSprite *pSprite);
static CCGraySprite* createWithNodeAndItChild(CCNode *pNode);
void setGray(bool isGray);
virtual void draw();

private:
bool m_isGray;
CCGLProgram* pProgram;
};
  • m_isGray 是否置灰
  • pProgram 保存置灰的GL程序

2. shader字符串

可以像官方那样写在一个.h文件中

1
2
3
4
5
6
7
8
9
10
11
12
13
14
GLchar* pszFragSource = (GLchar*)
"#ifdef GL_ES \n \
precision mediump float; \n \
#endif \n \
uniform sampler2D u_texture; \n \
varying vec2 v_texCoord; \n \
varying vec4 v_fragmentColor; \n \
void main(void) \n \
{ \n \
// Convert to greyscale using NTSC weightings \n \
vec4 col = texture2D(u_texture, v_texCoord); \n \
float grey = dot(col.rgb, vec3(0.299, 0.587, 0.114)); \n \
gl_FragColor = vec4(grey, grey, grey, col.a); \n \
}";

3. 核心代码

  • 覆盖父类的initWithTexturedraw方法,根据m_isGray调用对应GL程序进行渲染绘制精灵
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
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
void CCGraySprite::setGray(bool isGray) {
m_isGray = isGray;
if (isGray == true) {
this->setShaderProgram(pProgram);
}else
{
this->setShaderProgram(CCShaderCache::sharedShaderCache()->programForKey(kCCShader_PositionTextureColor));
}
}

bool CCGraySprite::initWithTexture(CCTexture2D* pTexture, const CCRect& tRect ){
m_isGray = true;
do{
CC_BREAK_IF(!CCSprite::initWithTexture(pTexture, tRect));

GLchar* pszFragSource = (GLchar*)
"#ifdef GL_ES \n \
precision mediump float; \n \
#endif \n \
uniform sampler2D u_texture; \n \
varying vec2 v_texCoord; \n \
varying vec4 v_fragmentColor; \n \
void main(void) \n \
{ \n \
// Convert to greyscale using NTSC weightings \n \
vec4 col = texture2D(u_texture, v_texCoord); \n \
float grey = dot(col.rgb, vec3(0.299, 0.587, 0.114)); \n \
gl_FragColor = vec4(grey, grey, grey, col.a); \n \
}";



pProgram = new CCGLProgram();
pProgram->initWithVertexShaderByteArray(ccPositionTextureColor_vert, pszFragSource);
this->setShaderProgram(pProgram);
CHECK_GL_ERROR_DEBUG();

this->getShaderProgram()->addAttribute(kCCAttributeNamePosition, kCCVertexAttrib_Position);
this->getShaderProgram()->addAttribute(kCCAttributeNameColor, kCCVertexAttrib_Color);
this->getShaderProgram()->addAttribute(kCCAttributeNameTexCoord, kCCVertexAttrib_TexCoords);
CHECK_GL_ERROR_DEBUG();

this->getShaderProgram()->link();
CHECK_GL_ERROR_DEBUG();

this->getShaderProgram()->updateUniforms();
CHECK_GL_ERROR_DEBUG();

return true;
} while (0);
return false;
}

void CCGraySprite::draw(){
if (this->getTexture() == NULL) {
return;
}
if(m_isGray == false){
CCSprite::draw();
return;
}

ccGLEnableVertexAttribs(kCCVertexAttribFlag_PosColorTex );
ccGLBlendFunc( m_sBlendFunc.src, m_sBlendFunc.dst );

this->getShaderProgram()->use();
this->getShaderProgram()->setUniformsForBuiltins();

ccGLBindTexture2D( this->getTexture()->getName() );

#define kQuadSize sizeof(m_sQuad.bl)
long offset = (long)&m_sQuad;

// vertex
int diff = offsetof( ccV3F_C4B_T2F, vertices);
glVertexAttribPointer(kCCVertexAttrib_Position, 3, GL_FLOAT, GL_FALSE, kQuadSize, (void*) (offset + diff));

// texCoods
diff = offsetof( ccV3F_C4B_T2F, texCoords);
glVertexAttribPointer(kCCVertexAttrib_TexCoords, 2, GL_FLOAT, GL_FALSE, kQuadSize, (void*)(offset + diff));

// color
diff = offsetof( ccV3F_C4B_T2F, colors);
glVertexAttribPointer(kCCVertexAttrib_Color, 4, GL_UNSIGNED_BYTE, GL_TRUE, kQuadSize, (void*)(offset + diff));
glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);
CC_INCREMENT_GL_DRAWS(1);
}

4. lua使用示例

  • 需手写.pkg文件,使用tolua++工具导出Lua binding文件,在AppDelegate::applicationDidFinishLaunching()注册后,方可在lua层使用
1
local nameBgSp = CCGraySprite:create("images/namebg.png")
1
2
local normalSp = CCSprite:create("images/namebg.png")
local graySp = CCGraySprite:createWithNodeAndItChild(normalSp)
1
2
local frameSp = CCGraySprite:create("images/frame.png")
frameSp:setGray(isGray)

扫描二维码或在微信中搜索 KeepMovingXin
欢迎关注微信公众号!


版权声明
KeepMoving by KP_小新 采用 创作共用保留署名-非商业-禁止演绎4.0国际许可证
Copyright © 2018 KeepMoving. All rights reserved.