OpenGLES入門 5 -- 紋理貼圖

步驟

1玲销、初始化上下文;
2输拇、設(shè)置緩沖區(qū)
3、設(shè)置著色器
4贤斜、創(chuàng)建圖片紋理
5策吠、確定頂點(diǎn)坐標(biāo),激活紋理瘩绒,渲染繪制

1猴抹、初始化

_eaglContext =[[EAGLContext alloc]initWithAPI:kEAGLRenderingAPIOpenGLES2];
[EAGLContext setCurrentContext:_eaglContext];

_glLayer = (CAEAGLLayer*) self.layer;
// CALayer 默認(rèn)是透明的,必須將它設(shè)為不透明才能讓其可見(jiàn)
_glLayer.opaque = YES;
// 設(shè)置描繪屬性锁荔,在這里設(shè)置不維持渲染內(nèi)容以及顏色格式為 RGBA8
_glLayer.drawableProperties = [NSDictionary dictionaryWithObjectsAndKeys:
                                     [NSNumber numberWithBool:NO], kEAGLDrawablePropertyRetainedBacking, kEAGLColorFormatRGBA8, kEAGLDrawablePropertyColorFormat, nil];

kEAGLColorFormatRGBA8:使用8位來(lái)保存RGBA的值;
kEAGLDrawablePropertyRetainedBacking:設(shè)置NO不保留之前繪制的圖像以用來(lái)重用;

2蟀给、綁定渲染緩沖及幀緩沖區(qū)

glGenRenderbuffers(1, &_colorRenderBuffer);
glBindRenderbuffer(GL_RENDERBUFFER, _colorRenderBuffer);
[_eaglContext renderbufferStorage:GL_RENDERBUFFER fromDrawable:_glLayer];
    
glGenFramebuffers(1,&_frameBuffer);
glBindFramebuffer(GL_FRAMEBUFFER,_frameBuffer);
glFramebufferRenderbuffer(GL_FRAMEBUFFER,GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, _frameBuffer);

渲染緩存:存儲(chǔ)繪制結(jié)果的緩沖區(qū)
幀緩存:接收渲染結(jié)果的緩沖區(qū),為GPU指定存儲(chǔ)渲染結(jié)果的區(qū)域阳堕。

3跋理、設(shè)置著色器

//shader
GLuint vertext  =[self compileWithShaderName:@"Vertex" shaderType:GL_VERTEX_SHADER];
GLuint fragment =[self compileWithShaderName:@"Fragment" shaderType:GL_FRAGMENT_SHADER];
    
_glProgram =glCreateProgram();
glAttachShader(_glProgram, vertext);
glAttachShader(_glProgram, fragment);

//操作產(chǎn)生最后的可執(zhí)行程序,它包含最后可以在硬件上執(zhí)行的硬件指令恬总。
glLinkProgram(_glProgram);
    
GLint linkSuccess = GL_TRUE;
glGetProgramiv(_glProgram, GL_LINK_STATUS,&linkSuccess);
if (linkSuccess ==GL_FALSE) {
     GLchar glMessage[256];
     glGetProgramInfoLog(_glProgram, sizeof(glMessage), 0, &glMessage[0]);
     NSString *messageString = [NSString stringWithUTF8String:glMessage];
     NSLog(@"program error %@", messageString);
     exit(1);
}
    
