2014-12-01 04:34:12 +00:00
|
|
|
|
/-
|
|
|
|
|
Copyright (c) 2014 Microsoft Corporation. All rights reserved.
|
|
|
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
|
|
|
|
Authors: Leonardo de Moura, Jeremy Avigad, Floris van Doorn
|
|
|
|
|
-/
|
|
|
|
|
prelude
|
|
|
|
|
import init.datatypes init.reserved_notation
|
|
|
|
|
|
2015-11-21 00:38:10 +00:00
|
|
|
|
definition id [reducible] [unfold_full] {A : Type} (a : A) : A :=
|
|
|
|
|
a
|
|
|
|
|
|
2014-12-12 23:22:19 +00:00
|
|
|
|
/- implication -/
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-05-07 19:28:47 +00:00
|
|
|
|
definition implies (a b : Prop) := a → b
|
|
|
|
|
|
|
|
|
|
lemma implies.trans [trans] {p q r : Prop} (h₁ : implies p q) (h₂ : implies q r) : implies p r :=
|
|
|
|
|
assume hp, h₂ (h₁ hp)
|
|
|
|
|
|
2014-12-01 04:34:12 +00:00
|
|
|
|
definition trivial := true.intro
|
|
|
|
|
|
|
|
|
|
definition not (a : Prop) := a → false
|
|
|
|
|
prefix `¬` := not
|
|
|
|
|
|
2015-07-29 04:56:35 +00:00
|
|
|
|
definition absurd {a : Prop} {b : Type} (H1 : a) (H2 : ¬a) : b :=
|
2014-12-01 04:34:12 +00:00
|
|
|
|
false.rec b (H2 H1)
|
|
|
|
|
|
2015-08-13 01:37:33 +00:00
|
|
|
|
theorem mt {a b : Prop} (H1 : a → b) (H2 : ¬b) : ¬a :=
|
|
|
|
|
assume Ha : a, absurd (H1 Ha) H2
|
|
|
|
|
|
2014-12-12 23:22:19 +00:00
|
|
|
|
/- not -/
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-05-09 19:15:30 +00:00
|
|
|
|
theorem not_false : ¬false :=
|
2014-12-01 04:34:12 +00:00
|
|
|
|
assume H : false, H
|
|
|
|
|
|
2015-04-07 14:46:11 +00:00
|
|
|
|
definition non_contradictory (a : Prop) : Prop := ¬¬a
|
|
|
|
|
|
|
|
|
|
theorem non_contradictory_intro {a : Prop} (Ha : a) : ¬¬a :=
|
|
|
|
|
assume Hna : ¬a, absurd Ha Hna
|
|
|
|
|
|
2015-08-13 01:37:33 +00:00
|
|
|
|
/- false -/
|
|
|
|
|
|
|
|
|
|
theorem false.elim {c : Prop} (H : false) : c :=
|
|
|
|
|
false.rec c H
|
|
|
|
|
|
|
|
|
|
|
2014-12-12 23:22:19 +00:00
|
|
|
|
/- eq -/
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
|
|
|
|
notation a = b := eq a b
|
2015-06-16 20:01:24 +00:00
|
|
|
|
definition rfl {A : Type} {a : A} : a = a := eq.refl a
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
|
|
|
|
-- proof irrelevance is built in
|
|
|
|
|
theorem proof_irrel {a : Prop} (H₁ H₂ : a) : H₁ = H₂ :=
|
|
|
|
|
rfl
|
|
|
|
|
|
2015-06-04 00:16:10 +00:00
|
|
|
|
-- Remark: we provide the universe levels explicitly to make sure `eq.drec` has the same type of `eq.rec` in the HoTT library
|
|
|
|
|
protected theorem eq.drec.{l₁ l₂} {A : Type.{l₂}} {a : A} {C : Π (x : A), a = x → Type.{l₁}} (h₁ : C a (eq.refl a)) {b : A} (h₂ : a = b) : C b h₂ :=
|
|
|
|
|
eq.rec (λh₂ : a = a, show C a h₂, from h₁) h₂ h₂
|
|
|
|
|
|
2014-12-01 04:34:12 +00:00
|
|
|
|
namespace eq
|
|
|
|
|
variables {A : Type}
|
|
|
|
|
variables {a b c a': A}
|
|
|
|
|
|
2015-06-04 00:16:10 +00:00
|
|
|
|
protected theorem drec_on {a : A} {C : Π (x : A), a = x → Type} {b : A} (h₂ : a = b) (h₁ : C a (refl a)) : C b h₂ :=
|
|
|
|
|
eq.drec h₁ h₂
|
|
|
|
|
|
2014-12-01 04:34:12 +00:00
|
|
|
|
theorem subst {P : A → Prop} (H₁ : a = b) (H₂ : P a) : P b :=
|
2015-02-11 20:49:27 +00:00
|
|
|
|
eq.rec H₂ H₁
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
|
|
|
|
theorem trans (H₁ : a = b) (H₂ : b = c) : a = c :=
|
|
|
|
|
subst H₂ H₁
|
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem symm : a = b → b = a :=
|
|
|
|
|
eq.rec (refl a)
|
|
|
|
|
|
|
|
|
|
theorem substr {P : A → Prop} (H₁ : b = a) : P a → P b :=
|
|
|
|
|
subst (symm H₁)
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-11-14 23:40:47 +00:00
|
|
|
|
theorem mp {a b : Type} : (a = b) → a → b :=
|
|
|
|
|
eq.rec_on
|
|
|
|
|
|
|
|
|
|
theorem mpr {a b : Type} : (a = b) → b → a :=
|
|
|
|
|
assume H₁ H₂, eq.rec_on (eq.symm H₁) H₂
|
|
|
|
|
|
2014-12-01 04:34:12 +00:00
|
|
|
|
namespace ops
|
|
|
|
|
notation H `⁻¹` := symm H --input with \sy or \-1 or \inv
|
|
|
|
|
notation H1 ⬝ H2 := trans H1 H2
|
|
|
|
|
notation H1 ▸ H2 := subst H1 H2
|
2015-07-20 21:56:50 +00:00
|
|
|
|
notation H1 ▹ H2 := eq.rec H2 H1
|
2014-12-01 04:34:12 +00:00
|
|
|
|
end ops
|
2014-12-12 21:50:53 +00:00
|
|
|
|
end eq
|
|
|
|
|
|
2015-05-02 18:29:31 +00:00
|
|
|
|
theorem congr {A B : Type} {f₁ f₂ : A → B} {a₁ a₂ : A} (H₁ : f₁ = f₂) (H₂ : a₁ = a₂) : f₁ a₁ = f₂ a₂ :=
|
|
|
|
|
eq.subst H₁ (eq.subst H₂ rfl)
|
|
|
|
|
|
2015-07-06 14:29:56 +00:00
|
|
|
|
theorem congr_fun {A : Type} {B : A → Type} {f g : Π x, B x} (H : f = g) (a : A) : f a = g a :=
|
|
|
|
|
eq.subst H (eq.refl (f a))
|
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem congr_arg {A B : Type} {a₁ a₂ : A} (f : A → B) : a₁ = a₂ → f a₁ = f a₂ :=
|
|
|
|
|
congr rfl
|
2015-07-06 14:29:56 +00:00
|
|
|
|
|
2015-04-04 15:58:35 +00:00
|
|
|
|
section
|
|
|
|
|
variables {A : Type} {a b c: A}
|
|
|
|
|
open eq.ops
|
|
|
|
|
|
2015-05-09 19:15:30 +00:00
|
|
|
|
theorem trans_rel_left (R : A → A → Prop) (H₁ : R a b) (H₂ : b = c) : R a c :=
|
2015-04-04 15:58:35 +00:00
|
|
|
|
H₂ ▸ H₁
|
|
|
|
|
|
2015-05-09 19:15:30 +00:00
|
|
|
|
theorem trans_rel_right (R : A → A → Prop) (H₁ : a = b) (H₂ : R b c) : R a c :=
|
2015-04-04 15:58:35 +00:00
|
|
|
|
H₁⁻¹ ▸ H₂
|
|
|
|
|
end
|
|
|
|
|
|
2014-12-12 21:50:53 +00:00
|
|
|
|
section
|
2014-12-01 04:34:12 +00:00
|
|
|
|
variable {p : Prop}
|
2014-12-12 21:50:53 +00:00
|
|
|
|
open eq.ops
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2014-12-12 21:50:53 +00:00
|
|
|
|
theorem of_eq_true (H : p = true) : p :=
|
2014-12-01 04:34:12 +00:00
|
|
|
|
H⁻¹ ▸ trivial
|
|
|
|
|
|
2014-12-12 21:50:53 +00:00
|
|
|
|
theorem not_of_eq_false (H : p = false) : ¬p :=
|
2014-12-01 04:34:12 +00:00
|
|
|
|
assume Hp, H ▸ Hp
|
2014-12-12 21:50:53 +00:00
|
|
|
|
end
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-05-02 22:15:35 +00:00
|
|
|
|
attribute eq.subst [subst]
|
|
|
|
|
attribute eq.refl [refl]
|
|
|
|
|
attribute eq.trans [trans]
|
|
|
|
|
attribute eq.symm [symm]
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2014-12-12 23:22:19 +00:00
|
|
|
|
/- ne -/
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-11-15 23:08:20 +00:00
|
|
|
|
definition ne [reducible] {A : Type} (a b : A) := ¬(a = b)
|
2014-12-01 04:34:12 +00:00
|
|
|
|
notation a ≠ b := ne a b
|
|
|
|
|
|
|
|
|
|
namespace ne
|
|
|
|
|
open eq.ops
|
|
|
|
|
variable {A : Type}
|
|
|
|
|
variables {a b : A}
|
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem intro (H : a = b → false) : a ≠ b := H
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem elim (H : a ≠ b) : a = b → false := H
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem irrefl (H : a ≠ a) : false := H rfl
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem symm (H : a ≠ b) : b ≠ a :=
|
|
|
|
|
assume (H₁ : b = a), H (H₁⁻¹)
|
2014-12-01 04:34:12 +00:00
|
|
|
|
end ne
|
|
|
|
|
|
2015-11-16 18:26:09 +00:00
|
|
|
|
theorem false_of_ne {A : Type} {a : A} : a ≠ a → false := ne.irrefl
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-08-13 01:37:33 +00:00
|
|
|
|
section
|
|
|
|
|
open eq.ops
|
|
|
|
|
variables {p : Prop}
|
|
|
|
|
|
|
|
|
|
theorem ne_false_of_self : p → p ≠ false :=
|
|
|
|
|
assume (Hp : p) (Heq : p = false), Heq ▸ Hp
|
|
|
|
|
|
|
|
|
|
theorem ne_true_of_not : ¬p → p ≠ true :=
|
|
|
|
|
assume (Hnp : ¬p) (Heq : p = true), (Heq ▸ Hnp) trivial
|
|
|
|
|
|
|
|
|
|
theorem true_ne_false : ¬true = false :=
|
|
|
|
|
ne_false_of_self trivial
|
|
|
|
|
end
|
|
|
|
|
|
2015-09-30 15:06:31 +00:00
|
|
|
|
infixl ` == `:50 := heq
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
|
|
|
|
namespace heq
|
|
|
|
|
universe variable u
|
|
|
|
|
variables {A B C : Type.{u}} {a a' : A} {b b' : B} {c : C}
|
|
|
|
|
|
2015-05-09 19:15:30 +00:00
|
|
|
|
theorem to_eq (H : a == a') : a = a' :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
have H₁ : ∀ (Ht : A = A), eq.rec a Ht = a, from
|
|
|
|
|
λ Ht, eq.refl a,
|
|
|
|
|
heq.rec H₁ H (eq.refl A)
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem elim {A : Type} {a : A} {P : A → Type} {b : A} (H₁ : a == b)
|
|
|
|
|
: P a → P b := eq.rec_on (to_eq H₁)
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem subst {P : ∀T : Type, T → Prop} : a == b → P A a → P B b :=
|
|
|
|
|
heq.rec_on
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
|
|
|
|
theorem symm (H : a == b) : b == a :=
|
2015-02-11 20:49:27 +00:00
|
|
|
|
heq.rec_on H (refl a)
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2014-12-12 21:20:27 +00:00
|
|
|
|
theorem of_eq (H : a = a') : a == a' :=
|
2014-12-01 04:34:12 +00:00
|
|
|
|
eq.subst H (refl a)
|
|
|
|
|
|
|
|
|
|
theorem trans (H₁ : a == b) (H₂ : b == c) : a == c :=
|
|
|
|
|
subst H₂ H₁
|
|
|
|
|
|
2014-12-12 21:20:27 +00:00
|
|
|
|
theorem of_heq_of_eq (H₁ : a == b) (H₂ : b = b') : a == b' :=
|
|
|
|
|
trans H₁ (of_eq H₂)
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2014-12-12 21:20:27 +00:00
|
|
|
|
theorem of_eq_of_heq (H₁ : a = a') (H₂ : a' == b) : a == b :=
|
|
|
|
|
trans (of_eq H₁) H₂
|
2015-05-09 16:49:41 +00:00
|
|
|
|
|
2015-07-20 21:56:50 +00:00
|
|
|
|
definition type_eq (H : a == b) : A = B :=
|
|
|
|
|
heq.rec_on H (eq.refl A)
|
2014-12-01 04:34:12 +00:00
|
|
|
|
end heq
|
|
|
|
|
|
2015-07-20 21:56:50 +00:00
|
|
|
|
open eq.ops
|
|
|
|
|
theorem eq_rec_heq {A : Type} {P : A → Type} {a a' : A} (H : a = a') (p : P a) : H ▹ p == p :=
|
2015-05-09 16:49:41 +00:00
|
|
|
|
eq.drec_on H !heq.refl
|
|
|
|
|
|
2015-07-20 21:56:50 +00:00
|
|
|
|
theorem heq_of_eq_rec_left {A : Type} {P : A → Type} : ∀ {a a' : A} {p₁ : P a} {p₂ : P a'} (e : a = a') (h₂ : e ▹ p₁ = p₂), p₁ == p₂
|
|
|
|
|
| a a p₁ p₂ (eq.refl a) h := eq.rec_on h !heq.refl
|
|
|
|
|
|
|
|
|
|
theorem heq_of_eq_rec_right {A : Type} {P : A → Type} : ∀ {a a' : A} {p₁ : P a} {p₂ : P a'} (e : a' = a) (h₂ : p₁ = e ▹ p₂), p₁ == p₂
|
|
|
|
|
| a a p₁ p₂ (eq.refl a) h := eq.rec_on h !heq.refl
|
|
|
|
|
|
2014-12-12 21:50:53 +00:00
|
|
|
|
theorem of_heq_true {a : Prop} (H : a == true) : a :=
|
|
|
|
|
of_eq_true (heq.to_eq H)
|
|
|
|
|
|
2015-07-20 21:56:50 +00:00
|
|
|
|
theorem eq_rec_compose : ∀ {A B C : Type} (p₁ : B = C) (p₂ : A = B) (a : A), p₁ ▹ (p₂ ▹ a : B) = (p₂ ⬝ p₁) ▹ a
|
|
|
|
|
| A A A (eq.refl A) (eq.refl A) a := calc
|
|
|
|
|
eq.refl A ▹ eq.refl A ▹ a = eq.refl A ▹ a : rfl
|
|
|
|
|
... = (eq.refl A ⬝ eq.refl A) ▹ a : {proof_irrel (eq.refl A) (eq.refl A ⬝ eq.refl A)}
|
|
|
|
|
|
2015-07-29 22:00:50 +00:00
|
|
|
|
theorem eq_rec_eq_eq_rec {A₁ A₂ : Type} {p : A₁ = A₂} : ∀ {a₁ : A₁} {a₂ : A₂}, p ▹ a₁ = a₂ → a₁ = p⁻¹ ▹ a₂ :=
|
|
|
|
|
eq.drec_on p (λ a₁ a₂ h, eq.drec_on h rfl)
|
|
|
|
|
|
|
|
|
|
theorem eq_rec_of_heq_left : ∀ {A₁ A₂ : Type} {a₁ : A₁} {a₂ : A₂} (h : a₁ == a₂), heq.type_eq h ▹ a₁ = a₂
|
|
|
|
|
| A A a a (heq.refl a) := rfl
|
|
|
|
|
|
|
|
|
|
theorem eq_rec_of_heq_right {A₁ A₂ : Type} {a₁ : A₁} {a₂ : A₂} (h : a₁ == a₂) : a₁ = (heq.type_eq h)⁻¹ ▹ a₂ :=
|
|
|
|
|
eq_rec_eq_eq_rec (eq_rec_of_heq_left h)
|
|
|
|
|
|
2015-05-25 23:48:33 +00:00
|
|
|
|
attribute heq.refl [refl]
|
2015-05-02 22:15:35 +00:00
|
|
|
|
attribute heq.trans [trans]
|
|
|
|
|
attribute heq.of_heq_of_eq [trans]
|
|
|
|
|
attribute heq.of_eq_of_heq [trans]
|
|
|
|
|
attribute heq.symm [symm]
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2014-12-12 23:22:19 +00:00
|
|
|
|
/- and -/
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
|
|
|
|
notation a /\ b := and a b
|
|
|
|
|
notation a ∧ b := and a b
|
|
|
|
|
|
|
|
|
|
variables {a b c d : Prop}
|
|
|
|
|
|
2014-12-12 21:50:53 +00:00
|
|
|
|
theorem and.elim (H₁ : a ∧ b) (H₂ : a → b → c) : c :=
|
|
|
|
|
and.rec H₂ H₁
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-11-21 00:38:10 +00:00
|
|
|
|
theorem and.swap : a ∧ b → b ∧ a :=
|
|
|
|
|
and.rec (λHa Hb, and.intro Hb Ha)
|
|
|
|
|
|
2014-12-12 23:22:19 +00:00
|
|
|
|
/- or -/
|
|
|
|
|
|
2015-09-30 15:06:31 +00:00
|
|
|
|
notation a \/ b := or a b
|
2014-12-01 04:34:12 +00:00
|
|
|
|
notation a ∨ b := or a b
|
|
|
|
|
|
|
|
|
|
namespace or
|
2015-05-09 19:15:30 +00:00
|
|
|
|
theorem elim (H₁ : a ∨ b) (H₂ : a → c) (H₃ : b → c) : c :=
|
2015-02-11 20:49:27 +00:00
|
|
|
|
or.rec H₂ H₃ H₁
|
2014-12-01 04:34:12 +00:00
|
|
|
|
end or
|
|
|
|
|
|
2015-04-07 14:46:11 +00:00
|
|
|
|
theorem non_contradictory_em (a : Prop) : ¬¬(a ∨ ¬a) :=
|
|
|
|
|
assume not_em : ¬(a ∨ ¬a),
|
|
|
|
|
have neg_a : ¬a, from
|
|
|
|
|
assume pos_a : a, absurd (or.inl pos_a) not_em,
|
|
|
|
|
absurd (or.inr neg_a) not_em
|
|
|
|
|
|
2015-11-21 00:38:10 +00:00
|
|
|
|
theorem or.swap : a ∨ b → b ∨ a := or.rec or.inr or.inl
|
|
|
|
|
|
2014-12-12 23:22:19 +00:00
|
|
|
|
/- iff -/
|
|
|
|
|
|
2014-12-01 04:34:12 +00:00
|
|
|
|
definition iff (a b : Prop) := (a → b) ∧ (b → a)
|
|
|
|
|
|
|
|
|
|
notation a <-> b := iff a b
|
|
|
|
|
notation a ↔ b := iff a b
|
|
|
|
|
|
|
|
|
|
namespace iff
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem intro : (a → b) → (b → a) → (a ↔ b) := and.intro
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem elim : ((a → b) → (b → a) → c) → (a ↔ b) → c := and.rec
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem elim_left : (a ↔ b) → a → b := and.left
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
|
|
|
|
definition mp := @elim_left
|
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem elim_right : (a ↔ b) → b → a := and.right
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-07-18 09:28:53 +00:00
|
|
|
|
definition mpr := @elim_right
|
2014-12-12 23:22:19 +00:00
|
|
|
|
|
2015-05-09 19:15:30 +00:00
|
|
|
|
theorem refl (a : Prop) : a ↔ a :=
|
2014-12-01 04:34:12 +00:00
|
|
|
|
intro (assume H, H) (assume H, H)
|
|
|
|
|
|
2015-05-09 19:15:30 +00:00
|
|
|
|
theorem rfl {a : Prop} : a ↔ a :=
|
2014-12-01 04:34:12 +00:00
|
|
|
|
refl a
|
|
|
|
|
|
|
|
|
|
theorem trans (H₁ : a ↔ b) (H₂ : b ↔ c) : a ↔ c :=
|
|
|
|
|
intro
|
2015-07-24 15:56:18 +00:00
|
|
|
|
(assume Ha, mp H₂ (mp H₁ Ha))
|
|
|
|
|
(assume Hc, mpr H₁ (mpr H₂ Hc))
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
|
|
|
|
theorem symm (H : a ↔ b) : b ↔ a :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
intro (elim_right H) (elim_left H)
|
|
|
|
|
|
|
|
|
|
theorem comm : (a ↔ b) ↔ (b ↔ a) :=
|
|
|
|
|
intro symm symm
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
|
|
|
|
open eq.ops
|
|
|
|
|
theorem of_eq {a b : Prop} (H : a = b) : a ↔ b :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
H ▸ rfl
|
2014-12-01 04:34:12 +00:00
|
|
|
|
end iff
|
|
|
|
|
|
2015-05-09 19:15:30 +00:00
|
|
|
|
theorem not_iff_not_of_iff (H₁ : a ↔ b) : ¬a ↔ ¬b :=
|
2014-12-16 03:17:51 +00:00
|
|
|
|
iff.intro
|
2015-07-24 15:56:18 +00:00
|
|
|
|
(assume (Hna : ¬ a) (Hb : b), Hna (iff.elim_right H₁ Hb))
|
|
|
|
|
(assume (Hnb : ¬ b) (Ha : a), Hnb (iff.elim_left H₁ Ha))
|
2014-12-16 03:17:51 +00:00
|
|
|
|
|
2014-12-12 21:50:53 +00:00
|
|
|
|
theorem of_iff_true (H : a ↔ true) : a :=
|
|
|
|
|
iff.mp (iff.symm H) trivial
|
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem not_of_iff_false : (a ↔ false) → ¬a := iff.mp
|
2014-12-12 21:50:53 +00:00
|
|
|
|
|
2015-04-07 14:46:11 +00:00
|
|
|
|
theorem iff_true_intro (H : a) : a ↔ true :=
|
|
|
|
|
iff.intro
|
|
|
|
|
(λ Hl, trivial)
|
|
|
|
|
(λ Hr, H)
|
|
|
|
|
|
|
|
|
|
theorem iff_false_intro (H : ¬a) : a ↔ false :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
iff.intro H !false.rec
|
2015-04-07 14:46:11 +00:00
|
|
|
|
|
|
|
|
|
theorem not_non_contradictory_iff_absurd (a : Prop) : ¬¬¬a ↔ ¬a :=
|
|
|
|
|
iff.intro
|
2015-07-24 15:56:18 +00:00
|
|
|
|
(λ (Hl : ¬¬¬a) (Ha : a), Hl (non_contradictory_intro Ha))
|
|
|
|
|
absurd
|
2015-04-07 14:46:11 +00:00
|
|
|
|
|
2015-05-02 22:15:35 +00:00
|
|
|
|
attribute iff.refl [refl]
|
|
|
|
|
attribute iff.symm [symm]
|
|
|
|
|
attribute iff.trans [trans]
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-11-21 00:38:10 +00:00
|
|
|
|
theorem imp_congr [congr] (H1 : a ↔ c) (H2 : b ↔ d) : (a → b) ↔ (c → d) :=
|
|
|
|
|
iff.intro
|
|
|
|
|
(λHab Hc, iff.mp H2 (Hab (iff.mpr H1 Hc)))
|
|
|
|
|
(λHcd Ha, iff.mpr H2 (Hcd (iff.mp H1 Ha)))
|
|
|
|
|
|
2015-11-16 18:26:09 +00:00
|
|
|
|
theorem not_not_intro (Ha : a) : ¬¬a :=
|
|
|
|
|
assume Hna : ¬a, Hna Ha
|
|
|
|
|
|
2015-11-21 00:38:10 +00:00
|
|
|
|
theorem not_true [simp] : (¬ true) ↔ false :=
|
2015-11-16 18:26:09 +00:00
|
|
|
|
iff_false_intro (not_not_intro trivial)
|
|
|
|
|
|
2015-11-21 00:38:10 +00:00
|
|
|
|
theorem not_false_iff [simp] : (¬ false) ↔ true :=
|
2015-11-16 18:26:09 +00:00
|
|
|
|
iff_true_intro not_false
|
|
|
|
|
|
2015-11-21 00:38:10 +00:00
|
|
|
|
theorem not_congr [congr] (H : a ↔ b) : ¬a ↔ ¬b :=
|
|
|
|
|
iff.intro (λ H₁ H₂, H₁ (iff.mpr H H₂)) (λ H₁ H₂, H₁ (iff.mp H H₂))
|
|
|
|
|
|
2015-11-16 18:26:09 +00:00
|
|
|
|
theorem ne_self_iff_false {A : Type} (a : A) : (a ≠ a) ↔ false :=
|
|
|
|
|
iff.intro false_of_ne false.elim
|
|
|
|
|
|
2015-11-21 00:38:10 +00:00
|
|
|
|
theorem eq_self_iff_true [simp] {A : Type} (a : A) : (a = a) ↔ true :=
|
2015-11-16 18:26:09 +00:00
|
|
|
|
iff_true_intro rfl
|
|
|
|
|
|
2015-11-21 00:38:10 +00:00
|
|
|
|
theorem heq_self_iff_true [simp] {A : Type} (a : A) : (a == a) ↔ true :=
|
2015-11-16 18:26:09 +00:00
|
|
|
|
iff_true_intro (heq.refl a)
|
|
|
|
|
|
2015-11-21 00:38:10 +00:00
|
|
|
|
theorem iff_not_self [simp] (a : Prop) : (a ↔ ¬a) ↔ false :=
|
2015-11-16 18:26:09 +00:00
|
|
|
|
iff_false_intro (λ H,
|
|
|
|
|
have H' : ¬a, from (λ Ha, (iff.mp H Ha) Ha),
|
|
|
|
|
H' (iff.mpr H H'))
|
|
|
|
|
|
2015-11-21 00:38:10 +00:00
|
|
|
|
theorem true_iff_false [simp] : (true ↔ false) ↔ false :=
|
2015-11-16 18:26:09 +00:00
|
|
|
|
iff_false_intro (λ H, iff.mp H trivial)
|
|
|
|
|
|
2015-11-21 00:38:10 +00:00
|
|
|
|
theorem false_iff_true [simp] : (false ↔ true) ↔ false :=
|
2015-11-16 18:26:09 +00:00
|
|
|
|
iff_false_intro (λ H, iff.mpr H trivial)
|
|
|
|
|
|
2015-11-18 02:44:11 +00:00
|
|
|
|
theorem false_of_true_iff_false : (true ↔ false) → false :=
|
|
|
|
|
assume H, iff.mp H trivial
|
|
|
|
|
|
2015-11-21 00:38:10 +00:00
|
|
|
|
/- and simp rules -/
|
|
|
|
|
theorem and.imp (H₂ : a → c) (H₃ : b → d) : a ∧ b → c ∧ d :=
|
|
|
|
|
and.rec (λHa Hb, and.intro (H₂ Ha) (H₃ Hb))
|
|
|
|
|
|
|
|
|
|
theorem and_congr [congr] (H1 : a ↔ c) (H2 : b ↔ d) : (a ∧ b) ↔ (c ∧ d) :=
|
|
|
|
|
iff.intro (and.imp (iff.mp H1) (iff.mp H2)) (and.imp (iff.mpr H1) (iff.mpr H2))
|
|
|
|
|
|
|
|
|
|
theorem and.comm [simp] : a ∧ b ↔ b ∧ a :=
|
|
|
|
|
iff.intro and.swap and.swap
|
|
|
|
|
|
|
|
|
|
theorem and.assoc [simp] : (a ∧ b) ∧ c ↔ a ∧ (b ∧ c) :=
|
|
|
|
|
iff.intro
|
|
|
|
|
(and.rec (λ H' Hc, and.rec (λ Ha Hb, and.intro Ha (and.intro Hb Hc)) H'))
|
|
|
|
|
(and.rec (λ Ha, and.rec (λ Hb Hc, and.intro (and.intro Ha Hb) Hc)))
|
|
|
|
|
|
|
|
|
|
theorem and.left_comm [simp] : a ∧ (b ∧ c) ↔ b ∧ (a ∧ c) :=
|
|
|
|
|
iff.trans (iff.symm !and.assoc) (iff.trans (and_congr !and.comm !iff.refl) !and.assoc)
|
|
|
|
|
|
|
|
|
|
theorem and_iff_left {a b : Prop} (Hb : b) : (a ∧ b) ↔ a :=
|
|
|
|
|
iff.intro and.left (λHa, and.intro Ha Hb)
|
|
|
|
|
|
|
|
|
|
theorem and_iff_right {a b : Prop} (Ha : a) : (a ∧ b) ↔ b :=
|
|
|
|
|
iff.intro and.right (and.intro Ha)
|
|
|
|
|
|
|
|
|
|
theorem and_true [simp] (a : Prop) : a ∧ true ↔ a :=
|
|
|
|
|
and_iff_left trivial
|
|
|
|
|
|
|
|
|
|
theorem true_and [simp] (a : Prop) : true ∧ a ↔ a :=
|
|
|
|
|
and_iff_right trivial
|
|
|
|
|
|
|
|
|
|
theorem and_false [simp] (a : Prop) : a ∧ false ↔ false :=
|
|
|
|
|
iff_false_intro and.right
|
|
|
|
|
|
|
|
|
|
theorem false_and [simp] (a : Prop) : false ∧ a ↔ false :=
|
|
|
|
|
iff_false_intro and.left
|
|
|
|
|
|
|
|
|
|
theorem and_self [simp] (a : Prop) : a ∧ a ↔ a :=
|
|
|
|
|
iff.intro and.left (assume H, and.intro H H)
|
|
|
|
|
|
|
|
|
|
/- or simp rules -/
|
|
|
|
|
|
|
|
|
|
theorem or.imp (H₂ : a → c) (H₃ : b → d) : a ∨ b → c ∨ d :=
|
|
|
|
|
or.rec (λ H, or.inl (H₂ H)) (λ H, or.inr (H₃ H))
|
|
|
|
|
|
|
|
|
|
theorem or.imp_left (H : a → b) : a ∨ c → b ∨ c :=
|
|
|
|
|
or.imp H id
|
|
|
|
|
|
|
|
|
|
theorem or.imp_right (H : a → b) : c ∨ a → c ∨ b :=
|
|
|
|
|
or.imp id H
|
|
|
|
|
|
|
|
|
|
theorem or_congr [congr] (H1 : a ↔ c) (H2 : b ↔ d) : (a ∨ b) ↔ (c ∨ d) :=
|
|
|
|
|
iff.intro (or.imp (iff.mp H1) (iff.mp H2)) (or.imp (iff.mpr H1) (iff.mpr H2))
|
|
|
|
|
|
|
|
|
|
theorem or.comm [simp] : a ∨ b ↔ b ∨ a := iff.intro or.swap or.swap
|
|
|
|
|
|
|
|
|
|
theorem or.assoc [simp] : (a ∨ b) ∨ c ↔ a ∨ (b ∨ c) :=
|
|
|
|
|
iff.intro
|
|
|
|
|
(or.rec (or.imp_right or.inl) (λ H, or.inr (or.inr H)))
|
|
|
|
|
(or.rec (λ H, or.inl (or.inl H)) (or.imp_left or.inr))
|
|
|
|
|
|
|
|
|
|
theorem or.left_comm [simp] : a ∨ (b ∨ c) ↔ b ∨ (a ∨ c) :=
|
|
|
|
|
iff.trans (iff.symm !or.assoc) (iff.trans (or_congr !or.comm !iff.refl) !or.assoc)
|
|
|
|
|
|
|
|
|
|
theorem or_true [simp] (a : Prop) : a ∨ true ↔ true :=
|
|
|
|
|
iff_true_intro (or.inr trivial)
|
|
|
|
|
|
|
|
|
|
theorem true_or [simp] (a : Prop) : true ∨ a ↔ true :=
|
|
|
|
|
iff_true_intro (or.inl trivial)
|
|
|
|
|
|
|
|
|
|
theorem or_false [simp] (a : Prop) : a ∨ false ↔ a :=
|
|
|
|
|
iff.intro (or.rec id false.elim) or.inl
|
|
|
|
|
|
|
|
|
|
theorem false_or [simp] (a : Prop) : false ∨ a ↔ a :=
|
|
|
|
|
iff.trans or.comm !or_false
|
|
|
|
|
|
|
|
|
|
theorem or_self [simp] (a : Prop) : a ∨ a ↔ a :=
|
|
|
|
|
iff.intro (or.rec id id) or.inl
|
|
|
|
|
|
|
|
|
|
/- iff simp rules -/
|
|
|
|
|
|
|
|
|
|
theorem iff_true [simp] (a : Prop) : (a ↔ true) ↔ a :=
|
|
|
|
|
iff.intro (assume H, iff.mpr H trivial) iff_true_intro
|
|
|
|
|
|
|
|
|
|
theorem true_iff [simp] (a : Prop) : (true ↔ a) ↔ a :=
|
|
|
|
|
iff.trans iff.comm !iff_true
|
|
|
|
|
|
|
|
|
|
theorem iff_false [simp] (a : Prop) : (a ↔ false) ↔ ¬ a :=
|
|
|
|
|
iff.intro and.left iff_false_intro
|
|
|
|
|
|
|
|
|
|
theorem false_iff [simp] (a : Prop) : (false ↔ a) ↔ ¬ a :=
|
|
|
|
|
iff.trans iff.comm !iff_false
|
|
|
|
|
|
|
|
|
|
theorem iff_self [simp] (a : Prop) : (a ↔ a) ↔ true :=
|
|
|
|
|
iff_true_intro iff.rfl
|
|
|
|
|
|
|
|
|
|
theorem iff_congr [congr] (H1 : a ↔ c) (H2 : b ↔ d) : (a ↔ b) ↔ (c ↔ d) :=
|
|
|
|
|
and_congr (imp_congr H1 H2) (imp_congr H2 H1)
|
|
|
|
|
|
|
|
|
|
/- exists -/
|
|
|
|
|
|
2014-12-01 04:34:12 +00:00
|
|
|
|
inductive Exists {A : Type} (P : A → Prop) : Prop :=
|
|
|
|
|
intro : ∀ (a : A), P a → Exists P
|
|
|
|
|
|
2014-12-16 03:05:03 +00:00
|
|
|
|
definition exists.intro := @Exists.intro
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-09-30 15:06:31 +00:00
|
|
|
|
notation `exists` binders `, ` r:(scoped P, Exists P) := r
|
|
|
|
|
notation `∃` binders `, ` r:(scoped P, Exists P) := r
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem exists.elim {A : Type} {p : A → Prop} {B : Prop}
|
|
|
|
|
(H1 : ∃x, p x) (H2 : ∀ (a : A), p a → B) : B :=
|
2014-12-01 04:34:12 +00:00
|
|
|
|
Exists.rec H2 H1
|
|
|
|
|
|
2015-11-21 00:38:10 +00:00
|
|
|
|
/- exists unique -/
|
|
|
|
|
|
|
|
|
|
definition exists_unique {A : Type} (p : A → Prop) :=
|
|
|
|
|
∃x, p x ∧ ∀y, p y → y = x
|
|
|
|
|
|
|
|
|
|
notation `∃!` binders `, ` r:(scoped P, exists_unique P) := r
|
|
|
|
|
|
|
|
|
|
theorem exists_unique.intro {A : Type} {p : A → Prop} (w : A) (H1 : p w) (H2 : ∀y, p y → y = w) :
|
|
|
|
|
∃!x, p x :=
|
|
|
|
|
exists.intro w (and.intro H1 H2)
|
|
|
|
|
|
|
|
|
|
theorem exists_unique.elim {A : Type} {p : A → Prop} {b : Prop}
|
|
|
|
|
(H2 : ∃!x, p x) (H1 : ∀x, p x → (∀y, p y → y = x) → b) : b :=
|
|
|
|
|
exists.elim H2 (λ w Hw, H1 w (and.left Hw) (and.right Hw))
|
|
|
|
|
|
|
|
|
|
theorem exists_unique_of_exists_of_unique {A : Type} {p : A → Prop}
|
|
|
|
|
(Hex : ∃ x, p x) (Hunique : ∀ y₁ y₂, p y₁ → p y₂ → y₁ = y₂) : ∃! x, p x :=
|
|
|
|
|
exists.elim Hex (λ x px, exists_unique.intro x px (take y, suppose p y, Hunique y x this px))
|
|
|
|
|
|
|
|
|
|
theorem exists_of_exists_unique {A : Type} {p : A → Prop} (H : ∃! x, p x) : ∃ x, p x :=
|
|
|
|
|
exists.elim H (λ x Hx, exists.intro x (and.left Hx))
|
|
|
|
|
|
|
|
|
|
theorem unique_of_exists_unique {A : Type} {p : A → Prop}
|
|
|
|
|
(H : ∃! x, p x) {y₁ y₂ : A} (py₁ : p y₁) (py₂ : p y₂) : y₁ = y₂ :=
|
|
|
|
|
exists_unique.elim H
|
|
|
|
|
(take x, suppose p x,
|
|
|
|
|
assume unique : ∀ y, p y → y = x,
|
|
|
|
|
show y₁ = y₂, from eq.trans (unique _ py₁) (eq.symm (unique _ py₂)))
|
|
|
|
|
|
|
|
|
|
/- exists, forall, exists unique congruences -/
|
|
|
|
|
section
|
|
|
|
|
variables {A : Type} {p₁ p₂ : A → Prop}
|
|
|
|
|
|
|
|
|
|
theorem forall_congr [congr] {A : Type} {P Q : A → Prop} (H : ∀a, (P a ↔ Q a)) : (∀a, P a) ↔ ∀a, Q a :=
|
|
|
|
|
iff.intro (λp a, iff.mp (H a) (p a)) (λq a, iff.mpr (H a) (q a))
|
|
|
|
|
|
|
|
|
|
theorem exists_imp_exists {A : Type} {P Q : A → Prop} (H : ∀a, (P a → Q a)) (p : ∃a, P a) : ∃a, Q a :=
|
|
|
|
|
exists.elim p (λa Hp, exists.intro a (H a Hp))
|
|
|
|
|
|
|
|
|
|
theorem exists_congr [congr] {A : Type} {P Q : A → Prop} (H : ∀a, (P a ↔ Q a)) : (∃a, P a) ↔ ∃a, Q a :=
|
|
|
|
|
iff.intro
|
|
|
|
|
(exists_imp_exists (λa, iff.mp (H a)))
|
|
|
|
|
(exists_imp_exists (λa, iff.mpr (H a)))
|
|
|
|
|
|
|
|
|
|
theorem exists_unique_congr [congr] (H : ∀ x, p₁ x ↔ p₂ x) : (∃! x, p₁ x) ↔ (∃! x, p₂ x) :=
|
|
|
|
|
exists_congr (λx, and_congr (H x) (forall_congr (λy, imp_congr (H y) iff.rfl)))
|
|
|
|
|
end
|
|
|
|
|
|
2015-03-29 19:41:46 +00:00
|
|
|
|
/- decidable -/
|
|
|
|
|
|
2014-12-01 04:34:12 +00:00
|
|
|
|
inductive decidable [class] (p : Prop) : Type :=
|
2015-02-26 01:00:10 +00:00
|
|
|
|
| inl : p → decidable p
|
|
|
|
|
| inr : ¬p → decidable p
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-02-25 20:34:49 +00:00
|
|
|
|
definition decidable_true [instance] : decidable true :=
|
2014-12-01 04:34:12 +00:00
|
|
|
|
decidable.inl trivial
|
|
|
|
|
|
2015-02-25 20:34:49 +00:00
|
|
|
|
definition decidable_false [instance] : decidable false :=
|
2014-12-01 04:34:12 +00:00
|
|
|
|
decidable.inr not_false
|
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
-- We use "dependent" if-then-else to be able to communicate the if-then-else condition
|
|
|
|
|
-- to the branches
|
|
|
|
|
definition dite (c : Prop) [H : decidable c] {A : Type} : (c → A) → (¬ c → A) → A :=
|
|
|
|
|
decidable.rec_on H
|
|
|
|
|
|
|
|
|
|
/- if-then-else -/
|
|
|
|
|
|
|
|
|
|
definition ite (c : Prop) [H : decidable c] {A : Type} (t e : A) : A :=
|
|
|
|
|
decidable.rec_on H (λ Hc, t) (λ Hnc, e)
|
|
|
|
|
|
2014-12-01 04:34:12 +00:00
|
|
|
|
namespace decidable
|
|
|
|
|
variables {p q : Prop}
|
|
|
|
|
|
|
|
|
|
definition rec_on_true [H : decidable p] {H1 : p → Type} {H2 : ¬p → Type} (H3 : p) (H4 : H1 H3)
|
2015-02-11 20:49:27 +00:00
|
|
|
|
: decidable.rec_on H H1 H2 :=
|
|
|
|
|
decidable.rec_on H (λh, H4) (λh, !false.rec (h H3))
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
|
|
|
|
definition rec_on_false [H : decidable p] {H1 : p → Type} {H2 : ¬p → Type} (H3 : ¬p) (H4 : H2 H3)
|
2015-02-11 20:49:27 +00:00
|
|
|
|
: decidable.rec_on H H1 H2 :=
|
|
|
|
|
decidable.rec_on H (λh, false.rec _ (H3 h)) (λh, H4)
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
definition by_cases {q : Type} [C : decidable p] : (p → q) → (¬p → q) → q := !dite
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem em (p : Prop) [H : decidable p] : p ∨ ¬p := by_cases or.inl or.inr
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
|
|
|
|
theorem by_contradiction [Hp : decidable p] (H : ¬p → false) : p :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
if H1 : p then H1 else false.rec _ (H H1)
|
2014-12-16 03:17:51 +00:00
|
|
|
|
end decidable
|
|
|
|
|
|
|
|
|
|
section
|
|
|
|
|
variables {p q : Prop}
|
|
|
|
|
open decidable
|
|
|
|
|
definition decidable_of_decidable_of_iff (Hp : decidable p) (H : p ↔ q) : decidable q :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
if Hp : p then inl (iff.mp H Hp)
|
|
|
|
|
else inr (iff.mp (not_iff_not_of_iff H) Hp)
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2014-12-16 03:17:51 +00:00
|
|
|
|
definition decidable_of_decidable_of_eq (Hp : decidable p) (H : p = q) : decidable q :=
|
|
|
|
|
decidable_of_decidable_of_iff Hp (iff.of_eq H)
|
2015-07-15 19:49:47 +00:00
|
|
|
|
|
|
|
|
|
protected definition or.by_cases [Hp : decidable p] [Hq : decidable q] {A : Type}
|
2015-07-24 15:56:18 +00:00
|
|
|
|
(h : p ∨ q) (h₁ : p → A) (h₂ : q → A) : A :=
|
|
|
|
|
if hp : p then h₁ hp else
|
|
|
|
|
if hq : q then h₂ hq else
|
|
|
|
|
false.rec _ (or.elim h hp hq)
|
2014-12-16 03:17:51 +00:00
|
|
|
|
end
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
|
|
|
|
section
|
|
|
|
|
variables {p q : Prop}
|
|
|
|
|
open decidable (rec_on inl inr)
|
|
|
|
|
|
2015-02-25 20:34:49 +00:00
|
|
|
|
definition decidable_and [instance] [Hp : decidable p] [Hq : decidable q] : decidable (p ∧ q) :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
if hp : p then
|
|
|
|
|
if hq : q then inl (and.intro hp hq)
|
|
|
|
|
else inr (assume H : p ∧ q, hq (and.right H))
|
|
|
|
|
else inr (assume H : p ∧ q, hp (and.left H))
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-02-25 20:34:49 +00:00
|
|
|
|
definition decidable_or [instance] [Hp : decidable p] [Hq : decidable q] : decidable (p ∨ q) :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
if hp : p then inl (or.inl hp) else
|
|
|
|
|
if hq : q then inl (or.inr hq) else
|
|
|
|
|
inr (or.rec hp hq)
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-02-25 20:34:49 +00:00
|
|
|
|
definition decidable_not [instance] [Hp : decidable p] : decidable (¬p) :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
if hp : p then inr (absurd hp) else inl hp
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-02-25 20:34:49 +00:00
|
|
|
|
definition decidable_implies [instance] [Hp : decidable p] [Hq : decidable q] : decidable (p → q) :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
if hp : p then
|
|
|
|
|
if hq : q then inl (assume H, hq)
|
|
|
|
|
else inr (assume H : p → q, absurd (H hp) hq)
|
|
|
|
|
else inl (assume Hp, absurd Hp hp)
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-02-25 20:34:49 +00:00
|
|
|
|
definition decidable_iff [instance] [Hp : decidable p] [Hq : decidable q] : decidable (p ↔ q) :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
decidable_and
|
2015-02-24 22:09:20 +00:00
|
|
|
|
|
2014-12-01 04:34:12 +00:00
|
|
|
|
end
|
|
|
|
|
|
2015-02-24 22:09:20 +00:00
|
|
|
|
definition decidable_pred [reducible] {A : Type} (R : A → Prop) := Π (a : A), decidable (R a)
|
|
|
|
|
definition decidable_rel [reducible] {A : Type} (R : A → A → Prop) := Π (a b : A), decidable (R a b)
|
|
|
|
|
definition decidable_eq [reducible] (A : Type) := decidable_rel (@eq A)
|
2015-07-24 15:56:18 +00:00
|
|
|
|
definition decidable_ne [instance] {A : Type} [H : decidable_eq A] (a b : A) : decidable (a ≠ b) :=
|
|
|
|
|
decidable_implies
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-03-29 19:41:46 +00:00
|
|
|
|
namespace bool
|
2015-05-09 19:15:30 +00:00
|
|
|
|
theorem ff_ne_tt : ff = tt → false
|
2015-03-29 19:41:46 +00:00
|
|
|
|
| [none]
|
|
|
|
|
end bool
|
|
|
|
|
|
|
|
|
|
open bool
|
|
|
|
|
definition is_dec_eq {A : Type} (p : A → A → bool) : Prop := ∀ ⦃x y : A⦄, p x y = tt → x = y
|
|
|
|
|
definition is_dec_refl {A : Type} (p : A → A → bool) : Prop := ∀x, p x x = tt
|
|
|
|
|
|
|
|
|
|
open decidable
|
|
|
|
|
protected definition bool.has_decidable_eq [instance] : ∀a b : bool, decidable (a = b)
|
|
|
|
|
| ff ff := inl rfl
|
|
|
|
|
| ff tt := inr ff_ne_tt
|
|
|
|
|
| tt ff := inr (ne.symm ff_ne_tt)
|
|
|
|
|
| tt tt := inl rfl
|
|
|
|
|
|
|
|
|
|
definition decidable_eq_of_bool_pred {A : Type} {p : A → A → bool} (H₁ : is_dec_eq p) (H₂ : is_dec_refl p) : decidable_eq A :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
take x y : A, if Hp : p x y = tt then inl (H₁ Hp)
|
|
|
|
|
else inr (assume Hxy : x = y, (eq.subst Hxy Hp) (H₂ y))
|
2015-03-29 19:41:46 +00:00
|
|
|
|
|
2015-04-03 06:31:40 +00:00
|
|
|
|
theorem decidable_eq_inl_refl {A : Type} [H : decidable_eq A] (a : A) : H a a = inl (eq.refl a) :=
|
|
|
|
|
match H a a with
|
|
|
|
|
| inl e := rfl
|
|
|
|
|
| inr n := absurd rfl n
|
|
|
|
|
end
|
|
|
|
|
|
2015-04-22 01:56:28 +00:00
|
|
|
|
open eq.ops
|
2015-04-03 06:31:40 +00:00
|
|
|
|
theorem decidable_eq_inr_neg {A : Type} [H : decidable_eq A] {a b : A} : Π n : a ≠ b, H a b = inr n :=
|
|
|
|
|
assume n,
|
|
|
|
|
match H a b with
|
|
|
|
|
| inl e := absurd e n
|
|
|
|
|
| inr n₁ := proof_irrel n n₁ ▸ rfl
|
|
|
|
|
end
|
|
|
|
|
|
2015-03-29 19:41:46 +00:00
|
|
|
|
/- inhabited -/
|
|
|
|
|
|
2014-12-01 04:34:12 +00:00
|
|
|
|
inductive inhabited [class] (A : Type) : Type :=
|
|
|
|
|
mk : A → inhabited A
|
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
protected definition inhabited.value {A : Type} : inhabited A → A :=
|
|
|
|
|
inhabited.rec (λa, a)
|
2015-01-08 02:45:58 +00:00
|
|
|
|
|
2014-12-13 22:26:44 +00:00
|
|
|
|
protected definition inhabited.destruct {A : Type} {B : Type} (H1 : inhabited A) (H2 : A → B) : B :=
|
2014-12-01 04:34:12 +00:00
|
|
|
|
inhabited.rec H2 H1
|
|
|
|
|
|
2015-01-08 02:45:58 +00:00
|
|
|
|
definition default (A : Type) [H : inhabited A] : A :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
inhabited.value H
|
2014-12-13 22:26:44 +00:00
|
|
|
|
|
2015-05-08 23:40:03 +00:00
|
|
|
|
definition arbitrary [irreducible] (A : Type) [H : inhabited A] : A :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
inhabited.value H
|
2015-01-05 21:27:09 +00:00
|
|
|
|
|
2015-02-25 20:34:49 +00:00
|
|
|
|
definition Prop.is_inhabited [instance] : inhabited Prop :=
|
2014-12-13 22:26:44 +00:00
|
|
|
|
inhabited.mk true
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-02-25 20:34:49 +00:00
|
|
|
|
definition inhabited_fun [instance] (A : Type) {B : Type} [H : inhabited B] : inhabited (A → B) :=
|
2014-12-13 22:26:44 +00:00
|
|
|
|
inhabited.rec_on H (λb, inhabited.mk (λa, b))
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-02-25 20:34:49 +00:00
|
|
|
|
definition inhabited_Pi [instance] (A : Type) {B : A → Type} [H : Πx, inhabited (B x)] :
|
2014-12-01 04:34:12 +00:00
|
|
|
|
inhabited (Πx, B x) :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
inhabited.mk (λa, !default)
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-03-29 19:41:46 +00:00
|
|
|
|
protected definition bool.is_inhabited [instance] : inhabited bool :=
|
|
|
|
|
inhabited.mk ff
|
|
|
|
|
|
2014-12-01 04:34:12 +00:00
|
|
|
|
inductive nonempty [class] (A : Type) : Prop :=
|
|
|
|
|
intro : A → nonempty A
|
|
|
|
|
|
2014-12-13 22:26:44 +00:00
|
|
|
|
protected definition nonempty.elim {A : Type} {B : Prop} (H1 : nonempty A) (H2 : A → B) : B :=
|
|
|
|
|
nonempty.rec H2 H1
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-03-25 21:40:55 +00:00
|
|
|
|
theorem nonempty_of_inhabited [instance] {A : Type} [H : inhabited A] : nonempty A :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
nonempty.intro !default
|
2014-12-01 04:34:12 +00:00
|
|
|
|
|
2015-08-13 01:37:33 +00:00
|
|
|
|
theorem nonempty_of_exists {A : Type} {P : A → Prop} : (∃x, P x) → nonempty A :=
|
|
|
|
|
Exists.rec (λw H, nonempty.intro w)
|
|
|
|
|
|
2015-04-01 18:48:18 +00:00
|
|
|
|
/- subsingleton -/
|
|
|
|
|
|
|
|
|
|
inductive subsingleton [class] (A : Type) : Prop :=
|
|
|
|
|
intro : (∀ a b : A, a = b) → subsingleton A
|
|
|
|
|
|
|
|
|
|
protected definition subsingleton.elim {A : Type} [H : subsingleton A] : ∀(a b : A), a = b :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
subsingleton.rec (λp, p) H
|
2015-04-01 18:48:18 +00:00
|
|
|
|
|
|
|
|
|
definition subsingleton_prop [instance] (p : Prop) : subsingleton p :=
|
|
|
|
|
subsingleton.intro (λa b, !proof_irrel)
|
|
|
|
|
|
|
|
|
|
definition subsingleton_decidable [instance] (p : Prop) : subsingleton (decidable p) :=
|
|
|
|
|
subsingleton.intro (λ d₁,
|
|
|
|
|
match d₁ with
|
|
|
|
|
| inl t₁ := (λ d₂,
|
|
|
|
|
match d₂ with
|
|
|
|
|
| inl t₂ := eq.rec_on (proof_irrel t₁ t₂) rfl
|
|
|
|
|
| inr f₂ := absurd t₁ f₂
|
|
|
|
|
end)
|
|
|
|
|
| inr f₁ := (λ d₂,
|
|
|
|
|
match d₂ with
|
|
|
|
|
| inl t₂ := absurd t₂ f₁
|
|
|
|
|
| inr f₂ := eq.rec_on (proof_irrel f₁ f₂) rfl
|
|
|
|
|
end)
|
|
|
|
|
end)
|
|
|
|
|
|
|
|
|
|
protected theorem rec_subsingleton {p : Prop} [H : decidable p]
|
|
|
|
|
{H1 : p → Type} {H2 : ¬p → Type}
|
|
|
|
|
[H3 : Π(h : p), subsingleton (H1 h)] [H4 : Π(h : ¬p), subsingleton (H2 h)]
|
|
|
|
|
: subsingleton (decidable.rec_on H H1 H2) :=
|
|
|
|
|
decidable.rec_on H (λh, H3 h) (λh, H4 h) --this can be proven using dependent version of "by_cases"
|
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem if_pos {c : Prop} [H : decidable c] (Hc : c) {A : Type} {t e : A} : (ite c t e) = t :=
|
2014-12-01 04:34:12 +00:00
|
|
|
|
decidable.rec
|
|
|
|
|
(λ Hc : c, eq.refl (@ite c (decidable.inl Hc) A t e))
|
|
|
|
|
(λ Hnc : ¬c, absurd Hc Hnc)
|
|
|
|
|
H
|
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem if_neg {c : Prop} [H : decidable c] (Hnc : ¬c) {A : Type} {t e : A} : (ite c t e) = e :=
|
2014-12-01 04:34:12 +00:00
|
|
|
|
decidable.rec
|
|
|
|
|
(λ Hc : c, absurd Hc Hnc)
|
|
|
|
|
(λ Hnc : ¬c, eq.refl (@ite c (decidable.inr Hnc) A t e))
|
|
|
|
|
H
|
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem if_t_t [simp] (c : Prop) [H : decidable c] {A : Type} (t : A) : (ite c t t) = t :=
|
2014-12-01 04:34:12 +00:00
|
|
|
|
decidable.rec
|
|
|
|
|
(λ Hc : c, eq.refl (@ite c (decidable.inl Hc) A t t))
|
|
|
|
|
(λ Hnc : ¬c, eq.refl (@ite c (decidable.inr Hnc) A t t))
|
|
|
|
|
H
|
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem implies_of_if_pos {c t e : Prop} [H : decidable c] (h : ite c t e) : c → t :=
|
2015-05-26 23:08:01 +00:00
|
|
|
|
assume Hc, eq.rec_on (if_pos Hc) h
|
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem implies_of_if_neg {c t e : Prop} [H : decidable c] (h : ite c t e) : ¬c → e :=
|
2015-05-26 23:08:01 +00:00
|
|
|
|
assume Hnc, eq.rec_on (if_neg Hnc) h
|
|
|
|
|
|
2015-06-02 21:06:15 +00:00
|
|
|
|
theorem if_ctx_congr {A : Type} {b c : Prop} [dec_b : decidable b] [dec_c : decidable c]
|
|
|
|
|
{x y u v : A}
|
2015-06-02 20:14:38 +00:00
|
|
|
|
(h_c : b ↔ c) (h_t : c → x = u) (h_e : ¬c → y = v) :
|
2015-07-24 15:56:18 +00:00
|
|
|
|
ite b x y = ite c u v :=
|
2015-06-02 20:14:38 +00:00
|
|
|
|
decidable.rec_on dec_b
|
|
|
|
|
(λ hp : b, calc
|
2015-07-24 15:56:18 +00:00
|
|
|
|
ite b x y = x : if_pos hp
|
|
|
|
|
... = u : h_t (iff.mp h_c hp)
|
|
|
|
|
... = ite c u v : if_pos (iff.mp h_c hp))
|
2015-06-02 20:14:38 +00:00
|
|
|
|
(λ hn : ¬b, calc
|
2015-07-24 15:56:18 +00:00
|
|
|
|
ite b x y = y : if_neg hn
|
|
|
|
|
... = v : h_e (iff.mp (not_iff_not_of_iff h_c) hn)
|
|
|
|
|
... = ite c u v : if_neg (iff.mp (not_iff_not_of_iff h_c) hn))
|
2015-06-02 20:14:38 +00:00
|
|
|
|
|
2015-06-02 21:06:15 +00:00
|
|
|
|
theorem if_congr {A : Type} {b c : Prop} [dec_b : decidable b] [dec_c : decidable c]
|
|
|
|
|
{x y u v : A}
|
|
|
|
|
(h_c : b ↔ c) (h_t : x = u) (h_e : y = v) :
|
2015-07-24 15:56:18 +00:00
|
|
|
|
ite b x y = ite c u v :=
|
2015-06-02 21:06:15 +00:00
|
|
|
|
@if_ctx_congr A b c dec_b dec_c x y u v h_c (λ h, h_t) (λ h, h_e)
|
|
|
|
|
|
2015-07-23 00:21:47 +00:00
|
|
|
|
theorem if_ctx_simp_congr {A : Type} {b c : Prop} [dec_b : decidable b] {x y u v : A}
|
2015-06-02 21:06:15 +00:00
|
|
|
|
(h_c : b ↔ c) (h_t : c → x = u) (h_e : ¬c → y = v) :
|
2015-07-24 15:56:18 +00:00
|
|
|
|
ite b x y = (@ite c (decidable_of_decidable_of_iff dec_b h_c) A u v) :=
|
2015-06-02 21:06:15 +00:00
|
|
|
|
@if_ctx_congr A b c dec_b (decidable_of_decidable_of_iff dec_b h_c) x y u v h_c h_t h_e
|
|
|
|
|
|
2015-07-23 00:21:47 +00:00
|
|
|
|
theorem if_simp_congr [congr] {A : Type} {b c : Prop} [dec_b : decidable b] {x y u v : A}
|
2015-06-02 20:14:38 +00:00
|
|
|
|
(h_c : b ↔ c) (h_t : x = u) (h_e : y = v) :
|
2015-07-24 15:56:18 +00:00
|
|
|
|
ite b x y = (@ite c (decidable_of_decidable_of_iff dec_b h_c) A u v) :=
|
2015-07-23 00:21:47 +00:00
|
|
|
|
@if_ctx_simp_congr A b c dec_b x y u v h_c (λ h, h_t) (λ h, h_e)
|
2015-06-02 21:06:15 +00:00
|
|
|
|
|
2015-11-21 00:38:10 +00:00
|
|
|
|
definition if_true [simp] {A : Type} (t e : A) : (if true then t else e) = t :=
|
|
|
|
|
if_pos trivial
|
|
|
|
|
|
|
|
|
|
definition if_false [simp] {A : Type} (t e : A) : (if false then t else e) = e :=
|
|
|
|
|
if_neg not_false
|
|
|
|
|
|
2015-06-02 21:06:15 +00:00
|
|
|
|
theorem if_congr_prop {b c x y u v : Prop} [dec_b : decidable b] [dec_c : decidable c]
|
|
|
|
|
(h_c : b ↔ c) (h_t : c → (x ↔ u)) (h_e : ¬c → (y ↔ v)) :
|
2015-07-24 15:56:18 +00:00
|
|
|
|
ite b x y ↔ ite c u v :=
|
2015-06-02 21:06:15 +00:00
|
|
|
|
decidable.rec_on dec_b
|
|
|
|
|
(λ hp : b, calc
|
2015-07-24 15:56:18 +00:00
|
|
|
|
ite b x y ↔ x : iff.of_eq (if_pos hp)
|
|
|
|
|
... ↔ u : h_t (iff.mp h_c hp)
|
|
|
|
|
... ↔ ite c u v : iff.of_eq (if_pos (iff.mp h_c hp)))
|
2015-06-02 21:06:15 +00:00
|
|
|
|
(λ hn : ¬b, calc
|
2015-07-24 15:56:18 +00:00
|
|
|
|
ite b x y ↔ y : iff.of_eq (if_neg hn)
|
|
|
|
|
... ↔ v : h_e (iff.mp (not_iff_not_of_iff h_c) hn)
|
|
|
|
|
... ↔ ite c u v : iff.of_eq (if_neg (iff.mp (not_iff_not_of_iff h_c) hn)))
|
2015-06-02 21:06:15 +00:00
|
|
|
|
|
2015-07-23 00:21:47 +00:00
|
|
|
|
theorem if_ctx_simp_congr_prop {b c x y u v : Prop} [dec_b : decidable b]
|
|
|
|
|
(h_c : b ↔ c) (h_t : c → (x ↔ u)) (h_e : ¬c → (y ↔ v)) :
|
2015-07-24 15:56:18 +00:00
|
|
|
|
ite b x y ↔ (@ite c (decidable_of_decidable_of_iff dec_b h_c) Prop u v) :=
|
2015-06-02 21:06:15 +00:00
|
|
|
|
@if_congr_prop b c x y u v dec_b (decidable_of_decidable_of_iff dec_b h_c) h_c h_t h_e
|
2015-06-02 20:14:38 +00:00
|
|
|
|
|
2015-07-23 00:21:47 +00:00
|
|
|
|
theorem if_simp_congr_prop [congr] {b c x y u v : Prop} [dec_b : decidable b]
|
|
|
|
|
(h_c : b ↔ c) (h_t : x ↔ u) (h_e : y ↔ v) :
|
2015-07-24 15:56:18 +00:00
|
|
|
|
ite b x y ↔ (@ite c (decidable_of_decidable_of_iff dec_b h_c) Prop u v) :=
|
2015-07-23 00:21:47 +00:00
|
|
|
|
@if_ctx_simp_congr_prop b c x y u v dec_b h_c (λ h, h_t) (λ h, h_e)
|
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem dif_pos {c : Prop} [H : decidable c] (Hc : c) {A : Type} {t : c → A} {e : ¬ c → A} : dite c t e = t Hc :=
|
2014-12-01 04:34:12 +00:00
|
|
|
|
decidable.rec
|
|
|
|
|
(λ Hc : c, eq.refl (@dite c (decidable.inl Hc) A t e))
|
|
|
|
|
(λ Hnc : ¬c, absurd Hc Hnc)
|
|
|
|
|
H
|
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem dif_neg {c : Prop} [H : decidable c] (Hnc : ¬c) {A : Type} {t : c → A} {e : ¬ c → A} : dite c t e = e Hnc :=
|
2014-12-01 04:34:12 +00:00
|
|
|
|
decidable.rec
|
|
|
|
|
(λ Hc : c, absurd Hc Hnc)
|
|
|
|
|
(λ Hnc : ¬c, eq.refl (@dite c (decidable.inr Hnc) A t e))
|
|
|
|
|
H
|
|
|
|
|
|
2015-06-02 21:06:15 +00:00
|
|
|
|
theorem dif_ctx_congr {A : Type} {b c : Prop} [dec_b : decidable b] [dec_c : decidable c]
|
2015-06-02 20:14:38 +00:00
|
|
|
|
{x : b → A} {u : c → A} {y : ¬b → A} {v : ¬c → A}
|
|
|
|
|
(h_c : b ↔ c)
|
2015-07-18 09:28:53 +00:00
|
|
|
|
(h_t : ∀ (h : c), x (iff.mpr h_c h) = u h)
|
|
|
|
|
(h_e : ∀ (h : ¬c), y (iff.mpr (not_iff_not_of_iff h_c) h) = v h) :
|
2015-06-02 21:06:15 +00:00
|
|
|
|
(@dite b dec_b A x y) = (@dite c dec_c A u v) :=
|
2015-06-02 20:14:38 +00:00
|
|
|
|
decidable.rec_on dec_b
|
|
|
|
|
(λ hp : b, calc
|
2015-07-24 15:56:18 +00:00
|
|
|
|
dite b x y = x hp : dif_pos hp
|
|
|
|
|
... = x (iff.mpr h_c (iff.mp h_c hp)) : proof_irrel
|
|
|
|
|
... = u (iff.mp h_c hp) : h_t
|
|
|
|
|
... = dite c u v : dif_pos (iff.mp h_c hp))
|
|
|
|
|
(λ hn : ¬b, let h_nc : ¬b ↔ ¬c := not_iff_not_of_iff h_c in calc
|
|
|
|
|
dite b x y = y hn : dif_neg hn
|
|
|
|
|
... = y (iff.mpr h_nc (iff.mp h_nc hn)) : proof_irrel
|
|
|
|
|
... = v (iff.mp h_nc hn) : h_e
|
|
|
|
|
... = dite c u v : dif_neg (iff.mp h_nc hn))
|
2015-06-02 20:14:38 +00:00
|
|
|
|
|
2015-07-23 00:21:47 +00:00
|
|
|
|
theorem dif_ctx_simp_congr {A : Type} {b c : Prop} [dec_b : decidable b]
|
2015-06-02 21:06:15 +00:00
|
|
|
|
{x : b → A} {u : c → A} {y : ¬b → A} {v : ¬c → A}
|
|
|
|
|
(h_c : b ↔ c)
|
2015-07-18 09:28:53 +00:00
|
|
|
|
(h_t : ∀ (h : c), x (iff.mpr h_c h) = u h)
|
|
|
|
|
(h_e : ∀ (h : ¬c), y (iff.mpr (not_iff_not_of_iff h_c) h) = v h) :
|
2015-06-02 21:06:15 +00:00
|
|
|
|
(@dite b dec_b A x y) = (@dite c (decidable_of_decidable_of_iff dec_b h_c) A u v) :=
|
|
|
|
|
@dif_ctx_congr A b c dec_b (decidable_of_decidable_of_iff dec_b h_c) x u y v h_c h_t h_e
|
|
|
|
|
|
2014-12-01 04:34:12 +00:00
|
|
|
|
-- Remark: dite and ite are "definitionally equal" when we ignore the proofs.
|
|
|
|
|
theorem dite_ite_eq (c : Prop) [H : decidable c] {A : Type} (t : A) (e : A) : dite c (λh, t) (λh, e) = ite c t e :=
|
|
|
|
|
rfl
|
2014-12-12 02:14:03 +00:00
|
|
|
|
|
|
|
|
|
definition is_true (c : Prop) [H : decidable c] : Prop :=
|
|
|
|
|
if c then true else false
|
|
|
|
|
|
|
|
|
|
definition is_false (c : Prop) [H : decidable c] : Prop :=
|
|
|
|
|
if c then false else true
|
|
|
|
|
|
2015-07-29 23:11:23 +00:00
|
|
|
|
definition of_is_true {c : Prop} [H₁ : decidable c] (H₂ : is_true c) : c :=
|
2014-12-12 21:20:27 +00:00
|
|
|
|
decidable.rec_on H₁ (λ Hc, Hc) (λ Hnc, !false.rec (if_neg Hnc ▸ H₂))
|
2014-12-12 02:14:03 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
notation `dec_trivial` := of_is_true trivial
|
|
|
|
|
|
2014-12-12 02:14:03 +00:00
|
|
|
|
theorem not_of_not_is_true {c : Prop} [H₁ : decidable c] (H₂ : ¬ is_true c) : ¬ c :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
if Hc : c then absurd trivial (if_pos Hc ▸ H₂) else Hc
|
2014-12-12 02:14:03 +00:00
|
|
|
|
|
|
|
|
|
theorem not_of_is_false {c : Prop} [H₁ : decidable c] (H₂ : is_false c) : ¬ c :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
if Hc : c then !false.rec (if_pos Hc ▸ H₂) else Hc
|
2014-12-12 02:14:03 +00:00
|
|
|
|
|
|
|
|
|
theorem of_not_is_false {c : Prop} [H₁ : decidable c] (H₂ : ¬ is_false c) : c :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
if Hc : c then Hc else absurd trivial (if_neg Hc ▸ H₂)
|
2015-07-23 00:21:47 +00:00
|
|
|
|
|
|
|
|
|
-- namespace used to collect congruence rules for "contextual simplification"
|
|
|
|
|
namespace contextual
|
|
|
|
|
attribute if_ctx_simp_congr [congr]
|
|
|
|
|
attribute if_ctx_simp_congr_prop [congr]
|
|
|
|
|
attribute dif_ctx_simp_congr [congr]
|
|
|
|
|
end contextual
|