{-# Language QuasiQuotes #-}
{-|
Module      : Main
Description : Day 22 solution
Copyright   : (c) Eric Mertens, 2021
License     : ISC
Maintainer  : emertens@gmail.com

<https://adventofcode.com/2015/day/22>

-}
module Main where

import Advent.Format (format)
import Advent.Search (astar, AStep(..))

-- |
-- >>> :main
-- 900
-- 1216
main :: IO ()
IO ()
main =
 do (hp,dmg) <- [format|2015 22 Hit Points: %u%nDamage: %u%n|]
    print (runSim (initialState hp dmg))
    print (runSim (initialState hp dmg) { difficulty = 1 })

runSim :: GameState -> Int
runSim :: GameState -> Int
runSim GameState
s = [Int] -> Int
forall a. HasCallStack => [a] -> a
head [Int
cost | (GameState
s1, Int
cost) <- (GameState -> [AStep GameState]) -> GameState -> [(GameState, Int)]
forall a. Ord a => (a -> [AStep a]) -> a -> [(a, Int)]
astar GameState -> [AStep GameState]
advance GameState
s, GameState -> Bool
bossDead GameState
s1]

data Spell = Recharge | Poison | Shield | Drain | MagicMissile
  deriving Spell -> Spell -> Bool
(Spell -> Spell -> Bool) -> (Spell -> Spell -> Bool) -> Eq Spell
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: Spell -> Spell -> Bool
== :: Spell -> Spell -> Bool
$c/= :: Spell -> Spell -> Bool
/= :: Spell -> Spell -> Bool
Eq

spellDamage :: Spell -> Int
spellDamage :: Spell -> Int
spellDamage Spell
spell =
  case Spell
spell of
    Spell
MagicMissile -> Int
4
    Spell
Drain        -> Int
2
    Spell
_            -> Int
0

spellHeal :: Spell -> Int
spellHeal :: Spell -> Int
spellHeal Spell
spell =
  case Spell
spell of
    Spell
Drain        -> Int
2
    Spell
_            -> Int
0

spellCost :: Spell -> Int
spellCost :: Spell -> Int
spellCost Spell
s =
  case Spell
s of
    Spell
Recharge -> Int
229
    Spell
Poison   -> Int
173
    Spell
Shield   -> Int
113
    Spell
Drain    -> Int
73
    Spell
MagicMissile -> Int
53

data GameState = GameState
  { GameState -> Int
manaPool     
  , GameState -> Int
poisonTimer  
  , GameState -> Int
rechargeTimer
  , GameState -> Int
shieldTimer  
  , GameState -> Int
playerHp     
  , GameState -> Int
bossHp, GameState -> Int
bossDamage, GameState -> Int
difficulty :: !Int
  }
  deriving (GameState -> GameState -> Bool
(GameState -> GameState -> Bool)
-> (GameState -> GameState -> Bool) -> Eq GameState
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: GameState -> GameState -> Bool
== :: GameState -> GameState -> Bool
$c/= :: GameState -> GameState -> Bool
/= :: GameState -> GameState -> Bool
Eq, Eq GameState
Eq GameState =>
(GameState -> GameState -> Ordering)
-> (GameState -> GameState -> Bool)
-> (GameState -> GameState -> Bool)
-> (GameState -> GameState -> Bool)
-> (GameState -> GameState -> Bool)
-> (GameState -> GameState -> GameState)
-> (GameState -> GameState -> GameState)
-> Ord GameState
GameState -> GameState -> Bool
GameState -> GameState -> Ordering
GameState -> GameState -> GameState
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 :: GameState -> GameState -> Ordering
compare :: GameState -> GameState -> Ordering
$c< :: GameState -> GameState -> Bool
< :: GameState -> GameState -> Bool
$c<= :: GameState -> GameState -> Bool
<= :: GameState -> GameState -> Bool
$c> :: GameState -> GameState -> Bool
> :: GameState -> GameState -> Bool
$c>= :: GameState -> GameState -> Bool
>= :: GameState -> GameState -> Bool
$cmax :: GameState -> GameState -> GameState
max :: GameState -> GameState -> GameState
$cmin :: GameState -> GameState -> GameState
min :: GameState -> GameState -> GameState
Ord, Int -> GameState -> ShowS
[GameState] -> ShowS
GameState -> String
(Int -> GameState -> ShowS)
-> (GameState -> String)
-> ([GameState] -> ShowS)
-> Show GameState
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> GameState -> ShowS
showsPrec :: Int -> GameState -> ShowS
$cshow :: GameState -> String
show :: GameState -> String
$cshowList :: [GameState] -> ShowS
showList :: [GameState] -> ShowS
Show)

initialState :: Int -> Int -> GameState
initialState :: Int -> Int -> GameState
initialState Int
hp Int
dmg = GameState
  { manaPool :: Int
manaPool      = Int
500
  , poisonTimer :: Int
poisonTimer   = Int
0
  , rechargeTimer :: Int
rechargeTimer = Int
0
  , shieldTimer :: Int
shieldTimer   = Int
0
  , playerHp :: Int
playerHp      = Int
50
  , bossHp :: Int
bossHp        = Int
hp
  , bossDamage :: Int
bossDamage    = Int
dmg
  , difficulty :: Int
difficulty    = Int
0
  }

stepTimers :: GameState -> GameState
stepTimers :: GameState -> GameState
stepTimers GameState
s = GameState
s
  { manaPool      = manaPool s +
                    if rechargeTimer s > 0 then 101 else 0
  , bossHp        = bossHp s -
                    if poisonTimer s > 0 then 3 else 0
  , poisonTimer   = dec (poisonTimer s)
  , rechargeTimer = dec (rechargeTimer s)
  , shieldTimer   = dec (shieldTimer s)
  }

bossAttack :: GameState -> GameState
bossAttack :: GameState -> GameState
bossAttack GameState
s = GameState
s { playerHp = playerHp s - effectiveAttack }
  where
  effectiveAttack :: Int
effectiveAttack = Int -> Int -> Int
forall a. Ord a => a -> a -> a
max Int
1 (GameState -> Int
bossDamage GameState
s Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
armor)
  armor :: Int
armor | GameState -> Int
shieldTimer GameState
s Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
> Int
0 = Int
7
        | Bool
otherwise         = Int
0

applyDifficulty :: GameState -> GameState
applyDifficulty :: GameState -> GameState
applyDifficulty GameState
s = GameState
s { playerHp = playerHp s - difficulty s }

advance :: GameState -> [AStep GameState]
advance :: GameState -> [AStep GameState]
advance GameState
s =
  [ GameState -> Int -> Int -> AStep GameState
forall a. a -> Int -> Int -> AStep a
AStep GameState
s5 (Spell -> Int
spellCost Spell
spell) Int
0
  | GameState
s1 <- GameState
s GameState -> (GameState -> GameState) -> [GameState]
--> GameState -> GameState
stepTimers (GameState -> GameState)
-> (GameState -> GameState) -> GameState -> GameState
forall b c a. (b -> c) -> (a -> b) -> a -> c
. GameState -> GameState
applyDifficulty
  , Spell
spell <- GameState -> [Spell]
availableSpells GameState
s1
  , GameState
s2 <- GameState
s1 GameState -> (GameState -> GameState) -> [GameState]
--> Spell -> GameState -> GameState
applySpell Spell
spell
  , GameState
s3 <- GameState
s2 GameState -> (GameState -> GameState) -> [GameState]
--> GameState -> GameState
stepTimers
  , GameState
s4 <- GameState
s3 GameState -> (GameState -> GameState) -> [GameState]
--> GameState -> GameState
bossAttack
  , GameState
s5 <- GameState
s4 GameState -> (GameState -> GameState) -> [GameState]
--> GameState -> GameState
forall a. a -> a
id
  ]

infix 1 -->

(-->) :: GameState -> (GameState -> GameState) -> [GameState]
GameState
s --> :: GameState -> (GameState -> GameState) -> [GameState]
--> GameState -> GameState
k
  | GameState -> Bool
playerDead GameState
s = []
  | GameState -> Bool
bossDead GameState
s   = [GameState
s]
  | Bool
otherwise    = [GameState -> GameState
k GameState
s]

applySpell :: Spell -> GameState -> GameState
applySpell :: Spell -> GameState -> GameState
applySpell Spell
spell GameState
s =
  GameState
s { manaPool      = manaPool s - spellCost spell
    , rechargeTimer = if spell == Recharge then 5 else rechargeTimer s
    , poisonTimer   = if spell == Poison   then 6 else poisonTimer s
    , shieldTimer   = if spell == Shield   then 6 else shieldTimer s
    , bossHp        = bossHp s - spellDamage spell
    , playerHp      = playerHp s + spellHeal spell
    }

availableSpells :: GameState -> [Spell]
availableSpells :: GameState -> [Spell]
availableSpells GameState
s =
    (Spell -> Bool) -> [Spell] -> [Spell]
forall a. (a -> Bool) -> [a] -> [a]
filter (\Spell
spell -> Spell -> Int
spellCost Spell
spell Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= GameState -> Int
manaPool GameState
s)
    ([Spell] -> [Spell]) -> [Spell] -> [Spell]
forall a b. (a -> b) -> a -> b
$ [Spell
Poison   | GameState -> Int
poisonTimer GameState
s Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
0]
   [Spell] -> [Spell] -> [Spell]
forall a. [a] -> [a] -> [a]
++ [Spell
Recharge | GameState -> Int
rechargeTimer GameState
s Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
0]
   [Spell] -> [Spell] -> [Spell]
forall a. [a] -> [a] -> [a]
++ [Spell
Shield   | GameState -> Int
shieldTimer GameState
s Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
0]
   [Spell] -> [Spell] -> [Spell]
forall a. [a] -> [a] -> [a]
++ [Spell
MagicMissile, Spell
Drain]

dec :: Int -> Int
dec :: Int -> Int
dec Int
x | Int
x Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
1 = Int
0
dec Int
x          = Int
xInt -> Int -> Int
forall a. Num a => a -> a -> a
-Int
1

bossDead :: GameState -> Bool
bossDead :: GameState -> Bool
bossDead GameState
s = GameState -> Int
bossHp GameState
s Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
0

playerDead :: GameState -> Bool
playerDead :: GameState -> Bool
playerDead GameState
s = GameState -> Int
playerHp GameState
s Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
0