深入分析Cocos2d-x 中的“纹理”

本博为什么要花这么多时间在纹理上呢?因为本博一次再一次的看到很多的Cocos2d开发者在说:“为什么我的游戏占内存这么大?”。“为什么我的游戏跑起来效率这么低?”。其实这里面最根本的问题是你思想上没有对资源进行优化的一种流程在里面。在进行游戏开发的过程中,你被美术牵着鼻子跑,你没有自然而然的想到该怎么进行图片的格式,尺寸,拼合,复用方式,以及绘制时的方式。这些因素对游戏产生的具体影响在你的脑子里还没有留下完整的概念。所以,我希望经过对于这些源码的分析,让大家去了解这些因素,从而学会思考如何优化自已的游戏。好了,大家先将今天关于纹理代码的分析博文学习完,下一篇我讲对于这些优化因素做一个具体的讲解。


           当一张图片被加载到内存后,它是以纹理的形式存在的。纹理是什么东西呢?纹理就是一块内存,这块内存中存放的是按照指定的像素格式填充的图片像素信息。它被最终作为三角面着色所依据的数据源。


           我们来看一下cocos2d-x中的libcocos2d库,其下有许多目录,找到textures展开,可以看到有CCTexture2D,CCTextureAtlas,CCTextureCache,CCTexturePVR四个类。


这四个类的功能分别是:

   CCTexture2D: 纹理,即图片加载入内存后供CPU和GPU操作的贴图对象。

  CCTexturePVR:处理PVR文件生成纹理的类,提示: 大家可以用它解析愤怒的小鸟中的图片。

  CCTextureCache:纹理管理器,负责加载图片并对生成的纹理进行管理。通过“字典”来进行快速的查询。

  CCTextureAtlas:纹理块管理器,如果图片是由多个小图块组成的,则纹理块管理器用来存储这些小图块的相关信息,以方便绘制相应图块。


      为了让大家更好的学习纹理,在讲解纹理的代码之前我已经先给大家分析了本章用到的两个功能类:

CCImageCCDictionary。这两个类分别在纹理模块中担任加载图片和管理纹理指针的作用。希望大家先顶一下这两篇贴子之后再开始下面的代码学习,你一定会感到非常容易。

一.CCTexture2D:

好,咱们现在开始看CCTexture2D:

[cpp]  view plain copy
  1. #ifndef __CCTEXTURE2D_H__  
  2. #define __CCTEXTURE2D_H__  
  3.   
  4. #include   
  5. #include "cocoa/CCObject.h"  
  6. #include "cocoa/CCGeometry.h"  
  7. #include "ccTypes.h"  
  8.   
  9. //Cocos2d命名空间  
  10. NS_CC_BEGIN  
  11. //需要用到CCImage,这里声明一下。  
  12. class CCImage;  
  13.   
  14. //纹理格式:即每个纹理中的像素单位分别是怎么为颜色值进行实际内存分配的。这个非常重要,我们在进行游戏开发的过程中,会常常与各种图片类型打交通。每种图片往往也有各自的像素格式。但当它们一旦加载到游戏中后,就会根据我们的要求变成以下某种类型的纹理。不同的纹理格式所占据的内存大小可能不同,我们要根据实际情况和需求来选择相应的纹理格式。比如我们用RGBA8888纹理格式来创建纹理,它占据的内存容量很大,如果我们要显示的纹理中没有ALPHA值,那就不应该使用带ALPHA通道的纹理格式。我们就可以改成RGB565像素格式。  
  15.   
  16. typedef enum {  
  17.   
  18.     //32位真彩色,最真但最耗内存  
  19.     kCCTexture2DPixelFormat_RGBA8888,  
  20.     //24位真彩色,去掉了ALPHA通道  
  21.     kCCTexture2DPixelFormat_RGB888,  
  22.     //16位色,将RGB压缩在一个字中。绿色多了1位,因为人眼对绿色更敏感。  
  23.     kCCTexture2DPixelFormat_RGB565,  
  24.     //8位色,只存ALPHA值,做遮罩图用  
  25.     kCCTexture2DPixelFormat_A8,  
  26.     //8位色,只存灰度或者强度值,做灰度图用  
  27.     kCCTexture2DPixelFormat_I8,  
  28.     //16位色,只存ALPHA值与强度值,双功能  
  29.     kCCTexture2DPixelFormat_AI88,  
  30.     //16位色,RGBA四通道各占4位。  
  31.     kCCTexture2DPixelFormat_RGBA4444,  
  32.     //16位色,RGB三通道各占5位,多1位留做ALPHA镂空使用  
  33.     kCCTexture2DPixelFormat_RGB5A1,      
  34.     // PVR的PVRTC4压缩格式  
  35.     kCCTexture2DPixelFormat_PVRTC4,  
  36.     // PVRTC的PVRTC2压缩格式  
  37.     kCCTexture2DPixelFormat_PVRTC2,  
  38.   
  39.     //默认格式RGBA8888  
  40.     kCCTexture2DPixelFormat_Default = kCCTexture2DPixelFormat_RGBA8888,  
  41.   
  42.     // 为了兼容性而保留的枚举值  
  43.     kTexture2DPixelFormat_RGBA8888 = kCCTexture2DPixelFormat_RGBA8888,  
  44.     kTexture2DPixelFormat_RGB888 = kCCTexture2DPixelFormat_RGB888,  
  45.     kTexture2DPixelFormat_RGB565 = kCCTexture2DPixelFormat_RGB565,  
  46.     kTexture2DPixelFormat_A8 = kCCTexture2DPixelFormat_A8,  
  47.     kTexture2DPixelFormat_RGBA4444 = kCCTexture2DPixelFormat_RGBA4444,  
  48.     kTexture2DPixelFormat_RGB5A1 = kCCTexture2DPixelFormat_RGB5A1,  
  49.     kTexture2DPixelFormat_Default = kCCTexture2DPixelFormat_Default  
  50.   
  51. } CCTexture2DPixelFormat;  
  52.   
  53. //需要使用Shader代码片段,这里声明一下  
  54. class CCGLProgram;  
  55.   
  56. //定义了纹理的一些参数  
  57. typedef struct _ccTexParams {  
  58.     GLuint    minFilter;//纹理过滤器:缩小过滤器  
  59.     GLuint    magFilter;//纹理过滤器:放大过滤器  
  60.     GLuint    wrapS;//横向纹理寻址模式  
  61.     GLuint    wrapT;//纵向纹理寻址模式  
  62. } ccTexParams;  
  63.   
  64.   
  65. // CCTexture2D类可以方便的从图片,文本或raw数据文件中创建OpenGL所用贴图,创建的贴图会自动转为2的幂次方大小,所以要注意对于贴图坐标的影响。  
  66.   
  67. class CC_DLL CCTexture2D : public CCObject  
  68. {  
  69. public:  
  70.     //构造  
  71. CCTexture2D();  
  72. //析构  
  73.     virtual ~CCTexture2D();  
  74.     //取得纹理的描述  
  75.     const char* description(void);  
  76.   
  77.     //释放数据  
  78. void releaseData(void *data);  
  79.  //保存数据  
  80.     void* keepData(void *data, unsigned int length);  
  81.   
  82.     //由数据指针和指定的像素格式,图片宽高,来生成OpenGL贴图。  
  83.     bool initWithData(const void* data, CCTexture2DPixelFormat pixelFormat, unsigned int pixelsWide, unsigned int pixelsHigh, const CCSize& contentSize);  
  84.   
  85. //在指定的位置绘制贴图  
  86. void drawAtPoint(const CCPoint& point);  
  87.     //纹制贴图上的一个图像块  
  88. void drawInRect(const CCRect& rect);  
  89.   
  90.     //由CCImage指针生成OpenGL贴图  
  91.     bool initWithImage(CCImage * uiImage);  
  92.   
  93.     //由一个字符串生成OpenGL贴图。  
  94.     bool initWithString(const char *text, const CCSize& dimensions, CCTextAlignment hAlignment, CCVerticalTextAlignment vAlignment, const char *fontName, float fontSize);  
  95.     //由一个字符串和指定的字体与大小生成OpenGL贴图  
  96.     bool initWithString(const char *text, const char *fontName, float fontSize);  
  97.   
  98. //如果支持PVR的压缩格式  
  99. #ifdef CC_SUPPORT_PVRTC      
  100.     //由一个PVR压缩格式的数据生成OpenGL贴图  
  101.     bool initWithPVRTCData(const void *data, int level, int bpp, bool hasAlpha, int length, CCTexture2DPixelFormat pixelFormat);  
  102. #endif // CC_SUPPORT_PVRTC  
  103.       
  104.     //从普通PVR文件生成OpenGL贴图  
  105.     bool initWithPVRFile(const char* file);  
  106.   
  107.    //设置贴图参数  
  108.     void setTexParameters(ccTexParams* texParams);  
  109.   
  110.     //设置为抗锯齿的贴图过滤方式(线性过滤)  
  111.     void setAntiAliasTexParameters();  
  112.   
  113.     //设置为非抗锯齿的贴图过滤方式(最近点采样)  
  114.     void setAliasTexParameters();  
  115.   
  116.   
  117.     //生成多级贴图: 由图片数据生成一系列尺寸为2的幂次方直至当前贴图大小的贴图。系统会根据距离自动选择纹理图片。可以解决大图片显示在小空间时的闪烁问题。  
  118.     void generateMipmap();  
  119.   
  120.     //取得像素格式名称  
  121.     const char* stringForFormat();  
  122.   
  123.     //返回当前贴图色深,即每个像素占多少位  
  124.     unsigned int bitsPerPixelForFormat();    
  125.   
  126. //通过参数贴图格式返回纹理色深  
  127. unsigned int bitsPerPixelForFormat(CCTexture2DPixelFormat format);  
  128.   
  129.   
  130.     //静态函数,用于设置默认带ALPHA通道的贴图像素格式。则图片创建为贴图时,如果有ALPHA通道,则生成此默认贴图像素格式。  
  131.     static void setDefaultAlphaPixelFormat(CCTexture2DPixelFormat format);  
  132.   
  133.     //静态函数,取得默认带ALPHA通道的贴图像素格式。  
  134.     static CCTexture2DPixelFormat defaultAlphaPixelFormat();  
  135.   
  136.     //静态函数,设置载入PVR时是否开启ALPHA渐变,默认不开启,则ALPHA值只有是与否,无渐变。  
  137.     static void PVRImagesHavePremultipliedAlpha(bool haveAlphaPremultiplied);  
  138.   
  139.     //取得图片大小(以像素为单位)  
  140.     const CCSize& getContentSizeInPixels();  
  141.     //是否有ALPHA渐变值  
  142. bool hasPremultipliedAlpha();  
  143. //是否有多级贴图  
  144.     bool hasMipmaps();  
  145. private:  
  146.     //加载一个带ALPHA渐变的图片生成OpenGL贴图  
  147.     bool initPremultipliedATextureWithImage(CCImage * image, unsigned int pixelsWide, unsigned int pixelsHigh);  
  148.       
  149.     //ALPHA渐变开关  
  150.     bool m_bPVRHaveAlphaPremultiplied;  
  151.   
  152.     //贴图格式变量及get接口  
  153.     CC_PROPERTY_READONLY(CCTexture2DPixelFormat, m_ePixelFormat, PixelFormat)  
  154.     //贴图宽度及get接口  
  155.     CC_PROPERTY_READONLY(unsigned int, m_uPixelsWide, PixelsWide)  
  156.     //贴图高度及get接口  
  157.     CC_PROPERTY_READONLY(unsigned int, m_uPixelsHigh, PixelsHigh)  
  158.   
  159.     //OpenGL贴图索引及get接口  
  160.     CC_PROPERTY_READONLY(GLuint, m_uName, Name)  
  161.   
  162.     //横向贴图坐标终点。因为图片如果不是2的幂次方,图片大小会小于贴图的大小,贴图一定是2的幂次方嘛,这时候横向的贴图坐标终点不是1.0。  
  163.     CC_PROPERTY(GLfloat, m_fMaxS, MaxS)  
  164.     //纵向贴图坐标终点。  
  165.     CC_PROPERTY(GLfloat, m_fMaxT, MaxT)  
  166.     //图片大小及get接口  
  167.     CC_PROPERTY_READONLY(CCSize, m_tContentSize, ContentSize)  
  168.   
  169.     // ALPHA渐变开关  
  170.     bool m_bHasPremultipliedAlpha;  
  171.     // 多级纹理开关  
  172.     bool m_bHasMipmaps;  
  173.   
  174.     //Shader代码片段指针  
  175.     CC_PROPERTY(CCGLProgram*, m_pShaderProgram, ShaderProgram);  
  176. };  
  177.   
  178. NS_CC_END  
  179.   
  180. #endif //__CCTEXTURE2D_H__  

