protot/src/modules/RenderModule.h

484 lines
9.3 KiB
C
Raw Normal View History

#pragma once
#include <cstdint>
#include <map>
#include <vector>
#include "math_types.h"
#include <bgfx/bgfx.h>
#include "RenderUtils.h"
struct Entity;
struct InputState {
int32_t mousedX;
int32_t mousedY;
int32_t mouseX;
int32_t mouseY;
uint8_t mouseButton;
int32_t mouseScroll;
char key;
InputState() :
mouseX(0),
mouseY(0),
mouseButton(0),
mouseScroll(0),
key(0) {
}
};
struct Camera {
Vector3f eye;
Vector3f poi;
Vector3f up;
float near;
float far;
float fov;
bool orthographic;
float width;
float height;
float mtxProj[16];
float mtxView[16];
float mtxEnv[16];
Camera() :
eye {5.f, 4.f, 5.f},
poi {0.f, 2.f, 0.f},
up {0.f, 1.f, 0.f},
near (0.1f),
far (150.f),
fov (60.f),
orthographic (false),
width (-1.f),
height (-1.f),
mtxProj {
1.f, 0.f, 0.f, 0.f,
0.f, 1.f, 0.f, 0.f,
0.f, 0.f, 1.f, 0.f,
0.f, 0.f, 0.f, 1.f},
mtxView {
1.f, 0.f, 0.f, 0.f,
0.f, 1.f, 0.f, 0.f,
0.f, 0.f, 1.f, 0.f,
0.f, 0.f, 0.f, 1.f},
mtxEnv {
1.f, 0.f, 0.f, 0.f,
0.f, 1.f, 0.f, 0.f,
0.f, 0.f, 1.f, 0.f,
0.f, 0.f, 0.f, 1.f}
{}
void updateMatrices();
};
struct Light {
bgfx::UniformHandle u_shadowMap;
bgfx::UniformHandle u_shadowMapParams;
bgfx::UniformHandle u_lightPos;
bgfx::UniformHandle u_lightMtx;
bgfx::TextureHandle shadowMapTexture;
bgfx::FrameBufferHandle shadowMapFB;
float pos[4];
float dir[3];
float mtxView[16];
float mtxProj[16];
float mtxLight[16];
float mtxShadow[16];
float shadowMapBias;
uint16_t shadowMapSize;
bool enabled;
float near;
float far;
float area;
Light() :
u_shadowMap (BGFX_INVALID_HANDLE),
u_lightPos (BGFX_INVALID_HANDLE),
u_lightMtx (BGFX_INVALID_HANDLE),
shadowMapTexture (BGFX_INVALID_HANDLE),
shadowMapFB (BGFX_INVALID_HANDLE),
pos {0.f, 10.f, 10.f, 1.0f},
dir {-1.f, -1.f, -1.f},
mtxView {
1.f, 0.f, 0.f, 0.f,
0.f, 1.f, 0.f, 0.f,
0.f, 0.f, 1.f, 0.f,
0.f, 0.f, 0.f, 1.f
},
mtxProj {
1.f, 0.f, 0.f, 0.f,
0.f, 1.f, 0.f, 0.f,
0.f, 0.f, 1.f, 0.f,
0.f, 0.f, 0.f, 1.f
},
mtxShadow {
1.f, 0.f, 0.f, 0.f,
0.f, 1.f, 0.f, 0.f,
0.f, 0.f, 1.f, 0.f,
0.f, 0.f, 0.f, 1.f
},
2016-11-10 21:29:47 +01:00
shadowMapBias (0.004f),
shadowMapSize (2048),
near (0.1f),
far (100.f),
area (10.f),
enabled (false)
{
}
};
struct Transform {
Quaternion rotation = Quaternion (0.0f, 0.0f, 0.0f, 1.0f);
Vector3f translation = Vector3f (0.0f, 0.0f, 0.0f);
Vector3f scale = Vector3f (1.0f, 1.0f, 1.0f);
Transform () {};
Transform (
const Vector3f &translation,
const Quaternion &rotation,
const Vector3f &scale)
:
translation(translation),
rotation(rotation),
scale(scale)
{}
Transform (const Matrix44f& mat) {
fromMatrix(mat);
}
Matrix44f toMatrix() const {
Matrix44f result;
Matrix33f scale_mat (
scale[0], 0.0f, 0.0f,
0.0f, scale[1], 0.0f,
0.0f, 0.0f, scale[2]
);
result.block<3,3>(0,0) = scale_mat * rotation.toMatrix();
result.block<1,3>(3,0) = translation.transpose();
result.block<3,1>(0,3) = Vector3f::Zero();
result(3,3) = 1.0f;
return result;
}
void fromMatrix(const Matrix44f &matrix) {
// Extract rotation matrix and the quaternion
Matrix33f rot_matrix (matrix.block<3,3>(0,0));
Vector3f row0 = rot_matrix.block<1,3>(0,0).transpose();
Vector3f row1 = rot_matrix.block<1,3>(1,0).transpose();
Vector3f row2 = rot_matrix.block<1,3>(2,0).transpose();
scale.set(
row0.norm(),
row1.norm(),
row2.norm()
);
rot_matrix.block<1,3>(0,0) = (row0 / scale[0]).transpose();
rot_matrix.block<1,3>(1,0) = (row1 / scale[1]).transpose();
rot_matrix.block<1,3>(2,0) = (row2 / scale[2]).transpose();
rotation = Quaternion::fromMatrix(rot_matrix).normalize();
row0 = rot_matrix.block<1,3>(0,0).transpose();
row1 = rot_matrix.block<1,3>(1,0).transpose();
row2 = rot_matrix.block<1,3>(2,0).transpose();
Vector3f trans (
matrix(3,0),
matrix(3,1),
matrix(3,2)
);
translation = trans;
}
Transform operator*(const Transform &other) const {
Matrix44f this_mat (toMatrix());
Matrix44f other_mat (other.toMatrix());
return Transform(this_mat * other_mat);
}
Vector3f operator*(const Vector3f &vec) const {
assert(false);
return Vector3f::Zero();
}
static Transform fromTrans(
const Vector3f &translation
) {
return Transform (
translation,
Quaternion(0.0f, 0.0f, 0.0f, 1.0f),
Vector3f(1.0f, 1.0f, 1.0f)
);
}
static Transform fromTransRot(
const Vector3f &translation,
const Quaternion &rotation
) {
return Transform (
translation,
rotation,
Vector3f(1.0f, 1.0f, 1.0f)
);
}
static Transform fromTransRotScale(
const Vector3f &translation,
const Quaternion &rotation,
const Vector3f &scale
) {
return Transform (translation, rotation, scale);
}
};
struct MeshHierarchy {
~MeshHierarchy()
{
for (bgfxutils::Mesh* mesh : meshes) {
meshUnload(mesh);
}
}
std::vector<bgfxutils::Mesh*> meshes;
/// index of the parent. Children must have higher indices than heir
// parents
std::vector<int> parent;
/// Transforms relative to their parents.
std::vector<Transform> localTransforms;
/// Absolute transforms.
std::vector<Matrix44f> meshMatrices;
void addMesh(
const int parent_idx,
const Transform& transform,
bgfxutils::Mesh* mesh) {
assert (parent_idx == -1 || parent_idx < parent.size());
parent.push_back(parent_idx);
localTransforms.push_back(transform);
meshes.push_back(mesh);
if (parent_idx != -1) {
meshMatrices.push_back(transform.toMatrix() * meshMatrices[parent_idx]);
} else {
meshMatrices.push_back(transform.toMatrix());
}
}
void updateMatrices(const Matrix44f &world_transform);
void submit(const RenderState *state);
};
struct Entity {
Transform transform;
float color[4];
MeshHierarchy mesh;
Entity() :
color { 1.f, 1.f, 1.f, 1.f }
{}
};
struct LightProbe
{
enum Enum
{
Bolonga,
Kyoto,
Count
};
void load(const char* _name);
void destroy()
{
bgfx::destroyTexture(m_tex);
bgfx::destroyTexture(m_texIrr);
}
bgfx::TextureHandle m_tex;
bgfx::TextureHandle m_texIrr;
};
2016-12-19 22:15:29 +01:00
struct Path {
std::vector<Vector3f> points;
std::vector<Vector4f> colors;
float thickness = 0.1f;
float miter = 0.0f;
bgfx::DynamicVertexBufferHandle mVertexBufferHandle = BGFX_INVALID_HANDLE;
bgfx::DynamicIndexBufferHandle mIndexBufferHandle = BGFX_INVALID_HANDLE;
~Path();
void UpdateBuffers();
2016-12-19 22:15:29 +01:00
};
2016-11-27 22:02:17 +01:00
struct DebugCommand {
enum CommandType {
Line,
Axes,
Sphere
2016-11-27 22:02:17 +01:00
};
CommandType type;
Vector3f from;
Vector3f to;
Vector4f color = Vector4f(1.f, 1.f, 1.f, 1.f);
2016-11-27 22:02:17 +01:00
};
struct Renderer {
bool initialized;
bool drawDebug;
2016-12-17 23:04:50 +01:00
bool drawFloor = true;
2016-12-19 22:15:29 +01:00
bool drawSkybox = true;
uint32_t width;
uint32_t height;
bgfx::UniformHandle sceneDefaultTextureSampler;
bgfx::TextureHandle sceneDefaultTexture;
std::vector<bgfxutils::Mesh*> meshes;
typedef std::map<std::string, unsigned int> MeshIdMap;
MeshIdMap meshIdMap;
LightProbe mLightProbes[LightProbe::Count];
LightProbe::Enum mCurrentLightProbe;
std::vector<Entity*> entities;
std::vector<Camera> cameras;
std::vector<Light> lights;
std::vector<Path> debugPaths;
2016-11-27 22:02:17 +01:00
std::vector<DebugCommand> debugCommands;
uint16_t activeCameraIndex;
// needed to forward inputs to IMGUI
InputState inputState;
Renderer() :
initialized(false),
drawDebug(false),
width (0),
height (0)
{}
// initialize simple geometries (cube, sphere, ...)
void createGeometries();
// create uniforms, load shaders, and create render targets
void setupShaders();
// setup renderpasses and wire up render targets
void setupRenderPasses();
void initialize(int width, int height);
void shutdown();
void paintGL();
void paintGLSimple();
void resize (int width, int height);
2016-12-17 23:04:50 +01:00
// check whether shader files were modified and reload them. Returns
// true on success, otherwise false
bool updateShaders();
Entity* createEntity();
bool destroyEntity (Entity* entity);
bgfxutils::Mesh* loadMesh(const char* filename);
2016-11-27 22:02:17 +01:00
// debug commands
void drawDebugLine (
const Vector3f &from,
const Vector3f &to,
const Vector3f &color);
void drawDebugAxes (
const Vector3f &pos,
const Matrix33f &orientation,
const float &scale);
void drawDebugSphere (
const Vector3f &pos,
float radius,
const Vector4f &color = Vector4f(1.f, 1.f, 1.f, 1.f));
};
2016-12-17 23:04:50 +01:00
struct RenderProgram {
bgfx::ProgramHandle program;
std::string vertexShaderFileName;
int vertexShaderFileModTime;
std::string fragmentShaderFileName;
int fragmentShaderFileModTime;
RenderProgram () :
vertexShaderFileName(""),
vertexShaderFileModTime(-1),
fragmentShaderFileName(""),
fragmentShaderFileModTime(-1)
{
program = BGFX_INVALID_HANDLE;
}
RenderProgram (
const char* vertex_shader_file_name,
const char* fragment_shader_file_name
)
:
vertexShaderFileName(vertex_shader_file_name),
vertexShaderFileModTime(-1),
fragmentShaderFileName(fragment_shader_file_name),
fragmentShaderFileModTime(-1)
{
program = BGFX_INVALID_HANDLE;
}
bool reload();
bool checkModified() const;
bool valid() const {
return bgfx::isValid(program);
}
};
struct RenderState {
enum {
Skybox,
ShadowMap,
Scene,
SceneTextured,
2016-12-17 23:04:50 +01:00
Lines,
LinesOccluded,
Debug,
Count
};
struct Texture {
uint32_t m_flags;
bgfx::UniformHandle m_sampler;
bgfx::TextureHandle m_texture;
uint8_t m_stage;
};
uint64_t m_state;
uint8_t m_numTextures;
2016-12-17 23:04:50 +01:00
RenderProgram m_program;
uint8_t m_viewId;
Texture m_textures[4];
};