//綁定著色器參數(shù)
glUseProgram(_glProgram);
_glPosition = glGetAttribLocation(_glProgram,"Position");
-(GLuint)compileWithShaderName:(NSString*)name shaderType:(GLenum)shaderType
{
    //獲取著色器文件
    NSString *shaderPath =[[NSBundle mainBundle]pathForResource:name ofType:@"glsl"];
    NSError *error;
    NSString *strShader =[NSString stringWithContentsOfFile:shaderPath encoding:NSUTF8StringEncoding error:&error];
    NSLog(@"strShader %@",strShader);
    if (!strShader) {
        NSLog(@"shader error %@",error.localizedDescription);
        exit(1);
    }
    
    // 2 創(chuàng)建一個(gè)代表shader的OpenGL對(duì)象, 指定vertex或fragment shader
    GLuint shaderHandler = glCreateShader(shaderType);
    
    // 3 獲取shader的source
    const char* shaderString = [strShader UTF8String];
    int shaderStringLength = (int)[strShader length];
    glShaderSource(shaderHandler, 1, &shaderString, &shaderStringLength);
    
    // 4 編譯shader
    glCompileShader(shaderHandler);
    
    // 5 查詢shader對(duì)象的信息
    GLint compileSuccess;
    glGetShaderiv(shaderHandler, GL_COMPILE_STATUS, &compileSuccess);
    if (compileSuccess == GL_FALSE) {
        GLchar messages[256];
        glGetShaderInfoLog(shaderHandler, sizeof(messages), 0, &messages[0]);
        NSString *messageString = [NSString stringWithUTF8String:messages];
        NSLog(@"%@", messageString);
        exit(1);
    }
    return shaderHandler;
}

著色器: 分為Vertex Shader 和Fragment Shader

  • 頂點(diǎn)著色器(Vertex Shader):用于確定圖形形狀
attribute vec4 Position;
attribute vec2 TextureCoords;
varying   vec2 TextureCoordsFrag;

void main(void)
{
    gl_Position = Position;
    TextureCoordsFrag = TextureCoords;
}

  • 片段著色器(Fragment Shader):用于確定圖像繪制渲染的顏色
precision mediump float;
uniform sampler2D Texture;
varying vec2 TextureCoordsFrag;

void main(void)
{
    vec4 mask = texture2D(Texture, TextureCoordsFrag);
    gl_FragColor = vec4(mask.rgb, 1.0);
}

這里推薦一個(gè)介紹GLSL語(yǔ)言的博客前普,講的還是比較詳細(xì)的

4、創(chuàng)建圖片紋理

/**
 * 創(chuàng)建圖片紋理
 */
