{-# LANGUAGE CPP #-} {-# LANGUAGE DeriveDataTypeable #-} {-# LANGUAGE EmptyDataDecls #-} {-# LANGUAGE TypeFamilies #-} #if __GLASGOW_HASKELL__ >= 702 {-# LANGUAGE DeriveGeneric #-} #endif #if __GLASGOW_HASKELL__ >= 706 {-# LANGUAGE PolyKinds #-} #endif #if __GLASGOW_HASKELL__ >= 708 {-# LANGUAGE Safe #-} #elif __GLASGOW_HASKELL__ >= 702 {-# LANGUAGE Trustworthy #-} #endif #include "bifunctors-common.h" ----------------------------------------------------------------------------- -- | -- Copyright : (C) 2008-2016 Edward Kmett -- License : BSD-style (see the file LICENSE) -- -- Maintainer : Edward Kmett -- Stability : provisional -- Portability : portable -- -- From the Functional Pearl \"Clowns to the Left of me, Jokers to the Right: Dissecting Data Structures\" -- by Conor McBride. ---------------------------------------------------------------------------- module Data.Bifunctor.Clown ( Clown(..) ) where #if __GLASGOW_HASKELL__ < 710 import Control.Applicative #endif import Data.Biapplicative import Data.Bifoldable import Data.Bitraversable import Data.Functor.Classes #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 -- | Make a 'Functor' over the first argument of a 'Bifunctor'. -- -- Mnemonic: C__l__owns to the __l__eft (parameter of the Bifunctor), -- joke__r__s to the __r__ight. newtype Clown f a b = Clown { runClown :: f a } deriving ( Eq, Ord, Show, Read #if __GLASGOW_HASKELL__ >= 702 , Generic #endif #if __GLASGOW_HASKELL__ >= 708 , Generic1 , Typeable #endif ) #if __GLASGOW_HASKELL__ >= 702 && __GLASGOW_HASKELL__ < 708 data ClownMetaData data ClownMetaCons data ClownMetaSel instance Datatype ClownMetaData where datatypeName _ = "Clown" moduleName _ = "Data.Bifunctor.Clown" instance Constructor ClownMetaCons where conName _ = "Clown" conIsRecord _ = True instance Selector ClownMetaSel where selName _ = "runClown" instance Generic1 (Clown f a) where type Rep1 (Clown f a) = D1 ClownMetaData (C1 ClownMetaCons (S1 ClownMetaSel (Rec0 (f a)))) from1 = M1 . M1 . M1 . K1 . runClown to1 = Clown . unK1 . unM1 . unM1 . unM1 #endif #if LIFTED_FUNCTOR_CLASSES instance (Eq1 f, Eq a) => Eq1 (Clown f a) where liftEq = liftEq2 (==) instance Eq1 f => Eq2 (Clown f) where liftEq2 f _ = eqClown (liftEq f) instance (Ord1 f, Ord a) => Ord1 (Clown f a) where liftCompare = liftCompare2 compare instance Ord1 f => Ord2 (Clown f) where liftCompare2 f _ = compareClown (liftCompare f) instance (Read1 f, Read a) => Read1 (Clown f a) where liftReadsPrec = liftReadsPrec2 readsPrec readList instance Read1 f => Read2 (Clown f) where liftReadsPrec2 rp1 rl1 _ _ = readsPrecClown (liftReadsPrec rp1 rl1) instance (Show1 f, Show a) => Show1 (Clown f a) where liftShowsPrec = liftShowsPrec2 showsPrec showList instance Show1 f => Show2 (Clown f) where liftShowsPrec2 sp1 sl1 _ _ = showsPrecClown (liftShowsPrec sp1 sl1) #else instance (Eq1 f, Eq a) => Eq1 (Clown f a) where eq1 = eqClown eq1 instance (Ord1 f, Ord a) => Ord1 (Clown f a) where compare1 = compareClown compare1 instance (Read1 f, Read a) => Read1 (Clown f a) where readsPrec1 = readsPrecClown readsPrec1 instance (Show1 f, Show a) => Show1 (Clown f a) where showsPrec1 = showsPrecClown showsPrec1 #endif eqClown :: (f a1 -> f a2 -> Bool) -> Clown f a1 b1 -> Clown f a2 b2 -> Bool eqClown eqA (Clown x) (Clown y) = eqA x y compareClown :: (f a1 -> f a2 -> Ordering) -> Clown f a1 b1 -> Clown f a2 b2 -> Ordering compareClown compareA (Clown x) (Clown y) = compareA x y readsPrecClown :: (Int -> ReadS (f a)) -> Int -> ReadS (Clown f a b) readsPrecClown rpA p = readParen (p > 10) $ \s0 -> do ("Clown", s1) <- lex s0 ("{", s2) <- lex s1 ("runClown", s3) <- lex s2 (x, s4) <- rpA 0 s3 ("}", s5) <- lex s4 return (Clown x, s5) showsPrecClown :: (Int -> f a -> ShowS) -> Int -> Clown f a b -> ShowS showsPrecClown spA p (Clown x) = showParen (p > 10) $ showString "Clown {runClown = " . spA 0 x . showChar '}' instance Functor f => Bifunctor (Clown f) where first f = Clown . fmap f . runClown {-# INLINE first #-} second _ = Clown . runClown {-# INLINE second #-} bimap f _ = Clown . fmap f . runClown {-# INLINE bimap #-} instance Functor (Clown f a) where fmap _ = Clown . runClown {-# INLINE fmap #-} instance Applicative f => Biapplicative (Clown f) where bipure a _ = Clown (pure a) {-# INLINE bipure #-} Clown mf <<*>> Clown mx = Clown (mf <*> mx) {-# INLINE (<<*>>) #-} instance Foldable f => Bifoldable (Clown f) where bifoldMap f _ = foldMap f . runClown {-# INLINE bifoldMap #-} instance Foldable (Clown f a) where foldMap _ = mempty {-# INLINE foldMap #-} instance Traversable f => Bitraversable (Clown f) where bitraverse f _ = fmap Clown . traverse f . runClown {-# INLINE bitraverse #-} instance Traversable (Clown f a) where traverse _ = pure . Clown . runClown {-# INLINE traverse #-}