module Air.Light where import Control.Arrow ((&&&), (>>>), (<<<)) import Control.Category (Category) import Data.Char import Data.Foldable (elem, foldl, foldl', toList, Foldable) import Data.Function (on) import Debug.Trace import Prelude hiding ((.), (^), (>), (<), (/), (-), elem, foldl, foldl1, length, drop, take, splitAt, replicate, (!!)) import qualified Prelude as P import System.FilePath (()) import qualified Data.Array as A import qualified Data.List as L import qualified Data.Map as M import qualified Data.Set as S import Data.List ( genericDrop, genericLength ) import qualified Control.Monad as Monad import Control.Monad.Trans (liftIO, MonadIO) import Control.Concurrent import System.Exit ( exitWith, ExitCode(ExitSuccess) ) import Control.Applicative ((<$>), (<*>)) import qualified Data.ByteString.Lazy.Char8 as LazyByteString import qualified Data.ByteString.Char8 as StrictByteString import Data.Maybe (listToMaybe) import Air.Data.Default (def, Default) -- base DSL {-# INLINE (.) #-} (.) :: a -> (a -> b) -> b a . f = f a infixl 9 . (>) :: (Category cat) => cat a b -> cat b c -> cat a c (>) = (>>>) infixl 8 > (<) :: (Category cat) => cat b c -> cat a b -> cat a c (<) = (<<<) infixr 8 < (^) :: (Functor f) => f a -> (a -> b) -> f b (^) = flip fmap infixl 8 ^ (/) :: FilePath -> FilePath -> FilePath (/) = () infixl 5 / {-# INLINE (-) #-} (-) :: (a -> b) -> a -> b f - x = f x infixr 0 - (<->) :: (Num a) => a -> a -> a (<->) = (P.-) infix 6 <-> -- List join :: [a] -> [[a]] -> [a] join = L.intercalate at :: (Integral i) => i -> [a] -> Maybe a at i xs = xs.drop i.first first, second, third, forth, fifth :: [a] -> Maybe a sixth, seventh, eighth, ninth, tenth :: [a] -> Maybe a first = listToMaybe second = at 1 third = at 2 forth = at 3 fifth = at 4 sixth = at 5 seventh = at 6 eighth = at 7 ninth = at 8 tenth = at 10 -- Set requires Ord instance, so use nub when -- xs is not comparable unique :: (Ord a) => [a] -> [a] unique = to_set > to_list is_unique :: (Ord a) => [a] -> Bool is_unique xs = xs.unique.length == xs.length same :: (Ord a) => [a] -> Bool same = unique > length > is 1 times :: (Integral i) => b -> i -> [b] times = flip replicate upto :: (Enum a) => a -> a -> [a] upto = flip enumFromTo downto :: (Num t, Enum t) => t -> t -> [t] downto m n = [n, n <-> 1.. m] remove_at :: (Integral i) => i -> [a] -> [a] remove_at n xs = xs.take n ++ xs.drop (n+1) insert_at, replace_at :: (Integral i) => i -> a -> [a] -> [a] insert_at n x xs = splitted.fst ++ [x] ++ splitted.snd where splitted = xs.splitAt n replace_at n x xs = xs.take n ++ [x] ++ xs.drop (n+1) slice :: (Integral i) => i -> i -> [a] -> [a] slice l r = take r > drop l cherry_pick :: (Integral i) => [i] -> [a] -> [Maybe a] cherry_pick ids xs = ids.map(\i -> xs.at i) inject, inject' :: (Foldable t) => a -> (a -> b -> a) -> t b -> a inject = flip foldl inject' = flip foldl' reduce, reduce' :: (Default a, Foldable t) => (a -> a -> a) -> t a -> a reduce = inject def reduce' = inject' def select, reject :: (a -> Bool) -> [a] -> [a] select = filter reject f = filter (f > not) label_by :: (a -> c) -> [a] -> [(c, a)] label_by f = map (f &&& id) labeling :: (a -> c') -> [a] -> [(a, c')] labeling f = map(id &&& f) in_group_of :: (Integral i) => i -> [t] -> [[t]] in_group_of _ [] = [] in_group_of n xs = h : t.in_group_of(n) where (h, t) = xs.splitAt(n) split_to :: (Integral i) => i -> [a] -> [[a]] split_to n xs = xs.in_group_of(size) where l = xs.length size = if l P.< n then 1 else l `div` n belongs_to :: (Foldable t, Eq a) => t a -> a -> Bool belongs_to = flip elem has :: (Foldable t, Eq b) => b -> t b -> Bool has = flip belongs_to indexed :: (Num t, Enum t) => [b] -> [(t, b)] indexed = zip [0..] ljust, rjust :: (Integral i) => i -> a -> [a] -> [a] rjust n x xs | n P.< xs.length = xs | otherwise = ( n.times x ++ xs ).reverse.take n.reverse ljust n x xs | n P.< xs.length = xs | otherwise = ( xs ++ n.times x ).take n -- faster reverse sort rsort :: (Ord a) => [a] -> [a] rsort xs = xs.L.sortBy(\a b -> b `compare` a) concat_map :: (a -> [b]) -> [a] -> [b] concat_map = concatMap -- Fold to_list :: (Foldable t) => t a -> [a] to_list = toList -- Set to_set :: (Ord a) => [a] -> S.Set a to_set = S.fromList -- Map to_h :: (Ord k) => [(k, a)] -> M.Map k a to_h xs = xs.M.fromList -- Array to_a :: [a] -> A.Array Int a to_a xs = A.listArray (0, xs.length <-> 1) xs to_a' :: (A.Ix i) => (i, i) -> [e] -> A.Array i e to_a' i xs = A.listArray i xs -- Ord compare_by :: (Ord b) => (a -> b) -> a -> a -> Ordering compare_by = on compare eq, is, is_not, isn't, aren't :: (Eq a) => a -> a -> Bool eq = flip (==) is = eq is_not a b = not (is a b) isn't = is_not aren't = is_not -- Tuple swap :: (a, b) -> (b, a) swap (x,y) = (y,x) tuple2 :: [a] -> Maybe (a, a) tuple2 xs = do x <- xs.first y <- xs.second return (x,y) tuple3 :: (Show a) => [a] -> Maybe (a, a, a) tuple3 xs = do x <- xs.first y <- xs.second z <- xs.third return (x,y,z) list2 :: (a, a) -> [a] list2 (x,y) = [x,y] list3 :: (a, a, a) -> [a] list3 (x,y,z) = [x,y,z] filter_fst :: (a -> Bool) -> [(a, b)] -> [(a, b)] filter_fst f = filter(fst > f) filter_snd :: (b -> Bool) -> [(a, b)] -> [(a, b)] filter_snd f = filter(snd > f) map_fst :: (a -> b) -> [(a, c)] -> [(b, c)] map_fst f = map(\(a,b) -> (f a, b)) map_snd :: (a -> b) -> [(c, a)] -> [(c, b)] map_snd f = map(\(a,b) -> (a, f b)) splat :: (a -> b -> c) -> (a, b) -> c splat f (a,b) = f a b splat3 :: (a -> b -> c -> d) -> (a, b, c) -> d splat3 f (a,b,c) = f a b c clone :: a -> (a, a) clone x = (x,x) -- Integer from_i :: (Integral a, Num b) => a -> b from_i = fromIntegral -- String lower, upper :: String -> String lower = map toLower upper = map toUpper starts_with, ends_with :: String -> String -> Bool starts_with = L.isPrefixOf ends_with = L.isSuffixOf capitalize :: String -> String capitalize [] = [] capitalize (x:xs) = [x].upper ++ xs.lower to_s :: (Show a) => a -> String to_s = show -- Debug trace' :: (Show a) => a -> a trace' x = trace (x.show) x -- New from Lab void :: (Monad m) => m a -> m () void x = x >>= const () > return don't :: (Monad m) => m a -> m () don't = const - return () length :: (Num i) => [a] -> i length = L.genericLength drop :: (Integral i) => i -> [a] -> [a] drop = L.genericDrop take :: Integral i => i -> [a] -> [a] take = L.genericTake splitAt :: Integral i => i -> [b] -> ([b], [b]) splitAt = L.genericSplitAt index :: Integral a => [b] -> a -> b index = L.genericIndex replicate :: Integral i => i -> a -> [a] replicate = L.genericReplicate (!!) :: Integral a => [b] -> a -> Maybe b (!!) = flip at to_f :: (Real a, Fractional b) => a -> b to_f = realToFrac sleep :: (RealFrac a) => a -> IO () sleep x = threadDelay - round - (x * 1000000) puts :: String -> IO () puts = putStrLn exit_success :: IO () exit_success = exitWith ExitSuccess fork :: IO a -> IO () fork io = void - forkIO - void io insert_unique :: (Eq a) => a -> [a] -> [a] insert_unique x xs = x : xs.reject (is x) end :: (Monad m) => m () end = return () io :: (MonadIO m) => IO a -> m a io = liftIO l2s :: LazyByteString.ByteString -> StrictByteString.ByteString l2s x = StrictByteString.concat - LazyByteString.toChunks x s2l :: StrictByteString.ByteString -> LazyByteString.ByteString s2l x = LazyByteString.fromChunks [x] ap2 f x1 z = f <$> x1 <*> z ap3 f x1 x2 z = ap2 f x1 x2 <*> z ap4 f x1 x2 x3 z = ap3 f x1 x2 x3 <*> z ap5 f x1 x2 x3 x4 z = ap4 f x1 x2 x3 x4 <*> z ap6 f x1 x2 x3 x4 x5 z = ap5 f x1 x2 x3 x4 x5 <*> z ap7 f x1 x2 x3 x4 x5 x6 z = ap6 f x1 x2 x3 x4 x5 x6 <*> z ap8 f x1 x2 x3 x4 x5 x6 x7 z = ap7 f x1 x2 x3 x4 x5 x6 x7 <*> z ap9 f x1 x2 x3 x4 x5 x6 x7 x8 z = ap8 f x1 x2 x3 x4 x5 x6 x7 x8 <*> z