{-# OPTIONS --safe #-}

module Cubical.Data.SumFin.Properties where

open import Cubical.Foundations.Prelude
open import Cubical.Foundations.Equiv renaming (_∙ₑ_ to _⋆_)
open import Cubical.Foundations.HLevels
open import Cubical.Foundations.Function
open import Cubical.Foundations.Isomorphism
open import Cubical.Foundations.Univalence

open import Cubical.Data.Empty as 
open import Cubical.Data.Unit
open import Cubical.Data.Bool hiding (_≤_)
open import Cubical.Data.Nat
open import Cubical.Data.Nat.Order
import Cubical.Data.Fin as Fin
import Cubical.Data.Fin.LehmerCode as LehmerCode
open import Cubical.Data.SumFin.Base as SumFin
open import Cubical.Data.Sum
open import Cubical.Data.Sigma

open import Cubical.HITs.PropositionalTruncation as Prop

open import Cubical.Relation.Nullary

private
  variable
     : Level
    k : 

SumFin→Fin : Fin k  Fin.Fin k
SumFin→Fin = SumFin.elim  {k} _  Fin.Fin k) Fin.fzero Fin.fsuc

Fin→SumFin : Fin.Fin k  Fin k
Fin→SumFin = Fin.elim  {k} _  Fin k) fzero fsuc

Fin→SumFin-fsuc : (fk : Fin.Fin k)  Fin→SumFin (Fin.fsuc fk)  fsuc (Fin→SumFin fk)
Fin→SumFin-fsuc = Fin.elim-fsuc  {k} _  Fin k) fzero fsuc

SumFin→Fin→SumFin : (fk : Fin k)  Fin→SumFin (SumFin→Fin fk)  fk
SumFin→Fin→SumFin = SumFin.elim  fk  Fin→SumFin (SumFin→Fin fk)  fk)
                                refl λ {k} {fk} eq 
  Fin→SumFin (Fin.fsuc (SumFin→Fin fk)) ≡⟨ Fin→SumFin-fsuc (SumFin→Fin fk) 
  fsuc (Fin→SumFin (SumFin→Fin fk))     ≡⟨ cong fsuc eq 
  fsuc fk                               

Fin→SumFin→Fin : (fk : Fin.Fin k)  SumFin→Fin (Fin→SumFin fk)  fk
Fin→SumFin→Fin = Fin.elim  fk  SumFin→Fin (Fin→SumFin fk)  fk)
                          refl λ {k} {fk} eq 
  SumFin→Fin (Fin→SumFin (Fin.fsuc fk)) ≡⟨ cong SumFin→Fin (Fin→SumFin-fsuc fk) 
  Fin.fsuc (SumFin→Fin (Fin→SumFin fk)) ≡⟨ cong Fin.fsuc eq 
  Fin.fsuc fk                           

SumFin≃Fin :  k  Fin k  Fin.Fin k
SumFin≃Fin _ =
  isoToEquiv (iso SumFin→Fin Fin→SumFin Fin→SumFin→Fin SumFin→Fin→SumFin)

SumFin≡Fin :  k  Fin k  Fin.Fin k
SumFin≡Fin k = ua (SumFin≃Fin k)

enum : (n : )(p : n < k)  Fin k
enum n p = Fin→SumFin (n , p)

enumElim : (P : Fin k  Type )  ((n : )(p : n < k)  P (enum _ p))  (i : Fin k)  P i
enumElim P f i = subst P (SumFin→Fin→SumFin i) (f (SumFin→Fin i .fst) (SumFin→Fin i .snd))

-- Closure properties of SumFin under type constructors

SumFin⊎≃ : (m n : )  (Fin m  Fin n)  (Fin (m + n))
SumFin⊎≃ 0 n = ⊎-swap-≃  ⊎-⊥-≃
SumFin⊎≃ (suc m) n = ⊎-assoc-≃  ⊎-equiv (idEquiv ) (SumFin⊎≃ m n)

