{-# Language ImportQualifiedPost #-}
module Main (main) where
import Advent
import Advent.Coord
import Data.Array.Base qualified as AB
import Data.Array.Unboxed qualified as A
import Data.Maybe (mapMaybe)
type Seating = A.UArray Coord Char
type Neighbors = A.Array Coord [Int]
main :: IO ()
IO ()
main =
do inp <- Int -> Int -> IO (UArray Coord Char)
getInputArray Int
2020 Int
11
let run UArray Coord Char -> Maybe (UArray Coord Char)
f = Int -> IO ()
forall a. Show a => a -> IO ()
print (Char -> [Char] -> Int
forall (f :: * -> *) a. (Foldable f, Eq a) => a -> f a -> Int
count Char
'#' (UArray Coord Char -> [Char]
forall (a :: * -> * -> *) e i. (IArray a e, Ix i) => a i e -> [e]
A.elems ((UArray Coord Char -> Maybe (UArray Coord Char))
-> UArray Coord Char -> UArray Coord Char
forall a. (a -> Maybe a) -> a -> a
stable UArray Coord Char -> Maybe (UArray Coord Char)
f UArray Coord Char
inp)))
run (adv 4 (adjacent inp))
run (adv 5 (lineOfSight inp))
stable :: (a -> Maybe a) -> a -> a
stable :: forall a. (a -> Maybe a) -> a -> a
stable a -> Maybe a
f a
x = a -> (a -> a) -> Maybe a -> a
forall b a. b -> (a -> b) -> Maybe a -> b
maybe a
x ((a -> Maybe a) -> a -> a
forall a. (a -> Maybe a) -> a -> a
stable a -> Maybe a
f) (a -> Maybe a
f a
x)
adjacent :: Seating -> Neighbors
adjacent :: UArray Coord Char -> Neighbors
adjacent UArray Coord Char
a = (Coord, Coord) -> [[Int]] -> Neighbors
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
(i, i) -> [e] -> a i e
A.listArray (Coord, Coord)
b [[(Coord, Coord) -> Coord -> Int
forall a. Ix a => (a, a) -> a -> Int
A.index (Coord, Coord)
b Coord
j | Coord
j <- Coord -> [Coord]
neighbors Coord
i, (Coord, Coord) -> Coord -> Bool
forall a. Ix a => (a, a) -> a -> Bool
A.inRange (Coord, Coord)
b Coord
j] | Coord
i <- (Coord, Coord) -> [Coord]
forall a. Ix a => (a, a) -> [a]
A.range (Coord, Coord)
b]
where
b :: (Coord, Coord)
b = UArray Coord Char -> (Coord, Coord)
forall i. Ix i => UArray i Char -> (i, i)
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> (i, i)
A.bounds UArray Coord Char
a
lineOfSight :: Seating -> Neighbors
lineOfSight :: UArray Coord Char -> Neighbors
lineOfSight UArray Coord Char
a = (Coord, Coord) -> [[Int]] -> Neighbors
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
(i, i) -> [e] -> a i e
A.listArray (Coord, Coord)
b [(Coord -> Maybe Int) -> [Coord] -> [Int]
forall a b. (a -> Maybe b) -> [a] -> [b]
mapMaybe (Coord -> Coord -> Maybe Int
look Coord
i) (Coord -> [Coord]
neighbors Coord
origin) | Coord
i <- (Coord, Coord) -> [Coord]
forall a. Ix a => (a, a) -> [a]
A.range (Coord, Coord)
b]
where
b :: (Coord, Coord)
b = UArray Coord Char -> (Coord, Coord)
forall i. Ix i => UArray i Char -> (i, i)
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> (i, i)
A.bounds UArray Coord Char
a
look :: Coord -> Coord -> Maybe Int
look Coord
i Coord
d =
do let j :: Coord
j = Coord
i Coord -> Coord -> Coord
forall a. Num a => a -> a -> a
+ Coord
d
v <- UArray Coord Char -> Coord -> Maybe Char
forall (a :: * -> * -> *) e i (f :: * -> *).
(IArray a e, Ix i, Alternative f) =>
a i e -> i -> f e
arrIx UArray Coord Char
a Coord
j
case v of
Char
'.' -> Coord -> Coord -> Maybe Int
look Coord
j Coord
d
Char
_ -> Int -> Maybe Int
forall a. a -> Maybe a
Just ((Coord, Coord) -> Coord -> Int
forall a. Ix a => (a, a) -> a -> Int
A.index (Coord, Coord)
b Coord
j)
adv ::
Int ->
Neighbors ->
Seating ->
Maybe Seating
adv :: Int -> Neighbors -> UArray Coord Char -> Maybe (UArray Coord Char)
adv Int
t Neighbors
ns UArray Coord Char
a
| [(Coord, Char)] -> Bool
forall a. [a] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null [(Coord, Char)]
changes = Maybe (UArray Coord Char)
forall a. Maybe a
Nothing
| Bool
otherwise = UArray Coord Char -> Maybe (UArray Coord Char)
forall a. a -> Maybe a
Just (UArray Coord Char -> Maybe (UArray Coord Char))
-> UArray Coord Char -> Maybe (UArray Coord Char)
forall a b. (a -> b) -> a -> b
$! UArray Coord Char
a UArray Coord Char -> [(Coord, Char)] -> UArray Coord Char
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> [(i, e)] -> a i e
A.// [(Coord, Char)]
changes
where
changes :: [(Coord, Char)]
changes = [(Coord
i, Char
v) | (Coord
i, Char
e) <- UArray Coord Char -> [(Coord, Char)]
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> [(i, e)]
A.assocs UArray Coord Char
a, Char
v <- Coord -> Char -> [Char]
valueAt Coord
i Char
e]
occupied :: Int -> Coord -> Bool
occupied :: Int -> Coord -> Bool
occupied Int
n Coord
i = Int -> [Int] -> Bool
forall {t}. (Eq t, Num t) => t -> [Int] -> Bool
occupied1 Int
n (Neighbors
ns Neighbors -> Coord -> [Int]
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> i -> e
A.! Coord
i)
occupied1 :: t -> [Int] -> Bool
occupied1 t
0 [Int]
_ = Bool
True
occupied1 t
_ [] = Bool
False
occupied1 t
n (Int
i:[Int]
is) =
case UArray Coord Char -> Int -> Char
forall i. Ix i => UArray i Char -> Int -> Char
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> Int -> e
AB.unsafeAt UArray Coord Char
a Int
i of
Char
'#' -> t -> [Int] -> Bool
occupied1 (t
nt -> t -> t
forall a. Num a => a -> a -> a
-t
1) [Int]
is
Char
_ -> t -> [Int] -> Bool
occupied1 t
n [Int]
is
valueAt :: Coord -> Char -> [Char]
valueAt Coord
i Char
'#' | Int -> Coord -> Bool
occupied Int
t Coord
i = [Char]
"L"
valueAt Coord
i Char
'L' | Bool -> Bool
not (Int -> Coord -> Bool
occupied Int
1 Coord
i) = [Char]
"#"
valueAt Coord
_ Char
_ = [Char]
""