{-# OPTIONS --cubical --no-import-sorts --safe #-}
module Cubical.Algebra.CommRing.Base where

open import Cubical.Foundations.Prelude
open import Cubical.Foundations.Equiv
open import Cubical.Foundations.Equiv.HalfAdjoint
open import Cubical.Foundations.HLevels
open import Cubical.Foundations.Isomorphism
open import Cubical.Foundations.Univalence
open import Cubical.Foundations.Transport
open import Cubical.Foundations.SIP

open import Cubical.Data.Sigma

open import Cubical.Structures.Axioms
open import Cubical.Algebra.Semigroup
open import Cubical.Algebra.Monoid
open import Cubical.Algebra.AbGroup
open import Cubical.Algebra.Ring.Base

open Iso

private
  variable
     : Level

record IsCommRing {R : Type }
                  (0r 1r : R) (_+_ _·_ : R  R  R) (-_ : R  R) : Type  where

  constructor iscommring

  field
    isRing : IsRing 0r 1r _+_ _·_ -_
    ·-comm : (x y : R)  x · y  y · x

  open IsRing isRing public

record CommRingStr (A : Type ) : Type (ℓ-suc ) where

  constructor commringstr

  field
    0r         : A
    1r         : A
    _+_        : A  A  A
    _·_        : A  A  A
    -_         : A  A
    isCommRing : IsCommRing 0r 1r _+_ _·_ -_

  infix  8 -_
  infixl 7 _·_
  infixl 6 _+_

  open IsCommRing isCommRing public

CommRing : Type (ℓ-suc )
CommRing = TypeWithStr _ CommRingStr

makeIsCommRing : {R : Type } {0r 1r : R} {_+_ _·_ : R  R  R} { -_ : R  R}
                 (is-setR : isSet R)
                 (+-assoc : (x y z : R)  x + (y + z)  (x + y) + z)
                 (+-rid : (x : R)  x + 0r  x)
                 (+-rinv : (x : R)  x + (- x)  0r)
                 (+-comm : (x y : R)  x + y  y + x)
                 (·-assoc : (x y z : R)  x · (y · z)  (x · y) · z)
                 (·-rid : (x : R)  x · 1r  x)
                 (·-rdist-+ : (x y z : R)  x · (y + z)  (x · y) + (x · z))
                 (·-comm : (x y : R)  x · y  y · x)
                IsCommRing 0r 1r _+_ _·_ -_
makeIsCommRing {_+_ = _+_} is-setR +-assoc +-rid +-rinv +-comm ·-assoc ·-rid ·-rdist-+ ·-comm =
  iscommring (makeIsRing is-setR +-assoc +-rid +-rinv +-comm ·-assoc ·-rid
                          x  ·-comm _ _  ·-rid x) ·-rdist-+
                          x y z  ·-comm _ _ ∙∙ ·-rdist-+ z x y ∙∙ λ i  (·-comm z x i) + (·-comm z y i))) ·-comm

makeCommRing : {R : Type } (0r 1r : R) (_+_ _·_ : R  R  R) (-_ : R  R)
               (is-setR : isSet R)
               (+-assoc : (x y z : R)  x + (y + z)  (x + y) + z)
               (+-rid : (x : R)  x + 0r  x)
               (+-rinv : (x : R)  x + (- x)  0r)
               (+-comm : (x y : R)  x + y  y + x)
               (·-assoc : (x y z : R)  x · (y · z)  (x · y) · z)
               (·-rid : (x : R)  x · 1r  x)
               (·-rdist-+ : (x y z : R)  x · (y + z)  (x · y) + (x · z))
               (·-comm : (x y : R)  x · y  y · x)
              CommRing
makeCommRing 0r 1r _+_ _·_ -_ is-setR +-assoc +-rid +-rinv +-comm ·-assoc ·-rid ·-rdist-+ ·-comm =
  _ , commringstr _ _ _ _ _ (makeIsCommRing is-setR +-assoc +-rid +-rinv +-comm ·-assoc ·-rid ·-rdist-+ ·-comm)

CommRing→Ring : CommRing {}  Ring
CommRing→Ring (_ , commringstr _ _ _ _ _ H) = _ , ringstr _ _ _ _ _ (IsCommRing.isRing H)

CommRingEquiv : (R S : CommRing) (e :  R    S )  Type 
CommRingEquiv R S e = RingEquiv (CommRing→Ring R) (CommRing→Ring S) e

