TP: tic-tac-toe en réseau¶
Prise en main (30 min)¶
L’objectif¶
Nous voulons programmer un tic-tac-toe en réseau.
Le projet est déjà entamé. Vous devez télécharger ce code
puis l’enrichir.
Le projet est organisé en 4 packages:
fr.insalyon.tc.elp.noyau pour les classes de base, comme celle modélisant la configuration des marques.
fr.insalyon.tc.elp.entree pour les classes modifiant la configuration courante.
fr.insalyon.tc.elp.sortie pour les classes offrant une vue sur la configuration courante.
fr.insalyon.tc.elp.app pour les applications combinant les classes des 3 packages précédents.
Package noyau¶
Il contient:
Couleur : couleur binaire (noir ou blanc)
Marque : marque, identifiée par une couleur, déposée sur la grille par un joueur
Position : position donnée par un couple d’indices
EnsembleDeMarques : configuration des marques
Regardez attentivement ces classes que vous utiliserez.
Positionnement¶
Nous avons les conventions suivantes:
Grille 3x3,
Identification des cases par un couple d’indices. On a repère absolu, dont le centre se trouve en haut à gauche, le premier vecteur est horizontal, pointé vers la gauche, le second est vertical pointé vers le bas.
Package Entree¶
Il contient:
Joueur : modélise un joueur qui, à son tour, dépose une marque de sa couleur sur la grille
Arbitre : modélise un arbitre qui s’assure que les joueurs jouent à tour de rôle
LecteurPosition : interface caractérisant les objets à qui on peut demander une position à laquelle déposer une nouvelle marque.
LecteurPositionDeFlux : LecteurPosition dans lequel la position est lue depuis un flux (entrée standard par exemple).
Joueur vs Utilisateur¶
Un objet Joueur, a une couleur, connait la configuration des pions, l’arbitre, ainsi qu’un objet de type LecteurPosition. Si cet objet est du sous-type LecteurPositionDeFlux, alors le joueur est en attente d’une chaine de caractères sur un flux d’entrée contenant la position qu’il doit jouer.
Un utilisateur peut donc transmettre par le flux d’entrée standard son déplacement à l’objet Joueur qui le représente.
Transmission textuelle¶
Pour transmettre une position sur le flux d’entrée standard, un utilisateur doit respecter la convention suivante:
une position s’écrit:
i,j
(abscisse, virgule, ordonnée). Par exemple1,2
.
NB. la méthode toString
des objets de type Position
respecte cette convention.
Package Sortie¶
Il contient pour l’instant:
VueShell: classe permettant d’afficher à l’écran la configuration des marques.
A l’avenir, nous voulons aussi avoir:
VueGraphique: classe permettant d’afficher dans une fenêtre graphique la configuration des marques.
Package App¶
Il est vide! Créez vite une première application TicTacToeShell
, dans laquelle
les deux utilisateurs donnent la position de leur nouvelle marque, sous forme textuelle,
dans le shell.
Dans la méthode main
:
créez une configuration (instance
EnsembleDeMarques
),créez un lecteur de position (instance de
LecteurPositionDeFlux
),créez un arbitre, un joueur blanc, un joueur noir (instances de
Arbitre
etJoueur
),lancez deux threads, un pour chaque joueur.
Vous pouvez jouer à l’aveugle: la grille n’est pas affichée!
Modèle et vue (30 min)¶
Objectif¶
La classe EnsembleDeMarques
(le modèle) ne connait pas la classe VueShell
(la vue).
On veut pourtant qu’elles puissent communiquer entre elles, pour que la vue soit mise à jour après la modification des données (ajout ou suppression d’un pion sur une position).
En même temps, on ne veut pas que le modèle dépende d’une vue particulière et on voudrait avoir la possibilité d’offrir aisément zéro, une ou plusieurs vues sur le modèle.
Pour réaliser cela, nous allons mettre en oeuvre le principe Observable/Observer.
Observable¶
Notre modèle (EnsembleDeMarques
) doit être observable (et observé par la vue).
Pour être observable, il doit dériver de la classe Observable.
Il hérite notamment des méthodes
setChanged()
: indique que l’état du modèle à changernotifyObservers(Object arg)
: avertit les objets qui observent le modèleclearChanged()
: oublie le changement
que vous devez appeler, dans l’ordre, à la fin de la méthode ajouterMarque
,
pour demander à la vue de procéder à une mise à jour de l’affichage.
Observer¶
Notre vue (VueShell
) doit observer notre modèle et déclencher l’affichage à
chaque fois que le modèle est mis à jour après l’ajout d’une nouvelle marque.
Elle doit satisfaire l’interface Observer,
c’est-à-dire posséder une méthode update(Observable o, Object arg)
(appelée par notifyObservers(Object arg)
).
C’est dans cette méthode que vous devez déclencher l’affichage.
L’objet o
est l’objet observable qui appelle update
(EnsembleDeMarques
).
Le paramètre arg
permet d’ajouter une information pour la mise à jour; par exemple,
la position à laquelle à été ajouté la nouvelle marque.
Dans l’application TicTacToeShell
¶
Dans la méthode main
:
créez une vue (instance de
VueShell
),puis ajoutez-la comme observateur du modèle (instance de
EnsembleDeMarques
) avec la méthodeaddObserver(Observer o)
, héritée de Observable.
Vous pouvez commencer à jouer!
Serveur/Client (30 min)¶
Objectif¶
Jusqu’à maintenant, les deux utilisateurs jouent sur la même machine.
Nous allons maintenant faire deux applications: TicTacToeShellServeur
et TicTacToeShellClient
.
Les deux applications auront toutes deux, un ensemble de marques, deux joueurs synchronisés par un arbitre, ainsi qu’une vue. Cependant, l’un des deux joueurs lira ses positions depuis la machine distante, tandis que l’autre enverra ses positions sur la machine distante.
Ainsi, seuls des positions, sous forme textuelle, seront communiquées.
Serveur¶
Pour lire un déplacement depuis la machine distante, il suffit, pour le joueur distant, de fournir à l’objet de type
LecteurPositionDeFlux
le flux d’entrée du socket. Testez cette partie avectelnet
.Pour transmettre un déplacement, nous allons créer une nouvelle classe
JoueurTransmetteur
qui dérive deJoueur
et qui redéfinit la méthodejouer
de façon à, une fois une position choisie, l’écrire sur le flux de sortie du socket. Testez encore avectelnet
.
Client¶
Une fois que vous avez un serveur qui fonctionne, écrire
l’application TicTacToeShellClient
est un jeu d’enfant,
car le code est parfaitement symétrique au code du serveur.
Vous pouvez maintenant jouer depuis deux machines différentes. Mais l’interface du jeu n’est pas confortable.
Interface graphique (45 min)¶
Objectif¶
Nous voulons créer une interface graphique pour notre jeu.
Le package Sortie du projet sera enrichi:
d’une classe
VueGraphique
satisfaisant les interfacesRunnable
etObserver
,éventuellement accompagnée d’autres classes pour décomposer le code.
La package App sera enrichi de nouvelles applications:
TicTacToeGraphique
pour jouer à deux sur la même machine,les variantes client/serveur pour jouer sur des machines distinctes.
Conception¶
La conception est libre, mais si vous ne savez pas quoi faire, vous pouvez:
écrire une classe
Case
dérivant deJButton
avec un champs contenant la position correspondante.dans la classe
VueGraphique
, organisez les cases dans un panel (JPanel
) sous forme tabulaire (GridLayout
).
Vous pouvez commencer par tester votre affichage en donnant les positions par le shell. Mais comment lire les positions directement sur l’interface graphique ?
Evénements (45 min)¶
Objectif¶
Maintenant que nous avons une interface graphique, nous voulons aussi que les utilisateurs puissent prescrire leurs déplacements par des actions (clics) sur l’interface graphique.
Conception¶
Vous devez écrire un listener, par exemple
CaseListener
. Une instance de cette classe devrait récupérer la position de la case choisie (et c’est pourquoi il est utile qu’une case connaisse sa position).Puis écrire une classe
LecteurPositionDeVueGraphique
qui satisfait l’interfaceLecteurPosition
. La vue, comme le listener, doivent connaitre le lecteur afin de lui transmettre la position de la case choisie.