/
Opposite.lean
255 lines (197 loc) · 12.3 KB
/
Opposite.lean
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
/-
Copyright (c) 2018 Kenny Lau. All rights reserved.
Released under Apache 2.0 license as described in the file LICENSE.
Authors: Kenny Lau
-/
import Mathlib.Algebra.Group.Opposite
import Mathlib.Algebra.Ring.Hom.Defs
#align_import algebra.ring.opposite from "leanprover-community/mathlib"@"76de8ae01554c3b37d66544866659ff174e66e1f"
/-!
# Ring structures on the multiplicative opposite
-/
universe u v
variable (α : Type u)
namespace MulOpposite
instance instDistrib [Distrib α] : Distrib αᵐᵒᵖ :=
{ MulOpposite.add α, MulOpposite.mul α with
left_distrib := fun x y z => unop_injective <| add_mul (unop y) (unop z) (unop x),
right_distrib := fun x y z => unop_injective <| mul_add (unop z) (unop x) (unop y) }
instance instMulZeroClass [MulZeroClass α] : MulZeroClass αᵐᵒᵖ where
zero := 0
mul := (· * ·)
zero_mul x := unop_injective <| mul_zero <| unop x
mul_zero x := unop_injective <| zero_mul <| unop x
instance instMulZeroOneClass [MulZeroOneClass α] : MulZeroOneClass αᵐᵒᵖ :=
{ MulOpposite.instMulZeroClass α, MulOpposite.mulOneClass α with }
instance instSemigroupWithZero [SemigroupWithZero α] : SemigroupWithZero αᵐᵒᵖ :=
{ MulOpposite.semigroup α, MulOpposite.instMulZeroClass α with }
instance instMonoidWithZero [MonoidWithZero α] : MonoidWithZero αᵐᵒᵖ :=
{ MulOpposite.monoid α, MulOpposite.instMulZeroOneClass α with }
instance instNonUnitalNonAssocSemiring [NonUnitalNonAssocSemiring α] :
NonUnitalNonAssocSemiring αᵐᵒᵖ :=
{ MulOpposite.addCommMonoid α, MulOpposite.instMulZeroClass α, MulOpposite.instDistrib α with }
instance instNonUnitalSemiring [NonUnitalSemiring α] : NonUnitalSemiring αᵐᵒᵖ :=
{ MulOpposite.instSemigroupWithZero α, MulOpposite.instNonUnitalNonAssocSemiring α with }
instance instNonAssocSemiring [NonAssocSemiring α] : NonAssocSemiring αᵐᵒᵖ :=
{ MulOpposite.addMonoidWithOne α, MulOpposite.instMulZeroOneClass α,
MulOpposite.instNonUnitalNonAssocSemiring α with }
instance instSemiring [Semiring α] : Semiring αᵐᵒᵖ :=
{ MulOpposite.instNonUnitalSemiring α, MulOpposite.instNonAssocSemiring α,
MulOpposite.instMonoidWithZero α with }
instance instNonUnitalCommSemiring [NonUnitalCommSemiring α] : NonUnitalCommSemiring αᵐᵒᵖ :=
{ MulOpposite.instNonUnitalSemiring α, MulOpposite.commSemigroup α with }
instance instCommSemiring [CommSemiring α] : CommSemiring αᵐᵒᵖ :=
{ MulOpposite.instSemiring α, MulOpposite.commSemigroup α with }
instance instNonUnitalNonAssocRing [NonUnitalNonAssocRing α] : NonUnitalNonAssocRing αᵐᵒᵖ :=
{ MulOpposite.addCommGroup α, MulOpposite.instMulZeroClass α,
MulOpposite.instDistrib α with }
instance instNonUnitalRing [NonUnitalRing α] : NonUnitalRing αᵐᵒᵖ :=
{ MulOpposite.addCommGroup α, MulOpposite.instSemigroupWithZero α,
MulOpposite.instDistrib α with }
instance instNonAssocRing [NonAssocRing α] : NonAssocRing αᵐᵒᵖ :=
{ MulOpposite.addCommGroup α, MulOpposite.instMulZeroOneClass α,
MulOpposite.instDistrib α, MulOpposite.addGroupWithOne α with }
instance instRing [Ring α] : Ring αᵐᵒᵖ :=
{ MulOpposite.monoid α, MulOpposite.instNonAssocRing α with }
instance instNonUnitalCommRing [NonUnitalCommRing α] : NonUnitalCommRing αᵐᵒᵖ :=
{ MulOpposite.instNonUnitalRing α,
MulOpposite.instNonUnitalCommSemiring α with }
instance instCommRing [CommRing α] : CommRing αᵐᵒᵖ :=
{ MulOpposite.instRing α, MulOpposite.instCommSemiring α with }
instance instNoZeroDivisors [Zero α] [Mul α] [NoZeroDivisors α] : NoZeroDivisors αᵐᵒᵖ where
eq_zero_or_eq_zero_of_mul_eq_zero (H : op (_ * _) = op (0 : α)) :=
Or.casesOn (eq_zero_or_eq_zero_of_mul_eq_zero <| op_injective H)
(fun hy => Or.inr <| unop_injective <| hy) fun hx => Or.inl <| unop_injective <| hx
instance instIsDomain [Ring α] [IsDomain α] : IsDomain αᵐᵒᵖ :=
NoZeroDivisors.to_isDomain _
instance instGroupWithZero [GroupWithZero α] : GroupWithZero αᵐᵒᵖ :=
{ MulOpposite.instMonoidWithZero α, MulOpposite.divInvMonoid α,
MulOpposite.nontrivial α with
mul_inv_cancel := fun _ hx => unop_injective <| inv_mul_cancel <| unop_injective.ne hx,
inv_zero := unop_injective inv_zero }
end MulOpposite
namespace AddOpposite
instance instDistrib [Distrib α] : Distrib αᵃᵒᵖ :=
{ AddOpposite.add α, @AddOpposite.mul α _ with
left_distrib := fun x y z => unop_injective <| mul_add (unop x) (unop z) (unop y),
right_distrib := fun x y z => unop_injective <| add_mul (unop y) (unop x) (unop z) }
instance instMulZeroClass [MulZeroClass α] : MulZeroClass αᵃᵒᵖ where
zero := 0
mul := (· * ·)
zero_mul x := unop_injective <| zero_mul <| unop x
mul_zero x := unop_injective <| mul_zero <| unop x
instance instMulZeroOneClass [MulZeroOneClass α] : MulZeroOneClass αᵃᵒᵖ :=
{ AddOpposite.instMulZeroClass α, AddOpposite.mulOneClass α with }
instance instSemigroupWithZero [SemigroupWithZero α] : SemigroupWithZero αᵃᵒᵖ :=
{ AddOpposite.semigroup α, AddOpposite.instMulZeroClass α with }
instance instMonoidWithZero [MonoidWithZero α] : MonoidWithZero αᵃᵒᵖ :=
{ AddOpposite.monoid α, AddOpposite.instMulZeroOneClass α with }
instance instNonUnitalNonAssocSemiring [NonUnitalNonAssocSemiring α] :
NonUnitalNonAssocSemiring αᵃᵒᵖ :=
{ AddOpposite.addCommMonoid α, AddOpposite.instMulZeroClass α, AddOpposite.instDistrib α with }
instance instNonUnitalSemiring [NonUnitalSemiring α] : NonUnitalSemiring αᵃᵒᵖ :=
{ AddOpposite.instSemigroupWithZero α, AddOpposite.instNonUnitalNonAssocSemiring α with }
instance instNonAssocSemiring [NonAssocSemiring α] : NonAssocSemiring αᵃᵒᵖ :=
{ AddOpposite.instMulZeroOneClass α, AddOpposite.instNonUnitalNonAssocSemiring α,
AddOpposite.addCommMonoidWithOne _ with }
instance instSemiring [Semiring α] : Semiring αᵃᵒᵖ :=
{ AddOpposite.instNonUnitalSemiring α, AddOpposite.instNonAssocSemiring α,
AddOpposite.instMonoidWithZero α with }
instance instNonUnitalCommSemiring [NonUnitalCommSemiring α] : NonUnitalCommSemiring αᵃᵒᵖ :=
{ AddOpposite.instNonUnitalSemiring α, AddOpposite.commSemigroup α with }
instance instCommSemiring [CommSemiring α] : CommSemiring αᵃᵒᵖ :=
{ AddOpposite.instSemiring α, AddOpposite.commSemigroup α with }
instance instNonUnitalNonAssocRing [NonUnitalNonAssocRing α] : NonUnitalNonAssocRing αᵃᵒᵖ :=
{ AddOpposite.addCommGroup α, AddOpposite.instMulZeroClass α, AddOpposite.instDistrib α with }
instance instNonUnitalRing [NonUnitalRing α] : NonUnitalRing αᵃᵒᵖ :=
{ AddOpposite.addCommGroup α, AddOpposite.instSemigroupWithZero α,
AddOpposite.instDistrib α with }
instance instNonAssocRing [NonAssocRing α] : NonAssocRing αᵃᵒᵖ :=
{ AddOpposite.addCommGroupWithOne α, AddOpposite.instMulZeroOneClass α,
AddOpposite.instDistrib α with }
instance instRing [Ring α] : Ring αᵃᵒᵖ :=
{ AddOpposite.instNonAssocRing α, AddOpposite.instSemiring α with }
instance instNonUnitalCommRing [NonUnitalCommRing α] : NonUnitalCommRing αᵃᵒᵖ :=
{ AddOpposite.instNonUnitalRing α, AddOpposite.instNonUnitalCommSemiring α with }
instance instCommRing [CommRing α] : CommRing αᵃᵒᵖ :=
{ AddOpposite.instRing α, AddOpposite.instCommSemiring α with }
instance instNoZeroDivisors [Zero α] [Mul α] [NoZeroDivisors α] : NoZeroDivisors αᵃᵒᵖ where
eq_zero_or_eq_zero_of_mul_eq_zero (H : op (_ * _) = op (0 : α)) :=
Or.imp (fun hx => unop_injective hx) (fun hy => unop_injective hy)
(@eq_zero_or_eq_zero_of_mul_eq_zero α _ _ _ _ _ <| op_injective H)
instance instIsDomain [Ring α] [IsDomain α] : IsDomain αᵃᵒᵖ :=
NoZeroDivisors.to_isDomain _
instance instGroupWithZero [GroupWithZero α] : GroupWithZero αᵃᵒᵖ :=
{ AddOpposite.instMonoidWithZero α, AddOpposite.divInvMonoid α,
AddOpposite.nontrivial α with
mul_inv_cancel := fun _ hx => unop_injective <| mul_inv_cancel <| unop_injective.ne hx,
inv_zero := unop_injective inv_zero }
end AddOpposite
open MulOpposite
/-- A non-unital ring homomorphism `f : R →ₙ+* S` such that `f x` commutes with `f y` for all `x, y`
defines a non-unital ring homomorphism to `Sᵐᵒᵖ`. -/
@[simps (config := .asFn)]
def NonUnitalRingHom.toOpposite {R S : Type*} [NonUnitalNonAssocSemiring R]
[NonUnitalNonAssocSemiring S] (f : R →ₙ+* S) (hf : ∀ x y, Commute (f x) (f y)) : R →ₙ+* Sᵐᵒᵖ :=
{ ((opAddEquiv : S ≃+ Sᵐᵒᵖ).toAddMonoidHom.comp ↑f : R →+ Sᵐᵒᵖ), f.toMulHom.toOpposite hf with
toFun := MulOpposite.op ∘ f }
#align non_unital_ring_hom.to_opposite NonUnitalRingHom.toOpposite
/-- A non-unital ring homomorphism `f : R →ₙ* S` such that `f x` commutes with `f y` for all `x, y`
defines a non-unital ring homomorphism from `Rᵐᵒᵖ`. -/
@[simps (config := .asFn)]
def NonUnitalRingHom.fromOpposite {R S : Type*} [NonUnitalNonAssocSemiring R]
[NonUnitalNonAssocSemiring S] (f : R →ₙ+* S) (hf : ∀ x y, Commute (f x) (f y)) : Rᵐᵒᵖ →ₙ+* S :=
{ (f.toAddMonoidHom.comp (opAddEquiv : R ≃+ Rᵐᵒᵖ).symm.toAddMonoidHom : Rᵐᵒᵖ →+ S),
f.toMulHom.fromOpposite hf with toFun := f ∘ MulOpposite.unop }
#align non_unital_ring_hom.from_opposite NonUnitalRingHom.fromOpposite
/-- A non-unital ring hom `α →ₙ+* β` can equivalently be viewed as a non-unital ring hom
`αᵐᵒᵖ →+* βᵐᵒᵖ`. This is the action of the (fully faithful) `ᵐᵒᵖ`-functor on morphisms. -/
@[simps]
def NonUnitalRingHom.op {α β} [NonUnitalNonAssocSemiring α] [NonUnitalNonAssocSemiring β] :
(α →ₙ+* β) ≃ (αᵐᵒᵖ →ₙ+* βᵐᵒᵖ) where
toFun f := { AddMonoidHom.mulOp f.toAddMonoidHom, MulHom.op f.toMulHom with }
invFun f := { AddMonoidHom.mulUnop f.toAddMonoidHom, MulHom.unop f.toMulHom with }
left_inv _ := rfl
right_inv _ := rfl
#align non_unital_ring_hom.op NonUnitalRingHom.op
/-- The 'unopposite' of a non-unital ring hom `αᵐᵒᵖ →ₙ+* βᵐᵒᵖ`. Inverse to
`NonUnitalRingHom.op`. -/
@[simp]
def NonUnitalRingHom.unop {α β} [NonUnitalNonAssocSemiring α] [NonUnitalNonAssocSemiring β] :
(αᵐᵒᵖ →ₙ+* βᵐᵒᵖ) ≃ (α →ₙ+* β) :=
NonUnitalRingHom.op.symm
#align non_unital_ring_hom.unop NonUnitalRingHom.unop
/-- A ring homomorphism `f : R →+* S` such that `f x` commutes with `f y` for all `x, y` defines
a ring homomorphism to `Sᵐᵒᵖ`. -/
@[simps (config := .asFn)]
def RingHom.toOpposite {R S : Type*} [Semiring R] [Semiring S] (f : R →+* S)
(hf : ∀ x y, Commute (f x) (f y)) : R →+* Sᵐᵒᵖ :=
{ ((opAddEquiv : S ≃+ Sᵐᵒᵖ).toAddMonoidHom.comp ↑f : R →+ Sᵐᵒᵖ), f.toMonoidHom.toOpposite hf with
toFun := MulOpposite.op ∘ f }
#align ring_hom.to_opposite RingHom.toOpposite
#align ring_hom.to_opposite_apply RingHom.toOpposite_apply
/-- A ring homomorphism `f : R →+* S` such that `f x` commutes with `f y` for all `x, y` defines
a ring homomorphism from `Rᵐᵒᵖ`. -/
@[simps (config := .asFn)]
def RingHom.fromOpposite {R S : Type*} [Semiring R] [Semiring S] (f : R →+* S)
(hf : ∀ x y, Commute (f x) (f y)) : Rᵐᵒᵖ →+* S :=
{ (f.toAddMonoidHom.comp (opAddEquiv : R ≃+ Rᵐᵒᵖ).symm.toAddMonoidHom : Rᵐᵒᵖ →+ S),
f.toMonoidHom.fromOpposite hf with toFun := f ∘ MulOpposite.unop }
#align ring_hom.from_opposite RingHom.fromOpposite
#align ring_hom.from_opposite_apply RingHom.fromOpposite_apply
/-- A ring hom `α →+* β` can equivalently be viewed as a ring hom `αᵐᵒᵖ →+* βᵐᵒᵖ`. This is the
action of the (fully faithful) `ᵐᵒᵖ`-functor on morphisms. -/
@[simps!]
def RingHom.op {α β} [NonAssocSemiring α] [NonAssocSemiring β] :
(α →+* β) ≃ (αᵐᵒᵖ →+* βᵐᵒᵖ) where
toFun f := { AddMonoidHom.mulOp f.toAddMonoidHom, MonoidHom.op f.toMonoidHom with }
invFun f := { AddMonoidHom.mulUnop f.toAddMonoidHom, MonoidHom.unop f.toMonoidHom with }
left_inv _ := rfl
right_inv _ := rfl
#align ring_hom.op RingHom.op
#align ring_hom.op_symm_apply_apply RingHom.op_symm_apply_apply
/-- The 'unopposite' of a ring hom `αᵐᵒᵖ →+* βᵐᵒᵖ`. Inverse to `RingHom.op`. -/
@[simp]
def RingHom.unop {α β} [NonAssocSemiring α] [NonAssocSemiring β] : (αᵐᵒᵖ →+* βᵐᵒᵖ) ≃ (α →+* β) :=
RingHom.op.symm
#align ring_hom.unop RingHom.unop