3 import qualified Data.Set as S
5 data Cell = Cell { level :: Int
8 } deriving (Show, Eq, Ord)
18 let finalGrid = head $ drop 200 $ iterate update grid0
19 print $ S.size finalGrid
23 do gs <- readFile "data/advent24.txt"
25 let isBug r c = (grid!!(r - 1))!!(c - 1) == '#'
27 return $ S.fromList [Cell {..} | row <- [1..gridSize], column <- [1..gridSize], isBug row column]
29 neighbourSpaces :: Cell -> Grid
30 neighbourSpaces cell =
31 ( (neighbourSpacesLeft cell)
32 <> (neighbourSpacesRight cell)
33 <> (neighbourSpacesAbove cell)
34 <> (neighbourSpacesBelow cell)
37 neighbourSpacesLeft :: Cell -> Grid
38 neighbourSpacesLeft (Cell {..})
39 | column == 4 && row == 3 = S.fromList [ Cell { level = (level + 1),
41 | r <- [1..gridSize] ]
42 | column == 1 = S.singleton ( Cell { level = (level - 1),
44 | otherwise = S.singleton ( Cell { column = (column - 1), ..})
46 neighbourSpacesRight :: Cell -> Grid
47 neighbourSpacesRight (Cell {..})
48 | column == 2 && row == 3 = S.fromList [ Cell { level = (level + 1),
50 | r <- [1..gridSize] ]
51 | column == 5 = S.singleton ( Cell { level = (level - 1),
53 | otherwise = S.singleton ( Cell { column = (column + 1), ..})
55 neighbourSpacesAbove :: Cell -> Grid
56 neighbourSpacesAbove (Cell {..})
57 | row == 4 && column == 3 = S.fromList [ Cell { level = (level + 1),
59 | c <- [1..gridSize] ]
60 | row == 1 = S.singleton ( Cell { level = (level - 1),
62 | otherwise = S.singleton ( Cell { row = (row - 1), ..})
64 neighbourSpacesBelow :: Cell -> Grid
65 neighbourSpacesBelow (Cell {..})
66 | row == 2 && column == 3 = S.fromList [ Cell { level = (level + 1),
68 | c <- [1..gridSize] ]
69 | row == 5 = S.singleton ( Cell { level = (level - 1),
71 | otherwise = S.singleton ( Cell { row = (row + 1), ..})
74 countOccupiedNeighbours :: Cell -> Grid -> Int
75 countOccupiedNeighbours cell grid = S.size $ S.intersection grid $ neighbourSpaces cell
77 bugSurvives :: Grid -> Cell -> Bool
78 bugSurvives grid cell = alive && oneNeighbour
79 where alive = cell `S.member` grid
80 oneNeighbour = (countOccupiedNeighbours cell grid) == 1
82 bugBorn :: Grid -> Cell -> Bool
83 bugBorn grid cell = dead && (nNbrs == 1 || nNbrs == 2)
84 where dead = cell `S.notMember` grid
85 nNbrs = countOccupiedNeighbours cell grid
87 update :: Grid -> Grid
88 update grid = S.union (S.filter (bugSurvives grid) bugs) (S.filter (bugBorn grid) empties)
90 empties = (S.foldr mergeEmpties S.empty grid) `S.difference` bugs
91 mergeEmpties cell acc = S.union acc $ neighbourSpaces cell