再来看CCTexture2D.cpp:


[cpp]  view plain copy
  1. #include "CCTexture2D.h"  
  2. #include "ccConfig.h"  
  3. #include "ccMacros.h"  
  4. #include "CCConfiguration.h"  
  5. #include "platform/platform.h"  
  6. #include "platform/CCImage.h"  
  7. #include "CCGL.h"  
  8. #include "support/ccUtils.h"  
  9. #include "platform/CCPlatformMacros.h"  
  10. #include "textures/CCTexturePVR.h"  
  11. #include "CCDirector.h"  
  12. #include "shaders/CCGLProgram.h"  
  13. #include "shaders/ccGLStateCache.h"  
  14. #include "shaders/CCShaderCache.h"  
  15. //这里定义是否使用可变纹理  
  16. #if CC_ENABLE_CACHE_TEXTURE_DATA  
  17.     #include "CCTextureCache.h"  
  18. #endif  
  19. //Cocos2d-x命名空间  
  20. NS_CC_BEGIN  
  21.   
  22.   
  23. //静态全局的默认贴图像素格式。缺省为kCCTexture2DPixelFormat_Default,即RGBA8888。  
  24. static CCTexture2DPixelFormat g_defaultAlphaPixelFormat = kCCTexture2DPixelFormat_Default;  
  25.   
  26. //静态全局的PVR是否有ALPHA渐变的开关变量,默认为否。  
  27. static bool PVRHaveAlphaPremultiplied_ = false;  
  28.   
  29. //构造函数。  
  30. CCTexture2D::CCTexture2D()  
  31. : m_uPixelsWide(0)  
  32. , m_uPixelsHigh(0)  
  33. , m_uName(0)  
  34. , m_fMaxS(0.0)  
  35. , m_fMaxT(0.0)  
  36. , m_bHasPremultipliedAlpha(false)  
  37. , m_bHasMipmaps(false)  
  38. , m_bPVRHaveAlphaPremultiplied(true)  
  39. , m_pShaderProgram(NULL)  
  40. {  
  41. }  
  42. //析构  
  43. CCTexture2D::~CCTexture2D()  
  44. {  
  45. //如果使用可变纹理,删除此可变纹理中的数据。  
  46. #if CC_ENABLE_CACHE_TEXTURE_DATA  
  47.     VolatileTexture::removeTexture(this);  
  48. #endif  
  49. //打印日志。  
  50. CCLOGINFO("cocos2d: deallocing CCTexture2D %u.", m_uName);  
  51. //释放所用到的Shader代码片段  
  52.     CC_SAFE_RELEASE(m_pShaderProgram);  
  53. //释放OpenGL所用到的贴图。  
  54.     if(m_uName)  
  55.     {  
  56.         ccGLDeleteTexture(m_uName);  
  57.     }  
  58. }  
  59. //取得当前纹理的贴图像素格式。  
  60. CCTexture2DPixelFormat CCTexture2D::getPixelFormat()  
  61. {  
  62.     return m_ePixelFormat;  
  63. }  
  64. //取得贴图宽度。  
  65. unsigned int CCTexture2D::getPixelsWide()  
  66. {  
  67.     return m_uPixelsWide;  
  68. }  
  69. //取得贴图高度。  
  70. unsigned int CCTexture2D::getPixelsHigh()  
  71. {  
  72.     return m_uPixelsHigh;  
  73. }  
  74. //取得贴图索引。  
  75. GLuint CCTexture2D::getName()  
  76. {  
  77.     return m_uName;  
  78. }  
  79. //取得图片大小(以点为单位)  
  80. CCSize CCTexture2D::getContentSize()  
  81. {  
  82. // CC_CONTENT_SCALE_FACTOR宏返回的是在不同屏幕下的点与像素的比率。Mac电脑上返回1.而使用Retina显示屏的iphone上返回2。  
  83.     CCSize ret;  
  84.     ret.width = m_tContentSize.width / CC_CONTENT_SCALE_FACTOR();  
  85.     ret.height = m_tContentSize.height / CC_CONTENT_SCALE_FACTOR();  
  86.       
  87.     return ret;  
  88. }  
  89. //取得图片大小(以像素为单位)  
  90. const CCSize& CCTexture2D::getContentSizeInPixels()  
  91. {  
  92.     return m_tContentSize;  
  93. }  
  94. //取得横向的贴图坐标终点  
  95. GLfloat CCTexture2D::getMaxS()  
  96. {  
  97.     return m_fMaxS;  
  98. }  
  99. //设置横向的贴图坐标终点  
  100. void CCTexture2D::setMaxS(GLfloat maxS)  
  101. {  
  102.     m_fMaxS = maxS;  
  103. }  
  104. //取得纵向的贴图坐标终点  
  105. GLfloat CCTexture2D::getMaxT()  
  106. {  
  107.     return m_fMaxT;  
  108. }  
  109. //设置纵向的贴图坐标终点  
  110. void CCTexture2D::setMaxT(GLfloat maxT)  
  111. {  
  112.     m_fMaxT = maxT;  
  113. }  
  114. //所用到的Shader代码片段。  
  115. CCGLProgram* CCTexture2D::getShaderProgram(void)  
  116. {  
  117.     return m_pShaderProgram;  
  118. }  
  119. //设置用到的Shader代码片段。  
  120. void CCTexture2D::setShaderProgram(CCGLProgram* pShaderProgram)  
  121. {  
  122.     CC_SAFE_RETAIN(pShaderProgram);  
  123.     CC_SAFE_RELEASE(m_pShaderProgram);  
  124.     m_pShaderProgram = pShaderProgram;  
  125. }  
  126. //释放数据  
  127. void CCTexture2D::releaseData(void *data)  
  128. {  
  129.     free(data);  
  130. }  
  131. //保存数据  
  132. void* CCTexture2D::keepData(void *data, unsigned int length)  
  133. {  
  134. //这里只是使用CC_UNUSED_PARAM宏用一下length,没什么实际功能。作者给出这个函数是预备未来供子类重载。  
  135.     CC_UNUSED_PARAM(length);  
  136.     return data;  
  137. }  
  138. //是否有ALPHA渐变的通道数据。  
  139. bool CCTexture2D::hasPremultipliedAlpha()  
  140. {  
  141.     return m_bHasPremultipliedAlpha;  
  142. }  
  143. //由数据指针创建指定大小和格式的贴图。取得创建成功后图片在贴图中的实际区域 。  
  144. bool CCTexture2D::initWithData(const void *data, CCTexture2DPixelFormat pixelFormat, unsigned int pixelsWide, unsigned int pixelsHigh, const CCSize& contentSize)  
  145. {  
  146. //如果是RGBA8888格式或者大小正好就是2的幂次方。像素数据按四字节(DWORD)对齐。否则按1字节(BYTE)进行对齐。  
  147.     if( pixelFormat == kCCTexture2DPixelFormat_RGBA8888 || ( ccNextPOT(pixelsWide)==pixelsWide && ccNextPOT(pixelsHigh)==pixelsHigh) )  
  148.     {  
  149.         glPixelStorei(GL_UNPACK_ALIGNMENT,4);  
  150.     }  
  151.     else  
  152.     {  
  153.         glPixelStorei(GL_UNPACK_ALIGNMENT,1);  
  154.     }  
  155. //产生一个OpenGL的贴图索引。  
  156. glGenTextures(1, &m_uName);  
  157. //将此贴图绑定为GL_TEXTURE_2D纹理。  
  158.     ccGLBindTexture2D(m_uName);  
  159. //设置OpenGL中的贴图的过滤参数。  
  160.     glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR );  
  161. glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR );  
  162. //设置贴图的横向纹理寻址模式为边缘截取模式。总是忽略边界。  
  163. glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE );  
  164. //设置贴图的纵向纹理寻址模式为边缘截取模式。总是忽略边界。  
  165.     glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE );  
  166.   
  167. //这里根据不同的像素格式来生成不同的OpenGL所用的贴图。注意:传入的宽和高在成功生成贴图后会返回实际贴图的宽和高。如果图片不是2的幂次方,这个数值会改成2的幂次方。比如你传入的图片宽高是148x245,则调用完成后宽高会转成256x256。  
  168.     switch(pixelFormat)  
  169.     {  
  170.     case kCCTexture2DPixelFormat_RGBA8888:  
  171.         glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, (GLsizei)pixelsWide, (GLsizei)pixelsHigh, 0, GL_RGBA, GL_UNSIGNED_BYTE, data);  
  172.         break;  
  173.     case kCCTexture2DPixelFormat_RGB888:  
  174.         glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, (GLsizei)pixelsWide, (GLsizei)pixelsHigh, 0, GL_RGB, GL_UNSIGNED_BYTE, data);  
  175.         break;  
  176.     case kCCTexture2DPixelFormat_RGBA4444:  
  177.         glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, (GLsizei)pixelsWide, (GLsizei)pixelsHigh, 0, GL_RGBA, GL_UNSIGNED_SHORT_4_4_4_4, data);  
  178.         break;  
  179.     case kCCTexture2DPixelFormat_RGB5A1:  
  180.         glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, (GLsizei)pixelsWide, (GLsizei)pixelsHigh, 0, GL_RGBA, GL_UNSIGNED_SHORT_5_5_5_1, data);  
  181.         break;  
  182.     case kCCTexture2DPixelFormat_RGB565:  
  183.         glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, (GLsizei)pixelsWide, (GLsizei)pixelsHigh, 0, GL_RGB, GL_UNSIGNED_SHORT_5_6_5, data);  
  184.         break;  
  185.     case kCCTexture2DPixelFormat_AI88:  
  186.         glTexImage2D(GL_TEXTURE_2D, 0, GL_LUMINANCE_ALPHA, (GLsizei)pixelsWide, (GLsizei)pixelsHigh, 0, GL_LUMINANCE_ALPHA, GL_UNSIGNED_BYTE, data);  
  187.         break;  
  188.     case kCCTexture2DPixelFormat_A8:  
  189.         glTexImage2D(GL_TEXTURE_2D, 0, GL_ALPHA, (GLsizei)pixelsWide, (GLsizei)pixelsHigh, 0, GL_ALPHA, GL_UNSIGNED_BYTE, data);  
  190.         break;  
  191.     case kCCTexture2DPixelFormat_I8:  
  192.         glTexImage2D(GL_TEXTURE_2D, 0, GL_LUMINANCE, (GLsizei)pixelsWide, (GLsizei)pixelsHigh, 0, GL_LUMINANCE, GL_UNSIGNED_BYTE, data);  
  193.         break;  
  194.     default:  
  195.         CCAssert(0, "NSInternalInconsistencyException");  
  196.   
  197.     }  
  198. //图片大小  
  199.     m_tContentSize = contentSize;  
  200. //保存实际的贴图宽高  
  201. m_uPixelsWide = pixelsWide;  
  202. m_uPixelsHigh = pixelsHigh;  
  203. //保存贴图的像素格式  
  204. m_ePixelFormat = pixelFormat;  
  205. //计算图片处于贴图中的横向和纵向的纹理坐标终点。  
  206.     m_fMaxS = contentSize.width / (float)(pixelsWide);  
  207.     m_fMaxT = contentSize.height / (float)(pixelsHigh);  
  208. //默认不使用ALPHA渐变通道。  
  209. m_bHasPremultipliedAlpha = false;  
  210. //默认不使用多级纹理。  
  211.     m_bHasMipmaps = false;  
  212.   
  213.     //设置使用kCCShader_PositionTexture对应类型的Shader。此Shader的顶点格式由位置和纹理坐标组成。  
  214. setShaderProgram(CCShaderCache::sharedShaderCache()->programForKey(kCCShader_PositionTexture));  
  215.   
  216.     return true;  
  217. }  
  218.   
  219. //取得纹理的描述  
  220. const char* CCTexture2D::description(void)  
  221. {  
  222.     return CCString::createWithFormat("", m_uName, m_uPixelsWide, m_uPixelsHigh, m_fMaxS, m_fMaxT)->getCString();  
  223. }  
  224.   
  225. // 由CCImage对象来初始化纹理生成OpenGL贴图。  
  226. bool CCTexture2D::initWithImage(CCImage *uiImage)  
  227. {  
  228.     //参数有效性判断。  
  229.     if (uiImage == NULL)  
  230.     {  
  231.         CCLOG("cocos2d: CCTexture2D. Can't create Texture. UIImage is nil");  
  232.         this->release();  
  233.         return false;  
  234.     }  
  235.     //取得图片的宽高  
  236.     unsigned int imageWidth = uiImage->getWidth();  
  237.     unsigned int imageHeight = uiImage->getHeight();  
  238.     //取得引擎的配置信息  
  239.     CCConfiguration *conf = CCConfiguration::sharedConfiguration();  
  240.     //取得配置信息中指定的最大纹理大小  
  241. unsigned maxTextureSize = conf->getMaxTextureSize();  
  242. //如果当前图片大于指定的最大纹理大小,提示错误警告交释放当前纹理返回NULL。  
  243.     if (imageWidth > maxTextureSize || imageHeight > maxTextureSize)   
  244.     {  
  245.         CCLOG("cocos2d: WARNING: Image (%u x %u) is bigger than the supported %u x %u", imageWidth, imageHeight, maxTextureSize, maxTextureSize);  
  246.         this->release();  
  247.         return NULL;  
  248.     }  
  249.       
  250.     //总是按加载ALPHA渐变的图片方式来生成OpenGL贴图  
  251.     return initPremultipliedATextureWithImage(uiImage, imageWidth, imageHeight);  
  252. }  
  253. //加载一个带ALPHA渐变的图片生成OpenGL贴图  
  254. bool CCTexture2D::initPremultipliedATextureWithImage(CCImage *image, unsigned int width, unsigned int height)  
  255. {  
  256. //取得图片的相关信息  
  257. //定义指针变量指向图片像素数据。  
  258.     unsigned char*            tempData = image->getData();  
  259.     //定义无符号int指针变量,也是为了指向32位色深的图片像素数据,以便使指针直接对应指定的一个像素数据位置。  
  260.     unsigned int*             inPixel32 = NULL;  
  261.     //定义无符号char指针变量,也是为了指向8位色深的图片像素数据,以便使指针直接对应指定的一个像素数据位置。  
  262.     unsigned char*            inPixel8 = NULL;  
  263.     //定义无符号short指针变量,指向16位色深的贴图像素数据,以便使指针直接对应指定的一个像素数据位置。  
  264.     unsigned short*           outPixel16 = NULL;  
  265.     //定义bool变量hasAlpha取得图片是否有Alpha通道。  
  266.     bool                      hasAlpha = image->hasAlpha();  
  267.     //定义变量imageSize保存图片大小。  
  268.     CCSize                    imageSize = CCSizeMake((float)(image->getWidth()), (float)(image->getHeight()));  
  269.     //定义变量pixelFormat用来保存贴图的像素格式。  
  270.     CCTexture2DPixelFormat    pixelFormat;  
  271.     //定义变量bpp保存图片的色深。  
  272.     size_t                    bpp = image->getBitsPerComponent();  
  273.   
  274. // 如果有ALPHA通道,使用默认的RGBA8888格式。  
  275. if(hasAlpha)  
  276.     {  
  277.         pixelFormat = g_defaultAlphaPixelFormat;  
  278.     }  
  279.     else  
  280.     {    //如果没有ALPHA通道  
  281.          //如果色深大于等于8,则转为RGB888格式,否则转为RGB565格式。这里有点问题,感觉应该按色深大于16来进行判断。即24和32位都转为RGB888,而16位及以下转为RGB565。  
  282.         if (bpp >= 8)  
  283.         {  
  284.             pixelFormat = kCCTexture2DPixelFormat_RGB888;  
  285.         }  
  286.         else   
  287.         {  
  288.             pixelFormat = kCCTexture2DPixelFormat_RGB565;  
  289.         }  
  290.           
  291.     }  
  292.       
  293.     // 取得数据的长度  
  294.     unsigned int length = width * height;  
  295.     //根据图片的不同格式和要创建的纹理格式,将数据填充到纹理中。  
  296.     if (pixelFormat == kCCTexture2DPixelFormat_RGB565)  
  297. {  
  298.  //根据是否有ALPHA通道来分别进行填充处理  
  299.         if (hasAlpha)  
  300.         {  
  301.              // 转换RGBA8888到RGB565  
  302.             // Convert "RRRRRRRRRGGGGGGGGBBBBBBBBAAAAAAAA" to "RRRRRGGGGGGBBBBB"  
  303.             //一个像素占2个字节,所以图像中所有像素占用的字节数为width*height*2。由此大小申请内存作为贴图的像素数据。  
  304.             tempData = new unsigned char[width * height * 2];  
  305.              //将贴图像素数据的地址返回给unsigned short指针。这样outPixel16就指向了贴图中的第一个像素的数据位置。  
  306.             outPixel16 = (unsigned short*)tempData;  
  307.              //因为有alpha,则图片是32位RGBA8888格式。取得图像的像素数据地址返回给unsigned int指针,则inPixel32指向了贴图中第一个像素的数据位置。  
  308.             inPixel32 = (unsigned int*)image->getData();  
  309.             //遍历图片中所有的像素,逐像素处理。  
  310.             for(unsigned int i = 0; i < length; ++i, ++inPixel32)  
  311.             {   //将inPixel32指向的unsigned int数据通过取出R,G,B各8位数据值,然后组成RGB565值。放入outPixel16指向的unsigned short数据中。  
  312.                 *outPixel16++ =   
  313.                 ((((*inPixel32 >>  0) & 0xFF) >> 3) << 11) |  // R  
  314.                 ((((*inPixel32 >>  8) & 0xFF) >> 2) << 5)  |  // G  
  315.                 ((((*inPixel32 >> 16) & 0xFF) >> 3) << 0);    // B  
  316.             }  
  317.         }  
  318.         else   
  319.         {    // 转换RGB888到RGB565  
  320.             // Convert "RRRRRRRRRGGGGGGGGBBBBBBBB" to "RRRRRGGGGGGBBBBB"  
  321.             //一个像素占2个字节,所以图像中所有像素占用的字节数为width*height*2。由此大小申请内存作为贴图的像素数据。  
  322.             tempData = new unsigned char[width * height * 2];  
  323.             //将贴图像素数据的地址返回给unsigned short指针。这样outPixel16就指向了贴图中的第一个像素的数据位置。  
  324.             outPixel16 = (unsigned short*)tempData;  
  325.             //如果图像的格式为RGB888。取得图像的像素数据地址返回给unsigned char指针,则inPixel8指向了贴图中第一个像素的R值位置。  
  326.             inPixel8 = (unsigned char*)image->getData();  
  327.             //遍历图片中所有的像素,逐像素处理。  
  328.             for(unsigned int i = 0; i < length; ++i)  
  329.             {   //inPixel8指向的是unsigned char值,通过++操作来取出R,G,B数据值,然后组成RGB565值。放入outPixel16指向的unsigned short数据中。  
  330.   
  331.                 *outPixel16++ =   
  332.                 (((*inPixel8++ & 0xFF) >> 3) << 11) |  // R  
  333.                 (((*inPixel8++ & 0xFF) >> 2) << 5)  |  // G  
  334.                 (((*inPixel8++ & 0xFF) >> 3) << 0);    // B  
  335.             }  
  336.         }      
  337.     }  
  338.     else if (pixelFormat == kCCTexture2DPixelFormat_RGBA4444)  
  339.     {   // 转换RGBA8888到RGBA4444  
  340.         // Convert "RRRRRRRRRGGGGGGGGBBBBBBBBAAAAAAAA" to "RRRRGGGGBBBBAAAA"  
  341.         //取得图像的像素数据地址返回给unsigned int指针,则inPixel32指向了贴图中第一个像素的数据位置。  
  342.         inPixel32 = (unsigned int*)image->getData();    
  343.          //一个像素占2个字节,所以图像中所有像素占用的字节数为width*height*2。由此大小申请内存作为贴图的像素数据。  
  344.         tempData = new unsigned char[width * height * 2];  
  345.         //将贴图像素数据的地址返回给unsigned short指针。这样outPixel16就指向了贴图中的第一个像素的数据位置。  
  346.         outPixel16 = (unsigned short*)tempData;  
  347.         //遍历图片中所有的像素,逐像素处理。  
  348.         for(unsigned int i = 0; i < length; ++i, ++inPixel32)  
  349.         {   //将inPixel32指向的unsigned int数据通过取出R,G,B,A各8位数据值,然后组成RGBA4444值。放入outPixel16指向的unsigned short数据中。  
  350.             *outPixel16++ =   
  351.             ((((*inPixel32 >> 0) & 0xFF) >> 4) << 12) | // R  
  352.             ((((*inPixel32 >> 8) & 0xFF) >> 4) <<  8) | // G  
  353.             ((((*inPixel32 >> 16) & 0xFF) >> 4) << 4) | // B  
  354.             ((((*inPixel32 >> 24) & 0xFF) >> 4) << 0);  // A  
  355.         }  
  356.     }  
  357.     else if (pixelFormat == kCCTexture2DPixelFormat_RGB5A1)  
  358.     {   // 转换RGBA8888到RGBA5551  
  359.         // Convert "RRRRRRRRRGGGGGGGGBBBBBBBBAAAAAAAA" to "RRRRRGGGGGBBBBBA"  
  360.         inPixel32 = (unsigned int*)image->getData();     
  361.          //一个像素占2个字节,所以图像中所有像素占用的字节数为width*height*2。由此大小申请内存作为贴图的像素数据。  
  362.         tempData = new unsigned char[width * height * 2];  
  363.         //将贴图像素数据的地址返回给unsigned short指针。这样outPixel16就指向了贴图中的第一个像素的数据位置。  
  364.         outPixel16 = (unsigned short*)tempData;  
  365.         //遍历图片中所有的像素,逐像素处理。  
  366.         for(unsigned int i = 0; i < length; ++i, ++inPixel32)  
  367.         {   //将inPixel32指向的unsigned int数据通过取出R,G,B,A各8位数据值,然后组成RGB5A1值。放入outPixel16指向的unsigned short数据中。  
  368.             *outPixel16++ =   
  369.             ((((*inPixel32 >> 0) & 0xFF) >> 3) << 11) | // R  
  370.             ((((*inPixel32 >> 8) & 0xFF) >> 3) <<  6) | // G  
  371.             ((((*inPixel32 >> 16) & 0xFF) >> 3) << 1) | // B  
  372.             ((((*inPixel32 >> 24) & 0xFF) >> 7) << 0);  // A  
  373.         }  
  374.     }  
  375.     else if (pixelFormat == kCCTexture2DPixelFormat_A8)  
  376.     {   // 转换RGBA8888到A8,同理,不再赘述  
  377.         // Convert "RRRRRRRRRGGGGGGGGBBBBBBBBAAAAAAAA" to "AAAAAAAA"  
  378.         inPixel32 = (unsigned int*)image->getData();  
  379.         tempData = new unsigned char[width * height];  
  380.         unsigned char *outPixel8 = tempData;  
  381.           
  382.         for(unsigned int i = 0; i < length; ++i, ++inPixel32)  
  383.         {  
  384.             *outPixel8++ = (*inPixel32 >> 24) & 0xFF;  // A  
  385.         }  
  386.     }  
  387.       
  388.     if (hasAlpha && pixelFormat == kCCTexture2DPixelFormat_RGB888)  
  389.     {   // 转换RGBA8888到RGB888,同理,不再赘述  
  390.         // Convert "RRRRRRRRRGGGGGGGGBBBBBBBBAAAAAAAA" to "RRRRRRRRGGGGGGGGBBBBBBBB"  
  391.         inPixel32 = (unsigned int*)image->getData();  
  392.         tempData = new unsigned char[width * height * 3];  
  393.         unsigned char *outPixel8 = tempData;  
  394.           
  395.         for(unsigned int i = 0; i < length; ++i, ++inPixel32)  
  396.         {  
  397.             *outPixel8++ = (*inPixel32 >> 0) & 0xFF; // R  
  398.             *outPixel8++ = (*inPixel32 >> 8) & 0xFF; // G  
  399.             *outPixel8++ = (*inPixel32 >> 16) & 0xFF; // B  
  400.         }  
  401.     }  
  402.     //因为最终相应像素格式的数据都存放在tempData中,所以这里通过像素数据来生成OpenGL贴图。  
  403.     initWithData(tempData, pixelFormat, width, height, imageSize);  
  404.     //如果是以上相应格式,则tempData都是新申请的内存块,则在这里释放申请的内存。  
  405.     if (tempData != image->getData())  
  406.     {  
  407.         delete [] tempData;  
  408.     }  
  409. //取得是否有ALPHA渐变通道数据  
  410.     m_bHasPremultipliedAlpha = image->isPremultipliedAlpha();  
  411.     return true;  
  412. }  
  413.   
  414. // 从字符串中创建OpenGL 贴图。  
  415. //参1:字符串  
  416. //参2:字体名称  
  417. //参3:字体大小  
  418. bool CCTexture2D::initWithString(const char *text, const char *fontName, float fontSize)  
  419. {  
  420.     return initWithString(text, CCSizeMake(0,0), kCCTextAlignmentCenter, kCCVerticalTextAlignmentTop, fontName, fontSize);  
  421. }  
  422. // 从字符串中创建OpenGL 贴图,可指定更多参数。  
  423. //参1:字符串  
  424. //参2:返回参数,代表在屏幕上占用的区域大小  
  425. //参3:文字的横向对齐方式  
  426. //参4:文字的纵向对齐方式  
  427. //参5:字体名称  
  428. //参6:字体大小  
  429. bool CCTexture2D::initWithString(const char *text, const CCSize& dimensions, CCTextAlignment hAlignment, CCVerticalTextAlignment vAlignment, const char *fontName, float fontSize)  
  430. {  
  431.     //如果定义使用可变纹理  
  432. #if CC_ENABLE_CACHE_TEXTURE_DATA  
  433.     // cache the texture data  
  434.     VolatileTexture::addStringTexture(this, text, dimensions, hAlignment, vAlignment, fontName, fontSize);  
  435. #endif  
  436.     //定义一个CCImage实例对象  
  437.     CCImage image;  
  438.     //定义一个CCImage进行由字符串创建图片时指定的文字对齐方式的变量eAlign  
  439.     CCImage::ETextAlign eAlign;  
  440.     //如果文字纵向对齐方式为顶部对齐。  
  441.     if (kCCVerticalTextAlignmentTop == vAlignment)  
  442.     {      
  443.         //根据文字横向对齐方式的不同分别对eAlign进行设置。  
  444.         eAlign = (kCCTextAlignmentCenter == hAlignment) ? CCImage::kAlignTop  
  445.             : (kCCTextAlignmentLeft == hAlignment) ? CCImage::kAlignTopLeft : CCImage::kAlignTopRight;  
  446.     }  
  447.     //如果文字纵向对齐方式为居中对齐。  
  448.     else if (kCCVerticalTextAlignmentCenter == vAlignment)  
  449.     {  
  450.         //根据文字横向对齐方式的不同分别对eAlign进行设置。  
  451.         eAlign = (kCCTextAlignmentCenter == hAlignment) ? CCImage::kAlignCenter  
  452.             : (kCCTextAlignmentLeft == hAlignment) ? CCImage::kAlignLeft : CCImage::kAlignRight;  
  453.     }  
  454.     //如果文字纵向对齐方式为底部对齐。  
  455.     else if (kCCVerticalTextAlignmentBottom == vAlignment)  
  456.     {  
  457.         //根据文字横向对齐方式的不同分别对eAlign进行设置。  
  458.         eAlign = (kCCTextAlignmentCenter == hAlignment) ? CCImage::kAlignBottom  
  459.             : (kCCTextAlignmentLeft == hAlignment) ? CCImage::kAlignBottomLeft : CCImage::kAlignBottomRight;  
  460.     }  
  461.     else  
  462.     {  
  463.          //其它对齐方式不应存在,故打印错误。  
  464.         CCAssert(false"Not supported alignment format!");  
  465.     }  
  466.     //调用CCImage的成员函数由字符串创建出图片数据。  
  467.     if (!image.initWithString(text, (int)dimensions.width, (int)dimensions.height, eAlign, fontName, (int)fontSize))  
  468.     {  
  469.         return false;  
  470.     }  
  471.     //再由CCImage实例对象来创建出OpenGL贴图,初始化纹理。  
  472.     return initWithImage(&image);  
  473. }  
  474.   
  475.   
  476. // 在指定的位置绘制贴图。  
  477. void CCTexture2D::drawAtPoint(const CCPoint& point)  
  478. {   //定义贴图中图像区域的UV坐标。从左上至右下。  
  479.     GLfloat    coordinates[] = {      
  480.         0.0f,    m_fMaxT,  
  481.         m_fMaxS,m_fMaxT,  
  482.         0.0f,    0.0f,  
  483.         m_fMaxS,0.0f };  
  484.     //取得贴图中图像区域的宽高  
  485.     GLfloat    width = (GLfloat)m_uPixelsWide * m_fMaxS,  
  486.         height = (GLfloat)m_uPixelsHigh * m_fMaxT;  
  487.     //定义对应的顶点坐标  
  488.     GLfloat        vertices[] = {      
  489.         point.x,            point.y,  
  490.         width + point.x,    point.y,  
  491.         point.x,            height  + point.y,  
  492.         width + point.x,    height  + point.y };  
  493.     //Shader中使用位置和纹理坐标通道。  
  494.     ccGLEnableVertexAttribs( kCCVertexAttribFlag_Position | kCCVertexAttribFlag_TexCoords );  
  495.     //后面的代码便用Shader进行渲染  
  496.     m_pShaderProgram->use();  
  497.     //设置Shader使用的最终结果矩阵  
  498.     m_pShaderProgram->setUniformForModelViewProjectionMatrix();  
  499.     //将贴图绑定  
  500.     ccGLBindTexture2D( m_uName );  
  501.   
  502.     //将vertices设置为顶点位置参数  
  503.     glVertexAttribPointer(kCCVertexAttrib_Position, 2, GL_FLOAT, GL_FALSE, 0, vertices);  
  504.     //将coordinates设置为顶点的纹理坐标参数  
  505.     glVertexAttribPointer(kCCVertexAttrib_TexCoords, 2, GL_FLOAT, GL_FALSE, 0, coordinates);  
  506.     //绘制三角形,参1为绘图方式,参2为顶点起始索引,参3为三角形面数。  
  507.     glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);  
  508. }  
  509. //绘制纹理上的一个区域。  
  510. void CCTexture2D::drawInRect(const CCRect& rect)  
  511. {   //定义贴图中图像区域的UV坐标。从左上至右下。  
  512.     GLfloat    coordinates[] = {      
  513.         0.0f,    m_fMaxT,  
  514.         m_fMaxS,m_fMaxT,  
  515.         0.0f,    0.0f,  
  516.         m_fMaxS,0.0f };  
  517.     //绘制到的区域  
  518.     GLfloat    vertices[] = {    rect.origin.x,        rect.origin.y,                            /*0.0f,*/  
  519.         rect.origin.x + rect.size.width,        rect.origin.y,                            /*0.0f,*/  
  520.         rect.origin.x,                            rect.origin.y + rect.size.height,        /*0.0f,*/  
  521.         rect.origin.x + rect.size.width,        rect.origin.y + rect.size.height,        /*0.0f*/ };  
  522.     //Shader中使用位置和纹理坐标通道。  
  523.     ccGLEnableVertexAttribs( kCCVertexAttribFlag_Position | kCCVertexAttribFlag_TexCoords );  
  524.     //后面的代码便用Shader进行渲染  
  525.     m_pShaderProgram->use();  
  526.     //设置Shader使用的最终结果矩阵  
  527.     m_pShaderProgram->setUniformForModelViewProjectionMatrix();  
  528.     //将贴图绑定  
  529.     ccGLBindTexture2D( m_uName );  
  530.     //将vertices设置为顶点位置参数  
  531.     glVertexAttribPointer(kCCVertexAttrib_Position, 2, GL_FLOAT, GL_FALSE, 0, vertices);  
  532.     //将coordinates设置为顶点的纹理坐标参数  
  533.     glVertexAttribPointer(kCCVertexAttrib_TexCoords, 2, GL_FLOAT, GL_FALSE, 0, coordinates);  
  534.     //绘制三角形,参1为绘图方式,参2为顶点起始索引,参3为三角形面数。  
  535.     glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);  
  536. }  
  537.   
  538. #ifdef CC_SUPPORT_PVRTC  
  539.   
  540. // 如果支持PVR文件的压缩格式。提供的读取PVR压缩文件的函数。     
  541. bool CCTexture2D::initWithPVRTCData(const void *data, int level, int bpp, bool hasAlpha, int length, CCTexture2DPixelFormat pixelFormat)  
  542. {  
  543.     if( !(CCConfiguration::sharedConfiguration()->supportsPVRTC()) )  
  544.     {  
  545.         CCLOG("cocos2d: WARNING: PVRTC images is not supported.");  
  546.         this->release();  
  547.         return false;  
  548.     }  
  549. //产生一个OpenGL的贴图索引。  
  550. glGenTextures(1, &m_uName);  
  551.     //绑定纹理  
  552.     glBindTexture(GL_TEXTURE_2D, m_uName);  
  553.     //设置纹理抗锯齿  
  554.     this->setAntiAliasTexParameters();  
  555.     //贴图格式  
  556.     GLenum format;  
  557.     //数据大小  
  558.     GLsizei size = length * length * bpp / 8;  
  559.     //根据是否有Alpha来取得贴图格式  
  560.     if(hasAlpha) {  
  561.         format = (bpp == 4) ? GL_COMPRESSED_RGBA_PVRTC_4BPPV1_IMG : GL_COMPRESSED_RGBA_PVRTC_2BPPV1_IMG;  
  562.     } else {  
  563.         format = (bpp == 4) ? GL_COMPRESSED_RGB_PVRTC_4BPPV1_IMG : GL_COMPRESSED_RGB_PVRTC_2BPPV1_IMG;  
  564.     }  
  565.     if(size < 32) {  
  566.         size = 32;  
  567.     }  
  568.     //加载压缩纹理。  
  569.     glCompressedTexImage2D(GL_TEXTURE_2D, level, format, length, length, 0, size, data);  
  570.     //设置其它属性。  
  571.     m_tContentSize = CCSizeMake((float)(length), (float)(length));  
  572.     m_uPixelsWide = length;  
  573.     m_uPixelsHigh = length;  
  574.     m_fMaxS = 1.0f;  
  575.     m_fMaxT = 1.0f;  
  576.     m_bHasPremultipliedAlpha = PVRHaveAlphaPremultiplied_;  
  577.     m_ePixelFormat = pixelFormat;  
  578.   
  579.     return true;  
  580. }  
  581. #endif // CC_SUPPORT_PVRTC  
  582. //加载PVR普通文件的函数。  
  583. bool CCTexture2D::initWithPVRFile(const char* file)  
  584. {  
  585.     bool bRet = false;  
  586.     // nothing to do with CCObject::init  
  587.       
  588.     CCTexturePVR *pvr = new CCTexturePVR;  
  589.     bRet = pvr->initWithContentsOfFile(file);  
  590.           
  591.     if (bRet)  
  592.     {  
  593.         pvr->setRetainName(true); // don't dealloc texture on release  
  594.           
  595.         m_uName = pvr->getName();  
  596.         m_fMaxS = 1.0f;  
  597.         m_fMaxT = 1.0f;  
  598.         m_uPixelsWide = pvr->getWidth();  
  599.         m_uPixelsHigh = pvr->getHeight();  
  600.         m_tContentSize = CCSizeMake((float)m_uPixelsWide, (float)m_uPixelsHigh);  
  601.         m_bHasPremultipliedAlpha = PVRHaveAlphaPremultiplied_;  
  602.         m_ePixelFormat = pvr->getFormat();  
  603.         m_bHasMipmaps = pvr->getNumberOfMipmaps() > 1;         
  604.   
  605.         pvr->release();  
  606.     }  
  607.     else  
  608.     {  
  609.         CCLOG("cocos2d: Couldn't load PVR image %s", file);  
  610.     }  
  611.   
  612.     return bRet;  
  613. }  
  614. //设置PVR文件加载时是否使用ALPHA渐变。  
  615. void CCTexture2D::PVRImagesHavePremultipliedAlpha(bool haveAlphaPremultiplied)  
  616. {  
  617.     PVRHaveAlphaPremultiplied_ = haveAlphaPremultiplied;  
  618. }  
  619.   
  620.       
  621. //生成多级纹理。  
  622. void CCTexture2D::generateMipmap()  
  623. {  
  624.     CCAssert( m_uPixelsWide == ccNextPOT(m_uPixelsWide) && m_uPixelsHigh == ccNextPOT(m_uPixelsHigh), "Mimpap texture only works in POT textures");  
  625.     ccGLBindTexture2D( m_uName );  
  626.     glGenerateMipmap(GL_TEXTURE_2D);  
  627.     m_bHasMipmaps = true;  
  628. }  
  629. //是否有多级纹理  
  630. bool CCTexture2D::hasMipmaps()  
  631. {  
  632.     return m_bHasMipmaps;  
  633. }  
  634. //设置纹理参数。  
  635. void CCTexture2D::setTexParameters(ccTexParams *texParams)  
  636. {  
  637.     CCAssert( (m_uPixelsWide == ccNextPOT(m_uPixelsWide) || texParams->wrapS == GL_CLAMP_TO_EDGE) &&  
  638.         (m_uPixelsHigh == ccNextPOT(m_uPixelsHigh) || texParams->wrapT == GL_CLAMP_TO_EDGE),  
  639.         "GL_CLAMP_TO_EDGE should be used in NPOT dimensions");  
  640.   
  641.     ccGLBindTexture2D( m_uName );  
  642.     glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, texParams->minFilter );  
  643.     glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, texParams->magFilter );  
  644.     glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, texParams->wrapS );  
  645.     glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, texParams->wrapT );  
  646. }  
  647. //设置为非抗锯齿纹理模式  
  648. void CCTexture2D::setAliasTexParameters()  
  649. {   //绑定纹理  
  650.     ccGLBindTexture2D( m_uName );  
  651.     //设置多级纹理  
  652.     if( ! m_bHasMipmaps )  
  653.     {     
  654.         //设置最小滤波方式为最近点采样,这种方式最快,但有锯齿。  
  655.         glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST );  
  656.     }  
  657.     else  
  658.     {    //设置最小滤波方式为多级纹理方式。  
  659.         glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST_MIPMAP_NEAREST );  
  660.     }  
  661.     //设置放大滤波方式为最近点采样  
  662.     glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST );  
  663. }  
  664. //设置为抗锯齿纹理模式  
  665. void CCTexture2D::setAntiAliasTexParameters()  
  666. {   //绑定纹理  
  667.     ccGLBindTexture2D( m_uName );  
  668.     //设置多级纹理  
  669.     if( ! m_bHasMipmaps )  
  670.     {  
  671.         //设置最小滤波方式为线性过滤,这种方式纹理会有一定程度模糊。  
  672.         glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR );  
  673.     }  
  674.     else  
  675.     {   //设置最小滤波方式为多级纹理方式。  
  676.         glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_NEAREST );  
  677.     }  
  678.     //设置放大滤波方式为最近点采样  
  679.     glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR );  
  680. }  
  681. //取得纹理格式的名字字符串  
  682. const char* CCTexture2D::stringForFormat()  
  683. {  
  684.     switch (m_ePixelFormat)   
  685.     {  
  686.         case kCCTexture2DPixelFormat_RGBA8888:  
  687.             return  "RGBA8888";  
  688.   
  689.         case kCCTexture2DPixelFormat_RGB888:  
  690.             return  "RGB888";  
  691.   
  692.         case kCCTexture2DPixelFormat_RGB565:  
  693.             return  "RGB565";  
  694.   
  695.         case kCCTexture2DPixelFormat_RGBA4444:  
  696.             return  "RGBA4444";  
  697.   
  698.         case kCCTexture2DPixelFormat_RGB5A1:  
  699.             return  "RGB5A1";  
  700.   
  701.         case kCCTexture2DPixelFormat_AI88:  
  702.             return  "AI88";  
  703.   
  704.         case kCCTexture2DPixelFormat_A8:  
  705.             return  "A8";  
  706.   
  707.         case kCCTexture2DPixelFormat_I8:  
  708.             return  "I8";  
  709.   
  710.         case kCCTexture2DPixelFormat_PVRTC4:  
  711.             return  "PVRTC4";  
  712.   
  713.         case kCCTexture2DPixelFormat_PVRTC2:  
  714.             return  "PVRTC2";  
  715.   
  716.         default:  
  717.             CCAssert(false , "unrecognised pixel format");  
  718.             CCLOG("stringForFormat: %ld, cannot give useful result", (long)m_ePixelFormat);  
  719.             break;  
  720.     }  
  721.   
  722.     return  NULL;  
  723. }  
  724.   
  725.   
  726. //设置默认带ALPHA通道的纹理格式  
  727. void CCTexture2D::setDefaultAlphaPixelFormat(CCTexture2DPixelFormat format)  
  728. {  
  729.     g_defaultAlphaPixelFormat = format;  
  730. }  
  731.   
  732. //取得默认带ALPHA通道的纹理格式  
  733. CCTexture2DPixelFormat CCTexture2D::defaultAlphaPixelFormat()  
  734. {  
  735.     return g_defaultAlphaPixelFormat;  
  736. }  
  737. //取得相应纹理格式的色深。  
  738. unsigned int CCTexture2D::bitsPerPixelForFormat(CCTexture2DPixelFormat format)  
  739. {  
  740.     unsigned int ret=0;  
  741.   
  742.     switch (format) {  
  743.         case kCCTexture2DPixelFormat_RGBA8888:  
  744.             ret = 32;//32位真彩色  
  745.             break;  
  746.         case kCCTexture2DPixelFormat_RGB888:  
  747.             // 看起来是用24位,但内部实际是用DWORD来存数据的,所以还是32位。只不过Alpha通道没用。  
  748.             ret = 32;  
  749.             break;  
  750.         case kCCTexture2DPixelFormat_RGB565:  
  751.             ret = 16;  
  752.             break;  
  753.         case kCCTexture2DPixelFormat_RGBA4444:  
  754.             ret = 16;  
  755.             break;  
  756.         case kCCTexture2DPixelFormat_RGB5A1:  
  757.             ret = 16;  
  758.             break;  
  759.         case kCCTexture2DPixelFormat_AI88:  
  760.             ret = 16;  
  761.             break;  
  762.         case kCCTexture2DPixelFormat_A8:  
  763.             ret = 8;  
  764.             break;  
  765.         case kCCTexture2DPixelFormat_I8:  
  766.             ret = 8;  
  767.             break;  
  768.         case kCCTexture2DPixelFormat_PVRTC4:  
  769.             ret = 4;  
  770.             break;  
  771.         case kCCTexture2DPixelFormat_PVRTC2:  
  772.             ret = 2;  
  773.             break;  
  774.         default:  
  775.             ret = -1;  
  776.             CCAssert(false , "unrecognised pixel format");  
  777.             CCLOG("bitsPerPixelForFormat: %ld, cannot give useful result", (long)format);  
  778.             break;  
  779.     }  
  780.     return ret;  
  781. }  
  782. //取得当前纹理格式的色深。  
  783. unsigned int CCTexture2D::bitsPerPixelForFormat()  
  784. {  
  785.     return this->bitsPerPixelForFormat(m_ePixelFormat);  
  786. }  
  787.   
  788.   
  789. NS_CC_END  

