include_str!("../input/day08.txt")
}
-#[derive(Debug, Hash, Ord, PartialOrd, Eq, PartialEq, Copy, Clone)]
+#[derive(Debug, Hash, Eq, PartialEq, Copy, Clone)]
struct Point(i32, i32);
+impl Point {
+ fn in_bounds(&self, bound: i32) -> bool {
+ self.0 >= 0 && self.1 >= 0 && self.0 < bound && self.1 < bound
+ }
+
+ fn scale(&self, multiplier: i32) -> Point {
+ Point(self.0 * multiplier, self.1 * multiplier)
+ }
+}
+
impl Add for Point {
type Output = Point;
fn get_nearby_colinear_points(bound: i32, a: Point, b: Point) -> Vec<Point> {
[a - b + a, b - a + b]
.iter()
- .filter(move |Point(i, j)| *i >= 0 && *i < bound && *j >= 0 && *j < bound)
+ .filter(|p| p.in_bounds(bound))
.copied()
.collect()
}
fn get_colinear_points(bound: i32, a: Point, b: Point) -> Vec<Point> {
- let d = a.min(b) - a.max(b);
- let mut p0 = a.min(b);
- while p0.0 >= 0 && p0.1 >= 0 && p0.0 < bound && p0.1 < bound {
- p0 = p0 + d;
- }
- p0 = p0 - d;
- [p0].iter()
+ let d = a - b;
+ [a].iter()
.cycle()
- .zip(0..)
- .map(|(p, n)| *p - Point(n * d.0, n * d.1))
- .take_while(|&Point(i, j)| i >= 0 && j >= 0 && i < bound && j < bound)
+ .zip(-bound..bound)
+ .map(|(p, n)| *p - d.scale(n))
+ .filter(|p| p.in_bounds(bound))
.collect()
}
-fn find_nearby_antinodes(size: i32, pos: &[Point]) -> impl Iterator<Item = Point> + use<'_> {
+fn find_nearby_antinodes(bound: i32, pos: &[Point]) -> impl Iterator<Item = Point> + use<'_> {
pos.iter()
.flat_map(|p| pos.iter().map(move |q| (p, q)))
.filter(|(a, b)| a != b)
- .flat_map(move |(a, b)| get_nearby_colinear_points(size, *a, *b))
+ .flat_map(move |(a, b)| get_nearby_colinear_points(bound, *a, *b))
}
-fn find_antinodes(size: i32, pos: &[Point]) -> impl Iterator<Item = Point> + use<'_> {
+fn find_antinodes(bound: i32, pos: &[Point]) -> impl Iterator<Item = Point> + use<'_> {
pos.iter()
.flat_map(|p| pos.iter().map(move |q| (p, q)))
.filter(|(a, b)| a != b)
- .flat_map(move |(a, b)| get_colinear_points(size, *a, *b))
+ .flat_map(move |(a, b)| get_colinear_points(bound, *a, *b))
}
pub fn part1() {