使用OpenGL显示UIImage

使用openGL显示UIImage的原理无非就是将UIImage通过openGL转换成纹理然后贴在一个空间坐标中,openGL可以将UIImage的CGImage转换成纹理,其中openGL需要使用到着色器,转换代码如下:

import UIKit
import GLKit
import CoreMotion
let TEX_COORD_MAX : Float = 1
class myGL: UIView {

    struct Vertex{
        
        var Position: (CFloat, CFloat, CFloat, CFloat);
        var TexCoord: (CFloat, CFloat);
    }
    
    var Vertices : [Vertex] = [
        Vertex(Position: (1, -1, 0, 1), TexCoord: (TEX_COORD_MAX, 0)),
        Vertex(Position: (1, 1, 0, 1), TexCoord: (TEX_COORD_MAX, TEX_COORD_MAX)),
        Vertex(Position: (-1, 1, 0, 1), TexCoord: (0, TEX_COORD_MAX)),
        Vertex(Position: (-1, -1, 0, 1), TexCoord: (0, 0))
    ]
    
    var Indices: [GLubyte] = [
        0, 1, 2,
        2, 3, 0
    ]
    
    //基本属性
    private var eaglLayer : CAEAGLLayer?
    private var context : EAGLContext?
    private var colorRenderBuffer : GLuint = 0
    
    //着色器
    private var fragmentShader : GLuint = 0
    private var vertexShader : GLuint = 0
    
    //着色器变量
    private var positionSlot : GLuint = 0
    private var textCoordSlot : GLuint = 0
    private var textureUni : GLuint = 0
    private var filterUni : GLuint = 0
    
    //纹理
    private var texture : GLuint = 0
    
    //VBO
    private var vertexBuffer : GLuint = 0
    private var indexBuffer : GLuint = 0
    
    //UIImage的CGImage,从外部接收
    var spriteImage : CGImage?

   override class func layerClass() -> AnyClass
    {
        return CAEAGLLayer.self
     }

    override init(frame: CGRect) {
        super.init(frame: frame)
        setUpLayer()
        setUpContext()
        setUpRenderBuffer()
        setUpFramebuffer()
        compileShaders()
        setUpVBOs()
    }

    required init?(coder aDecoder: NSCoder) {
        super.init(coder: aDecoder)
    }

    
    private func setUpLayer() {
        eaglLayer = layer as? CAEAGLLayer;
        eaglLayer!.opaque = true;
    }
    
