Problema: - Radio fijo de 80px funcionaba bien en 320x240 - En F4 fullscreen (1920x1080), radio de 360px era correcto visualmente - PERO las fuerzas físicas (spring_k, damping) seguían siendo para 80px - Resultado: pelotas nunca llegaban a pegarse en resoluciones altas Solución: 1. Radio proporcional a altura de pantalla (ROTOBALL_RADIUS_FACTOR = 0.333) 2. Escalar TODAS las constantes de física proporcionalmente al radio 3. Fórmula: scale = sphere_radius / BASE_RADIUS (80px) Cambios técnicos: - defines.h: ROTOBALL_RADIUS → ROTOBALL_RADIUS_FACTOR (0.333) - engine.cpp: Calcular radius dinámicamente en generate/update - ball.h: applyRotoBallForce() ahora recibe sphere_radius - ball.cpp: Escalar spring_k, damping_base, damping_near, near_threshold, max_force Resultado: - 320x240: Radio 80px, scale=1.0 (idéntico a antes) - 640x480: Radio 160px, scale=2.0 (fuerzas 2x) - 1920x1080: Radio 360px, scale=4.5 (fuerzas 4.5x) Comportamiento físico IDÉNTICO en todas las resoluciones ✅ 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
89 lines
3.7 KiB
C++
89 lines
3.7 KiB
C++
#pragma once
|
|
|
|
#include <SDL3/SDL_rect.h> // for SDL_FRect
|
|
|
|
#include <memory> // for shared_ptr, unique_ptr
|
|
|
|
#include "defines.h" // for Color
|
|
#include "external/sprite.h" // for Sprite
|
|
class Texture;
|
|
|
|
class Ball {
|
|
private:
|
|
std::unique_ptr<Sprite> sprite_; // Sprite para pintar la clase
|
|
SDL_FRect pos_; // Posición y tamaño de la pelota
|
|
float vx_, vy_; // Velocidad
|
|
float gravity_force_; // Gravedad base
|
|
float gravity_mass_factor_; // Factor de masa individual (0.7-1.3, afecta gravedad)
|
|
GravityDirection gravity_direction_; // Direcci\u00f3n de la gravedad
|
|
int screen_width_; // Ancho del terreno de juego
|
|
int screen_height_; // Alto del terreno de juego
|
|
Color color_; // Color de la pelota
|
|
bool on_surface_; // Indica si la pelota est\u00e1 en la superficie (suelo/techo/pared)
|
|
bool stopped_; // Indica si la pelota ha terminado de moverse;
|
|
float loss_; // Coeficiente de rebote. Pérdida de energía en cada rebote
|
|
|
|
// Datos para modo RotoBall (esfera 3D)
|
|
float pos_3d_x_, pos_3d_y_, pos_3d_z_; // Posición 3D en la esfera
|
|
float target_x_, target_y_; // Posición destino 2D (proyección)
|
|
float depth_brightness_; // Brillo según profundidad Z (0.0-1.0)
|
|
bool rotoball_attraction_active_; // ¿Está siendo atraída hacia la esfera?
|
|
|
|
public:
|
|
// Constructor
|
|
Ball(float x, float vx, float vy, Color color, std::shared_ptr<Texture> texture, int screen_width, int screen_height, GravityDirection gravity_dir = GravityDirection::DOWN, float mass_factor = 1.0f);
|
|
|
|
// Destructor
|
|
~Ball() = default;
|
|
|
|
// Actualiza la lógica de la clase
|
|
void update(float deltaTime);
|
|
|
|
// Pinta la clase
|
|
void render();
|
|
|
|
// Modifica la velocidad
|
|
void modVel(float vx, float vy);
|
|
|
|
// Cambia la gravedad
|
|
void switchGravity();
|
|
|
|
// Reactiva la gravedad si está desactivada
|
|
void enableGravityIfDisabled();
|
|
|
|
// Fuerza gravedad ON (siempre activa)
|
|
void forceGravityOn();
|
|
|
|
// Fuerza gravedad OFF (siempre desactiva)
|
|
void forceGravityOff();
|
|
|
|
// Cambia la direcci\u00f3n de gravedad
|
|
void setGravityDirection(GravityDirection direction);
|
|
|
|
// Aplica un peque\u00f1o empuje lateral aleatorio
|
|
void applyRandomLateralPush();
|
|
|
|
// Getters para debug
|
|
float getVelocityY() const { return vy_; }
|
|
float getVelocityX() const { return vx_; }
|
|
float getGravityForce() const { return gravity_force_; }
|
|
float getLossCoefficient() const { return loss_; }
|
|
GravityDirection getGravityDirection() const { return gravity_direction_; }
|
|
bool isOnSurface() const { return on_surface_; }
|
|
bool isStopped() const { return stopped_; }
|
|
|
|
// Getters para batch rendering
|
|
SDL_FRect getPosition() const { return pos_; }
|
|
Color getColor() const { return color_; }
|
|
|
|
// Funciones para modo RotoBall
|
|
void setRotoBallPosition3D(float x, float y, float z);
|
|
void setRotoBallTarget2D(float x, float y);
|
|
void setRotoBallScreenPosition(float x, float y); // Establecer posición directa en pantalla
|
|
void setDepthBrightness(float brightness);
|
|
float getDepthBrightness() const { return depth_brightness_; }
|
|
|
|
// Sistema de atracción física hacia esfera RotoBall
|
|
void enableRotoBallAttraction(bool enable);
|
|
void applyRotoBallForce(float target_x, float target_y, float sphere_radius, float deltaTime);
|
|
}; |