{-# Language ImportQualifiedPost, QuasiQuotes #-}
{-# Language OverloadedStrings #-}
module Main (main) where
import Advent (format, counts)
import Advent.Coord (Coord(C), cardinal, coordCol, coordRow,
above, below, right, left, manhattan, boundingBox)
import Advent.Search (dfs)
import Data.List (groupBy, sort, sortBy)
import Data.Function (on)
import Data.Ix (range)
import Data.Map qualified as Map
main :: IO ()
IO ()
main =
do let toCoord :: (Int, Int) -> Coord
toCoord (Int
x,Int
y) = Int -> Int -> Coord
C Int
y Int
x
input <- ((Int, Int) -> Coord) -> [(Int, Int)] -> [Coord]
forall a b. (a -> b) -> [a] -> [b]
map (Int, Int) -> Coord
toCoord ([(Int, Int)] -> [Coord]) -> IO [(Int, Int)] -> IO [Coord]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> [format|2018 6 (%u, %u%n)*|]
print (part1 input)
print (part2 input)
part1 ::
[Coord] ->
Int
part1 :: [Coord] -> Int
part1 [Coord]
input
= Map Coord Int -> Int
forall a. Ord a => Map Coord a -> a
forall (t :: * -> *) a. (Foldable t, Ord a) => t a -> a
maximum
(Map Coord Int -> Int) -> Map Coord Int -> Int
forall a b. (a -> b) -> a -> b
$ (Maybe Int -> Maybe Int) -> Map Coord (Maybe Int) -> Map Coord Int
forall a b k. (a -> Maybe b) -> Map k a -> Map k b
Map.mapMaybe Maybe Int -> Maybe Int
forall a. a -> a
id
(Map Coord (Maybe Int) -> Map Coord Int)
-> Map Coord (Maybe Int) -> Map Coord Int
forall a b. (a -> b) -> a -> b
$ (Int -> Int -> Maybe Int)
-> Map Coord Int -> Map Coord Int -> Map Coord (Maybe Int)
forall k a b c.
Ord k =>
(a -> b -> c) -> Map k a -> Map k b -> Map k c
Map.intersectionWith Int -> Int -> Maybe Int
forall {a}. Eq a => a -> a -> Maybe a
match ([Coord] -> Map Coord Int
regionSizes [Coord]
box0) ([Coord] -> Map Coord Int
regionSizes [Coord]
box1)
where
regionSizes :: [Coord] -> Map Coord Int
regionSizes = [Coord] -> Map Coord Int
forall (f :: * -> *) a. (Foldable f, Ord a) => f a -> Map a Int
counts ([Coord] -> Map Coord Int)
-> ([Coord] -> [Coord]) -> [Coord] -> Map Coord Int
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Coord -> [Coord]) -> [Coord] -> [Coord]
forall (t :: * -> *) a b. Foldable t => (a -> [b]) -> t a -> [b]
concatMap Coord -> [Coord]
toRegion
match :: a -> a -> Maybe a
match a
x a
y
| a
x a -> a -> Bool
forall a. Eq a => a -> a -> Bool
== a
y = a -> Maybe a
forall a. a -> Maybe a
Just a
x
| Bool
otherwise = Maybe a
forall a. Maybe a
Nothing
toRegion :: Coord -> [Coord]
toRegion Coord
c =
case [[(Coord, Int)]]
choices of
[(Coord
r,Int
_)]:[[(Coord, Int)]]
_ -> [Coord
r]
[[(Coord, Int)]]
_ -> []
where
choices :: [[(Coord, Int)]]
choices = ((Coord, Int) -> (Coord, Int) -> Bool)
-> [(Coord, Int)] -> [[(Coord, Int)]]
forall a. (a -> a -> Bool) -> [a] -> [[a]]
groupBy (Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
(==) (Int -> Int -> Bool)
-> ((Coord, Int) -> Int) -> (Coord, Int) -> (Coord, Int) -> Bool
forall b c a. (b -> b -> c) -> (a -> b) -> a -> a -> c
`on` (Coord, Int) -> Int
forall a b. (a, b) -> b
snd)
([(Coord, Int)] -> [[(Coord, Int)]])
-> [(Coord, Int)] -> [[(Coord, Int)]]
forall a b. (a -> b) -> a -> b
$ ((Coord, Int) -> (Coord, Int) -> Ordering)
-> [(Coord, Int)] -> [(Coord, Int)]
forall a. (a -> a -> Ordering) -> [a] -> [a]
sortBy (Int -> Int -> Ordering
forall a. Ord a => a -> a -> Ordering
compare (Int -> Int -> Ordering)
-> ((Coord, Int) -> Int)
-> (Coord, Int)
-> (Coord, Int)
-> Ordering
forall b c a. (b -> b -> c) -> (a -> b) -> a -> a -> c
`on` (Coord, Int) -> Int
forall a b. (a, b) -> b
snd)
[ (Coord
coord, Coord -> Coord -> Int
manhattan Coord
c Coord
coord) | Coord
coord <- [Coord]
input ]
Just (Coord
topLeft, Coord
bottomRight) = [Coord] -> Maybe (Coord, Coord)
forall (f :: * -> *). Foldable f => f Coord -> Maybe (Coord, Coord)
boundingBox [Coord]
input
box0 :: [Coord]
box0 = (Coord, Coord) -> [Coord]
forall a. Ix a => (a, a) -> [a]
range (Coord
topLeft, Coord
bottomRight)
box1 :: [Coord]
box1 = (Coord, Coord) -> [Coord]
forall a. Ix a => (a, a) -> [a]
range (Coord -> Coord
left (Coord -> Coord
above Coord
topLeft), Coord -> Coord
right (Coord -> Coord
below Coord
bottomRight))
part2 :: [Coord] -> Int
part2 :: [Coord] -> Int
part2 [Coord]
input = [Coord] -> Int
forall a. [a] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length ((Coord -> [Coord]) -> Coord -> [Coord]
forall a. Ord a => (a -> [a]) -> a -> [a]
dfs Coord -> [Coord]
step Coord
startingPoint)
where
distances :: Coord -> Int
distances :: Coord -> Int
distances Coord
c = [Int] -> Int
forall a. Num a => [a] -> a
forall (t :: * -> *) a. (Foldable t, Num a) => t a -> a
sum ((Coord -> Int) -> [Coord] -> [Int]
forall a b. (a -> b) -> [a] -> [b]
map (Coord -> Coord -> Int
manhattan Coord
c) [Coord]
input)
step :: Coord -> [Coord]
step Coord
c = [Coord
n | Coord
n <- Coord -> [Coord]
cardinal Coord
c, Coord -> Int
distances Coord
n Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
< Int
10000]
startingPoint :: Coord
startingPoint = Int -> Int -> Coord
C ([Int] -> Int
forall a. Ord a => [a] -> a
median ((Coord -> Int) -> [Coord] -> [Int]
forall a b. (a -> b) -> [a] -> [b]
map Coord -> Int
coordRow [Coord]
input)) ([Int] -> Int
forall a. Ord a => [a] -> a
median ((Coord -> Int) -> [Coord] -> [Int]
forall a b. (a -> b) -> [a] -> [b]
map Coord -> Int
coordCol [Coord]
input))
median :: Ord a => [a] -> a
median :: forall a. Ord a => [a] -> a
median [a]
xs = [a] -> [a]
forall a. Ord a => [a] -> [a]
sort [a]
xs [a] -> Int -> a
forall a. HasCallStack => [a] -> Int -> a
!! ([a] -> Int
forall a. [a] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length [a]
xs Int -> Int -> Int
forall a. Integral a => a -> a -> a
`quot` Int
2)