-(void)initImageTexture
{
    //獲取圖片
    NSString *imgPath =[[NSBundle mainBundle]pathForResource:@"3D" ofType:@"png"];
    NSData   *data    =[[NSData alloc]initWithContentsOfFile:imgPath];
    UIImage  *image   =[UIImage imageWithData:data];
    _textureID =[self createTextureWithImage:image];
}
-(GLuint)createTextureWithImage:(UIImage*)image
{
    //獲取圖片基本參數(shù)
    CGImageRef imageRef =[image CGImage];
    GLuint width   = (GLuint)CGImageGetWidth(imageRef);
    GLuint height  = (GLuint)CGImageGetHeight(imageRef);
    CGRect rect    = CGRectMake(0,0,width,height);
    
    //繪制
    CGColorSpaceRef  colorSpace =  CGColorSpaceCreateDeviceRGB();
    void *imageData  =  malloc(width*height*4);
 
   /**
     *  CGBitmapContextCreate(void * __nullable data,size_t width, size_t height, size_t
     *  bitsPerComponent, size_t bytesPerRow,CGColorSpaceRef cg_nullable space, uint32_t
     *  bitmapInfo)
     *  data:指向繪圖操作被渲染的內(nèi)存區(qū)域壹堰,這個(gè)內(nèi)存區(qū)域大小應(yīng)該為(bytesPerRow*height)個(gè)字節(jié)拭卿。如果對(duì)繪制操作被
     渲染的內(nèi)存區(qū)域并無(wú)特別的要求,那么可以傳遞NULL給參數(shù)data缀旁。
     *  width:代表被渲染內(nèi)存區(qū)域的寬度记劈。
     *  height:代表被渲染內(nèi)存區(qū)域的高度。
     *  bitsPerComponent:被渲染內(nèi)存區(qū)域中組件在屏幕每個(gè)像素點(diǎn)上需要使用的bits位并巍,舉例來(lái)說(shuō)目木,如果使用32-bit像素和
     RGB顏色格式,那么RGBA顏色格式中每個(gè)組件在屏幕每個(gè)像素點(diǎn)上需要使用的bits位就為32/4=8。
     *  bytesPerRow:代表被渲染內(nèi)存區(qū)域中每行所使用的bytes位數(shù)刽射。
     *  colorspace:用于被渲染內(nèi)存區(qū)域的“位圖上下文”军拟。
     *  bitmapInfo:指定被渲染內(nèi)存區(qū)域的“視圖”是否包含一個(gè)alpha(透視)通道以及每個(gè)像素相應(yīng)的位置,除此之外還
     可以指定組件式是浮點(diǎn)值還是整數(shù)值誓禁。
     */
    CGContextRef contextRef = CGBitmapContextCreate(imageData, width, height, 8, width * 4, colorSpace, kCGImageAlphaPremultipliedLast | kCGBitmapByteOrder32Big);

    /**
     *  void CGContextTranslateCTM ( CGContextRef c, CGFloat tx, CGFloat ty ):平移坐標(biāo)系統(tǒng)懈息。
     *  該方法相當(dāng)于把原來(lái)位于 (0, 0) 位置的坐標(biāo)原點(diǎn)平移到 (tx, ty) 點(diǎn)。在平移后的坐標(biāo)系統(tǒng)上繪制圖形時(shí)摹恰,所有坐標(biāo)點(diǎn)的 X 坐標(biāo)都相當(dāng)于增加了 tx辫继,所有點(diǎn)的 Y 坐標(biāo)都相當(dāng)于增加了 ty。
     */
    CGContextTranslateCTM(contextRef, 0, height);

    /**
     *  void CGContextScaleCTM ( CGContextRef c, CGFloat sx, CGFloat sy ):縮放坐標(biāo)系統(tǒng)俗慈。
     *  該方法控制坐標(biāo)系統(tǒng)水平方向上縮放 sx姑宽,垂直方向上縮放 sy。在縮放后的坐標(biāo)系統(tǒng)上繪制圖形時(shí)闺阱,所有點(diǎn)的 X 坐標(biāo)都相當(dāng)于乘以 sx 因子炮车,所有點(diǎn)的 Y 坐標(biāo)都相當(dāng)于乘以 sy 因子。
     */
    
    CGContextScaleCTM(contextRef, 1.0f, -1.0f);
    CGColorSpaceRelease(colorSpace);
    CGContextClearRect(contextRef, rect);
    CGContextDrawImage(contextRef, rect, imageRef);
    
    //生成紋理
    glEnable(GL_TEXTURE_2D);
    GLuint textureID;
    glGenTextures(1,&textureID);
    glBindTexture(GL_TEXTURE_2D, textureID);
    
    //紋理設(shè)置
    glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_WRAP_T,GL_CLAMP_TO_EDGE);
    glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_WRAP_S,GL_CLAMP_TO_EDGE);
    glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR);
    glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);
    
    /**
     *  void glTexImage2D(GLenum target,GLint level,GLint internalformat,GLsizei width,GLsizei
     height,GLint border,GLenum format,GLenum type,const GLvoid * pixels);
     *  target  指定目標(biāo)紋理酣溃,這個(gè)值必須是GL_TEXTURE_2D瘦穆。
     *  level   執(zhí)行細(xì)節(jié)級(jí)別。0是最基本的圖像級(jí)別赊豌,你表示第N級(jí)貼圖細(xì)化級(jí)別扛或。
     *  internalformat     指定紋理中的顏色組件,這個(gè)取值和后面的format取值必須相同亿絮「婧埃可選的值有
        GL_ALPHA,GL_RGB,GL_RGBA,GL_LUMINANCE,GL_LUMINANCE_ALPHA 等幾種。
     *  width   指定紋理圖像的寬度派昧,必須是2的n次方。紋理圖片至少要支持64個(gè)材質(zhì)元素的寬度
     *  height  指定紋理圖像的高度拢切,必須是2的m次方蒂萎。紋理圖片至少要支持64個(gè)材質(zhì)元素的高度
     *  border  指定邊框的寬度。必須為0淮椰。
     *  format  像素?cái)?shù)據(jù)的顏色格式五慈,必須和internalformatt取值必須相同≈魉耄可選的值有
     GL_ALPHA,GL_RGB,GL_RGBA,GL_LUMINANCE,GL_LUMINANCE_ALPHA 等幾種泻拦。
     *  type    指定像素?cái)?shù)據(jù)的數(shù)據(jù)類型『雒剑可以使用的值有
        GL_UNSIGNED_BYTE,
        GL_UNSIGNED_SHORT_5_6_5,
        GL_UNSIGNED_SHORT_4_4_4_4,
        GL_UNSIGNED_SHORT_5_5_5_1
     *  pixels  指定內(nèi)存中指向圖像數(shù)據(jù)的指針
     */
    glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, width, height, 0, GL_RGBA, GL_UNSIGNED_BYTE, imageData);
   
    //綁定紋理位置
    glBindTexture(GL_TEXTURE_2D, 0);
    //釋放內(nèi)存
    CGContextRelease(contextRef);
    free(imageData);
    
    return textureID;
}

