#include "game/scenes/ending.hpp" #include #include "core/audio/audio.hpp" // Para Audio #include "core/input/global_inputs.hpp" // Para check #include "core/input/input.hpp" // Para Input #include "core/rendering/pixel_reveal.hpp" // Para PixelReveal #include "core/rendering/screen.hpp" // Para Screen #include "core/rendering/surface.hpp" // Para Surface #include "core/rendering/surface_sprite.hpp" // Para SSprite #include "core/rendering/text.hpp" // Para Text, TEXT_STROKE #include "core/resources/resource_cache.hpp" // Para Resource #include "core/system/global_events.hpp" // Para check #include "game/options.hpp" // Para Options, options, OptionsGame, SectionS... #include "game/scene_manager.hpp" // Para SceneManager #include "utils/delta_timer.hpp" // Para DeltaTimer #include "utils/utils.hpp" // Para PaletteColor // Constructor Ending::Ending() : delta_timer_(std::make_unique()) { SceneManager::current = SceneManager::Scene::ENDING; SceneManager::options = SceneManager::Options::NONE; iniTexts(); // Inicializa los textos iniPics(); // Inicializa las imagenes iniScenes(); // Inicializa las escenas Screen::get()->setBorderColor(static_cast(PaletteColor::BLACK)); // Cambia el color del borde } // Destructor Ending::~Ending() = default; // Actualiza el objeto void Ending::update() { const float DELTA_TIME = delta_timer_->tick(); total_time_ += DELTA_TIME; // Actualiza el tiempo total handleEvents(); // Comprueba los eventos handleInput(); // Comprueba las entradas updateState(DELTA_TIME); // Actualiza la máquina de estados updateSpriteCovers(); // Actualiza las cortinillas de los elementos Audio::update(); // Actualiza el objeto Audio Screen::get()->update(DELTA_TIME); // Actualiza el objeto Screen } // Dibuja el final en pantalla void Ending::render() { // Prepara para empezar a dibujar en la textura de juego Screen::get()->start(); // Limpia la pantalla Screen::get()->clearSurface(static_cast(PaletteColor::BLACK)); // Skip rendering durante WARMING_UP if (state_ != State::WARMING_UP) { // Dibuja las imagenes de la escena const auto& pic = sprite_pics_.at(current_scene_); pic.image_sprite->render(); pic.pixel_reveal->render(pic.pos_x, pic.pos_y); // Dibuja los textos de la escena for (const auto& ti : scenes_.at(current_scene_).text_index) { // Convertir trigger de frames a segundos @ 60fps const float TRIGGER_TIME = static_cast(ti.trigger) / 60.0F; if (state_time_ > TRIGGER_TIME) { const auto& txt = sprite_texts_.at(ti.index); txt.image_sprite->render(); txt.pixel_reveal->render(txt.pos_x, txt.pos_y); } } // Dibuja la cortinilla de cambio de escena if (scene_cover_) { scene_cover_->render(0, 0); } } // Vuelca el contenido del renderizador en pantalla Screen::get()->render(); } // Comprueba el manejador de eventos void Ending::handleEvents() { SDL_Event event; while (SDL_PollEvent(&event)) { GlobalEvents::handle(event); } } // Comprueba las entradas void Ending::handleInput() { Input::get()->update(); GlobalInputs::handle(); } // Transición entre estados void Ending::transitionToState(State new_state) { state_ = new_state; state_time_ = 0.0F; // Al cambiar a una escena, resetear la cortinilla de salida y el contador de fade if (new_state != State::WARMING_UP && new_state != State::ENDING) { fadeout_time_ = 0.0F; scene_cover_.reset(); } } // Actualiza la máquina de estados void Ending::updateState(float delta_time) { state_time_ += delta_time; switch (state_) { case State::WARMING_UP: if (state_time_ >= WARMUP_DURATION) { transitionToState(State::SCENE_0); current_scene_ = 0; } break; case State::SCENE_0: checkChangeScene(); if (state_time_ >= SCENE_0_DURATION - FADEOUT_START_OFFSET) { fadeout_time_ += delta_time; if (!scene_cover_) { scene_cover_ = std::make_unique(Options::game.width, Options::game.height, COVER_PIXELS_PER_SECOND, STEP_DURATION, COVER_STEPS, true); } scene_cover_->update(fadeout_time_); } break; case State::SCENE_1: checkChangeScene(); if (state_time_ >= SCENE_1_DURATION - FADEOUT_START_OFFSET) { fadeout_time_ += delta_time; if (!scene_cover_) { scene_cover_ = std::make_unique(Options::game.width, Options::game.height, COVER_PIXELS_PER_SECOND, STEP_DURATION, COVER_STEPS, true); } scene_cover_->update(fadeout_time_); } break; case State::SCENE_2: checkChangeScene(); if (state_time_ >= SCENE_2_DURATION - FADEOUT_START_OFFSET) { fadeout_time_ += delta_time; if (!scene_cover_) { scene_cover_ = std::make_unique(Options::game.width, Options::game.height, COVER_PIXELS_PER_SECOND, STEP_DURATION, COVER_STEPS, true); } scene_cover_->update(fadeout_time_); } break; case State::SCENE_3: checkChangeScene(); if (state_time_ >= SCENE_3_DURATION - FADEOUT_START_OFFSET) { fadeout_time_ += delta_time; if (!scene_cover_) { scene_cover_ = std::make_unique(Options::game.width, Options::game.height, COVER_PIXELS_PER_SECOND, STEP_DURATION, COVER_STEPS, true); } scene_cover_->update(fadeout_time_); } break; case State::SCENE_4: checkChangeScene(); if (state_time_ >= SCENE_4_DURATION - FADEOUT_START_OFFSET) { fadeout_time_ += delta_time; if (!scene_cover_) { scene_cover_ = std::make_unique(Options::game.width, Options::game.height, COVER_PIXELS_PER_SECOND, STEP_DURATION, COVER_STEPS, true); } scene_cover_->update(fadeout_time_); } break; case State::ENDING: // Esperar ENDING_DURATION y luego transicionar a ENDING2 if (state_time_ >= ENDING_DURATION) { SceneManager::current = SceneManager::Scene::ENDING2; } break; } } // Inicializa los textos void Ending::iniTexts() { // Vector con los textos std::vector texts; // Escena #0 texts.push_back({"HE FINALLY MANAGED", 32}); texts.push_back({"TO GET TO THE JAIL", 42}); texts.push_back({"WITH ALL HIS PROJECTS", 142}); texts.push_back({"READY TO BE FREED", 152}); // Escena #1 texts.push_back({"ALL THE JAILERS WERE THERE", 1}); texts.push_back({"WAITING FOR THE JAILGAMES", 11}); texts.push_back({"TO BE RELEASED", 21}); texts.push_back({"THERE WERE EVEN BARRULLS AND", 161}); texts.push_back({"BEGINNERS AMONG THE CROWD", 171}); texts.push_back({"BRY WAS CRYING...", 181}); // Escena #2 texts.push_back({"BUT SUDDENLY SOMETHING", 19}); texts.push_back({"CAUGHT HIS ATTENTION", 29}); // Escena #3 texts.push_back({"A PILE OF JUNK!", 36}); texts.push_back({"FULL OF NON WORKING TRASH!!", 46}); // Escena #4 texts.push_back({"AND THEN,", 36}); texts.push_back({"FOURTY NEW PROJECTS", 46}); texts.push_back({"WERE BORN...", 158}); // Crea los sprites sprite_texts_.clear(); for (const auto& txt : texts) { auto text = Resource::Cache::get()->getText("smb2"); const float WIDTH = text->length(txt.caption, 1) + 2 + 2; const float HEIGHT = text->getCharacterSize() + 2 + 2; auto text_color = static_cast(PaletteColor::WHITE); auto shadow_color = static_cast(PaletteColor::BLACK); EndingSurface st; // Crea la textura st.image_surface = std::make_shared(WIDTH, HEIGHT); auto previuos_renderer = Screen::get()->getRendererSurface(); Screen::get()->setRendererSurface(st.image_surface); text->writeDX(Text::STROKE_FLAG, 2, 2, txt.caption, 1, text_color, 2, shadow_color); // Crea el sprite st.image_sprite = std::make_shared(st.image_surface, 0, 0, st.image_surface->getWidth(), st.image_surface->getHeight()); st.pos_x = static_cast((Options::game.width - st.image_surface->getWidth()) / 2); st.pos_y = txt.pos; st.image_sprite->setPosition(st.pos_x, st.pos_y); // Crea el efecto de revelado pixel a pixel st.pixel_reveal = std::make_unique(static_cast(WIDTH), static_cast(HEIGHT), TEXT_PIXELS_PER_SECOND, STEP_DURATION, REVEAL_STEPS); sprite_texts_.push_back(std::move(st)); Screen::get()->setRendererSurface(previuos_renderer); } } // Inicializa las imagenes void Ending::iniPics() { // Vector con las rutas y la posición std::vector pics; pics.push_back({"ending1.gif", 48}); pics.push_back({"ending2.gif", 26}); pics.push_back({"ending3.gif", 29}); pics.push_back({"ending4.gif", 63}); pics.push_back({"ending5.gif", 53}); // Crea los sprites sprite_pics_.clear(); for (const auto& pic : pics) { EndingSurface sp; // Crea la texture sp.image_surface = Resource::Cache::get()->getSurface(pic.caption); sp.image_surface->setTransparentColor(); const float WIDTH = sp.image_surface->getWidth(); const float HEIGHT = sp.image_surface->getHeight(); // Crea el sprite sp.pos_x = static_cast((Options::game.width - WIDTH) / 2); sp.pos_y = pic.pos; sp.image_sprite = std::make_shared(sp.image_surface, 0, 0, WIDTH, HEIGHT); sp.image_sprite->setPosition(sp.pos_x, sp.pos_y); // Crea el efecto de revelado pixel a pixel sp.pixel_reveal = std::make_unique(static_cast(WIDTH), static_cast(HEIGHT), IMAGE_PIXELS_PER_SECOND, STEP_DURATION, REVEAL_STEPS); sprite_pics_.push_back(std::move(sp)); } } // Inicializa las escenas void Ending::iniScenes() { // Variable para los tiempos int trigger; constexpr int LAPSE = 80; // Crea el contenedor SceneData sc; // Inicializa el vector scenes_.clear(); // Crea la escena #0 sc.counter_end = 1000; sc.picture_index = 0; sc.text_index.clear(); trigger = 85 * 2; trigger += LAPSE; sc.text_index.push_back({0, trigger}); trigger += LAPSE; sc.text_index.push_back({1, trigger}); trigger += LAPSE * 3; sc.text_index.push_back({2, trigger}); trigger += LAPSE; sc.text_index.push_back({3, trigger}); scenes_.push_back(sc); // Crea la escena #1 sc.counter_end = 1400; sc.picture_index = 1; sc.text_index.clear(); trigger = 140 * 2; trigger += LAPSE; sc.text_index.push_back({4, trigger}); trigger += LAPSE; sc.text_index.push_back({5, trigger}); trigger += LAPSE; sc.text_index.push_back({6, trigger}); trigger += LAPSE * 3; sc.text_index.push_back({7, trigger}); trigger += LAPSE; sc.text_index.push_back({8, trigger}); trigger += LAPSE * 3; sc.text_index.push_back({9, trigger}); scenes_.push_back(sc); // Crea la escena #2 sc.counter_end = 1000; sc.picture_index = 2; sc.text_index.clear(); trigger = 148 / 2; trigger += LAPSE; sc.text_index.push_back({10, trigger}); trigger += LAPSE; sc.text_index.push_back({11, trigger}); scenes_.push_back(sc); // Crea la escena #3 sc.counter_end = 800; sc.picture_index = 3; sc.text_index.clear(); trigger = 87 / 2; trigger += LAPSE; sc.text_index.push_back({12, trigger}); trigger += LAPSE / 2; sc.text_index.push_back({13, trigger}); scenes_.push_back(sc); // Crea la escena #4 sc.counter_end = 1000; sc.picture_index = 4; sc.text_index.clear(); trigger = 91 * 2; trigger += LAPSE; sc.text_index.push_back({14, trigger}); trigger += LAPSE * 2; sc.text_index.push_back({15, trigger}); trigger += LAPSE * 3; sc.text_index.push_back({16, trigger}); scenes_.push_back(sc); } // Bucle principal void Ending::run() { Audio::get()->playMusic("ending1.ogg"); while (SceneManager::current == SceneManager::Scene::ENDING) { update(); render(); } Audio::get()->stopMusic(); } // Actualiza las cortinillas de los elementos void Ending::updateSpriteCovers() { // Skip durante WARMING_UP if (state_ == State::WARMING_UP) { return; } // Actualiza el revelado de los textos for (const auto& ti : scenes_.at(current_scene_).text_index) { const float TRIGGER_TIME = static_cast(ti.trigger) / 60.0F; if (state_time_ > TRIGGER_TIME) { const float TIME_SINCE_TRIGGER = state_time_ - TRIGGER_TIME; sprite_texts_.at(ti.index).pixel_reveal->update(TIME_SINCE_TRIGGER); } } // Actualiza el revelado de la imagen (desde el inicio de la escena) sprite_pics_.at(current_scene_).pixel_reveal->update(state_time_); } // Comprueba si se ha de cambiar de escena void Ending::checkChangeScene() { // Obtener duración de la escena actual float current_duration = 0.0F; State next_state = State::ENDING; switch (state_) { case State::SCENE_0: current_duration = SCENE_0_DURATION; next_state = State::SCENE_1; break; case State::SCENE_1: current_duration = SCENE_1_DURATION; next_state = State::SCENE_2; break; case State::SCENE_2: current_duration = SCENE_2_DURATION; next_state = State::SCENE_3; break; case State::SCENE_3: current_duration = SCENE_3_DURATION; next_state = State::SCENE_4; break; case State::SCENE_4: current_duration = SCENE_4_DURATION; next_state = State::ENDING; break; default: return; } // Comprobar si ha pasado la duración de la escena if (state_time_ >= current_duration) { if (next_state == State::ENDING) { // Transición al estado ENDING con fade de audio transitionToState(State::ENDING); Audio::get()->fadeOutMusic(MUSIC_FADE_DURATION); } else { // Transición a la siguiente escena current_scene_++; transitionToState(next_state); } } }