open import 1Lab.Prelude

open import Data.List.Membership
open import Data.Finset.Base
open import Data.Fin.Finite
open import Data.Nat.Base
open import Data.Sum.Base
open import Data.Dec

import 1Lab.Reflection

open 1Lab.Reflection using (Idiom-TC ; lit ; nat ; con₀ ; List ; [] ; _∷_)

module Data.Finset.Properties where

private variable
   ℓ' ℓ'' : Level
  A B C : Type 

∈ᶠˢ-map : (f : A  B) {x : B} (xs : Finset A)  x  mapᶠˢ f xs  ∃[ (y , _)  fibreᵢ f x ] y  xs
∈ᶠˢ-map f {x} xs w = Finset-elim-prop  xs  x  mapᶠˢ f xs  ∃[ (y , _)  fibreᵢ f x ] y  xs)
   w  absurd (¬mem-[] w))
   y ind  ∈ᶠˢ-split  { p  inc ((y , symᵢ p) , hereₛ) }) λ w  case ind w of λ x p h  inc ((x , p) , thereₛ h))
  xs w

map-∈ᶠˢ :  (f : A  B) {x} (xs : Finset A)  x ∈ᶠˢ xs  f x ∈ᶠˢ mapᶠˢ f xs
map-∈ᶠˢ f {x} = ∈ᶠˢ-elim  xs _  f x ∈ᶠˢ map f xs) hereₛ  q  thereₛ)

map-∈ᶠˢ' :  (f : A  B) {x y} (xs : Finset A)  f x  y  x ∈ᶠˢ xs  y ∈ᶠˢ mapᶠˢ f xs
map-∈ᶠˢ' f {x} {y} xs p = ∈ᶠˢ-elim  xs _  y ∈ᶠˢ map f xs) (hereₛ' (Id≃path.from (sym p)))  q  thereₛ) xs

filter-∈ᶠˢ :  {P : A  Type }  d :  {x}  Dec (P x)  {x : A} xs  x ∈ᶠˢ xs  P x  x ∈ᶠˢ filter P xs
filter-∈ᶠˢ {P = P}  d  {x} xs mem px = ∈ᶠˢ-elim  xs _  x ∈ᶠˢ filter P xs)
   {xs}  case d {x} return  p  x ∈ᶠˢ cons-if p x (filter P xs)) of λ { (yes _)  hereₛ ; (no ¬px)  absurd (¬px px)})
   {y} {xs} q ind  case d {y} return  p  x ∈ᶠˢ cons-if p y (filter P xs)) of λ { (yes _)  thereₛ ind ; (no ¬px)  ind })
  xs mem

∈ᶠˢ-filter :  {P : A  Type }  d :  {x}  Dec (P x)  {x : A} xs  x ∈ᶠˢ filter P xs  x  xs ×  P x 
∈ᶠˢ-filter {P = P}  d = d  {x = x} = Finset-elim-prop  xs  x  filter P xs  x  xs ×  P x )
   w  absurd (¬mem-[] w))
   y {xs} ind  case d {y} return  p  x ∈ᶠˢ cons-if p y (filter P xs)  x  (y  xs) ×  P x ) of λ where
    (yes py)  ∈ᶠˢ-split  p  hereₛ' p , inc (substᵢ P (symᵢ p) py)) λ w  case ind w of λ a b  thereₛ a , inc b
    (no ¬py) q  case ind q of λ a b  thereₛ a , inc b)

uncons : (x : A) (xs : Finset A)  x ∈ᶠˢ xs  xs  x  xs
uncons x = Finset-elim-prop _  x  absurd (¬mem-[] x)) λ y {xs} ih  ∈ᶠˢ-split
   { reflᵢ  sym (∷-dup _ _) })
   w  ap (y ∷_) (ih w)  ∷-swap _ _ _)

unionl-∈ᶠˢ : (x : A) (xs ys : Finset A)  x ∈ᶠˢ xs  x ∈ᶠˢ (xs <> ys)
unionl-∈ᶠˢ x xs ys p = ∈ᶠˢ-elim  xs _  x ∈ᶠˢ (xs <> ys)) hereₛ  q x  thereₛ x) xs p

