{-# OPTIONS --without-K --safe #-}

open import Tools.Relation

module Definition.Typed.Weakening {a b} (M′ : Setoid a b) where

open Setoid M′ using () renaming (Carrier to M)

open import Definition.Untyped M as U hiding (wk ; _∷_)
open import Definition.Untyped.Properties M
open import Definition.Typed M′ hiding (_,_)

open import Tools.Fin
open import Tools.Nat
import Tools.PropositionalEquality as PE

private
  variable
     n m  : Nat
    A B t u : Term n
    Γ  : Con Term n
    Δ  : Con Term m
    Δ′ : Con Term 
    ρ  : Wk m n
    ρ′ : Wk n 

-- Weakening type

data _∷_⊆_ : Wk m n  Con Term m  Con Term n  Set a where
  id   :             id      Γ             Γ
  step : ρ  Δ  Γ  step ρ  Δ  A         Γ
  lift : ρ  Δ  Γ  lift ρ  Δ  U.wk ρ A  Γ  A


-- -- Weakening composition

_•ₜ_ : ρ  Γ  Δ  ρ′  Δ  Δ′  ρ  ρ′  Γ  Δ′
id     •ₜ η′ = η′
step η •ₜ η′ = step (η •ₜ η′)
lift η •ₜ id = lift η
lift η •ₜ step η′ = step (η •ₜ η′)
_•ₜ_ {ρ = lift ρ} {ρ′ = lift ρ′} {Δ′ = Δ′  A} (lift η) (lift η′) =
  PE.subst  x  lift (ρ  ρ′)  x  Δ′  A)
           (PE.cong₂ _∙_ PE.refl (PE.sym (wk-comp ρ ρ′ A)))
           (lift (η •ₜ η′))


-- Weakening of judgements

wkIndex :  {n}  ρ  Δ  Γ 
        let ρA = U.wk ρ A
            ρn = wkVar ρ n
        in   Δ  n  A  Γ  ρn  ρA  Δ
wkIndex id ⊢Δ i = PE.subst  x  _  x  _) (PE.sym (wk-id _)) i
wkIndex (step ρ) (⊢Δ  A) i = PE.subst  x  _  x  _)
                                       (wk1-wk _ _)
                                       (there (wkIndex ρ ⊢Δ i))
wkIndex (lift ρ) (⊢Δ  A) (there i) = PE.subst  x  _  x  _)
                                               (wk1-wk≡lift-wk1 _ _)
                                               (there (wkIndex ρ ⊢Δ i))
wkIndex (lift ρ) ⊢Δ here =
  let G = _
      n = _
  in  PE.subst  x  n  x  G)
               (wk1-wk≡lift-wk1 _ _)
               here