CommRingHom : (R S : CommRing)  Type 
CommRingHom R S = RingHom (CommRing→Ring R) (CommRing→Ring S)

module CommRingΣTheory {} where

  open RingΣTheory

  CommRingAxioms : (R : Type ) (s : RawRingStructure R)  Type 
  CommRingAxioms R (_+_ , 1r , _·_) = RingAxioms R (_+_ , 1r , _·_)
                                      × ((x y : R)  x · y  y · x)
  CommRingStructure : Type   Type 
  CommRingStructure = AxiomsStructure RawRingStructure CommRingAxioms

  CommRingΣ : Type (ℓ-suc )
  CommRingΣ = TypeWithStr  CommRingStructure

  CommRingEquivStr : StrEquiv CommRingStructure 
  CommRingEquivStr = AxiomsEquivStr RawRingEquivStr CommRingAxioms

  isPropCommRingAxioms : (R : Type ) (s : RawRingStructure R)
                        isProp (CommRingAxioms R s)
  isPropCommRingAxioms R (_·_ , 0r , _+_) =
    isPropΣ (isPropRingAxioms R (_·_ , 0r , _+_))
            λ { (_ , x , _)  isPropΠ2 λ _ _ 
                  x .IsMonoid.isSemigroup .IsSemigroup.is-set _ _}

  CommRing→CommRingΣ : CommRing  CommRingΣ
  CommRing→CommRingΣ (_ , commringstr _ _ _ _ _ (iscommring G C)) =
    _ , _ , Ring→RingΣ (_ , ringstr _ _ _ _ _ G) .snd .snd , C

  CommRingΣ→CommRing : CommRingΣ  CommRing
  CommRingΣ→CommRing (_ , _ , G , C) =
    _ , commringstr _ _ _ _ _ (iscommring (RingΣ→Ring (_ , _ , G) .snd .RingStr.isRing) C)

  CommRingIsoCommRingΣ : Iso CommRing CommRingΣ
  CommRingIsoCommRingΣ =
    iso CommRing→CommRingΣ CommRingΣ→CommRing  _  refl)  _  refl)

  commRingUnivalentStr : UnivalentStr CommRingStructure CommRingEquivStr
  commRingUnivalentStr = axiomsUnivalentStr _ isPropCommRingAxioms rawRingUnivalentStr

  CommRingΣPath : (R S : CommRingΣ)  (R ≃[ CommRingEquivStr ] S)  (R  S)
  CommRingΣPath = SIP commRingUnivalentStr

  CommRingEquivΣ : (R S : CommRing)  Type 
  CommRingEquivΣ R S = CommRing→CommRingΣ R ≃[ CommRingEquivStr ] CommRing→CommRingΣ S

  CommRingPath : (R S : CommRing)  (Σ[ e   R    S  ] CommRingEquiv R S e)  (R  S)
  CommRingPath R S =
    Σ[ e   R    S  ] CommRingEquiv R S e ≃⟨ isoToEquiv RingIsoΣPath 
    CommRingEquivΣ R S  ≃⟨ CommRingΣPath _ _ 
    CommRing→CommRingΣ R  CommRing→CommRingΣ S
      ≃⟨ isoToEquiv (invIso (congIso CommRingIsoCommRingΣ)) 
    R  S 

CommRingPath : (R S : CommRing {})  (Σ[ e   R    S  ] CommRingEquiv R S e)  (R  S)
CommRingPath = CommRingΣTheory.CommRingPath

isSetCommRing : ((R , str) : CommRing {})  isSet R
isSetCommRing (R , str) = str .CommRingStr.isRing .IsRing.·IsMonoid .IsMonoid.isSemigroup .IsSemigroup.is-set

isPropIsCommRing : {R : Type } (0r 1r : R) (_+_ _·_ : R  R  R) (-_ : R  R)
              isProp (IsCommRing 0r 1r _+_ _·_ -_)
isPropIsCommRing 0r 1r _+_ _·_ -_ (iscommring RR RC) (iscommring SR SC) =
  λ i  iscommring (isPropIsRing _ _ _ _ _ RR SR i)
                   (isPropComm RC SC i)
  where
  isSetR : isSet _
  isSetR = RR .IsRing.·IsMonoid .IsMonoid.isSemigroup .IsSemigroup.is-set

  isPropComm : isProp ((x y : _)  x · y  y · x)
  isPropComm = isPropΠ2 λ _ _  isSetR _ _