unionr-∈ᶠˢ : (x : A) (xs ys : Finset A)  x ∈ᶠˢ ys  x ∈ᶠˢ (xs <> ys)
unionr-∈ᶠˢ x xs ys p = Finset-elim-prop  xs  x ∈ᶠˢ (xs <> ys)) p  x p  thereₛ p) xs

∈ᶠˢ-union : (x : A) (xs ys : Finset A)  x ∈ᶠˢ (xs <> ys)   (x ∈ᶠˢ xs)  (x ∈ᶠˢ ys) 
∈ᶠˢ-union x = Finset-elim-prop _  ys w  inc (inr w)) λ x {xs} ind ys 
  ∈ᶠˢ-split  p  inc (inl (hereₛ' p)))
    λ w  case ind ys w of λ where
      (inl h)  inc (inl (thereₛ h))
      (inr t)  inc (inr t)

sigma-∈ᶠˢ
  :  { ℓ'} {A : Type } {P : A  Type ℓ'} {x y}
    (f :  x  Finset (P x)) xs
   x ∈ᶠˢ xs  y ∈ᶠˢ f x  (x , y) ∈ᶠˢ sigma xs f
sigma-∈ᶠˢ {P = P} {x = x₀} {y = y₀} f xs mx my = ∈ᶠˢ-elim  xs _  (y : P x₀)  y  f x₀  (x₀ , y)  sigma xs f)
   {xs} y ym  unionl-∈ᶠˢ (x₀ , y) (map (x₀ ,_) (f x₀)) (sigma xs f) (map-∈ᶠˢ (x₀ ,_) (f x₀) ym))
   {y} {xs} q ind py ym  unionr-∈ᶠˢ (x₀ , py) (map (y ,_) (f y)) (sigma xs f) (ind py ym))
  xs mx y₀ my

∈ᶠˢ-sigma
  :  { ℓ'} {A : Type } {P : A  Type ℓ'} {x y} (f :  x  Finset (P x)) xs
   (x , y) ∈ᶠˢ sigma xs f  x  xs × y  f x
∈ᶠˢ-sigma {P = P} {x = x} {y} f = Finset-elim-prop  xs  (x , y) ∈ᶠˢ sigma xs f  x  xs × y  f x)
   w  absurd (¬mem-[] w))
   x' {xs} ind w  case ∈ᶠˢ-union (x , y) (map (x' ,_) (f x')) (sigma xs f) w of λ where
    (inl w)  case ∈ᶠˢ-map (x' ,_) (f x') w of λ px' pf mem 
      let (p , q) = id-Σ (symᵢ pf) in
      Jᵢ  x' p   px' (q : Id-over P p y px')  px' ∈ᶠˢ f x'  (x ∈ᶠˢ (x'  xs)) × (y ∈ᶠˢ f x))
          px''  Jᵢ  px'' _  px''  f x  (x  (x  xs)) × (y  f x)) (hereₛ ,_))
         p px' q mem
    (inr w)  case ind w of λ h1 h2  thereₛ h1 , h2)

intersection-∈ᶠˢ :  _ : Discrete A  (x : A) (xs ys : Finset A)  x  xs  x  ys  x  intersection xs ys
intersection-∈ᶠˢ x xs ys mxs mys = ∈ᶠˢ-elim  xs _  x  intersection xs ys)
   {xs}  caseᵈ x  ys return  p  x  cons-if p x (intersection xs ys)) of λ where
    (yes _)    hereₛ
    (no ¬mys)  absurd (¬mys mys))
   {y} {xs} w q  there-cons-if (holds? _) y x (intersection xs ys) q)
  xs mxs

∈ᶠˢ-intersection :  d : Discrete A  (x : A) (xs ys : Finset A)  x  intersection xs ys  (x  xs) × (x  ys)
∈ᶠˢ-intersection x xs ys w = Finset-elim-prop  xs  x  intersection xs ys  x  xs × x  ys)
   w  absurd (¬mem-[] w))
   y {tl} ind  caseᵈ (y  ys) return  p  x  cons-if p y (intersection tl ys)  x  (y  tl) × x  ys) of λ where
    (yes q)  ∈ᶠˢ-split  p  substᵢ  e  x ∈ᶠˢ (e  tl)) p hereₛ , substᵢ (_∈ᶠˢ ys) (symᵢ p) q) λ w  case ind w of λ a b  thereₛ a , b
    (no ¬p) mem  case ind mem of λ a b  thereₛ a , b)
  xs w

