#include "engine.h" #include "../base/log.h" #include "../third_party/texture_compressor/texture_compressor.h" #include "animator.h" #include "audio/audio.h" #include "drawable.h" #include "font.h" #include "game.h" #include "game_factory.h" #include "image.h" #include "image_quad.h" #include "input_event.h" #include "mesh.h" #include "platform/platform.h" #include "renderer/geometry.h" #include "renderer/renderer.h" #include "renderer/shader.h" #include "renderer/texture.h" #include "shader_source.h" using namespace base; namespace eng { Engine* Engine::singleton = nullptr; Engine::Engine(Platform* platform, Renderer* renderer, Audio* audio) : platform_(platform), renderer_(renderer), audio_(audio) { DCHECK(!singleton); singleton = this; renderer_->SetContextLostCB(std::bind(&Engine::ContextLost, this)); quad_ = CreateRenderResource(); pass_through_shader_ = CreateRenderResource(); solid_shader_ = CreateRenderResource(); stats_ = std::make_unique(); stats_->SetZOrder(std::numeric_limits::max()); } Engine::~Engine() { game_.reset(); stats_.reset(); singleton = nullptr; } Engine& Engine::Get() { return *singleton; } bool Engine::Initialize() { // Normalize viewport. if (GetScreenWidth() > GetScreenHeight()) { float aspect_ratio = (float)GetScreenWidth() / (float)GetScreenHeight(); LOG << "aspect ratio: " << aspect_ratio; screen_size_ = {aspect_ratio * 2.0f, 2.0f}; projection_.CreateOrthoProjection(-aspect_ratio, aspect_ratio, -1.0f, 1.0f); } else { float aspect_ratio = (float)GetScreenHeight() / (float)GetScreenWidth(); LOG << "aspect_ratio: " << aspect_ratio; screen_size_ = {2.0f, aspect_ratio * 2.0f}; projection_.CreateOrthoProjection(-1.0, 1.0, -aspect_ratio, aspect_ratio); } LOG << "image scale factor: " << GetImageScaleFactor(); if (renderer_->SupportsDXT5()) { tex_comp_alpha_ = TextureCompressor::Create(TextureCompressor::kFormatDXT5); } else if (renderer_->SupportsATC()) { tex_comp_alpha_ = TextureCompressor::Create(TextureCompressor::kFormatATCIA); } if (renderer_->SupportsDXT1()) { tex_comp_opaque_ = TextureCompressor::Create(TextureCompressor::kFormatDXT1); } else if (renderer_->SupportsATC()) { tex_comp_opaque_ = TextureCompressor::Create(TextureCompressor::kFormatATC); } else if (renderer_->SupportsETC1()) { tex_comp_opaque_ = TextureCompressor::Create(TextureCompressor::kFormatETC1); } system_font_ = std::make_unique(); system_font_->Load("engine/RobotoMono-Regular.ttf"); if (!CreateRenderResources()) return false; SetImageSource("stats_tex", std::bind(&Engine::PrintStats, this)); game_ = GameFactoryBase::CreateGame(""); if (!game_) { LOG << "No game found to run."; return false; } if (!game_->Initialize()) { LOG << "Failed to initialize the game."; return false; } return true; } void Engine::Shutdown() { LOG << "Shutting down engine."; } void Engine::Update(float delta_time) { seconds_accumulated_ += delta_time; ++tick_; for (auto d : animators_) d->Update(delta_time); game_->Update(delta_time); // Destroy unused textures. for (auto& t : textures_) { if (!t.second.persistent && t.second.texture.use_count() == 1) t.second.texture->Destroy(); } fps_seconds_ += delta_time; if (fps_seconds_ >= 1) { fps_ = renderer_->GetAndResetFPS(); fps_seconds_ = 0; } if (stats_->IsVisible()) { RefreshImage("stats_tex"); stats_->AutoScale(); } } void Engine::Draw(float frame_frac) { for (auto d : animators_) d->Evaluate(time_step_ * frame_frac); drawables_.sort( [](auto& a, auto& b) { return a->GetZOrder() < b->GetZOrder(); }); renderer_->PrepareForDrawing(); for (auto d : drawables_) { if (d->IsVisible()) d->Draw(frame_frac); } renderer_->Present(); } void Engine::LostFocus() { audio_->Suspend(); if (game_) game_->LostFocus(); } void Engine::GainedFocus(bool from_interstitial_ad) { audio_->Resume(); if (game_) game_->GainedFocus(from_interstitial_ad); } void Engine::AddDrawable(Drawable* drawable) { DCHECK(std::find(drawables_.begin(), drawables_.end(), drawable) == drawables_.end()); drawables_.push_back(drawable); } void Engine::RemoveDrawable(Drawable* drawable) { auto it = std::find(drawables_.begin(), drawables_.end(), drawable); if (it != drawables_.end()) { drawables_.erase(it); return; } } void Engine::AddAnimator(Animator* animator) { DCHECK(std::find(animators_.begin(), animators_.end(), animator) == animators_.end()); animators_.push_back(animator); } void Engine::RemoveAnimator(Animator* animator) { auto it = std::find(animators_.begin(), animators_.end(), animator); if (it != animators_.end()) { animators_.erase(it); return; } } void Engine::Exit() { platform_->Exit(); } Vector2f Engine::ToScale(const Vector2f& vec) { return GetScreenSize() * vec / Vector2f((float)GetScreenWidth(), (float)GetScreenHeight()); } Vector2f Engine::ToPosition(const Vector2f& vec) { return ToScale(vec) - GetScreenSize() / 2.0f; } void Engine::SetImageSource(const std::string& asset_name, const std::string& file_name, bool persistent) { SetImageSource( asset_name, [file_name]() -> std::unique_ptr { auto image = std::make_unique(); if (!image->Load(file_name)) return nullptr; image->Compress(); return image; }, persistent); } void Engine::SetImageSource(const std::string& asset_name, CreateImageCB create_image, bool persistent) { std::shared_ptr texture; auto it = textures_.find(asset_name); if (it != textures_.end()) { texture = it->second.texture; it->second.create_image = create_image; it->second.persistent = persistent; } else { texture = CreateRenderResource(); textures_[asset_name] = {texture, create_image, persistent}; } if (persistent) { auto image = create_image(); if (image) texture->Update(std::move(image)); else texture->Destroy(); } } void Engine::RefreshImage(const std::string& asset_name) { auto it = textures_.find(asset_name); if (it == textures_.end()) return; auto image = it->second.create_image(); if (image) it->second.texture->Update(std::move(image)); else it->second.texture->Destroy(); } std::shared_ptr Engine::GetTexture(const std::string& asset_name) { auto it = textures_.find(asset_name); if (it != textures_.end()) { if (!it->second.texture->IsValid()) RefreshImage(it->first); return it->second.texture; } std::shared_ptr texture = CreateRenderResource(); textures_[asset_name] = {texture}; return texture; } void Engine::AddInputEvent(std::unique_ptr event) { if (replaying_) return; switch (event->GetType()) { case InputEvent::kDragEnd: if (((GetScreenSize() / 2) * 0.9f - event->GetVector()).Length() <= 0.25f) { SetSatsVisible(!stats_->IsVisible()); // TODO: Enqueue DragCancel so we can consume this event. } break; case InputEvent::kKeyPress: if (event->GetKeyPress() == 's') { SetSatsVisible(!stats_->IsVisible()); // Consume event. return; } break; case InputEvent::kDrag: if (stats_->IsVisible()) { if ((stats_->GetPosition() - event->GetVector()).Length() <= stats_->GetSize().y) stats_->SetPosition(event->GetVector()); // TODO: Enqueue DragCancel so we can consume this event. } break; default: break; } input_queue_.push_back(std::move(event)); } std::unique_ptr Engine::GetNextInputEvent() { std::unique_ptr event; if (replaying_) { if (replay_index_ < replay_data_.root()["input"].size()) { auto data = replay_data_.root()["input"][replay_index_]; if (data["tick"].asUInt64() == tick_) { event = std::make_unique( (InputEvent::Type)data["input_type"].asInt(), (size_t)data["pointer_id"].asUInt(), Vector2f(data["pos_x"].asFloat(), data["pos_y"].asFloat())); ++replay_index_; } return event; } replaying_ = false; replay_data_.root().clear(); } if (!input_queue_.empty()) { event.swap(input_queue_.front()); input_queue_.pop_front(); if (recording_) { Json::Value data; data["tick"] = tick_; data["input_type"] = event->GetType(); data["pointer_id"] = event->GetPointerId(); data["pos_x"] = event->GetVector().x; data["pos_y"] = event->GetVector().y; replay_data_.root()["input"].append(data); } } return event; } void Engine::StartRecording(const Json::Value& payload) { if (!replaying_ && !recording_) { recording_ = true; random_ = Random(); replay_data_.root()["seed"] = random_.seed(); replay_data_.root()["payload"] = payload; tick_ = 0; } } void Engine::EndRecording(const std::string file_name) { if (recording_) { DCHECK(!replaying_); recording_ = false; replay_data_.SaveAs(file_name, PersistentData::kShared); replay_data_.root().clear(); } } bool Engine::Replay(const std::string file_name, Json::Value& payload) { if (!replaying_ && !recording_ && replay_data_.Load(file_name, PersistentData::kShared)) { replaying_ = true; random_ = Random(replay_data_.root()["seed"].asUInt()); payload = replay_data_.root()["payload"]; tick_ = 0; replay_index_ = 0; } return replaying_; } void Engine::Vibrate(int duration) { if (vibration_enabled_) platform_->Vibrate(duration); } void Engine::ShowInterstitialAd() { platform_->ShowInterstitialAd(); } void Engine::ShareFile(const std::string& file_name) { platform_->ShareFile(file_name); } void Engine::SetKeepScreenOn(bool keep_screen_on) { platform_->SetKeepScreenOn(keep_screen_on); } void Engine::SetEnableAudio(bool enable) { audio_->SetEnableAudio(enable); } TextureCompressor* Engine::GetTextureCompressor(bool opacity) { return opacity ? tex_comp_alpha_.get() : tex_comp_opaque_.get(); } int Engine::GetScreenWidth() const { return renderer_->screen_width(); } int Engine::GetScreenHeight() const { return renderer_->screen_height(); } int Engine::GetDeviceDpi() const { return platform_->GetDeviceDpi(); } float Engine::GetImageScaleFactor() const { float width_inch = static_cast(renderer_->screen_width()) / static_cast(platform_->GetDeviceDpi()); return 2.57143f / width_inch; } const std::string& Engine::GetRootPath() const { return platform_->GetRootPath(); } const std::string& Engine::GetDataPath() const { return platform_->GetDataPath(); } const std::string& Engine::GetSharedDataPath() const { return platform_->GetSharedDataPath(); } int Engine::GetAudioHardwareSampleRate() { return audio_->GetHardwareSampleRate(); } bool Engine::IsMobile() const { return platform_->mobile_device(); } std::unique_ptr Engine::CreateRenderResourceInternal( RenderResourceFactoryBase& factory) { return renderer_->CreateResource(factory); } void Engine::ContextLost() { CreateRenderResources(); for (auto& t : textures_) RefreshImage(t.first); game_->ContextLost(); } bool Engine::CreateRenderResources() { // This creates a normalized unit sized quad. static const char vertex_description[] = "p2f;t2f"; static const float vertices[] = { -0.5f, -0.5f, 0.0f, 1.0f, 0.5f, -0.5f, 1.0f, 1.0f, -0.5f, 0.5f, 0.0f, 0.0f, 0.5f, 0.5f, 1.0f, 0.0f, }; // Create the quad geometry we can reuse for all sprites. auto quad_mesh = std::make_unique(); quad_mesh->Create(kPrimitive_TriangleStrip, vertex_description, 4, vertices); quad_->Create(std::move(quad_mesh)); // Create the shader we can reuse for texture rendering. auto source = std::make_unique(); if (!source->Load("engine/pass_through.glsl")) { LOG << "Could not create pass through shader."; return false; } pass_through_shader_->Create(std::move(source), quad_->vertex_description(), quad_->primitive(), false); // Create the shader we can reuse for solid rendering. source = std::make_unique(); if (!source->Load("engine/solid.glsl")) { LOG << "Could not create solid shader."; return false; } solid_shader_->Create(std::move(source), quad_->vertex_description(), quad_->primitive(), false); return true; } void Engine::SetSatsVisible(bool visible) { stats_->SetVisible(visible); if (visible) stats_->Create("stats_tex"); else stats_->Destory(); } std::unique_ptr Engine::PrintStats() { if (!stats_->IsVisible()) return nullptr; constexpr int width = 200; std::vector lines; std::string line; line = "fps: "; line += std::to_string(fps_); lines.push_back(line); // line = "cmd: "; // line += std::to_string(renderer_->global_queue_size() + // renderer_->render_queue_size()); // lines.push_back(line); constexpr int margin = 5; int line_height = system_font_->GetLineHeight(); int image_width = width + margin * 2; int image_height = (line_height + margin) * lines.size() + margin; auto image = std::make_unique(); image->Create(image_width, image_height); image->Clear({1, 1, 1, 0.08f}); int y = margin; for (auto& text : lines) { system_font_->Print(margin, y, text.c_str(), image->GetBuffer(), image->GetWidth()); y += line_height + margin; } return image; } } // namespace eng