二.CCTexturePVR:

CCTexturePVR.h:

[cpp]  view plain copy
  1. #ifndef __CCPVRTEXTURE_H__  
  2. #define __CCPVRTEXTURE_H__  
  3.   
  4. #include "CCStdC.h"  
  5. #include "CCGL.h"  
  6. #include "cocoa/CCObject.h"  
  7. #include "cocoa/CCArray.h"  
  8. //Cocos2d命名空间  
  9. NS_CC_BEGIN  
  10.   
  11. //用于多级纹理的结构  
  12. struct CCPVRMipmap {  
  13.     unsigned char *address;  
  14.     unsigned int len;  
  15. };  
  16.   
  17. //最大的多级纹理级数  
  18. enum {  
  19.     CC_PVRMIPMAP_MAX = 16,  
  20. };  
  21.   
  22.   
  23. //所支持的 PVR 图片格式:  
  24.     - RGBA8888  
  25.     - BGRA8888  
  26.     - RGBA4444  
  27.     - RGBA5551  
  28.     - RGB565  
  29.     - A8  
  30.     - I8  
  31.     - AI88  
  32.     - PVRTC 4BPP  
  33.     - PVRTC 2BPP  
  34.   
  35. class CCTexturePVR : public CCObject  
  36. {  
  37. public:  
  38.     //构造  
  39.     CCTexturePVR();  
  40.     //析构  
  41.     virtual ~CCTexturePVR();  
  42.     //成员函数:载入一个PVR图片文件  
  43.     bool initWithContentsOfFile(const char* path);  
  44.   
  45.     //导出的静态函数:载入一个PVR图片文件。此函数为可供js调用。  
  46.     CC_DEPRECATED_ATTRIB一个UTE static CCTexturePVR* pvrTextureWithContentsOfFile(const char* path);  
  47.     //静态函数:载入一个PVR图片文件  
  48.     static CCTexturePVR* create(const char* path);  
  49.       
  50.     //相关属性的获取  
  51.     //取得贴图索引  
  52.     inline unsigned int getName() { return m_uName; }  
  53.     //取得  
  54.     inline unsigned int getWidth() { return m_uWidth; }  
  55.     inline unsigned int getHeight() { return m_uHeight; }  
  56.     //取得是否有ALPHA通道  
  57.     inline bool hasAlpha() { return m_bHasAlpha; }  
  58.     //取得多级纹理的级数  
  59.     inline unsigned int getNumberOfMipmaps() { return m_uNumberOfMipmaps; }  
  60.     //取得图片格式  
  61.     inline CCTexture2DPixelFormat getFormat() { return m_eFormat; }  
  62.     //取得是否是供Retain显示屏使用的高清图片 。  
  63.     inline bool isRetainName() { return m_bRetainName; }  
  64.     //设置为供Retain显示屏使用的高清图片。  
  65.     inline void setRetainName(bool retainName) { m_bRetainName = retainName; }  
  66.   
  67. private:  
  68.     //解压PVR图片数据  
  69.     bool unpackPVRData(unsigned char* data, unsigned int len);  
  70.     //创建OpenGL纹理  
  71.     bool createGLTexture();  
  72.       
  73. protected:  
  74.     //多级纹理的各级信息结构。  
  75.     struct CCPVRMipmap m_asMipmaps[CC_PVRMIPMAP_MAX];   // pointer to mipmap images      
  76.     //多级纹理的最大级别数  
  77.     unsigned int m_uNumberOfMipmaps;                    // number of mipmap used  
  78.     //  
  79.     unsigned int m_uTableFormatIndex;  
  80.     unsigned int m_uWidth, m_uHeight;  
  81.     //贴图索引  
  82.     GLuint m_uName;  
  83.     //是否有Alpha通道  
  84.     bool m_bHasAlpha;  
  85.     //是否是供Retain显示屏使用的高清图片 。  
  86.     bool m_bRetainName;  
  87.     //图片格式  
  88.     CCTexture2DPixelFormat m_eFormat;  
  89. };  
  90.   
  91. NS_CC_END  
  92.   
  93.   
  94. #endif //__CCPVRTEXTURE_H__  