mutual
  wk : ρ  Δ  Γ 
     let ρA = U.wk ρ A
     in   Δ  Γ  A  Δ  ρA
  wk ρ ⊢Δ (ℕⱼ ⊢Γ) = ℕⱼ ⊢Δ
  wk ρ ⊢Δ (Emptyⱼ ⊢Γ) = Emptyⱼ ⊢Δ
  wk ρ ⊢Δ (Unitⱼ ⊢Γ) = Unitⱼ ⊢Δ
  wk ρ ⊢Δ (Uⱼ ⊢Γ) = Uⱼ ⊢Δ
  wk ρ ⊢Δ (Πⱼ F  G) = let ρF = wk ρ ⊢Δ F
                       in  Πⱼ ρF  (wk (lift ρ) (⊢Δ  ρF) G)
  wk ρ ⊢Δ (Σⱼ F  G) = let ρF = wk ρ ⊢Δ F
                       in  Σⱼ ρF  (wk (lift ρ) (⊢Δ  ρF) G)
  wk ρ ⊢Δ (univ A) = univ (wkTerm ρ ⊢Δ A)

  wkTerm : {Δ : Con Term m} {ρ : Wk m n}  ρ  Δ  Γ 
         let ρA = U.wk ρ A
             ρt = U.wk ρ t
         in  Δ  Γ  t  A  Δ  ρt  ρA
  wkTerm ρ ⊢Δ (ℕⱼ ⊢Γ) = ℕⱼ ⊢Δ
  wkTerm ρ ⊢Δ (Emptyⱼ ⊢Γ) = Emptyⱼ ⊢Δ
  wkTerm ρ ⊢Δ (Unitⱼ ⊢Γ) = Unitⱼ ⊢Δ
  wkTerm ρ ⊢Δ (Πⱼ F  G) = let ρF = wkTerm ρ ⊢Δ F
                           in  Πⱼ ρF  (wkTerm (lift ρ) (⊢Δ  univ ρF) G)
  wkTerm ρ ⊢Δ (Σⱼ F  G) = let ρF = wkTerm ρ ⊢Δ F
                           in  Σⱼ ρF  (wkTerm (lift ρ) (⊢Δ  univ ρF) G)
  wkTerm ρ ⊢Δ (var ⊢Γ x) = var ⊢Δ (wkIndex ρ ⊢Δ x)
  wkTerm ρ ⊢Δ (lamⱼ F t) = let ρF = wk ρ ⊢Δ F
                           in lamⱼ ρF (wkTerm (lift ρ) (⊢Δ  ρF) t)
  wkTerm ρ ⊢Δ (_∘ⱼ_ {G = G} g a) = PE.subst  x  _  _  x)
                                           (PE.sym (wk-β G))
                                           (wkTerm ρ ⊢Δ g ∘ⱼ wkTerm ρ ⊢Δ a)
  wkTerm ρ ⊢Δ (prodⱼ {G = G} ⊢F ⊢G t u) =
    let ρF = wk ρ ⊢Δ ⊢F
        ρG = wk (lift ρ) (⊢Δ  ρF) ⊢G
        ρt = wkTerm ρ ⊢Δ t
        ρu = wkTerm ρ ⊢Δ u
        ρu = PE.subst  x  _  _  x) (wk-β G) ρu
    in  prodⱼ ρF ρG ρt ρu
  wkTerm ρ ⊢Δ (fstⱼ {G = G} ⊢F ⊢G t) =
    let ρF = wk ρ ⊢Δ ⊢F
        ρG = wk (lift ρ) (⊢Δ  ρF) ⊢G
        ρt = wkTerm ρ ⊢Δ t
    in  fstⱼ ρF ρG ρt
  wkTerm ρ ⊢Δ (sndⱼ {G = G} ⊢F ⊢G t) =
    let ρF = wk ρ ⊢Δ ⊢F
        ρG = wk (lift ρ) (⊢Δ  ρF) ⊢G
        ρt = wkTerm ρ ⊢Δ t
    in  PE.subst  x  _  snd _  x) (PE.sym (wk-β G)) (sndⱼ ρF ρG ρt)
  wkTerm ρ ⊢Δ (zeroⱼ ⊢Γ) = zeroⱼ ⊢Δ
  wkTerm ρ ⊢Δ (sucⱼ n) = sucⱼ (wkTerm ρ ⊢Δ n)
  wkTerm {Δ = Δ} {ρ = ρ} [ρ] ⊢Δ (natrecⱼ {G = G} {s = s} ⊢G ⊢z ⊢s ⊢n) =
    PE.subst  x  _  natrec _ _ _ _ _ _  x) (PE.sym (wk-β G))
             (natrecⱼ (wk (lift [ρ]) (⊢Δ  ℕⱼ ⊢Δ) ⊢G)
                      (PE.subst  x  _  _  x) (wk-β G) (wkTerm [ρ] ⊢Δ ⊢z))
                      (PE.subst  x  ((Δ  )  (U.wk (lift ρ) G))  U.wk (lift (lift ρ)) s  x)
                                (wk-β-natrec ρ G)
                                (wkTerm (lift (lift [ρ])) (((⊢Δ  (ℕⱼ ⊢Δ))
                                         (wk (lift [ρ]) (⊢Δ  (ℕⱼ ⊢Δ)) ⊢G)))
                                        ⊢s))
                      (wkTerm [ρ] ⊢Δ ⊢n))
  wkTerm {Δ = Δ} {ρ = ρ} [ρ] ⊢Δ (prodrecⱼ {A = A} ⊢F ⊢G ⊢A ⊢t ⊢u) =
    let ⊢ρF = wk [ρ] ⊢Δ ⊢F
        ⊢ρG = wk (lift [ρ]) (⊢Δ  ⊢ρF) ⊢G
        ⊢ρA = wk (lift [ρ]) (⊢Δ  (Σⱼ ⊢ρF  ⊢ρG)) ⊢A
        ⊢ρt = wkTerm [ρ] ⊢Δ ⊢t
        ⊢ρu = wkTerm (lift (lift [ρ])) (⊢Δ  ⊢ρF  ⊢ρG) ⊢u
    in  PE.subst  x  _  prodrec _ _ _ _  x) (PE.sym (wk-β A))
                 (prodrecⱼ ⊢ρF ⊢ρG ⊢ρA ⊢ρt
                           (PE.subst  x  _  _  x)
                                     (wk-β-prodrec ρ A) ⊢ρu))
  wkTerm {Δ = Δ} {ρ = ρ} [ρ] ⊢Δ (Emptyrecⱼ {A = A} {e = e} ⊢A ⊢e) =
    (Emptyrecⱼ (wk [ρ] ⊢Δ ⊢A) (wkTerm [ρ] ⊢Δ ⊢e))
  wkTerm ρ ⊢Δ (starⱼ ⊢Γ) = starⱼ ⊢Δ
  wkTerm ρ ⊢Δ (conv t A≡B) = conv (wkTerm ρ ⊢Δ t) (wkEq ρ ⊢Δ A≡B)

  wkEq : ρ  Δ  Γ 
       let ρA = U.wk ρ A
           ρB = U.wk ρ B
       in  Δ  Γ  A  B  Δ  ρA  ρB
  wkEq ρ ⊢Δ (univ A≡B) = univ (wkEqTerm ρ ⊢Δ A≡B)
  wkEq ρ ⊢Δ (refl A) = refl (wk ρ ⊢Δ A)
  wkEq ρ ⊢Δ (sym A≡B) = sym (wkEq ρ ⊢Δ A≡B)
  wkEq ρ ⊢Δ (trans A≡B B≡C) = trans (wkEq ρ ⊢Δ A≡B) (wkEq ρ ⊢Δ B≡C)
  wkEq ρ ⊢Δ (Π-cong F F≡H G≡E p≈p′ q≈q′) = let ρF = wk ρ ⊢Δ F
                                           in  Π-cong ρF (wkEq ρ ⊢Δ F≡H)
                                                      (wkEq (lift ρ) (⊢Δ  ρF) G≡E)
                                                      p≈p′ q≈q′
  wkEq ρ ⊢Δ (Σ-cong F F≡H G≡E p≈p′) = let ρF = wk ρ ⊢Δ F
                                      in  Σ-cong ρF (wkEq ρ ⊢Δ F≡H)
                                                 (wkEq (lift ρ) (⊢Δ  ρF) G≡E)
                                                 p≈p′

  wkEqTerm : {Δ : Con Term m} {ρ : Wk m n}  ρ  Δ  Γ 
           let ρA = U.wk ρ A
               ρt = U.wk ρ t
               ρu = U.wk ρ u
           in  Δ  Γ  t  u  A  Δ  ρt  ρu  ρA
  wkEqTerm ρ ⊢Δ (refl t) = refl (wkTerm ρ ⊢Δ t)
  wkEqTerm ρ ⊢Δ (sym t≡u) = sym (wkEqTerm ρ ⊢Δ t≡u)
  wkEqTerm ρ ⊢Δ (trans t≡u u≡r) = trans (wkEqTerm ρ ⊢Δ t≡u) (wkEqTerm ρ ⊢Δ u≡r)
  wkEqTerm ρ ⊢Δ (conv t≡u A≡B) = conv (wkEqTerm ρ ⊢Δ t≡u) (wkEq ρ ⊢Δ A≡B)
  wkEqTerm ρ ⊢Δ (Π-cong F F≡H G≡E p≈p′ q≈q′) =
    let ρF = wk ρ ⊢Δ F
    in  Π-cong ρF (wkEqTerm ρ ⊢Δ F≡H)
                  (wkEqTerm (lift ρ) (⊢Δ  ρF) G≡E)
                  p≈p′ q≈q′
  wkEqTerm ρ ⊢Δ (Σ-cong F F≡H G≡E q≈q′) =
    let ρF = wk ρ ⊢Δ F
    in  Σ-cong ρF (wkEqTerm ρ ⊢Δ F≡H)
                  (wkEqTerm (lift ρ) (⊢Δ  ρF) G≡E)
                  q≈q′
  wkEqTerm ρ ⊢Δ (app-cong {G = G} f≡g a≡b p≈p₁ p≈p₂) =
    PE.subst  x  _  _  _  x)
             (PE.sym (wk-β G))
             (app-cong (wkEqTerm ρ ⊢Δ f≡g) (wkEqTerm ρ ⊢Δ a≡b) p≈p₁ p≈p₂)
  wkEqTerm ρ ⊢Δ (β-red {a = a} {t = t} {G = G} F ⊢G ⊢t ⊢a x) =
    let ρF = wk ρ ⊢Δ F
        ⊢ρG = wk (lift ρ) (⊢Δ  ρF) ⊢G
    in  PE.subst  x  _  _  _  x)
                 (PE.sym (wk-β G))
                 (PE.subst  x  _  U.wk _ ((lam _ t)  _  a)  x  _)
                           (PE.sym (wk-β t))
                           (β-red ρF ⊢ρG (wkTerm (lift ρ) (⊢Δ  ρF) ⊢t)
                                     (wkTerm ρ ⊢Δ ⊢a) x))
  wkEqTerm ρ ⊢Δ (η-eq F f g f0≡g0) =
    let ρF = wk ρ ⊢Δ F
    in  η-eq ρF (wkTerm ρ ⊢Δ f)
                (wkTerm ρ ⊢Δ g)
                λ x x₁  PE.subst₂  t u  _  t  _  _  u  _  _  _)
                                  (PE.sym (wk1-wk≡lift-wk1 _ _))
                                  (PE.sym (wk1-wk≡lift-wk1 _ _))
                                  (wkEqTerm (lift ρ) (⊢Δ  ρF) (f0≡g0 x x₁))
  wkEqTerm ρ ⊢Δ (fst-cong ⊢F ⊢G t≡t') =
    let ρF = wk ρ ⊢Δ ⊢F
        ρG = wk (lift ρ) (⊢Δ  ρF) ⊢G
    in  fst-cong ρF ρG (wkEqTerm ρ ⊢Δ t≡t')
  wkEqTerm ρ ⊢Δ (snd-cong {G = G} ⊢F ⊢G t≡t') =
    let ρF = wk ρ ⊢Δ ⊢F
        ρG = wk (lift ρ) (⊢Δ  ρF) ⊢G
        ρt≡t' = wkEqTerm ρ ⊢Δ t≡t'
    in  PE.subst  x  _  snd _  snd _  x) (PE.sym (wk-β G))
      (snd-cong ρF ρG ρt≡t')
  wkEqTerm {Δ = Δ} {ρ = ρ} [ρ] ⊢Δ (prod-cong {G = G} {t} {t′} {u} {u′} ⊢F ⊢G t≡t′ u≡u′) =
    let ρF = wk [ρ] ⊢Δ ⊢F
        ρG = wk (lift [ρ]) (⊢Δ  ρF) ⊢G
        ρt≡t′ = wkEqTerm [ρ] ⊢Δ t≡t′
        ρu≡u′ = wkEqTerm [ρ] ⊢Δ u≡u′
    in  prod-cong ρF ρG ρt≡t′ (PE.subst  x  Δ  _  _  x) (wk-β G) ρu≡u′)
  wkEqTerm ρ ⊢Δ (Σ-η {G = G} ⊢F ⊢G ⊢p ⊢r fst≡ snd≡) =
    let ρF = wk ρ ⊢Δ ⊢F
        ρG = wk (lift ρ) (⊢Δ  ρF) ⊢G
        ρp = wkTerm ρ ⊢Δ ⊢p
        ρr = wkTerm ρ ⊢Δ ⊢r
        ρfst≡ = wkEqTerm ρ ⊢Δ fst≡
        ρsnd≡ = wkEqTerm ρ ⊢Δ snd≡
        ρsnd≡ = PE.subst  x  _  _  _  x)
                         (wk-β G)
                         ρsnd≡
    in  Σ-η ρF ρG ρp ρr ρfst≡ ρsnd≡
  wkEqTerm {ρ = ρ} [ρ] ⊢Δ (Σ-β₁ {q = q} {G = G} ⊢F ⊢G t u p) =
    let ρF = wk [ρ] ⊢Δ ⊢F
        ρG = wk (lift [ρ]) (⊢Δ  ρF) ⊢G
        ρt = wkTerm [ρ] ⊢Δ t
        ρu = wkTerm [ρ] ⊢Δ u
        ρu = PE.subst  x  _  _  x) (wk-β G) ρu
    in  Σ-β₁ {q = q} ρF ρG ρt ρu (prodⱼ ρF ρG ρt ρu)
  wkEqTerm {ρ = ρ} [ρ] ⊢Δ (Σ-β₂ {q = q} {G = G} ⊢F ⊢G t u p) =
    let ρF = wk [ρ] ⊢Δ ⊢F
        ρG = wk (lift [ρ]) (⊢Δ  ρF) ⊢G
        ρt = wkTerm [ρ] ⊢Δ t
        ρu = wkTerm [ρ] ⊢Δ u
        ρu = PE.subst  x  _  _  x) (wk-β G) ρu
    in  PE.subst  x  _  _  _  x) (PE.sym (wk-β G))
      (Σ-β₂ {q = q} ρF ρG ρt ρu (prodⱼ ρF ρG ρt ρu))
  wkEqTerm ρ ⊢Δ (suc-cong m≡n) = suc-cong (wkEqTerm ρ ⊢Δ m≡n)
  wkEqTerm {Δ = Δ} {ρ = ρ} [ρ] ⊢Δ (natrec-cong {s = s} {s′ = s′} {F = F}
                                     ⊢F F≡F′ z≡z′ s≡s′ n≡n′ p≈p′ r≈r′) =
              PE.subst  x  Δ  natrec _ _ _ _ _ _  _  x) (PE.sym (wk-β F))
                       (natrec-cong (wk (lift [ρ]) (⊢Δ  (ℕⱼ ⊢Δ)) ⊢F)
                          (wkEq (lift [ρ]) (⊢Δ  ℕⱼ ⊢Δ) F≡F′)
                          (PE.subst  x  Δ  _  _  x) (wk-β F)
                                    (wkEqTerm [ρ] ⊢Δ z≡z′))
                          (PE.subst  x  ((Δ  ) 
                                    (U.wk (lift ρ) F))  U.wk (lift (lift ρ)) s
                                                        U.wk (lift (lift ρ)) s′  x)
                                    (wk-β-natrec _ F)
                                    (wkEqTerm (lift (lift [ρ]))
                                              ((⊢Δ  (ℕⱼ ⊢Δ)) 
                                              (wk (lift [ρ]) (⊢Δ  ℕⱼ ⊢Δ) ⊢F)) s≡s′))
                          (wkEqTerm [ρ] ⊢Δ n≡n′) p≈p′ r≈r′)
  wkEqTerm {Δ = Δ} {ρ = ρ} [ρ] ⊢Δ (natrec-zero {z = z} {s = s} {F = F} ⊢F ⊢z ⊢s) =
    PE.subst  x  Δ  natrec _ _ (U.wk (lift _) F) _ _ _  _  x)
             (PE.sym (wk-β F))
             (natrec-zero (wk (lift [ρ]) (⊢Δ  ℕⱼ ⊢Δ) ⊢F)
                          (PE.subst  x  Δ  U.wk ρ z  x)
                                    (wk-β F)
                                    (wkTerm [ρ] ⊢Δ ⊢z))
                          (PE.subst  x  ((Δ  )  (U.wk (lift ρ) F))  U.wk (lift (lift ρ)) s  x)
                                    (wk-β-natrec _ F)
                                    (wkTerm (lift (lift [ρ])) ((⊢Δ  (ℕⱼ ⊢Δ))
                                       (wk (lift [ρ]) (⊢Δ  (ℕⱼ ⊢Δ)) ⊢F)) ⊢s)))
  wkEqTerm {Δ = Δ} {ρ = ρ} [ρ] ⊢Δ (natrec-suc {p = p} {r = r} {n = n} {z = z} {s = s} {F = F} ⊢n ⊢F ⊢z ⊢s) =
    let ρn = wkTerm [ρ] ⊢Δ ⊢n
        ρF = wk (lift [ρ]) (⊢Δ  (ℕⱼ ⊢Δ)) ⊢F
        ρz = PE.subst  x  _  _  x) (wk-β F) (wkTerm [ρ] ⊢Δ ⊢z)
    in  PE.subst  x  _  natrec _ _ (U.wk (lift ρ) F) _ _ _
                           U.wk ρ (s [ n , natrec p r F z s n ])  x)
             (PE.sym (wk-β F))
             (PE.subst  x  Δ  natrec _ _ _ _ _ _  x  _)
                       (PE.sym (wk-β-doubleSubst ρ s (natrec p r F z s n) n))
                       (natrec-suc ρn ρF ρz
                                   (PE.subst  x  ((Δ  )  (U.wk (lift ρ) F))  _  x)
                                             (wk-β-natrec _ F)
                                             (wkTerm (lift (lift [ρ]))
                                                     ((⊢Δ  (ℕⱼ ⊢Δ))  ρF)
                                                     ⊢s))))
  wkEqTerm {Δ = Δ} {ρ = ρ} [ρ] ⊢Δ (prodrec-cong {A = A} ⊢F ⊢G A≡A′ t≡t′ u≡u′ p≈p′) =
    let ρF = wk [ρ] ⊢Δ ⊢F
        ρG = wk (lift [ρ]) (⊢Δ  ρF) ⊢G
        ρA≡A′ = wkEq (lift [ρ]) (⊢Δ  (Σⱼ ρF  ρG)) A≡A′
        ρt≡t′ = wkEqTerm [ρ] ⊢Δ t≡t′
        ρu≡u′ = wkEqTerm (lift (lift [ρ])) (⊢Δ  ρF  ρG) u≡u′
    in  PE.subst  x  Δ  prodrec _ _ _ _  _  x) (PE.sym (wk-β A))
                 (prodrec-cong ρF ρG ρA≡A′ ρt≡t′
                               (PE.subst  x  _  _  _  x)
                                         (wk-β-prodrec ρ A) ρu≡u′)
                               p≈p′)
  wkEqTerm {Δ = Δ} {ρ = ρ} [ρ] ⊢Δ (prodrec-β {u = u} {G = G} {A = A} ⊢F ⊢G ⊢A ⊢t ⊢t′ ⊢u) =
    let ρF = wk [ρ] ⊢Δ ⊢F
        ρG = wk (lift [ρ]) (⊢Δ  ρF) ⊢G
        ρA = wk (lift [ρ]) (⊢Δ  (Σⱼ ρF  ρG)) ⊢A
        ρt = wkTerm [ρ] ⊢Δ ⊢t
        ρt′ = wkTerm [ρ] ⊢Δ ⊢t′
        ρu = wkTerm (lift (lift [ρ])) (⊢Δ  ρF  ρG) ⊢u
    in  PE.subst₂  x y  _  prodrec _ _ _ _  x  y)
                  (PE.trans (subst-wk u)
                    (PE.trans (substVar-to-subst (λ{x0  PE.refl; (x0 +1)  PE.refl; (x +1 +1)  PE.refl}) u)
                              (PE.sym (wk-subst u))))
                  (PE.sym (wk-β A))
                 (prodrec-β ρF ρG ρA ρt
                           (PE.subst  x  _  _  x) (wk-β G) ρt′)
                           (PE.subst  x  _  _  x) (wk-β-prodrec ρ A) ρu))
  wkEqTerm {Δ = Δ} {ρ = ρ} [ρ] ⊢Δ (Emptyrec-cong {A = A} {A' = A'} {e = e} {e' = e'}
                                  A≡A' e≡e' p≈p′) =
    (Emptyrec-cong (wkEq [ρ] ⊢Δ A≡A')
                   (wkEqTerm [ρ] ⊢Δ e≡e') p≈p′)
  wkEqTerm ρ ⊢Δ (η-unit e e') = η-unit (wkTerm ρ ⊢Δ e) (wkTerm ρ ⊢Δ e')

mutual
  wkRed : ρ  Δ  Γ 
           let ρA = U.wk ρ A
               ρB = U.wk ρ B
           in  Δ  Γ  A  B  Δ  ρA  ρB
  wkRed ρ ⊢Δ (univ A⇒B) = univ (wkRedTerm ρ ⊢Δ A⇒B)

  wkRedTerm : {Δ : Con Term m} {ρ : Wk m n}  ρ  Δ  Γ 
           let ρA = U.wk ρ A
               ρt = U.wk ρ t
               ρu = U.wk ρ u
           in  Δ  Γ  t  u  A  Δ  ρt  ρu  ρA
  wkRedTerm ρ ⊢Δ (conv t⇒u A≡B) = conv (wkRedTerm ρ ⊢Δ t⇒u) (wkEq ρ ⊢Δ A≡B)
  wkRedTerm ρ ⊢Δ (app-subst {B = B} t⇒u a) =
    PE.subst  x  _  _  _  x) (PE.sym (wk-β B))
             (app-subst (wkRedTerm ρ ⊢Δ t⇒u) (wkTerm ρ ⊢Δ a))
  wkRedTerm ρ ⊢Δ (β-red {A = A} {B = B} {a = a} {t = t} ⊢A ⊢B ⊢t ⊢a p≡q) =
    let ⊢ρA = wk ρ ⊢Δ ⊢A
        ⊢ρB = wk (lift ρ) (⊢Δ  ⊢ρA) ⊢B
    in  PE.subst  x  _  _  _  x) (PE.sym (wk-β B))
                 (PE.subst  x  _  U.wk _ ((lam _ t)  _  a)  x  _)
                           (PE.sym (wk-β t))
                           (β-red ⊢ρA ⊢ρB (wkTerm (lift ρ) (⊢Δ  ⊢ρA) ⊢t)
                                      (wkTerm ρ ⊢Δ ⊢a) p≡q))
  wkRedTerm ρ ⊢Δ (fst-subst ⊢F ⊢G t⇒) =
    let ρF = wk ρ ⊢Δ ⊢F
        ρG = wk (lift ρ) (⊢Δ  ρF) ⊢G
        ρt⇒ = wkRedTerm ρ ⊢Δ t⇒
    in  fst-subst ρF ρG ρt⇒
  wkRedTerm ρ ⊢Δ (snd-subst {G = G} ⊢F ⊢G t⇒) =
    let ρF = wk ρ ⊢Δ ⊢F
        ρG = wk (lift ρ) (⊢Δ  ρF) ⊢G
        ρt⇒ = wkRedTerm ρ ⊢Δ t⇒
    in  PE.subst  x  _  snd _  snd _  x) (PE.sym (wk-β G))
      (snd-subst ρF ρG ρt⇒)
  wkRedTerm {ρ = ρ} [ρ] ⊢Δ (Σ-β₁ {G = G} ⊢F ⊢G t u ⊢tu) =
    let ρF = wk [ρ] ⊢Δ ⊢F
        ρG = wk (lift [ρ]) (⊢Δ  ρF) ⊢G
        ρt = wkTerm [ρ] ⊢Δ t
        ρu = wkTerm [ρ] ⊢Δ u
        ρu = PE.subst  x  _  _  x) (wk-β G) ρu
        ρtu = wkTerm [ρ] ⊢Δ ⊢tu
    in  Σ-β₁ ρF ρG ρt ρu ρtu
  wkRedTerm {ρ = ρ} [ρ] ⊢Δ (Σ-β₂ {G = G} ⊢F ⊢G t u ⊢tu) =
    let ρF = wk [ρ] ⊢Δ ⊢F
        ρG = wk (lift [ρ]) (⊢Δ  ρF) ⊢G
        ρt = wkTerm [ρ] ⊢Δ t
        ρu = wkTerm [ρ] ⊢Δ u
        ρu = PE.subst  x  _  _  x) (wk-β G) ρu
        ρtu = wkTerm [ρ] ⊢Δ ⊢tu
    in  PE.subst  x  _  _  _  x) (PE.sym (wk-β G))
      (Σ-β₂ ρF ρG ρt ρu ρtu)
  wkRedTerm {Δ = Δ} {ρ = ρ} [ρ] ⊢Δ (prodrec-subst {A = A} ⊢F ⊢G ⊢A ⊢u t⇒t′) =
    let ρF = wk [ρ] ⊢Δ ⊢F
        ρG = wk (lift [ρ]) (⊢Δ  ρF) ⊢G
        ρA = wk (lift [ρ]) (⊢Δ  (Σⱼ ρF  ρG)) ⊢A
        ρt⇒t′ = wkRedTerm [ρ] ⊢Δ t⇒t′
        ρu = wkTerm (lift (lift [ρ])) (⊢Δ  ρF  ρG) ⊢u
    in  PE.subst  x  Δ  prodrec _ _ _ _  _  x) (PE.sym (wk-β A))
                 (prodrec-subst ρF ρG ρA
                               (PE.subst  x  _  _  x)
                                         (wk-β-prodrec ρ A) ρu)
                               ρt⇒t′)
  wkRedTerm {Δ = Δ} {ρ = ρ} [ρ] ⊢Δ (prodrec-β {A = A} {G = G} {u = u} ⊢F ⊢G ⊢A ⊢t ⊢t′ ⊢u) =
    let ρF = wk [ρ] ⊢Δ ⊢F
        ρG = wk (lift [ρ]) (⊢Δ  ρF) ⊢G
        ρA = wk (lift [ρ]) (⊢Δ  (Σⱼ ρF  ρG)) ⊢A
        ρt = wkTerm [ρ] ⊢Δ ⊢t
        ρt′ = wkTerm [ρ] ⊢Δ ⊢t′
        ρu = wkTerm (lift (lift [ρ])) (⊢Δ  ρF  ρG) ⊢u
    in  PE.subst₂  x y  _  prodrec _ _ _ _  x  y)
                  (PE.trans (subst-wk u)
                    (PE.trans (substVar-to-subst (λ{x0  PE.refl;
                                                   (x0 +1)  PE.refl;
                                                   (x +1 +1)  PE.refl}) u)
                              (PE.sym (wk-subst u))))
                  (PE.sym (wk-β A))
                 (prodrec-β ρF ρG ρA ρt
                           (PE.subst  x  _  _  x) (wk-β G) ρt′)
                           (PE.subst  x  _  _  x) (wk-β-prodrec ρ A) ρu))
  wkRedTerm {Δ = Δ} {ρ = ρ} [ρ] ⊢Δ (natrec-subst {s = s} {F = F} ⊢F ⊢z ⊢s n⇒n′) =
    PE.subst  x  _  natrec _ _ _ _ _ _  _  x) (PE.sym (wk-β F))
             (natrec-subst (wk (lift [ρ]) (⊢Δ  ℕⱼ ⊢Δ) ⊢F)
                           (PE.subst  x  _  _  x) (wk-β F)
                                     (wkTerm [ρ] ⊢Δ ⊢z))
                                     (PE.subst  x  ((Δ  )  (U.wk (lift ρ) F))
                                                     U.wk (lift (lift ρ)) s  x)
                                               (wk-β-natrec _ F)
                                               (wkTerm (lift (lift [ρ])) ((⊢Δ  (ℕⱼ ⊢Δ)) 
                                                       (wk (lift [ρ]) (⊢Δ  (ℕⱼ ⊢Δ)) ⊢F)) ⊢s))
                           (wkRedTerm [ρ] ⊢Δ n⇒n′))
  wkRedTerm {Δ = Δ} {ρ = ρ} [ρ] ⊢Δ (natrec-zero {s = s} {F = F} ⊢F ⊢z ⊢s) =
    PE.subst  x  _  natrec _ _ (U.wk (lift ρ) F) _ _ _  _  x)
             (PE.sym (wk-β F))
             (natrec-zero (wk (lift [ρ]) (⊢Δ  ℕⱼ ⊢Δ) ⊢F)
                          (PE.subst  x  _  _  x)
                                    (wk-β F)
                                    (wkTerm [ρ] ⊢Δ ⊢z))
                          (PE.subst  x  ((Δ  )  (U.wk (lift ρ) F))
                                          U.wk (lift (lift ρ)) s  x)
                                    (wk-β-natrec ρ F)
                                    (wkTerm (lift (lift [ρ])) ((⊢Δ  (ℕⱼ ⊢Δ)) 
                                      (wk (lift [ρ]) (⊢Δ  ℕⱼ ⊢Δ) ⊢F)) ⊢s))
             )
  wkRedTerm {Δ = Δ} {ρ = ρ} [ρ] ⊢Δ (natrec-suc {p = p} {r = r} {n = n} {z = z} {s = s} {F = F} ⊢n ⊢F ⊢z ⊢s) =
    let ρn = wkTerm [ρ] ⊢Δ ⊢n
        ρF = wk (lift [ρ]) (⊢Δ  (ℕⱼ ⊢Δ)) ⊢F
        ρz = PE.subst  x  _  _  x) (wk-β F) (wkTerm [ρ] ⊢Δ ⊢z)
        ρs = U.wk ρ (s [ n , natrec p r F z s n ])
    in  PE.subst  x  _  natrec _ _ (U.wk (lift ρ) F) _ _ _  ρs  x)
             (PE.sym (wk-β F))
             (PE.subst  x  _  natrec _ _ _ _ _ _  x  _)
                       (PE.sym (wk-β-doubleSubst ρ s (natrec p r F z s n) n))
                       (natrec-suc ρn ρF ρz
                                   (PE.subst  x  ((Δ  )  (U.wk (lift ρ) F))  _  x)
                                             (wk-β-natrec _ F)
                                             (wkTerm (lift (lift [ρ]))
                                                     ((⊢Δ  (ℕⱼ ⊢Δ))  ρF)
                                                     ⊢s))))
  wkRedTerm {Δ = Δ} {ρ = ρ} [ρ] ⊢Δ (Emptyrec-subst {A = A} ⊢A n⇒n′) =
    (Emptyrec-subst (wk [ρ] ⊢Δ ⊢A)
                    (wkRedTerm [ρ] ⊢Δ n⇒n′))