SumFinΣ≃ : (n : )(f : Fin n  )  (Σ (Fin n)  x  Fin (f x)))  (Fin (totalSum f))
SumFinΣ≃ 0 f = ΣEmpty _
SumFinΣ≃ (suc n) f =
    Σ⊎≃
   ⊎-equiv (ΣUnit  tt  Fin (f (inl tt)))) (SumFinΣ≃ n  x  f (inr x)))
   SumFin⊎≃ (f (inl tt)) (totalSum  x  f (inr x)))

SumFin×≃ : (m n : )  (Fin m × Fin n)  (Fin (m · n))
SumFin×≃ m n = SumFinΣ≃ m  _  n)  pathToEquiv  i  Fin (totalSumConst {m = m} n i))

SumFinΠ≃ : (n : )(f : Fin n  )  ((x : Fin n)  Fin (f x))  (Fin (totalProd f))
SumFinΠ≃ 0 _ = isContr→≃Unit (isContrΠ⊥)  invEquiv (⊎-⊥-≃)
SumFinΠ≃ (suc n) f =
    Π⊎≃
   Σ-cong-equiv (ΠUnit  tt  Fin (f (inl tt))))  _  SumFinΠ≃ n  x  f (inr x)))
   SumFin×≃ (f (inl tt)) (totalProd  x  f (inr x)))

isNotZero :   
isNotZero 0 = 0
isNotZero (suc n) = 1

SumFin∥∥≃ : (n : )   Fin n ∥₁  Fin (isNotZero n)
SumFin∥∥≃ 0 = propTruncIdempotent≃ (isProp⊥)
SumFin∥∥≃ (suc n) =
    isContr→≃Unit (inhProp→isContr  inl tt ∣₁ isPropPropTrunc)
   isContr→≃Unit (isContrUnit)  invEquiv (⊎-⊥-≃)

ℕ→Bool :   Bool
ℕ→Bool 0 = false
ℕ→Bool (suc n) = true

SumFin∥∥DecProp : (n : )   Fin n ∥₁  Bool→Type (ℕ→Bool n)
SumFin∥∥DecProp 0 = uninhabEquiv (Prop.rec isProp⊥ ⊥.rec) ⊥.rec
SumFin∥∥DecProp (suc n) = isContr→≃Unit (inhProp→isContr  inl tt ∣₁ isPropPropTrunc)

-- negation of SumFin

SumFin¬ : (n : )  (¬ Fin n)  Bool→Type (isZero n)
SumFin¬ 0 = isContr→≃Unit isContr⊥→A
SumFin¬ (suc n) = uninhabEquiv  f  f fzero) ⊥.rec

-- SumFin 1 is equivalent to unit

Fin1≃Unit : Fin 1  Unit
Fin1≃Unit = ⊎-⊥-≃

isContrSumFin1 : isContr (Fin 1)
isContrSumFin1 = isOfHLevelRespectEquiv 0 (invEquiv Fin1≃Unit) isContrUnit

-- SumFin 2 is equivalent to Bool

SumFin2≃Bool : Fin 2  Bool
SumFin2≃Bool = ⊎-equiv (idEquiv _) ⊎-⊥-≃  isoToEquiv Iso-⊤⊎⊤-Bool

-- decidable predicate over SumFin

SumFinℙ≃ : (n : )  (Fin n  Bool)  Fin (2 ^ n)
SumFinℙ≃ 0 = isContr→≃Unit (isContrΠ⊥)  invEquiv (⊎-⊥-≃)
SumFinℙ≃ (suc n) =
    Π⊎≃
   Σ-cong-equiv (UnitToType≃ Bool  invEquiv SumFin2≃Bool)  _  SumFinℙ≃ n)
   SumFin×≃ 2 (2 ^ n)

-- decidable subsets of SumFin

Bool→ℕ : Bool  
Bool→ℕ true = 1
Bool→ℕ false = 0

