#pragma once //DGL #include "DGL_Enum.hpp" #include "DGL_Types.hpp" #include "DGL_Buffers.hpp" #include "DGL_Space.hpp" // Non-Standard C++ #include "Cpp_Alias.hpp" namespace DGL { // Forward Declarations sfn GetShaderInfo ( ID _shader , gSize _logLength , ptr _infoLengthRef , RawString _infoLogRef ) -> void; sfn QueryShader ( ID _shaderToQuery , EShaderInfo _shaderInfoDesired, ptr _requestedInfoObject ) -> void; sfn MakeShader (Ref(ID) _shaderIDHolder , EShaderType _typeOfShader , uInt64 _numberOfStringElements , ro ptr> _sourceCode, ro ptr _lengthsOfStrings) -> void; sfn MakeShaderProgram(Ref(ID) _shaderProgramIDHolder,ro ID _vertexShader , ro ID _fragShader ) -> void; // Functions sfn ActiveUniforms(ID _shaderToQueryForUniforms) -> GLint { GLint uniforms; glGetProgramiv(_shaderToQueryForUniforms, GL_ACTIVE_UNIFORMS, Address(uniforms)); for (int uniformIndex = 0; uniformIndex < uniforms; uniformIndex++) { int name_len = -1, num = -1; GLenum type = GL_ZERO; char name[100]; glGetActiveUniform(_shaderToQueryForUniforms, GLuint(uniformIndex), sizeof(name) - 1, Address(name_len), Address(num), Address(type), name); name[name_len] = 0; } return uniforms; } sfn AttachShader(ID _shaderProgram, ID _shaderToAttach) { glAttachShader(_shaderProgram, _shaderToAttach); return; } sfn BindShaderSource(ID _shader, gSize _numberOfStringElements, ro ptr> _sourceCode, ro ptr _lengthsOfStrings) { glShaderSource(_shader, _numberOfStringElements, _sourceCode, _lengthsOfStrings); //Address(_sourceCode), Address(_lengthsOfStrings)); return; } sfn CompileShader(ID _shaderToCompile) { glCompileShader(_shaderToCompile); gInt Result = gInt(EBool::False); gSize InfoLogLength; QueryShader(_shaderToCompile, EShaderInfo::CompileStatus, Address(Result)); if (!Result) { QueryShader(_shaderToCompile, EShaderInfo::InfoLogLength, Address(InfoLogLength)); if (InfoLogLength > 0) { std::vector ErrorMessage(InfoLogLength + 1); GetShaderInfo(_shaderToCompile, InfoLogLength, nullptr, Address(ErrorMessage.at(0))); throw std::runtime_error(Address(ErrorMessage.at(0))); } else { throw std::runtime_error("Shader compilation failed and did not get a proper info log."); } } return; } sfn CreateShader(EShaderType _typeOfShader) -> ID { return glCreateShader(GLenum(_typeOfShader)); } sfn CreateShaderProgram() -> ID { return glCreateProgram(); } sfn DeleteShader(ID _shaderToDelete) { glDeleteShader(_shaderToDelete); return; } sfn DetachShader(ID _shaderProgram, ID _shaderToDetach) { glDetachShader(_shaderProgram, _shaderToDetach); return; } sfn GetShaderInfo(ID _shader, gSize _logLength, ro ptr _infoLengthRef, RawString _infoLogRef) -> void { glGetShaderInfoLog(_shader, _logLength, _infoLengthRef, _infoLogRef); return; } sfn GetShaderProgramInfo(ID _shaderProgram, gSize _logLength, ro ptr _infoLengthRef, RawString _infoLogRef) -> void { glGetProgramInfoLog(_shaderProgram, _logLength, _infoLengthRef, _infoLogRef); return; } sfn GetUniformVariable(const ID _programID, ro RawString _nameOfVariable) -> ID { return glGetUniformLocation(_programID, _nameOfVariable); } sfn LinkProgramShader(ID _shaderProgramToLink) { glLinkProgram(_shaderProgramToLink); return; } sfn QueryShader(ID _shaderToQuery, EShaderInfo _shaderInfoDesired, ptr _requestedInfoObject) -> void { glGetShaderiv(_shaderToQuery, GLenum(_shaderInfoDesired), _requestedInfoObject); return; } sfn QueryShaderProgram(ID _shaderToQuery, EShaderProgramInfo _shaderProgramInfoDesired, ptr _requestedInfoObject) -> void { glGetProgramiv(_shaderToQuery, GLenum(_shaderProgramInfoDesired), _requestedInfoObject); return; } sfn LoadShaders(RawString _vertexShaderFilePath, RawString _fragmentShaderFilePath) -> ID { using std::vector; string vertexShaderCode ; string fragmentShaderCode ; ifstream vertexShaderFileStream ; ifstream fragmentShaderFileStream; vertexShaderFileStream .open(_vertexShaderFilePath); fragmentShaderFileStream.open(_fragmentShaderFilePath); if (vertexShaderFileStream.is_open() and fragmentShaderFileStream.is_open()) { stringstream vertSourceStrStream; stringstream fragSourceStrStream; vertSourceStrStream << vertexShaderFileStream .rdbuf(); fragSourceStrStream << fragmentShaderFileStream.rdbuf(); vertexShaderFileStream .close(); fragmentShaderFileStream.close(); vertexShaderCode = vertSourceStrStream.str(); fragmentShaderCode = fragSourceStrStream.str(); } else { throw std::runtime_error("Impossible to open% s.Are you in the right directory ? Don't forget to read the FAQ !"); } RawString vertexSourcePtr = vertexShaderCode .c_str(); RawString fragmentSourcePtr = fragmentShaderCode.c_str(); cout << "Compiling shader: " << _vertexShaderFilePath << endl; ID vertexShader = CreateShader(EShaderType::Vertex); BindShaderSource(vertexShader, 1, Address(vertexSourcePtr), NULL); CompileShader (vertexShader ); cout << "Compiling shader: " << _fragmentShaderFilePath << endl; ID fragmentShader = CreateShader(EShaderType::Fragment); BindShaderSource(fragmentShader, 1, Address(fragmentSourcePtr), NULL); CompileShader (fragmentShader ); cout << "Making Shader Program and Linking..." << endl; ID generatedProgram; MakeShaderProgram(generatedProgram, vertexShader, fragmentShader); DeleteShader(vertexShader ); DeleteShader(fragmentShader); return generatedProgram; } sfn MakeShader ( Ref(ID ) _shaderIDHolder , EShaderType _typeOfShader , gSize _numberOfStringElements, ro ptr> _sourceCode , ro ptr< gInt > _lengthsOfStrings ) -> void { _shaderIDHolder = CreateShader(_typeOfShader); BindShaderSource(_shaderIDHolder, _numberOfStringElements, _sourceCode, _lengthsOfStrings); CompileShader(_shaderIDHolder); return; } sfn MakeShaderProgram(Ref(ID) _shaderProgramIDHolder, ID _vertexShader, ID _fragShader) -> void { _shaderProgramIDHolder = CreateShaderProgram(); AttachShader(_shaderProgramIDHolder, _vertexShader); AttachShader(_shaderProgramIDHolder, _fragShader ); LinkProgramShader(_shaderProgramIDHolder); gInt Result = false; QueryShaderProgram(_shaderProgramIDHolder, EShaderProgramInfo::LinkStatus, Address(Result)); if (!Result) { gInt infoLogLength; QueryShaderProgram(_shaderProgramIDHolder, EShaderProgramInfo::InfoLogLength, Address(infoLogLength)); if (infoLogLength > 0) { std::vector ErrorMessage(infoLogLength + 1); GetShaderProgramInfo(_shaderProgramIDHolder, infoLogLength, NULL, Address(ErrorMessage.at(0))); cout << (char*)&ErrorMessage[0] << endl; throw std::runtime_error(Address(ErrorMessage.at(0))); } else { throw std::runtime_error("ShaderProgram compilation failed and did not get a proper info log."); } } DetachShader(_shaderProgramIDHolder, _vertexShader); DetachShader(_shaderProgramIDHolder, _fragShader ); return; } // MVA: MatrixVariableArray sfn SetUniformVariable_MVA(ID _matrixID, gSize _numMatricies, gBoolean _shouldTransposeValues, ro Ref(gFloat) _dataPtr) { glUniformMatrix4fv(_matrixID, _numMatricies, _shouldTransposeValues, Address(_dataPtr)); return; } sfn SetUniformVariable_Vector3(ID _ID, gSize _num, ro Ref(gFloat) _dataPtr) { glUniform3fv(_ID, _num, Address(_dataPtr)); return; } sfn SetUniformVariable_Float(ID _ID, gFloat _data) { glUniform1f(_ID, _data); return; } sfn UseProgramShader(ID _shaderProgramToUse) { glUseProgram(_shaderProgramToUse); return; } namespace Basic_LightShader { ID Shader; ID ScreenSpaceVarID; sfn LoadShader() { Shader = LoadShaders("./Shaders/BasicLight.vert", "./Shaders/BasicLight.frag"); ScreenSpaceVarID = GetUniformVariable(Shader, "ScreenSpaceTransform"); return; } sfn Use(ro Ref(CoordSpace) _lampTransform) { UseProgramShader(Shader); SetUniformVariable_MVA(ScreenSpaceVarID, 1, false, _lampTransform[0][0]); EnableVertexAttributeArray(0); EnableVertexAttributeArray(1); return; } sfn Stop() { DisableVertexAttributeArray(0); DisableVertexAttributeArray(1); return; } } struct Material_Phong { VecColor Color ; gFloat Ambience; gFloat Diffuse ; gFloat Specular; }; namespace PhongShader { ID ShaderID; ID ModelSpaceID, InverseModelSpaceID, ViewportID, ProjectionID; ID LightPositionID, ObjectColorID, LightColorID; ID AmbientStrengthID, DiffuseStrengthID, SpecularStrengthID; gInt VertexIndexID, NormalIndexID; sfn LoadShader() { ShaderID = LoadShaders("./Shaders/PhongShader.vert", "./Shaders/PhongShader.frag"); InverseModelSpaceID = GetUniformVariable(ShaderID, "InverseModelSpace"); ModelSpaceID = GetUniformVariable(ShaderID, "ModelSpace" ); ProjectionID = GetUniformVariable(ShaderID, "Projection" ); ViewportID = GetUniformVariable(ShaderID, "Viewport" ); ObjectColorID = GetUniformVariable(ShaderID, "ObjectColor" ); LightColorID = GetUniformVariable(ShaderID, "LightColor" ); LightPositionID = GetUniformVariable(ShaderID, "LightPosition"); AmbientStrengthID = GetUniformVariable(ShaderID, "AmbientStrength" ); DiffuseStrengthID = GetUniformVariable(ShaderID, "DiffuseStrength" ); SpecularStrengthID = GetUniformVariable(ShaderID, "SpecularStrength"); return; } sfn Use ( ro Ref(CoordSpace ) _projection , ro Ref(CoordSpace ) _viewport , ro Ref(CoordSpace ) _objectTransform , ro Ref(Vector3 ) _lightPosition , ro Ref(Vector3 ) _lightColor , ro Ref(Material_Phong) _material ) { CoordSpace inverseTransform = Inverse(_viewport * _objectTransform); UseProgramShader(ShaderID); SetUniformVariable_MVA(InverseModelSpaceID, 1, false, inverseTransform[0][0]); SetUniformVariable_MVA(ModelSpaceID , 1, false, _objectTransform[0][0]); SetUniformVariable_MVA(ProjectionID , 1, false, _projection [0][0]); SetUniformVariable_MVA(ViewportID , 1, false, _viewport [0][0]); SetUniformVariable_Vector3(LightPositionID, 1, _lightPosition[0]); SetUniformVariable_Vector3(ObjectColorID , 1, _material.Color[0]); SetUniformVariable_Vector3(LightColorID , 1, _lightColor [0]); SetUniformVariable_Float(AmbientStrengthID , _material.Ambience); SetUniformVariable_Float(DiffuseStrengthID , _material.Diffuse ); SetUniformVariable_Float(SpecularStrengthID, _material.Specular); EnableVertexAttributeArray(0); EnableVertexAttributeArray(1); return; } sfn Stop() { DisableVertexAttributeArray(1); DisableVertexAttributeArray(0); return; } } sfn LoadDefaultShaders() { Basic_LightShader::LoadShader(); PhongShader ::LoadShader(); return; } }