基于OpenGL编写一个简易的2D渲染框架-10 重构渲染器-Pass

  

Pass,渲染通路,一个渲染通路指的是一次像素处理和一次顶点处理,也就是指的是一次绘制。简单来说就是顶点数据在渲染管线中走一遍最后绘制。

渲染粒子系统的粒子时,需要开启 OpenGL 的混合模式,并使两个颜色相加。如果同一时间进行多张图片的绘制,并且这些图片的渲染并不需要开启混合模式。这时渲染的最终结果就是图片的渲染出现问题,这并不是我们想要的结果。一个解决方法就是进行两次 draw,第一次开启混合模式渲染粒子系统的粒子,第二次则关闭混合模式渲染图片。

上面解决方法的实现可以通过两个 Pass 渲染,第一个 Pass 设置开启混模式,第二个 Pass 关闭混合模式,进行两个 Pass 的渲染就可以了。

 

Simple2D 的 Pass 设计比较简单,主要实现

  1、裁剪测试、Alpha 测试、模板测试、深度测试和混合模式的设置。

  2、填充模式、面剔除模式、顶点正面方向的设置。

 

Pass 的成员属性添加这些测试是否开启的开关量:

 

bool bEnableScissor;        /* 裁剪测试 */
bool bEnableAlphaTest;      /* Alpha 测试 */
bool bEnableStencilTest;    /* 模板测试 */
bool bEnableDepthTest;      /* 深度测试 */
bool bEnableBlend;          /* 混合模式 */
bool bEnableCullFace;

 

 

接下来就是各种测试的属性变量:

    /* 剔除模式 */
    enum CullMode
    {
        CULL_MODE_BACK,             /* 只剔除背面 */
        CULL_MODE_FRONT,            /* 只剔除正面 */
        CULL_MODE_FRONT_AND_BACK    /* 剔除背面和正面 */
    };

    /* 正面的顶点顺序 */
    enum FrontFace
    {
        FRONT_FACE_CLOCK_WISE,            /* 顺时针为正面 */
        FRONT_FACE_COUNTER_CLOCK_WISE,    /* 逆时针为正面 */
    };

    /* 填充模式 */
    enum FillMode
    {
        FILL_LINE,        /* 线框 */
        FILL_POINT,       /**/
        FILL_FILL         /* 实体 */
    };

    /* 混合方程 */
    enum BlendEquation
    {
        BLEND_ADD,                 /* 彼此元素相加 */
        BLEND_SUNTRACT,            /* 彼此元素相减 */
        BLEND_REVERSE_SUBTRACT     /* 彼此元素相减,但顺序相反 */
    };

    /* 混合因子 */
    enum BlendFunc
    {
        BLEND_ZERO,
        BLEND_ONE,
        BLEND_SRC_COLOR,
        BLEND_ONE_MINUS_SRC_COLOR,
        BLEND_DST_COLOR,
        BLEND_ONE_MINUS_DST_COLOR,
        BLEND_SRC_ALPHA,
        BLEND_ONE_MINUS_SRC_ALPHA,
        BLEND_DST_ALPHA,
        BLEND_ONE_MINUS_DST_ALPHA,
        BLEND_CONSTANT_COLOR,
        BLEND_ONE_MINUS_CONSTANT_COLOR,
        BLEND_CONSTANT_ALPHA,
        BLEND_ONE_MINUS_CONSTANT_ALPHA
    };

    /* 比较函数 */
    enum CompareFunction
    {
        COMPARE_LESS,          /* 在片段深度值小于缓冲区的深度时通过测试 */
        COMPARE_LEQUAL,        /* 在片段深度值小于等于缓冲区的深度时通过测试 */

        COMPARE_GREATER,       /* 在片段深度值大于缓冲区的深度时通过测试 */
        COMPARE_GEQUAL,        /* 在片段深度值大于等于缓冲区的深度时通过测试 */

        COMPARE_EQUAL,         /* 在片段深度值等于缓冲区的深度时通过测试 */
        COMPARE_NOT_EQUAL,     /* 在片段深度值不等于缓冲区的深度时通过测试 */

        COMPARE_ALWAYS,        /* 永远通过测试 */
        COMPARE_NEVER          /* 永远不通过测试 */
    };

    /* 模板操作 */
    enum StencilOp
    {
        STENCIL_OP_KEEP,           /* 不改变,这也是默认值 */
        STENCIL_OP_ZERO,           /* 回零 */
        STENCIL_OP_REPLACE,        /* 使用测试条件中的设定值来代替当前模板值 */
        STENCIL_OP_INCR,           /* 增加1,但如果已经是最大值,则保持不变 */
        STENCIL_OP_INCR_WRAP,      /* 增加1,但如果已经是最大值,则从零重新开始 */
        STENCIL_OP_DECR,           /* 减少1,但如果已经是零,则保持不变 */
        STENCIL_OP_DECR_WRAP,      /* 减少1,但如果已经是零,则重新设置为最大值 */
        STENCIL_OP_INVERT          /* 按位取反 */
    };

 

        /* 裁剪测试,能够被绘制像素的区域 */
        int nScissorX, nScissorY, nScissorW, nScissorH;

        /* Alpha 测试 */
        CompareFunction alphaCompareFunction;
        float fClampRef;

        /* 模板测试 */
        CompareFunction stencilCompareFunction;
        unsigned int nStencilMask;
        int nStencilRef;

        StencilOp failStencilFailDepth;
        StencilOp passStencilFailDepth;
        StencilOp passStencilPassDepth;

        /* 深度测试,比较函数 */
        CompareFunction depthCompareFunction;

        /* 混合方程和混合因子 */
        BlendEquation blendEquation;
        BlendFunc blendSrc, blendDst, blendSrcAlpha, blendDstAlpha;

 

 