∈ᶠˢ-from-list :  {x : A} {xs}  x ∈ᶠˢ from-list xs   x ∈ₗ xs 
∈ᶠˢ-from-list {xs = []} p = absurd (¬mem-[] p)
∈ᶠˢ-from-list {xs = x  xs} = ∈ᶠˢ-split  p  inc (here p))  w  there <$> ∈ᶠˢ-from-list w)

from-list-∈ᶠˢ :  {x : A} {xs}  x ∈ₗ xs  x ∈ᶠˢ from-list xs
from-list-∈ᶠˢ (here p)  = hereₛ' p
from-list-∈ᶠˢ (there x) = thereₛ (from-list-∈ᶠˢ x)

instance
  Dec-All
    :  {P : A  Type }  _ :  {x}  H-Level (P x) 1   d :  {x}  Dec (P x)  {xs : Finset A}
     Dec (All P xs)
  Dec-All {P = P} {xs} = Finset-elim-prop  xs  Dec (All P xs)) (yes all-nil)
     { x (yes xs)  caseᵈ P x of λ { (yes x)  yes (all-cons x xs) ; (no ¬x)  no  xxs  ¬x (from-all _ xxs hereₛ) )}
       ; _ (no ¬xs)  no  xxs  ¬xs (to-all _  x w  from-all _ xxs (thereₛ w))))
       })
    xs

abstract
  subset→is-union : (xs ys : Finset A)  xs  ys  ys  (xs <> ys)
  subset→is-union = Finset-elim-prop _  ys p  refl) λ x {xs} ih ys sube 
    ys             ≡⟨ uncons x ys (sube x hereₛ) 
    x  ys         ≡⟨ ap (x ∷_) (ih ys λ x x∈xs  sube x (thereₛ x∈xs)) 
    x  (xs <> ys) 

  finset-ext : {xs ys : Finset A}  xs  ys  ys  xs  xs  ys
  finset-ext {xs = xs} {ys} p1 p2 =
    let
      p : xs  (ys <> xs)
      p = subset→is-union ys xs p2

      q : ys  (xs <> ys)
      q = subset→is-union xs ys p1
    in p ∙∙ union-comm ys xs ∙∙ sym q

instance
  Discrete-Finset :  _ : Discrete A   Discrete (Finset A)
  Discrete-Finset {x = xs} {ys} = case holds? (All (_∈ᶠˢ ys) xs × All (_∈ᶠˢ xs) ys) of λ where
    (yes (s1 , s2))  yes $ finset-ext  a  from-all _ s1)  a  from-all _ s2)
    (no ¬sub)        no λ p  ¬sub (to-all xs  a  subst (a ∈_) p) , to-all ys  a  subst (a ∈_) (sym p)))

Finite→finset : Finite A  Finset A
Finite→finset {A = A} = ∥-∥-rec-set (hlevel 2)  l  from-list (Listing.univ l)) const where abstract
  const : (x y : Listing A)  from-list (Listing.univ x)  from-list (Listing.univ y)
  const x y = finset-ext
     a _  from-list-∈ᶠˢ (Listing.find y a))
     a _  from-list-∈ᶠˢ (Listing.find x a))

Finite→finset-has : (x : Finite A) (y : A)  y ∈ᶠˢ Finite→finset x
Finite→finset-has = elim! λ x y  from-list-∈ᶠˢ (Listing.find x y)

abstract
  map-union
    : {A : Type } {B : Type ℓ'} (xs ys : Finset A) (f : A  B)
     map f (xs <> ys)  map f xs <> map f ys
  map-union = Finset-elim-prop _  ys f  refl)  x ih ys f  ap (f x ∷_) (ih ys f))

  map-map
    : {A : Type } {B : Type ℓ'} {C : Type ℓ''} (xs : Finset A) (f : B  C) (g : A  B)
     map f (map g xs)  map (f  g) xs
  map-map = Finset-elim-prop _  f g  refl)  x ih f g  ap (f (g x) ∷_) (ih f g))

delete :  _ : Discrete A   A  Finset A  Finset A
delete x xs = filter (x ≠_) xs

