{-# Language MultiWayIf, ImportQualifiedPost #-}
module Main (main) where
import Advent ( getInputArray, same )
import Advent.Coord ( cardinal, Coord )
import Data.Array.Unboxed ( UArray, (!), amap, assocs )
import Data.List ( foldl', minimumBy )
import Data.Map (Map)
import Data.Map qualified as Map
import Data.Maybe ( fromMaybe )
import Data.Ord ( comparing )
import Data.Set (Set)
import Data.Set qualified as Set
data Team = Elf | Goblin
deriving (Team -> Team -> Bool
(Team -> Team -> Bool) -> (Team -> Team -> Bool) -> Eq Team
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: Team -> Team -> Bool
== :: Team -> Team -> Bool
$c/= :: Team -> Team -> Bool
/= :: Team -> Team -> Bool
Eq, Eq Team
Eq Team =>
(Team -> Team -> Ordering)
-> (Team -> Team -> Bool)
-> (Team -> Team -> Bool)
-> (Team -> Team -> Bool)
-> (Team -> Team -> Bool)
-> (Team -> Team -> Team)
-> (Team -> Team -> Team)
-> Ord Team
Team -> Team -> Bool
Team -> Team -> Ordering
Team -> Team -> Team
forall a.
Eq a =>
(a -> a -> Ordering)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> a)
-> (a -> a -> a)
-> Ord a
$ccompare :: Team -> Team -> Ordering
compare :: Team -> Team -> Ordering
$c< :: Team -> Team -> Bool
< :: Team -> Team -> Bool
$c<= :: Team -> Team -> Bool
<= :: Team -> Team -> Bool
$c> :: Team -> Team -> Bool
> :: Team -> Team -> Bool
$c>= :: Team -> Team -> Bool
>= :: Team -> Team -> Bool
$cmax :: Team -> Team -> Team
max :: Team -> Team -> Team
$cmin :: Team -> Team -> Team
min :: Team -> Team -> Team
Ord, Int -> Team -> ShowS
[Team] -> ShowS
Team -> String
(Int -> Team -> ShowS)
-> (Team -> String) -> ([Team] -> ShowS) -> Show Team
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> Team -> ShowS
showsPrec :: Int -> Team -> ShowS
$cshow :: Team -> String
show :: Team -> String
$cshowList :: [Team] -> ShowS
showList :: [Team] -> ShowS
Show)
data Unit = Unit { Unit -> Int
attack, Unit -> Int
hp :: !Int, Unit -> Team
team :: !Team }
deriving (Int -> Unit -> ShowS
[Unit] -> ShowS
Unit -> String
(Int -> Unit -> ShowS)
-> (Unit -> String) -> ([Unit] -> ShowS) -> Show Unit
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> Unit -> ShowS
showsPrec :: Int -> Unit -> ShowS
$cshow :: Unit -> String
show :: Unit -> String
$cshowList :: [Unit] -> ShowS
showList :: [Unit] -> ShowS
Show)
type Dungeon = UArray Coord Bool
main :: IO ()
IO ()
main =
do input <- Int -> Int -> IO (UArray Coord Char)
getInputArray Int
2018 Int
15
let dungeon = UArray Coord Char -> Dungeon
parseMap UArray Coord Char
input
units = UArray Coord Char -> Map Coord Team
parseUnits UArray Coord Char
input
print (part1 dungeon units)
print (part2 dungeon units)
part1 :: Dungeon -> Map Coord Team -> Int
part1 :: Dungeon -> Map Coord Team -> Int
part1 Dungeon
dungeon Map Coord Team
units = Map Coord Unit -> Int -> Int
outcome Map Coord Unit
units2 Int
turns
where
units1 :: Map Coord Unit
units1 = (Team -> Unit) -> Map Coord Team -> Map Coord Unit
forall a b. (a -> b) -> Map Coord a -> Map Coord b
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (Int -> Int -> Team -> Unit
Unit Int
3 Int
200) Map Coord Team
units
(Map Coord Unit
units2, Int
turns) = [(Map Coord Unit, Int)] -> (Map Coord Unit, Int)
forall a. HasCallStack => [a] -> a
last (Dungeon -> Map Coord Unit -> Int -> [(Map Coord Unit, Int)]
simulate Dungeon
dungeon Map Coord Unit
units1 Int
0)
part2 :: Dungeon -> Map Coord Team -> Int
part2 :: Dungeon -> Map Coord Team -> Int
part2 Dungeon
dungeon Map Coord Team
units = Int -> Int -> Maybe (Int, Int) -> Int
search Int
elfCount Int
3 Maybe (Int, Int)
forall a. Maybe a
Nothing
where
elfCount :: Int
elfCount = Map Coord Team -> Int
forall k a. Map k a -> Int
Map.size ((Team -> Bool) -> Map Coord Team -> Map Coord Team
forall a k. (a -> Bool) -> Map k a -> Map k a
Map.filter (Team
ElfTeam -> Team -> Bool
forall a. Eq a => a -> a -> Bool
==) Map Coord Team
units)
search :: Int -> Int -> Maybe (Int, Int) -> Int
search Int
elfCount Int
atkLo (Just (Int
atkHi, Int
answer))
| Int
atkLo Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1 Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
atkHi = Int
answer
search Int
elfCount Int
atkLo Maybe (Int, Int)
mbAtkHi
| [(Map Coord Unit, Int)] -> Bool
forall a. [a] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null [(Map Coord Unit, Int)]
trimmed
= Int -> Int -> Maybe (Int, Int) -> Int
search Int
elfCount Int
atkLo ((Int, Int) -> Maybe (Int, Int)
forall a. a -> Maybe a
Just (Int
atk, Map Coord Unit -> Int -> Int
outcome Map Coord Unit
units2 Int
turns))
| Bool
otherwise = Int -> Int -> Maybe (Int, Int) -> Int
search Int
elfCount Int
atk Maybe (Int, Int)
mbAtkHi
where
atk :: Int
atk = case Maybe (Int, Int)
mbAtkHi of
Maybe (Int, Int)
Nothing -> Int
atkLo Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
2
Just (Int
atkHi, Int
_) -> (Int
atkLo Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
atkHi) Int -> Int -> Int
forall a. Integral a => a -> a -> a
`quot` Int
2
toUnit :: Team -> Unit
toUnit Team
t = case Team
t of Team
Elf -> Int -> Int -> Team -> Unit
Unit Int
atk Int
200 Team
t; Team
Goblin -> Int -> Int -> Team -> Unit
Unit Int
3 Int
200 Team
t
units1 :: Map Coord Unit
units1 = (Team -> Unit) -> Map Coord Team -> Map Coord Unit
forall a b. (a -> b) -> Map Coord a -> Map Coord b
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap Team -> Unit
toUnit Map Coord Team
units
allElves :: (Map k Unit, b) -> Bool
allElves (Map k Unit
us,b
_) = Map k Unit -> Int
forall k a. Map k a -> Int
Map.size ((Unit -> Bool) -> Map k Unit -> Map k Unit
forall a k. (a -> Bool) -> Map k a -> Map k a
Map.filter (\Unit
u -> Unit -> Team
team Unit
u Team -> Team -> Bool
forall a. Eq a => a -> a -> Bool
== Team
Elf) Map k Unit
us)
Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
elfCount
([(Map Coord Unit, Int)]
ticks, [(Map Coord Unit, Int)]
trimmed) = ((Map Coord Unit, Int) -> Bool)
-> [(Map Coord Unit, Int)]
-> ([(Map Coord Unit, Int)], [(Map Coord Unit, Int)])
forall a. (a -> Bool) -> [a] -> ([a], [a])
span (Map Coord Unit, Int) -> Bool
forall {k} {b}. (Map k Unit, b) -> Bool
allElves (Dungeon -> Map Coord Unit -> Int -> [(Map Coord Unit, Int)]
simulate Dungeon
dungeon Map Coord Unit
units1 Int
0)
(Map Coord Unit
units2, Int
turns) = [(Map Coord Unit, Int)] -> (Map Coord Unit, Int)
forall a. HasCallStack => [a] -> a
last [(Map Coord Unit, Int)]
ticks
outcome :: Map Coord Unit -> Int -> Int
outcome :: Map Coord Unit -> Int -> Int
outcome Map Coord Unit
units Int
turns = Int
turns Int -> Int -> Int
forall a. Num a => a -> a -> a
* Map Coord Int -> Int
forall a. Num a => Map Coord a -> a
forall (t :: * -> *) a. (Foldable t, Num a) => t a -> a
sum ((Unit -> Int) -> Map Coord Unit -> Map Coord Int
forall a b. (a -> b) -> Map Coord a -> Map Coord b
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap Unit -> Int
hp Map Coord Unit
units)
parseMap :: UArray Coord Char -> Dungeon
parseMap :: UArray Coord Char -> Dungeon
parseMap = (Char -> Bool) -> UArray Coord Char -> Dungeon
forall (a :: * -> * -> *) e' e i.
(IArray a e', IArray a e, Ix i) =>
(e' -> e) -> a i e' -> a i e
amap (Char
'#' Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
/=)
parseUnits :: UArray Coord Char -> Map Coord Team
parseUnits :: UArray Coord Char -> Map Coord Team
parseUnits UArray Coord Char
rs = [(Coord, Team)] -> Map Coord Team
forall k a. Ord k => [(k, a)] -> Map k a
Map.fromList
[ (Coord
c, Team
unit)
| (Coord
c,Char
v) <- UArray Coord Char -> [(Coord, Char)]
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> [(i, e)]
assocs UArray Coord Char
rs
, Team
unit <- case Char
v of
Char
'G' -> [Team
Goblin]
Char
'E' -> [Team
Elf ]
Char
_ -> [ ]
]
simulate :: Dungeon -> Map Coord Unit -> Int -> [(Map Coord Unit, Int)]
simulate :: Dungeon -> Map Coord Unit -> Int -> [(Map Coord Unit, Int)]
simulate Dungeon
dungeon Map Coord Unit
units Int
turns =
(Map Coord Unit
units, Int
turns) (Map Coord Unit, Int)
-> [(Map Coord Unit, Int)] -> [(Map Coord Unit, Int)]
forall a. a -> [a] -> [a]
: Dungeon
-> Set Coord -> Map Coord Unit -> Int -> [(Map Coord Unit, Int)]
tick Dungeon
dungeon (Map Coord Unit -> Set Coord
forall k a. Map k a -> Set k
Map.keysSet Map Coord Unit
units) Map Coord Unit
units Int
turns
tick :: Dungeon -> Set Coord -> Map Coord Unit -> Int -> [(Map Coord Unit, Int)]
tick :: Dungeon
-> Set Coord -> Map Coord Unit -> Int -> [(Map Coord Unit, Int)]
tick Dungeon
dungeon Set Coord
schedule Map Coord Unit
units Int
turns =
case Set Coord -> Maybe (Coord, Set Coord)
forall a. Set a -> Maybe (a, Set a)
Set.minView Set Coord
schedule of
Maybe (Coord, Set Coord)
Nothing -> Dungeon -> Map Coord Unit -> Int -> [(Map Coord Unit, Int)]
simulate Dungeon
dungeon Map Coord Unit
units (Int
turnsInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1)
Maybe (Coord, Set Coord)
_ | Map Coord Team -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => t a -> Bool
same ((Unit -> Team) -> Map Coord Unit -> Map Coord Team
forall a b. (a -> b) -> Map Coord a -> Map Coord b
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap Unit -> Team
team Map Coord Unit
units) -> [(Map Coord Unit
units,Int
turns)]
Just (Coord
pos, Set Coord
schedule) ->
let unit :: Unit
unit = Map Coord Unit
units Map Coord Unit -> Coord -> Unit
forall k a. Ord k => Map k a -> k -> a
Map.! Coord
pos
pos' :: Coord
pos' = Coord -> Maybe Coord -> Coord
forall a. a -> Maybe a -> a
fromMaybe Coord
pos (Coord -> Unit -> Map Coord Unit -> Dungeon -> Maybe Coord
route Coord
pos Unit
unit (Coord -> Map Coord Unit -> Map Coord Unit
forall k a. Ord k => k -> Map k a -> Map k a
Map.delete Coord
pos Map Coord Unit
units) Dungeon
dungeon)
units' :: Map Coord Unit
units' = Coord -> Unit -> Map Coord Unit -> Map Coord Unit
forall k a. Ord k => k -> a -> Map k a -> Map k a
Map.insert Coord
pos' Unit
unit (Coord -> Map Coord Unit -> Map Coord Unit
forall k a. Ord k => k -> Map k a -> Map k a
Map.delete Coord
pos Map Coord Unit
units)
in case Coord -> Unit -> Map Coord Unit -> Maybe Coord
target Coord
pos' Unit
unit Map Coord Unit
units of
Just Coord
tgt ->
let units'' :: Map Coord Unit
units'' = Int -> Coord -> Map Coord Unit -> Map Coord Unit
melee (Unit -> Int
attack Unit
unit) Coord
tgt Map Coord Unit
units'
schedule' :: Set Coord
schedule' = Set Coord -> Set Coord -> Set Coord
forall a. Ord a => Set a -> Set a -> Set a
Set.intersection Set Coord
schedule (Map Coord Unit -> Set Coord
forall k a. Map k a -> Set k
Map.keysSet Map Coord Unit
units'')
in Dungeon
-> Set Coord -> Map Coord Unit -> Int -> [(Map Coord Unit, Int)]
tick Dungeon
dungeon Set Coord
schedule' Map Coord Unit
units'' Int
turns
Maybe Coord
Nothing ->
Dungeon
-> Set Coord -> Map Coord Unit -> Int -> [(Map Coord Unit, Int)]
tick Dungeon
dungeon Set Coord
schedule Map Coord Unit
units' Int
turns
melee ::
Int ->
Coord ->
Map Coord Unit ->
Map Coord Unit
melee :: Int -> Coord -> Map Coord Unit -> Map Coord Unit
melee Int
atk = (Unit -> Maybe Unit) -> Coord -> Map Coord Unit -> Map Coord Unit
forall k a. Ord k => (a -> Maybe a) -> k -> Map k a -> Map k a
Map.update ((Unit -> Maybe Unit) -> Coord -> Map Coord Unit -> Map Coord Unit)
-> (Unit -> Maybe Unit)
-> Coord
-> Map Coord Unit
-> Map Coord Unit
forall a b. (a -> b) -> a -> b
$ \Unit
tgt ->
if Unit -> Int
hp Unit
tgt Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
atk
then Maybe Unit
forall a. Maybe a
Nothing
else Unit -> Maybe Unit
forall a. a -> Maybe a
Just (Unit -> Maybe Unit) -> Unit -> Maybe Unit
forall a b. (a -> b) -> a -> b
$! Unit
tgt { hp = hp tgt - atk }
target ::
Coord ->
Unit -> Map Coord Unit -> Maybe Coord
target :: Coord -> Unit -> Map Coord Unit -> Maybe Coord
target Coord
pos Unit
unit Map Coord Unit
units
| [Coord] -> Bool
forall a. [a] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null [Coord]
possible = Maybe Coord
forall a. Maybe a
Nothing
| Bool
otherwise = Coord -> Maybe Coord
forall a. a -> Maybe a
Just (Coord -> Maybe Coord) -> Coord -> Maybe Coord
forall a b. (a -> b) -> a -> b
$! (Coord -> Coord -> Ordering) -> [Coord] -> Coord
forall (t :: * -> *) a.
Foldable t =>
(a -> a -> Ordering) -> t a -> a
minimumBy Coord -> Coord -> Ordering
ordering [Coord]
possible
where
ordering :: Coord -> Coord -> Ordering
ordering = (Coord -> (Int, Coord)) -> Coord -> Coord -> Ordering
forall a b. Ord a => (b -> a) -> b -> b -> Ordering
comparing (\Coord
i -> (Unit -> Int
hp (Map Coord Unit
units Map Coord Unit -> Coord -> Unit
forall k a. Ord k => Map k a -> k -> a
Map.! Coord
i), Coord
i))
possible :: [Coord]
possible = (Coord -> Bool) -> [Coord] -> [Coord]
forall a. (a -> Bool) -> [a] -> [a]
filter (Unit -> Map Coord Unit -> Coord -> Bool
isEnemy Unit
unit Map Coord Unit
units) (Coord -> [Coord]
cardinal Coord
pos)
isEnemy ::
Unit ->
Map Coord Unit ->
Coord ->
Bool
isEnemy :: Unit -> Map Coord Unit -> Coord -> Bool
isEnemy Unit
unit Map Coord Unit
units Coord
loc =
case Coord -> Map Coord Unit -> Maybe Unit
forall k a. Ord k => k -> Map k a -> Maybe a
Map.lookup Coord
loc Map Coord Unit
units of
Just Unit
u -> Unit -> Team
team Unit
u Team -> Team -> Bool
forall a. Eq a => a -> a -> Bool
/= Unit -> Team
team Unit
unit
Maybe Unit
Nothing -> Bool
False
route ::
Coord ->
Unit ->
Map Coord Unit ->
Dungeon ->
Maybe Coord
route :: Coord -> Unit -> Map Coord Unit -> Dungeon -> Maybe Coord
route Coord
pos Unit
unit Map Coord Unit
units Dungeon
dungeon
| Coord -> Bool
isNear Coord
pos = Maybe Coord
forall a. Maybe a
Nothing
| Bool
otherwise = Set Coord -> Set (Integer, Coord, Coord) -> Maybe Coord
forall {a} {b}.
(Ord a, Ord b, Num a) =>
Set Coord -> Set (a, Coord, b) -> Maybe b
search Set Coord
forall a. Set a
Set.empty Set (Integer, Coord, Coord)
candidates
where
candidates :: Set (Integer, Coord, Coord)
candidates = [(Integer, Coord, Coord)] -> Set (Integer, Coord, Coord)
forall a. Ord a => [a] -> Set a
Set.fromList [ (Integer
0, Coord
start, Coord
start) | Coord
start <- Coord -> [Coord]
cardinal Coord
pos ]
isOpen :: Coord -> Bool
isOpen Coord
loc = Dungeon
dungeon Dungeon -> Coord -> Bool
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> i -> e
! Coord
loc Bool -> Bool -> Bool
&& Coord -> Map Coord Unit -> Bool
forall k a. Ord k => k -> Map k a -> Bool
Map.notMember Coord
loc Map Coord Unit
units
isNear :: Coord -> Bool
isNear = (Coord -> Bool) -> [Coord] -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
any (Unit -> Map Coord Unit -> Coord -> Bool
isEnemy Unit
unit Map Coord Unit
units) ([Coord] -> Bool) -> (Coord -> [Coord]) -> Coord -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Coord -> [Coord]
cardinal
search :: Set Coord -> Set (a, Coord, b) -> Maybe b
search Set Coord
seen Set (a, Coord, b)
q =
do ((dist, dest, start), q) <- Set (a, Coord, b) -> Maybe ((a, Coord, b), Set (a, Coord, b))
forall a. Set a -> Maybe (a, Set a)
Set.minView Set (a, Coord, b)
q
if | Set.member dest seen || not (isOpen dest) -> search seen q
| isNear dest -> Just start
| otherwise -> search (Set.insert dest seen)
(foldl' (flip Set.insert) q
[ (dist+1, dest, start) | dest <- cardinal dest ])