123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339 |
- #include "client/rendering/Engine.h"
- #include "client/Game.h"
- #include "client/rendering/Mesh.h"
- #include "math/Frustum.h"
- #include "rendering/Framebuffer.h"
- #include "rendering/Window.h"
- #include "utils/Logger.h"
- #include "utils/Random.h"
- #include "wrapper/GL.h"
- static Window window;
- static Shader worldShader;
- static Shader ssaoShader;
- static Shader ssaoBlurShader;
- static Shader shadowShader;
- static Shader postWorldShader;
- static Shader overlayShader;
- static Framebuffer<5> worldBuffer;
- static Framebuffer<1> ssaoBuffer;
- static Framebuffer<1> ssaoBlurBuffer;
- static Framebuffer<1> shadowBuffer;
- static Size lastSize{0, 0};
- static Frustum frustum{60.0f, 0.1f, 1000.0f, window.getSize()};
- static MatrixStack<16> model;
- Renderer Engine::renderer;
- ShaderMatrix Engine::matrix{nullptr, model, nullptr};
- float Engine::lag = 0.0f;
- static Texture ssaoNoise;
- static Mesh rectangle;
- static Matrix worldProj;
- static Matrix worldView;
- static Matrix worldShadowProj;
- static Matrix worldShadowView;
- static Matrix worldShadowProjView;
- static bool useSsao = true;
- static bool useShadows = false;
- static float shadowRadius = 0.01f;
- static float shadowBias = 0.0002f;
- static bool running = true;
- static Error compileShader(Shader& s, const char* name) {
- constexpr const char* prefix = "resources/shader/";
- return s.compile(StringBuffer<50>(prefix).append(name).append(".vs"),
- StringBuffer<50>(prefix).append(name).append(".fs"));
- }
- static Error initShaders() {
- Error error = compileShader(worldShader, "world");
- if(error.has()) {
- return error;
- }
- error = compileShader(ssaoShader, "ssao");
- if(error.has()) {
- return error;
- }
- error = compileShader(ssaoBlurShader, "ssaoBlur");
- if(error.has()) {
- return error;
- }
- error = compileShader(shadowShader, "worldShadow");
- if(error.has()) {
- return error;
- }
- error = compileShader(postWorldShader, "worldPost");
- if(error.has()) {
- return error;
- }
- return compileShader(overlayShader, "overlay");
- }
- static void resizeFramebuffers(const Size& size) {
- worldBuffer.resize(size);
- ssaoBuffer.resize(size);
- ssaoBlurBuffer.resize(size);
- shadowBuffer.resize(size);
- }
- static Error initFramebuffers(const Size& size) {
- Error error = worldBuffer.init(
- size, TextureFormat::float32(3), TextureFormat::float32(3),
- TextureFormat::color8(4), TextureFormat::float32(1),
- TextureFormat::depth32(true));
- if(error.has()) {
- return error;
- }
- error = ssaoBuffer.init(size, TextureFormat::float32(1));
- if(error.has()) {
- return error;
- }
- error = ssaoBlurBuffer.init(size, TextureFormat::float32(1));
- if(error.has()) {
- return error;
- }
- return shadowBuffer.init(size, TextureFormat::depth32());
- }
- static bool initRectangle() {
- if(rectangle.init()) {
- return true;
- }
- rectangle.build(TypedBuffer<Mesh::Triangle>(2)
- .add({{{-1.0f, -1.0f, +0.0f}, {0.0f, 0.0f}},
- {{+1.0f, +1.0f, +0.0f}, {1.0f, 1.0f}},
- {{-1.0f, +1.0f, +0.0f}, {0.0f, 1.0f}}})
- .add({{{-1.0f, -1.0f, +0.0f}, {0.0f, 0.0f}},
- {{+1.0f, -1.0f, +0.0f}, {1.0f, 0.0f}},
- {{+1.0f, +1.0f, +0.0f}, {1.0f, 1.0f}}}));
- return false;
- }
- static void initNoise() {
- ssaoNoise.init(TextureFormat::float32(3), 0);
- Random r(1);
- Array<float, 48> data;
- for(int i = 0; i < 48; i++) {
- data[i] = r.nextFloat() * 2.0f - 1.0f;
- }
- ssaoNoise.setData(4, 4, data.begin());
- }
- bool Engine::init() {
- WindowOptions options(4, 0, {1024, 620}, false, "test");
- Error error = window.open(options);
- if(error.has()) {
- LOG_ERROR(error.message);
- return true;
- }
- lastSize = window.getSize();
- error = initShaders();
- if(error.has()) {
- LOG_ERROR(error.message);
- return true;
- }
- error = initFramebuffers(window.getSize());
- if(error.has()) {
- LOG_ERROR(error.message);
- return true;
- }
- initNoise();
- if(renderer.init() || initRectangle()) {
- return true;
- }
- return false;
- }
- static void renderShadow() {
- shadowBuffer.bindAndClear();
- GL::enableDepthTesting();
- shadowShader.use();
- worldShadowProjView = worldShadowProj;
- worldShadowProjView *= worldShadowView;
- shadowShader.setMatrix("projView", worldShadowProjView.getValues());
- model.clear();
- shadowShader.setMatrix("model", model.peek().getValues());
- Engine::matrix = {&shadowShader, model, &worldView};
- Game::renderWorld();
- }
- static void renderWorld() {
- worldBuffer.bindAndClear();
- GL::enableDepthTesting();
- worldShader.use();
- Matrix rWorldShadowProjView;
- rWorldShadowProjView.scale(0.5f).translate(Vector3(0.5f, 0.5f, 0.5f));
- rWorldShadowProjView *= worldShadowProjView;
- worldShader.setMatrix("projViewShadow", rWorldShadowProjView.getValues());
- worldShader.setMatrix("proj", worldProj.getValues());
- worldView = Matrix();
- worldShader.setMatrix("view", worldView.getValues());
- model.clear();
- worldShader.setMatrix("model", model.peek().getValues());
- shadowBuffer.bindTextureTo(0, 1);
- worldShader.setInt("shadows", useShadows);
- worldShader.setFloat("radius", shadowRadius);
- worldShader.setFloat("zbias", shadowBias);
- Engine::matrix = {&worldShader, model, &worldView};
- Game::renderWorld();
- }
- static void renderSSAO() {
- ssaoShader.use();
- Matrix rProj;
- rProj.scale(0.5f).translate(Vector3(0.5f, 0.5f, 0.5f));
- rProj *= worldProj;
- ssaoShader.setMatrix("proj", rProj.getValues());
- const Size& size = window.getSize();
- ssaoShader.setInt("width", size.width);
- ssaoShader.setInt("height", size.height);
- worldBuffer.bindTextureTo(0, 0);
- worldBuffer.bindTextureTo(4, 1);
- ssaoNoise.bindTo(2);
- ssaoBuffer.bindAndClear();
- rectangle.draw();
- ssaoBlurShader.use();
- ssaoBuffer.bindTextureTo(0, 0);
- ssaoBlurBuffer.bindAndClear();
- rectangle.draw();
- }
- static void renderPostWorld() {
- GL::bindMainFramebuffer();
- GL::clear();
- postWorldShader.use();
- worldBuffer.bindTextureTo(2, 0);
- ssaoBlurBuffer.bindTextureTo(0, 1);
- worldBuffer.bindTextureTo(3, 2);
- worldBuffer.bindTextureTo(1, 3);
- postWorldShader.setInt("ssao", useSsao);
- postWorldShader.setInt("shadows", useShadows);
- rectangle.draw();
- }
- static void renderOverlay() {
- GL::disableDepthTesting();
- overlayShader.use();
- const Size& size = window.getSize();
- Matrix m;
- m.scale(Vector3(2.0f / size.width, -2.0f / size.height, 1.0f))
- .translate(Vector3(-1.0f, 1.0f, 0.0f));
- overlayShader.setMatrix("view", m.getValues());
- model.clear();
- overlayShader.setMatrix("model", model.peek().getValues());
- GL::enableBlending();
- Engine::matrix = {&overlayShader, model, &m};
- Game::renderOverlay();
- GL::disableBlending();
- }
- static void updateWorldProjection() {
- worldProj = frustum.updateProjection();
- if(!useShadows) {
- return;
- }
- worldShadowProj.set(0, Vector4(2.0f / 40.0f, 0.0f, 0.0f, 0.0f));
- worldShadowProj.set(1, Vector4(0.0f, 2.0f / 30.0f, 0.0f, 0.0f));
- worldShadowProj.set(2, Vector4(0.0f, 0.0f, -2.0f / (1000.0f - 0.1f), 0.0f));
- worldShadowProj.set(3, Vector4(0.0f, 0.0f, 0.0f, 1.0f));
- }
- static void updateWorldView() {
- if(!useShadows) {
- return;
- }
- Vector3 right(0.939693f, 0.0f, -0.34202f);
- Vector3 back(0.280166f, 0.573576f, 0.769751f);
- Vector3 up(-0.196175f, 0.819152f, -0.538986f);
- Vector3 center(16.0f, 24.0f, 24.0f);
- worldShadowView.set(
- 0, Vector4(right[0], right[1], right[2], right.dot(-center)));
- worldShadowView.set(1, Vector4(up[0], up[1], up[2], up.dot(-center)));
- worldShadowView.set(2,
- Vector4(back[0], back[1], back[2], back.dot(-center)));
- worldShadowView.set(3, Vector4(0.0f, 0.0f, 0.0f, 1.0f));
- }
- static void startRender() {
- const Size& size = window.getSize();
- if(size.width != lastSize.width || size.height != lastSize.height) {
- GL::setViewport(size.width, size.height);
- resizeFramebuffers(size);
- lastSize = size;
- }
- GL::printError("loop error");
- updateWorldProjection();
- updateWorldView();
- if(useShadows) {
- renderShadow();
- }
- renderWorld();
- if(useSsao) {
- renderSSAO();
- }
- renderPostWorld();
- renderOverlay();
- }
- struct Loop final {
- void render(float lag) {
- Engine::lag = lag;
- startRender();
- }
- void tick() {
- Game::tick();
- }
- bool isRunning() const {
- return running;
- }
- };
- void Engine::run() {
- Loop loop;
- window.run(loop, 50'000'000);
- }
- void Engine::stop() {
- running = false;
- }
- void Engine::setTextInput(TextInput* input) {
- window.textInput = input;
- if(input != nullptr) {
- input->setActive(true);
- }
- }
- bool Engine::isActiveTextInput(TextInput* input) {
- return window.textInput == input;
- }
- Buttons& Engine::getButtons() {
- return window.buttons;
- }
- const Size& Engine::getSize() {
- return window.getSize();
- }
- const Clock& Engine::getFrameClock() {
- return window.getFrameClock();
- }
- const Clock& Engine::getTickClock() {
- return window.getTickClock();
- }
|