5争拐、確定頂點(diǎn)坐標(biāo),激活紋理晦雨,渲染繪制

//清屏
glClearColor(1.0, 1.0, 1.0, 1.0);
glClear(GL_COLOR_BUFFER_BIT);
    
//設(shè)置繪制區(qū)域
glViewport(self.frame.size.width/2-50,self.frame.size.height/2-50,100,100);
   
//激活
glActiveTexture(GL_TEXTURE5); // 指定紋理單元GL_TEXTURE5
glBindTexture(GL_TEXTURE_2D, _textureID); // 綁定架曹,即可從_textureID中取出圖像數(shù)據(jù)隘冲。
glUniform1i(_texture, 5); // 與紋理單元的序號(hào)對(duì)應(yīng)
    
//render
[self renderVertices];
    
// 使用完之后解綁GL_TEXTURE_2D
glBindTexture(GL_TEXTURE_2D, 0);
[_eaglContext presentRenderbuffer:GL_RENDERBUFFER];
-(void)renderVertices
{
    GLfloat texCoords[] = {
        0, 0,//左下
        1, 0,//右下
        0, 1,//左上
        1, 1,//右上
    };
   
    /**
     *void glVertexAttribPointer(GLuint index,GLint size,GLenum type,GLboolean normalized,GLsizei
     *                            stride,const void *ptr)
     *     index: 著色器腳本對(duì)應(yīng)變量ID
     *     size : 此類型數(shù)據(jù)的個(gè)數(shù)
     *     type : 此類型的sizeof值
     *     normalized : 是否對(duì)非float類型數(shù)據(jù)轉(zhuǎn)化到float時(shí)候進(jìn)行歸一化處理
     *     stride : 此類型數(shù)據(jù)在數(shù)組中的重復(fù)間隔寬度,byte類型計(jì)數(shù)
     *     ptr    : 數(shù)據(jù)指針绑雄, 這個(gè)值受到VBO的影響
     */
    glVertexAttribPointer(_textureCoords, 2, GL_FLOAT, GL_FALSE, 0, texCoords);
    glEnableVertexAttribArray(_textureCoords);
    
    GLfloat vertices[] = {
        -1, -1, 0, //左下
         1, -1, 0, //右下
        -1,  1, 0, //左上
         1,  1, 0  //右上
    };
    glVertexAttribPointer(_glPosition, 3, GL_FLOAT, GL_FALSE, 0, vertices);
    glEnableVertexAttribArray(_glPosition);
    glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);

}

效果如下:

