open import Cat.Diagram.Coproduct.Copower
open import Cat.Diagram.Coproduct.Indexed
open import Cat.Functor.Morphism
open import Cat.Diagram.Zero
open import Cat.Functor.Hom
open import Cat.Prelude

open import Data.Set.Projective
open import Data.Set.Surjection
open import Data.Dec

import Cat.Diagram.Separator.Strong
import Cat.Morphism.Strong.Epi
import Cat.Diagram.Projective
import Cat.Reasoning
module Cat.Diagram.Projective.Strong
  {o ℓ}
  (C : Precategory o ℓ)
  where
open Cat.Diagram.Projective C
open Cat.Morphism.Strong.Epi C
open Cat.Reasoning C

Strong projective objects🔗

Let be a precategory. An object is a strong projective object if it has the left-lifting property against strong epimorphisms.

More explicitly, is a strong projective object if for every morphism and strong epi there merely exists a such that as in the following diagram:

is-strong-projective : (P : Ob)  Type _
is-strong-projective P =
   {X Y} (p : Hom P Y) (e : Hom X Y)
   is-strong-epi e
   ∃[ s ∈ Hom P X ] (e ∘ s ≡ p)
\ Warning

Being a strong projective object is actually a weaker condition than being a projective object: strong projectives only need to lift against strong epis, whereas projectives need to lift against all epis.

projective→strong-projective
  :  {P}
   is-projective P
   is-strong-projective P
projective→strong-projective pro p e e-strong =
  pro p (record { mor = e ; epic = e-strong .fst })

A functorial definition🔗

Like their non-strong counterparts, we can give a functorial definition of strong projectives. In particular, an object is a strong projective if and only if the preserves strong epimorphisms.

preserves-strong-epis→strong-projective
  :  {P}
   preserves-strong-epis (Hom-from C P)
   is-strong-projective P

strong-projective→preserves-strong-epis
  :  {P}
   is-strong-projective P
   preserves-strong-epis (Hom-from C P)
These proofs are essentially the same as the corresponding ones for projective objects, so we omit the details.
preserves-strong-epis→strong-projective {P = P} hom-epi {X = X} {Y = Y} p e e-strong =
  epi→surjective (el! (Hom P X)) (el! (Hom P Y))
    (e ∘_)
     {c}  hom-epi e-strong .fst {c = c})
    p

strong-projective→preserves-strong-epis {P = P} pro {X} {Y} {f = f} f-strong =
    surjective→strong-epi (el! (Hom P X)) (el! (Hom P Y)) (f ∘_) $ λ p 
    pro p f f-strong

Closure of strong projectives🔗

Like projective objects, strong projectives are closed under coproducts indexed by set-projective types and retracts.

indexed-coproduct-strong-projective
  :  {κ} {Idx : Type κ}
   {P : Idx  Ob} {∐P : Ob} {ι :  i  Hom (P i) ∐P}
   is-set-projective Idx ℓ
   (∀ i  is-strong-projective (P i))
   is-indexed-coproduct C P ι
   is-strong-projective ∐P

retract→strong-projective
  :  {R P}
   is-strong-projective P
   (s : Hom R P)
   has-retract s
   is-strong-projective R
These proofs are more or less identical to the corresponding ones for projective objects.
indexed-coproduct-strong-projective {P = P} {ι = ι} Idx-pro P-pro coprod {X = X} {Y = Y} p e e-strong = do
  s ← Idx-pro
         i  Σ[ sᵢ ∈ Hom (P i) X ] (e ∘ sᵢ ≡ p ∘ ι i))  i  hlevel 2)
         i  P-pro i (p ∘ ι i) e e-strong)
  pure (match  i  s i .fst) , unique₂  i  pullr commute ∙ s i .snd))
  where open is-indexed-coproduct coprod

retract→strong-projective P-pro s r p e e-strong = do
  (t , t-factor) ← P-pro (p ∘ r .retract) e e-strong
  pure (t ∘ s , pulll t-factor ∙ cancelr (r .is-retract))

Moreover, if has a zero object and a strong projective coproduct indexed by a discrete type, then each component of the coproduct is a strong projective.

zero+indexed-coproduct-strong-projective→strong-projective
  :  {κ} {Idx : Type κ} ⦃ Idx-Discrete : Discrete Idx ⦄
   {P : Idx  Ob} {∐P : Ob} {ι :  i  Hom (P i) ∐P}
   Zero C
   is-indexed-coproduct C P ι
   is-strong-projective ∐P
    i  is-strong-projective (P i)
Following the general theme, the proof is identical to the non-strong case.
zero+indexed-coproduct-strong-projective→strong-projective {ι = ι} z coprod ∐P-pro i =
  retract→strong-projective ∐P-pro (ι i) $
  zero→ι-has-retract C coprod z i

Enough strong projectives🔗

A category is said to have enough strong projectives if for object there is some strong epi with strong projective. We will refer to these projectives as projective presentations of

Note that there are two variations on this condition: one where there merely exists a strong projective presentation for every and another where those presentations are provided as structure. We prefer to work with the latter, as it tends to be less painful to work with.

record Strong-projectives : Type (o ⊔ ℓ) where
  field
    Pro : Ob  Ob
    present :  {X}  Hom (Pro X) X
    present-strong-epi :  {X}  is-strong-epi (present {X})
    projective :  {X}  is-strong-projective (Pro X)
module _ (coprods : (Idx : Set)  has-coproducts-indexed-by C ∣ Idx ∣)
  where
  open Cat.Diagram.Separator.Strong C coprods
  open Copowers coprods

If has set-indexed coproducts, and is a strong separating family with each a strong projective, then has enough strong projectives if is a set-projective type.

  strong-projective-separating-faily→strong-projectives
    :  {Idx : Set} {Pᵢ : ∣ Idx ∣  Ob}
     (∀ X  is-set-projective (Σ[ i ∈ ∣ Idx ∣ ] (Hom (Pᵢ i) X)))
     (∀ i  is-strong-projective (Pᵢ i))
     is-strong-separating-family Idx Pᵢ
     Strong-projectives

The hypotheses of this theorem basically give the game away: by definition, there is a strong epimorphism for every Moreover, is set-projective, so the corresponding coproduct is a strong projective.

  strong-projective-separating-faily→strong-projectives
    {Idx} {Pᵢ} Idx-pro Pᵢ-pro strong-sep = strong-projectives where
    open Strong-projectives

    strong-projectives : Strong-projectives
    strong-projectives .Pro X =
      ∐! (Σ[ i ∈ ∣ Idx ∣ ] (Hom (Pᵢ i) X)) (Pᵢ ⊙ fst)
    strong-projectives .present {X = X} =
      ∐!.match (Σ[ i ∈ ∣ Idx ∣ ] (Hom (Pᵢ i) X)) (Pᵢ ⊙ fst) snd
    strong-projectives .present-strong-epi =
      strong-sep
    strong-projectives .projective {X = X} =
      indexed-coproduct-strong-projective
        (Idx-pro X)
        (Pᵢ-pro ⊙ fst)
        (∐!.has-is-ic (Σ[ i ∈ ∣ Idx ∣ ] (Hom (Pᵢ i) X)) (Pᵢ ⊙ fst))