自己封装的Shader类(后面继续改进)
shader.h
#pragma once
#include "sb7.h"
#include <string>
class Shader
{
public:
Shader();
Shader(const std::string vertexPath, const std::string fragmentPath);
//program对象使用
void use();
private:
GLuint program;
//源代码
std::string vertexString;
const char* vertexSource;
std::string fragmentString;
const char* fragmentSource;
private:
void checkCompileErrors(unsigned int ID, std::string type);
};
shader.cpp
#include "Shader.h"
#include <fstream>
#include <sstream>
#include <iostream>
#include <exception>
Shader::Shader(){}
Shader::Shader(const std::string vertexPath, const std::string fragmentPath)
{
//用于读取shader文件
std::ifstream vertexFile;
std::ifstream fragmentFile;
//转换源码
std::stringstream vertexSStream;
std::stringstream fragmentSStream;
//创建程序对象
program = glCreateProgram();
//vertex shader
if (!vertexPath.empty()) //判断文件路径是否为空
{
//打开文件
vertexFile.open(vertexPath);
//检测文件是否有误
vertexFile.exceptions(std::ifstream::badbit || std::ifstream::failbit);
try
{
if (!vertexFile.is_open()) //判断文件是否打开
{
throw std::exception("open vertex shader file fail");
}
//读取源代码到字符串流
vertexSStream << vertexFile.rdbuf();
//读取完毕,可以关闭文件了
vertexFile.close();
//将源代码输入到字符串中去
vertexString = vertexSStream.str();
//由于读取的是const char*类型的,所以用一个string转换一下
vertexSource = vertexString.c_str();
//创建顶点着色器对象
GLuint vertexShader;
vertexShader = glCreateShader(GL_VERTEX_SHADER);
//将shader的源代码传递给shader对象
glShaderSource(vertexShader, 1, &vertexSource, NULL);
//编译着色器中的源代码
glCompileShader(vertexShader);
//将shader对象附加到program对象上去
glAttachShader(program, vertexShader);
//连接program对象上的shader对象
glLinkProgram(program);
//查错
checkCompileErrors(program, "VERTEX");
//可以把已经附加并且链接好的shader对象删除了
glDeleteShader(vertexShader);
}
catch (const std::exception& e)
{
std::cout << e.what() << std::endl;
}
}
//fragment shader
if (!fragmentPath.empty()) //判断文件路径是否为空
{
//文件路径不为空,打开文件
fragmentFile.open(fragmentPath);
fragmentFile.exceptions(std::ifstream::badbit || std::ifstream::failbit);
try
{
if (!fragmentFile.is_open()) //文件打开失败
{
throw std::exception("open fragment shader file fail");
}
//文件成功打开
//字符串流读取文件中的数据
fragmentSStream << fragmentFile.rdbuf();
//读完了,把文件关了
fragmentFile.close();
//把源代码读入字符串中去
fragmentString = fragmentSStream.str();
//把string转成系数需要的const char *类型
fragmentSource = fragmentString.c_str();
//创建shader对象
GLuint fragmentShader;
fragmentShader = glCreateShader(GL_FRAGMENT_SHADER);
//将源代码读入shader对象
glShaderSource(fragmentShader, 1, &fragmentSource, NULL);
//编译shader对象中的源码
glCompileShader(fragmentShader);
//将shader对象附加到program对象上去
glAttachShader(program, fragmentShader);
//链接program对象上的shader对象
glLinkProgram(program);
//查错
checkCompileErrors(program, "PROGRAM");
//完事,可以把shader对象删了
glDeleteShader(fragmentShader);
}
catch (const std::exception& e)
{
std::cout << e.what() << std::endl;
}
}
}
void Shader::use()
{
glUseProgram(program);
}
/*
* 查错
*/
void Shader::checkCompileErrors(unsigned int ID, std::string type)
{
int success;
char infoLog[512];
if (type != "PROGRAM")
{
glGetShaderiv(ID, GL_COMPILE_STATUS, &success);
if (!success)
{
glGetShaderInfoLog(ID, 512, NULL, infoLog);
std::cout << "shader compile error: " << infoLog << std::endl;
}
}
else
{
glGetProgramiv(ID, GL_LINK_STATUS, &success);
if (!success)
{
glGetProgramInfoLog(ID, 512, NULL, infoLog);
std::cout << "program linking error: " << infoLog << std::endl;
}
}
}
shader
vertex shader
#version 450 core
//VAO的第0个位置的数据
layout(location = 0) in vec4 offset;
//VAO的第1个位置的数据
layout(location = 1) in vec4 color;
//输出的数据,输出到下一个shader
out vec4 vs_color;
void main()
{
const vec4 vertices[3] = vec4[3](vec4(0.25, -0.25, 0.5, 1.0),
vec4(-0.25, -0.25, 0.5, 1.0),
vec4(0.25, 0.25, 0.5, 1.0));
vs_color = color;
//gl_VertexID是内置对象,每次自动+1,直到数组的大小为止,offset是偏移量,这里是一个椭圆的参数方程
gl_Position = vertices[gl_VertexID] + offset;
}
fragment shader
#version 450 core
//从vertex shader传过来的数据,in就是上一个shader的out
in vec4 vs_color;
//输出到屏幕或者窗口
out vec4 color;
void main()
{
color = vs_color;
}
代码
#include "sb7.h"
#include "Shader.h"
#include <math.h>
class my_application : public sb7::application
{
void startup()
{
shader = new Shader("vertexShader.vert", "fragShader.frag");
//创建数组对象
glCreateVertexArrays(2, vertex_arrays_object);
//绑定array对象到OpenGL上下文,让OpenGL知道这些数据
glBindVertexArray(vertex_arrays_object[0]);
glBindVertexArray(vertex_arrays_object[1]);
}
void render(double currentTime)
{
const GLfloat color[] = {
(float)sin(currentTime) * 0.5f + 0.5f,
(float)cos(currentTime) * 0.5f + 0.5f, 0.0f, 1.0f
};
glClearBufferfv(GL_COLOR, 0, color);
shader->use();
//三角形的位置变化,这个就是vertex shader传入的offset
GLfloat attrib[]{
(float)sin(currentTime) * 0.5f,
(float)cos(currentTime) * 0.6f, 0.0, 0.0f
};
//三角形的颜色变化,这个是vertex shader里面传入的color
GLfloat triangleColor[] = {
(float)sin(currentTime) * 0.2f + 0.5f,
(float)cos(currentTime) * 0.2f + 0.5f, 0.0f, 1.0f
};
//使用glVertexAttrub4fv(GLuint index, const GLfloat* v)填充变量,这里的index与vertex shader里面的location对应
glVertexAttrib4fv(0, attrib);
glVertexAttrib4fv(1, triangleColor);
glDrawArrays(GL_TRIANGLES, 0, 3);
}
void shutdown()
{
}
private:
Shader* shader;
//顶点数组对象
GLuint vertex_arrays_object[2];
};
DECLARE_MAIN(my_application);
出现过的问题
忽略了vao的创建和绑定
//创建数组对象 glCreateVertexArrays(2, vertex_arrays_object); //绑定array对象到OpenGL上下文,让OpenGL知道这些数据 glBindVertexArray(vertex_arrays_object[0]); glBindVertexArray(vertex_arrays_object[1]);
一开始我没有写上面的代码,导致一直无法绘制出图形,并且出现错误。
GL_INVALID_OPERATION error generated. Array object is not active.