summaryrefslogtreecommitdiff
path: root/Common/Parsing.lean
diff options
context:
space:
mode:
Diffstat (limited to 'Common/Parsing.lean')
-rw-r--r--Common/Parsing.lean46
1 files changed, 46 insertions, 0 deletions
diff --git a/Common/Parsing.lean b/Common/Parsing.lean
new file mode 100644
index 0000000..a89d785
--- /dev/null
+++ b/Common/Parsing.lean
@@ -0,0 +1,46 @@
+/- This file contains various parsing helpers. Started _after_ Day10. -/
+
+namespace Parsing
+
+structure RectangularGrid (Element : Type) where
+ width : Nat
+ height : Nat
+ elements : Array Element
+ valid : elements.size = width * height
+
+structure RectangularGrid.Coordinate (grid : RectangularGrid Element) where
+ x : Fin grid.width
+ y : Fin grid.height
+
+private def RectangularGrid.Coordinate.toIndex {grid : RectangularGrid Element} (coordinate : grid.Coordinate) : Fin (grid.width * grid.height) :=
+ ⟨coordinate.x + grid.width * coordinate.y,
+ Nat.le_pred_of_lt coordinate.y.isLt
+ |> Nat.mul_le_mul_left grid.width
+ |> Nat.add_le_add_iff_right.mpr
+ |> (Nat.mul_pred grid.width grid.height).subst (motive :=λx↦grid.width * coordinate.y.val + grid.width ≤ x + grid.width)
+ |> (Nat.sub_add_cancel (Nat.le_mul_of_pos_right grid.width (Nat.zero_lt_of_lt coordinate.y.isLt))).subst
+ |> (Nat.add_comm _ _).subst (motive := λx↦x ≤ grid.width*grid.height)
+ |> Nat.le_sub_of_add_le
+ |> Nat.lt_of_lt_of_le coordinate.x.isLt
+ |> λx↦(Nat.add_lt_add_right) x (grid.width * coordinate.y.val)
+ |> (Nat.sub_add_cancel (Nat.le_of_lt ((Nat.mul_lt_mul_left (Nat.zero_lt_of_lt coordinate.x.isLt)).mpr coordinate.y.isLt))).subst⟩
+
+private def RectangularGrid.Coordinate.ofIndex (grid : RectangularGrid Element) (index : Fin (grid.width * grid.height)) : grid.Coordinate :=
+ have : grid.width > 0 :=
+ have := index.isLt
+ match h : grid.width with
+ | Nat.zero => absurd ((Nat.zero_mul grid.height).subst (h.subst (motive := λx↦index<x*grid.height) this)) (Nat.not_lt_zero index.val)
+ | Nat.succ ww => Nat.succ_pos ww
+ {
+ x := ⟨index.val % grid.width, Nat.mod_lt index.val this⟩
+ y := ⟨index.val / grid.width, Nat.div_lt_of_lt_mul index.isLt⟩
+ }
+
+theorem Coordinate.toIndex_inv_fromIndex {grid : RectangularGrid Element} (index : Fin (grid.width * grid.height)) : RectangularGrid.Coordinate.toIndex (RectangularGrid.Coordinate.ofIndex grid index) = index := by
+ simp only [RectangularGrid.Coordinate.toIndex, RectangularGrid.Coordinate.ofIndex, Nat.mod_add_div, Fin.eta]
+
+def RectangularGrid.Get {grid : RectangularGrid Element} (coordinate : grid.Coordinate) : Element :=
+ grid.elements[coordinate.toIndex]'(grid.valid.substr coordinate.toIndex.isLt)
+
+instance : GetElem (RectangularGrid Element) (RectangularGrid.Coordinate grid) Element (λg _ ↦ g = grid) where
+ getElem := λ g c h ↦ g.Get (h▸c)