-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathScanner.hs
45 lines (30 loc) · 1.05 KB
/
Scanner.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
-- https://byorgey.wordpress.com/2019/05/22/competitive-programming-in-haskell-scanner/
{-# LANGUAGE LambdaCase #-}
module Scanner where
import Control.Monad (replicateM)
import Control.Monad.State
type Scanner = State [String]
runScanner :: Scanner a -> String -> a
runScanner = runScannerWith words
runScannerWith :: (String -> [String]) -> Scanner a -> String -> a
runScannerWith t s = evalState s . t
str :: Scanner String
str = get >>= \case s : ss -> put ss >> return s
int :: Scanner Int
int = read <$> str
integer :: Scanner Integer
integer = read <$> str
double :: Scanner Double
double = read <$> str
decimal :: Int -> Scanner Int
decimal p = (round . ((10 ^ p) *)) <$> double
numberOf :: Scanner a -> Scanner [a]
numberOf s = int >>= flip replicateM s
many :: Scanner a -> Scanner [a]
many s = get >>= \case [] -> return []; _ -> (:) <$> s <*> many s
times :: Int -> Scanner a -> Scanner [a]
times = replicateM
(><) :: Int -> Scanner a -> Scanner [a]
(><) = times
two, three, four :: Scanner a -> Scanner [a]
[two, three, four] = map times [2 .. 4]