trueCount : {n : }(f : Fin n  Bool)  
trueCount {n = 0} _ = 0
trueCount {n = suc n} f = Bool→ℕ (f (inl tt)) + (trueCount (f  inr))

SumFinDec⊎≃ : (n : )(t : Bool)  (Bool→Type t  Fin n)  (Fin (Bool→ℕ t + n))
SumFinDec⊎≃ _ true = idEquiv _
SumFinDec⊎≃ _ false = ⊎-swap-≃  ⊎-⊥-≃

SumFinSub≃ : (n : )(f : Fin n  Bool)  Σ _ (Bool→Type  f)  Fin (trueCount f)
SumFinSub≃ 0 _ = ΣEmpty _
SumFinSub≃ (suc n) f =
    Σ⊎≃
   ⊎-equiv (ΣUnit (Bool→Type  f  inl)) (SumFinSub≃ n (f  inr))
   SumFinDec⊎≃ _ (f (inl tt))

-- decidable quantifier

trueForSome : (n : )(f : Fin n  Bool)  Bool
trueForSome 0 _ = false
trueForSome (suc n) f = f (inl tt) or trueForSome n (f  inr)

trueForAll : (n : )(f : Fin n  Bool)  Bool
trueForAll 0 _ = true
trueForAll (suc n) f = f (inl tt) and trueForAll n (f  inr)

SumFin∃→ : (n : )(f : Fin n  Bool)  Σ _ (Bool→Type  f)  Bool→Type (trueForSome n f)
SumFin∃→ 0 _ = ΣEmpty _ .fst
SumFin∃→ (suc n) f =
    Bool→Type⊎' _ _
   map-⊎ (ΣUnit (Bool→Type  f  inl) .fst) (SumFin∃→ n (f  inr))
   Σ⊎≃ .fst

SumFin∃← : (n : )(f : Fin n  Bool)  Bool→Type (trueForSome n f)  Σ _ (Bool→Type  f)
SumFin∃← 0 _ = ⊥.rec
SumFin∃← (suc n) f =
    invEq Σ⊎≃
   map-⊎ (invEq (ΣUnit (Bool→Type  f  inl))) (SumFin∃← n (f  inr))
   Bool→Type⊎ _ _

SumFin∃≃ : (n : )(f : Fin n  Bool)   Σ (Fin n) (Bool→Type  f) ∥₁  Bool→Type (trueForSome n f)
SumFin∃≃ n f =
  propBiimpl→Equiv isPropPropTrunc isPropBool→Type
    (Prop.rec isPropBool→Type (SumFin∃→ n f))
    (∣_∣₁  SumFin∃← n f)

SumFin∀≃ : (n : )(f : Fin n  Bool)  ((x : Fin n)  Bool→Type (f x))  Bool→Type (trueForAll n f)
SumFin∀≃ 0 _ = isContr→≃Unit (isContrΠ⊥)
SumFin∀≃ (suc n) f =
    Π⊎≃
   Σ-cong-equiv (ΠUnit (Bool→Type  f  inl))  _  SumFin∀≃ n (f  inr))
   Bool→Type×≃ _ _

-- internal equality

SumFin≡ : (n : )  (a b : Fin n)  Bool
SumFin≡ 0 _ _ = true
SumFin≡ (suc n) (inl tt) (inl tt) = true
SumFin≡ (suc n) (inl tt) (inr y) = false
SumFin≡ (suc n) (inr x) (inl tt) = false
SumFin≡ (suc n) (inr x) (inr y) = SumFin≡ n x y

isSetSumFin : (n : )→ isSet (Fin n)
isSetSumFin 0 = isProp→isSet isProp⊥
isSetSumFin (suc n) = isSet⊎ (isProp→isSet isPropUnit) (isSetSumFin n)

