-- Martin Escardo, 3rd August 2015

{-

 We use type-in-type to get impredicativity à la Voevodsky
 (propositional resizing) in Agda.

 No other module in this development uses type-in-type.

 Because of limitations of Agda, or perhaps of our ability to exploit
 Agda, we can only resize from the second universe to the first.

 Curiously, if we write "U" rather than "Set" below, then other
 modules don't type check, even though U=Set by definition. I am not
 sure why this is the case. But it is possible to still use U in other
 modules, and we do.

-}

{-# OPTIONS --type-in-type #-}
{-# OPTIONS --without-K #-}

module prop where

open import preliminaries

-- A proposition is a type with at most one element:

open import isprop public

-- A truth value is a type together with a witness that it is a
-- proposition. "Prop" is an Agda reserved, but unused,
-- keyword, and Agda won't let us use it as an identifier:

Prp : Set
Prp = Σ \(P : Set)  isProp P

-- We could use the pair constructor _,_ from Σ, but this doesn't type
-- check when used in a module without the option type-in-type. The
-- reason is that the types of Σ and _,_ are different with(out)
-- type-in-type. To circumvent this, we define another another binary
-- operator ₍_,_₎, which does type check when used without
-- type-in-type. The reason this works is that the type of this new
-- operator is the same with or without type-in-type:

₍_,_₎ : (P : Set)  isProp P  Prp
₍_,_₎ = _,_

-- It is also necessary, for the same reason, to specialize the
-- types of the projections:

_holds : Prp  Set
_holds = pr₁

infix 20 _holds

holdsIsProp : (p : Prp)  isProp(p holds)
holdsIsProp = pr₂

-- NB. holdsIsProp shows that _holds is an embedding in the sense of
-- the HoTT book.

Prp-≡ : {p q : Prp}  p holds  q holds  p  q
Prp-≡ {p} {q} e = Σ-≡ e (isProp-isProp (transport isProp e (holdsIsProp p)) (holdsIsProp q))

-- open import propua -- <-- Doesn't work. We have to repeat the postulate literally:

postulate prop-ua' : {P Q : Set}  isProp P  isProp Q  (P  Q)  (Q  P)  P  Q

-- (Why? Moreover, there is an example in which we need to use both
-- prop-ua and prop-ua'.)

propext : {p q : Prp}  (p holds  q holds)  (q holds  p holds)  p  q
propext {p} {q} f g = Prp-≡ e
 where
  e : p holds  q holds
  e = prop-ua' (holdsIsProp p) (holdsIsProp q) f g