mirror of
https://github.com/MaSzyna-EU07/maszyna.git
synced 2026-03-22 15:05:03 +01:00
408 lines
14 KiB
C++
408 lines
14 KiB
C++
/*
|
|
This Source Code Form is subject to the
|
|
terms of the Mozilla Public License, v.
|
|
2.0. If a copy of the MPL was not
|
|
distributed with this file, You can
|
|
obtain one at
|
|
http://mozilla.org/MPL/2.0/.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include "GL/glew.h"
|
|
#include "openglgeometrybank.h"
|
|
#include "material.h"
|
|
#include "lightarray.h"
|
|
#include "dumb3d.h"
|
|
#include "frustum.h"
|
|
#include "scene.h"
|
|
#include "world.h"
|
|
#include "memcell.h"
|
|
|
|
#define EU07_USE_PICKING_FRAMEBUFFER
|
|
//#define EU07_USE_DEBUG_SHADOWMAP
|
|
//#define EU07_USE_DEBUG_CAMERA
|
|
//#define EU07_USE_DEBUG_CULLING
|
|
|
|
struct opengl_light {
|
|
|
|
GLuint id{ (GLuint)-1 };
|
|
glm::vec3 direction;
|
|
glm::vec4
|
|
position { 0.f, 0.f, 0.f, 1.f }, // 4th parameter specifies directional(0) or omni-directional(1) light source
|
|
ambient { 0.f, 0.f, 0.f, 1.f },
|
|
diffuse { 1.f, 1.f, 1.f, 1.f },
|
|
specular { 1.f, 1.f, 1.f, 1.f };
|
|
|
|
inline
|
|
void apply_intensity( float const Factor = 1.0f ) {
|
|
|
|
if( Factor == 1.0 ) {
|
|
|
|
glLightfv( id, GL_AMBIENT, glm::value_ptr(ambient) );
|
|
glLightfv( id, GL_DIFFUSE, glm::value_ptr(diffuse) );
|
|
glLightfv( id, GL_SPECULAR, glm::value_ptr(specular) );
|
|
}
|
|
else {
|
|
// temporary light scaling mechanics (ultimately this work will be left to the shaders
|
|
glm::vec4 scaledambient( ambient.r * Factor, ambient.g * Factor, ambient.b * Factor, ambient.a );
|
|
glm::vec4 scaleddiffuse( diffuse.r * Factor, diffuse.g * Factor, diffuse.b * Factor, diffuse.a );
|
|
glm::vec4 scaledspecular( specular.r * Factor, specular.g * Factor, specular.b * Factor, specular.a );
|
|
glLightfv( id, GL_AMBIENT, glm::value_ptr(scaledambient) );
|
|
glLightfv( id, GL_DIFFUSE, glm::value_ptr(scaleddiffuse) );
|
|
glLightfv( id, GL_SPECULAR, glm::value_ptr(scaledspecular) );
|
|
}
|
|
}
|
|
inline
|
|
void apply_angle() {
|
|
|
|
glLightfv( id, GL_POSITION, glm::value_ptr(position) );
|
|
if( position.w == 1.f ) {
|
|
glLightfv( id, GL_SPOT_DIRECTION, glm::value_ptr(direction) );
|
|
}
|
|
}
|
|
inline
|
|
void set_position( glm::vec3 const &Position ) {
|
|
|
|
position = glm::vec4( Position, position.w );
|
|
}
|
|
};
|
|
|
|
// encapsulates basic rendering setup.
|
|
// for modern opengl this translates to a specific collection of glsl shaders,
|
|
// for legacy opengl this is combination of blending modes, active texture units etc
|
|
struct opengl_technique {
|
|
|
|
};
|
|
|
|
// simple camera object. paired with 'virtual camera' in the scene
|
|
class opengl_camera {
|
|
|
|
public:
|
|
// methods:
|
|
inline
|
|
void
|
|
update_frustum() { update_frustum( m_projection, m_modelview ); }
|
|
void
|
|
update_frustum( glm::mat4 const &Projection, glm::mat4 const &Modelview );
|
|
bool
|
|
visible( scene::bounding_area const &Area ) const;
|
|
bool
|
|
visible( TDynamicObject const *Dynamic ) const;
|
|
inline
|
|
glm::dvec3 const &
|
|
position() const { return m_position; }
|
|
inline
|
|
glm::dvec3 &
|
|
position() { return m_position; }
|
|
inline
|
|
glm::mat4 const &
|
|
projection() const { return m_projection; }
|
|
inline
|
|
glm::mat4 &
|
|
projection() { return m_projection; }
|
|
inline
|
|
glm::mat4 const &
|
|
modelview() const { return m_modelview; }
|
|
inline
|
|
glm::mat4 &
|
|
modelview() { return m_modelview; }
|
|
inline
|
|
std::vector<glm::vec4> &
|
|
frustum_points() { return m_frustumpoints; }
|
|
// transforms provided set of clip space points to world space
|
|
template <class Iterator_>
|
|
void
|
|
transform_to_world( Iterator_ First, Iterator_ Last ) const {
|
|
std::for_each(
|
|
First, Last,
|
|
[this]( glm::vec4 &point ) {
|
|
// transform each point using the cached matrix...
|
|
point = this->m_inversetransformation * point;
|
|
// ...and scale by transformed w
|
|
point = glm::vec4{ glm::vec3{ point } / point.w, 1.f }; } ); }
|
|
// debug helper, draws shape of frustum in world space
|
|
void
|
|
draw( glm::vec3 const &Offset ) const;
|
|
|
|
private:
|
|
// members:
|
|
cFrustum m_frustum;
|
|
std::vector<glm::vec4> m_frustumpoints; // visualization helper; corners of defined frustum, in world space
|
|
glm::dvec3 m_position;
|
|
glm::mat4 m_projection;
|
|
glm::mat4 m_modelview;
|
|
glm::mat4 m_inversetransformation; // cached transformation to world space
|
|
};
|
|
|
|
// bare-bones render controller, in lack of anything better yet
|
|
class opengl_renderer {
|
|
|
|
public:
|
|
// types
|
|
|
|
// destructor
|
|
~opengl_renderer() { gluDeleteQuadric( m_quadric ); }
|
|
|
|
// methods
|
|
bool
|
|
Init( GLFWwindow *Window );
|
|
// main draw call. returns false on error
|
|
bool
|
|
Render();
|
|
inline
|
|
float
|
|
Framerate() { return m_framerate; }
|
|
// geometry methods
|
|
// NOTE: hands-on geometry management is exposed as a temporary measure; ultimately all visualization data should be generated/handled automatically by the renderer itself
|
|
// creates a new geometry bank. returns: handle to the bank or NULL
|
|
gfx::geometrybank_handle
|
|
Create_Bank();
|
|
// creates a new geometry chunk of specified type from supplied vertex data, in specified bank. returns: handle to the chunk or NULL
|
|
gfx::geometry_handle
|
|
Insert( gfx::vertex_array &Vertices, gfx::geometrybank_handle const &Geometry, int const Type );
|
|
// replaces data of specified chunk with the supplied vertex data, starting from specified offset
|
|
bool
|
|
Replace( gfx::vertex_array &Vertices, gfx::geometry_handle const &Geometry, std::size_t const Offset = 0 );
|
|
// adds supplied vertex data at the end of specified chunk
|
|
bool
|
|
Append( gfx::vertex_array &Vertices, gfx::geometry_handle const &Geometry );
|
|
// provides direct access to vertex data of specfied chunk
|
|
gfx::vertex_array const &
|
|
Vertices( gfx::geometry_handle const &Geometry ) const;
|
|
// material methods
|
|
material_handle
|
|
Fetch_Material( std::string const &Filename, bool const Loadnow = true );
|
|
void
|
|
Bind_Material( material_handle const Material );
|
|
opengl_material const &
|
|
Material( material_handle const Material ) const;
|
|
// texture methods
|
|
texture_handle
|
|
Fetch_Texture( std::string const &Filename, bool const Loadnow = true );
|
|
void
|
|
Bind_Texture( texture_handle const Texture );
|
|
opengl_texture const &
|
|
Texture( texture_handle const Texture ) const;
|
|
// light methods
|
|
void
|
|
Disable_Lights();
|
|
// utility methods
|
|
TSubModel const *
|
|
Pick_Control() const { return m_pickcontrolitem; }
|
|
editor::basic_node const *
|
|
Pick_Node() const { return m_picksceneryitem; }
|
|
// maintenance methods
|
|
void
|
|
Update( double const Deltatime );
|
|
TSubModel const *
|
|
Update_Pick_Control();
|
|
editor::basic_node const *
|
|
Update_Pick_Node();
|
|
// debug methods
|
|
std::string const &
|
|
info_times() const;
|
|
std::string const &
|
|
info_stats() const;
|
|
|
|
// members
|
|
GLenum static const sunlight{ GL_LIGHT0 };
|
|
std::size_t m_drawcount { 0 };
|
|
|
|
private:
|
|
// types
|
|
enum class rendermode {
|
|
none,
|
|
color,
|
|
shadows,
|
|
reflections,
|
|
pickcontrols,
|
|
pickscenery
|
|
};
|
|
|
|
enum textureunit {
|
|
helper = 0,
|
|
shadows,
|
|
normals,
|
|
diffuse
|
|
};
|
|
|
|
struct debug_stats {
|
|
int dynamics { 0 };
|
|
int models { 0 };
|
|
int submodels { 0 };
|
|
int paths { 0 };
|
|
int traction { 0 };
|
|
int shapes { 0 };
|
|
int lines { 0 };
|
|
int drawcalls { 0 };
|
|
};
|
|
|
|
using section_sequence = std::vector<scene::basic_section *>;
|
|
using distancecell_pair = std::pair<double, scene::basic_cell *>;
|
|
using cell_sequence = std::vector<distancecell_pair>;
|
|
|
|
struct renderpass_config {
|
|
|
|
opengl_camera camera;
|
|
rendermode draw_mode { rendermode::none };
|
|
float draw_range { 0.0f };
|
|
};
|
|
|
|
struct units_state {
|
|
|
|
bool diffuse { false };
|
|
bool shadows { false };
|
|
bool reflections { false };
|
|
};
|
|
|
|
typedef std::vector<opengl_light> opengllight_array;
|
|
|
|
// methods
|
|
bool
|
|
Init_caps();
|
|
void
|
|
setup_pass( renderpass_config &Config, rendermode const Mode, float const Znear = 0.f, float const Zfar = 1.f, bool const Ignoredebug = false );
|
|
void
|
|
setup_matrices();
|
|
void
|
|
setup_drawing( bool const Alpha = false );
|
|
void
|
|
setup_units( bool const Diffuse, bool const Shadows, bool const Reflections );
|
|
void
|
|
setup_shadow_color( glm::vec4 const &Shadowcolor );
|
|
void
|
|
switch_units( bool const Diffuse, bool const Shadows, bool const Reflections );
|
|
// helper, texture manager method; activates specified texture unit
|
|
void
|
|
select_unit( GLint const Textureunit );
|
|
// runs jobs needed to generate graphics for specified render pass
|
|
void
|
|
Render_pass( rendermode const Mode );
|
|
// creates dynamic environment cubemap
|
|
bool
|
|
Render_reflections();
|
|
bool
|
|
Render( world_environment *Environment );
|
|
void
|
|
Render( scene::basic_region *Region );
|
|
void
|
|
Render( section_sequence::iterator First, section_sequence::iterator Last );
|
|
void
|
|
Render( cell_sequence::iterator First, cell_sequence::iterator Last );
|
|
void
|
|
Render( scene::shape_node const &Shape, bool const Ignorerange );
|
|
void
|
|
Render( TAnimModel *Instance );
|
|
bool
|
|
Render( TDynamicObject *Dynamic );
|
|
bool
|
|
Render( TModel3d *Model, material_data const *Material, float const Squaredistance, Math3D::vector3 const &Position, Math3D::vector3 const &Angle );
|
|
bool
|
|
Render( TModel3d *Model, material_data const *Material, float const Squaredistance );
|
|
void
|
|
Render( TSubModel *Submodel );
|
|
void
|
|
Render( TTrack *Track );
|
|
void
|
|
Render( scene::basic_cell::path_sequence::const_iterator First, scene::basic_cell::path_sequence::const_iterator Last );
|
|
bool
|
|
Render_cab( TDynamicObject *Dynamic, bool const Alpha = false );
|
|
void
|
|
Render( TMemCell *Memcell );
|
|
void
|
|
Render_Alpha( scene::basic_region *Region );
|
|
void
|
|
Render_Alpha( cell_sequence::reverse_iterator First, cell_sequence::reverse_iterator Last );
|
|
void
|
|
Render_Alpha( TAnimModel *Instance );
|
|
void
|
|
Render_Alpha( TTraction *Traction );
|
|
void
|
|
Render_Alpha( scene::lines_node const &Lines );
|
|
bool
|
|
Render_Alpha( TDynamicObject *Dynamic );
|
|
bool
|
|
Render_Alpha( TModel3d *Model, material_data const *Material, float const Squaredistance, Math3D::vector3 const &Position, Math3D::vector3 const &Angle );
|
|
bool
|
|
Render_Alpha( TModel3d *Model, material_data const *Material, float const Squaredistance );
|
|
void
|
|
Render_Alpha( TSubModel *Submodel );
|
|
void
|
|
Update_Lights( light_array &Lights );
|
|
glm::vec3
|
|
pick_color( std::size_t const Index );
|
|
std::size_t
|
|
pick_index( glm::ivec3 const &Color );
|
|
|
|
// members
|
|
GLFWwindow *m_window { nullptr };
|
|
gfx::geometrybank_manager m_geometry;
|
|
material_manager m_materials;
|
|
texture_manager m_textures;
|
|
opengllight_array m_lights;
|
|
|
|
gfx::geometry_handle m_billboardgeometry { 0, 0 };
|
|
texture_handle m_glaretexture { -1 };
|
|
texture_handle m_suntexture { -1 };
|
|
texture_handle m_moontexture { -1 };
|
|
texture_handle m_reflectiontexture { -1 };
|
|
GLUquadricObj *m_quadric { nullptr }; // helper object for drawing debug mode scene elements
|
|
// TODO: refactor framebuffer stuff into an object
|
|
bool m_framebuffersupport { false };
|
|
#ifdef EU07_USE_PICKING_FRAMEBUFFER
|
|
GLuint m_pickframebuffer { 0 };
|
|
GLuint m_picktexture { 0 };
|
|
GLuint m_pickdepthbuffer { 0 };
|
|
#endif
|
|
int m_shadowbuffersize { 2048 };
|
|
GLuint m_shadowframebuffer { 0 };
|
|
GLuint m_shadowtexture { 0 };
|
|
#ifdef EU07_USE_DEBUG_SHADOWMAP
|
|
GLuint m_shadowdebugtexture{ 0 };
|
|
#endif
|
|
glm::mat4 m_shadowtexturematrix; // conversion from camera-centric world space to light-centric clip space
|
|
GLuint m_environmentframebuffer { 0 };
|
|
GLuint m_environmentcubetexture { 0 };
|
|
GLuint m_environmentdepthbuffer { 0 };
|
|
bool m_environmentcubetexturesupport { false }; // indicates whether we can use the dynamic environment cube map
|
|
int m_environmentcubetextureface { 0 }; // helper, currently processed cube map face
|
|
int m_environmentupdatetime { 0 }; // time of the most recent environment map update
|
|
glm::dvec3 m_environmentupdatelocation; // coordinates of most recent environment map update
|
|
|
|
int m_helpertextureunit { GL_TEXTURE0 };
|
|
int m_shadowtextureunit { GL_TEXTURE1 };
|
|
int m_normaltextureunit { GL_TEXTURE2 };
|
|
int m_diffusetextureunit{ GL_TEXTURE3 };
|
|
units_state m_unitstate;
|
|
|
|
unsigned int m_framestamp; // id of currently rendered gfx frame
|
|
float m_framerate;
|
|
double m_updateaccumulator { 0.0 };
|
|
std::string m_debugtimestext;
|
|
std::string m_pickdebuginfo;
|
|
debug_stats m_debugstats;
|
|
std::string m_debugstatstext;
|
|
|
|
glm::vec4 m_baseambient { 0.0f, 0.0f, 0.0f, 1.0f };
|
|
glm::vec4 m_shadowcolor { 0.65f, 0.65f, 0.65f, 1.f };
|
|
float m_specularopaquescalefactor { 1.f };
|
|
float m_speculartranslucentscalefactor { 1.f };
|
|
bool m_renderspecular{ false }; // controls whether to include specular component in the calculations
|
|
|
|
renderpass_config m_renderpass;
|
|
section_sequence m_sectionqueue; // list of sections in current render pass
|
|
cell_sequence m_cellqueue;
|
|
std::vector<TSubModel const *> m_pickcontrolsitems;
|
|
TSubModel const *m_pickcontrolitem { nullptr };
|
|
std::vector<editor::basic_node const *> m_picksceneryitems;
|
|
editor::basic_node const *m_picksceneryitem { nullptr };
|
|
#ifdef EU07_USE_DEBUG_CAMERA
|
|
renderpass_config m_worldcamera; // debug item
|
|
#endif
|
|
};
|
|
|
|
extern opengl_renderer GfxRenderer;
|
|
|
|
//---------------------------------------------------------------------------
|