Mi primer videojuego C++ y SFML
Hasta los momentos hemos construido gran parte de nuestro videojuego.
Básicamente lo hemos realizado en varios pasos (Parte I + Parte II) y que vamos a resumir a continuación:
Parte I
1) Creamos una ventana de dibujo.
2) Dibujamos una figura en pantalla (Circulo)
3) Movemos el circulo en la pantalla cambiando sus coordenadas con circulo.setPosition(400, 300)
4) Mover el circulo repetidamente usando el bucle while de la ventana creada y utilizando variables para las coordenadas del circulo circulo.setPosition(x, y) incrementando los valores de x e y.
4.1) Establecer limites al movimiento del circulo con la condición if.
5) Mover el circulo con una variable de incremento que cambia de sentido con el signo (+ o -) y utilizar la condición if para establecer los limites superior, inferior, izquierdo y derecho.
6) Dibujar un rectángulo en la parte inferior.
7) Mover el rectángulo con el teclado (Usando las teclas A y D)
8) Eliminar el limite inferior y establecer el rectángulo como el limite manejando la colisión entre el circulo y el rectángulo.
Parte II
9) Dibujar los cuadrado en la parte superior manualmente.
10) Dibujar cuadrado en la parte superior aplicando matemáticas y utilizando arrays.
Para simplificar he eliminado los limites laterales (derecho e izquierdo) de la barra inferior y hemos dejado el color por defecto de las figuras,
es importante concentrarnos mas en la mecánica del juego y mas adelante ocuparnos del diseño y la apariencia aspectos también muy importantes.
Código fuente Parte I y Parte II
El resultado del código fuente desarrollado en la parte I y II debe darte como resultado algo parecido a la siguiente imagen. En en código establecimos que para mover la barra inferior que hace rebotar a la pelota se debe utiliza las teclas A (izquierda) y D (derecha).
Parte III Destruir los cuadrados cuando es tocado por el circulo.
Como vemos la pelota representada por un circulo solo rebota en lo limites de la ventana y no con cada cuadrado y en la parte inferior, sólo cuando colisiona con la barra controlada por el jugador. Hasta ahora no hemos manejado la colisión con cada ladrillo que son representado con un cuadrado.
Para lograr que nuestra pelota destruya cada ladrillo debemos hacer algo similar a lo realizado con la barra y el circulo; debemos crear una caja de bordes para cada cuadrado, es decir para cada uno de los cuadrados almacenados en cuadrado[i] donde i representa el índice de cada cuadrado que va desde 0 hasta n.
Ahora además del array de cuadrado que creamos con la siguiente sentencia:
sf::RectangleShape cuadrados[n];// array de cuadrados
debemos crear un array de cajas de bordes para guardar el área de cada cuadrado
sf::FloatRect cuadradosBox[n];// array de caja de colisiones
para aprovecha el bloque del dibujo de cada cuadrado en la parte del código que dice //aquí dibujamos vamos a guardar el borde de cada cuadrado[i] en cuadradoBox[i]
//Aquí dibujamos
for(int i = 0; i < n; ++i){
xs = i * (ls + ls/4 );
cuadrados[i].setPosition(xs,ys);
cuadradosBox[i] = cuadrados[i].getGlobalBounds();
ventana.draw(cuadrados[i]);
}
y finalmente similar a la detección de la colisión del circulo con el rectángulo usamos una condición if para detectar la colisión del circulo con cada cuadrado recorriendo todo el array con la sentencia de control for.
//colisión del círculo con cada cuadrado
for (int i = 0; i < n; ++i){
if (circuloBox.intersects(cuadradosBox[i])){
incre_yc = -incre_yc;// incremento de la variable "y" del circulo
}
}
agregando los fragmentos de código en colo azul al código completo desarrollado en la parte I y II obtendremos un resultado igual al que muestra la siguiente imagen.
se puede observar a diferencia de la imagen anterior, el circulo al tocar un cuadrado rebota, y ese es el resultado que queremos por lo momentos. El siguiente paso es agregar las sentencias necesarias para destruir los cuadrados que toque el circulo. Daremos todo el detalle necesarios y el código fuente completo.
En este enlace encontraras el código fuente para obtener el resultado de la imagen anterior.
Ya teniendo el control de la colisión del circulo con cada cuadrado el siguiente paso es destruir el cuadrado. Para ello vamos a darle valor a cada cuadrado el valor uno "1" cuando esta activo y cero "0" cuando esta destruido. Es decir cada cuadrado va a tener dos estados "0" (vivo) o "1" (muerto). Para almacenar el estado de cada cuadrado utilizamos un arreglo (array) de estado de cuadrados y lo declaramos de la siguiente manera:
// cuadrados
int estadocuadrado[n]={};
y dentro de bucle for que usamos para crear y almacenar cada cuadrado al mismo tiempo se inicializa su valor en 1.
for(int i = 0; i < n; ++i){
cuadrados[i] = sf::RectangleShape (sf::Vector2f(ls, ls));
estadocuadrado[i] = 1;
}
Esto quiere decir que todos los cuadrados inician todos prendidos. Cuando el circulo toca por primera vez a un cuadrado es decir hay una intersección entre sus coordenadas y su estado es 1 e inmediatamente debe pasar a estado 0.
Si el circulo vuelve a pasar por donde estaba un cuadrado que está en estado 0 no debe haber colisión y debe rebota en el borde superior de la ventana. Debemos asegurar que para que pueda haber una colisión debe cumplirse dos condiciones: la primera es que debe haber una intersección y la segunda es que el estado del cuadrado debe ser igual a 1. Traducido en lenguaje C++ es:
//manejo de colisiones del circulo con los cuadrados
for (int i = 0; i < n; ++i){
if (circuloBox.intersects(cuadradosBox[i]) && estadocuadrado[i] == 1){
incre_yc = -incre_yc;
estadocuadrado[i] = 0;
}
}
y si un cuadrado esta en estado = 0, debemos asegurarnos de no dibujarlo en pantalla
//Aqui dibujamos
for(int i = 0; i < n; ++i){
xs = i * (ls + ls/4 );
cuadrados[i].setPosition(xs,ys);
if (estadocuadrado[i] == 1){
ventana.draw(cuadrados[i]);
}
}
A continuacion el codigo fuente completo:
Comentarios
Publicar un comentario