{-# LANGUAGE CPP #-}
{-# LANGUAGE GADTs #-}
#if __GLASGOW_HASKELL__ >= 706
{-# LANGUAGE PolyKinds #-}
#endif
{-# LANGUAGE MultiParamTypeClasses #-}
#if __GLASGOW_HASKELL__ >= 702 && __GLASGOW_HASKELL__ < 710
{-# LANGUAGE Trustworthy #-}
#endif
-----------------------------------------------------------------------------
-- |
-- Module      :  Control.Comonad.Density
-- Copyright   :  (C) 2008-2016 Edward Kmett
-- License     :  BSD-style (see the file LICENSE)
--
-- Maintainer  :  Edward Kmett <[email protected]>
-- Stability   :  experimental
-- Portability :  non-portable (GADTs, MPTCs)
--
-- The 'Density' 'Comonad' for a 'Functor' (aka the 'Comonad generated by a 'Functor')
-- The 'Density' term dates back to Dubuc''s 1974 thesis. The term
-- 'Monad' generated by a 'Functor' dates back to 1972 in Street''s
-- ''Formal Theory of Monads''.
--
-- The left Kan extension of a 'Functor' along itself (@'Lan' f f@) forms a 'Comonad'. This is
-- that 'Comonad'.
----------------------------------------------------------------------------
module Control.Comonad.Density
  ( Density(..)
  , liftDensity
  , densityToAdjunction, adjunctionToDensity
  , densityToLan, lanToDensity
  ) where

import Control.Applicative
import Control.Comonad
import Control.Comonad.Trans.Class
import Data.Functor.Apply
import Data.Functor.Adjunction
import Data.Functor.Extend
import Data.Functor.Kan.Lan

data Density k a where
  Density :: (k b -> a) -> k b -> Density k a

instance Functor (Density f) where
  fmap :: forall a b. (a -> b) -> Density f a -> Density f b
fmap a -> b
f (Density f b -> a
g f b
h) = forall {k} (k :: k -> *) (b :: k) a.
(k b -> a) -> k b -> Density k a
Density (a -> b
f forall b c a. (b -> c) -> (a -> b) -> a -> c
. f b -> a
g) f b
h
  {-# INLINE fmap #-}

instance Extend (Density f) where
  duplicated :: forall a. Density f a -> Density f (Density f a)
duplicated (Density f b -> a
f f b
ws) = forall {k} (k :: k -> *) (b :: k) a.
(k b -> a) -> k b -> Density k a
Density (forall {k} (k :: k -> *) (b :: k) a.
(k b -> a) -> k b -> Density k a
Density f b -> a
f) f b
ws
  {-# INLINE duplicated #-}

instance Comonad (Density f) where
  duplicate :: forall a. Density f a -> Density f (Density f a)
duplicate (Density f b -> a
f f b
ws) = forall {k} (k :: k -> *) (b :: k) a.
(k b -> a) -> k b -> Density k a
Density (forall {k} (k :: k -> *) (b :: k) a.
(k b -> a) -> k b -> Density k a
Density f b -> a
f) f b
ws
  {-# INLINE duplicate #-}
  extract :: forall a. Density f a -> a
extract (Density f b -> a
f f b
a) = f b -> a
f f b
a
  {-# INLINE extract #-}

instance ComonadTrans Density where
  lower :: forall (w :: * -> *) a. Comonad w => Density w a -> w a
lower (Density w b -> a
f w b
c) = forall (w :: * -> *) a b. Comonad w => (w a -> b) -> w a -> w b
extend w b -> a
f w b
c
  {-# INLINE lower #-}

instance Apply f => Apply (Density f) where
  Density f b -> a -> b
kxf f b
x <.> :: forall a b. Density f (a -> b) -> Density f a -> Density f b
<.> Density f b -> a
kya f b
y =
    forall {k} (k :: k -> *) (b :: k) a.
(k b -> a) -> k b -> Density k a
Density (\f (b, b)
k -> f b -> a -> b
kxf (forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap forall a b. (a, b) -> a
fst f (b, b)
k) (f b -> a
kya (forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap forall a b. (a, b) -> b
snd f (b, b)
k))) ((,) forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> f b
x forall (f :: * -> *) a b. Apply f => f (a -> b) -> f a -> f b
<.> f b
y)
  {-# INLINE (<.>) #-}

instance Applicative f => Applicative (Density f) where
  pure :: forall a. a -> Density f a
pure a
a = forall {k} (k :: k -> *) (b :: k) a.
(k b -> a) -> k b -> Density k a
Density (forall a b. a -> b -> a
const a
a) (forall (f :: * -> *) a. Applicative f => a -> f a
pure ())
  {-# INLINE pure #-}
  Density f b -> a -> b
kxf f b
x <*> :: forall a b. Density f (a -> b) -> Density f a -> Density f b
<*> Density f b -> a
kya f b
y =
    forall {k} (k :: k -> *) (b :: k) a.
(k b -> a) -> k b -> Density k a
Density (\f (b, b)
k -> f b -> a -> b
kxf (forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap forall a b. (a, b) -> a
fst f (b, b)
k) (f b -> a
kya (forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap forall a b. (a, b) -> b
snd f (b, b)
k))) (forall (f :: * -> *) a b c.
Applicative f =>
(a -> b -> c) -> f a -> f b -> f c
liftA2 (,) f b
x f b
y)
  {-# INLINE (<*>) #-}

-- | The natural transformation from a @'Comonad' w@ to the 'Comonad' generated by @w@ (forwards).
--
-- This is merely a right-inverse (section) of 'lower', rather than a full inverse.
--
-- @
-- 'lower' . 'liftDensity' ≡ 'id'
-- @
liftDensity :: Comonad w => w a -> Density w a
liftDensity :: forall (w :: * -> *) a. Comonad w => w a -> Density w a
liftDensity = forall {k} (k :: k -> *) (b :: k) a.
(k b -> a) -> k b -> Density k a
Density forall (w :: * -> *) a. Comonad w => w a -> a
extract
{-# INLINE liftDensity #-}

-- | The Density 'Comonad' of a left adjoint is isomorphic to the 'Comonad' formed by that 'Adjunction'.
--
-- This isomorphism is witnessed by 'densityToAdjunction' and 'adjunctionToDensity'.
--
-- @
-- 'densityToAdjunction' . 'adjunctionToDensity' ≡ 'id'
-- 'adjunctionToDensity' . 'densityToAdjunction' ≡ 'id'
-- @
densityToAdjunction :: Adjunction f g => Density f a -> f (g a)
densityToAdjunction :: forall (f :: * -> *) (g :: * -> *) a.
Adjunction f g =>
Density f a -> f (g a)
densityToAdjunction (Density f b -> a
f f b
v) = forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (forall (f :: * -> *) (u :: * -> *) a b.
Adjunction f u =>
(f a -> b) -> a -> u b
leftAdjunct f b -> a
f) f b
v
{-# INLINE densityToAdjunction #-}

adjunctionToDensity :: Adjunction f g => f (g a) -> Density f a
adjunctionToDensity :: forall (f :: * -> *) (g :: * -> *) a.
Adjunction f g =>
f (g a) -> Density f a
adjunctionToDensity = forall {k} (k :: k -> *) (b :: k) a.
(k b -> a) -> k b -> Density k a
Density forall (f :: * -> *) (u :: * -> *) a.
Adjunction f u =>
f (u a) -> a
counit
{-# INLINE adjunctionToDensity #-}

-- | The 'Density' 'Comonad' of a 'Functor' @f@ is obtained by taking the left Kan extension
-- ('Lan') of @f@ along itself. This isomorphism is witnessed by 'lanToDensity' and 'densityToLan'
--
-- @
-- 'lanToDensity' . 'densityToLan' ≡ 'id'
-- 'densityToLan' . 'lanToDensity' ≡ 'id'
-- @
lanToDensity :: Lan f f a -> Density f a
lanToDensity :: forall {k} (f :: k -> *) a. Lan f f a -> Density f a
lanToDensity (Lan f b -> a
f f b
v) = forall {k} (k :: k -> *) (b :: k) a.
(k b -> a) -> k b -> Density k a
Density f b -> a
f f b
v
{-# INLINE lanToDensity #-}

densityToLan :: Density f a -> Lan f f a
densityToLan :: forall {k} (f :: k -> *) a. Density f a -> Lan f f a
densityToLan (Density f b -> a
f f b
v) = forall {k} (g :: k -> *) (b :: k) a (h :: k -> *).
(g b -> a) -> h b -> Lan g h a
Lan f b -> a
f f b
v
{-# INLINE densityToLan #-}