    private func setUpContext()
    {
        context = EAGLContext(API: .OpenGLES2)
        if context == nil {
            print("fail load OpenGLES2")
            return
        }
        
        if EAGLContext.setCurrentContext(context) == false
            print("fail to set currentContext")
        }
    }
    
    private func compileShaders()
    {
       fragmentShader = compileShader("FragmentShader", type: GLenum(GL_FRAGMENT_SHADER))
       vertexShader = compileShader("VertexShader", type: GLenum(GL_VERTEX_SHADER))
       
       let programHandle = glCreateProgram()
        glAttachShader(programHandle, vertexShader);
        glAttachShader(programHandle, fragmentShader);
        glLinkProgram(programHandle);
       
        var linkSuccess : GLint = 0
        glGetProgramiv(programHandle, GLenum(GL_LINK_STATUS), &linkSuccess)
        if linkSuccess == GL_FALSE
        {
            var value: GLint = 0
            glGetProgramiv(programHandle, GLenum(GL_INFO_LOG_LENGTH), &value)
            var infoLog: [GLchar] = [GLchar](count: Int(value), repeatedValue: 0)
            var infoLogLength: GLsizei = 0
            glGetProgramInfoLog(programHandle, value, &infoLogLength, &infoLog)
            let s = NSString(bytes: infoLog, length: Int(infoLogLength), encoding: NSASCIIStringEncoding)
            print(s)
        }
        glUseProgram(programHandle)
        
        //关联shader的参数
        positionSlot = GLuint(glGetAttribLocation(programHandle, "position"))
        glEnableVertexAttribArray(GLuint(positionSlot))

    
        glGetAttribLocation(programHandle, "texCoordIn")
        glEnableVertexAttribArray(GLuint(textCoordSlot))
        
        textureUni = GLuint(glGetUniformLocation(programHandle, "textureImage"))
        
        filterUni = GLuint(glGetUniformLocation(programHandle, "rgbaFilter"))
        glUniformMatrix4fv(GLint(filterUni), 1, GLboolean(GL_FALSE), filter!)
    }
    
    private func compileShader(shaderName : String, type : GLenum) -> GLuint
    {
        //1
        let shaderPath = NSBundle.mainBundle().pathForResource(shaderName, ofType: "glsl")
        var shaderString : NSString?
        do{
            shaderString = try NSString(contentsOfFile: shaderPath!, encoding: NSUTF8StringEncoding)
        }
        catch{
            print("fail to compile shader")
        }
        
        //2
        let shaderHandle = glCreateShader(type)
        
        //3
        var shaderStringUTF8 = shaderString?.cStringUsingEncoding(NSUTF8StringEncoding)
        glShaderSource(shaderHandle, 1, &shaderStringUTF8!, nil)

        //4
        glCompileShader(shaderHandle)
        
        //5
        var compileSuccess  : GLint = 0
        glGetShaderiv(shaderHandle, GLenum(GL_COMPILE_STATUS), &compileSuccess)
        if compileSuccess == GL_FALSE
        {
            var value: GLint = 0
            glGetShaderiv(shaderHandle, GLenum(GL_INFO_LOG_LENGTH), &value)
            var infoLog: [GLchar] = [GLchar](count: Int(value), repeatedValue: 0)
            var infoLogLength: GLsizei = 0
            glGetShaderInfoLog(shaderHandle, value, &infoLogLength, &infoLog)
            let s = NSString(bytes: infoLog, length: Int(infoLogLength), encoding: NSASCIIStringEncoding)
            print(s)
        }
        return shaderHandle
    }
    
    private func setUpVBOs()
    {
        glGenBuffers(1, &vertexBuffer)
        glBindBuffer(GLenum(GL_ARRAY_BUFFER), vertexBuffer);
        glBufferData(GLenum(GL_ARRAY_BUFFER), Vertices.count * sizeofValue(Vertices[0]), Vertices, GLenum(GL_STATIC_DRAW))
        
        glGenBuffers(1, &indexBuffer);
        glBindBuffer(GLenum(GL_ELEMENT_ARRAY_BUFFER), indexBuffer);
        glBufferData(GLenum(GL_ELEMENT_ARRAY_BUFFER), Indices.count * sizeofValue(Indices[0]), Indices, GLenum(GL_STATIC_DRAW))
    }
    
    private func setUpRenderBuffer()
    {
        glGenRenderbuffers(1, &colorRenderBuffer);
        glBindRenderbuffer(GLenum(GL_RENDERBUFFER), colorRenderBuffer);
        context!.renderbufferStorage(Int(GL_RENDERBUFFER), fromDrawable: eaglLayer);
    }
    
    private func setUpFramebuffer()
    {
        var framebuffer = GLuint()
        glGenFramebuffers(1, &framebuffer)
        glBindFramebuffer(GLenum(GL_FRAMEBUFFER), framebuffer)
        glFramebufferRenderbuffer(GLenum(GL_FRAMEBUFFER), GLenum(GL_COLOR_ATTACHMENT0),
                                  GLenum(GL_RENDERBUFFER), colorRenderBuffer)
    }