wkRed* : ρ  Δ  Γ 
           let ρA = U.wk ρ A
               ρB = U.wk ρ B
           in  Δ  Γ  A ⇒* B  Δ  ρA ⇒* ρB
wkRed* ρ ⊢Δ (id A) = id (wk ρ ⊢Δ A)
wkRed* ρ ⊢Δ (A⇒A′  A′⇒*B) = wkRed ρ ⊢Δ A⇒A′  wkRed* ρ ⊢Δ A′⇒*B

wkRed*Term : ρ  Δ  Γ 
           let ρA = U.wk ρ A
               ρt = U.wk ρ t
               ρu = U.wk ρ u
           in  Δ  Γ  t ⇒* u  A  Δ  ρt ⇒* ρu  ρA
wkRed*Term ρ ⊢Δ (id t) = id (wkTerm ρ ⊢Δ t)
wkRed*Term ρ ⊢Δ (t⇒t′  t′⇒*u) = wkRedTerm ρ ⊢Δ t⇒t′  wkRed*Term ρ ⊢Δ t′⇒*u

wkRed:*: : ρ  Δ  Γ 
         let ρA = U.wk ρ A
             ρB = U.wk ρ B
         in  Δ  Γ  A :⇒*: B  Δ  ρA :⇒*: ρB
wkRed:*: ρ ⊢Δ [ ⊢A , ⊢B , D ] = [ wk ρ ⊢Δ ⊢A , wk ρ ⊢Δ ⊢B , wkRed* ρ ⊢Δ D ]

wkRed:*:Term : ρ  Δ  Γ 
             let ρA = U.wk ρ A
                 ρt = U.wk ρ t
                 ρu = U.wk ρ u
             in  Δ  Γ  t :⇒*: u  A  Δ  ρt :⇒*: ρu  ρA
wkRed:*:Term ρ ⊢Δ [ ⊢t , ⊢u , d ] =
  [ wkTerm ρ ⊢Δ ⊢t , wkTerm ρ ⊢Δ ⊢u , wkRed*Term ρ ⊢Δ d ]