abstract
  cons-delete :  _ : Discrete A  (x : A) (xs : Finset A)  x  xs  x  delete x xs  xs
  cons-delete x xs m = finset-ext to from where
    to :  a  a ∈ᶠˢ (x  delete x xs)  a ∈ᶠˢ xs
    to a = ∈ᶠˢ-split
       a=x  substᵢ (_∈ᶠˢ xs) (symᵢ a=x) m)
       w  ∈ᶠˢ-filter xs w .fst)

    from :  a  a ∈ᶠˢ xs  a ∈ᶠˢ (x  delete x xs)
    from a h with a ≡ᵢ? x
    ... | yes a=x = hereₛ' a=x
    ... | no a≠x  = thereₛ (filter-∈ᶠˢ xs h λ x=a  a≠x (Id≃path.from (sym x=a)))

no-members→is-[] : (a : Finset A) (ha : (e : A)  e  a)  a ≡ᵢ []
no-members→is-[] = Finset-elim-prop _  _  reflᵢ)  x {xs} _ h  absurd (h x hereₛ))

private
  powercons : A  Finset (Finset A)  Finset (Finset A)
  powercons x xs = xs <> map (x ∷_) xs

private abstract
  powercons-dup : (x : A) (xs : Finset (Finset A))  powercons x (powercons x xs)  powercons x xs
  powercons-dup x xs =
    (xs <> map (x ∷_) xs) <>  map (x ∷_) (xs <> map (x ∷_) xs)               ≡⟨ ap! (map-union xs (map (x ∷_) xs) (x ∷_)) 
    (xs <> map (x ∷_) xs) <> (map (x ∷_) xs <>  map (x ∷_) (map (x ∷_) xs) ) ≡⟨ ap! (map-map xs (x ∷_) (x ∷_)  ap  e  map e xs) (funext λ a  ∷-dup x a)) 
    (xs <> map (x ∷_) xs) <>  map (x ∷_) xs <> map (x ∷_) xs                 ≡⟨ ap! (union-idem (map (x ∷_) xs)) 
    (xs <> map (x ∷_) xs) <> map (x ∷_) xs                                     ≡˘⟨ union-assoc xs (map (x ∷_) xs) (map (x ∷_) xs) 
    xs <> map (x ∷_) xs <> map (x ∷_) xs                                       ≡⟨ ap (xs <>_) (union-idem (map (x ∷_) xs)) 
    xs <> map (x ∷_) xs                                                        

  powercons-swap : (x : A) (y : A) (xs : Finset (Finset A))  powercons x (powercons y xs)  powercons y (powercons x xs)
  powercons-swap x y xs =
    let
      p1 =
        (xs <> map (y ∷_) xs) <>  map (x ∷_) (xs <> map (y ∷_) xs)             ≡⟨ ap! (map-union xs (map (y ∷_) xs) (x ∷_)) 
        (xs <> map (y ∷_) xs) <> (map (x ∷_) xs <> map (x ∷_) (map (y ∷_) xs))   ≡˘⟨ union-assoc xs (map (y ∷_) xs) _ 
        xs <> (map (y ∷_) xs <> (map (x ∷_) xs) <> (map (x ∷_) (map (y ∷_) xs))) ≡⟨⟩
        xs <> map (y ∷_) xs <> map (x ∷_) xs <> map (x ∷_) (map (y ∷_) xs)       

      p2 =
        (xs <> map (x ∷_) xs) <>  map (y ∷_) (xs <> map (x ∷_) xs)           ≡⟨ ap! (map-union xs (map (x ∷_) xs) (y ∷_)) 
        (xs <> map (x ∷_) xs) <> (map (y ∷_) xs) <> map (y ∷_) (map (x ∷_) xs) ≡˘⟨ union-assoc xs (map (x ∷_) xs) _ 
        xs <> map (x ∷_) xs <> map (y ∷_) xs <> map (y ∷_) (map (x ∷_) xs)     ≡⟨ ap (xs <>_) (union-assoc (map (x ∷_) xs) (map (y ∷_) xs) _) 
        xs <> ((map (x ∷_) xs <> map (y ∷_) xs)) <> map (y ∷_) (map (x ∷_) xs) ≡⟨ ap (xs <>_) (ap₂ _<>_ (union-comm (map (x ∷_) xs) (map (y ∷_) xs)) (map-map xs (y ∷_) (x ∷_) ∙∙ ap  e  map e xs) (funext (∷-swap y x)) ∙∙ sym (map-map xs (x ∷_) (y ∷_)))) 
        xs <> ((map (y ∷_) xs <> map (x ∷_) xs)) <> map (x ∷_) (map (y ∷_) xs) ≡⟨ ap (xs <>_) (sym (union-assoc (map (y ∷_) xs) (map (x ∷_) xs) _)) 
        xs <> map (y ∷_) xs <> map (x ∷_) xs <> map (x ∷_) (map (y ∷_) xs)     
    in p1  sym p2

