{-# Language QuasiQuotes, ImportQualifiedPost, TemplateHaskell #-}
module Main where
import Advent (format, partialSums, stageTH)
import Advent.Coord (Coord, manhattan, north, origin, turnLeft, turnRight)
import Data.List (mapAccumL)
import Data.Set qualified as Set
data D = DL | DR
stageTH
main :: IO ()
IO ()
main =
do cmds <- [format|2016 1 (@D%d)&(, )%n|]
let path = [(D, Int)] -> [Coord]
computePath [(D, Int)]
cmds
print (part1 path)
print (part2 path)
part1 :: [Coord] -> Int
part1 :: [Coord] -> Int
part1 = Coord -> Coord -> Int
manhattan Coord
origin (Coord -> Int) -> ([Coord] -> Coord) -> [Coord] -> Int
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Coord] -> Coord
forall a. HasCallStack => [a] -> a
last
part2 :: [Coord] -> Maybe Int
part2 :: [Coord] -> Maybe Int
part2 = (Coord -> Int) -> Maybe Coord -> Maybe Int
forall a b. (a -> b) -> Maybe a -> Maybe b
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (Coord -> Coord -> Int
manhattan Coord
origin) (Maybe Coord -> Maybe Int)
-> ([Coord] -> Maybe Coord) -> [Coord] -> Maybe Int
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Coord] -> Maybe Coord
forall a. Ord a => [a] -> Maybe a
duplicate
computePath :: [(D,Int)] -> [Coord]
computePath :: [(D, Int)] -> [Coord]
computePath = [Coord] -> [Coord]
forall a. Num a => [a] -> [a]
partialSums ([Coord] -> [Coord])
-> ([(D, Int)] -> [Coord]) -> [(D, Int)] -> [Coord]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Coord -> [(D, Int)] -> [Coord]
toSteps Coord
north
duplicate :: Ord a => [a] -> Maybe a
duplicate :: forall a. Ord a => [a] -> Maybe a
duplicate = Set a -> [a] -> Maybe a
forall {a}. Ord a => Set a -> [a] -> Maybe a
aux Set a
forall a. Set a
Set.empty
where
aux :: Set a -> [a] -> Maybe a
aux Set a
_ [] = Maybe a
forall a. Maybe a
Nothing
aux Set a
seen (a
x:[a]
xs)
| a -> Set a -> Bool
forall a. Ord a => a -> Set a -> Bool
Set.member a
x Set a
seen = a -> Maybe a
forall a. a -> Maybe a
Just a
x
| Bool
otherwise = Set a -> [a] -> Maybe a
aux (a -> Set a -> Set a
forall a. Ord a => a -> Set a -> Set a
Set.insert a
x Set a
seen) [a]
xs
toSteps ::
Coord ->
[(D,Int)] ->
[Coord]
toSteps :: Coord -> [(D, Int)] -> [Coord]
toSteps Coord
dir0 [(D, Int)]
cmds = [[Coord]] -> [Coord]
forall (t :: * -> *) a. Foldable t => t [a] -> [a]
concat ((Coord, [[Coord]]) -> [[Coord]]
forall a b. (a, b) -> b
snd ((Coord -> (D, Int) -> (Coord, [Coord]))
-> Coord -> [(D, Int)] -> (Coord, [[Coord]])
forall (t :: * -> *) s a b.
Traversable t =>
(s -> a -> (s, b)) -> s -> t a -> (s, t b)
mapAccumL Coord -> (D, Int) -> (Coord, [Coord])
aux Coord
dir0 [(D, Int)]
cmds))
where
aux :: Coord -> (D, Int) -> (Coord, [Coord])
aux Coord
dir (D
lr, Int
steps) =
let dir' :: Coord
dir' = D -> Coord -> Coord
turn D
lr Coord
dir
in (Coord
dir', Int -> Coord -> [Coord]
forall a. Int -> a -> [a]
replicate Int
steps Coord
dir')
turn :: D -> Coord -> Coord
turn :: D -> Coord -> Coord
turn D
DL = Coord -> Coord
turnLeft
turn D
DR = Coord -> Coord
turnRight