接下来是设置各个测试的变量值:

 

void enableScissor(bool enable) { bEnableScissor = enable; }
void enableAlphaTest(bool enable) { bEnableAlphaTest = enable; }
void enableStencilTest(bool enable) { bEnableStencilTest = enable; }
void enableDepthTest(bool enable) { bEnableDepthTest = enable; }
void enableBlend(bool enable) { bEnableBlend = enable; }
void enableCullFace(bool enable) { bEnableCullFace = enable; }

void setFillMode(FillMode fm) { fillMode = fm; }
void setCullMode(CullMode cm) { cullMode = cm; }
void setFrontFace(FrontFace ff) { frontFace = ff; }

/* 设置裁剪区域 */
void setScissorRect(int x, int y, int w, int h) { nScissorX = x; nScissorY = y; nScissorW = w; nScissorH = h; }

void setAlphaCompareFunc(CompareFunction compare) { alphaCompareFunction = compare; }
void setAlphaClamf(float clamf) { fClampRef = clamf; }

void setStencilCompareFunc(CompareFunction compare) { stencilCompareFunction = compare; }
void setStencilMask(unsigned int mask) { nStencilMask = mask; }
void setStencilRef(int ref) { nStencilRef = ref; }

/* 更新模板缓冲操作 */
void setStencilOp(StencilOp sfail, StencilOp dpfail, StencilOp dppass) {
            failStencilFailDepth = sfail;
            passStencilFailDepth = dpfail;
            passStencilPassDepth = dppass;
}

/* 设置深度测试比较函数 */
void setDepthCompareFunc(CompareFunction compare) { depthCompareFunction = compare; }

/* 设置混合操作的混合方程 */
void setBlendEquation(BlendEquation equation) { blendEquation = equation; }

/* 设置混合模式的混合因子 */
void setBlendFunc(BlendFunc srcColor, BlendFunc dstColor, BlendFunc srcAlpha, BlendFunc dstAlpha) {
            blendSrc = srcColor; 
            blendDst = dstColor;
            blendSrcAlpha = srcAlpha; 
            blendDstAlpha = dstAlpha;
}

 

 

上面没有什么好说的,关键的地方在于如何设置 OpenGL 的状态,

    static int toEnum(CullMode cullmode)
    {
        switch ( cullmode ) {
        case Simple2D::CULL_MODE_BACK:              return GL_BACK;
        case Simple2D::CULL_MODE_FRONT:             return GL_FRONT;
        case Simple2D::CULL_MODE_FRONT_AND_BACK:    return GL_FRONT_AND_BACK;
        }
        return GL_BACK;
    }

    static int toEnum(FrontFace frontface)
    {
        switch ( frontface ) {
        case Simple2D::FRONT_FACE_CLOCK_WISE:            return GL_CW;
        case Simple2D::FRONT_FACE_COUNTER_CLOCK_WISE:    return GL_CCW;
        }
        return GL_CCW;
    }

    static int toEnum(FillMode fillmode)
    {
        switch ( fillmode ) {
        case Simple2D::FILL_LINE:    return GL_LINE;
        case Simple2D::FILL_POINT:   return GL_POINT;
        case Simple2D::FILL_FILL:    return GL_FILL;
        }
        return GL_FILL;
    }

    static int toEnum(BlendEquation equation)
    {
        switch ( equation ) {
        case Simple2D::BLEND_ADD:                return GL_FUNC_ADD;
        case Simple2D::BLEND_SUNTRACT:           return GL_FUNC_SUBTRACT;
        case Simple2D::BLEND_REVERSE_SUBTRACT:   return GL_FUNC_REVERSE_SUBTRACT;
        }
        return GL_FUNC_ADD;
    }

    static int toEnum(BlendFunc func)
    {
        switch ( func ) {
        case Simple2D::BLEND_ZERO:                       return GL_ZERO;
        case Simple2D::BLEND_ONE:                        return GL_ONE;
        case Simple2D::BLEND_SRC_COLOR:                  return GL_SRC_COLOR;
        case Simple2D::BLEND_ONE_MINUS_SRC_COLOR:        return GL_ONE_MINUS_SRC_COLOR;
        case Simple2D::BLEND_DST_COLOR:                  return GL_DST_COLOR;
        case Simple2D::BLEND_ONE_MINUS_DST_COLOR:        return GL_ONE_MINUS_DST_COLOR;
        case Simple2D::BLEND_SRC_ALPHA:                  return GL_SRC_ALPHA;
        case Simple2D::BLEND_ONE_MINUS_SRC_ALPHA:        return GL_ONE_MINUS_SRC_ALPHA;
        case Simple2D::BLEND_DST_ALPHA:                  return GL_DST_ALPHA;
        case Simple2D::BLEND_ONE_MINUS_DST_ALPHA:        return GL_ONE_MINUS_DST_ALPHA;
        case Simple2D::BLEND_CONSTANT_COLOR:             return GL_CONSTANT_COLOR;
        case Simple2D::BLEND_ONE_MINUS_CONSTANT_COLOR:   return GL_ONE_MINUS_CONSTANT_COLOR;
        case Simple2D::BLEND_CONSTANT_ALPHA:             return GL_CONSTANT_ALPHA;
        case Simple2D::BLEND_ONE_MINUS_CONSTANT_ALPHA:   return GL_ONE_MINUS_CONSTANT_ALPHA;
        }
        return GL_ONE;
    }

    static int toEnum(CompareFunction compare)
    {
        switch ( compare ) {
        case Simple2D::COMPARE_LESS:        return GL_LESS;
        case Simple2D::COMPARE_LEQUAL:      return GL_LEQUAL;
        case Simple2D::COMPARE_GREATER:     return GL_GREATER;
        case Simple2D::COMPARE_GEQUAL:      return GL_GEQUAL;
        case Simple2D::COMPARE_EQUAL:       return GL_EQUAL;
        case Simple2D::COMPARE_NOT_EQUAL:   return GL_NOTEQUAL;
        case Simple2D::COMPARE_ALWAYS:      return GL_ALWAYS;
        case Simple2D::COMPARE_NEVER:       return GL_NEVER;
        }
        return GL_LESS;
    }

    static int toEnum(StencilOp stencilOp)
    {
        switch ( stencilOp ) {
        case Simple2D::STENCIL_OP_KEEP:         return GL_KEEP;
        case Simple2D::STENCIL_OP_ZERO:         return GL_ZERO;
        case Simple2D::STENCIL_OP_REPLACE:      return GL_REPLACE;
        case Simple2D::STENCIL_OP_INCR:         return GL_INCR;
        case Simple2D::STENCIL_OP_INCR_WRAP:    return GL_INCR_WRAP;
        case Simple2D::STENCIL_OP_DECR:         return GL_DECR;
        case Simple2D::STENCIL_OP_DECR_WRAP:    return GL_DECR_WRAP;
        case Simple2D::STENCIL_OP_INVERT:       return GL_INVERT;
        }
        return GL_KEEP;
    }

 

    void Pass::setOpenGLState()
    {
        /* 面剔除 */
        if ( bEnableCullFace ) {
            glEnable(GL_CULL_FACE);
            glCullFace(toEnum(cullMode));
            glFrontFace(toEnum(frontFace));
        }
        else {
            glDisable(GL_CULL_FACE);
        }

        /* 填充模式 */
        glPolygonMode(GL_FRONT_AND_BACK, toEnum(fillMode));

        /* 裁剪测试 */
        if ( bEnableScissor ) {
            glEnable(GL_SCISSOR_TEST);
            /* 左下角为坐标原点 */
            glScissor(nScissorX, nScissorY, nScissorW, nScissorH);
        }
        else {
            glDisable(GL_SCISSOR_TEST);
        }

        /* Alpha 测试 */
        if ( bEnableAlphaTest ) {
            glEnable(GL_ALPHA_TEST);
            glAlphaFunc(toEnum(alphaCompareFunction), fClampRef);
        }
        else {
            glDisable(GL_ALPHA_TEST);
        }

        /* 模板测试 */
        if ( bEnableStencilTest ) {
            glEnable(GL_STENCIL_TEST);
            glStencilFunc(toEnum(stencilCompareFunction), nStencilRef, nStencilMask);
            glStencilOp(toEnum(failStencilFailDepth), toEnum(passStencilFailDepth), toEnum(passStencilPassDepth));
        }
        else {
            glDisable(GL_STENCIL_TEST);
        }

        /* 深度测试 */
        if ( bEnableDepthTest ) {
            glEnable(GL_DEPTH_TEST);
            glDepthFunc(toEnum(depthCompareFunction));
        }
        else {
            glDisable(GL_DEPTH_TEST);
        }

        /* 混合模式 */
        if ( bEnableBlend ) {
            glEnable(GL_BLEND);
            glBlendEquation(toEnum(blendEquation));
            glBlendFuncSeparate(toEnum(blendSrc), toEnum(blendDst), toEnum(blendSrcAlpha), toEnum(blendDstAlpha));
        }
        else {
            glDisable(GL_BLEND);
        }
    }

 

根据先前设计的类图

每个 Pass 都有一个 Shader 的成员变量,所以 Pass 类还要添加 Shader 的成员变量,不仅如此,绘制时的图元类型也被设计在 Pass 中:

PrimType primType;
Shader* pShader;

 

整个 Pass 类的设计比较简单,源码在完成重构渲染器后给出。

 

posted @ 2017-07-02 16:08  为了邮箱5  阅读(1057)  评论(0编辑  收藏  举报