private func render()
    {
        glClearColor(0.0, 1.0, 0.0, 0.0)
        glClear(GLenum(GL_COLOR_BUFFER_BIT))
        
        glViewport(0, 0, GLsizei(frame.size.width), GLsizei(frame.size.height))
        
        glBindBuffer(GLenum(GL_ARRAY_BUFFER), vertexBuffer)
        glBindBuffer(GLenum(GL_ELEMENT_ARRAY_BUFFER), indexBuffer)
        
        glEnableVertexAttribArray(positionSlot)
        glVertexAttribPointer(GLuint(positionSlot), 4, GLenum(GL_FLOAT), GLboolean(GL_FALSE),
                              GLsizei(sizeof(Vertex)), UnsafePointer<Int>(bitPattern: 0))
        
        glVertexAttribPointer(textCoordSlot, 2, GLenum(GL_FLOAT), GLboolean(GL_FALSE), Int32(sizeof(Vertex)), UnsafePointer<Void>(bitPattern: sizeof(Float) * 4))
        
        glActiveTexture(GLenum(GL_TEXTURE0))
        glBindTexture(GLenum(GL_TEXTURE_2D), texture)
        glTexParameteri(GLenum(GL_TEXTURE_2D), GLenum(GL_TEXTURE_MAG_FILTER), GL_LINEAR)
        glTexParameteri(GLenum(GL_TEXTURE_2D), GLenum(GL_TEXTURE_MIN_FILTER), GL_LINEAR)
        glTexParameteri(GLenum(GL_TEXTURE_2D), GLenum(GL_TEXTURE_WRAP_S), GL_CLAMP_TO_EDGE)
        glTexParameteri(GLenum(GL_TEXTURE_2D), GLenum(GL_TEXTURE_WRAP_T), GL_CLAMP_TO_EDGE)
              
        glDrawElements(GLuint(GL_TRIANGLES), GLsizei(Indices.count), GLenum(GL_UNSIGNED_BYTE), UnsafePointer<Int>(bitPattern: 0))
        
        glDeleteTextures(1, &texture)
        
        context!.presentRenderbuffer(Int(GL_RENDERBUFFER))
    }
    
    //暴露接口
    func setUpTexture(spriteImage : CGImage)
    {
        // 1
        self.spriteImage = spriteImage
        // 2
        let width = CGImageGetWidth(spriteImage)
        let height = CGImageGetHeight(spriteImage)
        
        
        let spriteData = UnsafeMutablePointer<Void>(calloc(width * height * 4, sizeof(GLubyte)))
        
        let spriteContext = CGBitmapContextCreate(spriteData, width, height, 8, width * 4,CGImageGetColorSpace(spriteImage), CGImageAlphaInfo.PremultipliedLast.rawValue);
        
        // 3
        CGContextDrawImage(spriteContext, CGRectMake(0, 0, CGFloat(width), CGFloat(height)), spriteImage)
        
        // 4
        glGenTextures(1, &texture)
        glBindTexture(GLenum(GL_TEXTURE_2D), texture)
        glTexParameteri(GLenum(GL_TEXTURE_2D), GLenum(GL_TEXTURE_MIN_FILTER), GLint(GL_NEAREST))
        
        glTexImage2D(GLenum(GL_TEXTURE_2D), 0, GL_RGBA, GLsizei(width), GLsizei(height), 0, GLenum(GL_RGBA), GLenum(GL_UNSIGNED_BYTE), spriteData)
        GLenum(GL_UNSIGNED_BYTE), spriteData)
                
        free(spriteData)
        render()
    }

还有着色器相关的代码如下:
1.顶点着色器

attribute vec4 position;
attribute vec2 texCoordIn;

varying vec2 textureOut;

void main() {
    textureOut = vec2(texCoordIn.x, 1.0 - texCoordIn.y);
    gl_Position = position;
}

2.像素着色器

precision mediump float;
uniform sampler2D textureImage;
varying mediump vec2 textureOut;

void main() {
    mediump vec4 RGBA;
    RGBA = texture2D(textureImage, textureOut);
    gl_FragColor = RGBA ;
}

以上代码就是通过一个集成自UIView的类,然后接收下外部传进来的CGImage,之后通过OpenGL转换成纹理,渲染到UIView上面,然后再将UIView添加到合适的地方就可以了.
本人也是刚刚接触OpenGL,以上代码也是在网上东拼西凑弄出来的,这段代码亲测可用,很多坑都已经避免了,如果有什么疑惑的欢迎交流.
遗留的问题: 这段代码我是用来显示接收到的视频帧数据的,因为openGL使用的是GPU,这样效率更高,但是为了更一步提高效率,应该用glTexImage2D()预渲染一个空纹理 ,然后以后用glSubTexImage2D()更新纹理.不过我这样使用的时候发现后面使用glSubTexImage2D()显示不了新的纹理,直接黑屏了,有哪位大神指导其中缘由的还望不吝赐教

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

推荐阅读更多精彩内容