效果圖.png
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末展辞,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子万牺,更是在濱河造成了極大的恐慌罗珍,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,482評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件脚粟,死亡現(xiàn)場(chǎng)離奇詭異覆旱,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)珊楼,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,377評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門通殃,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人厕宗,你說(shuō)我怎么就攤上這事画舌。” “怎么了已慢?”我有些...
    開(kāi)封第一講書人閱讀 152,762評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵曲聂,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我佑惠,道長(zhǎng)朋腋,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書人閱讀 55,273評(píng)論 1 279
  • 正文 為了忘掉前任膜楷,我火速辦了婚禮旭咽,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘赌厅。我一直安慰自己穷绵,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,289評(píng)論 5 373
  • 文/花漫 我一把揭開(kāi)白布特愿。 她就那樣靜靜地躺著仲墨,像睡著了一般。 火紅的嫁衣襯著肌膚如雪揍障。 梳的紋絲不亂的頭發(fā)上目养,一...
    開(kāi)封第一講書人閱讀 49,046評(píng)論 1 285
  • 那天,我揣著相機(jī)與錄音毒嫡,去河邊找鬼癌蚁。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的匈勋。 我是一名探鬼主播礼旅,決...
    沈念sama閱讀 38,351評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼洽洁!你這毒婦竟也來(lái)了痘系?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 36,988評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤饿自,失蹤者是張志新(化名)和其女友劉穎汰翠,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體昭雌,經(jīng)...
    沈念sama閱讀 43,476評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡复唤,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,948評(píng)論 2 324
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了烛卧。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片佛纫。...
    茶點(diǎn)故事閱讀 38,064評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖总放,靈堂內(nèi)的尸體忽然破棺而出呈宇,到底是詐尸還是另有隱情,我是刑警寧澤局雄,帶...
    沈念sama閱讀 33,712評(píng)論 4 323
  • 正文 年R本政府宣布甥啄,位于F島的核電站,受9級(jí)特大地震影響炬搭,放射性物質(zhì)發(fā)生泄漏蜈漓。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,261評(píng)論 3 307
  • 文/蒙蒙 一宫盔、第九天 我趴在偏房一處隱蔽的房頂上張望融虽。 院中可真熱鬧,春花似錦灼芭、人聲如沸衣形。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 30,264評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至倒源,卻和暖如春苛预,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背笋熬。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 31,486評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工热某, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,511評(píng)論 2 354
  • 正文 我出身青樓昔馋,卻偏偏與公主長(zhǎng)得像筹吐,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子秘遏,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,802評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容

  • 第三章 管線一覽 本章我們會(huì)學(xué)到什么 OpenGL管線的每個(gè)階段做什么的 如果連接著色器和固定功能管線階段 如果創(chuàng)...
    葭五閱讀 6,218評(píng)論 2 18
  • 你好丘薛,三角形 圖形渲染管線(Pipeline) 3D坐標(biāo)轉(zhuǎn)為2D坐標(biāo)的處理過(guò)程是由OpenGL的圖形渲染管線(Pi...
    IceMJ閱讀 7,411評(píng)論 2 13
  • <轉(zhuǎn)>我也忘了轉(zhuǎn)自哪里,抱歉,感謝原作者 什么是Shader Shader(著色器)是一段能夠針對(duì)3D對(duì)象進(jìn)行操作...
    星易乾川閱讀 5,578評(píng)論 1 16
  • 一、頂點(diǎn)與片段著色器簡(jiǎn)介 Vertex and FragmentShader:最強(qiáng)大的Shader類型邦危,也是本系列...
    半閑書屋半閑人閱讀 9,644評(píng)論 6 10
  • “我們今天沒(méi)見(jiàn)過(guò)白天什么樣子”這是和室友一起去工廠打零工之后洋侨,她的一句話,想了想倦蚪,是真的希坚。 早上六點(diǎn)起床,收拾完畢...
    喝露水的鷹閱讀 301評(píng)論 0 0