{-# LANGUAGE ImportQualifiedPost #-}
module Main (main) where
import Data.Array.Unboxed qualified as A
import Data.Map (Map)
import Data.Map.Strict qualified as Map
import Data.Maybe (maybeToList)
import Advent.Coord
import Advent.Input (getInputArray)
data Turn = NextL | NextR | NextS deriving Int -> Turn -> ShowS
[Turn] -> ShowS
Turn -> String
(Int -> Turn -> ShowS)
-> (Turn -> String) -> ([Turn] -> ShowS) -> Show Turn
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> Turn -> ShowS
showsPrec :: Int -> Turn -> ShowS
$cshow :: Turn -> String
show :: Turn -> String
$cshowList :: [Turn] -> ShowS
showList :: [Turn] -> ShowS
Show
data Cart = Cart !Velocity !Turn deriving Int -> Cart -> ShowS
[Cart] -> ShowS
Cart -> String
(Int -> Cart -> ShowS)
-> (Cart -> String) -> ([Cart] -> ShowS) -> Show Cart
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> Cart -> ShowS
showsPrec :: Int -> Cart -> ShowS
$cshow :: Cart -> String
show :: Cart -> String
$cshowList :: [Cart] -> ShowS
showList :: [Cart] -> ShowS
Show
type Velocity = Coord
newtype Road = Road (A.UArray Coord Char)
type CartQueue = Map Coord Cart
main :: IO ()
IO ()
main =
do road <- UArray Coord Char -> Road
Road (UArray Coord Char -> Road) -> IO (UArray Coord Char) -> IO Road
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Int -> Int -> IO (UArray Coord Char)
getInputArray Int
2018 Int
13
let carts = Road -> CartQueue
findCarts Road
road
putStrLn (part1 road carts)
putStrLn (part2 road carts)
format :: Coord -> String
format :: Coord -> String
format (C Int
y Int
x) = Int -> String
forall a. Show a => a -> String
show Int
x String -> ShowS
forall a. [a] -> [a] -> [a]
++ String
"," String -> ShowS
forall a. [a] -> [a] -> [a]
++ Int -> String
forall a. Show a => a -> String
show Int
y
part1 :: Road -> CartQueue -> String
part1 :: Road -> CartQueue -> String
part1 Road
road CartQueue
carts = Coord -> String
format ((Coord -> CartQueue -> CartQueue -> Coord)
-> Road -> CartQueue -> Coord
simulate (\Coord
pos CartQueue
_ CartQueue
_ -> Coord
pos) Road
road CartQueue
carts)
part2 :: Road -> CartQueue -> String
part2 :: Road -> CartQueue -> String
part2 Road
road CartQueue
carts = Coord -> String
format ((Coord -> CartQueue -> CartQueue -> Coord)
-> Road -> CartQueue -> Coord
simulate Coord -> CartQueue -> CartQueue -> Coord
onCollision Road
road CartQueue
carts)
where
onCollision :: Coord -> CartQueue -> CartQueue -> Coord
onCollision Coord
pos CartQueue
ready CartQueue
done =
(Coord -> CartQueue -> CartQueue -> Coord)
-> Road -> CartQueue -> CartQueue -> Coord
tick Coord -> CartQueue -> CartQueue -> Coord
onCollision Road
road (Coord -> CartQueue -> CartQueue
forall k a. Ord k => k -> Map k a -> Map k a
Map.delete Coord
pos CartQueue
ready) (Coord -> CartQueue -> CartQueue
forall k a. Ord k => k -> Map k a -> Map k a
Map.delete Coord
pos CartQueue
done)
indexRoad :: Road -> Coord -> Char
indexRoad :: Road -> Coord -> Char
indexRoad (Road UArray Coord Char
v) Coord
c = UArray Coord Char
v UArray Coord Char -> Coord -> Char
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> i -> e
A.! Coord
c
findCarts :: Road -> CartQueue
findCarts :: Road -> CartQueue
findCarts (Road UArray Coord Char
rs) =
[(Coord, Cart)] -> CartQueue
forall k a. Ord k => [(k, a)] -> Map k a
Map.fromList
[ (Coord
pos, Coord -> Turn -> Cart
Cart Coord
vel Turn
NextL)
| (Coord
pos, Char
c) <- UArray Coord Char -> [(Coord, Char)]
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> [(i, e)]
A.assocs UArray Coord Char
rs
, Coord
vel <- Maybe Coord -> [Coord]
forall a. Maybe a -> [a]
maybeToList (Char -> Maybe Coord
charToVec Char
c)
]
simulate ::
(Coord -> CartQueue -> CartQueue -> Coord)
->
Road ->
CartQueue ->
Coord
simulate :: (Coord -> CartQueue -> CartQueue -> Coord)
-> Road -> CartQueue -> Coord
simulate Coord -> CartQueue -> CartQueue -> Coord
onCollision Road
road CartQueue
carts
| [Coord
pos] <- CartQueue -> [Coord]
forall k a. Map k a -> [k]
Map.keys CartQueue
carts = Coord
pos
| Bool
otherwise = (Coord -> CartQueue -> CartQueue -> Coord)
-> Road -> CartQueue -> CartQueue -> Coord
tick Coord -> CartQueue -> CartQueue -> Coord
onCollision Road
road CartQueue
carts CartQueue
forall k a. Map k a
Map.empty
tick ::
(Coord -> CartQueue -> CartQueue -> Coord)
->
Road ->
CartQueue ->
CartQueue ->
Coord
tick :: (Coord -> CartQueue -> CartQueue -> Coord)
-> Road -> CartQueue -> CartQueue -> Coord
tick Coord -> CartQueue -> CartQueue -> Coord
onCollision Road
road CartQueue
carts CartQueue
done =
case CartQueue -> Maybe ((Coord, Cart), CartQueue)
forall k a. Map k a -> Maybe ((k, a), Map k a)
Map.minViewWithKey CartQueue
carts of
Maybe ((Coord, Cart), CartQueue)
Nothing -> (Coord -> CartQueue -> CartQueue -> Coord)
-> Road -> CartQueue -> Coord
simulate Coord -> CartQueue -> CartQueue -> Coord
onCollision Road
road CartQueue
done
Just ((Coord
pos, Cart
cart), CartQueue
carts')
| Bool
collision -> Coord -> CartQueue -> CartQueue -> Coord
onCollision Coord
pos' CartQueue
carts' CartQueue
done
| Bool
otherwise -> (Coord -> CartQueue -> CartQueue -> Coord)
-> Road -> CartQueue -> CartQueue -> Coord
tick Coord -> CartQueue -> CartQueue -> Coord
onCollision Road
road CartQueue
carts' (Coord -> Cart -> CartQueue -> CartQueue
forall k a. Ord k => k -> a -> Map k a -> Map k a
Map.insert Coord
pos' Cart
cart' CartQueue
done)
where
collision :: Bool
collision = Coord -> CartQueue -> Bool
forall k a. Ord k => k -> Map k a -> Bool
Map.member Coord
pos' CartQueue
done Bool -> Bool -> Bool
|| Coord -> CartQueue -> Bool
forall k a. Ord k => k -> Map k a -> Bool
Map.member Coord
pos' CartQueue
carts'
(Coord
pos', Cart
cart') = Road -> Coord -> Cart -> (Coord, Cart)
drive Road
road Coord
pos Cart
cart
drive :: Road -> Coord -> Cart -> (Coord, Cart)
drive :: Road -> Coord -> Cart -> (Coord, Cart)
drive Road
road Coord
pos (Cart Coord
vel Turn
next) = (Coord
pos', Cart
cart')
where
pos' :: Coord
pos' = Coord
pos Coord -> Coord -> Coord
forall a. Num a => a -> a -> a
+ Coord
vel
cart' :: Cart
cart' =
case Road -> Coord -> Char
indexRoad Road
road Coord
pos' of
Char
'\\' -> Coord -> Turn -> Cart
Cart (Coord -> Coord
invert Coord
vel) Turn
next
Char
'/' -> Coord -> Turn -> Cart
Cart (Coord -> Coord
invert' Coord
vel) Turn
next
Char
'+' -> Coord -> Turn -> Cart
Cart (Turn -> Coord -> Coord
turn Turn
next Coord
vel) (Turn -> Turn
nextTurn Turn
next)
Char
_ -> Coord -> Turn -> Cart
Cart Coord
vel Turn
next
turn :: Turn -> Velocity -> Velocity
turn :: Turn -> Coord -> Coord
turn Turn
NextL = Coord -> Coord
turnLeft
turn Turn
NextR = Coord -> Coord
turnRight
turn Turn
NextS = Coord -> Coord
forall a. a -> a
id
nextTurn :: Turn -> Turn
nextTurn :: Turn -> Turn
nextTurn Turn
NextL = Turn
NextS
nextTurn Turn
NextS = Turn
NextR
nextTurn Turn
NextR = Turn
NextL