{-# LANGUAGE ScopedTypeVariables
           , TypeApplications
           , DataKinds
           , ConstraintKinds
           , AllowAmbiguousTypes
           , MultiParamTypeClasses
           , FlexibleInstances
           , FlexibleContexts
           , RankNTypes
           , TypeFamilies
           #-}
module Data.Generics.ClassyPlate.Common where
import Control.Monad
import Data.Generics.ClassyPlate.Core
import Data.Generics.ClassyPlate.TypePrune
class MonoMatch a b where
  
  monoApp :: (a -> a) -> b -> b
instance MonoMatch a a where
  monoApp = id
type instance AppSelector (MonoMatch a) b = TypEq a b
type family TypEq a b :: Bool where
  TypEq a a = 'True
  TypEq a b = 'False
descend :: forall c b . ClassyPlate c b => (forall a . (ClassyPlate c a, c a) => a -> a) -> b -> b
descend = descend_ (undefined :: ClsToken c)
descendM :: forall c b m . (ClassyPlate c b, Monad m) => (forall a . (ClassyPlate c a, c a) => a -> m a) -> b -> m b
descendM = descendM_ (undefined :: ClsToken c)
topDown :: forall c b . ClassyPlate c b => (forall a . (ClassyPlate c a, c a) => a -> a) -> b -> b
topDown = topDown_ (undefined :: ClsToken c)
topDownM :: forall c b m . (ClassyPlate c b, Monad m) => (forall a . (ClassyPlate c a, c a) => a -> m a) -> b -> m b
topDownM = topDownM_ (undefined :: ClsToken c)
bottomUp :: forall c b . ClassyPlate c b => (forall a . (ClassyPlate c a, c a) => a -> a) -> b -> b
bottomUp = bottomUp_ (undefined :: ClsToken c)
bottomUpM :: forall c b m . (ClassyPlate c b, Monad m)
                => (forall a . (ClassyPlate c a, c a) => a -> m a) -> b -> m b
bottomUpM = bottomUpM_ (undefined :: ClsToken c)
selectiveTraverse :: forall c b . ClassyPlate c b => (forall a . (ClassyPlate c a, c a) => a -> (a, Bool)) -> b -> b
selectiveTraverse trf = descend @c ((\(e, go) -> if go then selectiveTraverse @c trf e else e) . trf)
selectiveTraverseM :: forall c b m . (Monad m, ClassyPlate c b) => (forall a . (ClassyPlate c a, c a) => a -> m (a, Bool)) -> b -> m b
selectiveTraverseM trf = descendM @c ((\(e, go) -> if go then selectiveTraverseM @c trf e else return e) <=< trf)
smartTraverse :: forall c b . SmartClassyPlate c (ClassIgnoresSubtree c b) b
              => (forall a . (ClassyPlate c a, c a) => a -> a) -> b -> b
smartTraverse = smartTraverse_ (undefined :: FlagToken (ClassIgnoresSubtree c b)) (undefined :: ClsToken c)
smartTraverseM :: forall c b m . (SmartClassyPlate c (ClassIgnoresSubtree c b) b, Monad m)
               => (forall a . (ClassyPlate c a, c a) => a -> m a) -> b -> m b
smartTraverseM = smartTraverseM_ (undefined :: FlagToken (ClassIgnoresSubtree c b)) (undefined :: ClsToken c)