Solutions and Write-Ups for my Advent Of Code adventures (mainly in Haskell)
2D grid manipulation day :x
The input is separated in two parts: a 2D grid and a list of directions.
I start by creating a data type describing the current state of my grid, it contains:
data World = World { grid :: Array (Int, Int) Char, position :: (Int, Int) } deriving Show
I separate my input when there is two consecutive line breaks. I retrive the grid from the first part as an array indexed with pair of coordinates (starting at 0), and I retrieve the coordinated of the robot represented by @:
type Input = (World, [Char])
parseInput :: String -> Input
parseInput input = (World grid start, moves)
where [gridStr, moveLines] = splitOn "\n\n" input
moves = filter (/= '\n') moveLines
gridLines = lines gridStr
height = length gridLines
width = length $ head gridLines
grid = listArray ((0, 0), (height - 1, width - 1)) $ filter (/= '\n') gridStr
start = fst . head . filter ((== '@') . snd) . assocs $ grid
I want to be able to move my robot by following the directions.
If the robot moves into a box, it should push that box, and propagate that movement to other boxes behind the pushed box.
If this results in trying to push the box into a wall, then no movement happens.
In order to move around, I first need a function describing a movement given a direction:
getStep :: Char -> (Int, Int)
getStep '<' = (0, -1)
getStep '>' = (0, 1)
getStep 'v' = (1, 0)
getStep '^' = (-1, 0)
I also made a helper function to apply that movement to given coordinates:
applyTuple :: (a -> b -> c) -> (a, a) -> (b, b) -> (c, c)
applyTuple f (a, b) (c, d) = (f a c, f b d)
Using this function, getting the next position of the robot can be done like that:
applyTuple (+) (position world) (getStep direction)
Now, what I need to do is to propagate this movement across boxes:
I perform this computation in a BFS way:
-- Propagates the movement along boxes
propagateStep [] v = Set.toAscList v
propagateStep (x : xs) v
| current `elem` "#." = propagateStep xs v
| otherwise = propagateStep xs' v'
where current = g ! x
neighbours | dir `elem` "v^" && current == '[' = nub [applyTuple (+) x (0, 1), applyTuple (+) x step]
| dir `elem` "v^" && current == ']' = nub [applyTuple (-) x (0, 1), applyTuple (+) x step]
| otherwise = [applyTuple (+) x step]
neighbours' = filter (`Set.notMember` v) neighbours
xs' = xs ++ neighbours'
v' = foldr Set.insert v neighbours'
Now that I have found the tiles that are going to be updated, I need to find their new state:
-- Get a tile's new state
getNew p | prev `elem` moves = g ! prev -- If the previous tile was moved, then this tile's new state is the previous tile's
| otherwise = '.' -- Otherwise, this tile is now empty
where prev = applyTuple (-) p step
With that done, all I need to do is to update the grid accordingly:
move :: World -> Char -> World
move world dir
| any ((== '#') . (g !)) moves = world
| otherwise = World g' pos'
where step = getStep dir
g = grid world
pos = position world
pos' = applyTuple (+) pos step
moves = pos : propagateStep [pos'] (Set.singleton pos')
g' = g // [(p, getNew p ) | p <- moves]
-- Get a tile's new state
getNew p | prev `elem` moves = g ! prev -- If the previous tile was moved, then this tile's new state is the previous tile's
| otherwise = '.' -- Otherwise, this tile is now empty
where prev = applyTuple (-) p step
-- Propagates the movement along boxes
propagateStep [] v = Set.toAscList v
propagateStep (x : xs) v
| current `elem` "#." = propagateStep xs v
| otherwise = propagateStep xs' v'
where current = g ! x
neighbours | dir `elem` "v^" && current == '[' = nub [applyTuple (+) x (0, 1), applyTuple (+) x step]
| dir `elem` "v^" && current == ']' = nub [applyTuple (-) x (0, 1), applyTuple (+) x step]
| otherwise = [applyTuple (+) x step]
neighbours' = filter (`Set.notMember` v) neighbours
xs' = xs ++ neighbours'
v' = foldr Set.insert v neighbours'
Now that I am able to move for one step, I also need to be able to compute the total gps score of a given grid.
This is simply done by getting the boxes’ coordinates and applying the computational rule on these coordinates:
computeGPSScore :: Array (Int, Int) Char -> Output
computeGPSScore = sum . map (computeScore . fst) . filter ((`elem` "O[") . snd) . assocs
where computeScore (i, j) = 100 * i + j
Now, I simply need to apply every move from the input to the grid, and then compute the score:
partOne :: Input -> Output
partOne = computeGPSScore . grid . uncurry (foldl move)
Part two is done exactly the same way. The only difference is that I need to expand the tiles first:
expandWorld :: World -> World
expandWorld world = World g' start
where expandChar '#' = "##"
expandChar 'O' = "[]"
expandChar '.' = ".."
expandChar '@' = "@."
g = grid world
(height, width) = snd $ bounds g
width' = (width + 1) * 2
g' = listArray ((0, 0), (height, width' - 1)) . concatMap expandChar $ elems g
start = fst . head . filter ((== '@') . snd) . assocs $ g'
partTwo :: Input -> Output
partTwo = partOne . first expandWorld
My solution is quite slow (about 1s). I suspect that the (//) operator is the reason for this, as it is O(n) where n is the size of the array (so the number of tiles in the grid).
In order to improve that, I see two solutions: