2013-12-21 22:48:54 +00:00
|
|
|
#include "OpenGLShaderProgram.h"
|
|
|
|
|
|
|
|
#include <QOpenGLShaderProgram>
|
2013-12-21 23:10:18 +00:00
|
|
|
#include <QOpenGLFunctions_3_2_Core>
|
2013-12-21 22:48:54 +00:00
|
|
|
#include <QDir>
|
2013-12-22 14:04:33 +00:00
|
|
|
#include "OpenGLRenderer.h"
|
|
|
|
#include "OpenGLSharedState.h"
|
2013-12-21 22:48:54 +00:00
|
|
|
#include "Texture2D.h"
|
|
|
|
#include "Texture3D.h"
|
|
|
|
#include "Texture4D.h"
|
|
|
|
#include "Color.h"
|
|
|
|
|
2013-12-22 14:04:33 +00:00
|
|
|
OpenGLShaderProgram::OpenGLShaderProgram(QString name, OpenGLRenderer* renderer):
|
|
|
|
renderer(renderer), name(name)
|
2013-12-21 22:48:54 +00:00
|
|
|
{
|
|
|
|
program = new QOpenGLShaderProgram();
|
2013-12-22 14:04:33 +00:00
|
|
|
functions = renderer->getOpenGlFunctions();
|
2013-12-21 22:48:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
OpenGLShaderProgram::~OpenGLShaderProgram()
|
|
|
|
{
|
|
|
|
delete program;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OpenGLShaderProgram::addVertexSource(QString path)
|
|
|
|
{
|
|
|
|
program->addShaderFromSourceFile(QOpenGLShader::Vertex, QString(":/shaders/%1.vert").arg(path));
|
|
|
|
}
|
|
|
|
|
|
|
|
void OpenGLShaderProgram::addFragmentSource(QString path)
|
|
|
|
{
|
|
|
|
program->addShaderFromSourceFile(QOpenGLShader::Fragment, QString(":/shaders/%1.frag").arg(path));
|
|
|
|
}
|
|
|
|
|
|
|
|
void OpenGLShaderProgram::compile()
|
|
|
|
{
|
|
|
|
if (not program->link())
|
|
|
|
{
|
|
|
|
qWarning() << "Error while compiling shader " << name << "\n" << program->log() << "\n";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
qDebug() << "Shader " << name << " compilation output:\n" << program->log() << "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void OpenGLShaderProgram::addTexture(QString sampler_name, Texture2D* texture)
|
|
|
|
{
|
|
|
|
GLuint texid;
|
|
|
|
|
|
|
|
if (textures.contains(sampler_name))
|
|
|
|
{
|
|
|
|
texid = textures[sampler_name].second;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-12-21 23:10:18 +00:00
|
|
|
functions->glGenTextures(1, &texid);
|
2013-12-21 22:48:54 +00:00
|
|
|
textures[sampler_name] = QPair<int, unsigned int>(GL_TEXTURE_2D, texid);
|
|
|
|
}
|
|
|
|
|
2013-12-21 23:10:18 +00:00
|
|
|
functions->glBindTexture(GL_TEXTURE_2D, texid);
|
|
|
|
functions->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
|
|
|
functions->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
|
|
|
functions->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
|
|
|
|
functions->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
|
2013-12-21 22:48:54 +00:00
|
|
|
|
|
|
|
int sx, sy;
|
|
|
|
texture->getSize(&sx, &sy);
|
|
|
|
float* pixels = new float[sx * sy * 4];
|
|
|
|
for (int x = 0; x < sx; x++)
|
|
|
|
{
|
|
|
|
for (int y = 0; y < sy; y++)
|
|
|
|
{
|
|
|
|
float* pixel = pixels + (y * sx + x) * 4;
|
|
|
|
Color col = texture->getPixel(x, y);
|
|
|
|
pixel[0] = (float)col.r;
|
|
|
|
pixel[1] = (float)col.g;
|
|
|
|
pixel[2] = (float)col.b;
|
|
|
|
pixel[3] = (float)col.a;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-21 23:10:18 +00:00
|
|
|
functions->glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, sx, sy, 0, GL_RGBA, GL_FLOAT, pixels);
|
2013-12-21 22:48:54 +00:00
|
|
|
delete[] pixels;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OpenGLShaderProgram::addTexture(QString sampler_name, Texture3D* texture)
|
|
|
|
{
|
|
|
|
GLuint texid;
|
|
|
|
|
|
|
|
if (textures.contains(sampler_name))
|
|
|
|
{
|
|
|
|
texid = textures[sampler_name].second;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-12-21 23:10:18 +00:00
|
|
|
functions->glGenTextures(1, &texid);
|
2013-12-21 22:48:54 +00:00
|
|
|
textures[sampler_name] = QPair<int, unsigned int>(GL_TEXTURE_3D, texid);
|
|
|
|
}
|
|
|
|
|
2013-12-21 23:10:18 +00:00
|
|
|
functions->glBindTexture(GL_TEXTURE_3D, texid);
|
|
|
|
functions->glTexParameteri(GL_TEXTURE_3D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
|
|
|
functions->glTexParameteri(GL_TEXTURE_3D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
|
|
|
functions->glTexParameteri(GL_TEXTURE_3D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
|
|
|
|
functions->glTexParameteri(GL_TEXTURE_3D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
|
|
|
|
functions->glTexParameteri(GL_TEXTURE_3D, GL_TEXTURE_WRAP_R, GL_CLAMP_TO_EDGE);
|
2013-12-21 22:48:54 +00:00
|
|
|
|
|
|
|
int sx, sy, sz;
|
|
|
|
texture->getSize(&sx, &sy, &sz);
|
|
|
|
float* pixels = new float[sx * sy * sz * 4];
|
|
|
|
for (int x = 0; x < sx; x++)
|
|
|
|
{
|
|
|
|
for (int y = 0; y < sy; y++)
|
|
|
|
{
|
|
|
|
for (int z = 0; z < sz; z++)
|
|
|
|
{
|
|
|
|
float* pixel = pixels + (z * (sx * sy) + y * sx + x) * 4;
|
|
|
|
Color col = texture->getPixel(x, y, z);
|
|
|
|
pixel[0] = (float)col.r;
|
|
|
|
pixel[1] = (float)col.g;
|
|
|
|
pixel[2] = (float)col.b;
|
|
|
|
pixel[3] = (float)col.a;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-21 23:10:18 +00:00
|
|
|
functions->glTexImage3D(GL_TEXTURE_3D, 0, GL_RGBA, sx, sy, sz, 0, GL_RGBA, GL_FLOAT, pixels);
|
2013-12-21 22:48:54 +00:00
|
|
|
delete[] pixels;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OpenGLShaderProgram::addTexture(QString sampler_name, Texture4D* texture)
|
|
|
|
{
|
|
|
|
GLuint texid;
|
|
|
|
|
|
|
|
if (textures.contains(sampler_name))
|
|
|
|
{
|
|
|
|
texid = textures[sampler_name].second;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-12-21 23:10:18 +00:00
|
|
|
functions->glGenTextures(1, &texid);
|
2013-12-21 22:48:54 +00:00
|
|
|
textures[sampler_name] = QPair<int, unsigned int>(GL_TEXTURE_3D, texid);
|
|
|
|
}
|
|
|
|
|
2013-12-21 23:10:18 +00:00
|
|
|
functions->glBindTexture(GL_TEXTURE_3D, texid);
|
|
|
|
functions->glTexParameteri(GL_TEXTURE_3D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
|
|
|
functions->glTexParameteri(GL_TEXTURE_3D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
|
|
|
functions->glTexParameteri(GL_TEXTURE_3D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
|
|
|
|
functions->glTexParameteri(GL_TEXTURE_3D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
|
|
|
|
functions->glTexParameteri(GL_TEXTURE_3D, GL_TEXTURE_WRAP_R, GL_CLAMP_TO_EDGE);
|
2013-12-21 22:48:54 +00:00
|
|
|
|
|
|
|
int sx, sy, sz, sw;
|
|
|
|
texture->getSize(&sx, &sy, &sz, &sw);
|
|
|
|
float* pixels = new float[sx * sy * sz * sw * 4];
|
|
|
|
for (int x = 0; x < sx; x++)
|
|
|
|
{
|
|
|
|
for (int y = 0; y < sy; y++)
|
|
|
|
{
|
|
|
|
for (int z = 0; z < sz; z++)
|
|
|
|
{
|
|
|
|
for (int w = 0; w < sw; w++)
|
|
|
|
{
|
|
|
|
float* pixel = pixels + (w * (sx * sy * sz) + z * (sx * sy) + y * sx + x) * 4;
|
|
|
|
Color col = texture->getPixel(x, y, z, w);
|
|
|
|
pixel[0] = (float)col.r;
|
|
|
|
pixel[1] = (float)col.g;
|
|
|
|
pixel[2] = (float)col.b;
|
|
|
|
pixel[3] = (float)col.a;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-21 23:10:18 +00:00
|
|
|
functions->glTexImage3D(GL_TEXTURE_3D, 0, GL_RGBA, sx, sy, sz * sw, 0, GL_RGBA, GL_FLOAT, pixels);
|
2013-12-21 22:48:54 +00:00
|
|
|
delete[] pixels;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OpenGLShaderProgram::bind()
|
|
|
|
{
|
|
|
|
program->bind();
|
|
|
|
|
2013-12-22 14:04:33 +00:00
|
|
|
renderer->getSharedState()->apply(this);
|
|
|
|
|
2013-12-21 22:48:54 +00:00
|
|
|
QMapIterator<QString, QPair<int, unsigned int> > iter(textures);
|
|
|
|
int i = 0;
|
|
|
|
while (iter.hasNext())
|
|
|
|
{
|
|
|
|
iter.next();
|
|
|
|
int textureSampler = program->uniformLocation(iter.key());
|
|
|
|
if (textureSampler >= 0)
|
|
|
|
{
|
2013-12-21 23:10:18 +00:00
|
|
|
functions->glActiveTexture(GL_TEXTURE0 + i);
|
|
|
|
functions->glBindTexture(iter.value().first, iter.value().second);
|
2013-12-21 22:48:54 +00:00
|
|
|
program->setUniformValue(textureSampler, i);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void OpenGLShaderProgram::release()
|
|
|
|
{
|
|
|
|
program->release();
|
|
|
|
}
|
|
|
|
|
|
|
|
void OpenGLShaderProgram::drawTriangles(float* vertices, int triangle_count)
|
|
|
|
{
|
|
|
|
bind();
|
|
|
|
|
|
|
|
GLuint vertex = program->attributeLocation("vertex");
|
|
|
|
program->setAttributeArray(vertex, GL_FLOAT, vertices, 3);
|
|
|
|
program->enableAttributeArray(vertex);
|
|
|
|
|
2013-12-21 23:10:18 +00:00
|
|
|
functions->glDrawArrays(GL_TRIANGLES, 0, triangle_count * 3);
|
2013-12-21 22:48:54 +00:00
|
|
|
|
|
|
|
program->disableAttributeArray(vertex);
|
|
|
|
|
|
|
|
release();
|
|
|
|
}
|
|
|
|
|
|
|
|
void OpenGLShaderProgram::drawTriangleStrip(float* vertices, int vertex_count)
|
|
|
|
{
|
|
|
|
bind();
|
|
|
|
|
|
|
|
GLuint vertex = program->attributeLocation("vertex");
|
|
|
|
program->setAttributeArray(vertex, GL_FLOAT, vertices, 3);
|
|
|
|
program->enableAttributeArray(vertex);
|
|
|
|
|
2013-12-21 23:10:18 +00:00
|
|
|
functions->glDrawArrays(GL_TRIANGLE_STRIP, 0, vertex_count);
|
2013-12-21 22:48:54 +00:00
|
|
|
|
|
|
|
program->disableAttributeArray(vertex);
|
|
|
|
|
|
|
|
release();
|
|
|
|
}
|