powerset : Finset A  Finset (Finset A)
powerset []                = []  []
powerset (x  xs)          = powercons x (powerset xs)
powerset (∷-dup x xs i)    = powercons-dup x (powerset xs) i
powerset (∷-swap x y xs i) = powercons-swap x y (powerset xs) i
powerset (squash x y p q i j) = hlevel 2 (powerset x) (powerset y)  i  powerset (p i))  i  powerset (q i)) i j

∈ᶠˢ-powerset : (xs ys : Finset A)  ys  powerset xs  ys  xs
∈ᶠˢ-powerset = Finset-elim-prop _
   ys  ∈ᶠˢ-split  { reflᵢ a p  p }) λ m  absurd (¬mem-[] m))
  λ x {xs} ih ys mem  case ∈ᶠˢ-union _ (powerset xs) (map (x ∷_) (powerset xs)) mem of λ where
    (inl h) y m  thereₛ (ih ys h y m)
    (inr t) y m  case ∈ᶠˢ-map (x ∷_) (powerset xs) t of λ where
      ys' p n  ∈ᶠˢ-case (substᵢ (y ∈ᶠˢ_) (symᵢ p) m)
        hereₛ'  w  thereₛ (ih ys' n y w))

powerset-∈ᶠˢ :  _ : Discrete A  (xs ys : Finset A)  ys  xs  ys  powerset xs
powerset-∈ᶠˢ = Finset-elim-prop _
   ys sube  hereₛ' (Id≃path.from (finset-ext sube  a m  absurd (¬mem-[] m)))))
  λ x {xs} ih ys sube  caseᵈ x  ys of λ where
    (yes x∈ys) 
      let
        ys' = delete x ys

        s' : delete x ys  xs
        s' a m = let (m' , a≠x) = ∈ᶠˢ-filter ys m in ∈ᶠˢ-case (sube a m')
           p  case a≠x of λ ¬x=a  absurd (¬x=a (Id≃path.to (symᵢ p))))
           w  w)
      in unionr-∈ᶠˢ _ (powerset xs) _ $ map-∈ᶠˢ' (x ∷_) (powerset xs) (cons-delete x ys x∈ys) (ih ys' s')
    (no x∉ys)  unionl-∈ᶠˢ _ (powerset xs) _ $ ih ys λ a m  ∈ᶠˢ-split {P = λ _  a  xs}
       a=x  absurd (x∉ys (substᵢ (_∈ ys) a=x m)))  w  w) (sube a m)

-- List of finite subsets of elements drawn from a given list. Best
-- behaved when the list is nubbed.

powerlist : List A  List (Finset A)
powerlist [] = []  []
powerlist (x  xs) using ppowerlist xs =
  map (x ∷_) p <> p

-- If a finset belongs to the powerlist of a nubbed list, then it's a
-- subset of the list. The list must be nubbed if we want to get a
-- proper index `e ∈ xs` instead of the weaker ∥ e ∈ xs ∥.

member-powerlist
  : {xs : List A} (hxs : is-nubbed xs) {ys : Finset A}
   ys ∈ₗ powerlist xs  (e : A)  e  ys  e  xs
member-powerlist {xs = []} hxs {ys} (here reflᵢ) e me = absurd (¬mem-[] me)
member-powerlist {xs = x  xs} hxxs {ys} mys e me
  using (_ , hxs)uncons-is-nubbed hxxs
  with member-++-view _ (powerlist xs) mys