SumFin≡≃ : (n : )  (a b : Fin n)  (a  b)  Bool→Type (SumFin≡ n a b)
SumFin≡≃ 0 _ _ = isContr→≃Unit (isProp→isContrPath isProp⊥ _ _)
SumFin≡≃ (suc n) (inl tt) (inl tt) = isContr→≃Unit (inhProp→isContr refl (isSetSumFin _ _ _))
SumFin≡≃ (suc n) (inl tt) (inr y) = invEquiv (⊎Path.Cover≃Path _ _)  uninhabEquiv ⊥.rec* ⊥.rec
SumFin≡≃ (suc n) (inr x) (inl tt) = invEquiv (⊎Path.Cover≃Path _ _)  uninhabEquiv ⊥.rec* ⊥.rec
SumFin≡≃ (suc n) (inr x) (inr y) = invEquiv (_ , isEmbedding-inr x y)  SumFin≡≃ n x y

-- propositional truncation of Fin

-- decidability of Fin

DecFin : (n : )  Dec (Fin n)
DecFin 0 = no (idfun _)
DecFin (suc n) = yes fzero

-- propositional truncation of Fin

Dec∥Fin∥ : (n : )  Dec  Fin n ∥₁
Dec∥Fin∥ n = Dec∥∥ (DecFin n)

-- some properties about cardinality

fzero≠fone : {n : }  ¬ (fzero  fsuc fzero)
fzero≠fone {n = n} = SumFin≡≃ (suc (suc n)) fzero (fsuc fzero) .fst

Fin>0→isInhab : (n : )  0 < n  Fin n
Fin>0→isInhab 0 p = ⊥.rec (¬-<-zero p)
Fin>0→isInhab (suc n) p = fzero

Fin>1→hasNonEqualTerm : (n : )  1 < n  Σ[ i  Fin n ] Σ[ j  Fin n ] ¬ i  j
Fin>1→hasNonEqualTerm 0 p = ⊥.rec (snotz (≤0→≡0 p))
Fin>1→hasNonEqualTerm 1 p = ⊥.rec (snotz (≤0→≡0 (pred-≤-pred p)))
Fin>1→hasNonEqualTerm (suc (suc n)) _ = fzero , fsuc fzero , fzero≠fone

isEmpty→Fin≡0 : (n : )  ¬ Fin n  0  n
isEmpty→Fin≡0 0 _ = refl
isEmpty→Fin≡0 (suc n) p = ⊥.rec (p fzero)

isInhab→Fin>0 : (n : )  Fin n  0 < n
isInhab→Fin>0 0 i = ⊥.rec i
isInhab→Fin>0 (suc n) _ = suc-≤-suc zero-≤

hasNonEqualTerm→Fin>1 : (n : )  (i j : Fin n)  ¬ i  j  1 < n
hasNonEqualTerm→Fin>1 0 i _ _ = ⊥.rec i
hasNonEqualTerm→Fin>1 1 i j p = ⊥.rec (p (isContr→isProp isContrSumFin1 i j))
hasNonEqualTerm→Fin>1 (suc (suc n)) _ _ _ = suc-≤-suc (suc-≤-suc zero-≤)

Fin≤1→isProp : (n : )  n  1  isProp (Fin n)
Fin≤1→isProp 0 _ = isProp⊥
Fin≤1→isProp 1 _ = isContr→isProp isContrSumFin1
Fin≤1→isProp (suc (suc n)) p = ⊥.rec (¬-<-zero (pred-≤-pred p))

isProp→Fin≤1 : (n : )  isProp (Fin n)  n  1
isProp→Fin≤1 0 _ = ≤-solver 0 1
isProp→Fin≤1 1 _ = ≤-solver 1 1
isProp→Fin≤1 (suc (suc n)) p = ⊥.rec (fzero≠fone (p fzero (fsuc fzero)))

-- automorphisms of SumFin

SumFin≃≃ : (n : )  (Fin n  Fin n)  Fin (LehmerCode.factorial n)
SumFin≃≃ _ =
    equivComp (SumFin≃Fin _) (SumFin≃Fin _)
   LehmerCode.lehmerEquiv
   LehmerCode.lehmerFinEquiv
   invEquiv (SumFin≃Fin _)