CCTexturePVR.cpp:

[cpp]  view plain copy
  1. #include "CCTexture2D.h"  
  2. #include "CCTexturePVR.h"  
  3. #include "ccMacros.h"  
  4. #include "CCConfiguration.h"  
  5. #include "support/ccUtils.h"  
  6. #include "CCStdC.h"  
  7. #include "platform/CCFileUtils.h"  
  8. #include "support/zip_support/ZipUtils.h"  
  9. #include "shaders/ccGLStateCache.h"  
  10. #include   
  11. #include   
  12. //Cocos2d命名空间  
  13. NS_CC_BEGIN  
  14.   
  15. #define PVR_TEXTURE_FLAG_TYPE_MASK    0xff  
  16.   
  17. //PVR文件的信息标志位的各bit位意义  
  18. enum {  
  19.     kPVRTextureFlagMipmap         = (1<<8),        // 有多级纹理   
  20.     kPVRTextureFlagTwiddle        = (1<<9),        // is twiddled  
  21.     kPVRTextureFlagBumpmap        = (1<<10),       // 是法线贴图(用于产生凹凸感)  
  22.     kPVRTextureFlagTiling         = (1<<11),       // is bordered for tiled pvr  
  23.     kPVRTextureFlagCubemap        = (1<<12),       // 是立方体环境映射贴图(一般用于做天空盒)  
  24.     kPVRTextureFlagFalseMipCol    = (1<<13),       // are there false coloured MIP levels  
  25.     kPVRTextureFlagVolume         = (1<<14),       // 立体纹理,相当于有多层的纹理。  
  26.     kPVRTextureFlagAlpha          = (1<<15),       // v2.1 is there transparency info in the texture  
  27.     kPVRTextureFlagVerticalFlip   = (1<<16),       // v2.1 is the texture vertically flipped  
  28. };  
  29. //PVR文件头标识  
  30. static char gPVRTexIdentifier[5] = "PVR!";  
  31. //所有PVR文件的格式  
  32. enum  
  33. {  
  34.     kPVRTexturePixelTypeRGBA_4444= 0x10,  
  35.     kPVRTexturePixelTypeRGBA_5551,  
  36.     kPVRTexturePixelTypeRGBA_8888,  
  37.     kPVRTexturePixelTypeRGB_565,  
  38.     kPVRTexturePixelTypeRGB_555,                // 这个Cocos2d-x暂不支持  
  39.     kPVRTexturePixelTypeRGB_888,  
  40.     kPVRTexturePixelTypeI_8,  
  41.     kPVRTexturePixelTypeAI_88,  
  42.     kPVRTexturePixelTypePVRTC_2,  
  43.     kPVRTexturePixelTypePVRTC_4,  
  44.     kPVRTexturePixelTypeBGRA_8888,  
  45.     kPVRTexturePixelTypeA_8,  
  46. };  
  47. //信息数组  
  48. static const unsigned int tableFormats[][7] = {  
  49.     //数组元素的结构为:  
  50.     // 1- PVR 文件格式  
  51.     // 2- OpenGL 内部格式  
  52.     // 3- OpenGL 格式  
  53.     // 4- OpenGL 数据类型  
  54.     // 5- 色深  
  55.     // 6- 是否压缩  
  56.     // 7- Cocos2d 像素格式  
  57.     { kPVRTexturePixelTypeRGBA_4444, GL_RGBA,   GL_RGBA, GL_UNSIGNED_SHORT_4_4_4_4,         16, false, kCCTexture2DPixelFormat_RGBA4444 },  
  58.     { kPVRTexturePixelTypeRGBA_5551, GL_RGBA,   GL_RGBA, GL_UNSIGNED_SHORT_5_5_5_1,         16, false, kCCTexture2DPixelFormat_RGB5A1   },  
  59.     { kPVRTexturePixelTypeRGBA_8888, GL_RGBA,   GL_RGBA, GL_UNSIGNED_BYTE,                  32, false, kCCTexture2DPixelFormat_RGBA8888 },  
  60.     { kPVRTexturePixelTypeRGB_565,   GL_RGB,    GL_RGB,  GL_UNSIGNED_SHORT_5_6_5,           16, false, kCCTexture2DPixelFormat_RGB565   },  
  61.     { kPVRTexturePixelTypeRGB_888,   GL_RGB,    GL_RGB,  GL_UNSIGNED_BYTE,                  24, false,  kCCTexture2DPixelFormat_RGB888  },  
  62.     { kPVRTexturePixelTypeA_8,       GL_ALPHA,  GL_ALPHA,   GL_UNSIGNED_BYTE,               8,  false, kCCTexture2DPixelFormat_A8       },  
  63.     { kPVRTexturePixelTypeI_8,       GL_LUMINANCE,  GL_LUMINANCE,   GL_UNSIGNED_BYTE,       8,  false, kCCTexture2DPixelFormat_I8       },  
  64.     { kPVRTexturePixelTypeAI_88,     GL_LUMINANCE_ALPHA,    GL_LUMINANCE_ALPHA, GL_UNSIGNED_BYTE,16,    false, kCCTexture2DPixelFormat_AI88 },  
  65. #if (CC_TARGET_PLATFORM == CC_PLATFORM_IOS)    
  66.     //如果程序运行在IOS上,还可支持以下信息  
  67.     { kPVRTexturePixelTypePVRTC_2,   GL_COMPRESSED_RGBA_PVRTC_2BPPV1_IMG, (unsigned int)-1, (unsigned int)-1,           2,  true, kCCTexture2DPixelFormat_PVRTC2    },  
  68.     { kPVRTexturePixelTypePVRTC_4,   GL_COMPRESSED_RGBA_PVRTC_4BPPV1_IMG, (unsigned int)-1, (unsigned int)-1,           4,  true, kCCTexture2DPixelFormat_PVRTC4    },  
  69.   
  70.     { kPVRTexturePixelTypeBGRA_8888, GL_RGBA,   GL_BGRA, GL_UNSIGNED_BYTE,                  32, false, kCCTexture2DPixelFormat_RGBA8888 },  
  71. #endif   
  72. };  
  73.   
  74. //信息表的元素数量  
  75. #define MAX_TABLE_ELEMENTS (sizeof(tableFormats) / sizeof(tableFormats[0]))  
  76. //  
  77. enum {  
  78.     kCCInternalPVRTextureFormat,  
  79.     kCCInternalOpenGLInternalFormat,  
  80.     kCCInternalOpenGLFormat,  
  81.     kCCInternalOpenGLType,  
  82.     kCCInternalBPP,  
  83.     kCCInternalCompressedImage,  
  84.     kCCInternalCCTexture2DPixelFormat,  
  85. };  
  86. //PVR文件头信息结构  
  87. typedef struct _PVRTexHeader  
  88. {  
  89.     unsigned int headerLength;  //头信息长度  
  90.     unsigned int height;            //高  
  91.     unsigned int width;         //宽  
  92.     unsigned int numMipmaps;    //是否有多级纹理  
  93.     unsigned int flags;         //标记位  
  94.     unsigned int dataLength;    //后面的像素数据长度  
  95.     unsigned int bpp;           //色深  
  96.     unsigned int bitmaskRed;    //对应红色的像素位  
  97.     unsigned int bitmaskGreen;  //对应绿色的像素位  
  98.     unsigned int bitmaskBlue;   //对应蓝色的像素位  
  99.     unsigned int bitmaskAlpha; //对应ALPHA色的像素位  
  100.     unsigned int pvrTag;            //  
  101.     unsigned int numSurfs;      //是否有多层  
  102. } PVRTexHeader;  
  103. //构造函数  
  104. CCTexturePVR::CCTexturePVR()   
  105. : m_uTableFormatIndex(0)  
  106. , m_uNumberOfMipmaps(0)  
  107. , m_uWidth(0)  
  108. , m_uHeight(0)  
  109. , m_bRetainName(false)  
  110. , m_bHasAlpha(false)  
  111. , m_uName(0)  
  112. , m_eFormat(kCCTexture2DPixelFormat_Default)  
  113. {  
  114. }  
  115. //析构函数  
  116. CCTexturePVR::~CCTexturePVR()  
  117. {  
  118.     CCLOGINFO( "cocos2d: deallocing CCTexturePVR" );  
  119.     //释放OpenGL贴图  
  120.     if (m_uName != 0 && ! m_bRetainName)  
  121.     {  
  122.         ccGLDeleteTexture(m_uName);  
  123.     }  
  124. }  
  125. //解压PVR像素数据  
  126. bool CCTexturePVR::unpackPVRData(unsigned char* data, unsigned int len)  
  127. {  
  128.     bool success = false;  
  129.     PVRTexHeader *header = NULL;  
  130.     unsigned int flags, pvrTag;  
  131.     unsigned int dataLength = 0, dataOffset = 0, dataSize = 0;  
  132.     unsigned int blockSize = 0, widthBlocks = 0, heightBlocks = 0;  
  133.     unsigned int width = 0, height = 0, bpp = 4;  
  134.     unsigned char *bytes = NULL;  
  135.     unsigned int formatFlags;  
  136.   
  137.     //将数据地址转为头信息指针,这个就可以直接通过结构指针进行存取。  
  138.     header = (PVRTexHeader *)data;  
  139.   
  140.     //格式有效性检查  
  141.     pvrTag = CC_SWAP_INT32_LITTLE_TO_HOST(header->pvrTag);  
  142.   
  143.     /* 
  144.         Check that given data really represents pvrtexture 
  145.  
  146.         [0] = 'P' 
  147.         [1] = 'V' 
  148.         [2] = 'R' 
  149.         [3] = '!' 
  150.     */  
  151.     if (gPVRTexIdentifier[0] != ((pvrTag >>  0) & 0xff) ||  
  152.         gPVRTexIdentifier[1] != ((pvrTag >>  8) & 0xff) ||  
  153.         gPVRTexIdentifier[2] != ((pvrTag >> 16) & 0xff) ||  
  154.         gPVRTexIdentifier[3] != ((pvrTag >> 24) & 0xff))  
  155.     {  
  156.         CCLOG("Unsupported PVR format. Use the Legacy format until the new format is supported");  
  157.         return false;  
  158.     }  
  159.     //取得配置信息  
  160.     CCConfiguration *configuration = CCConfiguration::sharedConfiguration();  
  161.     //取得标记  
  162.     flags = CC_SWAP_INT32_LITTLE_TO_HOST(header->flags);  
  163.     formatFlags = flags & PVR_TEXTURE_FLAG_TYPE_MASK;  
  164.     bool flipped = (flags & kPVRTextureFlagVerticalFlip) ? true : false;  
  165.     if (flipped)  
  166.     {  
  167.         CCLOG("cocos2d: WARNING: Image is flipped. Regenerate it using PVRTexTool");  
  168.     }  
  169.     //查看是否支持非2的幂次方大小的纹理  
  170.     if (! configuration->supportsNPOT() &&  
  171.         (header->width != ccNextPOT(header->width) || header->height != ccNextPOT(header->height)))  
  172.     {  
  173.         CCLOG("cocos2d: ERROR: Loding an NPOT texture (%dx%d) but is not supported on this device", header->width, header->height);  
  174.         return false;  
  175.     }  
  176.     //遍历所有的格式信息数据,找到对应的格式信息。  
  177.     for (m_uTableFormatIndex = 0; m_uTableFormatIndex < (unsigned int)MAX_TABLE_ELEMENTS; m_uTableFormatIndex++)  
  178.     {  
  179.         if (tableFormats[m_uTableFormatIndex][kCCInternalPVRTextureFormat] == formatFlags)  
  180.         {  
  181.             //Reset num of mipmaps  
  182.             m_uNumberOfMipmaps = 0;  
  183.   
  184.             //取得图片大小  
  185.             m_uWidth = width = CC_SWAP_INT32_LITTLE_TO_HOST(header->width);  
  186.             m_uHeight = height = CC_SWAP_INT32_LITTLE_TO_HOST(header->height);  
  187.               
  188.             //检查是否有ALPHA通道  
  189.             if (CC_SWAP_INT32_LITTLE_TO_HOST(header->bitmaskAlpha))  
  190.             {  
  191.                 m_bHasAlpha = true;  
  192.             }  
  193.             else  
  194.             {  
  195.                 m_bHasAlpha = false;  
  196.             }  
  197.               
  198.             //取得数据长度  
  199.             dataLength = CC_SWAP_INT32_LITTLE_TO_HOST(header->dataLength);  
  200.   
  201.             //将数据指针偏移到头信息之后,即像素数据所在位置  
  202.             bytes = ((unsigned char *)data) + sizeof(PVRTexHeader);  
  203.             m_eFormat = (CCTexture2DPixelFormat)(tableFormats[m_uTableFormatIndex][kCCInternalCCTexture2DPixelFormat]);  
  204.             bpp = tableFormats[m_uTableFormatIndex][kCCInternalBPP];  
  205.               
  206.             // 遍历每个多级纹理图像块  
  207.             while (dataOffset < dataLength)  
  208.             {  
  209.                 switch (formatFlags) {  
  210.                     case kPVRTexturePixelTypePVRTC_2:  
  211.                         blockSize = 8 * 4; // Pixel by pixel block size for 2bpp  
  212.                         widthBlocks = width / 8;  
  213.                         heightBlocks = height / 4;  
  214.                         break;  
  215.                     case kPVRTexturePixelTypePVRTC_4:  
  216.                         blockSize = 4 * 4; // Pixel by pixel block size for 4bpp  
  217.                         widthBlocks = width / 4;  
  218.                         heightBlocks = height / 4;  
  219.                         break;  
  220.                     case kPVRTexturePixelTypeBGRA_8888:  
  221.                         if (CCConfiguration::sharedConfiguration()->supportsBGRA8888() == false)   
  222.                         {  
  223.                             CCLOG("cocos2d: TexturePVR. BGRA8888 not supported on this device");  
  224.                             return false;  
  225.                         }  
  226.                     default:  
  227.                         blockSize = 1;  
  228.                         widthBlocks = width;  
  229.                         heightBlocks = height;  
  230.                         break;  
  231.                 }  
  232.                   
  233.                 // Clamp to minimum number of blocks  
  234.                 if (widthBlocks < 2)  
  235.                 {  
  236.                     widthBlocks = 2;  
  237.                 }  
  238.                 if (heightBlocks < 2)  
  239.                 {  
  240.                     heightBlocks = 2;  
  241.                 }  
  242.   
  243.                 dataSize = widthBlocks * heightBlocks * ((blockSize  * bpp) / 8);  
  244.                 unsigned int packetLength = (dataLength - dataOffset);  
  245.                 packetLength = packetLength > dataSize ? dataSize : packetLength;  
  246.                   
  247.                 //记录每个多级纹理图像块的像素数据地址和长度。  
  248.                 m_asMipmaps[m_uNumberOfMipmaps].address = bytes + dataOffset;  
  249.                 m_asMipmaps[m_uNumberOfMipmaps].len = packetLength;  
  250.                 m_uNumberOfMipmaps++;  
  251.                   
  252.                 //检查是否超出最大级数  
  253.                 CCAssert(m_uNumberOfMipmaps < CC_PVRMIPMAP_MAX,   
  254.                          "TexturePVR: Maximum number of mimpaps reached. Increate the CC_PVRMIPMAP_MAX value");  
  255.                 //偏移到下一个多级纹理图像块。  
  256.                 dataOffset += packetLength;  
  257.                   
  258.                 //大小减为原来1/2。  
  259.                 width = MAX(width >> 1, 1);  
  260.                 height = MAX(height >> 1, 1);  
  261.             }  
  262.               
  263.             //读取完成  
  264.             success = true;  
  265.             break;  
  266.         }  
  267.     }  
  268.     //成功判断  
  269.     if (! success)  
  270.     {  
  271.         CCLOG("cocos2d: WARNING: Unsupported PVR Pixel Format: 0x%2x. Re-encode it with a OpenGL pixel format variant", formatFlags);  
  272.     }  
  273.       
  274.     return success;  
  275. }  
  276. //创建OpenGL贴图  
  277. bool CCTexturePVR::createGLTexture()  
  278. {     
  279.     //保存宽高  
  280.     unsigned int width = m_uWidth;  
  281.     unsigned int height = m_uHeight;  
  282.     GLenum err;  
  283.     //如果文件中有多级纹理  
  284.     if (m_uNumberOfMipmaps > 0)  
  285.     {  
  286.         //先释放原来的纹理  
  287.         if (m_uName != 0)  
  288.         {  
  289.             ccGLDeleteTexture(m_uName);  
  290.         }  
  291.         //像素数据每字节对齐  
  292.         glPixelStorei(GL_UNPACK_ALIGNMENT,1);  
  293.         //产生OpenGL贴图  
  294.         glGenTextures(1, &m_uName);  
  295.         glBindTexture(GL_TEXTURE_2D, m_uName);  
  296.           
  297.         // 如果无多级纹理,设置最小滤波方式为线性过滤,产生抗锯齿效果。  
  298.         if (m_uNumberOfMipmaps == 1)  
  299.         {  
  300.             glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);  
  301.         }  
  302.         else  
  303.         {  //设置最小滤波方式为最近点采样,这种方式有锯齿。  
  304.             glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_NEAREST);  
  305.         }  
  306.         //设置最大滤波方式为线性过滤  
  307.         glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);  
  308.     //设置贴图的横向纹理寻址模式为边缘截取模式。总是忽略边界。  
  309.         glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);  
  310.         //设置贴图的纵向纹理寻址模式为边缘截取模式。总是忽略边界。  
  311.         glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);  
  312.     }  
  313.     //检错  
  314.     CHECK_GL_ERROR_DEBUG(); // clean possible GL error  
  315.     //取得对应格式的相关信息。  
  316.     GLenum internalFormat = tableFormats[m_uTableFormatIndex][kCCInternalOpenGLInternalFormat];  
  317.     GLenum format = tableFormats[m_uTableFormatIndex][kCCInternalOpenGLFormat];  
  318.     GLenum type = tableFormats[m_uTableFormatIndex][kCCInternalOpenGLType];  
  319.     bool compressed = (0 == tableFormats[m_uTableFormatIndex][kCCInternalCompressedImage]) ? false : true;  
  320.   
  321.     // 循环产生多级纹理贴图。  
  322.     for (unsigned int i = 0; i < m_uNumberOfMipmaps; ++i)  
  323.     {     
  324.         //检查配置信息是否支持压缩格式。  
  325.         if (compressed && ! CCConfiguration::sharedConfiguration()->supportsPVRTC())   
  326.         {  
  327.             CCLOG("cocos2d: WARNING: PVRTC images are not supported");  
  328.             return false;  
  329.         }  
  330.         //取得当前级别的图像块的像素数据地址和长度。  
  331.         unsigned char *data = m_asMipmaps[i].address;  
  332.         unsigned int datalen = m_asMipmaps[i].len;  
  333.         //如果是压缩格式,则产生压缩格式的贴图。  
  334.         if (compressed)  
  335.         {  
  336.             glCompressedTexImage2D(GL_TEXTURE_2D, i, internalFormat, width, height, 0, datalen, data);  
  337.         }  
  338.         else  
  339.         {//产生一般格式的贴图。  
  340.             glTexImage2D(GL_TEXTURE_2D, i, internalFormat, width, height, 0, format, type, data);  
  341.         }  
  342.         //如果是非2的幂次方大小,则提示。  
  343.         if (i > 0 && (width != height || ccNextPOT(width) != width ))  
  344.         {  
  345.             CCLOG("cocos2d: TexturePVR. WARNING. Mipmap level %u is not squared. Texture won't render correctly. width=%u != height=%u", i, width, height);  
  346.         }  
  347.         //检错  
  348.         err = glGetError();  
  349.         if (err != GL_NO_ERROR)  
  350.         {  
  351.             CCLOG("cocos2d: TexturePVR: Error uploading compressed texture level: %u . glError: 0x%04X", i, err);  
  352.             return false;  
  353.         }  
  354.         //变为原来大小的1/2。  
  355.         width = MAX(width >> 1, 1);  
  356.         height = MAX(height >> 1, 1);  
  357.     }  
  358.           
  359.     return true;  
  360. }  
  361.   
  362. //成员函数:加载PVR图片  
  363. bool CCTexturePVR::initWithContentsOfFile(const char* path)  
  364. {  
  365.     unsigned char* pvrdata = NULL;  
  366.     int pvrlen = 0;  
  367.     //定义字符串变量,存储转换为小写的路径字符串。     
  368.     std::string lowerCase(path);  
  369.     for (unsigned int i = 0; i < lowerCase.length(); ++i)  
  370.     {  
  371.         lowerCase[i] = tolower(lowerCase[i]);  
  372.     }  
  373.     //检查是否是ccz格式。  
  374.     if (lowerCase.find(".ccz") != std::string::npos)  
  375.     {  
  376.         //调用压缩库函数将文件数据读取到为pvrdata申请的内存中。取得其大小存入pvrlen变量中。  
  377.         pvrlen = ZipUtils::ccInflateCCZFile(path, &pvrdata);  
  378.     }  
  379.     //检查是否是gz格式。  
  380.     else if (lowerCase.find(".gz") != std::string::npos)  
  381.     {     
  382.         //调用压缩库函数将文件数据读取到为pvrdata申请的内存中。取得其大小存入pvrlen变量中。  
  383.         pvrlen = ZipUtils::ccInflateGZipFile(path, &pvrdata);  
  384.     }  
  385.     else  
  386.     {  
  387.          //普通PVR文件,将文件数据读取到一块内存中返回给pvrdata。取得其大小存入pvrlen变量中。  
  388.         pvrdata = CCFileUtils::sharedFileUtils()->getFileData(path, "rb", (unsigned long *)(&pvrlen));  
  389.     }  
  390.     //如果读取失败则返回  
  391.     if (pvrlen < 0)  
  392.     {  
  393.         this->release();  
  394.         return false;  
  395.     }  
  396.       
  397.     m_uNumberOfMipmaps = 0;  
  398.   
  399.     m_uName = 0;  
  400.     m_uWidth = m_uHeight = 0;  
  401.     m_bHasAlpha = false;  
  402.   
  403.     m_bRetainName = false;   
  404.     //如果解压数据失败则返回。  
  405.     //解压后,如果创建贴图失败则返回。  
  406.     if (!unpackPVRData(pvrdata, pvrlen)  || !createGLTexture())  
  407.     {  
  408.         CC_SAFE_DELETE_ARRAY(pvrdata);  
  409.         this->release();  
  410.         return false;  
  411.     }  
  412.     //释放数据占用的内存。  
  413.     CC_SAFE_DELETE_ARRAY(pvrdata);  
  414.       
  415.     return true;  
  416. }  
  417. //导出的静态函数:加载PVR图片  
  418. CCTexturePVR * CCTexturePVR::pvrTextureWithContentsOfFile(const char* path)  
  419. {  
  420.     return CCTexturePVR::create(path);  
  421. }  
  422. //静态函数:加载PVR图片  
  423. CCTexturePVR * CCTexturePVR::create(const char* path)  
  424. {  
  425.     //new 一个新的CCTexturePVR  
  426.     CCTexturePVR * pTexture = new CCTexturePVR();  
  427.     if (pTexture)  
  428.     {  
  429.         //调用成员函数加载PVR图片  
  430.         if (pTexture->initWithContentsOfFile(path))  
  431.         {  
  432.             //如果成功,设置交由内存管理器使用引用计数器进行内存管理。  
  433.             pTexture->autorelease();  
  434.         }  
  435.         else  
  436.         {     
  437.               //否则释放纹理  
  438.             delete pTexture;  
  439.             pTexture = NULL;  
  440.         }  
  441.     }  
  442.     //返回这个CCTexturePVR  
  443.     return pTexture;  
  444. }  
  445.   
  446. NS_CC_END  

