{-# Language QuasiQuotes #-}
module Main where
import Advent (format)
import Advent.Coord (Coord(..), cardinal)
import Advent.Search (bfsOn)
import Data.Bits (Bits(popCount))
data Entry = Entry { Entry -> Int
entrySteps :: !Int, Entry -> Coord
entryCoord :: Coord }
deriving (Entry -> Entry -> Bool
(Entry -> Entry -> Bool) -> (Entry -> Entry -> Bool) -> Eq Entry
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: Entry -> Entry -> Bool
== :: Entry -> Entry -> Bool
$c/= :: Entry -> Entry -> Bool
/= :: Entry -> Entry -> Bool
Eq, Int -> Entry -> ShowS
[Entry] -> ShowS
Entry -> String
(Int -> Entry -> ShowS)
-> (Entry -> String) -> ([Entry] -> ShowS) -> Show Entry
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> Entry -> ShowS
showsPrec :: Int -> Entry -> ShowS
$cshow :: Entry -> String
show :: Entry -> String
$cshowList :: [Entry] -> ShowS
showList :: [Entry] -> ShowS
Show)
main :: IO ()
IO ()
main =
do input <- [format|2016 13 %u%n|]
let entries = (Entry -> Coord) -> (Entry -> [Entry]) -> Entry -> [Entry]
forall r a. Ord r => (a -> r) -> (a -> [a]) -> a -> [a]
bfsOn Entry -> Coord
entryCoord (Int -> Entry -> [Entry]
nextEntries Int
input) Entry
initialEntry
print $ head [steps | Entry steps (C 39 31) <- entries]
print $ length $ takeWhile (<= 50) $ map entrySteps entries
initialEntry :: Entry
initialEntry :: Entry
initialEntry = Int -> Coord -> Entry
Entry Int
0 (Int -> Int -> Coord
C Int
1 Int
1)
isValidCoord :: Int -> Coord -> Bool
isValidCoord :: Int -> Coord -> Bool
isValidCoord Int
input (C Int
y Int
x) =
Int
x Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
>= Int
0 Bool -> Bool -> Bool
&& Int
y Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
>= Int
0 Bool -> Bool -> Bool
&&
Int -> Bool
forall a. Integral a => a -> Bool
even (Int -> Int
forall a. Bits a => a -> Int
popCount (Int
xInt -> Int -> Int
forall a. Num a => a -> a -> a
*Int
x Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
3Int -> Int -> Int
forall a. Num a => a -> a -> a
*Int
x Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
2Int -> Int -> Int
forall a. Num a => a -> a -> a
*Int
xInt -> Int -> Int
forall a. Num a => a -> a -> a
*Int
y Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
y Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
yInt -> Int -> Int
forall a. Num a => a -> a -> a
*Int
y Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
input))
nextEntries :: Int -> Entry -> [Entry]
nextEntries :: Int -> Entry -> [Entry]
nextEntries Int
input (Entry Int
steps Coord
coord) =
[Int -> Coord -> Entry
Entry (Int
stepsInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1) Coord
c | Coord
c <- Coord -> [Coord]
cardinal Coord
coord, Int -> Coord -> Bool
isValidCoord Int
input Coord
c]