... | inl (a , _) with ((ays' , reflᵢ) , b)member-map (x ∷_) (powerlist xs) a =
  ∈ᶠˢ-case  prop-instance (hxxs _)  me here λ w 
    there (member-powerlist hxs b e w)

... | inr (a , _) = there (member-powerlist hxs a e me)

-- Conversely, if we know that a finset ys is a subset of a given list
-- xs, we can find an index for it in the powerlist of xs as long as we
-- can decide memberhsip in ys. This lets us find which 'branches' of the
-- powerlist the finset takes.

powerlist-member
  :  _ : Discrete A  (xs : List A) (ys : Finset A)
   ((e : A)  e  ys  e  xs)
   ys ∈ₗ powerlist xs
powerlist-member [] ys ys⊆[] = here $ no-members→is-[] ys λ e he 
  case ys⊆[] e he of λ ()

powerlist-member {A = A} (x  xs) ys ys⊆xxs with holds? (x ∈ᶠˢ ys)
... | yes x∈ys =
  ++-memberₗ {ys = powerlist xs} $ map-member' (x ∷_) (powerlist xs) let
    drop : (e : A)  e ∈ᶠˢ delete x ys  e ∈ₗ xs
    drop e he = let (m , x≠e) = ∈ᶠˢ-filter ys he in case ys⊆xxs e m of λ where
      (here reflᵢ)  absurd (case x≠e of λ f  f refl)
      (there x)     x
  in (delete x ys , Id≃path.from (cons-delete x ys x∈ys))
   , powerlist-member xs (delete x ys) drop

... | no  x∉ys =
  ++-memberᵣ {ys = powerlist xs} let
    drop : (e : A)  e  ys  e  xs
    drop e he = case ys⊆xxs e he of λ where
      (here p)   absurd (x∉ys (substᵢ (_∈ᶠˢ ys) p he))
      (there x)  x
  in powerlist-member xs ys drop

powerlist-is-nubbed :  _ : Discrete A  (xs : List A)  is-nubbed xs  is-nubbed (powerlist xs)
powerlist-is-nubbed [] _ ys (here p) (here p') = ap here prop!
powerlist-is-nubbed {A = A} (x  xs) hxxs =
  ++-is-nubbed
    (map-is-nubbed (x ∷_)
       b f f' p q  Σ-prop-path! (work b (f , p)  sym (work b (f' , q))))
      hpxs)
    hpxs disj
  where
  open Σ (uncons-is-nubbed hxxs) renaming (fst to x∉xs ; snd to hxs)
  hpxs = powerlist-is-nubbed xs hxs

  disj : (ys : Finset A) (h : ys ∈ₗ map (x ∷_) (powerlist xs))  ys  powerlist xs
  disj ys hx ¬hx with ((ys' , reflᵢ) , _)member-map (x ∷_) (powerlist xs) hx =
    x∉xs (member-powerlist hxs ¬hx x hereₛ)

  work
    : (ys : Finset A) (zs : Σ[ f  fibreᵢ (x ∷_) ys ] (f .fst ∈ₗ powerlist xs))
     zs .fst .fst  delete x ys
  work ys ((zs , reflᵢ) , pxs) = finset-ext to from where
    from : (a : A)  a  delete x (x  zs)  a  zs
    from a m using (m' , a≠x)∈ᶠˢ-filter ys m = ∈ᶠˢ-case m'
       { reflᵢ  case a≠x of λ a≠a  absurd (a≠a refl) })
       w  w)

    to : (a : A)  a  zs  a  delete x ys
    to a azs with holds? (x  x)
    ... | yes x≠x = absurd (x≠x refl)
    ... | no  x=x  =
      let a∈xs = member-powerlist hxs pxs a azs
        in filter-∈ᶠˢ zs azs λ x=a  x∉xs (subst (_∈ xs) (sym x=a) a∈xs)

module _  l : Listing A  where instance
  private
    module l = Listing l
    instance _ = Listing→Discrete l
  open Listing

  Listing-Finset : Listing (Finset A)
  Listing-Finset .univ = powerlist l.univ
  Listing-Finset .has-member a .centre = powerlist-member l.univ a λ _ _  l.find _
  Listing-Finset .has-member a .paths  = powerlist-is-nubbed l.univ  _  is-contr→is-prop (l.has-member _)) a _