module Data.Edison.Seq.SizedSeq (
Sized,
empty,singleton,lcons,rcons,append,lview,lhead,ltail,rview,rhead,rtail,
lheadM,ltailM,rheadM,rtailM,
null,size,concat,reverse,reverseOnto,fromList,toList,map,concatMap,
fold,fold',fold1,fold1',foldr,foldr',foldl,foldl',foldr1,foldr1',foldl1,foldl1',
reducer,reducer',reducel,reducel',reduce1,reduce1',
copy,inBounds,lookup,lookupM,lookupWithDefault,update,adjust,
mapWithIndex,foldrWithIndex,foldlWithIndex,foldrWithIndex',foldlWithIndex',
take,drop,splitAt,subseq,filter,partition,takeWhile,dropWhile,splitWhile,
zip,zip3,zipWith,zipWith3,unzip,unzip3,unzipWith,unzipWith3,
strict, strictWith,
structuralInvariant,
moduleName,instanceName,
fromSeq,toSeq
) where
import Prelude hiding (concat,reverse,map,concatMap,foldr,foldl,foldr1,foldl1,
filter,takeWhile,dropWhile,lookup,take,drop,splitAt,
zip,zip3,zipWith,zipWith3,unzip,unzip3,null)
import qualified Control.Applicative as App
import qualified Data.Edison.Seq as S
import qualified Data.Edison.Seq.ListSeq as L
import Data.Edison.Seq.Defaults
import Data.Monoid
import Data.Semigroup as SG
import Control.Monad
import Test.QuickCheck
moduleName :: String
instanceName :: S.Sequence s => Sized s a -> String
empty :: S.Sequence s => Sized s a
singleton :: S.Sequence s => a -> Sized s a
lcons :: S.Sequence s => a -> Sized s a -> Sized s a
rcons :: S.Sequence s => a -> Sized s a -> Sized s a
append :: S.Sequence s => Sized s a -> Sized s a -> Sized s a
lview :: (S.Sequence s, Monad m) => Sized s a -> m (a, Sized s a)
lhead :: S.Sequence s => Sized s a -> a
lheadM :: (S.Sequence s, Monad m) => Sized s a -> m a
ltail :: S.Sequence s => Sized s a -> Sized s a
ltailM :: (S.Sequence s, Monad m) => Sized s a -> m (Sized s a)
rview :: (S.Sequence s, Monad m) => Sized s a -> m (a, Sized s a)
rhead :: S.Sequence s => Sized s a -> a
rheadM :: (S.Sequence s, Monad m) => Sized s a -> m a
rtail :: S.Sequence s => Sized s a -> Sized s a
rtailM :: (S.Sequence s, Monad m) => Sized s a -> m (Sized s a)
null :: S.Sequence s => Sized s a -> Bool
size :: S.Sequence s => Sized s a -> Int
concat :: S.Sequence s => Sized s (Sized s a) -> Sized s a
reverse :: S.Sequence s => Sized s a -> Sized s a
reverseOnto :: S.Sequence s => Sized s a -> Sized s a -> Sized s a
fromList :: S.Sequence s => [a] -> Sized s a
toList :: S.Sequence s => Sized s a -> [a]
map :: S.Sequence s => (a -> b) -> Sized s a -> Sized s b
concatMap :: S.Sequence s => (a -> Sized s b) -> Sized s a -> Sized s b
fold :: S.Sequence s => (a -> b -> b) -> b -> Sized s a -> b
fold' :: S.Sequence s => (a -> b -> b) -> b -> Sized s a -> b
fold1 :: S.Sequence s => (a -> a -> a) -> Sized s a -> a
fold1' :: S.Sequence s => (a -> a -> a) -> Sized s a -> a
foldr :: S.Sequence s => (a -> b -> b) -> b -> Sized s a -> b
foldl :: S.Sequence s => (b -> a -> b) -> b -> Sized s a -> b
foldr1 :: S.Sequence s => (a -> a -> a) -> Sized s a -> a
foldl1 :: S.Sequence s => (a -> a -> a) -> Sized s a -> a
reducer :: S.Sequence s => (a -> a -> a) -> a -> Sized s a -> a
reducel :: S.Sequence s => (a -> a -> a) -> a -> Sized s a -> a
reduce1 :: S.Sequence s => (a -> a -> a) -> Sized s a -> a
foldr' :: S.Sequence s => (a -> b -> b) -> b -> Sized s a -> b
foldl' :: S.Sequence s => (b -> a -> b) -> b -> Sized s a -> b
foldr1' :: S.Sequence s => (a -> a -> a) -> Sized s a -> a
foldl1' :: S.Sequence s => (a -> a -> a) -> Sized s a -> a
reducer' :: S.Sequence s => (a -> a -> a) -> a -> Sized s a -> a
reducel' :: S.Sequence s => (a -> a -> a) -> a -> Sized s a -> a
reduce1' :: S.Sequence s => (a -> a -> a) -> Sized s a -> a
copy :: S.Sequence s => Int -> a -> Sized s a
inBounds :: S.Sequence s => Int -> Sized s a -> Bool
lookup :: S.Sequence s => Int -> Sized s a -> a
lookupM :: (S.Sequence s, Monad m) => Int -> Sized s a -> m a
lookupWithDefault :: S.Sequence s => a -> Int -> Sized s a -> a
update :: S.Sequence s => Int -> a -> Sized s a -> Sized s a
adjust :: S.Sequence s => (a -> a) -> Int -> Sized s a -> Sized s a
mapWithIndex :: S.Sequence s => (Int -> a -> b) -> Sized s a -> Sized s b
foldrWithIndex :: S.Sequence s => (Int -> a -> b -> b) -> b -> Sized s a -> b
foldlWithIndex :: S.Sequence s => (b -> Int -> a -> b) -> b -> Sized s a -> b
foldrWithIndex' :: S.Sequence s => (Int -> a -> b -> b) -> b -> Sized s a -> b
foldlWithIndex' :: S.Sequence s => (b -> Int -> a -> b) -> b -> Sized s a -> b
take :: S.Sequence s => Int -> Sized s a -> Sized s a
drop :: S.Sequence s => Int -> Sized s a -> Sized s a
splitAt :: S.Sequence s => Int -> Sized s a -> (Sized s a, Sized s a)
subseq :: S.Sequence s => Int -> Int -> Sized s a -> Sized s a
filter :: S.Sequence s => (a -> Bool) -> Sized s a -> Sized s a
partition :: S.Sequence s => (a -> Bool) -> Sized s a -> (Sized s a, Sized s a)
takeWhile :: S.Sequence s => (a -> Bool) -> Sized s a -> Sized s a
dropWhile :: S.Sequence s => (a -> Bool) -> Sized s a -> Sized s a
splitWhile :: S.Sequence s => (a -> Bool) -> Sized s a -> (Sized s a, Sized s a)
zip :: S.Sequence s => Sized s a -> Sized s b -> Sized s (a,b)
zip3 :: S.Sequence s => Sized s a -> Sized s b -> Sized s c -> Sized s (a,b,c)
zipWith :: S.Sequence s => (a -> b -> c) -> Sized s a -> Sized s b -> Sized s c
zipWith3 :: S.Sequence s => (a -> b -> c -> d) -> Sized s a -> Sized s b -> Sized s c -> Sized s d
unzip :: S.Sequence s => Sized s (a,b) -> (Sized s a, Sized s b)
unzip3 :: S.Sequence s => Sized s (a,b,c) -> (Sized s a, Sized s b, Sized s c)
unzipWith :: S.Sequence s => (a -> b) -> (a -> c) -> Sized s a -> (Sized s b, Sized s c)
unzipWith3 :: S.Sequence s => (a -> b) -> (a -> c) -> (a -> d) -> Sized s a -> (Sized s b, Sized s c, Sized s d)
strict :: S.Sequence s => Sized s a -> Sized s a
strictWith :: S.Sequence s => (a -> b) -> Sized s a -> Sized s a
structuralInvariant :: S.Sequence s => Sized s a -> Bool
fromSeq :: S.Sequence s => s a -> Sized s a
toSeq :: S.Sequence s => Sized s a -> s a
moduleName = "Data.Edison.Seq.SizedSeq"
instanceName (N _ s) = "SizedSeq(" ++ S.instanceName s ++ ")"
data Sized s a = N !Int (s a)
fromSeq xs = N (S.size xs) xs
toSeq (N _ xs) = xs
empty = N 0 S.empty
singleton x = N 1 (S.singleton x)
lcons x (N n xs) = N (n+1) (S.lcons x xs)
rcons x (N n xs) = N (n+1) (S.rcons x xs)
append (N m xs) (N n ys) = N (m+n) (S.append xs ys)
lview (N n xs) = case S.lview xs of
Nothing -> fail "SizedSeq.lview: empty sequence"
Just (x,xs) -> return (x, N (n1) xs)
lhead (N _ xs) = S.lhead xs
lheadM (N _ xs) = S.lheadM xs
ltail (N 0 _) = error "SizedSeq.ltail: empty sequence"
ltail (N n xs) = N (n1) (S.ltail xs)
ltailM (N 0 _) = fail "SizedSeq.ltailM: empty sequence"
ltailM (N n xs) = return (N (n1) (S.ltail xs))
rview (N n xs) = case S.rview xs of
Nothing -> fail "SizedSeq.rview: empty sequence"
Just (x,xs) -> return (x, N (n1) xs)
rhead (N _ xs) = S.rhead xs
rheadM (N _ xs) = S.rheadM xs
rtail (N 0 _) = error "SizedSeq.rtail: empty sequence"
rtail (N n xs) = N (n1) (S.rtail xs)
rtailM (N 0 _) = fail "SizedSeq.rtailM: empty sequence"
rtailM (N n xs) = return (N (n1) (S.rtail xs))
null (N n _) = n == 0
size (N n _) = n
concat (N _ xss) = fromSeq (S.concat (S.map toSeq xss))
reverse (N n xs) = N n (S.reverse xs)
reverseOnto (N m xs) (N n ys) = N (m+n) (S.reverseOnto xs ys)
fromList = fromSeq . S.fromList
toList (N _ xs) = S.toList xs
map f (N n xs) = N n (S.map f xs)
concatMap = concatMapUsingFoldr
fold f e (N _ xs) = S.fold f e xs
fold' f e (N _ xs) = S.fold' f e xs
fold1 f (N _ xs) = S.fold1 f xs
fold1' f (N _ xs) = S.fold1' f xs
foldr f e (N _ xs) = S.foldr f e xs
foldr' f e (N _ xs) = S.foldr' f e xs
foldl f e (N _ xs) = S.foldl f e xs
foldl' f e (N _ xs) = S.foldl' f e xs
foldr1 f (N _ xs) = S.foldr1 f xs
foldr1' f (N _ xs) = S.foldr1' f xs
foldl1 f (N _ xs) = S.foldl1 f xs
foldl1' f (N _ xs) = S.foldl1' f xs
reducer f e (N _ xs) = S.reducer f e xs
reducer' f e (N _ xs) = S.reducer' f e xs
reducel f e (N _ xs) = S.reducel f e xs
reducel' f e (N _ xs) = S.reducel' f e xs
reduce1 f (N _ xs) = S.reduce1 f xs
reduce1' f (N _ xs) = S.reduce1' f xs
copy n x
| n <= 0 = empty
| otherwise = N n (S.copy n x)
inBounds i (N n _) = (i >= 0) && (i < n)
lookup i (N _ xs) = S.lookup i xs
lookupM i (N _ xs) = S.lookupM i xs
lookupWithDefault d i (N _ xs) = S.lookupWithDefault d i xs
update i x (N n xs) = N n (S.update i x xs)
adjust f i (N n xs) = N n (S.adjust f i xs)
mapWithIndex f (N n xs) = N n (S.mapWithIndex f xs)
foldrWithIndex f e (N _ xs) = S.foldrWithIndex f e xs
foldrWithIndex' f e (N _ xs) = S.foldrWithIndex' f e xs
foldlWithIndex f e (N _ xs) = S.foldlWithIndex f e xs
foldlWithIndex' f e (N _ xs) = S.foldlWithIndex' f e xs
take i original@(N n xs)
| i <= 0 = empty
| i >= n = original
| otherwise = N i (S.take i xs)
drop i original@(N n xs)
| i <= 0 = original
| i >= n = empty
| otherwise = N (ni) (S.drop i xs)
splitAt i original@(N n xs)
| i <= 0 = (empty, original)
| i >= n = (original, empty)
| otherwise = let (ys,zs) = S.splitAt i xs
in (N i ys, N (ni) zs)
subseq i len original@(N n xs)
| i <= 0 = take len original
| i >= n || len <= 0 = empty
| i+len >= n = N (ni) (S.drop i xs)
| otherwise = N len (S.subseq i len xs)
filter p = fromSeq . S.filter p . toSeq
partition p (N n xs) = (N m ys, N (nm) zs)
where (ys,zs) = S.partition p xs
m = S.size ys
takeWhile p = fromSeq . S.takeWhile p . toSeq
dropWhile p = fromSeq . S.dropWhile p . toSeq
splitWhile p (N n xs) = (N m ys, N (nm) zs)
where (ys,zs) = S.splitWhile p xs
m = S.size ys
zip (N m xs) (N n ys) = N (min m n) (S.zip xs ys)
zip3 (N l xs) (N m ys) (N n zs) = N (min l (min m n)) (S.zip3 xs ys zs)
zipWith f (N m xs) (N n ys) = N (min m n) (S.zipWith f xs ys)
zipWith3 f (N l xs) (N m ys) (N n zs) = N (min l (min m n)) (S.zipWith3 f xs ys zs)
unzip (N n xys) = (N n xs, N n ys)
where (xs,ys) = S.unzip xys
unzip3 (N n xyzs) = (N n xs, N n ys, N n zs)
where (xs,ys,zs) = S.unzip3 xyzs
unzipWith f g (N n xys) = (N n xs, N n ys)
where (xs,ys) = S.unzipWith f g xys
unzipWith3 f g h (N n xyzs) = (N n xs, N n ys, N n zs)
where (xs,ys,zs) = S.unzipWith3 f g h xyzs
strict s@(N _ s') = S.strict s' `seq` s
strictWith f s@(N _ s') = S.strictWith f s' `seq` s
structuralInvariant (N i s) = i == S.size s
instance S.Sequence s => S.Sequence (Sized s) where
{lcons = lcons; rcons = rcons;
lview = lview; lhead = lhead; ltail = ltail;
lheadM = lheadM; ltailM = ltailM; rheadM = rheadM; rtailM = rtailM;
rview = rview; rhead = rhead; rtail = rtail; null = null;
size = size; concat = concat; reverse = reverse;
reverseOnto = reverseOnto; fromList = fromList; toList = toList;
fold = fold; fold' = fold'; fold1 = fold1; fold1' = fold1';
foldr = foldr; foldr' = foldr'; foldl = foldl; foldl' = foldl';
foldr1 = foldr1; foldr1' = foldr1'; foldl1 = foldl1; foldl1' = foldl1';
reducer = reducer; reducer' = reducer'; reducel = reducel;
reducel' = reducel'; reduce1 = reduce1; reduce1' = reduce1';
copy = copy; inBounds = inBounds; lookup = lookup;
lookupM = lookupM; lookupWithDefault = lookupWithDefault;
update = update; adjust = adjust; mapWithIndex = mapWithIndex;
foldrWithIndex = foldrWithIndex; foldrWithIndex' = foldrWithIndex';
foldlWithIndex = foldlWithIndex; foldlWithIndex' = foldlWithIndex';
take = take; drop = drop; splitAt = splitAt; subseq = subseq;
filter = filter; partition = partition; takeWhile = takeWhile;
dropWhile = dropWhile; splitWhile = splitWhile; zip = zip;
zip3 = zip3; zipWith = zipWith; zipWith3 = zipWith3; unzip = unzip;
unzip3 = unzip3; unzipWith = unzipWith; unzipWith3 = unzipWith3;
strict = strict; strictWith = strictWith;
structuralInvariant = structuralInvariant; instanceName = instanceName}
instance S.Sequence s => Functor (Sized s) where
fmap = map
instance S.Sequence s => App.Alternative (Sized s) where
empty = empty
(<|>) = append
instance S.Sequence s => App.Applicative (Sized s) where
pure = return
x <*> y = do
x' <- x
y' <- y
return (x' y')
instance S.Sequence s => Monad (Sized s) where
return = singleton
xs >>= k = concatMap k xs
instance S.Sequence s => MonadPlus (Sized s) where
mplus = append
mzero = empty
instance Eq (s a) => Eq (Sized s a) where
(N m xs) == (N n ys) = (m == n) && (xs == ys)
instance (S.Sequence s, Ord a, Eq (s a)) => Ord (Sized s a) where
compare = defaultCompare
instance (S.Sequence s, Show (s a)) => Show (Sized s a) where
showsPrec i xs rest
| i == 0 = L.concat [ moduleName,".fromSeq ",showsPrec 10 (toSeq xs) rest]
| otherwise = L.concat ["(",moduleName,".fromSeq ",showsPrec 10 (toSeq xs) (')':rest)]
instance (S.Sequence s, Read (s a)) => Read (Sized s a) where
readsPrec _ xs = maybeParens p xs
where p xs = tokenMatch (moduleName++".fromSeq") xs
>>= readsPrec 10
>>= \(l,rest) -> return (fromSeq l, rest)
instance (S.Sequence s, Arbitrary (s a)) => Arbitrary (Sized s a) where
arbitrary = do xs <- arbitrary
return (fromSeq xs)
instance (S.Sequence s, CoArbitrary (s a)) => CoArbitrary (Sized s a) where
coarbitrary xs = coarbitrary (toSeq xs)
instance S.Sequence s => Semigroup (Sized s a) where
(<>) = append
instance S.Sequence s => Monoid (Sized s a) where
mempty = empty
mappend = (SG.<>)