{-# Language QuasiQuotes, NumericUnderscores, ImportQualifiedPost #-}
module Main where
import Data.Array.Unboxed (Array, (!), listArray, rangeSize, bounds)
import Data.Map qualified as Map
import Data.Set (Set)
import Data.Set qualified as Set
import Advent (format)
import Advent.Coord (Coord(C), coordRow, east, west, cardinal, coordCol, south)
import Advent.Search (bfsN)
pieces :: Array Int (Set Coord)
pieces :: Array Int (Set Coord)
pieces = (Int, Int) -> [Set Coord] -> Array Int (Set Coord)
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
(i, i) -> [e] -> a i e
listArray (Int
0,Int
4) [
[Coord] -> Set Coord
forall a. Ord a => [a] -> Set a
Set.fromList [Int -> Int -> Coord
C Int
0 Int
0, Int -> Int -> Coord
C Int
0 Int
1, Int -> Int -> Coord
C Int
0 Int
2, Int -> Int -> Coord
C Int
0 Int
3],
[Coord] -> Set Coord
forall a. Ord a => [a] -> Set a
Set.fromList [Int -> Int -> Coord
C (-Int
2) Int
1, Int -> Int -> Coord
C (-Int
1) Int
0, Int -> Int -> Coord
C (-Int
1) Int
2, Int -> Int -> Coord
C Int
0 Int
1],
[Coord] -> Set Coord
forall a. Ord a => [a] -> Set a
Set.fromList [Int -> Int -> Coord
C Int
0 Int
0, Int -> Int -> Coord
C Int
0 Int
1, Int -> Int -> Coord
C Int
0 Int
2, Int -> Int -> Coord
C (-Int
1) Int
2, Int -> Int -> Coord
C (-Int
2) Int
2],
[Coord] -> Set Coord
forall a. Ord a => [a] -> Set a
Set.fromList [Int -> Int -> Coord
C Int
0 Int
0, Int -> Int -> Coord
C (-Int
1) Int
0, Int -> Int -> Coord
C (-Int
2) Int
0, Int -> Int -> Coord
C (-Int
3) Int
0],
[Coord] -> Set Coord
forall a. Ord a => [a] -> Set a
Set.fromList [Int -> Int -> Coord
C (-Int
1) Int
0, Int -> Int -> Coord
C (-Int
1) Int
1, Int -> Int -> Coord
C Int
0 Int
0, Int -> Int -> Coord
C Int
0 Int
1]
]
initialStuff :: Set Coord
initialStuff :: Set Coord
initialStuff = [Coord] -> Set Coord
forall a. Ord a => [a] -> Set a
Set.fromList [Int -> Int -> Coord
C Int
0 Int
x | Int
x <- [Int
0..Int
6]]
main :: IO ()
IO ()
main =
do moves <- (Char -> Coord) -> [Char] -> [Coord]
forall a b. (a -> b) -> [a] -> [b]
map Char -> Coord
dir ([Char] -> [Coord]) -> IO [Char] -> IO [Coord]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> [format|2022 17 %s%n|]
let movesArray = (Int, Int) -> [Coord] -> Array Int Coord
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
(i, i) -> [e] -> a i e
listArray (Int
0, [Coord] -> Int
forall a. [a] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length [Coord]
movesInt -> Int -> Int
forall a. Num a => a -> a -> a
-Int
1) [Coord]
moves
let states = ((Int, Int, Set Coord) -> (Int, Int, Set Coord))
-> (Int, Int, Set Coord) -> [(Int, Int, Set Coord)]
forall a. (a -> a) -> a -> [a]
iterate (Array Int Coord -> (Int, Int, Set Coord) -> (Int, Int, Set Coord)
place Array Int Coord
movesArray) (Int
0, Int
0, Set Coord
initialStuff)
let heightAt Int
i = case [(Int, Int, Set Coord)]
states [(Int, Int, Set Coord)] -> Int -> (Int, Int, Set Coord)
forall a. HasCallStack => [a] -> Int -> a
!! Int
i of (Int
_,Int
_,Set Coord
stuff) -> Set Coord -> Int
height Set Coord
stuff
print (heightAt 2022)
let (cyc1,cyc2) = findCycle [(i,j,normalize stuff) | (i,j,stuff) <- states]
let cycLen = Int
cyc2 Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
cyc1
let (cycCnt, overflow) = (1_000_000_000_000 - cyc1) `divMod` cycLen
let cycHeight = Int -> Int
heightAt Int
cyc2 Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int -> Int
heightAt Int
cyc1
print (heightAt (cyc2 + overflow) + cycHeight * (cycCnt-1))
height :: Set Coord -> Int
height :: Set Coord -> Int
height Set Coord
stuff = - Coord -> Int
coordRow (Set Coord -> Coord
forall a. Ord a => Set a -> a
forall (t :: * -> *) a. (Foldable t, Ord a) => t a -> a
minimum Set Coord
stuff)
normalize :: Set Coord -> Set Coord
normalize :: Set Coord -> Set Coord
normalize Set Coord
stuff = Set Coord -> Coord -> Set Coord
translate Set Coord
stuff (Int -> Int -> Coord
C (Set Coord -> Int
height Set Coord
stuff) Int
0)
findCycle :: Ord a => [a] -> (Int,Int)
findCycle :: forall a. Ord a => [a] -> (Int, Int)
findCycle = Map a Int -> Int -> [a] -> (Int, Int)
forall {k} {b}. (Ord k, Num b) => Map k b -> b -> [k] -> (b, b)
go Map a Int
forall k a. Map k a
Map.empty Int
0
where
go :: Map k b -> b -> [k] -> (b, b)
go Map k b
_ b
_ [] = [Char] -> (b, b)
forall a. HasCallStack => [Char] -> a
error [Char]
"no cycle"
go Map k b
seen b
i (k
x:[k]
xs) =
case k -> Map k b -> Maybe b
forall k a. Ord k => k -> Map k a -> Maybe a
Map.lookup k
x Map k b
seen of
Maybe b
Nothing -> Map k b -> b -> [k] -> (b, b)
go (k -> b -> Map k b -> Map k b
forall k a. Ord k => k -> a -> Map k a -> Map k a
Map.insert k
x b
i Map k b
seen)(b
ib -> b -> b
forall a. Num a => a -> a -> a
+b
1) [k]
xs
Just b
j -> (b
j,b
i)
dir :: Char -> Coord
dir :: Char -> Coord
dir Char
'>' = Coord
east
dir Char
'<' = Coord
west
dir Char
_ = [Char] -> Coord
forall a. HasCallStack => [Char] -> a
error [Char]
"bad dir"
inWalls :: Coord -> Bool
inWalls :: Coord -> Bool
inWalls (C Int
_ Int
x) = Int
0 Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
x Bool -> Bool -> Bool
&& Int
x Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
6
translate :: Set Coord -> Coord -> Set Coord
translate :: Set Coord -> Coord -> Set Coord
translate Set Coord
p Coord
c = (Coord -> Coord) -> Set Coord -> Set Coord
forall a b. (a -> b) -> Set a -> Set b
Set.mapMonotonic (Coord
cCoord -> Coord -> Coord
forall a. Num a => a -> a -> a
+) Set Coord
p
clean :: Set Coord -> Set Coord
clean :: Set Coord -> Set Coord
clean Set Coord
stuff = (Coord -> Bool) -> Set Coord -> Set Coord
forall a. (a -> Bool) -> Set a -> Set a
Set.filter Coord -> Bool
alive Set Coord
stuff
where
ymin :: Int
ymin = Coord -> Int
coordRow (Set Coord -> Coord
forall a. Ord a => Set a -> a
forall (t :: * -> *) a. (Foldable t, Ord a) => t a -> a
minimum Set Coord
stuff)
step :: Coord -> [Coord]
step Coord
c = [Coord
n | Coord
n <- Coord -> [Coord]
cardinal Coord
c, Int
0 Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Coord -> Int
coordCol Coord
n, Coord -> Int
coordCol Coord
n Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
6, Coord -> Int
coordRow Coord
c Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
>= Int
ymin, Coord -> Set Coord -> Bool
forall a. Ord a => a -> Set a -> Bool
Set.notMember Coord
n Set Coord
stuff]
air :: [Coord]
air = (Coord -> [Coord]) -> [Coord] -> [Coord]
forall a. Ord a => (a -> [a]) -> [a] -> [a]
bfsN Coord -> [Coord]
step [Int -> Int -> Coord
C Int
ymin Int
x | Int
x <- [Int
0..Int
6], Coord -> Set Coord -> Bool
forall a. Ord a => a -> Set a -> Bool
Set.notMember (Int -> Int -> Coord
C Int
ymin Int
x) Set Coord
stuff]
alive :: Coord -> Bool
alive Coord
x = (Coord -> Bool) -> [Coord] -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
any (Coord -> [Coord] -> Bool
forall a. Eq a => a -> [a] -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool
`elem` [Coord]
air) (Coord -> [Coord]
cardinal Coord
x) Bool -> Bool -> Bool
|| Coord -> Int
coordRow Coord
x Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
ymin
place ::
Array Int Coord ->
(Int, Int, Set Coord) ->
(Int, Int, Set Coord)
place :: Array Int Coord -> (Int, Int, Set Coord) -> (Int, Int, Set Coord)
place Array Int Coord
jets (Int
i,Int
j,Set Coord
stuff) =
case Int -> Set Coord -> (Set Coord, Int)
drive Int
j Set Coord
start of
(Set Coord
stuck, Int
j') -> (Int
i', Int
j', Set Coord -> Set Coord
clean (Set Coord -> Set Coord -> Set Coord
forall a. Ord a => Set a -> Set a -> Set a
Set.union Set Coord
stuff Set Coord
stuck))
where
i' :: Int
i' = (Int
iInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1)Int -> Int -> Int
forall a. Integral a => a -> a -> a
`mod`Int
5
start :: Set Coord
start = Set Coord -> Coord -> Set Coord
translate (Array Int (Set Coord)
pieces Array Int (Set Coord) -> Int -> Set Coord
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> i -> e
! Int
i) (Int -> Int -> Coord
C (-Set Coord -> Int
height Set Coord
stuffInt -> Int -> Int
forall a. Num a => a -> a -> a
-Int
4) Int
2)
isCrashed :: Set Coord -> Bool
isCrashed Set Coord
piece = Bool -> Bool
not ((Coord -> Bool) -> Set Coord -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
all Coord -> Bool
inWalls Set Coord
piece Bool -> Bool -> Bool
&& Set Coord -> Set Coord -> Bool
forall a. Ord a => Set a -> Set a -> Bool
Set.disjoint Set Coord
stuff Set Coord
piece)
drive :: Int -> Set Coord -> (Set Coord, Int)
drive Int
dj Set Coord
p1
| Set Coord -> Set Coord -> Bool
forall a. Ord a => Set a -> Set a -> Bool
Set.disjoint Set Coord
stuff Set Coord
p4 = Int -> Set Coord -> (Set Coord, Int)
drive Int
dj' Set Coord
p4
| Bool
otherwise = (Set Coord
p3, Int
dj')
where
dj' :: Int
dj' = (Int
djInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1) Int -> Int -> Int
forall a. Integral a => a -> a -> a
`mod` (Int, Int) -> Int
forall a. Ix a => (a, a) -> Int
rangeSize (Array Int Coord -> (Int, Int)
forall i. Ix i => Array i Coord -> (i, i)
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> (i, i)
bounds Array Int Coord
jets)
p2 :: Set Coord
p2 = Set Coord -> Coord -> Set Coord
translate Set Coord
p1 (Array Int Coord
jets Array Int Coord -> Int -> Coord
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> i -> e
! Int
dj)
p3 :: Set Coord
p3 | Set Coord -> Bool
isCrashed Set Coord
p2 = Set Coord
p1
| Bool
otherwise = Set Coord
p2
p4 :: Set Coord
p4 = Set Coord -> Coord -> Set Coord
translate Set Coord
p3 Coord
south