Lint: rename públicos al inglés + refactor cognitive-complexity + unused-includes
Identifier-naming: rename de métodos públicos y cross-file al inglés
(camelBack), traducción de campos y locales en el proceso (TitleShip,
StageManager, SpawnController, ShipAnimator, helpers de PlayArea, etc.).
Refactor por cognitive-complexity (>25): GameScene::draw (59→3) con 9
helpers de estado, PhysicsWorld::resolveBodyCollisions (35→5) extrayendo
resolveBodyPair, Options::load{Window,Physics,Audio}ConfigFromYaml
(32/49/57→5/2/3) con templates readField, TitleScene::update (68→4) con
5 sub-pasos por estado + handleSkipInput/handleStartInput +
triggerExitForJoinedPlayers, DebrisManager::explode (39→3) con
extractSegments/spawnDebris/applyAngularVelocity/applyVisualRotation.
use-anyofallof: bucles → std::ranges::any_of/all_of en Input,
ShipAnimator y SpawnController.
readability-static-accessed-through-instance: Director::run y
VectorText::getTextWidth/Height invocados por clase.
readability-convert-member-functions-to-static: ResourcePack::decryptData.
unused-includes: eliminación de <utility>, <cstdint>, <vector>,
<iostream>, defaults.hpp y otros no usados directamente en headers y
unidades de traducción. Restablecido core/defaults.hpp en title_scene.cpp
(falsa "unused" del header).
Bug fix: eliminado isActive() duplicado en Bullet (redeclaración tras
rename de esta_activa→isActive que chocaba con el override de Entity).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -5,7 +5,6 @@
|
|||||||
#include <functional> // Para std::function
|
#include <functional> // Para std::function
|
||||||
#include <memory> // Para std::unique_ptr
|
#include <memory> // Para std::unique_ptr
|
||||||
#include <string> // Para string
|
#include <string> // Para string
|
||||||
#include <utility> // Para move
|
|
||||||
|
|
||||||
// Forward-declares per no incloure core/audio/jail_audio.hpp al header. Els
|
// Forward-declares per no incloure core/audio/jail_audio.hpp al header. Els
|
||||||
// tres símbols (Music/Sound para el punter que exposa la API i Engine per al
|
// tres símbols (Music/Sound para el punter que exposa la API i Engine per al
|
||||||
|
|||||||
@@ -2,7 +2,6 @@
|
|||||||
#include <SDL3/SDL.h>
|
#include <SDL3/SDL.h>
|
||||||
|
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
#include <cstdint>
|
|
||||||
#include <numbers>
|
#include <numbers>
|
||||||
|
|
||||||
namespace Defaults {
|
namespace Defaults {
|
||||||
@@ -439,16 +438,16 @@ constexpr float CENTER_Y = Game::HEIGHT / 2.0F; // auto-derivado de Game::HEIGH
|
|||||||
// Posicions target (calculades dinàmicament des dels parámetros base)
|
// Posicions target (calculades dinàmicament des dels parámetros base)
|
||||||
// Nota: std::cos/sin no són constexpr en C++20, pero funcionen en runtime
|
// Nota: std::cos/sin no són constexpr en C++20, pero funcionen en runtime
|
||||||
// Les funciones inline són optimitzades por el compilador (zero overhead)
|
// Les funciones inline són optimitzades por el compilador (zero overhead)
|
||||||
inline auto P1_TARGET_X() -> float {
|
inline auto p1TargetX() -> float {
|
||||||
return CENTER_X + (CLOCK_RADIUS * std::cos(CLOCK_8_ANGLE));
|
return CENTER_X + (CLOCK_RADIUS * std::cos(CLOCK_8_ANGLE));
|
||||||
}
|
}
|
||||||
inline auto P1_TARGET_Y() -> float {
|
inline auto p1TargetY() -> float {
|
||||||
return CENTER_Y + ((Game::HEIGHT / 2.0F) * TARGET_Y_RATIO);
|
return CENTER_Y + ((Game::HEIGHT / 2.0F) * TARGET_Y_RATIO);
|
||||||
}
|
}
|
||||||
inline auto P2_TARGET_X() -> float {
|
inline auto p2TargetX() -> float {
|
||||||
return CENTER_X + (CLOCK_RADIUS * std::cos(CLOCK_4_ANGLE));
|
return CENTER_X + (CLOCK_RADIUS * std::cos(CLOCK_4_ANGLE));
|
||||||
}
|
}
|
||||||
inline auto P2_TARGET_Y() -> float {
|
inline auto p2TargetY() -> float {
|
||||||
return CENTER_Y + ((Game::HEIGHT / 2.0F) * TARGET_Y_RATIO);
|
return CENTER_Y + ((Game::HEIGHT / 2.0F) * TARGET_Y_RATIO);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -37,11 +37,11 @@ class Shape {
|
|||||||
auto parseFile(const std::string& contingut) -> bool;
|
auto parseFile(const std::string& contingut) -> bool;
|
||||||
|
|
||||||
// Getters
|
// Getters
|
||||||
[[nodiscard]] auto get_primitives() const -> const std::vector<ShapePrimitive>& {
|
[[nodiscard]] auto getPrimitives() const -> const std::vector<ShapePrimitive>& {
|
||||||
return primitives_;
|
return primitives_;
|
||||||
}
|
}
|
||||||
[[nodiscard]] auto getCenter() const -> const Vec2& { return center_; }
|
[[nodiscard]] auto getCenter() const -> const Vec2& { return center_; }
|
||||||
[[nodiscard]] auto get_escala_defecte() const -> float { return escala_defecte_; }
|
[[nodiscard]] auto getDefaultScale() const -> float { return escala_defecte_; }
|
||||||
[[nodiscard]] auto isValid() const -> bool { return !primitives_.empty(); }
|
[[nodiscard]] auto isValid() const -> bool { return !primitives_.empty(); }
|
||||||
|
|
||||||
// Info de depuració
|
// Info de depuració
|
||||||
|
|||||||
@@ -10,13 +10,12 @@
|
|||||||
namespace Graphics {
|
namespace Graphics {
|
||||||
|
|
||||||
// Inicialización de variables estàtiques
|
// Inicialización de variables estàtiques
|
||||||
std::unordered_map<std::string, std::shared_ptr<Shape>> ShapeLoader::cache_;
|
std::unordered_map<std::string, std::shared_ptr<Shape>> ShapeLoader::cache;
|
||||||
std::string ShapeLoader::base_path_ = "data/shapes/";
|
|
||||||
|
|
||||||
auto ShapeLoader::load(const std::string& filename) -> std::shared_ptr<Shape> {
|
auto ShapeLoader::load(const std::string& filename) -> std::shared_ptr<Shape> {
|
||||||
// Check cache first
|
// Check cache first
|
||||||
auto it = cache_.find(filename);
|
auto it = cache.find(filename);
|
||||||
if (it != cache_.end()) {
|
if (it != cache.end()) {
|
||||||
std::cout << "[ShapeLoader] Cache hit: " << filename << '\n';
|
std::cout << "[ShapeLoader] Cache hit: " << filename << '\n';
|
||||||
return it->second; // Cache hit
|
return it->second; // Cache hit
|
||||||
}
|
}
|
||||||
@@ -56,17 +55,17 @@ auto ShapeLoader::load(const std::string& filename) -> std::shared_ptr<Shape> {
|
|||||||
std::cout << "[ShapeLoader] Carregat: " << normalized << " (" << shape->getName()
|
std::cout << "[ShapeLoader] Carregat: " << normalized << " (" << shape->getName()
|
||||||
<< ", " << shape->getNumPrimitives() << " primitives)" << '\n';
|
<< ", " << shape->getNumPrimitives() << " primitives)" << '\n';
|
||||||
|
|
||||||
cache_[filename] = shape;
|
cache[filename] = shape;
|
||||||
return shape;
|
return shape;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ShapeLoader::clear_cache() {
|
void ShapeLoader::clearCache() {
|
||||||
std::cout << "[ShapeLoader] Netejant caché (" << cache_.size() << " formes)"
|
std::cout << "[ShapeLoader] Netejant caché (" << cache.size() << " formes)"
|
||||||
<< '\n';
|
<< '\n';
|
||||||
cache_.clear();
|
cache.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto ShapeLoader::get_cache_size() -> size_t { return cache_.size(); }
|
auto ShapeLoader::getCacheSize() -> size_t { return cache.size(); }
|
||||||
|
|
||||||
auto ShapeLoader::resolvePath(const std::string& filename) -> std::string {
|
auto ShapeLoader::resolvePath(const std::string& filename) -> std::string {
|
||||||
// Si es un path absolut (comença con '/'), usar-lo directament
|
// Si es un path absolut (comença con '/'), usar-lo directament
|
||||||
@@ -75,12 +74,12 @@ auto ShapeLoader::resolvePath(const std::string& filename) -> std::string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Si ya conté el prefix base_path, usar-lo directament
|
// Si ya conté el prefix base_path, usar-lo directament
|
||||||
if (filename.starts_with(base_path_)) {
|
if (filename.starts_with(BASE_PATH)) {
|
||||||
return filename;
|
return filename;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Altrament, añadir base_path (ara suporta subdirectoris)
|
// Altrament, añadir base_path (ara suporta subdirectoris)
|
||||||
return base_path_ + filename;
|
return std::string(BASE_PATH) + filename;
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace Graphics
|
} // namespace Graphics
|
||||||
|
|||||||
@@ -23,14 +23,14 @@ class ShapeLoader {
|
|||||||
static auto load(const std::string& filename) -> std::shared_ptr<Shape>;
|
static auto load(const std::string& filename) -> std::shared_ptr<Shape>;
|
||||||
|
|
||||||
// Netejar caché (útil per debug/recàrrega)
|
// Netejar caché (útil per debug/recàrrega)
|
||||||
static void clear_cache();
|
static void clearCache();
|
||||||
|
|
||||||
// Estadístiques (debug)
|
// Estadístiques (debug)
|
||||||
static auto get_cache_size() -> size_t;
|
static auto getCacheSize() -> size_t;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static std::unordered_map<std::string, std::shared_ptr<Shape>> cache_;
|
static std::unordered_map<std::string, std::shared_ptr<Shape>> cache;
|
||||||
static std::string base_path_; // "data/shapes/"
|
static constexpr const char* BASE_PATH = "data/shapes/";
|
||||||
|
|
||||||
// Helpers privats
|
// Helpers privats
|
||||||
static auto resolvePath(const std::string& filename) -> std::string;
|
static auto resolvePath(const std::string& filename) -> std::string;
|
||||||
|
|||||||
@@ -137,7 +137,7 @@ void Starfield::update(float delta_time) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Establir multiplicador de brightness
|
// Establir multiplicador de brightness
|
||||||
void Starfield::set_brightness(float multiplier) {
|
void Starfield::setBrightness(float multiplier) {
|
||||||
multiplicador_brightness_ = std::max(0.0F, multiplier); // Evitar valors negatius
|
multiplicador_brightness_ = std::max(0.0F, multiplier); // Evitar valors negatius
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -153,7 +153,7 @@ void Starfield::draw() {
|
|||||||
float brightness = computeBrightness(estrella);
|
float brightness = computeBrightness(estrella);
|
||||||
|
|
||||||
// Renderizar estrella sin rotación
|
// Renderizar estrella sin rotación
|
||||||
Rendering::render_shape(
|
Rendering::renderShape(
|
||||||
renderer_,
|
renderer_,
|
||||||
shape_estrella_,
|
shape_estrella_,
|
||||||
estrella.position,
|
estrella.position,
|
||||||
|
|||||||
@@ -43,8 +43,8 @@ class Starfield {
|
|||||||
void draw();
|
void draw();
|
||||||
|
|
||||||
// Setters per ajustar parámetros en time real
|
// Setters per ajustar parámetros en time real
|
||||||
void set_punt_fuga(const Vec2& point) { punt_fuga_ = point; }
|
void setVanishingPoint(const Vec2& point) { punt_fuga_ = point; }
|
||||||
void set_brightness(float multiplier);
|
void setBrightness(float multiplier);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
// Estructura interna per cada estrella
|
// Estructura interna per cada estrella
|
||||||
|
|||||||
@@ -175,7 +175,7 @@ auto VectorText::getShapeFilename(char c) -> std::string {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto VectorText::is_supported(char c) const -> bool {
|
auto VectorText::isSupported(char c) const -> bool {
|
||||||
return chars_.contains(c);
|
return chars_.contains(c);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -221,7 +221,7 @@ void VectorText::render(const std::string& text, const Vec2& position, float sca
|
|||||||
// Ajustar X e Y para que position represente esquina superior izquierda
|
// Ajustar X e Y para que position represente esquina superior izquierda
|
||||||
// (render_shape espera el centro, así que sumamos la mitad de ancho y altura)
|
// (render_shape espera el centro, así que sumamos la mitad de ancho y altura)
|
||||||
Vec2 char_pos = {.x = current_x + (CHAR_WIDTH_SCALED / 2.0F), .y = position.y + (CHAR_HEIGHT_SCALED / 2.0F)};
|
Vec2 char_pos = {.x = current_x + (CHAR_WIDTH_SCALED / 2.0F), .y = position.y + (CHAR_HEIGHT_SCALED / 2.0F)};
|
||||||
Rendering::render_shape(renderer_, it->second, char_pos, 0.0F, scale, 1.0F, brightness);
|
Rendering::renderShape(renderer_, it->second, char_pos, 0.0F, scale, 1.0F, brightness);
|
||||||
|
|
||||||
// Avanzar posición
|
// Avanzar posición
|
||||||
current_x += CHAR_WIDTH_SCALED + SPACING_SCALED;
|
current_x += CHAR_WIDTH_SCALED + SPACING_SCALED;
|
||||||
@@ -236,8 +236,8 @@ void VectorText::render(const std::string& text, const Vec2& position, float sca
|
|||||||
|
|
||||||
void VectorText::renderCentered(const std::string& text, const Vec2& centre_punt, float scale, float spacing, float brightness) const {
|
void VectorText::renderCentered(const std::string& text, const Vec2& centre_punt, float scale, float spacing, float brightness) const {
|
||||||
// Calcular dimensions del text
|
// Calcular dimensions del text
|
||||||
float text_width = get_text_width(text, scale, spacing);
|
float text_width = getTextWidth(text, scale, spacing);
|
||||||
float text_height = get_text_height(scale);
|
float text_height = getTextHeight(scale);
|
||||||
|
|
||||||
// Calcular posición de l'esquina superior izquierda
|
// Calcular posición de l'esquina superior izquierda
|
||||||
// restant la meitat de las dimensions del point central
|
// restant la meitat de las dimensions del point central
|
||||||
@@ -249,7 +249,7 @@ void VectorText::renderCentered(const std::string& text, const Vec2& centre_punt
|
|||||||
render(text, posicio_esquerra, scale, spacing, brightness);
|
render(text, posicio_esquerra, scale, spacing, brightness);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto VectorText::get_text_width(const std::string& text, float scale, float spacing) -> float {
|
auto VectorText::getTextWidth(const std::string& text, float scale, float spacing) -> float {
|
||||||
if (text.empty()) {
|
if (text.empty()) {
|
||||||
return 0.0F;
|
return 0.0F;
|
||||||
}
|
}
|
||||||
@@ -276,7 +276,7 @@ auto VectorText::get_text_width(const std::string& text, float scale, float spac
|
|||||||
return (visual_chars * CHAR_WIDTH_SCALED) + ((visual_chars - 1) * SPACING_SCALED);
|
return (visual_chars * CHAR_WIDTH_SCALED) + ((visual_chars - 1) * SPACING_SCALED);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto VectorText::get_text_height(float scale) -> float {
|
auto VectorText::getTextHeight(float scale) -> float {
|
||||||
return BASE_CHAR_HEIGHT * scale;
|
return BASE_CHAR_HEIGHT * scale;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -40,13 +40,13 @@ class VectorText {
|
|||||||
// Calcular ancho total de un string (útil para centrado).
|
// Calcular ancho total de un string (útil para centrado).
|
||||||
// Es estático: no depende del estado del VectorText (el ancho viene de
|
// Es estático: no depende del estado del VectorText (el ancho viene de
|
||||||
// las constantes BASE_CHAR_WIDTH/BASE_CHAR_HEIGHT del archivo .cpp).
|
// las constantes BASE_CHAR_WIDTH/BASE_CHAR_HEIGHT del archivo .cpp).
|
||||||
[[nodiscard]] static auto get_text_width(const std::string& text, float scale = 1.0F, float spacing = 2.0F) -> float;
|
[[nodiscard]] static auto getTextWidth(const std::string& text, float scale = 1.0F, float spacing = 2.0F) -> float;
|
||||||
|
|
||||||
// Calcular altura del texto (útil para centrado vertical).
|
// Calcular altura del texto (útil para centrado vertical).
|
||||||
[[nodiscard]] static auto get_text_height(float scale = 1.0F) -> float;
|
[[nodiscard]] static auto getTextHeight(float scale = 1.0F) -> float;
|
||||||
|
|
||||||
// Verificar si un carácter está soportado
|
// Verificar si un carácter está soportado
|
||||||
[[nodiscard]] auto is_supported(char c) const -> bool;
|
[[nodiscard]] auto isSupported(char c) const -> bool;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Rendering::Renderer* renderer_;
|
Rendering::Renderer* renderer_;
|
||||||
|
|||||||
@@ -2,11 +2,11 @@
|
|||||||
|
|
||||||
#include <SDL3/SDL.h> // Para SDL_GetGamepadAxis, SDL_GamepadAxis, SDL_GamepadButton, SDL_GetError, SDL_JoystickID, SDL_AddGamepadMappingsFromFile, SDL_Event, SDL_EventType, SDL_GetGamepadButton, SDL_GetKeyboardState, SDL_INIT_GAMEPAD, SDL_InitSubSystem, SDL_LogError, SDL_OpenGamepad, SDL_PollEvent, SDL_WasInit, Sint16, SDL_Gamepad, SDL_LogCategory, SDL_Scancode
|
#include <SDL3/SDL.h> // Para SDL_GetGamepadAxis, SDL_GamepadAxis, SDL_GamepadButton, SDL_GetError, SDL_JoystickID, SDL_AddGamepadMappingsFromFile, SDL_Event, SDL_EventType, SDL_GetGamepadButton, SDL_GetKeyboardState, SDL_INIT_GAMEPAD, SDL_InitSubSystem, SDL_LogError, SDL_OpenGamepad, SDL_PollEvent, SDL_WasInit, Sint16, SDL_Gamepad, SDL_LogCategory, SDL_Scancode
|
||||||
|
|
||||||
|
#include <algorithm> // Para std::ranges::any_of
|
||||||
#include <iostream> // Para basic_ostream, operator<<, cout, cerr
|
#include <iostream> // Para basic_ostream, operator<<, cout, cerr
|
||||||
#include <memory> // Para shared_ptr, __shared_ptr_access, allocator, operator==, make_shared
|
#include <memory> // Para shared_ptr, __shared_ptr_access, allocator, operator==, make_shared
|
||||||
#include <ranges> // Para __find_if_fn, find_if
|
|
||||||
#include <unordered_map> // Para unordered_map, _Node_iterator, operator==, _Node_iterator_base, _Node_const_iterator
|
#include <unordered_map> // Para unordered_map, _Node_iterator, operator==, _Node_iterator_base, _Node_const_iterator
|
||||||
#include <utility> // Para pair, move
|
#include <utility> // Para move
|
||||||
|
|
||||||
#include "game/options.hpp" // Para Options::controls
|
#include "game/options.hpp" // Para Options::controls
|
||||||
|
|
||||||
@@ -190,12 +190,9 @@ auto Input::checkAnyButton(bool repeat) -> bool {
|
|||||||
|
|
||||||
// Comprueba si algún player (P1 o P2) presionó alguna acción de una lista
|
// Comprueba si algún player (P1 o P2) presionó alguna acción de una lista
|
||||||
auto Input::checkAnyPlayerAction(const std::span<const InputAction>& actions, bool repeat) -> bool {
|
auto Input::checkAnyPlayerAction(const std::span<const InputAction>& actions, bool repeat) -> bool {
|
||||||
for (const auto& action : actions) {
|
return std::ranges::any_of(actions, [this, repeat](const InputAction& action) {
|
||||||
if (checkActionPlayer1(action, repeat) || checkActionPlayer2(action, repeat)) {
|
return checkActionPlayer1(action, repeat) || checkActionPlayer2(action, repeat);
|
||||||
return true;
|
});
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Comprueba si hay algun mando conectado
|
// Comprueba si hay algun mando conectado
|
||||||
|
|||||||
@@ -7,7 +7,6 @@
|
|||||||
#include <span> // Para span
|
#include <span> // Para span
|
||||||
#include <string> // Para string, basic_string
|
#include <string> // Para string, basic_string
|
||||||
#include <unordered_map> // Para unordered_map
|
#include <unordered_map> // Para unordered_map
|
||||||
#include <utility> // Para pair
|
|
||||||
#include <vector> // Para vector
|
#include <vector> // Para vector
|
||||||
|
|
||||||
#include "core/input/input_types.hpp" // for InputAction
|
#include "core/input/input_types.hpp" // for InputAction
|
||||||
|
|||||||
@@ -1,7 +1,5 @@
|
|||||||
#include "input_types.hpp"
|
#include "input_types.hpp"
|
||||||
|
|
||||||
#include <utility> // Para pair
|
|
||||||
|
|
||||||
// Definición de los mapas
|
// Definición de los mapas
|
||||||
const std::unordered_map<InputAction, std::string> ACTION_TO_STRING = {
|
const std::unordered_map<InputAction, std::string> ACTION_TO_STRING = {
|
||||||
{InputAction::LEFT, "LEFT"},
|
{InputAction::LEFT, "LEFT"},
|
||||||
|
|||||||
@@ -8,21 +8,21 @@ namespace Easing {
|
|||||||
// Ease-out quadratic: empieza rápido, desacelera suavemente
|
// Ease-out quadratic: empieza rápido, desacelera suavemente
|
||||||
// t = progreso normalizado [0.0 - 1.0]
|
// t = progreso normalizado [0.0 - 1.0]
|
||||||
// retorna value interpolado [0.0 - 1.0]
|
// retorna value interpolado [0.0 - 1.0]
|
||||||
inline auto ease_out_quad(float t) -> float {
|
inline auto easeOutQuad(float t) -> float {
|
||||||
return 1.0F - ((1.0F - t) * (1.0F - t));
|
return 1.0F - ((1.0F - t) * (1.0F - t));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ease-in quadratic: empieza lento, acelera
|
// Ease-in quadratic: empieza lento, acelera
|
||||||
// t = progreso normalizado [0.0 - 1.0]
|
// t = progreso normalizado [0.0 - 1.0]
|
||||||
// retorna value interpolado [0.0 - 1.0]
|
// retorna value interpolado [0.0 - 1.0]
|
||||||
inline auto ease_in_quad(float t) -> float {
|
inline auto easeInQuad(float t) -> float {
|
||||||
return t * t;
|
return t * t;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ease-in-out quadratic: acelera al inicio, desacelera al final
|
// Ease-in-out quadratic: acelera al inicio, desacelera al final
|
||||||
// t = progreso normalizado [0.0 - 1.0]
|
// t = progreso normalizado [0.0 - 1.0]
|
||||||
// retorna value interpolado [0.0 - 1.0]
|
// retorna value interpolado [0.0 - 1.0]
|
||||||
inline auto ease_in_out_quad(float t) -> float {
|
inline auto easeInOutQuad(float t) -> float {
|
||||||
return (t < 0.5F)
|
return (t < 0.5F)
|
||||||
? 2.0F * t * t
|
? 2.0F * t * t
|
||||||
: 1.0F - ((-2.0F * t + 2.0F) * (-2.0F * t + 2.0F) / 2.0F);
|
: 1.0F - ((-2.0F * t + 2.0F) * (-2.0F * t + 2.0F) / 2.0F);
|
||||||
@@ -31,7 +31,7 @@ inline auto ease_in_out_quad(float t) -> float {
|
|||||||
// Ease-out cubic: desaceleración más suave que quadratic
|
// Ease-out cubic: desaceleración más suave que quadratic
|
||||||
// t = progreso normalizado [0.0 - 1.0]
|
// t = progreso normalizado [0.0 - 1.0]
|
||||||
// retorna value interpolado [0.0 - 1.0]
|
// retorna value interpolado [0.0 - 1.0]
|
||||||
inline auto ease_out_cubic(float t) -> float {
|
inline auto easeOutCubic(float t) -> float {
|
||||||
float t1 = 1.0F - t;
|
float t1 = 1.0F - t;
|
||||||
return 1.0F - (t1 * t1 * t1);
|
return 1.0F - (t1 * t1 * t1);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,7 +9,7 @@
|
|||||||
namespace Physics {
|
namespace Physics {
|
||||||
|
|
||||||
// Comprobación genèrica de colisión entre dues entidades
|
// Comprobación genèrica de colisión entre dues entidades
|
||||||
inline auto check_collision(const Entities::Entity& a, const Entities::Entity& b, float amplifier = 1.0F) -> bool {
|
inline auto checkCollision(const Entities::Entity& a, const Entities::Entity& b, float amplifier = 1.0F) -> bool {
|
||||||
// Comprovar si ambdós són col·lisionables
|
// Comprovar si ambdós són col·lisionables
|
||||||
if (!a.isCollidable() || !b.isCollidable()) {
|
if (!a.isCollidable() || !b.isCollidable()) {
|
||||||
return false;
|
return false;
|
||||||
|
|||||||
@@ -121,19 +121,24 @@ void PhysicsWorld::resolveBodyCollisions() {
|
|||||||
for (std::size_t j = i + 1; j < COUNT; ++j) {
|
for (std::size_t j = i + 1; j < COUNT; ++j) {
|
||||||
auto* a = bodies_[i];
|
auto* a = bodies_[i];
|
||||||
auto* b = bodies_[j];
|
auto* b = bodies_[j];
|
||||||
if (a == nullptr || b == nullptr) {
|
if (a != nullptr && b != nullptr) {
|
||||||
continue;
|
resolveBodyPair(*a, *b);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// Dos cuerpos estáticos no necesitan resolución
|
|
||||||
if (a->isStatic() && b->isStatic()) {
|
|
||||||
continue;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const Vec2 DELTA = b->position - a->position;
|
void PhysicsWorld::resolveBodyPair(RigidBody& a, RigidBody& b) {
|
||||||
|
// Dos cuerpos estáticos no necesitan resolución
|
||||||
|
if (a.isStatic() && b.isStatic()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const Vec2 DELTA = b.position - a.position;
|
||||||
const float DIST_SQ = DELTA.lengthSquared();
|
const float DIST_SQ = DELTA.lengthSquared();
|
||||||
const float SUM_R = a->radius + b->radius;
|
const float SUM_R = a.radius + b.radius;
|
||||||
if (DIST_SQ > SUM_R * SUM_R || DIST_SQ <= 0.0F) {
|
if (DIST_SQ > SUM_R * SUM_R || DIST_SQ <= 0.0F) {
|
||||||
continue;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const float DIST = std::sqrt(DIST_SQ);
|
const float DIST = std::sqrt(DIST_SQ);
|
||||||
@@ -141,37 +146,35 @@ void PhysicsWorld::resolveBodyCollisions() {
|
|||||||
|
|
||||||
// Corrección posicional (resolver penetración)
|
// Corrección posicional (resolver penetración)
|
||||||
const float PENETRATION = SUM_R - DIST;
|
const float PENETRATION = SUM_R - DIST;
|
||||||
const float TOTAL_INV_MASS = a->inverse_mass + b->inverse_mass;
|
const float TOTAL_INV_MASS = a.inverse_mass + b.inverse_mass;
|
||||||
if (TOTAL_INV_MASS > 0.0F) {
|
if (TOTAL_INV_MASS > 0.0F) {
|
||||||
const Vec2 CORRECTION = NORMAL * (PENETRATION / TOTAL_INV_MASS);
|
const Vec2 CORRECTION = NORMAL * (PENETRATION / TOTAL_INV_MASS);
|
||||||
if (!a->isStatic()) {
|
if (!a.isStatic()) {
|
||||||
a->position -= CORRECTION * a->inverse_mass;
|
a.position -= CORRECTION * a.inverse_mass;
|
||||||
}
|
}
|
||||||
if (!b->isStatic()) {
|
if (!b.isStatic()) {
|
||||||
b->position += CORRECTION * b->inverse_mass;
|
b.position += CORRECTION * b.inverse_mass;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Velocidad relativa proyectada sobre la normal
|
// Velocidad relativa proyectada sobre la normal
|
||||||
const Vec2 V_REL = b->velocity - a->velocity;
|
const Vec2 V_REL = b.velocity - a.velocity;
|
||||||
const float VEL_ALONG_NORMAL = V_REL.dot(NORMAL);
|
const float VEL_ALONG_NORMAL = V_REL.dot(NORMAL);
|
||||||
// Si se están separando, no aplicar impulso
|
// Si se están separando, no aplicar impulso
|
||||||
if (VEL_ALONG_NORMAL > 0.0F) {
|
if (VEL_ALONG_NORMAL > 0.0F) {
|
||||||
continue;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Restitución promedio (Box2D usa max; promedio es más permisivo)
|
// Restitución promedio (Box2D usa max; promedio es más permisivo)
|
||||||
const float E = (a->restitution + b->restitution) * 0.5F;
|
const float E = (a.restitution + b.restitution) * 0.5F;
|
||||||
const float J = -(1.0F + E) * VEL_ALONG_NORMAL / TOTAL_INV_MASS;
|
const float J = -(1.0F + E) * VEL_ALONG_NORMAL / TOTAL_INV_MASS;
|
||||||
const Vec2 IMPULSE = NORMAL * J;
|
const Vec2 IMPULSE = NORMAL * J;
|
||||||
|
|
||||||
if (!a->isStatic()) {
|
if (!a.isStatic()) {
|
||||||
a->velocity -= IMPULSE * a->inverse_mass;
|
a.velocity -= IMPULSE * a.inverse_mass;
|
||||||
}
|
|
||||||
if (!b->isStatic()) {
|
|
||||||
b->velocity += IMPULSE * b->inverse_mass;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
if (!b.isStatic()) {
|
||||||
|
b.velocity += IMPULSE * b.inverse_mass;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -59,6 +59,9 @@ class PhysicsWorld {
|
|||||||
void integrate(float dt);
|
void integrate(float dt);
|
||||||
void resolveBoundsCollisions();
|
void resolveBoundsCollisions();
|
||||||
void resolveBodyCollisions();
|
void resolveBodyCollisions();
|
||||||
|
// Resol un únic parell (a, b): correcció posicional + impulso elàstic.
|
||||||
|
// Estàtic: només toca els dos cossos rebuts, no consulta el world.
|
||||||
|
static void resolveBodyPair(RigidBody& a, RigidBody& b);
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace Physics
|
} // namespace Physics
|
||||||
|
|||||||
@@ -11,21 +11,21 @@ namespace Rendering {
|
|||||||
extern float g_current_scale_factor;
|
extern float g_current_scale_factor;
|
||||||
|
|
||||||
// Transforma coordenada lógica a física con arrodoniment
|
// Transforma coordenada lógica a física con arrodoniment
|
||||||
inline auto transform_x(int logical_x, float scale) -> int {
|
inline auto transformX(int logical_x, float scale) -> int {
|
||||||
return static_cast<int>(std::round(logical_x * scale));
|
return static_cast<int>(std::round(logical_x * scale));
|
||||||
}
|
}
|
||||||
|
|
||||||
inline auto transform_y(int logical_y, float scale) -> int {
|
inline auto transformY(int logical_y, float scale) -> int {
|
||||||
return static_cast<int>(std::round(logical_y * scale));
|
return static_cast<int>(std::round(logical_y * scale));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Variant que usa el factor de scale global
|
// Variant que usa el factor de scale global
|
||||||
inline auto transform_x(int logical_x) -> int {
|
inline auto transformX(int logical_x) -> int {
|
||||||
return transform_x(logical_x, g_current_scale_factor);
|
return transformX(logical_x, g_current_scale_factor);
|
||||||
}
|
}
|
||||||
|
|
||||||
inline auto transform_y(int logical_y) -> int {
|
inline auto transformY(int logical_y) -> int {
|
||||||
return transform_y(logical_y, g_current_scale_factor);
|
return transformY(logical_y, g_current_scale_factor);
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace Rendering
|
} // namespace Rendering
|
||||||
|
|||||||
@@ -362,7 +362,7 @@ void GpuFrameRenderer::compositePass() {
|
|||||||
ubo.flicker_amplitude = FLICKER_AMPLITUDE;
|
ubo.flicker_amplitude = FLICKER_AMPLITUDE;
|
||||||
ubo.flicker_frequency_hz = postfx_params_.flicker_frequency_hz;
|
ubo.flicker_frequency_hz = postfx_params_.flicker_frequency_hz;
|
||||||
ubo.background_pulse_freq_hz = postfx_params_.background_pulse_freq_hz;
|
ubo.background_pulse_freq_hz = postfx_params_.background_pulse_freq_hz;
|
||||||
ubo.pad_a_ = 0.0F;
|
ubo.pad_a = 0.0F;
|
||||||
ubo.background_min_r = BG_MIN_R;
|
ubo.background_min_r = BG_MIN_R;
|
||||||
ubo.background_min_g = BG_MIN_G;
|
ubo.background_min_g = BG_MIN_G;
|
||||||
ubo.background_min_b = BG_MIN_B;
|
ubo.background_min_b = BG_MIN_B;
|
||||||
@@ -373,8 +373,8 @@ void GpuFrameRenderer::compositePass() {
|
|||||||
ubo.background_max_a = 1.0F;
|
ubo.background_max_a = 1.0F;
|
||||||
ubo.texel_size_x = 1.0F / logical_w_;
|
ubo.texel_size_x = 1.0F / logical_w_;
|
||||||
ubo.texel_size_y = 1.0F / logical_h_;
|
ubo.texel_size_y = 1.0F / logical_h_;
|
||||||
ubo.pad_b_ = 0.0F;
|
ubo.pad_b = 0.0F;
|
||||||
ubo.pad_c_ = 0.0F;
|
ubo.pad_c = 0.0F;
|
||||||
|
|
||||||
SDL_PushGPUFragmentUniformData(cmd_buffer_, 0, &ubo, sizeof(ubo));
|
SDL_PushGPUFragmentUniformData(cmd_buffer_, 0, &ubo, sizeof(ubo));
|
||||||
|
|
||||||
|
|||||||
@@ -9,8 +9,6 @@
|
|||||||
|
|
||||||
#include <SDL3/SDL_gpu.h>
|
#include <SDL3/SDL_gpu.h>
|
||||||
|
|
||||||
#include <cstdint>
|
|
||||||
|
|
||||||
namespace Rendering::GPU {
|
namespace Rendering::GPU {
|
||||||
|
|
||||||
class GpuDevice;
|
class GpuDevice;
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ struct PostFxUniforms {
|
|||||||
float flicker_amplitude; // Profundidad del flicker (0..1)
|
float flicker_amplitude; // Profundidad del flicker (0..1)
|
||||||
float flicker_frequency_hz; // Hz
|
float flicker_frequency_hz; // Hz
|
||||||
float background_pulse_freq_hz; // Hz
|
float background_pulse_freq_hz; // Hz
|
||||||
float pad_a_;
|
float pad_a;
|
||||||
|
|
||||||
float background_min_r; // Color min RGB en [0..1], A=1
|
float background_min_r; // Color min RGB en [0..1], A=1
|
||||||
float background_min_g;
|
float background_min_g;
|
||||||
@@ -43,8 +43,8 @@ struct PostFxUniforms {
|
|||||||
|
|
||||||
float texel_size_x; // 1.0 / texture_width
|
float texel_size_x; // 1.0 / texture_width
|
||||||
float texel_size_y;
|
float texel_size_y;
|
||||||
float pad_b_;
|
float pad_b;
|
||||||
float pad_c_;
|
float pad_c;
|
||||||
};
|
};
|
||||||
|
|
||||||
class GpuPostFxPipeline {
|
class GpuPostFxPipeline {
|
||||||
|
|||||||
@@ -13,7 +13,6 @@
|
|||||||
#include "core/defaults.hpp"
|
#include "core/defaults.hpp"
|
||||||
#include "core/input/mouse.hpp"
|
#include "core/input/mouse.hpp"
|
||||||
#include "core/rendering/coordinate_transform.hpp"
|
#include "core/rendering/coordinate_transform.hpp"
|
||||||
#include "core/rendering/line_renderer.hpp"
|
|
||||||
#include "game/options.hpp"
|
#include "game/options.hpp"
|
||||||
#include "project.h"
|
#include "project.h"
|
||||||
|
|
||||||
|
|||||||
@@ -5,7 +5,6 @@
|
|||||||
|
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
|
|
||||||
#include "core/defaults.hpp"
|
|
||||||
#include "core/rendering/line_renderer.hpp"
|
#include "core/rendering/line_renderer.hpp"
|
||||||
|
|
||||||
namespace Rendering {
|
namespace Rendering {
|
||||||
@@ -48,7 +47,7 @@ static auto transformPoint(const Vec2& point, const Vec2& shape_centre, const Ve
|
|||||||
float centered_y = point.y - shape_centre.y;
|
float centered_y = point.y - shape_centre.y;
|
||||||
|
|
||||||
// 2. Aplicar rotación 3D (si es proporciona)
|
// 2. Aplicar rotación 3D (si es proporciona)
|
||||||
if ((rotation_3d != nullptr) && rotation_3d->has_rotation()) {
|
if ((rotation_3d != nullptr) && rotation_3d->hasRotation()) {
|
||||||
Vec2 rotated_3d = apply3dRotation(centered_x, centered_y, *rotation_3d);
|
Vec2 rotated_3d = apply3dRotation(centered_x, centered_y, *rotation_3d);
|
||||||
centered_x = rotated_3d.x;
|
centered_x = rotated_3d.x;
|
||||||
centered_y = rotated_3d.y;
|
centered_y = rotated_3d.y;
|
||||||
@@ -72,7 +71,7 @@ static auto transformPoint(const Vec2& point, const Vec2& shape_centre, const Ve
|
|||||||
return {.x = rotated_x + position.x, .y = rotated_y + position.y};
|
return {.x = rotated_x + position.x, .y = rotated_y + position.y};
|
||||||
}
|
}
|
||||||
|
|
||||||
void render_shape(Rendering::Renderer* renderer,
|
void renderShape(Rendering::Renderer* renderer,
|
||||||
const std::shared_ptr<Graphics::Shape>& shape,
|
const std::shared_ptr<Graphics::Shape>& shape,
|
||||||
const Vec2& position,
|
const Vec2& position,
|
||||||
float angle,
|
float angle,
|
||||||
@@ -90,7 +89,7 @@ void render_shape(Rendering::Renderer* renderer,
|
|||||||
|
|
||||||
const Vec2& shape_centre = shape->getCenter();
|
const Vec2& shape_centre = shape->getCenter();
|
||||||
|
|
||||||
for (const auto& primitive : shape->get_primitives()) {
|
for (const auto& primitive : shape->getPrimitives()) {
|
||||||
if (primitive.type == Graphics::PrimitiveType::POLYLINE) {
|
if (primitive.type == Graphics::PrimitiveType::POLYLINE) {
|
||||||
// POLYLINE: conectar puntos consecutivos.
|
// POLYLINE: conectar puntos consecutivos.
|
||||||
for (size_t i = 0; i < primitive.points.size() - 1; i++) {
|
for (size_t i = 0; i < primitive.points.size() - 1; i++) {
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ struct Rotation3D {
|
|||||||
yaw(y),
|
yaw(y),
|
||||||
roll(r) {}
|
roll(r) {}
|
||||||
|
|
||||||
[[nodiscard]] auto has_rotation() const -> bool {
|
[[nodiscard]] auto hasRotation() const -> bool {
|
||||||
return pitch != 0.0F || yaw != 0.0F || roll != 0.0F;
|
return pitch != 0.0F || yaw != 0.0F || roll != 0.0F;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@@ -42,7 +42,7 @@ struct Rotation3D {
|
|||||||
// - scale: factor de scale (1.0 = mida original)
|
// - scale: factor de scale (1.0 = mida original)
|
||||||
// - progress: progrés de l'animación (0.0-1.0, default 1.0 = tot visible)
|
// - progress: progrés de l'animación (0.0-1.0, default 1.0 = tot visible)
|
||||||
// - brightness: factor de brightness (0.0-1.0, default 1.0 = màxima brightness)
|
// - brightness: factor de brightness (0.0-1.0, default 1.0 = màxima brightness)
|
||||||
void render_shape(Rendering::Renderer* renderer,
|
void renderShape(Rendering::Renderer* renderer,
|
||||||
const std::shared_ptr<Graphics::Shape>& shape,
|
const std::shared_ptr<Graphics::Shape>& shape,
|
||||||
const Vec2& position,
|
const Vec2& position,
|
||||||
float angle,
|
float angle,
|
||||||
|
|||||||
@@ -4,7 +4,6 @@
|
|||||||
#include "resource_helper.hpp"
|
#include "resource_helper.hpp"
|
||||||
|
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <iostream>
|
|
||||||
|
|
||||||
#include "resource_loader.hpp"
|
#include "resource_loader.hpp"
|
||||||
|
|
||||||
|
|||||||
@@ -11,8 +11,8 @@ namespace Resource {
|
|||||||
|
|
||||||
// Singleton
|
// Singleton
|
||||||
auto Loader::get() -> Loader& {
|
auto Loader::get() -> Loader& {
|
||||||
static Loader instance;
|
static Loader instance_;
|
||||||
return instance;
|
return instance_;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Inicialitzar el sistema de recursos
|
// Inicialitzar el sistema de recursos
|
||||||
|
|||||||
@@ -62,7 +62,7 @@ class Pack {
|
|||||||
static auto readFile(const std::string& filepath) -> std::vector<uint8_t>;
|
static auto readFile(const std::string& filepath) -> std::vector<uint8_t>;
|
||||||
[[nodiscard]] static auto calculateChecksum(const std::vector<uint8_t>& data) -> uint32_t;
|
[[nodiscard]] static auto calculateChecksum(const std::vector<uint8_t>& data) -> uint32_t;
|
||||||
static void encryptData(std::vector<uint8_t>& data, const std::string& key);
|
static void encryptData(std::vector<uint8_t>& data, const std::string& key);
|
||||||
void decryptData(std::vector<uint8_t>& data, const std::string& key);
|
static void decryptData(std::vector<uint8_t>& data, const std::string& key);
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace Resource
|
} // namespace Resource
|
||||||
|
|||||||
@@ -19,29 +19,29 @@ struct MatchConfig {
|
|||||||
// Métodos auxiliars
|
// Métodos auxiliars
|
||||||
|
|
||||||
// Retorna true si solo hay un player active
|
// Retorna true si solo hay un player active
|
||||||
[[nodiscard]] auto es_un_jugador() const -> bool {
|
[[nodiscard]] auto isSinglePlayer() const -> bool {
|
||||||
return (jugador1_actiu && !jugador2_actiu) ||
|
return (jugador1_actiu && !jugador2_actiu) ||
|
||||||
(!jugador1_actiu && jugador2_actiu);
|
(!jugador1_actiu && jugador2_actiu);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Retorna true si hay dos jugadors active
|
// Retorna true si hay dos jugadors active
|
||||||
[[nodiscard]] auto son_dos_jugadors() const -> bool {
|
[[nodiscard]] auto isCoop() const -> bool {
|
||||||
return jugador1_actiu && jugador2_actiu;
|
return jugador1_actiu && jugador2_actiu;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Retorna true si no hay sin player active
|
// Retorna true si no hay sin player active
|
||||||
[[nodiscard]] auto cap_jugador() const -> bool {
|
[[nodiscard]] auto hasNoPlayers() const -> bool {
|
||||||
return !jugador1_actiu && !jugador2_actiu;
|
return !jugador1_actiu && !jugador2_actiu;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Compte de jugadors active (0, 1 o 2)
|
// Compte de jugadors active (0, 1 o 2)
|
||||||
[[nodiscard]] auto compte_jugadors() const -> uint8_t {
|
[[nodiscard]] auto getPlayerCount() const -> uint8_t {
|
||||||
return (jugador1_actiu ? 1 : 0) + (jugador2_actiu ? 1 : 0);
|
return (jugador1_actiu ? 1 : 0) + (jugador2_actiu ? 1 : 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Retorna l'ID de l'únic player active (0 o 1)
|
// Retorna l'ID de l'únic player active (0 o 1)
|
||||||
// Solo vàlid si es_un_jugador() retorna true
|
// Solo vàlid si es_un_jugador() retorna true
|
||||||
[[nodiscard]] auto id_unic_jugador() const -> uint8_t {
|
[[nodiscard]] auto getSinglePlayerId() const -> uint8_t {
|
||||||
if (jugador1_actiu && !jugador2_actiu) {
|
if (jugador1_actiu && !jugador2_actiu) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,39 +10,39 @@
|
|||||||
namespace Utils {
|
namespace Utils {
|
||||||
|
|
||||||
// Variables globals per guardar argv[0]
|
// Variables globals per guardar argv[0]
|
||||||
static std::string executable_path_;
|
static std::string executable_path;
|
||||||
static std::string executable_directory_;
|
static std::string executable_directory;
|
||||||
|
|
||||||
// Inicialitzar el sistema de rutes con argv[0]
|
// Inicialitzar el sistema de rutes con argv[0]
|
||||||
void initializePathSystem(const char* argv0) {
|
void initializePathSystem(const char* argv0) {
|
||||||
if (argv0 == nullptr) {
|
if (argv0 == nullptr) {
|
||||||
std::cerr << "[PathUtils] ADVERTÈNCIA: argv[0] es nullptr\n";
|
std::cerr << "[PathUtils] ADVERTÈNCIA: argv[0] es nullptr\n";
|
||||||
executable_path_ = "";
|
executable_path = "";
|
||||||
executable_directory_ = ".";
|
executable_directory = ".";
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
executable_path_ = argv0;
|
executable_path = argv0;
|
||||||
|
|
||||||
// Extreure el directori
|
// Extreure el directori
|
||||||
std::filesystem::path path(argv0);
|
std::filesystem::path path(argv0);
|
||||||
executable_directory_ = path.parent_path().string();
|
executable_directory = path.parent_path().string();
|
||||||
|
|
||||||
if (executable_directory_.empty()) {
|
if (executable_directory.empty()) {
|
||||||
executable_directory_ = ".";
|
executable_directory = ".";
|
||||||
}
|
}
|
||||||
|
|
||||||
std::cout << "[PathUtils] Executable: " << executable_path_ << "\n";
|
std::cout << "[PathUtils] Executable: " << executable_path << "\n";
|
||||||
std::cout << "[PathUtils] Directori: " << executable_directory_ << "\n";
|
std::cout << "[PathUtils] Directori: " << executable_directory << "\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
// Obtenir el directori de l'executable
|
// Obtenir el directori de l'executable
|
||||||
auto getExecutableDirectory() -> std::string {
|
auto getExecutableDirectory() -> std::string {
|
||||||
if (executable_directory_.empty()) {
|
if (executable_directory.empty()) {
|
||||||
std::cerr << "[PathUtils] ADVERTÈNCIA: Sistema de rutes no inicialitzat\n";
|
std::cerr << "[PathUtils] ADVERTÈNCIA: Sistema de rutes no inicialitzat\n";
|
||||||
return ".";
|
return ".";
|
||||||
}
|
}
|
||||||
return executable_directory_;
|
return executable_directory;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Detectar si estem dins un bundle de macOS
|
// Detectar si estem dins un bundle de macOS
|
||||||
|
|||||||
@@ -26,12 +26,12 @@ constexpr int VELOCITAT_MAX = static_cast<int>(Defaults::Physics::BULLET_SPEED);
|
|||||||
constexpr float PI = Defaults::Math::PI;
|
constexpr float PI = Defaults::Math::PI;
|
||||||
|
|
||||||
// Helpers per comprovar límits de zona
|
// Helpers per comprovar límits de zona
|
||||||
inline auto dins_zona_joc(float x, float y) -> bool {
|
inline auto isInPlayArea(float x, float y) -> bool {
|
||||||
const SDL_FPoint POINT = {x, y};
|
const SDL_FPoint POINT = {x, y};
|
||||||
return SDL_PointInRectFloat(&POINT, &Defaults::Zones::PLAYAREA);
|
return SDL_PointInRectFloat(&POINT, &Defaults::Zones::PLAYAREA);
|
||||||
}
|
}
|
||||||
|
|
||||||
inline void obtenir_limits_zona(float& min_x, float& max_x, float& min_y, float& max_y) {
|
inline void getPlayAreaBounds(float& min_x, float& max_x, float& min_y, float& max_y) {
|
||||||
const auto& zona = Defaults::Zones::PLAYAREA;
|
const auto& zona = Defaults::Zones::PLAYAREA;
|
||||||
min_x = zona.x;
|
min_x = zona.x;
|
||||||
max_x = zona.x + zona.w;
|
max_x = zona.x + zona.w;
|
||||||
@@ -40,7 +40,7 @@ inline void obtenir_limits_zona(float& min_x, float& max_x, float& min_y, float&
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Obtenir límits segurs (compensant radi de l'entidad)
|
// Obtenir límits segurs (compensant radi de l'entidad)
|
||||||
inline void obtenir_limits_zona_segurs(float radi, float& min_x, float& max_x, float& min_y, float& max_y) {
|
inline void getSafePlayAreaBounds(float radi, float& min_x, float& max_x, float& min_y, float& max_y) {
|
||||||
const auto& zona = Defaults::Zones::PLAYAREA;
|
const auto& zona = Defaults::Zones::PLAYAREA;
|
||||||
constexpr float MARGE_SEGURETAT = 10.0F; // Safety margin
|
constexpr float MARGE_SEGURETAT = 10.0F; // Safety margin
|
||||||
|
|
||||||
@@ -51,7 +51,7 @@ inline void obtenir_limits_zona_segurs(float radi, float& min_x, float& max_x, f
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Obtenir centro de l'àrea de juego
|
// Obtenir centro de l'àrea de juego
|
||||||
inline void obtenir_centre_zona(float& centre_x, float& centre_y) {
|
inline void getPlayAreaCenter(float& centre_x, float& centre_y) {
|
||||||
const auto& zona = Defaults::Zones::PLAYAREA;
|
const auto& zona = Defaults::Zones::PLAYAREA;
|
||||||
centre_x = zona.x + (zona.w / 2.0F);
|
centre_x = zona.x + (zona.w / 2.0F);
|
||||||
centre_y = zona.y + (zona.h / 2.0F);
|
centre_y = zona.y + (zona.h / 2.0F);
|
||||||
|
|||||||
@@ -62,122 +62,142 @@ void DebrisManager::explode(const std::shared_ptr<Graphics::Shape>& shape,
|
|||||||
// Reproducir sonido de explosión
|
// Reproducir sonido de explosión
|
||||||
Audio::get()->playSound(sound, Audio::Group::GAME);
|
Audio::get()->playSound(sound, Audio::Group::GAME);
|
||||||
|
|
||||||
// Obtenir centro de la shape para transformacions
|
|
||||||
const Vec2& shape_centre = shape->getCenter();
|
const Vec2& shape_centre = shape->getCenter();
|
||||||
|
|
||||||
// Iterar sobre todas las primitives de la shape
|
for (const auto& primitive : shape->getPrimitives()) {
|
||||||
for (const auto& primitive : shape->get_primitives()) {
|
for (const auto& [local_p1, local_p2] : extractSegments(primitive)) {
|
||||||
// Processar cada segment de línia
|
// Transformar points locals → coordenades mundials
|
||||||
|
Vec2 world_p1 = transformPoint(local_p1, shape_centre, centro, angle, scale);
|
||||||
|
Vec2 world_p2 = transformPoint(local_p2, shape_centre, centro, angle, scale);
|
||||||
|
|
||||||
|
// Si el pool es ple, no té sentit continuar amb la resta de segments
|
||||||
|
if (!spawnDebris(world_p1, world_p2, centro, velocitat_base, brightness,
|
||||||
|
velocitat_objecte, velocitat_angular,
|
||||||
|
factor_herencia_visual, color)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
auto DebrisManager::extractSegments(const Graphics::ShapePrimitive& primitive)
|
||||||
|
-> std::vector<std::pair<Vec2, Vec2>> {
|
||||||
std::vector<std::pair<Vec2, Vec2>> segments;
|
std::vector<std::pair<Vec2, Vec2>> segments;
|
||||||
|
|
||||||
if (primitive.type == Graphics::PrimitiveType::POLYLINE) {
|
if (primitive.type == Graphics::PrimitiveType::POLYLINE) {
|
||||||
// Polyline: extreure segments consecutius
|
// Polyline: extreure segments consecutius
|
||||||
for (size_t i = 0; i < primitive.points.size() - 1; i++) {
|
for (size_t i = 0; i + 1 < primitive.points.size(); i++) {
|
||||||
segments.emplace_back(primitive.points[i], primitive.points[i + 1]);
|
segments.emplace_back(primitive.points[i], primitive.points[i + 1]);
|
||||||
}
|
}
|
||||||
} else { // PrimitiveType::LINE
|
return segments;
|
||||||
// Line: un únic segment
|
}
|
||||||
|
// PrimitiveType::LINE: un únic segment (si té els 2 punts)
|
||||||
if (primitive.points.size() >= 2) {
|
if (primitive.points.size() >= 2) {
|
||||||
segments.emplace_back(primitive.points[0], primitive.points[1]);
|
segments.emplace_back(primitive.points[0], primitive.points[1]);
|
||||||
}
|
}
|
||||||
|
return segments;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Crear debris para cada segment
|
auto DebrisManager::spawnDebris(const Vec2& world_p1, const Vec2& world_p2,
|
||||||
for (const auto& [local_p1, local_p2] : segments) {
|
const Vec2& centro, float velocitat_base, float brightness,
|
||||||
// 1. Transformar points locals → coordenades mundials
|
const Vec2& velocitat_objecte, float velocitat_angular,
|
||||||
Vec2 world_p1 =
|
float factor_herencia_visual, SDL_Color color) -> bool {
|
||||||
transformPoint(local_p1, shape_centre, centro, angle, scale);
|
|
||||||
Vec2 world_p2 =
|
|
||||||
transformPoint(local_p2, shape_centre, centro, angle, scale);
|
|
||||||
|
|
||||||
// 2. Trobar slot lliure
|
|
||||||
Debris* debris = findFreeSlot();
|
Debris* debris = findFreeSlot();
|
||||||
if (debris == nullptr) {
|
if (debris == nullptr) {
|
||||||
std::cerr << "[DebrisManager] Warning: no debris slots disponibles\n";
|
std::cerr << "[DebrisManager] Warning: no debris slots disponibles\n";
|
||||||
return; // Pool ple
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3. Inicialitzar geometria
|
// Geometria
|
||||||
debris->p1 = world_p1;
|
debris->p1 = world_p1;
|
||||||
debris->p2 = world_p2;
|
debris->p2 = world_p2;
|
||||||
|
|
||||||
// 4. Calcular direcció de explosión (radial, des del centro hacia fuera)
|
// Direcció radial (desde el centro hacia el segment)
|
||||||
Vec2 direccio = computeExplosionDirection(world_p1, world_p2, centro);
|
Vec2 direccio = computeExplosionDirection(world_p1, world_p2, centro);
|
||||||
|
|
||||||
// 5. Velocidad inicial (base ± variació aleatòria + velocity heretada)
|
// Velocidad inicial (base ± variació aleatòria + velocity heretada de l'objecte)
|
||||||
float speed =
|
float speed =
|
||||||
velocitat_base +
|
velocitat_base +
|
||||||
(((std::rand() / static_cast<float>(RAND_MAX)) * 2.0F - 1.0F) *
|
(((std::rand() / static_cast<float>(RAND_MAX)) * 2.0F - 1.0F) *
|
||||||
Defaults::Physics::Debris::VARIACIO_VELOCITAT);
|
Defaults::Physics::Debris::VARIACIO_VELOCITAT);
|
||||||
|
|
||||||
// Heredar velocity de l'objecte original (suma vectorial)
|
|
||||||
debris->velocity.x = (direccio.x * speed) + velocitat_objecte.x;
|
debris->velocity.x = (direccio.x * speed) + velocitat_objecte.x;
|
||||||
debris->velocity.y = (direccio.y * speed) + velocitat_objecte.y;
|
debris->velocity.y = (direccio.y * speed) + velocitat_objecte.y;
|
||||||
debris->acceleration = Defaults::Physics::Debris::ACCELERACIO;
|
debris->acceleration = Defaults::Physics::Debris::ACCELERACIO;
|
||||||
|
|
||||||
// 6. Herència de velocity angular con sin + conversió de excés
|
// Rotación de trayectoria (con conversió a tangencial si excedeix cap)
|
||||||
|
applyAngularVelocity(*debris, direccio, velocitat_angular);
|
||||||
|
|
||||||
// 6a. Rotación de TRAYECTORIA con sin + conversió tangencial
|
// Rotación visual (proporcional o aleatòria)
|
||||||
if (std::abs(velocitat_angular) > 0.01F) {
|
applyVisualRotation(*debris, velocitat_angular, factor_herencia_visual);
|
||||||
// FASE 1: Aplicar herència i variació (igual que antes)
|
|
||||||
|
debris->angle_rotacio = 0.0F;
|
||||||
|
|
||||||
|
// Vida i shrinking
|
||||||
|
debris->temps_vida = 0.0F;
|
||||||
|
debris->temps_max = Defaults::Physics::Debris::TEMPS_VIDA;
|
||||||
|
debris->factor_shrink = Defaults::Physics::Debris::SHRINK_RATE;
|
||||||
|
|
||||||
|
// Visuals heretades
|
||||||
|
debris->brightness = brightness;
|
||||||
|
debris->color = color;
|
||||||
|
|
||||||
|
debris->active = true;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void DebrisManager::applyAngularVelocity(Debris& debris, const Vec2& direccio,
|
||||||
|
float velocitat_angular) {
|
||||||
|
if (std::abs(velocitat_angular) <= 0.01F) {
|
||||||
|
debris.velocitat_rot = 0.0F; // Nave: sin curvas
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// FASE 1: Aplicar herència i variació
|
||||||
float factor_herencia =
|
float factor_herencia =
|
||||||
Defaults::Physics::Debris::FACTOR_HERENCIA_MIN +
|
Defaults::Physics::Debris::FACTOR_HERENCIA_MIN +
|
||||||
((std::rand() / static_cast<float>(RAND_MAX)) *
|
((std::rand() / static_cast<float>(RAND_MAX)) *
|
||||||
(Defaults::Physics::Debris::FACTOR_HERENCIA_MAX -
|
(Defaults::Physics::Debris::FACTOR_HERENCIA_MAX -
|
||||||
Defaults::Physics::Debris::FACTOR_HERENCIA_MIN));
|
Defaults::Physics::Debris::FACTOR_HERENCIA_MIN));
|
||||||
|
|
||||||
float velocitat_ang_heretada = velocitat_angular * factor_herencia;
|
float velocitat_ang_heretada = velocitat_angular * factor_herencia;
|
||||||
|
float variacio = ((std::rand() / static_cast<float>(RAND_MAX)) * 0.2F) - 0.1F;
|
||||||
float variacio =
|
|
||||||
((std::rand() / static_cast<float>(RAND_MAX)) * 0.2F) - 0.1F;
|
|
||||||
velocitat_ang_heretada *= (1.0F + variacio);
|
velocitat_ang_heretada *= (1.0F + variacio);
|
||||||
|
|
||||||
// FASE 2: Aplicar sin i calcular excés
|
// FASE 2: Cap a la velocity màxima; l'excés es converteix en tangencial
|
||||||
constexpr float CAP = Defaults::Physics::Debris::VELOCITAT_ROT_MAX;
|
constexpr float CAP = Defaults::Physics::Debris::VELOCITAT_ROT_MAX;
|
||||||
float abs_ang = std::abs(velocitat_ang_heretada);
|
float abs_ang = std::abs(velocitat_ang_heretada);
|
||||||
float sign_ang = (velocitat_ang_heretada >= 0.0F) ? 1.0F : -1.0F;
|
float sign_ang = (velocitat_ang_heretada >= 0.0F) ? 1.0F : -1.0F;
|
||||||
|
|
||||||
if (abs_ang > CAP) {
|
if (abs_ang <= CAP) {
|
||||||
// Excés: convertir a velocity tangencial
|
debris.velocitat_rot = velocitat_ang_heretada;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Excés: converteix l'excés de velocitat angular en velocitat tangencial lineal
|
||||||
float excess = abs_ang - CAP;
|
float excess = abs_ang - CAP;
|
||||||
|
constexpr float RADIUS = 20.0F; // Radi típic de la shape (enemigos = 20 px)
|
||||||
|
float v_tangential = excess * RADIUS;
|
||||||
|
|
||||||
// Radi de la shape (enemigos = 20 px)
|
// Direcció tangencial: perpendicular a la radial (90° CCW): tangent = (-dy, dx)
|
||||||
float radius = 20.0F;
|
debris.velocity.x += -direccio.y * v_tangential;
|
||||||
|
debris.velocity.y += direccio.x * v_tangential;
|
||||||
|
|
||||||
// Velocidad tangencial = ω_excés × radi
|
// Velocitat angular limitada al cap (preservant el signe)
|
||||||
float v_tangential = excess * radius;
|
debris.velocitat_rot = sign_ang * CAP;
|
||||||
|
|
||||||
// Direcció tangencial: perpendicular a la radial (90° CCW)
|
|
||||||
// Si direccio = (dx, dy), tangent = (-dy, dx)
|
|
||||||
float tangent_x = -direccio.y;
|
|
||||||
float tangent_y = direccio.x;
|
|
||||||
|
|
||||||
// Añadir velocity tangencial (suma vectorial)
|
|
||||||
debris->velocity.x += tangent_x * v_tangential;
|
|
||||||
debris->velocity.y += tangent_y * v_tangential;
|
|
||||||
|
|
||||||
// Aplicar hacia velocity angular (preservar signe)
|
|
||||||
debris->velocitat_rot = sign_ang * CAP;
|
|
||||||
} else {
|
|
||||||
// Per sota del sin: comportament normal
|
|
||||||
debris->velocitat_rot = velocitat_ang_heretada;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
debris->velocitat_rot = 0.0F; // Nave: sin curvas
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// 6b. Rotación VISUAL (proporcional según factor_herencia_visual)
|
void DebrisManager::applyVisualRotation(Debris& debris, float velocitat_angular,
|
||||||
|
float factor_herencia_visual) {
|
||||||
if (factor_herencia_visual > 0.01F && std::abs(velocitat_angular) > 0.01F) {
|
if (factor_herencia_visual > 0.01F && std::abs(velocitat_angular) > 0.01F) {
|
||||||
// Heredar rotación visual con factor proporcional
|
// Heredar rotación visual con factor proporcional + ±5% de variació
|
||||||
debris->velocitat_rot_visual = debris->velocitat_rot * factor_herencia_visual;
|
debris.velocitat_rot_visual = debris.velocitat_rot * factor_herencia_visual;
|
||||||
|
|
||||||
// Variació aleatòria pequeña (±5%) per naturalitat
|
|
||||||
float variacio_visual =
|
float variacio_visual =
|
||||||
((std::rand() / static_cast<float>(RAND_MAX)) * 0.1F) - 0.05F;
|
((std::rand() / static_cast<float>(RAND_MAX)) * 0.1F) - 0.05F;
|
||||||
debris->velocitat_rot_visual *= (1.0F + variacio_visual);
|
debris.velocitat_rot_visual *= (1.0F + variacio_visual);
|
||||||
} else {
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Rotación visual aleatòria (factor = 0.0 o sin velocidad angular)
|
// Rotación visual aleatòria (factor = 0.0 o sin velocidad angular)
|
||||||
debris->velocitat_rot_visual =
|
debris.velocitat_rot_visual =
|
||||||
Defaults::Physics::Debris::ROTACIO_MIN +
|
Defaults::Physics::Debris::ROTACIO_MIN +
|
||||||
((std::rand() / static_cast<float>(RAND_MAX)) *
|
((std::rand() / static_cast<float>(RAND_MAX)) *
|
||||||
(Defaults::Physics::Debris::ROTACIO_MAX -
|
(Defaults::Physics::Debris::ROTACIO_MAX -
|
||||||
@@ -185,24 +205,7 @@ void DebrisManager::explode(const std::shared_ptr<Graphics::Shape>& shape,
|
|||||||
|
|
||||||
// 50% probabilitat de rotación en sentit contrari
|
// 50% probabilitat de rotación en sentit contrari
|
||||||
if (std::rand() % 2 == 0) {
|
if (std::rand() % 2 == 0) {
|
||||||
debris->velocitat_rot_visual = -debris->velocitat_rot_visual;
|
debris.velocitat_rot_visual = -debris.velocitat_rot_visual;
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
debris->angle_rotacio = 0.0F;
|
|
||||||
|
|
||||||
// 7. Configurar vida i shrinking
|
|
||||||
debris->temps_vida = 0.0F;
|
|
||||||
debris->temps_max = Defaults::Physics::Debris::TEMPS_VIDA;
|
|
||||||
debris->factor_shrink = Defaults::Physics::Debris::SHRINK_RATE;
|
|
||||||
|
|
||||||
// 8. Heredar brightness y color del padre
|
|
||||||
debris->brightness = brightness;
|
|
||||||
debris->color = color;
|
|
||||||
|
|
||||||
// 9. Activar
|
|
||||||
debris->active = true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -8,6 +8,8 @@
|
|||||||
|
|
||||||
#include <array>
|
#include <array>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
#include <utility>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
#include "core/defaults.hpp"
|
#include "core/defaults.hpp"
|
||||||
#include "core/graphics/shape.hpp"
|
#include "core/graphics/shape.hpp"
|
||||||
@@ -72,6 +74,21 @@ class DebrisManager {
|
|||||||
// Calcular direcció de explosión (radial, des del centro hacia el segment).
|
// Calcular direcció de explosión (radial, des del centro hacia el segment).
|
||||||
// Estático: solo opera sobre los puntos pasados, sin estado del manager.
|
// Estático: solo opera sobre los puntos pasados, sin estado del manager.
|
||||||
[[nodiscard]] static auto computeExplosionDirection(const Vec2& p1, const Vec2& p2, const Vec2& centre_objecte) -> Vec2;
|
[[nodiscard]] static auto computeExplosionDirection(const Vec2& p1, const Vec2& p2, const Vec2& centre_objecte) -> Vec2;
|
||||||
|
|
||||||
|
// Sub-pasos de explode() (descomposició per reduir complexitat cognitiva).
|
||||||
|
// extractSegments y los apply* son static (solo toquen el debris pasado).
|
||||||
|
[[nodiscard]] static auto extractSegments(const Graphics::ShapePrimitive& primitive)
|
||||||
|
-> std::vector<std::pair<Vec2, Vec2>>;
|
||||||
|
// Inicialitza un debris en un slot lliure i el deixa actiu. Retorna
|
||||||
|
// false si el pool está ple (la cridadora ha d'aturar el bucle).
|
||||||
|
auto spawnDebris(const Vec2& world_p1, const Vec2& world_p2,
|
||||||
|
const Vec2& centro, float velocitat_base, float brightness,
|
||||||
|
const Vec2& velocitat_objecte, float velocitat_angular,
|
||||||
|
float factor_herencia_visual, SDL_Color color) -> bool;
|
||||||
|
static void applyAngularVelocity(Debris& debris, const Vec2& direccio,
|
||||||
|
float velocitat_angular);
|
||||||
|
static void applyVisualRotation(Debris& debris, float velocitat_angular,
|
||||||
|
float factor_herencia_visual);
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace Effects
|
} // namespace Effects
|
||||||
|
|||||||
@@ -48,7 +48,7 @@ Bullet::Bullet(Rendering::Renderer* renderer)
|
|||||||
|
|
||||||
void Bullet::init() {
|
void Bullet::init() {
|
||||||
// Inicialment inactiva
|
// Inicialment inactiva
|
||||||
esta_ = false;
|
is_active_ = false;
|
||||||
center_ = {.x = 0.0F, .y = 0.0F};
|
center_ = {.x = 0.0F, .y = 0.0F};
|
||||||
angle_ = 0.0F;
|
angle_ = 0.0F;
|
||||||
grace_timer_ = 0.0F;
|
grace_timer_ = 0.0F;
|
||||||
@@ -63,7 +63,7 @@ void Bullet::init() {
|
|||||||
|
|
||||||
void Bullet::disparar(const Vec2& position, float angle, uint8_t owner_id) {
|
void Bullet::disparar(const Vec2& position, float angle, uint8_t owner_id) {
|
||||||
// Activar bullet
|
// Activar bullet
|
||||||
esta_ = true;
|
is_active_ = true;
|
||||||
|
|
||||||
// Almacenar propietario (0=P1, 1=P2)
|
// Almacenar propietario (0=P1, 1=P2)
|
||||||
owner_id_ = owner_id;
|
owner_id_ = owner_id;
|
||||||
@@ -90,7 +90,7 @@ void Bullet::disparar(const Vec2& position, float angle, uint8_t owner_id) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void Bullet::update(float delta_time) {
|
void Bullet::update(float delta_time) {
|
||||||
if (!esta_) {
|
if (!is_active_) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -106,7 +106,7 @@ void Bullet::update(float delta_time) {
|
|||||||
float max_x;
|
float max_x;
|
||||||
float min_y;
|
float min_y;
|
||||||
float max_y;
|
float max_y;
|
||||||
Constants::obtenir_limits_zona_segurs(Defaults::Entities::BULLET_RADIUS,
|
Constants::getSafePlayAreaBounds(Defaults::Entities::BULLET_RADIUS,
|
||||||
min_x,
|
min_x,
|
||||||
max_x,
|
max_x,
|
||||||
min_y,
|
min_y,
|
||||||
@@ -125,16 +125,16 @@ void Bullet::postUpdate(float /*delta_time*/) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void Bullet::desactivar() {
|
void Bullet::desactivar() {
|
||||||
esta_ = false;
|
is_active_ = false;
|
||||||
// Detener el cuerpo físico para que no acumule deriva mientras inactiva.
|
// Detener el cuerpo físico para que no acumule deriva mientras inactiva.
|
||||||
body_.velocity = Vec2{};
|
body_.velocity = Vec2{};
|
||||||
body_.angular_velocity = 0.0F;
|
body_.angular_velocity = 0.0F;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Bullet::draw() const {
|
void Bullet::draw() const {
|
||||||
if (esta_ && shape_) {
|
if (is_active_ && shape_) {
|
||||||
// Les bales roten segons l'angle de trayectòria (estático tras disparo)
|
// Les bales roten segons l'angle de trayectòria (estático tras disparo)
|
||||||
Rendering::render_shape(renderer_, shape_, center_, angle_, 1.0F, 1.0F, brightness_,
|
Rendering::renderShape(renderer_, shape_, center_, angle_, 1.0F, 1.0F, brightness_,
|
||||||
/*rotation_3d=*/nullptr, Defaults::Palette::BULLET);
|
/*rotation_3d=*/nullptr, Defaults::Palette::BULLET);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,18 +23,17 @@ class Bullet : public Entities::Entity {
|
|||||||
void draw() const override;
|
void draw() const override;
|
||||||
|
|
||||||
// Override: Interfaz de Entity
|
// Override: Interfaz de Entity
|
||||||
[[nodiscard]] auto isActive() const -> bool override { return esta_; }
|
[[nodiscard]] auto isActive() const -> bool override { return is_active_; }
|
||||||
|
|
||||||
// Override: Interfaz de colisión (gameplay-level: PLAYAREA bounds-check)
|
// Override: Interfaz de colisión (gameplay-level: PLAYAREA bounds-check)
|
||||||
[[nodiscard]] auto getCollisionRadius() const -> float override {
|
[[nodiscard]] auto getCollisionRadius() const -> float override {
|
||||||
return Defaults::Entities::BULLET_RADIUS;
|
return Defaults::Entities::BULLET_RADIUS;
|
||||||
}
|
}
|
||||||
[[nodiscard]] auto isCollidable() const -> bool override {
|
[[nodiscard]] auto isCollidable() const -> bool override {
|
||||||
return esta_ && grace_timer_ <= 0.0F;
|
return is_active_ && grace_timer_ <= 0.0F;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Getters (API pública sin cambios)
|
// Getters (API pública sin cambios)
|
||||||
[[nodiscard]] auto esta_activa() const -> bool { return esta_; }
|
|
||||||
[[nodiscard]] auto getOwnerId() const -> uint8_t { return owner_id_; }
|
[[nodiscard]] auto getOwnerId() const -> uint8_t { return owner_id_; }
|
||||||
[[nodiscard]] auto getGraceTimer() const -> float { return grace_timer_; }
|
[[nodiscard]] auto getGraceTimer() const -> float { return grace_timer_; }
|
||||||
void desactivar();
|
void desactivar();
|
||||||
@@ -43,7 +42,7 @@ class Bullet : public Entities::Entity {
|
|||||||
// Miembros específicos de Bullet (heredados: renderer_, shape_, center_, angle_, brightness_, body_).
|
// Miembros específicos de Bullet (heredados: renderer_, shape_, center_, angle_, brightness_, body_).
|
||||||
// Inicializados en la declaración para que tanto el ctor por defecto como el que toma renderer
|
// Inicializados en la declaración para que tanto el ctor por defecto como el que toma renderer
|
||||||
// dejen el objeto en estado coherente (proyectil inactivo, sin owner, sin grace timer).
|
// dejen el objeto en estado coherente (proyectil inactivo, sin owner, sin grace timer).
|
||||||
bool esta_{false};
|
bool is_active_{false};
|
||||||
uint8_t owner_id_{0}; // 0=P1, 1=P2
|
uint8_t owner_id_{0}; // 0=P1, 1=P2
|
||||||
float grace_timer_{0.0F}; // Grace period timer (0.0 = vulnerable)
|
float grace_timer_{0.0F}; // Grace period timer (0.0 = vulnerable)
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -113,7 +113,7 @@ void Enemy::init(EnemyType type, const Vec2* ship_pos) {
|
|||||||
float max_x;
|
float max_x;
|
||||||
float min_y;
|
float min_y;
|
||||||
float max_y;
|
float max_y;
|
||||||
Constants::obtenir_limits_zona_segurs(Defaults::Entities::ENEMY_RADIUS, min_x, max_x, min_y, max_y);
|
Constants::getSafePlayAreaBounds(Defaults::Entities::ENEMY_RADIUS, min_x, max_x, min_y, max_y);
|
||||||
|
|
||||||
if (ship_pos != nullptr) {
|
if (ship_pos != nullptr) {
|
||||||
bool found_safe_position = false;
|
bool found_safe_position = false;
|
||||||
@@ -229,7 +229,7 @@ void Enemy::draw() const {
|
|||||||
case EnemyType::QUADRAT: color = Defaults::Palette::QUADRAT; break;
|
case EnemyType::QUADRAT: color = Defaults::Palette::QUADRAT; break;
|
||||||
case EnemyType::MOLINILLO: color = Defaults::Palette::MOLINILLO; break;
|
case EnemyType::MOLINILLO: color = Defaults::Palette::MOLINILLO; break;
|
||||||
}
|
}
|
||||||
Rendering::render_shape(renderer_, shape_, center_, rotacio_, SCALE, 1.0F, brightness_,
|
Rendering::renderShape(renderer_, shape_, center_, rotacio_, SCALE, 1.0F, brightness_,
|
||||||
/*rotation_3d=*/nullptr, color);
|
/*rotation_3d=*/nullptr, color);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -436,7 +436,7 @@ auto Enemy::attemptSafeSpawn(const Vec2& ship_pos, float& out_x, float& out_y) -
|
|||||||
float max_x;
|
float max_x;
|
||||||
float min_y;
|
float min_y;
|
||||||
float max_y;
|
float max_y;
|
||||||
Constants::obtenir_limits_zona_segurs(Defaults::Entities::ENEMY_RADIUS, min_x, max_x, min_y, max_y);
|
Constants::getSafePlayAreaBounds(Defaults::Entities::ENEMY_RADIUS, min_x, max_x, min_y, max_y);
|
||||||
|
|
||||||
const int RANGE_X = static_cast<int>(max_x - min_x);
|
const int RANGE_X = static_cast<int>(max_x - min_x);
|
||||||
const int RANGE_Y = static_cast<int>(max_y - min_y);
|
const int RANGE_Y = static_cast<int>(max_y - min_y);
|
||||||
|
|||||||
@@ -46,7 +46,7 @@ void Ship::init(const Vec2* spawn_point, bool activar_invulnerabilitat) {
|
|||||||
} else {
|
} else {
|
||||||
float centre_x;
|
float centre_x;
|
||||||
float centre_y;
|
float centre_y;
|
||||||
Constants::obtenir_centre_zona(centre_x, centre_y);
|
Constants::getPlayAreaCenter(centre_x, centre_y);
|
||||||
center_ = {.x = centre_x, .y = centre_y};
|
center_ = {.x = centre_x, .y = centre_y};
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -158,6 +158,6 @@ void Ship::draw() const {
|
|||||||
const float VISUAL_PUSH = SPEED / 33.33F;
|
const float VISUAL_PUSH = SPEED / 33.33F;
|
||||||
const float SCALE = 1.0F + (VISUAL_PUSH / 12.0F);
|
const float SCALE = 1.0F + (VISUAL_PUSH / 12.0F);
|
||||||
|
|
||||||
Rendering::render_shape(renderer_, shape_, center_, angle_, SCALE, 1.0F, brightness_,
|
Rendering::renderShape(renderer_, shape_, center_, angle_, SCALE, 1.0F, brightness_,
|
||||||
/*rotation_3d=*/nullptr, Defaults::Palette::SHIP);
|
/*rotation_3d=*/nullptr, Defaults::Palette::SHIP);
|
||||||
}
|
}
|
||||||
|
|||||||
+72
-150
@@ -224,119 +224,71 @@ void setConfigFile(const std::string& path) { config_file_path = path; }
|
|||||||
|
|
||||||
// Funciones auxiliars per load seccions del YAML
|
// Funciones auxiliars per load seccions del YAML
|
||||||
|
|
||||||
|
// Lee un campo escalar del YAML aplicando un validador; si la clau no
|
||||||
|
// existe, deja `dest` intacto; si la conversió o la validació fallen,
|
||||||
|
// asigna `fallback`. Estàtic per quedar dins de la unitat de traducció.
|
||||||
|
template <typename T, typename Validator>
|
||||||
|
static void readField(const fkyaml::node& parent, const char* key, T& dest,
|
||||||
|
T fallback, Validator&& validate) {
|
||||||
|
if (!parent.contains(key)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
auto val = parent[key].template get_value<T>();
|
||||||
|
dest = validate(val) ? val : fallback;
|
||||||
|
} catch (...) {
|
||||||
|
dest = fallback;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Variant sin validador: només lectura amb fallback en cas d'error.
|
||||||
|
template <typename T>
|
||||||
|
static void readField(const fkyaml::node& parent, const char* key, T& dest, T fallback) {
|
||||||
|
if (!parent.contains(key)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
dest = parent[key].template get_value<T>();
|
||||||
|
} catch (...) {
|
||||||
|
dest = fallback;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void loadWindowConfigFromYaml(const fkyaml::node& yaml) {
|
static void loadWindowConfigFromYaml(const fkyaml::node& yaml) {
|
||||||
if (yaml.contains("window")) {
|
if (!yaml.contains("window")) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
const auto& win = yaml["window"];
|
const auto& win = yaml["window"];
|
||||||
|
|
||||||
if (win.contains("width")) {
|
readField(win, "width", window.width, Defaults::Window::WIDTH,
|
||||||
try {
|
[](int v) { return v >= Defaults::Window::MIN_WIDTH; });
|
||||||
auto val = win["width"].get_value<int>();
|
readField(win, "height", window.height, Defaults::Window::HEIGHT,
|
||||||
window.width = (val >= Defaults::Window::MIN_WIDTH)
|
[](int v) { return v >= Defaults::Window::MIN_HEIGHT; });
|
||||||
? val
|
readField(win, "fullscreen", window.fullscreen, false);
|
||||||
: Defaults::Window::WIDTH;
|
|
||||||
} catch (...) {
|
|
||||||
window.width = Defaults::Window::WIDTH;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (win.contains("height")) {
|
|
||||||
try {
|
|
||||||
auto val = win["height"].get_value<int>();
|
|
||||||
window.height = (val >= Defaults::Window::MIN_HEIGHT)
|
|
||||||
? val
|
|
||||||
: Defaults::Window::HEIGHT;
|
|
||||||
} catch (...) {
|
|
||||||
window.height = Defaults::Window::HEIGHT;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (win.contains("fullscreen")) {
|
|
||||||
try {
|
|
||||||
window.fullscreen = win["fullscreen"].get_value<bool>();
|
|
||||||
} catch (...) {
|
|
||||||
window.fullscreen = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (win.contains("zoom_factor")) {
|
if (win.contains("zoom_factor")) {
|
||||||
try {
|
readField(win, "zoom_factor", window.zoom_factor, Defaults::Window::BASE_ZOOM,
|
||||||
auto val = win["zoom_factor"].get_value<float>();
|
[](float v) { return v >= Defaults::Window::MIN_ZOOM && v <= 10.0F; });
|
||||||
window.zoom_factor = (val >= Defaults::Window::MIN_ZOOM && val <= 10.0F)
|
|
||||||
? val
|
|
||||||
: Defaults::Window::BASE_ZOOM;
|
|
||||||
} catch (...) {
|
|
||||||
window.zoom_factor = Defaults::Window::BASE_ZOOM;
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
// Legacy config: infer zoom from width
|
// Legacy config: infer zoom from width
|
||||||
window.zoom_factor = static_cast<float>(window.width) / Defaults::Window::WIDTH;
|
window.zoom_factor = static_cast<float>(window.width) / Defaults::Window::WIDTH;
|
||||||
window.zoom_factor = std::max(Defaults::Window::MIN_ZOOM, window.zoom_factor);
|
window.zoom_factor = std::max(Defaults::Window::MIN_ZOOM, window.zoom_factor);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
static void loadPhysicsConfigFromYaml(const fkyaml::node& yaml) {
|
static void loadPhysicsConfigFromYaml(const fkyaml::node& yaml) {
|
||||||
if (yaml.contains("physics")) {
|
if (!yaml.contains("physics")) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
const auto& phys = yaml["physics"];
|
const auto& phys = yaml["physics"];
|
||||||
|
constexpr auto POSITIVE = [](float v) { return v > 0.0F; };
|
||||||
|
|
||||||
if (phys.contains("rotation_speed")) {
|
readField(phys, "rotation_speed", physics.rotation_speed, Defaults::Physics::ROTATION_SPEED, POSITIVE);
|
||||||
try {
|
readField(phys, "acceleration", physics.acceleration, Defaults::Physics::ACCELERATION, POSITIVE);
|
||||||
auto val = phys["rotation_speed"].get_value<float>();
|
readField(phys, "max_velocity", physics.max_velocity, Defaults::Physics::MAX_VELOCITY, POSITIVE);
|
||||||
physics.rotation_speed =
|
readField(phys, "friction", physics.friction, Defaults::Physics::FRICTION, POSITIVE);
|
||||||
(val > 0) ? val : Defaults::Physics::ROTATION_SPEED;
|
readField(phys, "enemy_speed", physics.enemy_speed, Defaults::Physics::ENEMY_SPEED, POSITIVE);
|
||||||
} catch (...) {
|
readField(phys, "bullet_speed", physics.bullet_speed, Defaults::Physics::BULLET_SPEED, POSITIVE);
|
||||||
physics.rotation_speed = Defaults::Physics::ROTATION_SPEED;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (phys.contains("acceleration")) {
|
|
||||||
try {
|
|
||||||
auto val = phys["acceleration"].get_value<float>();
|
|
||||||
physics.acceleration =
|
|
||||||
(val > 0) ? val : Defaults::Physics::ACCELERATION;
|
|
||||||
} catch (...) {
|
|
||||||
physics.acceleration = Defaults::Physics::ACCELERATION;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (phys.contains("max_velocity")) {
|
|
||||||
try {
|
|
||||||
auto val = phys["max_velocity"].get_value<float>();
|
|
||||||
physics.max_velocity =
|
|
||||||
(val > 0) ? val : Defaults::Physics::MAX_VELOCITY;
|
|
||||||
} catch (...) {
|
|
||||||
physics.max_velocity = Defaults::Physics::MAX_VELOCITY;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (phys.contains("friction")) {
|
|
||||||
try {
|
|
||||||
auto val = phys["friction"].get_value<float>();
|
|
||||||
physics.friction = (val > 0) ? val : Defaults::Physics::FRICTION;
|
|
||||||
} catch (...) {
|
|
||||||
physics.friction = Defaults::Physics::FRICTION;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (phys.contains("enemy_speed")) {
|
|
||||||
try {
|
|
||||||
auto val = phys["enemy_speed"].get_value<float>();
|
|
||||||
physics.enemy_speed = (val > 0) ? val : Defaults::Physics::ENEMY_SPEED;
|
|
||||||
} catch (...) {
|
|
||||||
physics.enemy_speed = Defaults::Physics::ENEMY_SPEED;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (phys.contains("bullet_speed")) {
|
|
||||||
try {
|
|
||||||
auto val = phys["bullet_speed"].get_value<float>();
|
|
||||||
physics.bullet_speed =
|
|
||||||
(val > 0) ? val : Defaults::Physics::BULLET_SPEED;
|
|
||||||
} catch (...) {
|
|
||||||
physics.bullet_speed = Defaults::Physics::BULLET_SPEED;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void loadGameplayConfigFromYaml(const fkyaml::node& yaml) {
|
static void loadGameplayConfigFromYaml(const fkyaml::node& yaml) {
|
||||||
@@ -381,69 +333,39 @@ static void loadRenderingConfigFromYaml(const fkyaml::node& yaml) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void loadAudioConfigFromYaml(const fkyaml::node& yaml) {
|
static void loadAudioMusicSection(const fkyaml::node& aud) {
|
||||||
if (yaml.contains("audio")) {
|
if (!aud.contains("music")) {
|
||||||
const auto& aud = yaml["audio"];
|
return;
|
||||||
|
|
||||||
if (aud.contains("enabled")) {
|
|
||||||
try {
|
|
||||||
audio.enabled = aud["enabled"].get_value<bool>();
|
|
||||||
} catch (...) {
|
|
||||||
audio.enabled = Defaults::Audio::ENABLED;
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (aud.contains("volume")) {
|
|
||||||
try {
|
|
||||||
auto val = aud["volume"].get_value<float>();
|
|
||||||
audio.volume = (val >= 0.0F && val <= 1.0F) ? val : Defaults::Audio::VOLUME;
|
|
||||||
} catch (...) {
|
|
||||||
audio.volume = Defaults::Audio::VOLUME;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (aud.contains("music")) {
|
|
||||||
const auto& mus = aud["music"];
|
const auto& mus = aud["music"];
|
||||||
|
constexpr auto UNIT_RANGE = [](float v) { return v >= 0.0F && v <= 1.0F; };
|
||||||
|
|
||||||
if (mus.contains("enabled")) {
|
readField(mus, "enabled", audio.music.enabled, Defaults::Audio::MUSIC_ENABLED);
|
||||||
try {
|
readField(mus, "volume", audio.music.volume, Defaults::Audio::MUSIC_VOLUME, UNIT_RANGE);
|
||||||
audio.music.enabled = mus["enabled"].get_value<bool>();
|
|
||||||
} catch (...) {
|
|
||||||
audio.music.enabled = Defaults::Audio::MUSIC_ENABLED;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mus.contains("volume")) {
|
static void loadAudioSoundSection(const fkyaml::node& aud) {
|
||||||
try {
|
if (!aud.contains("sound")) {
|
||||||
auto val = mus["volume"].get_value<float>();
|
return;
|
||||||
audio.music.volume = (val >= 0.0F && val <= 1.0F) ? val : Defaults::Audio::MUSIC_VOLUME;
|
|
||||||
} catch (...) {
|
|
||||||
audio.music.volume = Defaults::Audio::MUSIC_VOLUME;
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (aud.contains("sound")) {
|
|
||||||
const auto& snd = aud["sound"];
|
const auto& snd = aud["sound"];
|
||||||
|
constexpr auto UNIT_RANGE = [](float v) { return v >= 0.0F && v <= 1.0F; };
|
||||||
|
|
||||||
if (snd.contains("enabled")) {
|
readField(snd, "enabled", audio.sound.enabled, Defaults::Audio::SOUND_ENABLED);
|
||||||
try {
|
readField(snd, "volume", audio.sound.volume, Defaults::Audio::SOUND_VOLUME, UNIT_RANGE);
|
||||||
audio.sound.enabled = snd["enabled"].get_value<bool>();
|
|
||||||
} catch (...) {
|
|
||||||
audio.sound.enabled = Defaults::Audio::SOUND_ENABLED;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (snd.contains("volume")) {
|
static void loadAudioConfigFromYaml(const fkyaml::node& yaml) {
|
||||||
try {
|
if (!yaml.contains("audio")) {
|
||||||
auto val = snd["volume"].get_value<float>();
|
return;
|
||||||
audio.sound.volume = (val >= 0.0F && val <= 1.0F) ? val : Defaults::Audio::SOUND_VOLUME;
|
|
||||||
} catch (...) {
|
|
||||||
audio.sound.volume = Defaults::Audio::SOUND_VOLUME;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
const auto& aud = yaml["audio"];
|
||||||
|
constexpr auto UNIT_RANGE = [](float v) { return v >= 0.0F && v <= 1.0F; };
|
||||||
|
|
||||||
|
readField(aud, "enabled", audio.enabled, Defaults::Audio::ENABLED);
|
||||||
|
readField(aud, "volume", audio.volume, Defaults::Audio::VOLUME, UNIT_RANGE);
|
||||||
|
loadAudioMusicSection(aud);
|
||||||
|
loadAudioSoundSection(aud);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Carregar controls del player 1 desde YAML
|
// Carregar controls del player 1 desde YAML
|
||||||
|
|||||||
@@ -8,13 +8,9 @@
|
|||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
#include <ctime>
|
#include <ctime>
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
#include "core/audio/audio.hpp"
|
#include "core/audio/audio.hpp"
|
||||||
#include "core/entities/entity.hpp"
|
|
||||||
#include "core/input/input.hpp"
|
#include "core/input/input.hpp"
|
||||||
#include "core/math/easing.hpp"
|
|
||||||
#include "core/physics/collision.hpp"
|
|
||||||
#include "core/rendering/line_renderer.hpp"
|
#include "core/rendering/line_renderer.hpp"
|
||||||
#include "core/system/scene_context.hpp"
|
#include "core/system/scene_context.hpp"
|
||||||
#include "game/stage_system/stage_loader.hpp"
|
#include "game/stage_system/stage_loader.hpp"
|
||||||
@@ -223,7 +219,7 @@ void GameScene::stepShootingInput() {
|
|||||||
|
|
||||||
void GameScene::stepMidGameJoin() {
|
void GameScene::stepMidGameJoin() {
|
||||||
// Permitir join solo durante PLAYING.
|
// Permitir join solo durante PLAYING.
|
||||||
if (stage_manager_->get_estat() != StageSystem::EstatStage::PLAYING) {
|
if (stage_manager_->getState() != StageSystem::EstatStage::PLAYING) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -361,7 +357,7 @@ void GameScene::stepDeathSequence(float delta_time) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void GameScene::stepStageStateMachine(float delta_time) {
|
void GameScene::stepStageStateMachine(float delta_time) {
|
||||||
const StageSystem::EstatStage STATE = stage_manager_->get_estat();
|
const StageSystem::EstatStage STATE = stage_manager_->getState();
|
||||||
switch (STATE) {
|
switch (STATE) {
|
||||||
case StageSystem::EstatStage::INIT_HUD:
|
case StageSystem::EstatStage::INIT_HUD:
|
||||||
runStageInitHud(delta_time);
|
runStageInitHud(delta_time);
|
||||||
@@ -382,11 +378,11 @@ void GameScene::runStageInitHud(float delta_time) {
|
|||||||
// Update stage manager timer (puede cambiar el state).
|
// Update stage manager timer (puede cambiar el state).
|
||||||
stage_manager_->update(delta_time);
|
stage_manager_->update(delta_time);
|
||||||
// Si el state cambió, salir para no usar el timer del nuevo state.
|
// Si el state cambió, salir para no usar el timer del nuevo state.
|
||||||
if (stage_manager_->get_estat() != StageSystem::EstatStage::INIT_HUD) {
|
if (stage_manager_->getState() != StageSystem::EstatStage::INIT_HUD) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
float global_progress = 1.0F - (stage_manager_->get_timer_transicio() / Defaults::Game::INIT_HUD_DURATION);
|
float global_progress = 1.0F - (stage_manager_->getTransitionTimer() / Defaults::Game::INIT_HUD_DURATION);
|
||||||
global_progress = std::min(1.0F, global_progress);
|
global_progress = std::min(1.0F, global_progress);
|
||||||
|
|
||||||
const float SHIP1_P = Systems::InitHud::computeRangeProgress(
|
const float SHIP1_P = Systems::InitHud::computeRangeProgress(
|
||||||
@@ -429,8 +425,8 @@ void GameScene::runStagePlaying(float delta_time) {
|
|||||||
|
|
||||||
// Stage completado: cuando al menos un jugador está vivo y todos los enemies muertos.
|
// Stage completado: cuando al menos un jugador está vivo y todos los enemies muertos.
|
||||||
const bool ALGU_VIU = (hit_timer_per_player_[0] == 0.0F || hit_timer_per_player_[1] == 0.0F);
|
const bool ALGU_VIU = (hit_timer_per_player_[0] == 0.0F || hit_timer_per_player_[1] == 0.0F);
|
||||||
if (ALGU_VIU && stage_manager_->getSpawnController().tots_enemics_destruits(enemies_)) {
|
if (ALGU_VIU && stage_manager_->getSpawnController().allEnemiesDestroyed(enemies_)) {
|
||||||
stage_manager_->stage_completat();
|
stage_manager_->markStageCompleted();
|
||||||
Audio::get()->playSound(Defaults::Sound::GOOD_JOB_COMMANDER, Audio::Group::GAME);
|
Audio::get()->playSound(Defaults::Sound::GOOD_JOB_COMMANDER, Audio::Group::GAME);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -488,50 +484,74 @@ void GameScene::runCollisionDetections() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void GameScene::draw() {
|
void GameScene::draw() {
|
||||||
// Handle CONTINUE screen
|
|
||||||
if (game_over_state_ == GameOverState::CONTINUE) {
|
if (game_over_state_ == GameOverState::CONTINUE) {
|
||||||
// Draw game background elements first
|
drawContinueState();
|
||||||
drawMargins();
|
|
||||||
|
|
||||||
for (const auto& enemy : enemies_) {
|
|
||||||
enemy.draw();
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const auto& bullet : bullets_) {
|
|
||||||
bullet.draw();
|
|
||||||
}
|
|
||||||
|
|
||||||
debris_manager_.draw();
|
|
||||||
floating_score_manager_.draw();
|
|
||||||
drawScoreboard();
|
|
||||||
|
|
||||||
// Draw CONTINUE screen overlay
|
|
||||||
drawContinue();
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handle final GAME OVER state
|
|
||||||
if (game_over_state_ == GameOverState::GAME_OVER) {
|
if (game_over_state_ == GameOverState::GAME_OVER) {
|
||||||
// Game over: draw enemies, bullets, debris, and "GAME OVER" text
|
drawGameOverState();
|
||||||
drawMargins();
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (stage_manager_->getState()) {
|
||||||
|
case StageSystem::EstatStage::INIT_HUD:
|
||||||
|
drawInitHudState();
|
||||||
|
break;
|
||||||
|
case StageSystem::EstatStage::LEVEL_START:
|
||||||
|
drawLevelStartState();
|
||||||
|
break;
|
||||||
|
case StageSystem::EstatStage::PLAYING:
|
||||||
|
drawPlayingState();
|
||||||
|
break;
|
||||||
|
case StageSystem::EstatStage::LEVEL_COMPLETED:
|
||||||
|
drawLevelCompletedState();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void GameScene::drawEnemies() const {
|
||||||
for (const auto& enemy : enemies_) {
|
for (const auto& enemy : enemies_) {
|
||||||
enemy.draw();
|
enemy.draw();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void GameScene::drawBullets() const {
|
||||||
for (const auto& bullet : bullets_) {
|
for (const auto& bullet : bullets_) {
|
||||||
bullet.draw();
|
bullet.draw();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void GameScene::drawActiveShipsAlive() const {
|
||||||
|
for (uint8_t i = 0; i < 2; i++) {
|
||||||
|
bool jugador_actiu = (i == 0) ? match_config_.jugador1_actiu : match_config_.jugador2_actiu;
|
||||||
|
if (jugador_actiu && hit_timer_per_player_[i] == 0.0F) {
|
||||||
|
ships_[i].draw();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void GameScene::drawContinueState() {
|
||||||
|
drawMargins();
|
||||||
|
drawEnemies();
|
||||||
|
drawBullets();
|
||||||
|
debris_manager_.draw();
|
||||||
|
floating_score_manager_.draw();
|
||||||
|
drawScoreboard();
|
||||||
|
drawContinue();
|
||||||
|
}
|
||||||
|
|
||||||
|
void GameScene::drawGameOverState() {
|
||||||
|
drawMargins();
|
||||||
|
drawEnemies();
|
||||||
|
drawBullets();
|
||||||
debris_manager_.draw();
|
debris_manager_.draw();
|
||||||
floating_score_manager_.draw();
|
floating_score_manager_.draw();
|
||||||
|
|
||||||
// Draw centered "GAME OVER" text
|
|
||||||
const std::string GAME_OVER_TEXT = "GAME OVER";
|
const std::string GAME_OVER_TEXT = "GAME OVER";
|
||||||
constexpr float SCALE = Defaults::Game::GameOverScreen::TEXT_SCALE;
|
constexpr float SCALE = Defaults::Game::GameOverScreen::TEXT_SCALE;
|
||||||
constexpr float SPACING = Defaults::Game::GameOverScreen::TEXT_SPACING;
|
constexpr float SPACING = Defaults::Game::GameOverScreen::TEXT_SPACING;
|
||||||
|
|
||||||
// Calcular centro de l'àrea de juego usant constants
|
|
||||||
const SDL_FRect& play_area = Defaults::Zones::PLAYAREA;
|
const SDL_FRect& play_area = Defaults::Zones::PLAYAREA;
|
||||||
float centre_x = play_area.x + (play_area.w / 2.0F);
|
float centre_x = play_area.x + (play_area.w / 2.0F);
|
||||||
float centre_y = play_area.y + (play_area.h / 2.0F);
|
float centre_y = play_area.y + (play_area.h / 2.0F);
|
||||||
@@ -539,20 +559,13 @@ void GameScene::draw() {
|
|||||||
text_.renderCentered(GAME_OVER_TEXT, {.x = centre_x, .y = centre_y}, SCALE, SPACING);
|
text_.renderCentered(GAME_OVER_TEXT, {.x = centre_x, .y = centre_y}, SCALE, SPACING);
|
||||||
|
|
||||||
drawScoreboard();
|
drawScoreboard();
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// [NEW] Stage state rendering
|
void GameScene::drawInitHudState() {
|
||||||
StageSystem::EstatStage state = stage_manager_->get_estat();
|
float timer = stage_manager_->getTransitionTimer();
|
||||||
|
|
||||||
switch (state) {
|
|
||||||
case StageSystem::EstatStage::INIT_HUD: {
|
|
||||||
// Calcular progrés de cada animación independent
|
|
||||||
float timer = stage_manager_->get_timer_transicio();
|
|
||||||
float total_time = Defaults::Game::INIT_HUD_DURATION;
|
float total_time = Defaults::Game::INIT_HUD_DURATION;
|
||||||
float global_progress = 1.0F - (timer / total_time);
|
float global_progress = 1.0F - (timer / total_time);
|
||||||
|
|
||||||
// [NEW] Calcular progress independiente para cada elemento
|
|
||||||
float rect_progress = Systems::InitHud::computeRangeProgress(
|
float rect_progress = Systems::InitHud::computeRangeProgress(
|
||||||
global_progress,
|
global_progress,
|
||||||
Defaults::Game::INIT_HUD_RECT_RATIO_INIT,
|
Defaults::Game::INIT_HUD_RECT_RATIO_INIT,
|
||||||
@@ -573,14 +586,11 @@ void GameScene::draw() {
|
|||||||
Defaults::Game::INIT_HUD_SHIP2_RATIO_INIT,
|
Defaults::Game::INIT_HUD_SHIP2_RATIO_INIT,
|
||||||
Defaults::Game::INIT_HUD_SHIP2_RATIO_END);
|
Defaults::Game::INIT_HUD_SHIP2_RATIO_END);
|
||||||
|
|
||||||
// Dibuixar elements animats
|
|
||||||
if (rect_progress > 0.0F) {
|
if (rect_progress > 0.0F) {
|
||||||
// [NOU] Reproduir so cuando comença l'animación del rectangle
|
|
||||||
if (!init_hud_rect_sound_played_) {
|
if (!init_hud_rect_sound_played_) {
|
||||||
Audio::get()->playSound(Defaults::Sound::INIT_HUD, Audio::Group::GAME);
|
Audio::get()->playSound(Defaults::Sound::INIT_HUD, Audio::Group::GAME);
|
||||||
init_hud_rect_sound_played_ = true;
|
init_hud_rect_sound_played_ = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
Systems::InitHud::drawBordersAnimated(sdl_.getRenderer(), rect_progress);
|
Systems::InitHud::drawBordersAnimated(sdl_.getRenderer(), rect_progress);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -588,7 +598,6 @@ void GameScene::draw() {
|
|||||||
Systems::InitHud::drawScoreboardAnimated(text_, buildScoreboard(), score_progress);
|
Systems::InitHud::drawScoreboardAnimated(text_, buildScoreboard(), score_progress);
|
||||||
}
|
}
|
||||||
|
|
||||||
// [MODIFICAT] Dibuixar naves con progress independent
|
|
||||||
if (ship1_progress > 0.0F && match_config_.jugador1_actiu && !ships_[0].isHit()) {
|
if (ship1_progress > 0.0F && match_config_.jugador1_actiu && !ships_[0].isHit()) {
|
||||||
ships_[0].draw();
|
ships_[0].draw();
|
||||||
}
|
}
|
||||||
@@ -596,82 +605,36 @@ void GameScene::draw() {
|
|||||||
if (ship2_progress > 0.0F && match_config_.jugador2_actiu && !ships_[1].isHit()) {
|
if (ship2_progress > 0.0F && match_config_.jugador2_actiu && !ships_[1].isHit()) {
|
||||||
ships_[1].draw();
|
ships_[1].draw();
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
case StageSystem::EstatStage::LEVEL_START:
|
void GameScene::drawLevelStartState() {
|
||||||
drawMargins();
|
drawMargins();
|
||||||
// [NEW] Draw both ships if active and alive
|
drawActiveShipsAlive();
|
||||||
for (uint8_t i = 0; i < 2; i++) {
|
drawBullets();
|
||||||
bool jugador_actiu = (i == 0) ? match_config_.jugador1_actiu : match_config_.jugador2_actiu;
|
|
||||||
if (jugador_actiu && hit_timer_per_player_[i] == 0.0F) {
|
|
||||||
ships_[i].draw();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// [NEW] Draw bullets
|
|
||||||
for (const auto& bullet : bullets_) {
|
|
||||||
bullet.draw();
|
|
||||||
}
|
|
||||||
|
|
||||||
// [NEW] Draw debris
|
|
||||||
debris_manager_.draw();
|
debris_manager_.draw();
|
||||||
floating_score_manager_.draw();
|
floating_score_manager_.draw();
|
||||||
|
drawStageMessage(stage_manager_->getLevelStartMessage());
|
||||||
// [EXISTING] Draw intro message and score
|
|
||||||
drawStageMessage(stage_manager_->get_missatge_level_start());
|
|
||||||
drawScoreboard();
|
drawScoreboard();
|
||||||
break;
|
}
|
||||||
|
|
||||||
case StageSystem::EstatStage::PLAYING:
|
void GameScene::drawPlayingState() {
|
||||||
drawMargins();
|
drawMargins();
|
||||||
|
drawActiveShipsAlive();
|
||||||
// [EXISTING] Normal rendering - active ships
|
drawEnemies();
|
||||||
for (uint8_t i = 0; i < 2; i++) {
|
drawBullets();
|
||||||
bool jugador_actiu = (i == 0) ? match_config_.jugador1_actiu : match_config_.jugador2_actiu;
|
|
||||||
if (jugador_actiu && hit_timer_per_player_[i] == 0.0F) {
|
|
||||||
ships_[i].draw();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const auto& enemy : enemies_) {
|
|
||||||
enemy.draw();
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const auto& bullet : bullets_) {
|
|
||||||
bullet.draw();
|
|
||||||
}
|
|
||||||
|
|
||||||
debris_manager_.draw();
|
debris_manager_.draw();
|
||||||
floating_score_manager_.draw();
|
floating_score_manager_.draw();
|
||||||
drawScoreboard();
|
drawScoreboard();
|
||||||
break;
|
}
|
||||||
|
|
||||||
case StageSystem::EstatStage::LEVEL_COMPLETED:
|
void GameScene::drawLevelCompletedState() {
|
||||||
drawMargins();
|
drawMargins();
|
||||||
// [NEW] Draw both ships if active and alive
|
drawActiveShipsAlive();
|
||||||
for (uint8_t i = 0; i < 2; i++) {
|
drawBullets();
|
||||||
bool jugador_actiu = (i == 0) ? match_config_.jugador1_actiu : match_config_.jugador2_actiu;
|
|
||||||
if (jugador_actiu && hit_timer_per_player_[i] == 0.0F) {
|
|
||||||
ships_[i].draw();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// [NEW] Draw bullets (allow last shots to be visible)
|
|
||||||
for (const auto& bullet : bullets_) {
|
|
||||||
bullet.draw();
|
|
||||||
}
|
|
||||||
|
|
||||||
// [NEW] Draw debris (from last destroyed enemies)
|
|
||||||
debris_manager_.draw();
|
debris_manager_.draw();
|
||||||
floating_score_manager_.draw();
|
floating_score_manager_.draw();
|
||||||
|
|
||||||
// [EXISTING] Draw completion message and score
|
|
||||||
drawStageMessage(StageSystem::Constants::MISSATGE_LEVEL_COMPLETED);
|
drawStageMessage(StageSystem::Constants::MISSATGE_LEVEL_COMPLETED);
|
||||||
drawScoreboard();
|
drawScoreboard();
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void GameScene::tocado(uint8_t player_id) {
|
void GameScene::tocado(uint8_t player_id) {
|
||||||
@@ -764,7 +727,7 @@ auto GameScene::buildScoreboard() const -> std::string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Nivel (2 dígits)
|
// Nivel (2 dígits)
|
||||||
uint8_t stage_num = stage_manager_->get_stage_actual();
|
uint8_t stage_num = stage_manager_->getCurrentStage();
|
||||||
std::string stage_str = (stage_num < 10) ? "0" + std::to_string(stage_num)
|
std::string stage_str = (stage_num < 10) ? "0" + std::to_string(stage_num)
|
||||||
: std::to_string(stage_num);
|
: std::to_string(stage_num);
|
||||||
|
|
||||||
@@ -801,7 +764,7 @@ void GameScene::drawStageMessage(const std::string& message) {
|
|||||||
float total_time;
|
float total_time;
|
||||||
float typing_ratio;
|
float typing_ratio;
|
||||||
|
|
||||||
if (stage_manager_->get_estat() == StageSystem::EstatStage::LEVEL_START) {
|
if (stage_manager_->getState() == StageSystem::EstatStage::LEVEL_START) {
|
||||||
total_time = Defaults::Game::LEVEL_START_DURATION;
|
total_time = Defaults::Game::LEVEL_START_DURATION;
|
||||||
typing_ratio = Defaults::Game::LEVEL_START_TYPING_RATIO;
|
typing_ratio = Defaults::Game::LEVEL_START_TYPING_RATIO;
|
||||||
} else { // LEVEL_COMPLETED
|
} else { // LEVEL_COMPLETED
|
||||||
@@ -810,7 +773,7 @@ void GameScene::drawStageMessage(const std::string& message) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Calculate progress from timer (0.0 at start → 1.0 at end)
|
// Calculate progress from timer (0.0 at start → 1.0 at end)
|
||||||
float remaining_time = stage_manager_->get_timer_transicio();
|
float remaining_time = stage_manager_->getTransitionTimer();
|
||||||
float progress = 1.0F - (remaining_time / total_time);
|
float progress = 1.0F - (remaining_time / total_time);
|
||||||
|
|
||||||
// Determine how many characters to show
|
// Determine how many characters to show
|
||||||
@@ -834,7 +797,7 @@ void GameScene::drawStageMessage(const std::string& message) {
|
|||||||
// ===================================================
|
// ===================================================
|
||||||
|
|
||||||
// Calculate text width at base scale (using FULL message for position calculation)
|
// Calculate text width at base scale (using FULL message for position calculation)
|
||||||
float text_width_at_base = text_.get_text_width(message, BASE_SCALE, SPACING);
|
float text_width_at_base = Graphics::VectorText::getTextWidth(message, BASE_SCALE, SPACING);
|
||||||
|
|
||||||
// Auto-scale if text exceeds max width
|
// Auto-scale if text exceeds max width
|
||||||
float scale = (text_width_at_base <= MAX_WIDTH)
|
float scale = (text_width_at_base <= MAX_WIDTH)
|
||||||
@@ -842,8 +805,8 @@ void GameScene::drawStageMessage(const std::string& message) {
|
|||||||
: MAX_WIDTH / text_width_at_base;
|
: MAX_WIDTH / text_width_at_base;
|
||||||
|
|
||||||
// Recalculate dimensions with final scale (using FULL message for centering)
|
// Recalculate dimensions with final scale (using FULL message for centering)
|
||||||
float full_text_width = text_.get_text_width(message, scale, SPACING);
|
float full_text_width = Graphics::VectorText::getTextWidth(message, scale, SPACING);
|
||||||
float text_height = text_.get_text_height(scale);
|
float text_height = Graphics::VectorText::getTextHeight(scale);
|
||||||
|
|
||||||
// Calculate position as if FULL text was there (for fixed position typewriter)
|
// Calculate position as if FULL text was there (for fixed position typewriter)
|
||||||
float x = play_area.x + ((play_area.w - full_text_width) / 2.0F);
|
float x = play_area.x + ((play_area.w - full_text_width) / 2.0F);
|
||||||
@@ -862,7 +825,7 @@ auto GameScene::getSpawnPoint(uint8_t player_id) const -> Vec2 {
|
|||||||
const SDL_FRect& zona = Defaults::Zones::PLAYAREA;
|
const SDL_FRect& zona = Defaults::Zones::PLAYAREA;
|
||||||
|
|
||||||
float x_ratio;
|
float x_ratio;
|
||||||
if (match_config_.es_un_jugador()) {
|
if (match_config_.isSinglePlayer()) {
|
||||||
// Un sol player: spawn al centro (50%)
|
// Un sol player: spawn al centro (50%)
|
||||||
x_ratio = 0.5F;
|
x_ratio = 0.5F;
|
||||||
} else {
|
} else {
|
||||||
@@ -901,7 +864,7 @@ void GameScene::fireBullet(uint8_t player_id) {
|
|||||||
// Buscar primera bullet inactiva en el pool del player
|
// Buscar primera bullet inactiva en el pool del player
|
||||||
int start_idx = player_id * 3; // P1=[0,1,2], P2=[3,4,5]
|
int start_idx = player_id * 3; // P1=[0,1,2], P2=[3,4,5]
|
||||||
for (int i = start_idx; i < start_idx + 3; i++) {
|
for (int i = start_idx; i < start_idx + 3; i++) {
|
||||||
if (!bullets_[i].esta_activa()) {
|
if (!bullets_[i].isActive()) {
|
||||||
bullets_[i].disparar(posicio_dispar, ship_angle, player_id);
|
bullets_[i].disparar(posicio_dispar, ship_angle, player_id);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -100,6 +100,18 @@ class GameScene final : public Scene {
|
|||||||
// [NEW] Stage system helpers
|
// [NEW] Stage system helpers
|
||||||
void drawStageMessage(const std::string& message);
|
void drawStageMessage(const std::string& message);
|
||||||
|
|
||||||
|
// Helpers de renderitzat (extracció de draw() per reduir complexitat).
|
||||||
|
// Cadascun gestiona un bloc concret; draw() només despatxa segons l'estat.
|
||||||
|
void drawEnemies() const;
|
||||||
|
void drawBullets() const;
|
||||||
|
void drawActiveShipsAlive() const;
|
||||||
|
void drawContinueState();
|
||||||
|
void drawGameOverState();
|
||||||
|
void drawInitHudState();
|
||||||
|
void drawLevelStartState();
|
||||||
|
void drawPlayingState();
|
||||||
|
void drawLevelCompletedState();
|
||||||
|
|
||||||
// [NEW] Función helper del marcador
|
// [NEW] Función helper del marcador
|
||||||
[[nodiscard]] auto buildScoreboard() const -> std::string;
|
[[nodiscard]] auto buildScoreboard() const -> std::string;
|
||||||
|
|
||||||
|
|||||||
@@ -104,7 +104,7 @@ void LogoScene::initLetters() {
|
|||||||
float min_x = FLT_MAX;
|
float min_x = FLT_MAX;
|
||||||
float max_x = -FLT_MAX;
|
float max_x = -FLT_MAX;
|
||||||
|
|
||||||
for (const auto& prim : shape->get_primitives()) {
|
for (const auto& prim : shape->getPrimitives()) {
|
||||||
for (const auto& point : prim.points) {
|
for (const auto& point : prim.points) {
|
||||||
min_x = std::min(min_x, point.x);
|
min_x = std::min(min_x, point.x);
|
||||||
max_x = std::max(max_x, point.x);
|
max_x = std::max(max_x, point.x);
|
||||||
@@ -321,7 +321,7 @@ void LogoScene::draw() {
|
|||||||
float current_scale =
|
float current_scale =
|
||||||
ESCALA_INICIAL + ((ESCALA_FINAL - ESCALA_INICIAL) * ease_factor);
|
ESCALA_INICIAL + ((ESCALA_FINAL - ESCALA_INICIAL) * ease_factor);
|
||||||
|
|
||||||
Rendering::render_shape(
|
Rendering::renderShape(
|
||||||
sdl_.getRenderer(),
|
sdl_.getRenderer(),
|
||||||
lletra.shape,
|
lletra.shape,
|
||||||
pos_actual,
|
pos_actual,
|
||||||
@@ -344,7 +344,7 @@ void LogoScene::draw() {
|
|||||||
if (!explotades.contains(i)) {
|
if (!explotades.contains(i)) {
|
||||||
const auto& lletra = lletres_[i];
|
const auto& lletra = lletres_[i];
|
||||||
|
|
||||||
Rendering::render_shape(
|
Rendering::renderShape(
|
||||||
sdl_.getRenderer(),
|
sdl_.getRenderer(),
|
||||||
lletra.shape,
|
lletra.shape,
|
||||||
lletra.position,
|
lletra.position,
|
||||||
|
|||||||
@@ -13,7 +13,6 @@
|
|||||||
|
|
||||||
#include "core/defaults.hpp"
|
#include "core/defaults.hpp"
|
||||||
#include "core/graphics/shape.hpp"
|
#include "core/graphics/shape.hpp"
|
||||||
#include "core/input/input_types.hpp"
|
|
||||||
#include "core/rendering/sdl_manager.hpp"
|
#include "core/rendering/sdl_manager.hpp"
|
||||||
#include "core/system/scene.hpp"
|
#include "core/system/scene.hpp"
|
||||||
#include "core/system/scene_context.hpp"
|
#include "core/system/scene_context.hpp"
|
||||||
|
|||||||
@@ -11,6 +11,7 @@
|
|||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
#include "core/audio/audio.hpp"
|
#include "core/audio/audio.hpp"
|
||||||
|
#include "core/defaults.hpp"
|
||||||
#include "core/graphics/shape_loader.hpp"
|
#include "core/graphics/shape_loader.hpp"
|
||||||
#include "core/input/input.hpp"
|
#include "core/input/input.hpp"
|
||||||
#include "core/rendering/shape_renderer.hpp"
|
#include "core/rendering/shape_renderer.hpp"
|
||||||
@@ -64,10 +65,10 @@ TitleScene::TitleScene(SDLManager& sdl, SceneContext& context)
|
|||||||
// Brightness depèn de l'opción
|
// Brightness depèn de l'opción
|
||||||
if (estat_actual_ == TitleState::MAIN) {
|
if (estat_actual_ == TitleState::MAIN) {
|
||||||
// Si saltem a MAIN, starfield instantàniament brillant
|
// Si saltem a MAIN, starfield instantàniament brillant
|
||||||
starfield_->set_brightness(BRIGHTNESS_STARFIELD);
|
starfield_->setBrightness(BRIGHTNESS_STARFIELD);
|
||||||
} else {
|
} else {
|
||||||
// Flux normal: comença con brightness 0.0 per fade-in
|
// Flux normal: comença con brightness 0.0 per fade-in
|
||||||
starfield_->set_brightness(0.0F);
|
starfield_->setBrightness(0.0F);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Inicialitzar animador de naves 3D
|
// Inicialitzar animador de naves 3D
|
||||||
@@ -76,11 +77,11 @@ TitleScene::TitleScene(SDLManager& sdl, SceneContext& context)
|
|||||||
|
|
||||||
if (estat_actual_ == TitleState::MAIN) {
|
if (estat_actual_ == TitleState::MAIN) {
|
||||||
// Jump to MAIN: empezar entrada inmediatamente
|
// Jump to MAIN: empezar entrada inmediatamente
|
||||||
ship_animator_->set_visible(true);
|
ship_animator_->setVisible(true);
|
||||||
ship_animator_->start_entry_animation();
|
ship_animator_->startEntryAnimation();
|
||||||
} else {
|
} else {
|
||||||
// Flux normal: NO empezar entrada todavía (esperaran a MAIN)
|
// Flux normal: NO empezar entrada todavía (esperaran a MAIN)
|
||||||
ship_animator_->set_visible(false);
|
ship_animator_->setVisible(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Inicialitzar lletres del título "ORNI ATTACK!"
|
// Inicialitzar lletres del título "ORNI ATTACK!"
|
||||||
@@ -126,7 +127,7 @@ void TitleScene::initTitle() {
|
|||||||
float min_y = FLT_MAX;
|
float min_y = FLT_MAX;
|
||||||
float max_y = -FLT_MAX;
|
float max_y = -FLT_MAX;
|
||||||
|
|
||||||
for (const auto& prim : shape->get_primitives()) {
|
for (const auto& prim : shape->getPrimitives()) {
|
||||||
for (const auto& point : prim.points) {
|
for (const auto& point : prim.points) {
|
||||||
min_x = std::min(min_x, point.x);
|
min_x = std::min(min_x, point.x);
|
||||||
max_x = std::max(max_x, point.x);
|
max_x = std::max(max_x, point.x);
|
||||||
@@ -200,7 +201,7 @@ void TitleScene::initTitle() {
|
|||||||
float min_y = FLT_MAX;
|
float min_y = FLT_MAX;
|
||||||
float max_y = -FLT_MAX;
|
float max_y = -FLT_MAX;
|
||||||
|
|
||||||
for (const auto& prim : shape->get_primitives()) {
|
for (const auto& prim : shape->getPrimitives()) {
|
||||||
for (const auto& point : prim.points) {
|
for (const auto& point : prim.points) {
|
||||||
min_x = std::min(min_x, point.x);
|
min_x = std::min(min_x, point.x);
|
||||||
max_x = std::max(max_x, point.x);
|
max_x = std::max(max_x, point.x);
|
||||||
@@ -284,7 +285,7 @@ void TitleScene::inicialitzarJailgames() {
|
|||||||
float max_x = -FLT_MAX;
|
float max_x = -FLT_MAX;
|
||||||
float min_y = FLT_MAX;
|
float min_y = FLT_MAX;
|
||||||
float max_y = -FLT_MAX;
|
float max_y = -FLT_MAX;
|
||||||
for (const auto& prim : shape->get_primitives()) {
|
for (const auto& prim : shape->getPrimitives()) {
|
||||||
for (const auto& point : prim.points) {
|
for (const auto& point : prim.points) {
|
||||||
min_x = std::min(min_x, point.x);
|
min_x = std::min(min_x, point.x);
|
||||||
max_x = std::max(max_x, point.x);
|
max_x = std::max(max_x, point.x);
|
||||||
@@ -326,7 +327,7 @@ void TitleScene::inicialitzarJailgames() {
|
|||||||
void TitleScene::dibuixarPeuTitol(float spacing) const {
|
void TitleScene::dibuixarPeuTitol(float spacing) const {
|
||||||
// Logo JAILGAMES pequeño sobre el copyright.
|
// Logo JAILGAMES pequeño sobre el copyright.
|
||||||
for (const auto& lletra : lletres_jailgames_) {
|
for (const auto& lletra : lletres_jailgames_) {
|
||||||
Rendering::render_shape(sdl_.getRenderer(), lletra.shape,
|
Rendering::renderShape(sdl_.getRenderer(), lletra.shape,
|
||||||
lletra.position, 0.0F,
|
lletra.position, 0.0F,
|
||||||
Defaults::Title::Layout::JAILGAMES_SCALE,
|
Defaults::Title::Layout::JAILGAMES_SCALE,
|
||||||
1.0F);
|
1.0F);
|
||||||
@@ -365,7 +366,28 @@ void TitleScene::update(float delta_time) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
switch (estat_actual_) {
|
switch (estat_actual_) {
|
||||||
case TitleState::STARFIELD_FADE_IN: {
|
case TitleState::STARFIELD_FADE_IN:
|
||||||
|
updateStarfieldFadeInState(delta_time);
|
||||||
|
break;
|
||||||
|
case TitleState::STARFIELD:
|
||||||
|
updateStarfieldState(delta_time);
|
||||||
|
break;
|
||||||
|
case TitleState::MAIN:
|
||||||
|
updateMainState(delta_time);
|
||||||
|
break;
|
||||||
|
case TitleState::PLAYER_JOIN_PHASE:
|
||||||
|
updatePlayerJoinPhaseState(delta_time);
|
||||||
|
break;
|
||||||
|
case TitleState::BLACK_SCREEN:
|
||||||
|
updateBlackScreenState(delta_time);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
handleSkipInput();
|
||||||
|
handleStartInput();
|
||||||
|
}
|
||||||
|
|
||||||
|
void TitleScene::updateStarfieldFadeInState(float delta_time) {
|
||||||
temps_acumulat_ += delta_time;
|
temps_acumulat_ += delta_time;
|
||||||
|
|
||||||
// Calcular progrés del fade (0.0 → 1.0)
|
// Calcular progrés del fade (0.0 → 1.0)
|
||||||
@@ -373,18 +395,17 @@ void TitleScene::update(float delta_time) {
|
|||||||
|
|
||||||
// Lerp brightness de 0.0 a BRIGHTNESS_STARFIELD
|
// Lerp brightness de 0.0 a BRIGHTNESS_STARFIELD
|
||||||
float brightness_actual = progress * BRIGHTNESS_STARFIELD;
|
float brightness_actual = progress * BRIGHTNESS_STARFIELD;
|
||||||
starfield_->set_brightness(brightness_actual);
|
starfield_->setBrightness(brightness_actual);
|
||||||
|
|
||||||
// Transición a STARFIELD cuando el fade es completa
|
// Transición a STARFIELD cuando el fade es completa
|
||||||
if (temps_acumulat_ >= DURACIO_FADE_IN) {
|
if (temps_acumulat_ >= DURACIO_FADE_IN) {
|
||||||
estat_actual_ = TitleState::STARFIELD;
|
estat_actual_ = TitleState::STARFIELD;
|
||||||
temps_acumulat_ = 0.0F; // Reset timer per al següent state
|
temps_acumulat_ = 0.0F; // Reset timer per al següent state
|
||||||
starfield_->set_brightness(BRIGHTNESS_STARFIELD); // Assegurar value final
|
starfield_->setBrightness(BRIGHTNESS_STARFIELD); // Assegurar value final
|
||||||
}
|
}
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
case TitleState::STARFIELD:
|
void TitleScene::updateStarfieldState(float delta_time) {
|
||||||
temps_acumulat_ += delta_time;
|
temps_acumulat_ += delta_time;
|
||||||
if (temps_acumulat_ >= DURACIO_INIT) {
|
if (temps_acumulat_ >= DURACIO_INIT) {
|
||||||
estat_actual_ = TitleState::MAIN;
|
estat_actual_ = TitleState::MAIN;
|
||||||
@@ -394,17 +415,16 @@ void TitleScene::update(float delta_time) {
|
|||||||
|
|
||||||
// Naves esperaran ENTRANCE_DELAY antes de entrar (no start aquí)
|
// Naves esperaran ENTRANCE_DELAY antes de entrar (no start aquí)
|
||||||
}
|
}
|
||||||
break;
|
}
|
||||||
|
|
||||||
case TitleState::MAIN: {
|
void TitleScene::updateMainState(float delta_time) {
|
||||||
temps_estat_main_ += delta_time;
|
temps_estat_main_ += delta_time;
|
||||||
|
|
||||||
// Iniciar animación de entrada de naves después del delay
|
// Iniciar animación de entrada de naves después del delay
|
||||||
if (temps_estat_main_ >= Defaults::Title::Ships::ENTRANCE_DELAY) {
|
if (temps_estat_main_ >= Defaults::Title::Ships::ENTRANCE_DELAY &&
|
||||||
if (ship_animator_ && !ship_animator_->is_visible()) {
|
ship_animator_ && !ship_animator_->isVisible()) {
|
||||||
ship_animator_->set_visible(true);
|
ship_animator_->setVisible(true);
|
||||||
ship_animator_->start_entry_animation();
|
ship_animator_->startEntryAnimation();
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fase 1: Estàtic (0-10s)
|
// Fase 1: Estàtic (0-10s)
|
||||||
@@ -426,17 +446,15 @@ void TitleScene::update(float delta_time) {
|
|||||||
|
|
||||||
// Actualitzar animación del logo
|
// Actualitzar animación del logo
|
||||||
updateLogoAnimation(delta_time);
|
updateLogoAnimation(delta_time);
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
case TitleState::PLAYER_JOIN_PHASE:
|
void TitleScene::updatePlayerJoinPhaseState(float delta_time) {
|
||||||
temps_acumulat_ += delta_time;
|
temps_acumulat_ += delta_time;
|
||||||
|
|
||||||
// Continuar animación orbital durante la transición
|
// Continuar animación orbital durante la transición
|
||||||
updateLogoAnimation(delta_time);
|
updateLogoAnimation(delta_time);
|
||||||
|
|
||||||
// [NOU] Continuar comprovant si l'altre player quiere unir-se durante la transición ("late join")
|
// [NOU] Continuar comprovant si l'altre player quiere unir-se durante la transición ("late join")
|
||||||
{
|
|
||||||
bool p1_actiu_abans = match_config_.jugador1_actiu;
|
bool p1_actiu_abans = match_config_.jugador1_actiu;
|
||||||
bool p2_actiu_abans = match_config_.jugador2_actiu;
|
bool p2_actiu_abans = match_config_.jugador2_actiu;
|
||||||
|
|
||||||
@@ -445,16 +463,7 @@ void TitleScene::update(float delta_time) {
|
|||||||
context_.setMatchConfig(match_config_);
|
context_.setMatchConfig(match_config_);
|
||||||
|
|
||||||
// Trigger animación de salida per la ship que acaba de unir-se
|
// Trigger animación de salida per la ship que acaba de unir-se
|
||||||
if (ship_animator_) {
|
triggerExitForJoinedPlayers(p1_actiu_abans, p2_actiu_abans, "late join - ");
|
||||||
if (match_config_.jugador1_actiu && !p1_actiu_abans) {
|
|
||||||
ship_animator_->trigger_exit_animation_for_player(1);
|
|
||||||
std::cout << "[TitleScene] P1 late join - ship exiting\n";
|
|
||||||
}
|
|
||||||
if (match_config_.jugador2_actiu && !p2_actiu_abans) {
|
|
||||||
ship_animator_->trigger_exit_animation_for_player(2);
|
|
||||||
std::cout << "[TitleScene] P2 late join - ship exiting\n";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Reproducir so de START cuando el segon player s'uneix
|
// Reproducir so de START cuando el segon player s'uneix
|
||||||
Audio::get()->playSound(Defaults::Sound::START, Audio::Group::GAME);
|
Audio::get()->playSound(Defaults::Sound::START, Audio::Group::GAME);
|
||||||
@@ -464,7 +473,6 @@ void TitleScene::update(float delta_time) {
|
|||||||
|
|
||||||
std::cout << "[TitleScene] Segon player s'ha unit - so i timer reiniciats\n";
|
std::cout << "[TitleScene] Segon player s'ha unit - so i timer reiniciats\n";
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (temps_acumulat_ >= DURACIO_TRANSITION) {
|
if (temps_acumulat_ >= DURACIO_TRANSITION) {
|
||||||
// Transición a pantalla negra
|
// Transición a pantalla negra
|
||||||
@@ -472,9 +480,9 @@ void TitleScene::update(float delta_time) {
|
|||||||
temps_acumulat_ = 0.0F;
|
temps_acumulat_ = 0.0F;
|
||||||
std::cout << "[TitleScene] Passant a BLACK_SCREEN\n";
|
std::cout << "[TitleScene] Passant a BLACK_SCREEN\n";
|
||||||
}
|
}
|
||||||
break;
|
}
|
||||||
|
|
||||||
case TitleState::BLACK_SCREEN:
|
void TitleScene::updateBlackScreenState(float delta_time) {
|
||||||
temps_acumulat_ += delta_time;
|
temps_acumulat_ += delta_time;
|
||||||
|
|
||||||
// No animation, no input checking - just wait
|
// No animation, no input checking - just wait
|
||||||
@@ -483,32 +491,41 @@ void TitleScene::update(float delta_time) {
|
|||||||
context_.setNextScene(SceneType::GAME);
|
context_.setNextScene(SceneType::GAME);
|
||||||
std::cout << "[TitleScene] Canviant a escena GAME\n";
|
std::cout << "[TitleScene] Canviant a escena GAME\n";
|
||||||
}
|
}
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TitleScene::handleSkipInput() {
|
||||||
// Verificar botones de skip (FIRE/THRUST/START) para saltar escenas ANTES de MAIN
|
// Verificar botones de skip (FIRE/THRUST/START) para saltar escenas ANTES de MAIN
|
||||||
if (estat_actual_ == TitleState::STARFIELD_FADE_IN || estat_actual_ == TitleState::STARFIELD) {
|
if (estat_actual_ != TitleState::STARFIELD_FADE_IN && estat_actual_ != TitleState::STARFIELD) {
|
||||||
if (checkSkipButtonPressed()) {
|
return;
|
||||||
|
}
|
||||||
|
if (!checkSkipButtonPressed()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
// Saltar a MAIN
|
// Saltar a MAIN
|
||||||
estat_actual_ = TitleState::MAIN;
|
estat_actual_ = TitleState::MAIN;
|
||||||
starfield_->set_brightness(BRIGHTNESS_STARFIELD);
|
starfield_->setBrightness(BRIGHTNESS_STARFIELD);
|
||||||
temps_estat_main_ = 0.0F;
|
temps_estat_main_ = 0.0F;
|
||||||
|
|
||||||
// Naves esperaran ENTRANCE_DELAY antes de entrar (no start aquí)
|
// Naves esperaran ENTRANCE_DELAY antes de entrar (no start aquí)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TitleScene::handleStartInput() {
|
||||||
|
// Verificar boton START para start match desde MAIN
|
||||||
|
if (estat_actual_ != TitleState::MAIN) {
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Verificar boton START para start match desde MAIN
|
|
||||||
if (estat_actual_ == TitleState::MAIN) {
|
|
||||||
// Guardar state anterior per detectar qui ha premut START AQUEST frame
|
// Guardar state anterior per detectar qui ha premut START AQUEST frame
|
||||||
bool p1_actiu_abans = match_config_.jugador1_actiu;
|
bool p1_actiu_abans = match_config_.jugador1_actiu;
|
||||||
bool p2_actiu_abans = match_config_.jugador2_actiu;
|
bool p2_actiu_abans = match_config_.jugador2_actiu;
|
||||||
|
|
||||||
if (checkStartGameButtonPressed()) {
|
if (!checkStartGameButtonPressed()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Si START es prem durante el delay (naves aún invisibles), saltar-las a FLOATING
|
// Si START es prem durante el delay (naves aún invisibles), saltar-las a FLOATING
|
||||||
if (ship_animator_ && !ship_animator_->is_visible()) {
|
if (ship_animator_ && !ship_animator_->isVisible()) {
|
||||||
ship_animator_->set_visible(true);
|
ship_animator_->setVisible(true);
|
||||||
ship_animator_->skip_to_floating_state();
|
ship_animator_->skipToFloatingState();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Configurar match antes de canviar de escena
|
// Configurar match antes de canviar de escena
|
||||||
@@ -525,20 +542,24 @@ void TitleScene::update(float delta_time) {
|
|||||||
temps_acumulat_ = 0.0F;
|
temps_acumulat_ = 0.0F;
|
||||||
|
|
||||||
// Trigger animación de salida NOMÉS per las naves que han premut START
|
// Trigger animación de salida NOMÉS per las naves que han premut START
|
||||||
if (ship_animator_) {
|
triggerExitForJoinedPlayers(p1_actiu_abans, p2_actiu_abans, "");
|
||||||
if (match_config_.jugador1_actiu && !p1_actiu_abans) {
|
|
||||||
ship_animator_->trigger_exit_animation_for_player(1);
|
|
||||||
std::cout << "[TitleScene] P1 ship exiting\n";
|
|
||||||
}
|
|
||||||
if (match_config_.jugador2_actiu && !p2_actiu_abans) {
|
|
||||||
ship_animator_->trigger_exit_animation_for_player(2);
|
|
||||||
std::cout << "[TitleScene] P2 ship exiting\n";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Audio::get()->fadeOutMusic(MUSIC_FADE);
|
Audio::get()->fadeOutMusic(MUSIC_FADE);
|
||||||
Audio::get()->playSound(Defaults::Sound::START, Audio::Group::GAME);
|
Audio::get()->playSound(Defaults::Sound::START, Audio::Group::GAME);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TitleScene::triggerExitForJoinedPlayers(bool p1_was_active, bool p2_was_active,
|
||||||
|
const char* log_prefix) {
|
||||||
|
if (ship_animator_ == nullptr) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (match_config_.jugador1_actiu && !p1_was_active) {
|
||||||
|
ship_animator_->triggerExitAnimationForPlayer(1);
|
||||||
|
std::cout << "[TitleScene] P1 " << log_prefix << "ship exiting\n";
|
||||||
|
}
|
||||||
|
if (match_config_.jugador2_actiu && !p2_was_active) {
|
||||||
|
ship_animator_->triggerExitAnimationForPlayer(2);
|
||||||
|
std::cout << "[TitleScene] P2 " << log_prefix << "ship exiting\n";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -618,7 +639,7 @@ void TitleScene::draw() {
|
|||||||
pos_shadow.x = posicions_originals_orni_[i].x + static_cast<int>(std::round(shadow_offset_x));
|
pos_shadow.x = posicions_originals_orni_[i].x + static_cast<int>(std::round(shadow_offset_x));
|
||||||
pos_shadow.y = posicions_originals_orni_[i].y + static_cast<int>(std::round(shadow_offset_y));
|
pos_shadow.y = posicions_originals_orni_[i].y + static_cast<int>(std::round(shadow_offset_y));
|
||||||
|
|
||||||
Rendering::render_shape(
|
Rendering::renderShape(
|
||||||
sdl_.getRenderer(),
|
sdl_.getRenderer(),
|
||||||
lletres_orni_[i].shape,
|
lletres_orni_[i].shape,
|
||||||
pos_shadow,
|
pos_shadow,
|
||||||
@@ -635,7 +656,7 @@ void TitleScene::draw() {
|
|||||||
pos_shadow.x = posicions_originals_attack_[i].x + static_cast<int>(std::round(shadow_offset_x));
|
pos_shadow.x = posicions_originals_attack_[i].x + static_cast<int>(std::round(shadow_offset_x));
|
||||||
pos_shadow.y = posicions_originals_attack_[i].y + static_cast<int>(std::round(shadow_offset_y));
|
pos_shadow.y = posicions_originals_attack_[i].y + static_cast<int>(std::round(shadow_offset_y));
|
||||||
|
|
||||||
Rendering::render_shape(
|
Rendering::renderShape(
|
||||||
sdl_.getRenderer(),
|
sdl_.getRenderer(),
|
||||||
lletres_attack_[i].shape,
|
lletres_attack_[i].shape,
|
||||||
pos_shadow,
|
pos_shadow,
|
||||||
@@ -650,7 +671,7 @@ void TitleScene::draw() {
|
|||||||
|
|
||||||
// Dibuixar "ORNI" (línia 1)
|
// Dibuixar "ORNI" (línia 1)
|
||||||
for (const auto& lletra : lletres_orni_) {
|
for (const auto& lletra : lletres_orni_) {
|
||||||
Rendering::render_shape(
|
Rendering::renderShape(
|
||||||
sdl_.getRenderer(),
|
sdl_.getRenderer(),
|
||||||
lletra.shape,
|
lletra.shape,
|
||||||
lletra.position,
|
lletra.position,
|
||||||
@@ -662,7 +683,7 @@ void TitleScene::draw() {
|
|||||||
|
|
||||||
// Dibuixar "ATTACK!" (línia 2)
|
// Dibuixar "ATTACK!" (línia 2)
|
||||||
for (const auto& lletra : lletres_attack_) {
|
for (const auto& lletra : lletres_attack_) {
|
||||||
Rendering::render_shape(
|
Rendering::renderShape(
|
||||||
sdl_.getRenderer(),
|
sdl_.getRenderer(),
|
||||||
lletra.shape,
|
lletra.shape,
|
||||||
lletra.position,
|
lletra.position,
|
||||||
|
|||||||
@@ -11,7 +11,6 @@
|
|||||||
#include <memory>
|
#include <memory>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
#include "core/defaults.hpp"
|
|
||||||
#include "core/graphics/shape.hpp"
|
#include "core/graphics/shape.hpp"
|
||||||
#include "core/graphics/starfield.hpp"
|
#include "core/graphics/starfield.hpp"
|
||||||
#include "core/graphics/vector_text.hpp"
|
#include "core/graphics/vector_text.hpp"
|
||||||
@@ -119,4 +118,18 @@ class TitleScene final : public Scene {
|
|||||||
void initTitle(); // Carrega i posiciona las lletres del título
|
void initTitle(); // Carrega i posiciona las lletres del título
|
||||||
void inicialitzarJailgames(); // Carrega i posiciona el logo JAILGAMES pequeño
|
void inicialitzarJailgames(); // Carrega i posiciona el logo JAILGAMES pequeño
|
||||||
void dibuixarPeuTitol(float spacing) const; // Logo JAILGAMES + línia de copyright
|
void dibuixarPeuTitol(float spacing) const; // Logo JAILGAMES + línia de copyright
|
||||||
|
|
||||||
|
// Sub-pasos de update() (extreure cada state per reduir complexitat).
|
||||||
|
void updateStarfieldFadeInState(float delta_time);
|
||||||
|
void updateStarfieldState(float delta_time);
|
||||||
|
void updateMainState(float delta_time);
|
||||||
|
void updatePlayerJoinPhaseState(float delta_time);
|
||||||
|
void updateBlackScreenState(float delta_time);
|
||||||
|
// Handlers de input globals (independents de l'state actual).
|
||||||
|
void handleSkipInput();
|
||||||
|
void handleStartInput();
|
||||||
|
// Helper compartit: dispara l'animación de salida per las naves del player que
|
||||||
|
// acaba de fer un join "en aquest frame" (jugadorX_actiu == true && !prev).
|
||||||
|
void triggerExitForJoinedPlayers(bool p1_was_active, bool p2_was_active,
|
||||||
|
const char* log_prefix);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
|
|
||||||
#include "spawn_controller.hpp"
|
#include "spawn_controller.hpp"
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
#include <array>
|
#include <array>
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
@@ -81,25 +82,18 @@ void SpawnController::update(float delta_time, std::array<Enemy, 15>& orni_array
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SpawnController::tots_enemics_spawnejats() const -> bool {
|
auto SpawnController::allEnemiesSpawned() const -> bool {
|
||||||
return index_spawn_actual_ >= spawn_queue_.size();
|
return index_spawn_actual_ >= spawn_queue_.size();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SpawnController::tots_enemics_destruits(const std::array<Enemy, 15>& orni_array) const -> bool {
|
auto SpawnController::allEnemiesDestroyed(const std::array<Enemy, 15>& orni_array) const -> bool {
|
||||||
if (!tots_enemics_spawnejats()) {
|
if (!allEnemiesSpawned()) {
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const auto& enemy : orni_array) {
|
|
||||||
if (enemy.isActive()) {
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
return std::ranges::all_of(orni_array, [](const Enemy& enemy) { return !enemy.isActive(); });
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
auto SpawnController::getAliveEnemyCount(const std::array<Enemy, 15>& orni_array) -> uint8_t {
|
||||||
}
|
|
||||||
|
|
||||||
auto SpawnController::get_enemics_vius(const std::array<Enemy, 15>& orni_array) -> uint8_t {
|
|
||||||
uint8_t count = 0;
|
uint8_t count = 0;
|
||||||
for (const auto& enemy : orni_array) {
|
for (const auto& enemy : orni_array) {
|
||||||
if (enemy.isActive()) {
|
if (enemy.isActive()) {
|
||||||
@@ -109,7 +103,7 @@ auto SpawnController::get_enemics_vius(const std::array<Enemy, 15>& orni_array)
|
|||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SpawnController::get_enemics_spawnejats() const -> uint8_t {
|
auto SpawnController::countSpawnedEnemies() const -> uint8_t {
|
||||||
return static_cast<uint8_t>(index_spawn_actual_);
|
return static_cast<uint8_t>(index_spawn_actual_);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -33,11 +33,11 @@ class SpawnController {
|
|||||||
void update(float delta_time, std::array<Enemy, 15>& orni_array, bool pausar = false);
|
void update(float delta_time, std::array<Enemy, 15>& orni_array, bool pausar = false);
|
||||||
|
|
||||||
// Status queries
|
// Status queries
|
||||||
[[nodiscard]] auto tots_enemics_spawnejats() const -> bool;
|
[[nodiscard]] auto allEnemiesSpawned() const -> bool;
|
||||||
[[nodiscard]] auto tots_enemics_destruits(const std::array<Enemy, 15>& orni_array) const -> bool;
|
[[nodiscard]] auto allEnemiesDestroyed(const std::array<Enemy, 15>& orni_array) const -> bool;
|
||||||
// Estático: solo recorre el array pasado; no consulta estado del controller.
|
// Estático: solo recorre el array pasado; no consulta estado del controller.
|
||||||
[[nodiscard]] static auto get_enemics_vius(const std::array<Enemy, 15>& orni_array) -> uint8_t;
|
[[nodiscard]] static auto getAliveEnemyCount(const std::array<Enemy, 15>& orni_array) -> uint8_t;
|
||||||
[[nodiscard]] auto get_enemics_spawnejats() const -> uint8_t;
|
[[nodiscard]] auto countSpawnedEnemies() const -> uint8_t;
|
||||||
|
|
||||||
// [NEW] Set ship position reference for safe spawn
|
// [NEW] Set ship position reference for safe spawn
|
||||||
void setShipPosition(const Vec2* ship_pos) { ship_position_ = ship_pos; }
|
void setShipPosition(const Vec2* ship_pos) { ship_position_ = ship_pos; }
|
||||||
|
|||||||
@@ -55,7 +55,7 @@ struct StageConfig {
|
|||||||
MultiplicadorsDificultat multiplicadors;
|
MultiplicadorsDificultat multiplicadors;
|
||||||
|
|
||||||
// Validació
|
// Validació
|
||||||
[[nodiscard]] auto es_valid() const -> bool {
|
[[nodiscard]] auto isValid() const -> bool {
|
||||||
// stage_id es uint8_t: el rango superior (<=255) está garantizado por
|
// stage_id es uint8_t: el rango superior (<=255) está garantizado por
|
||||||
// el tipo; basta con confirmar que no es 0 (sentinela "sin asignar").
|
// el tipo; basta con confirmar que no es 0 (sentinela "sin asignar").
|
||||||
return stage_id >= 1 &&
|
return stage_id >= 1 &&
|
||||||
@@ -70,7 +70,7 @@ struct StageSystemConfig {
|
|||||||
std::vector<StageConfig> stages; // Índex [0] = stage 1
|
std::vector<StageConfig> stages; // Índex [0] = stage 1
|
||||||
|
|
||||||
// Obtenir configuración de un stage específic
|
// Obtenir configuración de un stage específic
|
||||||
[[nodiscard]] auto obte_stage(uint8_t stage_id) const -> const StageConfig* {
|
[[nodiscard]] auto findStage(uint8_t stage_id) const -> const StageConfig* {
|
||||||
if (stage_id < 1 || stage_id > stages.size()) {
|
if (stage_id < 1 || stage_id > stages.size()) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -127,7 +127,7 @@ auto StageLoader::parseStage(const fkyaml::node& yaml, StageConfig& stage) -> bo
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!stage.es_valid()) {
|
if (!stage.isValid()) {
|
||||||
std::cerr << "[StageLoader] Error: stage " << static_cast<int>(stage.stage_id)
|
std::cerr << "[StageLoader] Error: stage " << static_cast<int>(stage.stage_id)
|
||||||
<< " no es vàlid" << '\n';
|
<< " no es vàlid" << '\n';
|
||||||
return false;
|
return false;
|
||||||
|
|||||||
@@ -51,20 +51,20 @@ void StageManager::update(float delta_time, bool pause_spawn) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void StageManager::stage_completat() {
|
void StageManager::markStageCompleted() {
|
||||||
std::cout << "[StageManager] Stage " << static_cast<int>(stage_actual_) << " completat!"
|
std::cout << "[StageManager] Stage " << static_cast<int>(stage_actual_) << " completat!"
|
||||||
<< '\n';
|
<< '\n';
|
||||||
changeState(EstatStage::LEVEL_COMPLETED);
|
changeState(EstatStage::LEVEL_COMPLETED);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto StageManager::tot_completat() const -> bool {
|
auto StageManager::isGameComplete() const -> bool {
|
||||||
return stage_actual_ >= config_->metadata.total_stages &&
|
return stage_actual_ >= config_->metadata.total_stages &&
|
||||||
estat_ == EstatStage::LEVEL_COMPLETED &&
|
estat_ == EstatStage::LEVEL_COMPLETED &&
|
||||||
timer_transicio_ <= 0.0F;
|
timer_transicio_ <= 0.0F;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto StageManager::get_config_actual() const -> const StageConfig* {
|
auto StageManager::getCurrentConfig() const -> const StageConfig* {
|
||||||
return config_->obte_stage(stage_actual_);
|
return config_->findStage(stage_actual_);
|
||||||
}
|
}
|
||||||
|
|
||||||
void StageManager::changeState(EstatStage nou_estat) {
|
void StageManager::changeState(EstatStage nou_estat) {
|
||||||
@@ -155,7 +155,7 @@ void StageManager::processLevelCompleted(float delta_time) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void StageManager::loadStage(uint8_t stage_id) {
|
void StageManager::loadStage(uint8_t stage_id) {
|
||||||
const StageConfig* stage_config = config_->obte_stage(stage_id);
|
const StageConfig* stage_config = config_->findStage(stage_id);
|
||||||
if (stage_config == nullptr) {
|
if (stage_config == nullptr) {
|
||||||
std::cerr << "[StageManager] Error: no es pot trobar stage " << static_cast<int>(stage_id)
|
std::cerr << "[StageManager] Error: no es pot trobar stage " << static_cast<int>(stage_id)
|
||||||
<< '\n';
|
<< '\n';
|
||||||
|
|||||||
@@ -28,15 +28,15 @@ class StageManager {
|
|||||||
void update(float delta_time, bool pause_spawn = false);
|
void update(float delta_time, bool pause_spawn = false);
|
||||||
|
|
||||||
// Stage progression
|
// Stage progression
|
||||||
void stage_completat(); // Call when all enemies destroyed
|
void markStageCompleted(); // Call when all enemies destroyed
|
||||||
[[nodiscard]] auto tot_completat() const -> bool; // All 10 stages done?
|
[[nodiscard]] auto isGameComplete() const -> bool; // All 10 stages done?
|
||||||
|
|
||||||
// Current state queries
|
// Current state queries
|
||||||
[[nodiscard]] auto get_estat() const -> EstatStage { return estat_; }
|
[[nodiscard]] auto getState() const -> EstatStage { return estat_; }
|
||||||
[[nodiscard]] auto get_stage_actual() const -> uint8_t { return stage_actual_; }
|
[[nodiscard]] auto getCurrentStage() const -> uint8_t { return stage_actual_; }
|
||||||
[[nodiscard]] auto get_config_actual() const -> const StageConfig*;
|
[[nodiscard]] auto getCurrentConfig() const -> const StageConfig*;
|
||||||
[[nodiscard]] auto get_timer_transicio() const -> float { return timer_transicio_; }
|
[[nodiscard]] auto getTransitionTimer() const -> float { return timer_transicio_; }
|
||||||
[[nodiscard]] auto get_missatge_level_start() const -> const std::string& { return missatge_level_start_actual_; }
|
[[nodiscard]] auto getLevelStartMessage() const -> const std::string& { return missatge_level_start_actual_; }
|
||||||
|
|
||||||
// Spawn control (delegate to SpawnController)
|
// Spawn control (delegate to SpawnController)
|
||||||
auto getSpawnController() -> SpawnController& { return spawn_controller_; }
|
auto getSpawnController() -> SpawnController& { return spawn_controller_; }
|
||||||
|
|||||||
@@ -16,7 +16,7 @@ void detectBulletEnemy(Context& ctx) {
|
|||||||
|
|
||||||
for (auto& bullet : ctx.bullets) {
|
for (auto& bullet : ctx.bullets) {
|
||||||
for (auto& enemy : ctx.enemies) {
|
for (auto& enemy : ctx.enemies) {
|
||||||
if (!Physics::check_collision(bullet, enemy, AMPLIFIER)) {
|
if (!Physics::checkCollision(bullet, enemy, AMPLIFIER)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -86,7 +86,7 @@ void detectShipEnemy(Context& ctx) {
|
|||||||
if (enemy.isInvulnerable()) {
|
if (enemy.isInvulnerable()) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
if (Physics::check_collision(ctx.ships[i], enemy, AMPLIFIER)) {
|
if (Physics::checkCollision(ctx.ships[i], enemy, AMPLIFIER)) {
|
||||||
ctx.on_player_hit(i);
|
ctx.on_player_hit(i);
|
||||||
break; // Solo una colisión por player por frame
|
break; // Solo una colisión por player por frame
|
||||||
}
|
}
|
||||||
@@ -102,7 +102,7 @@ void detectBulletPlayer(Context& ctx) {
|
|||||||
constexpr float AMPLIFIER = Defaults::Game::COLLISION_BULLET_PLAYER_AMPLIFIER;
|
constexpr float AMPLIFIER = Defaults::Game::COLLISION_BULLET_PLAYER_AMPLIFIER;
|
||||||
|
|
||||||
for (auto& bullet : ctx.bullets) {
|
for (auto& bullet : ctx.bullets) {
|
||||||
if (!bullet.esta_activa() || bullet.getGraceTimer() > 0.0F) {
|
if (!bullet.isActive() || bullet.getGraceTimer() > 0.0F) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
const uint8_t BULLET_OWNER = bullet.getOwnerId();
|
const uint8_t BULLET_OWNER = bullet.getOwnerId();
|
||||||
@@ -120,7 +120,7 @@ void detectBulletPlayer(Context& ctx) {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!Physics::check_collision(bullet, ctx.ships[player_id], AMPLIFIER)) {
|
if (!Physics::checkCollision(bullet, ctx.ships[player_id], AMPLIFIER)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,7 +5,6 @@
|
|||||||
#include <SDL3/SDL.h>
|
#include <SDL3/SDL.h>
|
||||||
|
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <cstdint>
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
#include "core/defaults.hpp"
|
#include "core/defaults.hpp"
|
||||||
@@ -30,7 +29,7 @@ auto computeRangeProgress(float global_progress,
|
|||||||
}
|
}
|
||||||
|
|
||||||
auto computeShipPosition(float progress, const Vec2& final_position) -> Vec2 {
|
auto computeShipPosition(float progress, const Vec2& final_position) -> Vec2 {
|
||||||
const float EASED = Easing::ease_out_quad(progress);
|
const float EASED = Easing::easeOutQuad(progress);
|
||||||
const SDL_FRect& zone = Defaults::Zones::PLAYAREA;
|
const SDL_FRect& zone = Defaults::Zones::PLAYAREA;
|
||||||
// Y inicial: 50 px bajo la zona de juego.
|
// Y inicial: 50 px bajo la zona de juego.
|
||||||
const float Y_INI = zone.y + zone.h + 50.0F;
|
const float Y_INI = zone.y + zone.h + 50.0F;
|
||||||
@@ -40,7 +39,7 @@ auto computeShipPosition(float progress, const Vec2& final_position) -> Vec2 {
|
|||||||
|
|
||||||
void drawBordersAnimated(Rendering::Renderer* renderer, float progress) {
|
void drawBordersAnimated(Rendering::Renderer* renderer, float progress) {
|
||||||
const SDL_FRect& zone = Defaults::Zones::PLAYAREA;
|
const SDL_FRect& zone = Defaults::Zones::PLAYAREA;
|
||||||
const float EASED = Easing::ease_out_quad(progress);
|
const float EASED = Easing::easeOutQuad(progress);
|
||||||
|
|
||||||
const int X1 = static_cast<int>(zone.x);
|
const int X1 = static_cast<int>(zone.x);
|
||||||
const int Y1 = static_cast<int>(zone.y);
|
const int Y1 = static_cast<int>(zone.y);
|
||||||
@@ -81,7 +80,7 @@ void drawBordersAnimated(Rendering::Renderer* renderer, float progress) {
|
|||||||
void drawScoreboardAnimated(const Graphics::VectorText& text,
|
void drawScoreboardAnimated(const Graphics::VectorText& text,
|
||||||
const std::string& scoreboard_text,
|
const std::string& scoreboard_text,
|
||||||
float progress) {
|
float progress) {
|
||||||
const float EASED = Easing::ease_out_quad(progress);
|
const float EASED = Easing::easeOutQuad(progress);
|
||||||
|
|
||||||
constexpr float SCALE = 0.85F;
|
constexpr float SCALE = 0.85F;
|
||||||
constexpr float SPACING = 0.0F;
|
constexpr float SPACING = 0.0F;
|
||||||
|
|||||||
@@ -61,7 +61,7 @@ void ShipAnimator::draw() const {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Renderizar ship (perspectiva ya incorporada a la shape)
|
// Renderizar ship (perspectiva ya incorporada a la shape)
|
||||||
Rendering::render_shape(
|
Rendering::renderShape(
|
||||||
renderer_,
|
renderer_,
|
||||||
ship.shape,
|
ship.shape,
|
||||||
ship.current_position,
|
ship.current_position,
|
||||||
@@ -73,7 +73,7 @@ void ShipAnimator::draw() const {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ShipAnimator::start_entry_animation() {
|
void ShipAnimator::startEntryAnimation() {
|
||||||
using namespace Defaults::Title::Ships;
|
using namespace Defaults::Title::Ships;
|
||||||
|
|
||||||
// Configurar ship P1 para l'animación de entrada
|
// Configurar ship P1 para l'animación de entrada
|
||||||
@@ -91,7 +91,7 @@ void ShipAnimator::start_entry_animation() {
|
|||||||
ships_[1].current_scale = ships_[1].initial_scale;
|
ships_[1].current_scale = ships_[1].initial_scale;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ShipAnimator::trigger_exit_animation() {
|
void ShipAnimator::triggerExitAnimation() {
|
||||||
// Configurar ambdues naves para l'animación de salida
|
// Configurar ambdues naves para l'animación de salida
|
||||||
for (auto& ship : ships_) {
|
for (auto& ship : ships_) {
|
||||||
// Canviar state a EXITING
|
// Canviar state a EXITING
|
||||||
@@ -106,7 +106,7 @@ void ShipAnimator::trigger_exit_animation() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ShipAnimator::skip_to_floating_state() {
|
void ShipAnimator::skipToFloatingState() {
|
||||||
// Posar ambdues naves directament en state FLOATING
|
// Posar ambdues naves directament en state FLOATING
|
||||||
for (auto& ship : ships_) {
|
for (auto& ship : ships_) {
|
||||||
ship.state = ShipState::FLOATING;
|
ship.state = ShipState::FLOATING;
|
||||||
@@ -122,17 +122,12 @@ void ShipAnimator::skip_to_floating_state() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto ShipAnimator::is_visible() const -> bool {
|
auto ShipAnimator::isVisible() const -> bool {
|
||||||
// Retorna true si almenys una ship es visible
|
// Retorna true si almenys una ship es visible
|
||||||
for (const auto& ship : ships_) {
|
return std::ranges::any_of(ships_, [](const TitleShip& ship) { return ship.visible; });
|
||||||
if (ship.visible) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ShipAnimator::trigger_exit_animation_for_player(int player_id) {
|
void ShipAnimator::triggerExitAnimationForPlayer(int player_id) {
|
||||||
// Trobar la ship del player especificat
|
// Trobar la ship del player especificat
|
||||||
for (auto& ship : ships_) {
|
for (auto& ship : ships_) {
|
||||||
if (ship.player_id == player_id) {
|
if (ship.player_id == player_id) {
|
||||||
@@ -150,20 +145,15 @@ void ShipAnimator::trigger_exit_animation_for_player(int player_id) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ShipAnimator::set_visible(bool visible) {
|
void ShipAnimator::setVisible(bool visible) {
|
||||||
for (auto& ship : ships_) {
|
for (auto& ship : ships_) {
|
||||||
ship.visible = visible;
|
ship.visible = visible;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto ShipAnimator::is_animation_complete() const -> bool {
|
auto ShipAnimator::isAnimationComplete() const -> bool {
|
||||||
// Comprovar si todas las naves són invisibles (han completat l'animación de salida)
|
// Comprovar si todas las naves són invisibles (han completat l'animación de salida)
|
||||||
for (const auto& ship : ships_) {
|
return std::ranges::all_of(ships_, [](const TitleShip& ship) { return !ship.visible; });
|
||||||
if (ship.visible) {
|
|
||||||
return false; // Aún hay alguna ship visible
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true; // Todas las naves són invisibles
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Métodos de animación (stubs)
|
// Métodos de animación (stubs)
|
||||||
@@ -185,7 +175,7 @@ void ShipAnimator::updateEntering(TitleShip& ship, float delta_time) {
|
|||||||
float progress = std::min(1.0F, elapsed / ENTRY_DURATION);
|
float progress = std::min(1.0F, elapsed / ENTRY_DURATION);
|
||||||
|
|
||||||
// Aplicar easing (ease_out_quad per arribada suau)
|
// Aplicar easing (ease_out_quad per arribada suau)
|
||||||
float eased_progress = Easing::ease_out_quad(progress);
|
float eased_progress = Easing::easeOutQuad(progress);
|
||||||
|
|
||||||
// Lerp posición (inicial → objetivo)
|
// Lerp posición (inicial → objetivo)
|
||||||
ship.current_position.x = Easing::lerp(ship.initial_position.x, ship.target_position.x, eased_progress);
|
ship.current_position.x = Easing::lerp(ship.initial_position.x, ship.target_position.x, eased_progress);
|
||||||
@@ -230,7 +220,7 @@ void ShipAnimator::updateExiting(TitleShip& ship, float delta_time) {
|
|||||||
float progress = std::min(1.0F, ship.state_time / EXIT_DURATION);
|
float progress = std::min(1.0F, ship.state_time / EXIT_DURATION);
|
||||||
|
|
||||||
// Aplicar easing (ease_in_quad per aceleración hacia el point de fuga)
|
// Aplicar easing (ease_in_quad per aceleración hacia el point de fuga)
|
||||||
float eased_progress = Easing::ease_in_quad(progress);
|
float eased_progress = Easing::easeInQuad(progress);
|
||||||
|
|
||||||
// Vec2 de fuga (centro del starfield)
|
// Vec2 de fuga (centro del starfield)
|
||||||
constexpr Vec2 VANISHING_POINT{.x = VANISHING_POINT_X, .y = VANISHING_POINT_Y};
|
constexpr Vec2 VANISHING_POINT{.x = VANISHING_POINT_X, .y = VANISHING_POINT_Y};
|
||||||
@@ -258,7 +248,7 @@ void ShipAnimator::configureShipP1(TitleShip& ship) {
|
|||||||
ship.state_time = 0.0F;
|
ship.state_time = 0.0F;
|
||||||
|
|
||||||
// Posicions (clock 8, bottom-left)
|
// Posicions (clock 8, bottom-left)
|
||||||
ship.target_position = {.x = P1_TARGET_X(), .y = P1_TARGET_Y()};
|
ship.target_position = {.x = p1TargetX(), .y = p1TargetY()};
|
||||||
|
|
||||||
// Calcular posición inicial (fuera de pantalla)
|
// Calcular posición inicial (fuera de pantalla)
|
||||||
ship.initial_position = computeOffscreenPosition(CLOCK_8_ANGLE);
|
ship.initial_position = computeOffscreenPosition(CLOCK_8_ANGLE);
|
||||||
@@ -293,7 +283,7 @@ void ShipAnimator::configureShipP2(TitleShip& ship) {
|
|||||||
ship.state_time = 0.0F;
|
ship.state_time = 0.0F;
|
||||||
|
|
||||||
// Posicions (clock 4, bottom-right)
|
// Posicions (clock 4, bottom-right)
|
||||||
ship.target_position = {.x = P2_TARGET_X(), .y = P2_TARGET_Y()};
|
ship.target_position = {.x = p2TargetX(), .y = p2TargetY()};
|
||||||
|
|
||||||
// Calcular posición inicial (fuera de pantalla)
|
// Calcular posición inicial (fuera de pantalla)
|
||||||
ship.initial_position = computeOffscreenPosition(CLOCK_4_ANGLE);
|
ship.initial_position = computeOffscreenPosition(CLOCK_4_ANGLE);
|
||||||
|
|||||||
@@ -75,15 +75,15 @@ class ShipAnimator {
|
|||||||
void draw() const;
|
void draw() const;
|
||||||
|
|
||||||
// Control de state (cridat per TitleScene)
|
// Control de state (cridat per TitleScene)
|
||||||
void start_entry_animation();
|
void startEntryAnimation();
|
||||||
void trigger_exit_animation(); // Anima todas las naves
|
void triggerExitAnimation(); // Anima todas las naves
|
||||||
void trigger_exit_animation_for_player(int player_id); // Anima solo una ship (P1=1, P2=2)
|
void triggerExitAnimationForPlayer(int player_id); // Anima solo una ship (P1=1, P2=2)
|
||||||
void skip_to_floating_state(); // Salta directament a FLOATING sin animación
|
void skipToFloatingState(); // Salta directament a FLOATING sin animación
|
||||||
|
|
||||||
// Control de visibilitat
|
// Control de visibilitat
|
||||||
void set_visible(bool visible);
|
void setVisible(bool visible);
|
||||||
[[nodiscard]] auto is_animation_complete() const -> bool;
|
[[nodiscard]] auto isAnimationComplete() const -> bool;
|
||||||
[[nodiscard]] auto is_visible() const -> bool; // Comprova si alguna ship es visible
|
[[nodiscard]] auto isVisible() const -> bool; // Comprova si alguna ship es visible
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Rendering::Renderer* renderer_;
|
Rendering::Renderer* renderer_;
|
||||||
|
|||||||
+1
-1
@@ -14,5 +14,5 @@ auto main(int argc, char* argv[]) -> int {
|
|||||||
Director director(args);
|
Director director(args);
|
||||||
|
|
||||||
// Executar bucle principal del juego
|
// Executar bucle principal del juego
|
||||||
return director.run();
|
return Director::run();
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user