{-# Language QuasiQuotes, OverloadedStrings, BlockArguments, ViewPatterns, TemplateHaskell #-}
module Main (main) where
import Advent (format, stageTH)
import Control.Applicative ((<|>))
import Data.List (foldl1', tails)
import Text.ParserCombinators.ReadP
data Tree
= Tree :+: Tree
| Leaf !Int
deriving Int -> Tree -> ShowS
[Tree] -> ShowS
Tree -> String
(Int -> Tree -> ShowS)
-> (Tree -> String) -> ([Tree] -> ShowS) -> Show Tree
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> Tree -> ShowS
showsPrec :: Int -> Tree -> ShowS
$cshow :: Tree -> String
show :: Tree -> String
$cshowList :: [Tree] -> ShowS
showList :: [Tree] -> ShowS
Show
t :: ReadP Tree
t :: ReadP Tree
t =
Int -> Tree
Leaf (Int -> Tree) -> ReadP Int -> ReadP Tree
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> ReadS Int -> ReadP Int
forall a. ReadS a -> ReadP a
readS_to_P ReadS Int
forall a. Read a => ReadS a
reads ReadP Tree -> ReadP Tree -> ReadP Tree
forall a. ReadP a -> ReadP a -> ReadP a
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|>
Tree -> Tree -> Tree
(:+:) (Tree -> Tree -> Tree)
-> ReadP String -> ReadP (Tree -> Tree -> Tree)
forall a b. a -> ReadP b -> ReadP a
forall (f :: * -> *) a b. Functor f => a -> f b -> f a
<$ ReadP String
"[" ReadP (Tree -> Tree -> Tree) -> ReadP Tree -> ReadP (Tree -> Tree)
forall a b. ReadP (a -> b) -> ReadP a -> ReadP b
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> ReadP Tree
t ReadP (Tree -> Tree) -> ReadP String -> ReadP (Tree -> Tree)
forall a b. ReadP a -> ReadP b -> ReadP a
forall (f :: * -> *) a b. Applicative f => f a -> f b -> f a
<* ReadP String
"," ReadP (Tree -> Tree) -> ReadP Tree -> ReadP Tree
forall a b. ReadP (a -> b) -> ReadP a -> ReadP b
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> ReadP Tree
t ReadP Tree -> ReadP String -> ReadP Tree
forall a b. ReadP a -> ReadP b -> ReadP a
forall (f :: * -> *) a b. Applicative f => f a -> f b -> f a
<* ReadP String
"]"
stageTH
main :: IO ()
IO ()
main =
do inp <- [format|2021 18 (@t%n)*|]
print (magnitude (foldl1' add inp))
print (maximum [magnitude (add x y) `max` magnitude (add y x)
| x:ys <- tails inp, y <- ys])
add :: Tree -> Tree -> Tree
add :: Tree -> Tree -> Tree
add Tree
x Tree
y = Tree -> Tree
reduce (Tree
x Tree -> Tree -> Tree
:+: Tree
y)
reduce :: Tree -> Tree
reduce :: Tree -> Tree
reduce (Tree -> Tree
explode -> Tree
x) = Tree -> (Tree -> Tree) -> Maybe Tree -> Tree
forall b a. b -> (a -> b) -> Maybe a -> b
maybe Tree
x Tree -> Tree
reduce (Tree -> Maybe Tree
split Tree
x)
explode :: Tree -> Tree
explode :: Tree -> Tree
explode = Int -> [(Side, Tree)] -> Tree -> Tree
forall {t}. (Num t, Ord t) => t -> [(Side, Tree)] -> Tree -> Tree
down (Int
0::Int) []
where
down :: t -> [(Side, Tree)] -> Tree -> Tree
down t
4 [(Side, Tree)]
z (Leaf Int
l :+: Leaf Int
r) = t -> [(Side, Tree)] -> Tree -> Tree
up t
4 (Side -> Int -> [(Side, Tree)] -> [(Side, Tree)]
sendUp Side
L Int
l (Side -> Int -> [(Side, Tree)] -> [(Side, Tree)]
sendUp Side
R Int
r [(Side, Tree)]
z)) (Int -> Tree
Leaf Int
0)
down t
d [(Side, Tree)]
z (Tree
l :+: Tree
r) | t
d t -> t -> Bool
forall a. Ord a => a -> a -> Bool
< t
4 = t -> [(Side, Tree)] -> Tree -> Tree
down (t
dt -> t -> t
forall a. Num a => a -> a -> a
+t
1) ((Side
R,Tree
r)(Side, Tree) -> [(Side, Tree)] -> [(Side, Tree)]
forall a. a -> [a] -> [a]
:[(Side, Tree)]
z) Tree
l
down t
d [(Side, Tree)]
z Tree
x = t -> [(Side, Tree)] -> Tree -> Tree
up t
d [(Side, Tree)]
z Tree
x
up :: t -> [(Side, Tree)] -> Tree -> Tree
up t
_ [] Tree
x = Tree
x
up t
d ((Side
R,Tree
r):[(Side, Tree)]
z) Tree
l = t -> [(Side, Tree)] -> Tree -> Tree
down t
d ((Side
L,Tree
l)(Side, Tree) -> [(Side, Tree)] -> [(Side, Tree)]
forall a. a -> [a] -> [a]
:[(Side, Tree)]
z) Tree
r
up t
d ((Side
L,Tree
l):[(Side, Tree)]
z) Tree
r = t -> [(Side, Tree)] -> Tree -> Tree
up (t
dt -> t -> t
forall a. Num a => a -> a -> a
-t
1) [(Side, Tree)]
z (Tree
l Tree -> Tree -> Tree
:+: Tree
r)
split :: Tree -> Maybe Tree
split :: Tree -> Maybe Tree
split (Tree
l :+: Tree
r) = (Tree -> Tree -> Tree
:+: Tree
r) (Tree -> Tree) -> Maybe Tree -> Maybe Tree
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Tree -> Maybe Tree
split Tree
l Maybe Tree -> Maybe Tree -> Maybe Tree
forall a. Maybe a -> Maybe a -> Maybe a
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> (Tree
l Tree -> Tree -> Tree
:+:) (Tree -> Tree) -> Maybe Tree -> Maybe Tree
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Tree -> Maybe Tree
split Tree
r
split (Leaf Int
x)
| Int
x Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
>= Int
10, (Int
q,Int
r) <- Int -> Int -> (Int, Int)
forall a. Integral a => a -> a -> (a, a)
quotRem Int
x Int
2 = Tree -> Maybe Tree
forall a. a -> Maybe a
Just (Int -> Tree
Leaf Int
q Tree -> Tree -> Tree
:+: Int -> Tree
Leaf (Int
qInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
r))
| Bool
otherwise = Maybe Tree
forall a. Maybe a
Nothing
magnitude :: Tree -> Int
magnitude :: Tree -> Int
magnitude (Leaf Int
x) = Int
x
magnitude (Tree
l :+: Tree
r) = Int
3 Int -> Int -> Int
forall a. Num a => a -> a -> a
* Tree -> Int
magnitude Tree
l Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
2 Int -> Int -> Int
forall a. Num a => a -> a -> a
* Tree -> Int
magnitude Tree
r
data Side = L | R deriving (Side -> Side -> Bool
(Side -> Side -> Bool) -> (Side -> Side -> Bool) -> Eq Side
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: Side -> Side -> Bool
== :: Side -> Side -> Bool
$c/= :: Side -> Side -> Bool
/= :: Side -> Side -> Bool
Eq, Int -> Side -> ShowS
[Side] -> ShowS
Side -> String
(Int -> Side -> ShowS)
-> (Side -> String) -> ([Side] -> ShowS) -> Show Side
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> Side -> ShowS
showsPrec :: Int -> Side -> ShowS
$cshow :: Side -> String
show :: Side -> String
$cshowList :: [Side] -> ShowS
showList :: [Side] -> ShowS
Show)
type Zip = [(Side, Tree)]
sendUp :: Side -> Int -> Zip -> Zip
sendUp :: Side -> Int -> [(Side, Tree)] -> [(Side, Tree)]
sendUp Side
L Int
x ((Side
L,Tree
l):[(Side, Tree)]
zs) = (Side
L, Int -> Tree -> Tree
sendR Int
x Tree
l)(Side, Tree) -> [(Side, Tree)] -> [(Side, Tree)]
forall a. a -> [a] -> [a]
:[(Side, Tree)]
zs
sendUp Side
R Int
x ((Side
R,Tree
r):[(Side, Tree)]
zs) = (Side
R, Int -> Tree -> Tree
sendL Int
x Tree
r)(Side, Tree) -> [(Side, Tree)] -> [(Side, Tree)]
forall a. a -> [a] -> [a]
:[(Side, Tree)]
zs
sendUp Side
h Int
x ((Side, Tree)
z :[(Side, Tree)]
zs) = (Side, Tree)
z (Side, Tree) -> [(Side, Tree)] -> [(Side, Tree)]
forall a. a -> [a] -> [a]
: Side -> Int -> [(Side, Tree)] -> [(Side, Tree)]
sendUp Side
h Int
x [(Side, Tree)]
zs
sendUp Side
_ Int
_ [] = []
sendL :: Int -> Tree -> Tree
sendL :: Int -> Tree -> Tree
sendL Int
x (Tree
l :+: Tree
r) = Int -> Tree -> Tree
sendL Int
x Tree
l Tree -> Tree -> Tree
:+: Tree
r
sendL Int
x (Leaf Int
y) = Int -> Tree
Leaf (Int
x Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
y)
sendR :: Int -> Tree -> Tree
sendR :: Int -> Tree -> Tree
sendR Int
x (Tree
l :+: Tree
r) = Tree
l Tree -> Tree -> Tree
:+: Int -> Tree -> Tree
sendR Int
x Tree
r
sendR Int
x (Leaf Int
y) = Int -> Tree
Leaf (Int
x Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
y)