Amélioration intéraction, game over + check cases numéro
This commit is contained in:
		
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 232 B After Width: | Height: | Size: 159 B | 
							
								
								
									
										
											BIN
										
									
								
								public/sHighlight.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								public/sHighlight.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 117 B | 
| @@ -1,31 +1,40 @@ | |||||||
| type CellGrid = string[][]                                                                                      // type CellGrid (grille de jeu) | type CellGrid = string[][]                                                                                      // type CellGrid (grille de jeu) | ||||||
| type MineGrid = number[][]                                                                                      // type MineGrid (grille de bombes) | type MineGrid = number[][]                                                                                      // type MineGrid (grille de bombes) | ||||||
|  | type Directions = number[][]                                                                                    // type Directions (directions possibles) | ||||||
|  |  | ||||||
|  | function getDirections(x: number, y: number): Directions { | ||||||
|  |     return [[x-1, y-1], [x-1, y], [x-1, y+1], [x, y-1], [x, y+1], [x+1, y-1], [x+1, y], [x+1, y+1]] | ||||||
|  | } | ||||||
|  |  | ||||||
| function genCellGrid(width: number, length: number): CellGrid {                                                 // fonction genCellGrid (génération de la grille de jeu) | function genCellGrid(width: number, length: number): CellGrid {                                                 // fonction genCellGrid (génération de la grille de jeu) | ||||||
|     return Array.from({ length: width }, () => Array.from({ length: length }, () => "/sUnknown.png")) |     return Array.from({ length: width }, () => Array.from({ length: length }, () => "/sUnknown.png")) | ||||||
| } | } | ||||||
|  |  | ||||||
| function genMineGrid(width: number, length: number, nbMines: number): MineGrid {                                // fonction genMineGrid (génération des bombes) | function genMineGrid(width: number, length: number, nbMines: number, fx: number, fy: number): MineGrid {        // fonction genMineGrid (génération des bombes) | ||||||
|     const mineGrid: MineGrid = []                                                                               // initialisation de la grille de bombes |     const mineGrid: MineGrid = []                                                                               // initialisation de la grille de bombes | ||||||
|  |     const directions = getDirections(fx, fy) | ||||||
|  |      | ||||||
|     while (mineGrid.length < nbMines) {                                                                         // boucle pour placer les bombes |     while (mineGrid.length < nbMines) {                                                                         // boucle pour placer les bombes | ||||||
|         const x = Math.floor(Math.random() * length)                                                            // position x de la bombe |         const x = Math.floor(Math.random() * length)                                                            // position x de la bombe | ||||||
|         const y = Math.floor(Math.random() * width)                                                             // position y de la bombe |         const y = Math.floor(Math.random() * width)                                                             // position y de la bombe | ||||||
|         const indice = [y, x]                                                                                   // id de la bombe |         const indice = [x, y] | ||||||
|         if (!mineGrid.includes(indice)) mineGrid.push(indice)                                                   // si la bombe n'est pas déjà placée, on la place |          | ||||||
|  |         if (directions.some(d => d[0] === x && d[1] === y) || (x === fx && y === fy)) continue                  // vérifie si la bombe n'est pas déjà placée et si ce n'est pas le premier clic | ||||||
|  |         if (!mineGrid.some(m => m[0] === x && m[1] === y)) mineGrid.push(indice)                                // vérifie si la bombe n'est pas déjà placée et si ce n'est pas le premier clic | ||||||
|     } |     } | ||||||
|     return mineGrid;                                                                                            // retour de la grille de bombes |     mineGrid.sort((a, b) => a[0] - b[0] || a[1] - b[1])                                                         // Trier les bombes par ordre x puis y | ||||||
|  |     return mineGrid                                                                                             // retour de la grille de bombes | ||||||
| } | } | ||||||
|  |  | ||||||
| function cliqueGauche(cellGrid: CellGrid, mineGrid: MineGrid, x: number, y: number) {                           // fonction main (début du jeu) |  | ||||||
|     //const indice = [y, x]                                                                                     // id de la case cliquée | function cliqueGauche(gameStatus: number, cellGrid: CellGrid, mineGrid: MineGrid, x: number, y: number) { | ||||||
|     //if (mineGrid.includes(indice)) bombe()                                                                    // si la case cliquée est une bombe, on va dans la fonction bombe    |     const checkDrop = ['/sEmpty.png', '/sUnknown.png', '/sFlag.png', '/sQuestion.png'] | ||||||
|     //else if (champDeMines[indice] === 0) {caseVide(indice);}                                                  // si la case cliquée est vide, on va dans la fonction caseVide |     if (checkDrop.includes(cellGrid[y][x])) return | ||||||
|     //else if (champDeMines[indice] !== 0) {decouvreCase(indice);}                                              // si la case cliquée est un numéro, on va dans la fonction decouvreCase |     else if (cellGrid[y][x] === '/sClick.png') checkCell(gameStatus, cellGrid, mineGrid, x, y) | ||||||
|     //decouvreCase(indice); |     else revealCell(gameStatus, cellGrid, mineGrid, x, y) | ||||||
|     if (cellGrid[y][x] === '/sClick.png') return cellGrid[y][x] = '/sEmpty.png'                                 // modification de l'image de la case |  | ||||||
| } | } | ||||||
|  |  | ||||||
| function cliqueDroit(cellGrid: string[][], x: number, y: number): void {                                        // fonction cliqueDroit (flag d'une case)                                                                                       // modification de l'image de la case | function cliqueDroit(cellGrid: string[][], x: number, y: number) {                                              // fonction cliqueDroit (flag d'une case)                                                                                       // modification de l'image de la case | ||||||
|     switch (cellGrid[y][x]) { |     switch (cellGrid[y][x]) { | ||||||
|         case '/sUnknown.png': |         case '/sUnknown.png': | ||||||
|             cellGrid[y][x] = "/sFlag.png" |             cellGrid[y][x] = "/sFlag.png" | ||||||
| @@ -39,30 +48,58 @@ function cliqueDroit(cellGrid: string[][], x: number, y: number): void { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | function checkCell(gameStatus: number, cellGrid: CellGrid, mineGrid: MineGrid, x: number, y: number) { | ||||||
|  |     if (mineGrid.some(m => m[0] === x && m[1] === y)) return gameOver(gameStatus, cellGrid, mineGrid, x, y) | ||||||
|  |  | ||||||
| /* |     let minesAdjacentes = 0 | ||||||
| const directions: number[] = [-11, -10, -9, -1, 1, 9, 10, 11];                                                                  // directions possibles pour découvrir les cases |     const directions = getDirections(x, y) | ||||||
| const champDeMines: number[] = [];                                                                                              // champ de mines (0 = case vide, 10 = bombe, 1 à 8 = nombre de bombes adjacentes) |     for (const [dx, dy] of directions) { | ||||||
| for (let i = 0; i < 100; i++) {champDeMines.push(0);}                                                                           // initialisation des cases à 0 |         if (dx < 0 || dx >= cellGrid[0].length || dy < 0 || dy >= cellGrid.length) continue | ||||||
| const casesDecouvertes: number[] = [];                                                                                          // cases découvertes (0 = case non découverte, 1 = case découverte, 2 = case flag) |         if (mineGrid.some(m => m[0] === dx && m[1] === dy)) minesAdjacentes++ | ||||||
| for (let i = 0; i < 100; i++) {casesDecouvertes.push(0);}                                                                       // initialisation des cases à 0 |  | ||||||
|  |  | ||||||
| function bombe(): void {                                                                                                        // fonction bombe (game over) |  | ||||||
|     for (let i = 0; i < casesDecouvertes.length; i++) {casesDecouvertes[i] = 1;} |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
| function caseVide(indice: number): void {                                                                                       // fonction caseVide (découverte des cases adjacentes à une case vide) |     if (minesAdjacentes > 0) cellGrid[y][x] = `/s${minesAdjacentes}.png` | ||||||
|     for (let i = 0; i < directions.length; i++) {                                                                               // parcours des directions |     else { | ||||||
|         if (champDeMines[indice + directions[i]] !== 0) {decouvreCase(indice + directions[i]); return; }                        // si la case est un numéro, on ne fait rien |         cellGrid[y][x] = '/sEmpty.png' | ||||||
|         else if (casesDecouvertes[indice + directions[i]] === 2) {decouvreCase(indice + directions[i]); return;}                // si la case est un drapeau, on ne fait rien |         for (const [dx, dy] of directions) { | ||||||
|         else {decouvreCase(indice + directions[i]); caseVide(indice + directions[i]);}                                          // si la case est vide, on va dans la fonction caseVide |             if (dx >= 0 && dx < cellGrid[0].length && dy >= 0 && dy < cellGrid.length && cellGrid[dy][dx] === '/sUnknown.png') { | ||||||
|  |                 checkCell(gameStatus, cellGrid, mineGrid, dx, dy) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| function decouvreCase(indice: number): void {                                                                                   // fonction decouvreCase (découverte d'une case) | function revealCell(gameStatus: number, cellGrid: CellGrid, mineGrid: MineGrid, x: number, y: number) { | ||||||
|     casesDecouvertes[indice] = 1;                                                                                               // modification dans le tableau des cases découvertes  |     const directions = getDirections(x, y) | ||||||
|  |     let flagCount = 0 | ||||||
|  |     for (const [dx, dy] of directions) { | ||||||
|  |         if (dx >= 0 && dx < cellGrid[0].length && dy >= 0 && dy < cellGrid.length && cellGrid[dy][dx] === '/sFlag.png') flagCount++ | ||||||
|  |     } | ||||||
|  |     const cellValue = parseInt(cellGrid[y][x].match(/\/s(\d)\.png$/)?.[1] || '0') | ||||||
|  |     if (flagCount === cellValue) { | ||||||
|  |         for (const [dx, dy] of directions) { | ||||||
|  |             if (dx >= 0 && dx < cellGrid[0].length && dy >= 0 && dy < cellGrid.length && cellGrid[dy][dx] === '/sHighlight.png') { | ||||||
|  |                 cellGrid[dy][dx] = '/sUnknown.png' | ||||||
|  |                 checkCell(gameStatus, cellGrid, mineGrid, dx, dy) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
| } | } | ||||||
| */                                                                          |  | ||||||
|  |  | ||||||
| export { genCellGrid, genMineGrid, cliqueGauche, cliqueDroit }                                                                  // export des variables et fonctions | function gameOver(gameStatus: number, cellGrid: CellGrid, mineGrid: MineGrid, x: number, y: number) { | ||||||
| export type { CellGrid, MineGrid }                                                                                              // export des types |     for (const [mx, my] of mineGrid) { | ||||||
|  |         if (cellGrid[my][mx] === '/sUnknown.png') cellGrid[my][mx] = '/sMine.png' | ||||||
|  |     } | ||||||
|  |     // Vérifier l'emplacement de tous les flags dans toute la grille et les remplacer par '/sFlagWrong.png' s'il y en a un qui n'est pas sur une bombe | ||||||
|  |     for (let y = 0; y < cellGrid.length; y++) { | ||||||
|  |         for (let x = 0; x < cellGrid[y].length; x++) { | ||||||
|  |             if (cellGrid[y][x] === '/sFlag.png') if (!mineGrid.some(m => m[0] === x && m[1] === y)) cellGrid[y][x] = '/sFlagWrong.png' | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     cellGrid[y][x] = '/sExploded.png' | ||||||
|  |     gameStatus = 3 | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export { getDirections, genCellGrid, genMineGrid, cliqueGauche, cliqueDroit }                                 // export des fonctions | ||||||
|  | export type { Directions, CellGrid, MineGrid }                                                                // export des types | ||||||
| @@ -1,8 +1,9 @@ | |||||||
| <script setup lang="ts"> | <script setup lang="ts"> | ||||||
| import { ref, onMounted, onUnmounted } from 'vue' | import { ref, watch, onMounted, onUnmounted } from 'vue' | ||||||
| import { genCellGrid, genMineGrid, cliqueGauche, cliqueDroit } from '@/utils/game' | import { getDirections, genCellGrid, genMineGrid, cliqueGauche, cliqueDroit } from '@/utils/game' | ||||||
| import type { CellGrid, MineGrid } from '@/utils/game' | import type { CellGrid, MineGrid } from '@/utils/game' | ||||||
|  |  | ||||||
|  | // Variables du jeu | ||||||
| const width = 10 | const width = 10 | ||||||
| const length = 15 | const length = 15 | ||||||
| const nbMines = 25 | const nbMines = 25 | ||||||
| @@ -13,48 +14,111 @@ cellGrid.value = genCellGrid(width, length) | |||||||
|  |  | ||||||
| // Création grille des mines | // Création grille des mines | ||||||
| let mineGrid: MineGrid = [] | let mineGrid: MineGrid = [] | ||||||
| mineGrid = genMineGrid(width, length, nbMines) |  | ||||||
|  | // État du jeu | ||||||
|  | const gameStatus = ref(0) // 0: vide, 1: en cours, 2: gagné, 3: perdu | ||||||
|  | const timer = ref(0) | ||||||
|  | let timerInterval: number | undefined = undefined | ||||||
|  |  | ||||||
|  | watch(gameStatus, async (status) => { | ||||||
|  |   if (status === 1) { | ||||||
|  |     timerInterval = setInterval(() => timer.value++, 1000) | ||||||
|  |   } else { | ||||||
|  |     if (timerInterval !== null) clearInterval(timerInterval) | ||||||
|  |     if (status === 2) alert('Vous avez gagné !') | ||||||
|  |     if (status === 3) alert('Vous avez perdu !') | ||||||
|  |   } | ||||||
|  | }) | ||||||
|  |  | ||||||
| // État pour gérer le clic et la position de la cellule | // État pour gérer le clic et la position de la cellule | ||||||
| const isMouseDown = ref(false) | const isMouseDown = ref(false) | ||||||
| const currentCell = ref<{ rowIndex: number, cellIndex: number } | null>(null) | const currentCell = ref<{ rowIndex: number, cellIndex: number } | null>(null) | ||||||
|  |  | ||||||
|  | // État pour suivre les cases surlignées | ||||||
|  | const highlightedCells = ref<number[][]>([]) | ||||||
|  |  | ||||||
|  | // Gestion des événements de clic et de la position de la cellule | ||||||
| const handleMouseDown = (rowIndex: number, cellIndex: number) => { | const handleMouseDown = (rowIndex: number, cellIndex: number) => { | ||||||
|   isMouseDown.value = true |   isMouseDown.value = true | ||||||
|   currentCell.value = { rowIndex, cellIndex } |   currentCell.value = { rowIndex, cellIndex } | ||||||
|   if (cellGrid.value[rowIndex][cellIndex] === '/sUnknown.png') cellGrid.value[rowIndex][cellIndex] = '/sClick.png' |   const cellValue = cellGrid.value[rowIndex][cellIndex] | ||||||
|  |  | ||||||
|  |   if (cellValue === '/sUnknown.png') cellGrid.value[rowIndex][cellIndex] = '/sClick.png' // Si la cellule est inconnue, on la vérifie | ||||||
|  |   else if (/\/s\d\.png$/.test(cellValue)) highlightAdjacentUnknownCells(cellIndex, rowIndex) // Vérifie si la cellule contient un chiffre | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
| const handleMouseUp = (rowIndex: number, cellIndex: number) => { | // Gestion des événements de relâchement de clic et de la position de la cellule | ||||||
|  | const handleMouseUp = async (rowIndex: number, cellIndex: number) => { | ||||||
|   isMouseDown.value = false |   isMouseDown.value = false | ||||||
|   cliqueGauche(cellGrid.value, mineGrid, cellIndex, rowIndex) |   if (!mineGrid.length) { | ||||||
|   currentCell.value = null |     mineGrid = genMineGrid(width, length, nbMines, cellIndex, rowIndex) | ||||||
|  |     gameStatus.value = 1 | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   cliqueGauche(gameStatus.value, cellGrid.value, mineGrid, cellIndex, rowIndex) | ||||||
|  |   currentCell.value = null | ||||||
|  |  | ||||||
|  |   // Réinitialiser les cases surlignées | ||||||
|  |   for (const [dx, dy] of highlightedCells.value) { if (cellGrid.value[dy][dx] === '/sHighlight.png') cellGrid.value[dy][dx] = '/sUnknown.png' } | ||||||
|  |   highlightedCells.value = [] | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Gestion des événements de déplacement de la souris et de la position de la cellule | ||||||
| const handleMouseMove = (rowIndex: number, cellIndex: number) => { | const handleMouseMove = (rowIndex: number, cellIndex: number) => { | ||||||
|   if (isMouseDown.value && currentCell.value) { |   if (isMouseDown.value && currentCell.value) { | ||||||
|     const { rowIndex: prevRowIndex, cellIndex: prevCellIndex } = currentCell.value |     const { rowIndex: prevRowIndex, cellIndex: prevCellIndex } = currentCell.value | ||||||
|  |  | ||||||
|     if (prevRowIndex !== rowIndex || prevCellIndex !== cellIndex) { |     if (prevRowIndex !== rowIndex || prevCellIndex !== cellIndex) { | ||||||
|       if (cellGrid.value[prevRowIndex][prevCellIndex] === '/sClick.png') cellGrid.value[prevRowIndex][prevCellIndex] = '/sUnknown.png' // Réinitialiser l'image précédente |       const cellValue = cellGrid.value[rowIndex][cellIndex] | ||||||
|       if (cellGrid.value[rowIndex][cellIndex] === '/sUnknown.png') cellGrid.value[rowIndex][cellIndex] = '/sClick.png' // Changer l'image actuelle |  | ||||||
|  |       // Réinitialiser les cases surlignées | ||||||
|  |       for (const [dx, dy] of highlightedCells.value) { cellGrid.value[dy][dx] = '/sUnknown.png' } | ||||||
|  |       highlightedCells.value = [] | ||||||
|  |  | ||||||
|  |       // Changer l'image actuelle | ||||||
|  |       if (cellValue === '/sUnknown.png') cellGrid.value[rowIndex][cellIndex] = '/sClick.png' | ||||||
|  |       else if (/\/s\d\.png$/.test(cellValue)) highlightAdjacentUnknownCells(cellIndex, rowIndex) | ||||||
|  |  | ||||||
|  |       // Réinitialiser l'image précédente | ||||||
|  |       const prevCellValue = cellGrid.value[prevRowIndex][prevCellIndex] | ||||||
|  |       if (prevCellValue === '/sClick.png') cellGrid.value[prevRowIndex][prevCellIndex] = '/sUnknown.png' | ||||||
|  |  | ||||||
|       currentCell.value = { rowIndex, cellIndex } |       currentCell.value = { rowIndex, cellIndex } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // Gestion des événements de déplacement de la souris et de la position de la cellule | ||||||
| const handleGlobalMouseMove = (event: MouseEvent) => { | const handleGlobalMouseMove = (event: MouseEvent) => { | ||||||
|   if (isMouseDown.value && currentCell.value) { |   if (isMouseDown.value && currentCell.value) { | ||||||
|     const target = event.target as HTMLElement |     const target = event.target as HTMLElement | ||||||
|  |  | ||||||
|     if (!target.classList.contains('cell')) { |     if (!target.classList.contains('cell')) { | ||||||
|       const { rowIndex, cellIndex } = currentCell.value |       const { rowIndex, cellIndex } = currentCell.value | ||||||
|       if (cellGrid.value[rowIndex][cellIndex] === '/sClick.png') { |  | ||||||
|         cellGrid.value[rowIndex][cellIndex] = '/sUnknown.png' |       if (cellGrid.value[rowIndex][cellIndex] === '/sClick.png') cellGrid.value[rowIndex][cellIndex] = '/sUnknown.png' | ||||||
|       } |  | ||||||
|       currentCell.value = null |       currentCell.value = null | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // Fonction pour surligner toutes les cases adjacentes inconnues | ||||||
|  | const highlightAdjacentUnknownCells = (cellIndex: number, rowIndex: number) => { | ||||||
|  |   const directions = getDirections(cellIndex, rowIndex) | ||||||
|  |    | ||||||
|  |   highlightedCells.value = [] // Réinitialiser la liste des cases surlignées | ||||||
|  |  | ||||||
|  |   for (const [dx, dy] of directions) { | ||||||
|  |     if (dx < 0 || dx >= cellGrid.value[0].length || dy < 0 || dy >= cellGrid.value.length) continue | ||||||
|  |     if (cellGrid.value[dy][dx] === '/sUnknown.png') { | ||||||
|  |       cellGrid.value[dy][dx] = '/sHighlight.png' | ||||||
|  |       highlightedCells.value.push([dx, dy]) | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Écoute des événements de déplacement de la souris | ||||||
| onMounted(() => document.addEventListener('mousemove', handleGlobalMouseMove)) | onMounted(() => document.addEventListener('mousemove', handleGlobalMouseMove)) | ||||||
| onUnmounted(() => document.removeEventListener('mousemove', handleGlobalMouseMove)) | onUnmounted(() => document.removeEventListener('mousemove', handleGlobalMouseMove)) | ||||||
| </script> | </script> | ||||||
| @@ -62,6 +126,7 @@ onUnmounted(() => document.removeEventListener('mousemove', handleGlobalMouseMov | |||||||
| <template> | <template> | ||||||
|   <div class="main"> |   <div class="main"> | ||||||
|     <h1>Solo</h1> |     <h1>Solo</h1> | ||||||
|  |     <div class="timer" :gameStatus>{{ timer }}</div> | ||||||
|     <div class="grid"> |     <div class="grid"> | ||||||
|       <div v-for="(row, rowIndex) in cellGrid" :key="rowIndex" class="row"> |       <div v-for="(row, rowIndex) in cellGrid" :key="rowIndex" class="row"> | ||||||
|         <div v-for="(cell, cellIndex) in row" :key="cellIndex"> |         <div v-for="(cell, cellIndex) in row" :key="cellIndex"> | ||||||
| @@ -77,6 +142,13 @@ onUnmounted(() => document.removeEventListener('mousemove', handleGlobalMouseMov | |||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|  |     <div class="grid"> | ||||||
|  |       <div v-for="(row, rowIndex) in cellGrid" :key="rowIndex" class="row"> | ||||||
|  |         <div v-for="(cell, cellIndex) in row" :key="cellIndex"> | ||||||
|  |           <img class="cell" :src="mineGrid.some(mine => mine[0] === cellIndex && mine[1] === rowIndex) ? '/sMine.png' : '/sEmpty.png'" /> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|   </div> |   </div> | ||||||
| </template> | </template> | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user