{-# Language FlexibleContexts, FlexibleInstances, MultiParamTypeClasses, RankNTypes, TypeFamilies, TypeOperators #-}

-- | Type classes 'Functor', 'Foldable', and 'Traversable' that correspond to the standard type classes of the same
-- name, but applying the given transformation to the given tree node and all its descendants. The corresponding classes
-- in the "Transformation.Shallow" moduleo perate only on the immediate children, while those from the
-- "Transformation.Deep" module exclude the argument node itself.

module Transformation.Full where

import qualified Data.Functor
import           Data.Functor.Compose (Compose(getCompose))
import           Data.Functor.Const (Const(getConst))
import qualified Data.Foldable
import qualified Data.Traversable
import qualified Rank2
import qualified Transformation
import           Transformation (Transformation, Domain, Codomain)
import {-# SOURCE #-} qualified Transformation.Deep as Deep

import Prelude hiding (Foldable(..), Traversable(..), Functor(..), Applicative(..), (<$>), fst, snd)

-- | Like "Transformation.Deep".'Deep.Functor' except it maps an additional wrapper around the entire tree
class (Transformation t, Rank2.Functor (g (Domain t))) => Functor t g where
   (<$>) :: t -> Domain t (g (Domain t) (Domain t)) -> Codomain t (g (Codomain t) (Codomain t))
   infixl 4 <$>

-- | Like "Transformation.Deep".'Deep.Foldable' except the entire tree is also wrapped
class (Transformation t, Rank2.Foldable (g (Domain t))) => Foldable t g where
   foldMap :: (Codomain t ~ Const m, Monoid m) => t -> Domain t (g (Domain t) (Domain t)) -> m

-- | Like "Transformation.Deep".'Deep.Traversable' except it traverses an additional wrapper around the entire tree
class (Transformation t, Rank2.Traversable (g (Domain t))) => Traversable t g where
   traverse :: Codomain t ~ Compose m f => t -> Domain t (g (Domain t) (Domain t)) -> m (f (g f f))

-- | Alphabetical synonym for '<$>'
fmap :: Functor t g => t -> Domain t (g (Domain t) (Domain t)) -> Codomain t (g (Codomain t) (Codomain t))
fmap :: t
-> Domain t (g (Domain t) (Domain t))
-> Codomain t (g (Codomain t) (Codomain t))
fmap = t
-> Domain t (g (Domain t) (Domain t))
-> Codomain t (g (Codomain t) (Codomain t))
forall t (g :: (* -> *) -> (* -> *) -> *).
Functor t g =>
t
-> Domain t (g (Domain t) (Domain t))
-> Codomain t (g (Codomain t) (Codomain t))
(<$>)

-- | Default implementation for '<$>' that maps the wrapper and then the tree
mapDownDefault :: (Deep.Functor t g, t `Transformation.At` g (Domain t) (Domain t), Data.Functor.Functor (Codomain t))
               => t -> Domain t (g (Domain t) (Domain t)) -> Codomain t (g (Codomain t) (Codomain t))
mapDownDefault :: t
-> Domain t (g (Domain t) (Domain t))
-> Codomain t (g (Codomain t) (Codomain t))
mapDownDefault t
t Domain t (g (Domain t) (Domain t))
x = (t
t t -> g (Domain t) (Domain t) -> g (Codomain t) (Codomain t)
forall t (g :: (* -> *) -> (* -> *) -> *).
Functor t g =>
t -> g (Domain t) (Domain t) -> g (Codomain t) (Codomain t)
Deep.<$>) (g (Domain t) (Domain t) -> g (Codomain t) (Codomain t))
-> Codomain t (g (Domain t) (Domain t))
-> Codomain t (g (Codomain t) (Codomain t))
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
Data.Functor.<$> (t
t t
-> Domain t (g (Domain t) (Domain t))
-> Codomain t (g (Domain t) (Domain t))
forall t x. At t x => t -> Domain t x -> Codomain t x
Transformation.$ Domain t (g (Domain t) (Domain t))
x)

-- | Default implementation for '<$>' that maps the tree and then the wrapper
mapUpDefault   :: (Deep.Functor t g, t `Transformation.At` g (Codomain t) (Codomain t), Data.Functor.Functor (Domain t))
               => t -> Domain t (g (Domain t) (Domain t)) -> Codomain t (g (Codomain t) (Codomain t))
mapUpDefault :: t
-> Domain t (g (Domain t) (Domain t))
-> Codomain t (g (Codomain t) (Codomain t))
mapUpDefault   t
t Domain t (g (Domain t) (Domain t))
x = t
t t
-> Domain t (g (Codomain t) (Codomain t))
-> Codomain t (g (Codomain t) (Codomain t))
forall t x. At t x => t -> Domain t x -> Codomain t x
Transformation.$ ((t
t t -> g (Domain t) (Domain t) -> g (Codomain t) (Codomain t)
forall t (g :: (* -> *) -> (* -> *) -> *).
Functor t g =>
t -> g (Domain t) (Domain t) -> g (Codomain t) (Codomain t)
Deep.<$>) (g (Domain t) (Domain t) -> g (Codomain t) (Codomain t))
-> Domain t (g (Domain t) (Domain t))
-> Domain t (g (Codomain t) (Codomain t))
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
Data.Functor.<$> Domain t (g (Domain t) (Domain t))
x)

foldMapDownDefault, foldMapUpDefault :: (t `Transformation.At` g (Domain t) (Domain t), Deep.Foldable t g,
                                         Codomain t ~ Const m, Data.Foldable.Foldable (Domain t), Monoid m)
                                     => t -> Domain t (g (Domain t) (Domain t)) -> m
-- | Default implementation for 'foldMap' that folds the wrapper and then the tree
foldMapDownDefault :: t -> Domain t (g (Domain t) (Domain t)) -> m
foldMapDownDefault t
t Domain t (g (Domain t) (Domain t))
x = Const m (g (Domain t) (Domain t)) -> m
forall a k (b :: k). Const a b -> a
getConst (t
t t
-> Domain t (g (Domain t) (Domain t))
-> Codomain t (g (Domain t) (Domain t))
forall t x. At t x => t -> Domain t x -> Codomain t x
Transformation.$ Domain t (g (Domain t) (Domain t))
x) m -> m -> m
forall a. Monoid a => a -> a -> a
`mappend` (g (Domain t) (Domain t) -> m)
-> Domain t (g (Domain t) (Domain t)) -> m
forall (t :: * -> *) m a.
(Foldable t, Monoid m) =>
(a -> m) -> t a -> m
Data.Foldable.foldMap (t -> g (Domain t) (Domain t) -> m
forall t (g :: (* -> *) -> (* -> *) -> *) m.
(Foldable t g, Codomain t ~ Const m, Monoid m) =>
t -> g (Domain t) (Domain t) -> m
Deep.foldMap t
t) Domain t (g (Domain t) (Domain t))
x
-- | Default implementation for 'foldMap' that folds the tree and then the wrapper
foldMapUpDefault :: t -> Domain t (g (Domain t) (Domain t)) -> m
foldMapUpDefault   t
t Domain t (g (Domain t) (Domain t))
x = (g (Domain t) (Domain t) -> m)
-> Domain t (g (Domain t) (Domain t)) -> m
forall (t :: * -> *) m a.
(Foldable t, Monoid m) =>
(a -> m) -> t a -> m
Data.Foldable.foldMap (t -> g (Domain t) (Domain t) -> m
forall t (g :: (* -> *) -> (* -> *) -> *) m.
(Foldable t g, Codomain t ~ Const m, Monoid m) =>
t -> g (Domain t) (Domain t) -> m
Deep.foldMap t
t) Domain t (g (Domain t) (Domain t))
x m -> m -> m
forall a. Monoid a => a -> a -> a
`mappend` Const m (g (Domain t) (Domain t)) -> m
forall a k (b :: k). Const a b -> a
getConst (t
t t
-> Domain t (g (Domain t) (Domain t))
-> Codomain t (g (Domain t) (Domain t))
forall t x. At t x => t -> Domain t x -> Codomain t x
Transformation.$ Domain t (g (Domain t) (Domain t))
x)

-- | Default implementation for 'traverse' that traverses the wrapper and then the tree
traverseDownDefault :: (Deep.Traversable t g, t `Transformation.At` g (Domain t) (Domain t),
                        Codomain t ~ Compose m f, Data.Traversable.Traversable f, Monad m)
                    => t -> Domain t (g (Domain t) (Domain t)) -> m (f (g f f))
traverseDownDefault :: t -> Domain t (g (Domain t) (Domain t)) -> m (f (g f f))
traverseDownDefault t
t Domain t (g (Domain t) (Domain t))
x = Compose m f (g (Domain t) (Domain t))
-> m (f (g (Domain t) (Domain t)))
forall k1 (f :: k1 -> *) k2 (g :: k2 -> k1) (a :: k2).
Compose f g a -> f (g a)
getCompose (t
t t
-> Domain t (g (Domain t) (Domain t))
-> Codomain t (g (Domain t) (Domain t))
forall t x. At t x => t -> Domain t x -> Codomain t x
Transformation.$ Domain t (g (Domain t) (Domain t))
x) m (f (g (Domain t) (Domain t)))
-> (f (g (Domain t) (Domain t)) -> m (f (g f f))) -> m (f (g f f))
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= (g (Domain t) (Domain t) -> m (g f f))
-> f (g (Domain t) (Domain t)) -> m (f (g f f))
forall (t :: * -> *) (f :: * -> *) a b.
(Traversable t, Applicative f) =>
(a -> f b) -> t a -> f (t b)
Data.Traversable.traverse (t -> g (Domain t) (Domain t) -> m (g f f)
forall t (g :: (* -> *) -> (* -> *) -> *) (m :: * -> *)
       (f :: * -> *).
(Traversable t g, Codomain t ~ Compose m f) =>
t -> g (Domain t) (Domain t) -> m (g f f)
Deep.traverse t
t)

-- | Default implementation for 'traverse' that traverses the tree and then the wrapper
traverseUpDefault   :: (Deep.Traversable t g, Codomain t ~ Compose m f, t `Transformation.At` g f f,
                        Data.Traversable.Traversable (Domain t), Monad m)
                    => t -> Domain t (g (Domain t) (Domain t)) -> m (f (g f f))
traverseUpDefault :: t -> Domain t (g (Domain t) (Domain t)) -> m (f (g f f))
traverseUpDefault   t
t Domain t (g (Domain t) (Domain t))
x = (g (Domain t) (Domain t) -> m (g f f))
-> Domain t (g (Domain t) (Domain t)) -> m (Domain t (g f f))
forall (t :: * -> *) (f :: * -> *) a b.
(Traversable t, Applicative f) =>
(a -> f b) -> t a -> f (t b)
Data.Traversable.traverse (t -> g (Domain t) (Domain t) -> m (g f f)
forall t (g :: (* -> *) -> (* -> *) -> *) (m :: * -> *)
       (f :: * -> *).
(Traversable t g, Codomain t ~ Compose m f) =>
t -> g (Domain t) (Domain t) -> m (g f f)
Deep.traverse t
t) Domain t (g (Domain t) (Domain t))
x m (Domain t (g f f))
-> (Domain t (g f f) -> m (f (g f f))) -> m (f (g f f))
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= Compose m f (g f f) -> m (f (g f f))
forall k1 (f :: k1 -> *) k2 (g :: k2 -> k1) (a :: k2).
Compose f g a -> f (g a)
getCompose (Compose m f (g f f) -> m (f (g f f)))
-> (Domain t (g f f) -> Compose m f (g f f))
-> Domain t (g f f)
-> m (f (g f f))
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (t
t t -> Domain t (g f f) -> Codomain t (g f f)
forall t x. At t x => t -> Domain t x -> Codomain t x
Transformation.$)