{-# LANGUAGE CPP #-} {-# LANGUAGE DeriveDataTypeable #-} {-# LANGUAGE EmptyDataDecls #-} {-# LANGUAGE FlexibleContexts #-} {-# LANGUAGE StandaloneDeriving #-} {-# LANGUAGE TypeFamilies #-} {-# LANGUAGE TypeOperators #-} #if __GLASGOW_HASKELL__ >= 702 {-# LANGUAGE DeriveGeneric #-} #endif #if __GLASGOW_HASKELL__ >= 706 {-# LANGUAGE PolyKinds #-} #endif ----------------------------------------------------------------------------- -- | -- Copyright : (C) 2008-2016 Edward Kmett -- License : BSD-style (see the file LICENSE) -- -- Maintainer : Edward Kmett -- Stability : provisional -- Portability : portable -- ---------------------------------------------------------------------------- module Data.Bifunctor.Biff ( Biff(..) ) where #if __GLASGOW_HASKELL__ < 710 import Control.Applicative #endif import Data.Biapplicative import Data.Bifoldable import Data.Bitraversable #if __GLASGOW_HASKELL__ < 710 import Data.Foldable import Data.Monoid import Data.Traversable #endif #if __GLASGOW_HASKELL__ >= 708 import Data.Typeable #endif #if __GLASGOW_HASKELL__ >= 702 import GHC.Generics #endif -- | Compose two 'Functor's on the inside of a 'Bifunctor'. newtype Biff p f g a b = Biff { runBiff :: p (f a) (g b) } deriving ( Eq, Ord, Show, Read #if __GLASGOW_HASKELL__ >= 702 , Generic #endif #if __GLASGOW_HASKELL__ >= 708 , Typeable #endif ) #if __GLASGOW_HASKELL__ >= 702 # if __GLASGOW_HASKELL__ >= 708 deriving instance Functor (p (f a)) => Generic1 (Biff p f g a) # else data BiffMetaData data BiffMetaCons data BiffMetaSel instance Datatype BiffMetaData where datatypeName = const "Biff" moduleName = const "Data.Bifunctor.Biff" instance Constructor BiffMetaCons where conName = const "Biff" conIsRecord = const True instance Selector BiffMetaSel where selName = const "runBiff" instance Functor (p (f a)) => Generic1 (Biff p f g a) where type Rep1 (Biff p f g a) = D1 BiffMetaData (C1 BiffMetaCons (S1 BiffMetaSel (p (f a) :.: Rec1 g))) from1 = M1 . M1 . M1 . Comp1 . fmap Rec1 . runBiff to1 = Biff . fmap unRec1 . unComp1 . unM1 . unM1 . unM1 # endif #endif instance (Bifunctor p, Functor f, Functor g) => Bifunctor (Biff p f g) where first f = Biff . first (fmap f) . runBiff {-# INLINE first #-} second f = Biff . second (fmap f) . runBiff {-# INLINE second #-} bimap f g = Biff . bimap (fmap f) (fmap g) . runBiff {-# INLINE bimap #-} instance (Bifunctor p, Functor g) => Functor (Biff p f g a) where fmap f = Biff . second (fmap f) . runBiff {-# INLINE fmap #-} instance (Biapplicative p, Applicative f, Applicative g) => Biapplicative (Biff p f g) where bipure a b = Biff (bipure (pure a) (pure b)) {-# INLINE bipure #-} Biff fg <<*>> Biff xy = Biff (bimap (<*>) (<*>) fg <<*>> xy) {-# INLINE (<<*>>) #-} instance (Bifoldable p, Foldable g) => Foldable (Biff p f g a) where foldMap f = bifoldMap (const mempty) (foldMap f) . runBiff {-# INLINE foldMap #-} instance (Bifoldable p, Foldable f, Foldable g) => Bifoldable (Biff p f g) where bifoldMap f g = bifoldMap (foldMap f) (foldMap g) . runBiff {-# INLINE bifoldMap #-} instance (Bitraversable p, Traversable g) => Traversable (Biff p f g a) where traverse f = fmap Biff . bitraverse pure (traverse f) . runBiff {-# INLINE traverse #-} instance (Bitraversable p, Traversable f, Traversable g) => Bitraversable (Biff p f g) where bitraverse f g = fmap Biff . bitraverse (traverse f) (traverse g) . runBiff {-# INLINE bitraverse #-}