School_Project/JournalDeBord/rapport/rapport.tex

275 lines
14 KiB
TeX
Raw Permalink Normal View History

2023-05-15 13:05:16 +02:00
\documentclass[12pt,a4paper]{article}
2023-05-16 09:45:41 +02:00
\usepackage{amsmath}
\usepackage{listings}
\usepackage{tikz}
2023-05-18 11:37:12 +02:00
\usepackage{csquotes}
2023-05-16 09:45:41 +02:00
2023-05-15 13:05:16 +02:00
\begin{document}
\title{Rapport du projet informatique 2023 \\ "Road to Master"}
\author{Debucquoy Anthony \and Matteo Di Leto}
\date{May 2023}
\maketitle
\newpage
\tableofcontents
\newpage
\section*{Introduction}
2023-05-21 17:54:47 +02:00
Lors de ce deuxième quadrimestre, le projet Informatique proposé par notre université fut partie intégrante de notre emploi du temps.
2023-05-22 02:12:50 +02:00
Régulièrement, nous nous sommes rassemblés pour nous organiser et trouver une direction dans laquelle nous voulions voir notre projet évoluer.
Grâce aux objectifs fixés par nos enseignants, nous sommes - nous le pensons - maintenant plus aptes à nous confronter à ce genre d'objectifs, tant au niveau personnel qu'en tant que groupe.
Il va sans dire que comme pour tout projet, notre chemin a été semé d'embûches. En l'occurrence, nous souhaitons faire part de l'abandon d'un de nos membre. Eddy Jiofak qui souhaite se réorienter.
2023-05-15 13:05:16 +02:00
Nous lui souhaitons une bonne reconversion.
2023-05-18 11:37:12 +02:00
\section*{Objectifs}
Voici l'objectif fixé par nos enseignants. (document de consignes)
\begin{displayquote}
Le but final de ce projet est de réaliser une application graphique en Java permettant de
jouer au jeu "Cats Organized Neatly". Ce jeu est un jeu de type "puzzle" où le joueur doit
placer des pièces de formes différentes pour combler l'aire de jeu. L'application devra permettre
de sauvegarder et charger une partie et de créer des niveaux automatiquement.
\end{displayquote}
2023-05-15 13:05:16 +02:00
\newpage
\section{Organisation}
2023-05-19 23:28:27 +02:00
Lors de nos rassemblement pour le projet, toutes les idées émises se sont retrouvés sur un blog afin de pouvoir y accéder de manière efficace.
2023-05-18 11:37:12 +02:00
Ce blog nous sert également à garder une trace de l'évolution du projet.
2023-05-15 13:05:16 +02:00
\subsection{Choix}
\begin{itemize}
2023-05-22 02:12:50 +02:00
\item{le VCS \verb|git| pour garder une trace de l'avancement du projet. Avec comme remote une instance privée de gitea nous permettant de vérifier les MR/PR plus efficacement.}
2023-05-19 23:28:27 +02:00
\item{Une instance de DroneCI permettant de vérifier que le projet soit toujours compilable et que les tests ne soient pas ratés sans que nous nous en rendions compte.}
\item{Javafx, comme recommandé par nos enseignants.}
2023-05-19 22:12:17 +02:00
\item{Les pièces et niveaux sont stockées sous forme de matrice de booléen}
2023-05-18 11:37:12 +02:00
\item{Un parser de fichiers efficace et donnant des fichier légers.}
\end{itemize}
\subsubsection*{Shapes}
2023-05-19 22:12:17 +02:00
Les pièces ainsi que la carte sont représentés par une matrice de booléen \verb|boolean[][]|.
Nous avons donc une classe \verb|shape|, parent de \verb|Map| et de \verb|Piece| dans lequel nous stockons notre matrice.
2023-05-20 14:17:42 +02:00
Ensuite, \verb|Map| Contient une liste de \verb|Piece|. Ces pièces contiennent une position représentée par la classe \verb|Vec2|.
2023-05-19 23:28:27 +02:00
Cette position est la position du carré supérieur gauche dans la \verb|Map|.
Avec toutes ces informations nous avons le nécessaire pour le moteur du jeu.
2023-05-19 23:28:27 +02:00
Il est facilement possible de manipuler la carte et les pièces. Il nous suffit alors de faire correspondre l'affichage avec
ces différentes classes. Ce qui est entrepris par la classe \verb|GameUI|.
2023-05-18 11:37:12 +02:00
Le tout est géré par la classe \verb|Controller| qui permet de choisir entre l'affichage d'un menu ou d'une partie en cours.
2023-05-20 14:17:42 +02:00
% \subsection{Difficultés}
2023-05-15 13:05:16 +02:00
\section{Points Forts}
2023-05-16 09:45:41 +02:00
\subsection{Parser de fichiers}
2023-05-22 02:12:50 +02:00
Pour la rétention des niveaux, plusieurs possibilités s'offraient à nous. Nous avons alors décidé d'accomplir une série d'objectifs propres à notre projet avec un parser de fichiers dédié.
2023-05-17 20:16:09 +02:00
Nous voulions que ce parser accomplisse les objectifs suivants:
2023-05-16 09:45:41 +02:00
\begin{itemize}
2023-05-17 20:16:09 +02:00
\item{Les données du niveau seront encapsulées dans un header/footer pour laisser la possibilité d'enregistrer plus d'informations (images/musiques) dans un seul fichier dans le futur.}
2023-05-22 02:12:50 +02:00
\item{La taille du fichier devra être aussi petite que possible, tout en gardant les informations nécessaires au bon fonctionnement du jeu.}
2023-05-16 09:45:41 +02:00
\item{Il sera possible d'enregistrer l'état d'une partie en cours.}
\end{itemize}
2023-05-18 11:37:12 +02:00
Ce parser est implémenté par la classe \verb|BinaryParser|.
2023-05-16 09:45:41 +02:00
\subsubsection*{spécification}
\begin{description}
\item[Header/Footer]{ Les données du niveau commencent par les 3 \emph{caractères} 'S', 'M', 'S' (ou \verb|0x534D53|) et se terminent par les 3 \emph{caractères} 'S', 'M', 'E' (ou \verb|0x534D45|)}
\item[Taille de carte]{ Le premier octet des données représente la largeur de la carte, le second sa hauteur.}
2023-05-22 02:12:50 +02:00
\item[Forme de la carte]{ Chaque cellule de la carte est représenté par un 1 ou un 0. Le 1 représente un emplacement libre, un 0 une cellule vide.
2023-05-16 09:45:41 +02:00
La forme de la carte peut alors être répartie sur un nombre indéterminé d'octets.
Nous pouvons déterminer ce nombre grace à
$$\frac{\text{largeur} * \text{hauteur } (+1 \text{ si multiple de } 8)}{8}$$
en division entière.}
2023-05-19 23:28:27 +02:00
\item[Nombre de pièce]{ L' (les) octet(s) qui forme(nt) la carte représente le nombre de pièce}
2023-05-21 17:54:47 +02:00
\item[Pour chaque pièces]{
2023-05-16 09:45:41 +02:00
\
\begin{description}
2023-05-19 22:12:17 +02:00
\item[Taille de la pièce]{La taille est représentée sur un seul octet sous forme de nibble\footnote{https://en.wikipedia.org/wiki/Nibble}. La première partie du nibble est la largeur. La seconde partie du nibble est la hauteur }
2023-05-21 17:54:47 +02:00
\item[Forme de la pièce]{ Chaque cellules de la pièce est représentée par un 1 ou un 0. La manière de le représenter et exactement la même que pour la forme de la carte }
2023-05-16 09:45:41 +02:00
\end{description}
2023-05-22 02:12:50 +02:00
Dans le cas où le fichier sauvegarde l'état de la partie, à la fin, et pour chaque pièce dans le même ordre que l'apparition des pièces:
2023-05-16 09:45:41 +02:00
\begin{description}
2023-05-19 22:12:17 +02:00
\item[Position de la pièce]{2 octets par pièces, 1 octet pour sa position en x et 1 octet pour sa position en y.
2023-05-21 17:54:47 +02:00
Dans le cas où la pièce est flottante (n'est pas placée dans la carte.), les octets contenant les caractères F puis L (0x464C) remplacent les coordonnées}
2023-05-16 09:45:41 +02:00
\end{description}
}
\end{description}
\subsubsection*{Exemple}
Voici le 'hexdump' du niveau 11
\begin{verbatim}
53 4d 53 05 05 e7 ff ff 80 06 33 ff 80 22 f0 22 |SMS.......3.."."|
b0 22 70 22 b0 11 80 53 4d 45 |."p"...SME|
\end{verbatim}
2023-05-19 14:46:17 +02:00
2023-05-21 17:54:47 +02:00
Représente une carte de la forme :
2023-05-16 09:45:41 +02:00
\begin{tikzpicture}
\filldraw[blue] (0,0) -- (0,5) -- (3,5) -- (3,4) -- (5,4) -- (5,0) -- cycle;
\draw[step=1cm,gray] (0, 0) grid (5,5);
\end{tikzpicture}
2023-05-21 17:54:47 +02:00
Avec les pièces :
2023-05-16 09:45:41 +02:00
\begin{tikzpicture}
\fill[red] (1,1) rectangle (4,4);
\fill[red] (5,1) rectangle (7,3);
\fill[red] (8,1) -- (8,3) -- (9,3) -- (9,2) -- (10,2) -- (10,1) -- cycle;
\fill[red] (11,1) -- (11,2) -- (12,2) -- (12,3) -- (13,3) -- (13,1) -- cycle;
\fill[red] (1,-2) -- (1,0) -- (2,0) -- (2,-1) -- (3,-1) -- (3,-2) -- cycle;
\fill[red] (4, -2) rectangle (5, -1);
\draw[step=1cm,gray] (0.5, -2.5) grid (13.5, 4.5);
\end{tikzpicture}
2023-05-22 02:12:50 +02:00
En plus de ce parser, et dans le cas où ce premier ne serait pas capable de stocker certaine carte (par exemple si une pièce mesure plus de 15x15),
nous avons également implémenté un parser très simple en utilisant l'interface \verb|Serialize| de java. Ce parser est implémenté et fonctionnel,
2023-05-18 11:37:12 +02:00
mais n'est pas utilisé dans le projet à l'heure actuelle.
2023-05-20 14:17:42 +02:00
Ces deux parseurs implémentent l'interface \verb|FileParser|.
2023-05-18 11:37:12 +02:00
2023-05-20 14:17:42 +02:00
Finalement, La classe \verb|FileParserFactory| permet une utilisation simple de ces parseurs. Celle-ci contient deux fonctions statiques.
2023-05-18 11:37:12 +02:00
\begin{itemize}
2023-05-20 14:17:42 +02:00
\item{\verb|Map loadMapFromFile(File)|}
2023-05-18 11:37:12 +02:00
permet de retourner la Map d'un fichier chargé.
2023-05-20 14:17:42 +02:00
\item{\verb|void saveFileFromMap(File, Map)|}
2023-05-18 11:37:12 +02:00
permet de sauvegarder une map dans un fichier.
\end{itemize}
Dans le cas d'une sauvegarde ou d'un chargement, le parser est choisi en fonction de l'extension de fichier ('.level', '.slevel', .'serialized', '.sserialized').
2023-05-22 02:12:50 +02:00
L'avantage de ce système est que nous pouvons facilement ajouter d'autres parser de fichiers dans le futur.
2023-05-18 11:37:12 +02:00
2023-05-20 14:17:42 +02:00
\subsection{Générateur de niveaux}
2023-05-22 02:12:50 +02:00
Le générateur de niveaux a été conçu de sorte à proposer 3 difficultés différentes:
2023-05-19 22:12:17 +02:00
2023-05-20 13:11:30 +02:00
\begin{itemize}
\item{Niveau Facile}
\item{Niveau Moyen}
\item{Niveau Difficile}
\end{itemize}
2023-05-19 22:12:17 +02:00
2023-05-22 02:12:50 +02:00
L'algorithme derrière est le même. En voici le principe :
2023-05-19 22:12:17 +02:00
2023-05-20 13:11:30 +02:00
\subsubsection*{Gestion du plateau}
2023-05-20 14:17:42 +02:00
2023-05-21 17:54:47 +02:00
Le joueur choisit une difficulté. En fonction de la difficulté choisie, la grandeur du plateau de jeu sera différente.
2023-05-22 02:12:50 +02:00
Si la difficulté choisie est facile ou moyenne, alors un curseur parcourt les extrémités du niveau.
Ce curseur sélectionne aléatoirement les cellules qui seront gardées ou non.
2023-05-20 14:17:42 +02:00
Grâce à ça, la forme du plateau n'est pas trop carrée.
2023-05-22 02:12:50 +02:00
Nous nous sommes basés sur le même principe pour le niveau de difficulté difficile mais en plus d'une taille encore plus grande,
le curseur parcourt les extrémités avec une profondeur de 2 afin de faire varier la carte plus grande.
Cela introduit le problème des cases isoléss. C'est pourquoi une boucle vérifie chaque cellule et la supprime si celle-ci est isolée.
2023-05-19 22:12:17 +02:00
2023-05-20 13:11:30 +02:00
\subsubsection*{Gestion des pièces}
2023-05-19 22:12:17 +02:00
2023-05-22 02:12:50 +02:00
Peu importe la difficulté du niveau, voici le fonctionnement :
2023-05-19 22:12:17 +02:00
2023-05-20 14:17:42 +02:00
Une taille maximum des pièces a été fixée au préalable à 3x3.
2023-05-22 02:12:50 +02:00
Par la suite, un curseur parcourt des cases de la carte préalablement conçue de manière aléatoire.
Pour chaque case, l'algorithme teste pour chaque case de la pièce, si l'espace est disponible.
Si ça n'est pas le cas, alors la pièce est modifiée afin de faire correspondre la pièce et la carte.
2023-05-19 22:12:17 +02:00
2023-05-20 14:17:42 +02:00
L'avantage de cette méthode est que les niveaux sont tous très différents.
Les désavantages sont que, par malchance, il est possible d'avoir énormément de piece 1x1.
2023-05-22 02:12:50 +02:00
il est aussi plus difficile d'appliquer des textures et dessins - à l'image du jeu de base - sur les pièces.
2023-05-20 13:11:30 +02:00
2023-05-22 02:12:50 +02:00
Malgré tout, avec nos nombreux tests, ce générateur de niveaux nous satisfait vraiment bien et la difficulté des niveaux correspond bien aux attentes.
2023-05-20 13:11:30 +02:00
2023-05-20 14:17:42 +02:00
\subsection{Interface graphique}
2023-05-20 23:37:22 +02:00
L'interface graphique du jeu tient sur 5 classes différentes.
2023-05-20 13:11:30 +02:00
2023-05-20 14:17:42 +02:00
\subsubsection*{Controller}
2023-05-22 02:12:50 +02:00
Classe principale. Elle s'occupe de la gestion des autres classes et de la cohérence entre elles.
2023-05-20 14:17:42 +02:00
Toutes les autres classes (présentes dans le package \verb|Scenes|) sont des sous classe de Parents.
2023-05-22 02:12:50 +02:00
Cela permet de les afficher grâce à la méthode statique \verb|switchRoot|.
2023-05-20 14:17:42 +02:00
C'est aussi le point d'entrée du programme.
2023-05-20 13:11:30 +02:00
2023-05-20 14:17:42 +02:00
\subsubsection*{MenuAccueil}
Classe s'occupant de générer la page d'accueil du jeu.
2023-05-21 17:54:47 +02:00
C'est-à-dire la première page que verra l'utilisateur.
2023-05-20 14:17:42 +02:00
Cette page permet d'accéder aux niveaux du jeu de base, mais également de générer les niveaux aléatoires.
2023-05-22 02:12:50 +02:00
De plus un dernier bouton "Load Game" permet de revenir sur la dernière partie en cours du joueur.
2023-05-20 13:11:30 +02:00
2023-05-20 14:17:42 +02:00
\subsubsection*{MenuLevel}
Classe s'occupant d'afficher les niveaux proposés dans le jeu, et qui,
2023-05-22 02:12:50 +02:00
en fonction du jour choisi, change les boutons et les niveaux disponibles.
2023-05-19 15:27:12 +02:00
2023-05-20 14:17:42 +02:00
\subsubsection*{ScreenLevelFinish}
Classe qui s'affiche à l'écran dès que le joueur a fini le niveau.
Celle-ci propose également de réessayer le niveau ou de retourner au menu principal.
\subsubsection*{GameUI}
Classe s'occupant de l'affichage d'un niveau.
2023-05-21 17:54:47 +02:00
S'occupe dans un premier temps d'afficher le plateau au milieu de l'écran.
2023-05-22 02:12:50 +02:00
Par la suite, les pièces sont dispersées à la gauche de l'écran et sont rendues disponibles à l'utilisateur.
2023-05-21 17:54:47 +02:00
Ces pièces sont déplaçables à l'aide de la souris.
2023-05-22 02:12:50 +02:00
Si elles sont lâchées sur un emplacement disponible du plateau, alors elles vont s'aligner avec ses cellules.
2023-05-20 14:17:42 +02:00
Pour se faire, nous regardons si le centre d'une cellule (la plus en haut à gauche) est présente dans une cellule du plateau.
2023-05-22 02:12:50 +02:00
Ensuite, on vérifie que toutes les cellules de la pièce ont bien la place dans le plateau pour se poser. Si c'est le cas, alors la pièce est posée.
2023-05-19 15:27:12 +02:00
2023-05-15 13:05:16 +02:00
\section{Points Faibles}
2023-05-20 14:17:42 +02:00
Bien que l'interface graphique permet de naviguer de manière fluide dans le jeu, il y a clairement un manque d'animation et de dynamisme
\emph{i.e}: transition de page à une autre, apparition des boutons ainsi qu'un manque de musique.
Toujours concernant l'affichage l'adaptation à la taille de l'écran peut être revue.
2023-05-19 23:28:27 +02:00
2023-05-20 14:17:42 +02:00
De plus suite à la perte de notre membre nous n'avons pas su gérer la partie du projet qui concerne le design/textures des pièces ainsi que l'histoire jeu préparée auparavant.
2023-05-19 15:03:31 +02:00
2023-05-15 13:05:16 +02:00
\section{Apports Positifs et négatifs}
\subsection{Anthony}
2023-05-21 17:54:47 +02:00
Personnellement, ce projet m'a permis de me plonger dans la conception d'un format de fichier personnalisé.
2023-05-22 02:12:50 +02:00
Cest un exercice que je navais pas encore fait jusqua maintenant. Malgré mes efforts pour prévoir un maximum de choses à lavance afin déviter de devoir modifier ma spécification pendant le développement, je me suis vite rendu compte que je navais pas pensé à tout et que je devrais changer des étapes pour pouvoir arriver à mes fins. Je pense que ce parser de fichier est vraiment améliorable mais je suis relativement fier du résultat.
J'ai pu présenter ce parser à Dr Quoitin qui a pu me conseiller sur différentes approches à ce problème.
2023-05-22 02:12:50 +02:00
J'en prends bonne note.
\subsection{Matteo}
2023-05-22 02:12:50 +02:00
Il est clair que je peux tirer plusieurs enseignements grâce à la réalisation de notre projet.
2023-05-19 23:28:27 +02:00
Tout d'abord, j'ai pu en apprendre beaucoup plus concernant la P.O.O en java, mais aussi j'en ai appris
2023-05-21 17:54:47 +02:00
d'avantage sur l'utilisation de la bibliothèque JavaFx.
2023-05-19 20:25:58 +02:00
2023-05-22 02:12:50 +02:00
De plus, durant la réalisation du projet, comme dit précédemment, nous avons utilisé plusieurs outils dont la plus grande découverte pour moi est
Git qui révèle son intérêt pour les travaux de groupes et qui, selon moi, se révélera tout aussi essentiel pour mes futurs projets
2023-05-19 20:25:58 +02:00
scolaires ainsi que professionnels.
2023-05-20 14:17:42 +02:00
\section{conclusion}
2023-05-19 20:25:58 +02:00
2023-05-22 02:12:50 +02:00
En conclusion, nous pouvons séparer notre travail en trois parties différentes:
2023-05-19 20:25:58 +02:00
\begin{enumerate}
2023-05-20 13:11:30 +02:00
\item Le parser de fichier (gestion sauvegarder/charger partie)
2023-05-20 14:17:42 +02:00
\item Logique du jeu (Map,Vec2,Shapes)
\item Liaison à l'UI (Javafx)
\end{enumerate}
2023-05-20 14:17:42 +02:00
2023-05-22 02:12:50 +02:00
Malgré notre travail concentré sur le bon fonctionnement du jeu avec un parser suivant nos objectifs, une utilisation de la P.O.O de manière très efficace
ainsi qu'une approche correcte de l'utilisation du framework Javafx, d'autres améliorations sont toujours possibles!
En effet, l'idée d'ajouter une histoire, des trophées, un Easter egg, des pièces spéciales ou un encore une table de score basée sur le temps,
2023-05-20 23:37:22 +02:00
reste possible afin de rendre notre jeu encore plus complet.
En conclusion, notre jeu a encore plein de possibilité afin d'être encore plus complet et amusant!
2023-05-16 09:45:41 +02:00
2023-05-15 13:05:16 +02:00
\end{document}