{-# Language ImportQualifiedPost, OverloadedStrings #-}
module Main (main) where
import Advent (countBy, getInputLines)
import Advent.Coord (Coord(C), above, below, left, right, origin)
import Advent.Search (bfsOn)
import Control.Applicative (Alternative((<|>), many))
import Control.Monad (foldM)
import Data.Set (Set)
import Data.Set qualified as Set
import Text.ParserCombinators.ReadP (ReadP, between, readP_to_S, sepBy1)
data Dir = N | S | E | W
deriving (Dir -> Dir -> Bool
(Dir -> Dir -> Bool) -> (Dir -> Dir -> Bool) -> Eq Dir
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: Dir -> Dir -> Bool
== :: Dir -> Dir -> Bool
$c/= :: Dir -> Dir -> Bool
/= :: Dir -> Dir -> Bool
Eq, Eq Dir
Eq Dir =>
(Dir -> Dir -> Ordering)
-> (Dir -> Dir -> Bool)
-> (Dir -> Dir -> Bool)
-> (Dir -> Dir -> Bool)
-> (Dir -> Dir -> Bool)
-> (Dir -> Dir -> Dir)
-> (Dir -> Dir -> Dir)
-> Ord Dir
Dir -> Dir -> Bool
Dir -> Dir -> Ordering
Dir -> Dir -> Dir
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 :: Dir -> Dir -> Ordering
compare :: Dir -> Dir -> Ordering
$c< :: Dir -> Dir -> Bool
< :: Dir -> Dir -> Bool
$c<= :: Dir -> Dir -> Bool
<= :: Dir -> Dir -> Bool
$c> :: Dir -> Dir -> Bool
> :: Dir -> Dir -> Bool
$c>= :: Dir -> Dir -> Bool
>= :: Dir -> Dir -> Bool
$cmax :: Dir -> Dir -> Dir
max :: Dir -> Dir -> Dir
$cmin :: Dir -> Dir -> Dir
min :: Dir -> Dir -> Dir
Ord, Int -> Dir -> ShowS
[Dir] -> ShowS
Dir -> String
(Int -> Dir -> ShowS)
-> (Dir -> String) -> ([Dir] -> ShowS) -> Show Dir
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> Dir -> ShowS
showsPrec :: Int -> Dir -> ShowS
$cshow :: Dir -> String
show :: Dir -> String
$cshowList :: [Dir] -> ShowS
showList :: [Dir] -> ShowS
Show)
newtype Regexp a = RE [[Either a (Regexp a)]]
deriving (Regexp a -> Regexp a -> Bool
(Regexp a -> Regexp a -> Bool)
-> (Regexp a -> Regexp a -> Bool) -> Eq (Regexp a)
forall a. Eq a => Regexp a -> Regexp a -> Bool
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: forall a. Eq a => Regexp a -> Regexp a -> Bool
== :: Regexp a -> Regexp a -> Bool
$c/= :: forall a. Eq a => Regexp a -> Regexp a -> Bool
/= :: Regexp a -> Regexp a -> Bool
Eq, Eq (Regexp a)
Eq (Regexp a) =>
(Regexp a -> Regexp a -> Ordering)
-> (Regexp a -> Regexp a -> Bool)
-> (Regexp a -> Regexp a -> Bool)
-> (Regexp a -> Regexp a -> Bool)
-> (Regexp a -> Regexp a -> Bool)
-> (Regexp a -> Regexp a -> Regexp a)
-> (Regexp a -> Regexp a -> Regexp a)
-> Ord (Regexp a)
Regexp a -> Regexp a -> Bool
Regexp a -> Regexp a -> Ordering
Regexp a -> Regexp a -> Regexp a
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
forall a. Ord a => Eq (Regexp a)
forall a. Ord a => Regexp a -> Regexp a -> Bool
forall a. Ord a => Regexp a -> Regexp a -> Ordering
forall a. Ord a => Regexp a -> Regexp a -> Regexp a
$ccompare :: forall a. Ord a => Regexp a -> Regexp a -> Ordering
compare :: Regexp a -> Regexp a -> Ordering
$c< :: forall a. Ord a => Regexp a -> Regexp a -> Bool
< :: Regexp a -> Regexp a -> Bool
$c<= :: forall a. Ord a => Regexp a -> Regexp a -> Bool
<= :: Regexp a -> Regexp a -> Bool
$c> :: forall a. Ord a => Regexp a -> Regexp a -> Bool
> :: Regexp a -> Regexp a -> Bool
$c>= :: forall a. Ord a => Regexp a -> Regexp a -> Bool
>= :: Regexp a -> Regexp a -> Bool
$cmax :: forall a. Ord a => Regexp a -> Regexp a -> Regexp a
max :: Regexp a -> Regexp a -> Regexp a
$cmin :: forall a. Ord a => Regexp a -> Regexp a -> Regexp a
min :: Regexp a -> Regexp a -> Regexp a
Ord, Int -> Regexp a -> ShowS
[Regexp a] -> ShowS
Regexp a -> String
(Int -> Regexp a -> ShowS)
-> (Regexp a -> String) -> ([Regexp a] -> ShowS) -> Show (Regexp a)
forall a. Show a => Int -> Regexp a -> ShowS
forall a. Show a => [Regexp a] -> ShowS
forall a. Show a => Regexp a -> String
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: forall a. Show a => Int -> Regexp a -> ShowS
showsPrec :: Int -> Regexp a -> ShowS
$cshow :: forall a. Show a => Regexp a -> String
show :: Regexp a -> String
$cshowList :: forall a. Show a => [Regexp a] -> ShowS
showList :: [Regexp a] -> ShowS
Show)
main :: IO ()
IO ()
main =
do [input] <- Int -> Int -> IO [String]
getInputLines Int
2018 Int
20
let [(re,_)] = readP_to_S parseRe0 input
let (doors, _) = route (Set.singleton origin) re
let ds = (Coord -> [Coord]) -> Coord -> [Int]
forall a. Ord a => (a -> [a]) -> a -> [Int]
distances (Set Coord -> Coord -> [Coord]
neighbor Set Coord
doors) (Int -> Int -> Coord
C Int
0 Int
0)
print (maximum ds)
print (countBy (>= 1000) ds)
parseRe0 :: ReadP (Regexp Dir)
parseRe0 :: ReadP (Regexp Dir)
parseRe0 = ReadP String
-> ReadP String -> ReadP (Regexp Dir) -> ReadP (Regexp Dir)
forall open close a.
ReadP open -> ReadP close -> ReadP a -> ReadP a
between ReadP String
"^" ReadP String
"$" ReadP (Regexp Dir)
parseRe1
parseRe1 :: ReadP (Regexp Dir)
parseRe1 :: ReadP (Regexp Dir)
parseRe1 = [[Either Dir (Regexp Dir)]] -> Regexp Dir
forall a. [[Either a (Regexp a)]] -> Regexp a
RE ([[Either Dir (Regexp Dir)]] -> Regexp Dir)
-> ReadP [[Either Dir (Regexp Dir)]] -> ReadP (Regexp Dir)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> ReadP (Either Dir (Regexp Dir)) -> ReadP [Either Dir (Regexp Dir)]
forall a. ReadP a -> ReadP [a]
forall (f :: * -> *) a. Alternative f => f a -> f [a]
many ReadP (Either Dir (Regexp Dir))
parseRe2 ReadP [Either Dir (Regexp Dir)]
-> ReadP String -> ReadP [[Either Dir (Regexp Dir)]]
forall a sep. ReadP a -> ReadP sep -> ReadP [a]
`sepBy1` ReadP String
"|"
parseRe2 :: ReadP (Either Dir (Regexp Dir))
parseRe2 :: ReadP (Either Dir (Regexp Dir))
parseRe2 = Regexp Dir -> Either Dir (Regexp Dir)
forall a b. b -> Either a b
Right (Regexp Dir -> Either Dir (Regexp Dir))
-> ReadP (Regexp Dir) -> ReadP (Either Dir (Regexp Dir))
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> ReadP String
-> ReadP String -> ReadP (Regexp Dir) -> ReadP (Regexp Dir)
forall open close a.
ReadP open -> ReadP close -> ReadP a -> ReadP a
between ReadP String
"(" ReadP String
")" ReadP (Regexp Dir)
parseRe1 ReadP (Either Dir (Regexp Dir))
-> ReadP (Either Dir (Regexp Dir))
-> ReadP (Either Dir (Regexp Dir))
forall a. ReadP a -> ReadP a -> ReadP a
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> Dir -> Either Dir (Regexp Dir)
forall a b. a -> Either a b
Left (Dir -> Either Dir (Regexp Dir))
-> ReadP Dir -> ReadP (Either Dir (Regexp Dir))
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> ReadP Dir
parseDir
parseDir :: ReadP Dir
parseDir :: ReadP Dir
parseDir = Dir
N Dir -> ReadP String -> ReadP Dir
forall a b. a -> ReadP b -> ReadP a
forall (f :: * -> *) a b. Functor f => a -> f b -> f a
<$ ReadP String
"N" ReadP Dir -> ReadP Dir -> ReadP Dir
forall a. ReadP a -> ReadP a -> ReadP a
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> Dir
S Dir -> ReadP String -> ReadP Dir
forall a b. a -> ReadP b -> ReadP a
forall (f :: * -> *) a b. Functor f => a -> f b -> f a
<$ ReadP String
"S" ReadP Dir -> ReadP Dir -> ReadP Dir
forall a. ReadP a -> ReadP a -> ReadP a
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> Dir
E Dir -> ReadP String -> ReadP Dir
forall a b. a -> ReadP b -> ReadP a
forall (f :: * -> *) a b. Functor f => a -> f b -> f a
<$ ReadP String
"E" ReadP Dir -> ReadP Dir -> ReadP Dir
forall a. ReadP a -> ReadP a -> ReadP a
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> Dir
W Dir -> ReadP String -> ReadP Dir
forall a b. a -> ReadP b -> ReadP a
forall (f :: * -> *) a b. Functor f => a -> f b -> f a
<$ ReadP String
"W"
move :: Dir -> Coord -> Coord
move :: Dir -> Coord -> Coord
move Dir
N = Coord -> Coord
above
move Dir
S = Coord -> Coord
below
move Dir
W = Coord -> Coord
left
move Dir
E = Coord -> Coord
right
neighbor :: Set Coord -> Coord -> [Coord]
neighbor :: Set Coord -> Coord -> [Coord]
neighbor Set Coord
doors Coord
here =
[ Dir -> Coord -> Coord
move Dir
dir (Dir -> Coord -> Coord
move Dir
dir Coord
here)
| Dir
dir <- [Dir
N,Dir
E,Dir
S,Dir
W]
, Coord -> Set Coord -> Bool
forall a. Ord a => a -> Set a -> Bool
Set.member (Dir -> Coord -> Coord
move Dir
dir Coord
here) Set Coord
doors ]
data WithLen a = WithLen { forall a. WithLen a -> Int
dist :: !Int, forall a. WithLen a -> a
loc :: a }
distances :: Ord a => (a -> [a]) -> a -> [Int]
distances :: forall a. Ord a => (a -> [a]) -> a -> [Int]
distances a -> [a]
next a
start = WithLen a -> Int
forall a. WithLen a -> Int
dist (WithLen a -> Int) -> [WithLen a] -> [Int]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> (WithLen a -> a)
-> (WithLen a -> [WithLen a]) -> WithLen a -> [WithLen a]
forall r a. Ord r => (a -> r) -> (a -> [a]) -> a -> [a]
bfsOn WithLen a -> a
forall a. WithLen a -> a
loc (\(WithLen Int
x a
y) -> Int -> a -> WithLen a
forall a. Int -> a -> WithLen a
WithLen (Int
xInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1) (a -> WithLen a) -> [a] -> [WithLen a]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> a -> [a]
next a
y) (Int -> a -> WithLen a
forall a. Int -> a -> WithLen a
WithLen Int
0 a
start)
route :: Set Coord -> Regexp Dir -> (Set Coord, Set Coord)
route :: Set Coord -> Regexp Dir -> (Set Coord, Set Coord)
route Set Coord
starts (RE [[Either Dir (Regexp Dir)]]
alts) = ([Either Dir (Regexp Dir)] -> (Set Coord, Set Coord))
-> [[Either Dir (Regexp Dir)]] -> (Set Coord, Set Coord)
forall m a. Monoid m => (a -> m) -> [a] -> m
forall (t :: * -> *) m a.
(Foldable t, Monoid m) =>
(a -> m) -> t a -> m
foldMap ((Set Coord -> Either Dir (Regexp Dir) -> (Set Coord, Set Coord))
-> Set Coord -> [Either Dir (Regexp Dir)] -> (Set Coord, Set Coord)
forall (t :: * -> *) (m :: * -> *) b a.
(Foldable t, Monad m) =>
(b -> a -> m b) -> b -> t a -> m b
foldM Set Coord -> Either Dir (Regexp Dir) -> (Set Coord, Set Coord)
routeFrom Set Coord
starts) [[Either Dir (Regexp Dir)]]
alts
routeFrom :: Set Coord -> Either Dir (Regexp Dir) -> (Set Coord, Set Coord)
routeFrom :: Set Coord -> Either Dir (Regexp Dir) -> (Set Coord, Set Coord)
routeFrom Set Coord
starts = (Dir -> (Set Coord, Set Coord))
-> (Regexp Dir -> (Set Coord, Set Coord))
-> Either Dir (Regexp Dir)
-> (Set Coord, Set Coord)
forall a c b. (a -> c) -> (b -> c) -> Either a b -> c
either (Set Coord -> Dir -> (Set Coord, Set Coord)
dirStep Set Coord
starts) (Set Coord -> Regexp Dir -> (Set Coord, Set Coord)
route Set Coord
starts)
dirStep :: Set Coord -> Dir -> (Set Coord, Set Coord)
dirStep :: Set Coord -> Dir -> (Set Coord, Set Coord)
dirStep Set Coord
starts Dir
d = ( (Coord -> Coord) -> Set Coord -> Set Coord
forall a b. (a -> b) -> Set a -> Set b
Set.mapMonotonic (Dir -> Coord -> Coord
move Dir
d) Set Coord
starts
, (Coord -> Coord) -> Set Coord -> Set Coord
forall a b. (a -> b) -> Set a -> Set b
Set.mapMonotonic (Dir -> Coord -> Coord
move Dir
d (Coord -> Coord) -> (Coord -> Coord) -> Coord -> Coord
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Dir -> Coord -> Coord
move Dir
d) Set Coord
starts)