module Data.Table.Base where
open import Data.Nat
open import Data.Fin
open import Data.Product using (_×_ ; _,_)
open import Data.List as List using (List)
open import Function using (_∘_)
record Table {a} (A : Set a) n : Set a where
  constructor tabulate
  field lookup : Fin n → A
open Table public
module _ {a} {A : Set a} {n : ℕ} where
  rearrange : ∀ {m} → (Fin m → Fin n) → Table A n → Table A m
  rearrange f t = tabulate (lookup t ∘ f)
  head : Table A (ℕ.suc n) → A
  head t = lookup t zero
  tail : Table A (ℕ.suc n) → Table A n
  tail t = tabulate (lookup t ∘ suc)
  uncons : Table A (ℕ.suc n) → A × Table A n
  uncons t = head t , tail t
  toList : Table A n → List A
  toList = List.tabulate ∘ lookup
fromList : ∀ {a} {A : Set a} (xs : List A) → Table A (List.length xs)
fromList = tabulate ∘ List.lookup
module _ {a b} {A : Set a} {B : Set b} where
  foldr : ∀ {n} → (A → B → B) → B → Table A n → B
  foldr {n = zero}  f z t = z
  foldr {n = suc n} f z t = f (head t) (foldr f z (tail t))
  foldl : ∀ {n} → (B → A → B) → B → Table A n → B
  foldl {n = zero}  f z t = z
  foldl {n = suc n} f z t = foldl f (f z (head t)) (tail t)
  map : ∀ {n} → (A → B) → Table A n → Table B n
  map f t = tabulate (f ∘ lookup t)
replicate : ∀ {n a} {A : Set a} → A → Table A n
replicate x = tabulate (λ _ → x)
_⊛_ : ∀ {n a b} {A : Set a} {B : Set b} → Table (A → B) n → Table A n → Table B n
fs ⊛ xs = tabulate λ i → lookup fs i (lookup xs i)