main.hs 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  1. -- Mine Sweeper Lab
  2. -- Import section
  3. import Data.Char
  4. import Data.Set ( Set )
  5. import qualified Data.Set as S
  6. import Data.Universe.Helpers ( cartesianProduct ) -- Need to be installed : `cabal install universe`
  7. import Data.List ( foldl1' )
  8. import Systeme.IO
  9. -- Types
  10. data Cell = Covered Int Bool Bool -- number of bombs around ; has bomb ; has flag
  11. | Uncovered Int -- number of bombs around
  12. | Selected
  13. data Grid = Grid [[Cell]]
  14. data StdGen = SG -- TODO : replace
  15. type Mat = [[Int]]
  16. -- Methods
  17. instance Show Cell where
  18. show ( Covered _ _ hasFlag ) = if hasFlag then "[F]" else "[ ]"
  19. show ( Uncovered n ) = ['(', intToDigit n, ')']
  20. show Selected = ">x<"
  21. instance Show Grid where
  22. show (Grid a) = unlines $ map (unwords . map show) a
  23. -- Functions
  24. randSet::Int -> StdGen -> StdGen -> Int -> Int -> Set (Int, Int)
  25. randSet n sg1 sg2 h w =
  26. let byl = randomRs (0, h - 1) sg1 in
  27. let bxl = randomRs (0, w - 1) sg2 in
  28. let bl = zip byl bxl in
  29. let biggerSets = scanl (flip S.insert) S.empty bl in
  30. head (dropWhile (\s -> S.size s < n) biggerSets)
  31. grid::Int -> Int -> Set (Int, Int) -> Grid
  32. grid h w s = Grid [ [Covered 0 (S.member (y, x) s) False | x <- [0..w-1] ] | y <- [0..h-1] ]
  33. randomRs (x, y) sg = [x + mod k (y - x) | k <- [1..y] ] -- TODO : replace
  34. mineIndic::Cell -> Int
  35. mineIndic c = let Covered _ b _ = c in if b then 1 else 0
  36. mines::Grid -> Mat
  37. mines g = let Grid m = g in map (map mineIndic) m
  38. moveUp::Mat -> Mat
  39. moveUp m = concat [tail m, [[0 | _ <- [1..length (m!!0)]]]]
  40. moveDown::Mat -> Mat
  41. moveDown m = concat [[[0 | _ <- [1..length (m!!0)]]], init m]
  42. moveRight::Mat -> Mat
  43. moveRight m = map (\l -> 0 : init l) m
  44. moveLeft::Mat -> Mat
  45. moveLeft m = map (\l -> concat [tail l, [0]]) m
  46. gridMoves::Mat -> [Mat]
  47. gridMoves m =
  48. let hMov = [id, moveUp, moveDown] in
  49. let wMov = [id, moveLeft, moveRight] in
  50. let allMov = tail $ cartesianProduct (.) hMov wMov in
  51. map (\f -> f m) allMov
  52. matrixSum::Mat -> Mat -> Mat
  53. matrixSum m1 m2 = zipWith (zipWith (+)) m1 m2
  54. neighbourMap::Grid -> Mat
  55. neighbourMap g =
  56. let Grid cm = g in
  57. let m = map (map mineIndic) cm in
  58. foldl1' matrixSum (gridMoves m)
  59. updateCell::Cell -> Int -> Cell
  60. updateCell c n = case c of
  61. Covered _ bomb flag -> Covered n bomb flag
  62. Uncovered _ -> Uncovered n
  63. Selected -> Selected
  64. updateGrid::Grid -> Mat -> Grid
  65. updateGrid g m =
  66. let Grid cm = g in
  67. Grid (zipWith (zipWith updateCell) cm m)
  68. applyi::(a -> a) -> Int -> [a] -> [a]
  69. applyi f i xs =
  70. let (xs1, xs2) = splitAt i xs in
  71. concat [xs1, ( (f (head xs2)) : tail xs2 )]
  72. applyij::(a -> a) -> Int -> Int -> [[a]] -> [[a]]
  73. applyij f i j xss = applyi (applyi f j) i xss
  74. uncover::(Int, Int) -> Grid -> Grid
  75. uncover (i, j) g =
  76. let Grid cm = g in
  77. let neighbours = tail [(i', j') | i' <- [0, -1, 1], j' <- [0, -1, 1]] in
  78. let uncoverOneCell = \c -> case c of
  79. Covered 0 _ _ -> Uncovered 0 -- TODO : recursion
  80. Covered n _ _ -> Uncovered n
  81. in
  82. Grid (applyij uncoverOneCell i j cm)
  83. -- 5) Main Loop
  84. covIndic::Cell -> Int
  85. covIndic (Covered _ _ _) = 1
  86. covIndic Selected = 1
  87. covIndic _ = 0
  88. won::Grid -> Int -> Bool
  89. won g nbMines =
  90. let Grid cm = g in
  91. let nbTotal = (length cm) * (length (cm!!0)) in
  92. let uncoBinMatrix = map (map covIndic) cm in
  93. let nbUnco = sum (map sum uncoBinMatrix) in
  94. nbUnco == nbTotal - nbMines
  95. toggleFlag::Cell -> Cell
  96. toggleFlag (Covered n b flag) = Covered n b (not flag)
  97. toggleFlag c = c
  98. loop::Int -> Int -> Int -> Grid -> IO ()
  99. loop i j n b@(Grid xs)
  100. | won b n = putStrLn "Victory !"
  101. | otherwise = do
  102. putStrLn $ show $ Grid $ applyij ( const Selected ) i j xs
  103. c <- getChar
  104. case c of
  105. 'i' -> loop (max (i - 1) 0) j n b -- move caret up
  106. 'k' -> loop (min (i + 1) ((length xs) - 1)) j n b -- move caret down
  107. 'j' -> loop i (max (j - 1) 0) n b -- move caret left
  108. 'l' -> loop i (min (j + 1) ((length (xs!!0)) - 1)) n b -- move caret right
  109. 'f' -> loop i j n (Grid $ applyij toggleFlag i j xs) -- toggle flag
  110. 'u' -> loop i j n (uncover (i, j) b) -- uncover tile
  111. otherwise -> loop i j n b -- do nothing
  112. main::IO()
  113. main = do
  114. hSetBuffering stdin NoBuffering -- disable the need to press enter
  115. hSetEcho stdin False -- dont print input in terminal
  116. -- random sequences
  117. sg <- newStdGen
  118. sg' <- newStdGen
  119. -- parameters
  120. let nmines = 5
  121. let h = 7
  122. let w = 10
  123. ... -- create grid, add mines, update neighbours
  124. loop ... -- start loop
  125. -- Testing data
  126. dtTinyGrid = Grid [
  127. [Covered 1 True False, Uncovered 2, Covered 0 True False],
  128. [Uncovered 0, Uncovered 0, Covered 1 True True]
  129. ]
  130. dtRS = randSet 3 SG SG 4 5
  131. dtGridCover = grid 4 5 dtRS
  132. dtMines = mines dtGridCover