module Data.Bifunctor.Joker
( Joker(..)
) where
import Control.Applicative
import Data.Biapplicative
import Data.Bifunctor.Apply
import Data.Bifoldable
import Data.Bitraversable
import Data.Foldable
import Data.Functor.Apply
import Data.Semigroup.Bifoldable
import Data.Semigroup.Bitraversable
import Data.Semigroup.Foldable
import Data.Semigroup.Traversable
import Data.Traversable
newtype Joker g a b = Joker { runJoker :: g b }
deriving (Eq,Ord,Show,Read)
instance Functor g => Bifunctor (Joker g) where
first _ = Joker . runJoker
second g = Joker . fmap g . runJoker
bimap _ g = Joker . fmap g . runJoker
instance Functor g => Functor (Joker g a) where
fmap g = Joker . fmap g . runJoker
instance Applicative g => Biapplicative (Joker g) where
bipure _ b = Joker (pure b)
Joker mf <<*>> Joker mx = Joker (mf <*> mx)
instance Apply g => Biapply (Joker g) where
Joker fg <<.>> Joker xy = Joker (fg <.> xy)
instance Foldable g => Bifoldable (Joker g) where
bifoldMap _ g = foldMap g . runJoker
instance Foldable g => Foldable (Joker g a) where
foldMap g = foldMap g . runJoker
instance Traversable g => Bitraversable (Joker g) where
bitraverse _ g = fmap Joker . traverse g . runJoker
instance Traversable g => Traversable (Joker g a) where
traverse g = fmap Joker . traverse g . runJoker
instance Foldable1 g => Bifoldable1 (Joker g) where
bifoldMap1 _ g = foldMap1 g . runJoker
instance Foldable1 g => Foldable1 (Joker g a) where
foldMap1 g = foldMap1 g . runJoker
instance Traversable1 g => Bitraversable1 (Joker g) where
bitraverse1 _ g = fmap Joker . traverse1 g . runJoker
instance Traversable1 g => Traversable1 (Joker g a) where
traverse1 g = fmap Joker . traverse1 g . runJoker