#ifdef KIND_POLYMORPHIC_TYPEABLE
#endif
#if MIN_VERSION_transformers(0,4,0)
#define HAVE_EXCEPTT
#endif
#if MIN_VERSION_base(4,7,0)
#define HAVE_PROXY
#endif
module Data.Monoid.Endo.Fold
(
foldEndo
, dualFoldEndo
, FoldEndoArgs(..)
, AnEndo(..)
, WrappedFoldable(..)
, (&$)
, (<&$>)
, embedEndoWith
, embedDualEndoWith
)
where
import Control.Applicative (Applicative(pure), Const(Const))
import Control.Monad (Monad(return))
import Data.Either (Either(Right))
import Data.Foldable (Foldable(foldMap))
import Data.Function ((.), id)
import Data.Functor (Functor(fmap))
import Data.Functor.Identity (Identity(Identity))
import Data.Maybe (Maybe(Just, Nothing))
import Data.Monoid (Dual(Dual, getDual), Endo(Endo), Monoid(mempty, mconcat), (<>))
import GHC.Generics (Generic)
import System.IO (IO)
import Text.Read (Read)
import Text.Show (Show)
#ifdef KIND_POLYMORPHIC_TYPEABLE
import Data.Data (Data)
import Data.Typeable (Typeable)
#endif
#ifdef HAVE_PROXY
import Data.Proxy (Proxy(Proxy))
#endif
#ifdef HAVE_EXCEPTT
import Control.Monad.Trans.Except (ExceptT)
#endif
import Control.Monad.Trans.Identity (IdentityT)
import Control.Monad.Trans.List (ListT)
import Control.Monad.Trans.Maybe (MaybeT)
import Control.Monad.Trans.Reader (ReaderT)
import Control.Monad.Trans.RWS (RWST)
import qualified Control.Monad.Trans.RWS.Strict as Strict (RWST)
import Control.Monad.Trans.State (StateT)
import qualified Control.Monad.Trans.State.Strict as Strict (StateT)
import Control.Monad.Trans.Writer (WriterT)
import qualified Control.Monad.Trans.Writer.Strict as Strict (WriterT)
import Data.Functor.Compose (Compose)
import Data.Functor.Product (Product)
import Data.Functor.Reverse (Reverse)
foldEndo :: FoldEndoArgs args => args
foldEndo = foldEndoArgs mempty
dualFoldEndo :: FoldEndoArgs args => args
dualFoldEndo = dualFoldEndoArgs mempty
class FoldEndoArgs a where
type ResultOperatesOn a
type Result a
foldEndoArgs :: Endo (ResultOperatesOn a) -> a
dualFoldEndoArgs :: Dual (Endo (ResultOperatesOn a)) -> a
#ifdef HAVE_MINIMAL_PRAGMA
#endif
instance
( AnEndo a
, FoldEndoArgs r
, EndoOperatesOn a ~ ResultOperatesOn r
) => FoldEndoArgs (a -> r)
where
type ResultOperatesOn (a -> r) = ResultOperatesOn r
type Result (a -> r) = Result r
foldEndoArgs e e' = foldEndoArgs (e <> anEndo e')
dualFoldEndoArgs e e' = dualFoldEndoArgs (e <> aDualEndo e')
instance FoldEndoArgs (Endo a) where
type ResultOperatesOn (Endo a) = a
type Result (Endo a) = Endo a
foldEndoArgs = id
dualFoldEndoArgs (Dual e) = e
instance (Monoid c, FoldEndoArgs r) => FoldEndoArgs (Const c r) where
type ResultOperatesOn (Const c r) = ResultOperatesOn r
type Result (Const c r) = Const c (Result r)
foldEndoArgs _ = Const mempty
dualFoldEndoArgs _ = Const mempty
instance FoldEndoArgs r => FoldEndoArgs (Either e r) where
type ResultOperatesOn (Either e r) = ResultOperatesOn r
type Result (Either e r) = Either e (Result r)
foldEndoArgs = Right . foldEndoArgs
dualFoldEndoArgs = Right . dualFoldEndoArgs
instance FoldEndoArgs r => FoldEndoArgs (Identity r) where
type ResultOperatesOn (Identity r) = ResultOperatesOn r
type Result (Identity r) = Identity (Result r)
foldEndoArgs = Identity . foldEndoArgs
dualFoldEndoArgs = Identity . dualFoldEndoArgs
instance FoldEndoArgs r => FoldEndoArgs (IO r) where
type ResultOperatesOn (IO r) = ResultOperatesOn r
type Result (IO r) = IO (Result r)
foldEndoArgs = pure . foldEndoArgs
dualFoldEndoArgs = pure . dualFoldEndoArgs
instance FoldEndoArgs r => FoldEndoArgs (Maybe r) where
type ResultOperatesOn (Maybe r) = ResultOperatesOn r
type Result (Maybe r) = Maybe (Result r)
foldEndoArgs = Just . foldEndoArgs
dualFoldEndoArgs = Just . dualFoldEndoArgs
instance
(Applicative f, Applicative g, FoldEndoArgs r)
=> FoldEndoArgs (Compose f g r)
where
type ResultOperatesOn (Compose f g r) = ResultOperatesOn r
type Result (Compose f g r) = Compose f g (Result r)
foldEndoArgs = pure . foldEndoArgs
dualFoldEndoArgs = pure . dualFoldEndoArgs
instance
(Applicative f, Applicative g, FoldEndoArgs r)
=> FoldEndoArgs (Product f g r)
where
type ResultOperatesOn (Product f g r) = ResultOperatesOn r
type Result (Product f g r) = Product f g (Result r)
foldEndoArgs = pure . foldEndoArgs
dualFoldEndoArgs = pure . dualFoldEndoArgs
instance (Applicative f, FoldEndoArgs r) => FoldEndoArgs (IdentityT f r) where
type ResultOperatesOn (IdentityT f r) = ResultOperatesOn r
type Result (IdentityT f r) = IdentityT f (Result r)
foldEndoArgs = pure . foldEndoArgs
dualFoldEndoArgs = pure . dualFoldEndoArgs
#ifdef HAVE_EXCEPTT
instance
( Monad m
#ifndef APPLICATIVE_MONAD
, Functor m
#endif
, FoldEndoArgs r
) => FoldEndoArgs (ExceptT e m r)
where
type ResultOperatesOn (ExceptT e m r) = ResultOperatesOn r
type Result (ExceptT e m r) = ExceptT e m (Result r)
foldEndoArgs = pure . foldEndoArgs
dualFoldEndoArgs = pure . dualFoldEndoArgs
#endif
instance (Applicative f, FoldEndoArgs r) => FoldEndoArgs (ListT f r) where
type ResultOperatesOn (ListT f r) = ResultOperatesOn r
type Result (ListT f r) = ListT f (Result r)
foldEndoArgs = pure . foldEndoArgs
dualFoldEndoArgs = pure . dualFoldEndoArgs
instance
( Monad m
#ifndef APPLICATIVE_MONAD
, Functor m
#endif
, FoldEndoArgs r
) => FoldEndoArgs (MaybeT m r) where
type ResultOperatesOn (MaybeT m r) = ResultOperatesOn r
type Result (MaybeT m r) = MaybeT m (Result r)
foldEndoArgs = return . foldEndoArgs
dualFoldEndoArgs = return . dualFoldEndoArgs
instance (Applicative f, FoldEndoArgs r) => FoldEndoArgs (ReaderT r' f r) where
type ResultOperatesOn (ReaderT r' f r) = ResultOperatesOn r
type Result (ReaderT r' f r) = ReaderT r' f (Result r)
foldEndoArgs = pure . foldEndoArgs
dualFoldEndoArgs = pure . dualFoldEndoArgs
instance
( Monad m
#ifndef APPLICATIVE_MONAD
, Functor m
#endif
, Monoid w
, FoldEndoArgs r
) => FoldEndoArgs (RWST r' w s m r)
where
type ResultOperatesOn (RWST r' w s m r) = ResultOperatesOn r
type Result (RWST r' w s m r) = RWST r' w s m (Result r)
foldEndoArgs = return . foldEndoArgs
dualFoldEndoArgs = return . dualFoldEndoArgs
instance
( Monad m
#ifndef APPLICATIVE_MONAD
, Functor m
#endif
, Monoid w
, FoldEndoArgs r
) => FoldEndoArgs (Strict.RWST r' w s m r)
where
type ResultOperatesOn (Strict.RWST r' w s m r) = ResultOperatesOn r
type Result (Strict.RWST r' w s m r) = Strict.RWST r' w s m (Result r)
foldEndoArgs = return . foldEndoArgs
dualFoldEndoArgs = return . dualFoldEndoArgs
instance (Monad m, FoldEndoArgs r) => FoldEndoArgs (StateT s m r) where
type ResultOperatesOn (StateT s m r) = ResultOperatesOn r
type Result (StateT s m r) = StateT s m (Result r)
foldEndoArgs = return . foldEndoArgs
dualFoldEndoArgs = return . dualFoldEndoArgs
instance (Monad m, FoldEndoArgs r) => FoldEndoArgs (Strict.StateT s m r) where
type ResultOperatesOn (Strict.StateT s m r) = ResultOperatesOn r
type Result (Strict.StateT s m r) = Strict.StateT s m (Result r)
foldEndoArgs = return . foldEndoArgs
dualFoldEndoArgs = return . dualFoldEndoArgs
instance
(Applicative f, FoldEndoArgs r, Monoid w) => FoldEndoArgs (WriterT w f r)
where
type ResultOperatesOn (WriterT w f r) = ResultOperatesOn r
type Result (WriterT w f r) = WriterT w f (Result r)
foldEndoArgs = pure . foldEndoArgs
dualFoldEndoArgs = pure . dualFoldEndoArgs
instance
(Applicative f, FoldEndoArgs r, Monoid w)
=> FoldEndoArgs (Strict.WriterT w f r)
where
type ResultOperatesOn (Strict.WriterT w f r) = ResultOperatesOn r
type Result (Strict.WriterT w f r) = Strict.WriterT w f (Result r)
foldEndoArgs = pure . foldEndoArgs
dualFoldEndoArgs = pure . dualFoldEndoArgs
class AnEndo a where
type EndoOperatesOn a
anEndo :: a -> Endo (EndoOperatesOn a)
anEndo = getDual . aDualEndo
aDualEndo :: a -> Dual (Endo (EndoOperatesOn a))
aDualEndo = Dual . anEndo
#if HAVE_MINIMAL_PRAGMA
#endif
instance AnEndo (Endo a) where
type EndoOperatesOn (Endo a) = a
anEndo = id
instance AnEndo (a -> a) where
type EndoOperatesOn (a -> a) = a
anEndo = Endo
instance AnEndo a => AnEndo (Maybe a) where
type EndoOperatesOn (Maybe a) = EndoOperatesOn a
anEndo Nothing = mempty
anEndo (Just e) = anEndo e
#ifdef HAVE_PROXY
instance AnEndo (Proxy a) where
type EndoOperatesOn (Proxy a) = a
anEndo Proxy = mempty
aDualEndo Proxy = mempty
#endif
newtype WrappedFoldable f a = WrapFoldable {getFoldable :: f a}
deriving
( Generic
, Read
, Show
#ifdef KIND_POLYMORPHIC_TYPEABLE
, Data
, Typeable
#endif
)
instance (Foldable f, AnEndo a) => AnEndo (WrappedFoldable f a) where
type EndoOperatesOn (WrappedFoldable f a) = EndoOperatesOn a
anEndo (WrapFoldable fa) = foldMap anEndo fa
aDualEndo (WrapFoldable fa) = foldMap aDualEndo fa
instance AnEndo a => AnEndo [a] where
type EndoOperatesOn [a] = EndoOperatesOn a
anEndo = anEndo . WrapFoldable
aDualEndo = aDualEndo . WrapFoldable
instance (Foldable f, AnEndo a) => AnEndo (Reverse f a) where
type EndoOperatesOn (Reverse f a) = EndoOperatesOn a
anEndo = anEndo . WrapFoldable
aDualEndo = aDualEndo . WrapFoldable
instance
( AnEndo a
, AnEndo b
, EndoOperatesOn a ~ EndoOperatesOn b
) => AnEndo (a, b)
where
type EndoOperatesOn (a, b) = EndoOperatesOn a
anEndo (a, b) = anEndo a <> anEndo b
aDualEndo (a, b) = aDualEndo a <> aDualEndo b
instance
( AnEndo a
, AnEndo b
, AnEndo c
, EndoOperatesOn a ~ EndoOperatesOn b
, EndoOperatesOn a ~ EndoOperatesOn c
) => AnEndo (a, b, c)
where
type EndoOperatesOn (a, b, c) = EndoOperatesOn a
anEndo (a, b, c) = anEndo a <> anEndo b <> anEndo c
aDualEndo (a, b, c) = aDualEndo a <> aDualEndo b <> aDualEndo c
instance
( AnEndo a1
, AnEndo a2
, AnEndo a3
, AnEndo a4
, EndoOperatesOn a1 ~ EndoOperatesOn a2
, EndoOperatesOn a1 ~ EndoOperatesOn a3
, EndoOperatesOn a1 ~ EndoOperatesOn a4
) => AnEndo (a1, a2, a3, a4)
where
type EndoOperatesOn (a1, a2, a3, a4) = EndoOperatesOn a1
anEndo (a1, a2, a3, a4) = mconcat
[ anEndo a1
, anEndo a2
, anEndo a3
, anEndo a4
]
aDualEndo (a1, a2, a3, a4) = mconcat
[ aDualEndo a1
, aDualEndo a2
, aDualEndo a3
, aDualEndo a4
]
instance
( AnEndo a1
, AnEndo a2
, AnEndo a3
, AnEndo a4
, AnEndo a5
, EndoOperatesOn a1 ~ EndoOperatesOn a2
, EndoOperatesOn a1 ~ EndoOperatesOn a3
, EndoOperatesOn a1 ~ EndoOperatesOn a4
, EndoOperatesOn a1 ~ EndoOperatesOn a5
) => AnEndo (a1, a2, a3, a4, a5)
where
type EndoOperatesOn (a1, a2, a3, a4, a5) = EndoOperatesOn a1
anEndo (a1, a2, a3, a4, a5) = mconcat
[ anEndo a1
, anEndo a2
, anEndo a3
, anEndo a4
, anEndo a5
]
aDualEndo (a1, a2, a3, a4, a5) = mconcat
[ aDualEndo a1
, aDualEndo a2
, aDualEndo a3
, aDualEndo a4
, aDualEndo a5
]
instance
( AnEndo a1
, AnEndo a2
, AnEndo a3
, AnEndo a4
, AnEndo a5
, AnEndo a6
, EndoOperatesOn a1 ~ EndoOperatesOn a2
, EndoOperatesOn a1 ~ EndoOperatesOn a3
, EndoOperatesOn a1 ~ EndoOperatesOn a4
, EndoOperatesOn a1 ~ EndoOperatesOn a5
, EndoOperatesOn a1 ~ EndoOperatesOn a6
) => AnEndo (a1, a2, a3, a4, a5, a6)
where
type EndoOperatesOn (a1, a2, a3, a4, a5, a6) = EndoOperatesOn a1
anEndo (a1, a2, a3, a4, a5, a6) = mconcat
[ anEndo a1
, anEndo a2
, anEndo a3
, anEndo a4
, anEndo a5
, anEndo a6
]
aDualEndo (a1, a2, a3, a4, a5, a6) = mconcat
[ aDualEndo a1
, aDualEndo a2
, aDualEndo a3
, aDualEndo a4
, aDualEndo a5
, aDualEndo a6
]
instance
( AnEndo a1
, AnEndo a2
, AnEndo a3
, AnEndo a4
, AnEndo a5
, AnEndo a6
, AnEndo a7
, EndoOperatesOn a1 ~ EndoOperatesOn a2
, EndoOperatesOn a1 ~ EndoOperatesOn a3
, EndoOperatesOn a1 ~ EndoOperatesOn a4
, EndoOperatesOn a1 ~ EndoOperatesOn a5
, EndoOperatesOn a1 ~ EndoOperatesOn a6
, EndoOperatesOn a1 ~ EndoOperatesOn a7
) => AnEndo (a1, a2, a3, a4, a5, a6, a7)
where
type EndoOperatesOn (a1, a2, a3, a4, a5, a6, a7) = EndoOperatesOn a1
anEndo (a1, a2, a3, a4, a5, a6, a7) = mconcat
[ anEndo a1
, anEndo a2
, anEndo a3
, anEndo a4
, anEndo a5
, anEndo a6
, anEndo a7
]
aDualEndo (a1, a2, a3, a4, a5, a6, a7) = mconcat
[ aDualEndo a1
, aDualEndo a2
, aDualEndo a3
, aDualEndo a4
, aDualEndo a5
, aDualEndo a6
, aDualEndo a7
]
instance
( AnEndo a1
, AnEndo a2
, AnEndo a3
, AnEndo a4
, AnEndo a5
, AnEndo a6
, AnEndo a7
, AnEndo a8
, EndoOperatesOn a1 ~ EndoOperatesOn a2
, EndoOperatesOn a1 ~ EndoOperatesOn a3
, EndoOperatesOn a1 ~ EndoOperatesOn a4
, EndoOperatesOn a1 ~ EndoOperatesOn a5
, EndoOperatesOn a1 ~ EndoOperatesOn a6
, EndoOperatesOn a1 ~ EndoOperatesOn a7
, EndoOperatesOn a1 ~ EndoOperatesOn a8
) => AnEndo (a1, a2, a3, a4, a5, a6, a7, a8)
where
type EndoOperatesOn (a1, a2, a3, a4, a5, a6, a7, a8) = EndoOperatesOn a1
anEndo (a1, a2, a3, a4, a5, a6, a7, a8) = mconcat
[ anEndo a1
, anEndo a2
, anEndo a3
, anEndo a4
, anEndo a5
, anEndo a6
, anEndo a7
, anEndo a8
]
aDualEndo (a1, a2, a3, a4, a5, a6, a7, a8) = mconcat
[ aDualEndo a1
, aDualEndo a2
, aDualEndo a3
, aDualEndo a4
, aDualEndo a5
, aDualEndo a6
, aDualEndo a7
, aDualEndo a8
]
instance
( AnEndo a1
, AnEndo a2
, AnEndo a3
, AnEndo a4
, AnEndo a5
, AnEndo a6
, AnEndo a7
, AnEndo a8
, AnEndo a9
, EndoOperatesOn a1 ~ EndoOperatesOn a2
, EndoOperatesOn a1 ~ EndoOperatesOn a3
, EndoOperatesOn a1 ~ EndoOperatesOn a4
, EndoOperatesOn a1 ~ EndoOperatesOn a5
, EndoOperatesOn a1 ~ EndoOperatesOn a6
, EndoOperatesOn a1 ~ EndoOperatesOn a7
, EndoOperatesOn a1 ~ EndoOperatesOn a8
, EndoOperatesOn a1 ~ EndoOperatesOn a9
) => AnEndo (a1, a2, a3, a4, a5, a6, a7, a8, a9)
where
type EndoOperatesOn (a1, a2, a3, a4, a5, a6, a7, a8, a9) = EndoOperatesOn a1
anEndo (a1, a2, a3, a4, a5, a6, a7, a8, a9) = mconcat
[ anEndo a1
, anEndo a2
, anEndo a3
, anEndo a4
, anEndo a5
, anEndo a6
, anEndo a7
, anEndo a8
, anEndo a9
]
aDualEndo (a1, a2, a3, a4, a5, a6, a7, a8, a9) = mconcat
[ aDualEndo a1
, aDualEndo a2
, aDualEndo a3
, aDualEndo a4
, aDualEndo a5
, aDualEndo a6
, aDualEndo a7
, aDualEndo a8
, aDualEndo a9
]
instance
( AnEndo a1
, AnEndo a2
, AnEndo a3
, AnEndo a4
, AnEndo a5
, AnEndo a6
, AnEndo a7
, AnEndo a8
, AnEndo a9
, AnEndo a10
, EndoOperatesOn a1 ~ EndoOperatesOn a2
, EndoOperatesOn a1 ~ EndoOperatesOn a3
, EndoOperatesOn a1 ~ EndoOperatesOn a4
, EndoOperatesOn a1 ~ EndoOperatesOn a5
, EndoOperatesOn a1 ~ EndoOperatesOn a6
, EndoOperatesOn a1 ~ EndoOperatesOn a7
, EndoOperatesOn a1 ~ EndoOperatesOn a8
, EndoOperatesOn a1 ~ EndoOperatesOn a9
, EndoOperatesOn a1 ~ EndoOperatesOn a10
) => AnEndo (a1, a2, a3, a4, a5, a6, a7, a8, a9, a10)
where
type EndoOperatesOn (a1, a2, a3, a4, a5, a6, a7, a8, a9, a10) = EndoOperatesOn a1
anEndo (a1, a2, a3, a4, a5, a6, a7, a8, a9, a10) = mconcat
[ anEndo a1
, anEndo a2
, anEndo a3
, anEndo a4
, anEndo a5
, anEndo a6
, anEndo a7
, anEndo a8
, anEndo a9
, anEndo a10
]
aDualEndo (a1, a2, a3, a4, a5, a6, a7, a8, a9, a10) = mconcat
[ aDualEndo a1
, aDualEndo a2
, aDualEndo a3
, aDualEndo a4
, aDualEndo a5
, aDualEndo a6
, aDualEndo a7
, aDualEndo a8
, aDualEndo a9
, aDualEndo a10
]
(&$) :: (a -> b) -> a -> b
f &$ a = f a
infixl 1 &$
(<&$>) :: Functor f => (a -> b) -> f a -> f b
(<&$>) = fmap
infixl 1 <&$>
embedEndoWith :: (AnEndo e, EndoOperatesOn e ~ a)
=> (Endo a -> b)
-> e -> b
embedEndoWith = (. anEndo)
embedDualEndoWith
:: (AnEndo e, EndoOperatesOn e ~ a)
=> (Dual (Endo a) -> b)
-> e -> b
embedDualEndoWith = (. aDualEndo)