三.CCTextureCache:

打开CCTextureCache.h:

[cpp]  view plain copy
  1. #ifndef __CCTEXTURE_CACHE_H__  
  2. #define __CCTEXTURE_CACHE_H__  
  3. //由CCObject派生  
  4. #include "cocoa/CCObject.h"  
  5. //需要用到字典  
  6. #include "cocoa/CCDictionary.h"  
  7. #include "textures/CCTexture2D.h"  
  8. #include   
  9.   
  10. //这里用到CCImage类和STL容器之一list  
  11. #if CC_ENABLE_CACHE_TEXTURE_DATA  
  12.     #include "platform/CCImage.h"  
  13.     #include   
  14. #endif  
  15.   
  16. //Cocos2d命名空间  
  17. NS_CC_BEGIN  
  18. //用到线程锁  
  19. class CCLock;  
  20. //用到CCImage处理图片  
  21. class CCImage;  
  22.   
  23. //纹理管理器  
  24. class CC_DLL CCTextureCache : public CCObject  
  25. {  
  26. protected:  
  27.     //字典对象指针。  
  28. CCDictionary*               m_pTextures;  
  29. //线程临界区。用于锁定字典访问,貌似用不到。这里屏蔽了~  
  30.     //pthread_mutex_t          *m_pDictLock;  
  31.   
  32.   
  33. private:  
  34.     // 设置多线程加载图片时的回调函数。  
  35.     void addImageAsyncCallBack(float dt);  
  36.   
  37. public:  
  38.     //构造函数  
  39. CCTextureCache();  
  40. //析构函数  
  41.     virtual ~CCTextureCache();  
  42.     //取得当前类描述  
  43.     const char* description(void);  
  44.     //取得当前字典的快照(拷贝)  
  45.     CCDictionary* snapshotTextures();  
  46.   
  47.     //返回唯一纹理管理器的实例指针  
  48.     static CCTextureCache * sharedTextureCache();  
  49.   
  50.     //销毁唯一纹理管理器的实例指针  
  51.     static void purgeSharedTextureCache();  
  52.   
  53. //加载一个图片生成纹理,文件名做为字典的查询对应关键字。返回生成的纹理指针,支持png,bmp,tiff,jpeg,pvr,gif等格式。  
  54.     CCTexture2D* addImage(const char* fileimage);  
  55.   
  56.     //此函数可以支持多线程载入图片,调用时会创建一个线程进行异步加载,加载成功后由主线程调用设置的回调函数,当然创建的纹理会做为参数传递。支持png和jpg  
  57.     void addImageAsync(const char *path, CCObject *target, SEL_CallFuncO selector);  
  58.   
  59.   
  60.     //加载一个图片生成纹理,指定参数key(其实要求是图片的相对路径字符串)做为字典的查询对应关键字。  
  61.     CCTexture2D* addUIImage(CCImage *image, const char *key);  
  62.   
  63.     //通过查询关键字(其实要求是图片的相对路径字符串)从字典里找到对应的纹理。  
  64. CCTexture2D* textureForKey(const char* key);  

你可能感兴趣的:(深入分析Cocos2d-x 中的“纹理”)