Ajouter des fonctionnalités à notre exercice des balles rebondissantes
Dans cet exercice, vous devrez utiliser le jeu des balles rebondissantes de l'article précédent comme base, pour y ajouter de nouvelles fonctionnalitées intéressantes.
Prérequis: | Avant de vous lancer dans cet exercice, il est fortement conseillé d'avoir vu et compris tous les précédents articles de ce module. |
---|---|
Objectifs: | Tester votre connaissance du Javascript orienté objet en conception et en pratique. |
Pour commencer
Pour commencer, faite une copie locale de index-finished.html, style.css, et main-finished.js de l'article précédent, dans un nouveau dossier.
Vous pouvez utiliser un site comme CodePen, JSFiddle ou Glitch. Vous pouvez copier vos codes HTML, CSS et JavaScript dans l'un d'entre eux. Si celui que vous utilisez ne possède pas de fenêtres séparées pour les différents langages, ajoutez les dans des balises <script>
/<style>
dans votre code HTML.
Le projet en bref
Notre jeu des balles est assez sympa, mais maintenant il s'agit de le rendre plus interactif en y ajoutant un viseur contrôlé par l'utilisateur, qui va détruire une balle s'il la touche. Nous voulons aussi tester votre capacité en programmation orientée objet en créant un object Shape()
dont le viseur et les balles peuvent hériter. Pour terminer, nous voulons créer un compteur qui permet d'afficher combien de balles il nous reste encore à détruire.
Ce screenshot vous donne une idée du résultat final :
Si vous voulez en savoir plus, regardez l'exemple fini (n'en profitez pas pour récupérer le code source !).
Vos objectifs
Cette section décrit ce que vous aurez à faire.
Créons nos nouveaux objets
Pour commencer, modifions le constructeur de l'objet Ball()
pour qu'il devienne le constructeur de Shape()
puis créons en un nouveau pour Ball()
:
- Le constructeur
Shape()
devra définir les propriétésx
,y
,velX
, etvelY
de la même manière que le constructeurBall()
auparavant, mais sans les propriétéscolor
etsize
. Shape()
doit aussi définir une nouvelle propriétéexists
, qui servira à identifier les balles qu'il reste à détruire dans la fenêtre (celles qui n'ont pas encore été détruites). Elle doit retourner un booléen (true
/false
).- Le constructeur
Ball()
doit hériter des propriétésx
,y
,velX
,velY
, etexists
du constructeurShape()
. Ball()
doit aussi définir les propriétéscolor
etsize
, comme à l'origine.- N'oubliez pas de définir le prototype de
Ball()
et son constructeur de manière approprié.
Les méthodes draw()
, update()
, et collisionDetect()
doivent fonctionnées comme avant, sans être modifiées.
Vous devrez ajouter un nouveau paramètre au constructeur new Ball() ( ... )
— le paramètre exists
doit être le 5ème et être égal à true
.
Vous pouvez recharger la page — tout doit fonctionner comme avant, même après les modifications que vous avez effectuées sur les objets.
Définition du EvilCircle() (viseur)
Il est temps de vous équipez ! — le EvilCircle()
! Dans notre jeu, nous allons créer un viseur mais nous allons nous servir de l'objet Shape()
pour le définir. Vous voudrez certainement en ajouter un (plusieurs) autre plus tard, qu'un autre joueur ou l'ordinateur pourra contrôler. Vous n'irez probablement pas bien loin avec un seul viseur, mais ce sera suffisant pour le moment !
Le constructeur du EvilCircle()
doit hériter des propriétés x
, y
, velX
, velY
, et exists
de Shape()
, mais velX
et velY
doivent toujours être égales à 20.
Vous devriez utiliser quelque chose comme Shape.call(this, x, y, 20, 20, exists);
Le constructeur doit aussi définir ses propres propriétés:
color
—'white'
size
—10
Une fois de plus, souvenez-vous de définir vos propriétés héritées en paramètre du constructeur et de définir le prototype et son constructeur de manière appropriée.
Définir les méthodes du EvilCircle() (viseur)
EvilCircle()
doit avoir quatre méthodes, comme définie en dessous.
draw()
Cette méthode doit avoir la même fonction que celle de Ball()
: soit dessiner l'objet dans le canvas. Elle fonctionnera quasiment de la même manière, copiez la fonction Ball.prototype.draw
. Puis appliquez les modifications suivantes:
- On ne veut pas que le viseur soit plein, mais qu'il ait seulement un contour. Changez
fillStyle
etfill()
pourstrokeStyle
etstroke()
. - On voudrait qu'il soit aussi un peu plus épais, pour être plus facile à voir. Pour ça on doit définir un attribut
lineWidth
à ctx après l'appel à la fonctionbeginPath()
(avec une valeur de 3).
checkBounds()
Cette méthode à la même fonction que la première partie de Ball()
update()
— Savoir si le viseur va hors de l'écran, et l'arrêter si besoin. Une fois encore, copié la méthode Ball.prototype.update
, mais en effectuant quelques changements:
- Débarrassez-vous des deux dernières lignes — on a pas besoin de connaître la position du viseur à chaque frame, car nous le déplacerons d'une manière différente comme vous pourrez le voir.
- Dans les conditions en
if()
, si la condition retourne true on ne veut pas modifier (update) les propriétésvelX
/velY
; mais plutôt changer les valeurs dex
/y
de manière à ce que le viseur revienne doucement dans l'écran. Ajouter ou soustraire de manière appropriée la taille (size)
du viseur sera suffisant.
setControls()
Cette méthode ajoute un écouteur d'évènement onkeydown
à l'objet window
ce qui permettra en enfonçant certaine touche du clavier de déplacer le viseur dans la fenêtre. Insérez le code suivant dans la méthode :
var _this = this;
window.onkeydown = function (e) {
if (e.keyCode === 65) {
_this.x -= _this.velX;
} else if (e.keyCode === 68) {
_this.x += _this.velX;
} else if (e.keyCode === 87) {
_this.y -= _this.velY;
} else if (e.keyCode === 83) {
_this.y += _this.velY;
}
};
Quand une touche est enfoncée, la propriété keyCode de l'objet event est consultée pour savoir quelle touche est enfoncée. Si c'est une des touches spécifiée, alors le viseur ce déplacera à gauche, à droite, en haut ou en bas.
- Pour un point bonus, faite apparaître à quel touche correspond le code de celle que l'utilisateur a enfoncé.
- Pour un second point bonus, pouvez vous nous dire pourquoi nous devons définir
var _this = this;
de cette façon ? Cela à quelque chose à voir avec la portée des fonction.
collisionDetect()
Cette méthode fonctionne d'une manière similaire à Ball()
collisionDetect()
, copier celle-ci pour vous en servir comme base. Il y a deux différences:
- Dans la condition extérieure
if
, nous n'avons plus besoin de vérifier si la balle actuellement dans la boucle est celle actuellement surveiller — Parce que ce n'est plus une balle, mais notre viseur ! A la place, on doit tester si la balle visée existe (avec quelle propriété pourrez vous faire cela?). Si elle n'existe pas, c'est qu'elle a déjà été détruite, on a donc pas besoin de la vérifier encore une fois. - Dans la condition intérieur
if
, on ne souhaite plus changer un élément de couleur lorsqu'une collision est détéctée — A la place, on veut détruire les balles qui entre en collision avec le viseur (encore une fois, comment pensez-vous faire cela ?).
Insérer le viseur dans notre programme
Maintenant que nous avons définit notre viseur, on a besoin de le faire apparaître à l'écran. Pour ce faire on doit appliquer quelques modifications à la fonction loop()
.
- Premièrement, créons une nouvelle instance de l'objet viseur (en spécifiant les paramètres nécessaire), et appelons sa méthode
setControls()
. On doit seulement effectuer ses deux actions une seule fois, pas à chaque itération. - Au moment où l'on boucle à travers toutes les balles et que l'on appelle les méthodes
draw()
,update()
, etcollisionDetect()
pour chacune d'entre elle, faite de manière à ce que ces fonctions soit appelées seulement si la balle existe. - Appellez les méthodes de l'instance du viseur
draw()
,checkBounds()
, etcollisionDetect()
à chaque itération de la boucle.
Implémenter le compteur de score
Pour implémenter le compteur de score, suivez les étapes suivantes:
- Dans votre fichier HTML, ajoutez un élement
<p>
qui contiendra le texte suivant "Ball count: ", juste en dessous de l'élément<h1>
. - Dans votre fichier CSS, ajouter les règlesz suivantes:
css
p { position: absolute; margin: 0; top: 35px; right: 5px; color: #aaa; }
- Dans votre JavaScript, effectuez les modifications suivante :
- Créez une variable qui contiendra la référence vers le paragraphe.
- Stocker et afficher le nombre de balle présentent à l'écran.
- Incrémentez le compteur de balle à chaque fois qu'une balle apparait à l'écran.
- Décrementez le compteur à chaque fois qu'une balle est détruite par le viseur.
Conseils et astuces
- Cet exercice est un bon challenge. Prenez le temps de faire et de comprendre chaque étape.
- Ce serait une bonne idée de garder une copie de chaque étape lorsque vous arrivez à la faire marcher correctement, pour vous y réferrer si vous n'arrivez plus à progresser ensuite.
Evaluation
Si vous effectuez cette évalutation dans le cadre d'un cours, vous devriez pouvoir fournir votre travail à votre professeur/mentor pour correction. Si vous apprenez par vous même, vous pouvez obtenir la correction sur discussion thread for this exercise, ou sur #mdn IRC channel sur Mozilla IRC. Tout d'abord effectuez cet exercice — vous n'obtiendrez jamais rien en trichant !