------------------------------------------------------------------------
-- The Agda standard library
--
-- A Categorical view of the Sum type (Right-biased)
------------------------------------------------------------------------

open import Level

module Data.Sum.Categorical.Right (a : Level) {b} (B : Set b) where

open import Data.Sum
open import Category.Functor
open import Category.Applicative
open import Category.Monad
open import Function
import Function.Identity.Categorical as Id

Sumᵣ : Set (a  b)  Set (a  b)
Sumᵣ A = A  B

functor : RawFunctor Sumᵣ
functor = record { _<$>_ = map₁ }

applicative : RawApplicative Sumᵣ
applicative = record
  { pure = inj₁
  ; _⊛_ = [ map₁ , const  inj₂ ]′
  }

monadT : RawMonadT (_∘′ Sumᵣ)
monadT M = record
  { return = M.pure ∘′ inj₁
  ; _>>=_  = λ ma f  ma M.>>= [ f , M.pure ∘′ inj₂ ]′
  } where module M = RawMonad M

monad : RawMonad Sumᵣ
monad = monadT Id.monad

------------------------------------------------------------------------
-- Get access to other monadic functions

module _ {F} (App : RawApplicative {a  b} F) where

  open RawApplicative App

  sequenceA :  {A}  Sumᵣ (F A)  F (Sumᵣ A)
  sequenceA (inj₂ a) = pure (inj₂ a)
  sequenceA (inj₁ x) = inj₁ <$> x

  mapA :  {A B}  (A  F B)  Sumᵣ A  F (Sumᵣ B)
  mapA f = sequenceA  map₁ f

  forA :  {A B}  Sumᵣ A  (A  F B)  F (Sumᵣ B)
  forA = flip mapA

module _ {M} (Mon : RawMonad {a  b} M) where

  private App = RawMonad.rawIApplicative Mon

  sequenceM :  {A}  Sumᵣ (M A)  M (Sumᵣ A)
  sequenceM = sequenceA App

  mapM :  {A B}  (A  M B)  Sumᵣ A  M (Sumᵣ B)
  mapM = mapA App

  forM :  {A B}  Sumᵣ A  (A  M B)  M (Sumᵣ B)
  forM = forA App