806 lines
25 KiB
Rust
806 lines
25 KiB
Rust
/*
|
|
THIS PLACE IS NOT A PLACE OF HONOR...
|
|
NO HIGHLY ESTEEMED DEED IS COMMEMORATED HERE...
|
|
NOTHING VALUED IS HERE.
|
|
|
|
THE DANGER IS PRESENT IN YOUR TIME AS IT WAS IN OURS.
|
|
|
|
THE DANGER IS UNLEASHED ONLY IF YOU SUBSTANTIALLY DISTURB THIS FILE...
|
|
THIS FILE IS BEST LEFT SHUNNED AND UNINHABITED.
|
|
*/
|
|
|
|
|
|
use std::collections::HashSet;
|
|
use std::collections::HashMap;
|
|
use std::env;
|
|
use std::fs;
|
|
|
|
use memoize::memoize;
|
|
|
|
use crate::Dir::{North, South, East, West, Stay };
|
|
|
|
|
|
type Coord = (i32, i32);
|
|
#[derive(Clone, Copy, Eq, PartialEq, Hash, Debug, PartialOrd, Ord)]
|
|
enum Dir { North, South, East, West, Stay }
|
|
|
|
fn step(start: &Coord, d: Dir, steps: i32) -> Coord {
|
|
match d {
|
|
North => (start.0, start.1 - steps),
|
|
South => (start.0, start.1 + steps),
|
|
East => (start.0 + steps, start.1),
|
|
West => (start.0 - steps, start.1),
|
|
Stay => *start,
|
|
}
|
|
}
|
|
|
|
fn walk_path(start: &Coord, path: &[Dir]) -> Coord {
|
|
let mut here = *start;
|
|
for d in path {
|
|
here = step(&here, *d, 1);
|
|
}
|
|
return here;
|
|
}
|
|
|
|
fn main() {
|
|
println!("Hello, AoC day 21!");
|
|
|
|
let args: Vec<String> = env::args().collect();
|
|
if args.len() != 2 {
|
|
println!("wrong number of arguments!");
|
|
std::process::exit(1);
|
|
}
|
|
|
|
let file_path = &args[1];
|
|
|
|
let file = fs::read_to_string(file_path)
|
|
.expect("should be able to read the file");
|
|
|
|
let keypad_door: HashMap<char, Coord> = HashMap::from([
|
|
('7', (0,0)), ('8', (1,0)), ('9', (2,0)),
|
|
('4', (0,1)), ('5', (1,1)), ('6', (2,1)),
|
|
('1', (0,2)), ('2', (1,2)), ('3', (2,2)),
|
|
('0', (1,3)), ('A', (2,3)),
|
|
]);
|
|
let mut door_map: HashMap<Coord, char> = HashMap::new();
|
|
for (key, loc) in &keypad_door { door_map.insert(*loc, *key); }
|
|
let mut directions_matrix: HashMap<(Coord, Coord), Vec<(Dir, i32)>> = HashMap::new();
|
|
for (_this, here) in &keypad_door {
|
|
for (_other, there) in &keypad_door {
|
|
let mut v = vec![];
|
|
if there.0 > here.0 {
|
|
v.push((East, there.0 - here.0));
|
|
}
|
|
if there.0 < here.0 {
|
|
v.push((West, here.0 - there.0));
|
|
}
|
|
if there.1 > here.1 {
|
|
v.push((South, there.1 - here.1));
|
|
}
|
|
if there.1 < here.1 {
|
|
v.push((North, here.1 - there.1));
|
|
}
|
|
if there == here {
|
|
v.push((Stay, 0));
|
|
}
|
|
directions_matrix.insert((*here, *there), v);
|
|
}
|
|
}
|
|
|
|
|
|
let keypad_arrows: HashMap<Dir, Coord> = HashMap::from([
|
|
(North, (1,0)), (Stay, (2,0)),
|
|
(West, (0,1)), (South, (1,1)), (East, (2,1)),
|
|
]);
|
|
let mut keypad_map: HashMap<Coord, Dir> = HashMap::new();
|
|
for (key, loc) in &keypad_arrows { keypad_map.insert(*loc, *key); }
|
|
let mut keypad_directions_matrix = HashMap::new();
|
|
for (_this, here) in &keypad_arrows {
|
|
for (_other, there) in &keypad_arrows {
|
|
let mut v = vec![];
|
|
if there.0 > here.0 {
|
|
v.push((East, there.0 - here.0));
|
|
}
|
|
if there.0 < here.0 {
|
|
v.push((West, here.0 - there.0));
|
|
}
|
|
if there.1 > here.1 {
|
|
v.push((South, there.1 - here.1));
|
|
}
|
|
if there.1 < here.1 {
|
|
v.push((North, here.1 - there.1));
|
|
}
|
|
if there == here {
|
|
v.push((Stay, 0));
|
|
}
|
|
keypad_directions_matrix.insert((*here, *there), v);
|
|
}
|
|
}
|
|
|
|
let mut sum = 0;
|
|
for line in file.lines() {
|
|
let door_code: Vec<char> = line.chars().collect();
|
|
let num: i32 = line[..line.len()-1].parse().expect("failed to parse door code as number");
|
|
let mut complexity: i64 = 100000000000;
|
|
|
|
for x in &door_code {
|
|
print!("{}", x);
|
|
}
|
|
print!("\n");
|
|
|
|
let options_door = build_options(
|
|
&directions_matrix,
|
|
&keypad_door,
|
|
&door_code,
|
|
'A');
|
|
for arrow_code_1 in sequences(options_door, &door_map, keypad_door[&'A']) {
|
|
print_code(&arrow_code_1);
|
|
|
|
let options_arrow_1 = build_options(
|
|
&keypad_directions_matrix,
|
|
&keypad_arrows,
|
|
&arrow_code_1,
|
|
Stay);
|
|
for arrow_code_2 in sequences(options_arrow_1, &keypad_map, keypad_arrows[&Stay]) {
|
|
print_code(&arrow_code_2);
|
|
|
|
let options_arrow_2 = build_options(
|
|
&keypad_directions_matrix,
|
|
&keypad_arrows,
|
|
&arrow_code_2,
|
|
Stay);
|
|
for arrow_code_3 in sequences(options_arrow_2, &keypad_map, keypad_arrows[&Stay]) {
|
|
complexity = complexity.min(arrow_code_3.len() as i64);
|
|
print_code(&arrow_code_3);
|
|
}
|
|
}
|
|
}
|
|
println!("shortest sequence is {} long", complexity);
|
|
sum += complexity * (num as i64);
|
|
}
|
|
|
|
println!("the sum of the complexity scores is {}", sum);
|
|
|
|
println!("okay let's play around");
|
|
let code_0 = "029A";
|
|
println!("starting sequence is {code_0}");
|
|
let mut seed = 0;
|
|
let code_1 = &expand(
|
|
&code_0.chars().collect(),
|
|
&door_map,
|
|
&keypad_door,
|
|
'A',
|
|
&mut seed);
|
|
print_code(code_1);
|
|
let code_2 = &expand(
|
|
&code_1,
|
|
&keypad_map,
|
|
&keypad_arrows,
|
|
Stay,
|
|
&mut seed);
|
|
print_code(code_2);
|
|
let code_3 = &expand(
|
|
&code_2,
|
|
&keypad_map,
|
|
&keypad_arrows,
|
|
Stay,
|
|
&mut seed);
|
|
print_code(code_3);
|
|
let code_4 = &expand(
|
|
&code_3,
|
|
&keypad_map,
|
|
&keypad_arrows,
|
|
Stay,
|
|
&mut seed);
|
|
print_code(code_4);
|
|
seed = 0b11111111111111111111111;
|
|
let code_1 = &expand(
|
|
&code_0.chars().collect(),
|
|
&door_map,
|
|
&keypad_door,
|
|
'A',
|
|
&mut seed);
|
|
print_code(code_1);
|
|
let code_2 = &expand(
|
|
&code_1,
|
|
&keypad_map,
|
|
&keypad_arrows,
|
|
Stay,
|
|
&mut seed);
|
|
print_code(code_2);
|
|
let code_3 = &expand(
|
|
&code_2,
|
|
&keypad_map,
|
|
&keypad_arrows,
|
|
Stay,
|
|
&mut seed);
|
|
print_code(code_3);
|
|
let code_4 = &expand(
|
|
&code_3,
|
|
&keypad_map,
|
|
&keypad_arrows,
|
|
Stay,
|
|
&mut seed);
|
|
print_code(code_4);
|
|
|
|
println!("\n\n\nokay, i think i got it. now let's do the memoized solution.");
|
|
|
|
|
|
let mut bests: HashMap<(Dir, Dir), Vec<Dir>> = HashMap::new();
|
|
for prev in [Stay, East, West, North, South] {
|
|
for next in [Stay, East, West, North, South] {
|
|
let four_deep = shortest_seq(
|
|
keypad_arrows[&prev],
|
|
keypad_arrows[&next],
|
|
7,
|
|
vec![(0,3)]);
|
|
let mut result = four_deep;
|
|
for _ in 0..6 {
|
|
result = execute_code(&result, &keypad_map, keypad_arrows[&Stay]);
|
|
}
|
|
bests.insert((prev, next), result);
|
|
}
|
|
}
|
|
for (key, value) in &bests {
|
|
print!("{key:?}:");
|
|
print_code(&value);
|
|
}
|
|
|
|
let mut sum = 0;
|
|
for line in file.lines() {
|
|
let num: i32 = line[..line.len()-1].parse().expect("failed to parse door code as number");
|
|
|
|
println!("looking at code {line}, generating:");
|
|
print_code(&thru_keypads(line, 2));
|
|
print_code(&thru_keypads(line, 3));
|
|
print_code(&thru_keypads(line, 4));
|
|
|
|
let extra = 25;
|
|
let mut length = 0;
|
|
let mut prev = 'A';
|
|
for d in line.chars() {
|
|
length += length_calc_numpad(prev, d, 1+extra);
|
|
prev = d;
|
|
}
|
|
println!("length should be {length}");
|
|
|
|
sum += length * (num as i64);
|
|
}
|
|
|
|
println!("the answer is {sum}");
|
|
}
|
|
|
|
fn length_calc_numpad(prev: char, next: char, depth: i32) -> i64 {
|
|
let optimals: HashMap<(char, char), Vec<Dir>> = HashMap::from([
|
|
(('A', 'A'), vec![Stay]),
|
|
(('A', '0'), vec![West, Stay]),
|
|
(('A', '1'), vec![North, West, West, Stay]),
|
|
(('A', '2'), vec![West, North, Stay]),
|
|
(('A', '3'), vec![North, Stay]),
|
|
(('A', '4'), vec![North, North, West, West, Stay]),
|
|
(('A', '5'), vec![West, North, North, Stay]),
|
|
(('A', '6'), vec![North, North, Stay]),
|
|
(('A', '7'), vec![North, North, North, West, West, Stay]),
|
|
(('A', '8'), vec![West, North, North, North, Stay]),
|
|
(('A', '9'), vec![North, North, North, Stay]),
|
|
|
|
(('0', 'A'), vec![East, Stay]),
|
|
(('0', '0'), vec![Stay]),
|
|
(('0', '1'), vec![North, West, Stay]),
|
|
(('0', '2'), vec![North, Stay]),
|
|
(('0', '3'), vec![North, East, Stay]),
|
|
(('0', '4'), vec![North, North, West, Stay]),
|
|
(('0', '5'), vec![North, North, Stay]),
|
|
(('0', '6'), vec![North, North, East, Stay]),
|
|
(('0', '7'), vec![North, North, North, West, Stay]),
|
|
(('0', '8'), vec![North, North, North, Stay]),
|
|
(('0', '9'), vec![North, North, North, East, Stay]),
|
|
|
|
(('1', 'A'), vec![East, East, South, Stay]),
|
|
(('1', '0'), vec![East, South, Stay]),
|
|
(('1', '1'), vec![Stay]),
|
|
(('1', '2'), vec![East, Stay]),
|
|
(('1', '3'), vec![East, East, Stay]),
|
|
(('1', '4'), vec![North, Stay]),
|
|
(('1', '5'), vec![North, East, Stay]),
|
|
(('1', '6'), vec![North, East, East, Stay]),
|
|
(('1', '7'), vec![North, North, Stay]),
|
|
(('1', '8'), vec![North, North, East, Stay]),
|
|
(('1', '9'), vec![North, North, East, East, Stay]),
|
|
|
|
(('2', 'A'), vec![South, East, Stay]),
|
|
(('2', '0'), vec![South, Stay]),
|
|
(('2', '1'), vec![West, Stay]),
|
|
(('2', '2'), vec![Stay]),
|
|
(('2', '3'), vec![East, Stay]),
|
|
(('2', '4'), vec![West, North, Stay]),
|
|
(('2', '5'), vec![North, Stay]),
|
|
(('2', '6'), vec![East, North, Stay]),
|
|
(('2', '7'), vec![West, North, North, Stay]),
|
|
(('2', '8'), vec![North, North, Stay]),
|
|
(('2', '9'), vec![East, North, North, Stay]),
|
|
|
|
(('3', 'A'), vec![South, Stay]),
|
|
(('3', '0'), vec![West, South, Stay]),
|
|
(('3', '1'), vec![West, West, Stay]),
|
|
(('3', '2'), vec![West, Stay]),
|
|
(('3', '3'), vec![Stay]),
|
|
(('3', '4'), vec![West, West, North, Stay]),
|
|
(('3', '5'), vec![West, North, Stay]),
|
|
(('3', '6'), vec![North, Stay]),
|
|
(('3', '7'), vec![West, West, North, North, Stay]),
|
|
(('3', '8'), vec![West, North, Stay]),
|
|
(('3', '9'), vec![North, North, Stay]),
|
|
|
|
(('4', 'A'), vec![East, East, South, South, Stay]),
|
|
(('4', '0'), vec![East, South, South, Stay]),
|
|
(('4', '1'), vec![South, Stay]),
|
|
(('4', '2'), vec![South, East, Stay]),
|
|
(('4', '3'), vec![South, East, East, Stay]),
|
|
(('4', '4'), vec![Stay]),
|
|
(('4', '5'), vec![East, Stay]),
|
|
(('4', '6'), vec![East, East, Stay]),
|
|
(('4', '7'), vec![North, Stay]),
|
|
(('4', '8'), vec![North, East, Stay]),
|
|
(('4', '9'), vec![North, East, East, Stay]),
|
|
|
|
(('5', 'A'), vec![South, South, East, Stay]),
|
|
(('5', '0'), vec![South, South, Stay]),
|
|
(('5', '1'), vec![West, South, Stay]),
|
|
(('5', '2'), vec![South, Stay]),
|
|
(('5', '3'), vec![South, East, Stay]),
|
|
(('5', '4'), vec![West, Stay]),
|
|
(('5', '5'), vec![Stay]),
|
|
(('5', '6'), vec![East, Stay]),
|
|
(('5', '7'), vec![West, North, Stay]),
|
|
(('5', '8'), vec![North, Stay]),
|
|
(('5', '9'), vec![North, East, Stay]),
|
|
|
|
(('6', 'A'), vec![South, South, Stay]),
|
|
(('6', '0'), vec![West, South, South, Stay]),
|
|
(('6', '1'), vec![West, West, South, Stay]),
|
|
(('6', '2'), vec![West, South, Stay]),
|
|
(('6', '3'), vec![South, Stay]),
|
|
(('6', '4'), vec![West, West, Stay]),
|
|
(('6', '5'), vec![West, Stay]),
|
|
(('6', '6'), vec![Stay]),
|
|
(('6', '7'), vec![West, West, North, Stay]),
|
|
(('6', '8'), vec![West, North, Stay]),
|
|
(('6', '9'), vec![North, Stay]),
|
|
|
|
(('7', 'A'), vec![East, East, South, South, South, Stay]),
|
|
(('7', '0'), vec![East, South, South, South, Stay]),
|
|
(('7', '1'), vec![South, South, Stay]),
|
|
(('7', '2'), vec![South, South, East, Stay]),
|
|
(('7', '3'), vec![South, South, East, East, Stay]),
|
|
(('7', '4'), vec![South, Stay]),
|
|
(('7', '5'), vec![South, East, Stay]),
|
|
(('7', '6'), vec![South, East, East, Stay]),
|
|
(('7', '7'), vec![Stay]),
|
|
(('7', '8'), vec![East, Stay]),
|
|
(('7', '9'), vec![East, East, Stay]),
|
|
|
|
(('8', 'A'), vec![South, South, South, East, Stay]),
|
|
(('8', '0'), vec![South, South, South, Stay]),
|
|
(('8', '1'), vec![West, South, South, Stay]),
|
|
(('8', '2'), vec![South, South, Stay]),
|
|
(('8', '3'), vec![South, South, East, Stay]),
|
|
(('8', '4'), vec![West, South, Stay]),
|
|
(('8', '5'), vec![South, Stay]),
|
|
(('8', '6'), vec![South, East, Stay]),
|
|
(('8', '7'), vec![West, West, Stay]),
|
|
(('8', '8'), vec![Stay]),
|
|
(('8', '9'), vec![East, Stay]),
|
|
|
|
(('9', 'A'), vec![South, South, South, Stay]),
|
|
(('9', '0'), vec![West, South, South, South, Stay]),
|
|
(('9', '1'), vec![West, West, South, South, Stay]),
|
|
(('9', '2'), vec![West, South, South, Stay]),
|
|
(('9', '3'), vec![South, South, Stay]),
|
|
(('9', '4'), vec![West, West, South, Stay]),
|
|
(('9', '5'), vec![West, South, Stay]),
|
|
(('9', '6'), vec![South, Stay]),
|
|
(('9', '7'), vec![West, West, Stay]),
|
|
(('9', '8'), vec![West, Stay]),
|
|
(('9', '9'), vec![Stay]),
|
|
]);
|
|
|
|
if depth == 0 {
|
|
return 1;
|
|
} else {
|
|
let mut p = Stay;
|
|
let mut len = 0;
|
|
for d in &optimals[&(prev, next)] {
|
|
len += length_calc(p, *d, depth - 1);
|
|
p = *d;
|
|
}
|
|
return len;
|
|
}
|
|
}
|
|
#[memoize]
|
|
fn length_calc(prev: Dir, next: Dir, depth: i32) -> i64 {
|
|
let optimals: HashMap<(Dir, Dir), Vec<Dir>> = HashMap::from([
|
|
((Stay, Stay), vec![Stay]),
|
|
((Stay, North), vec![West, Stay]),
|
|
((Stay, East), vec![South, Stay]),
|
|
((Stay, South), vec![West, South, Stay]),
|
|
((Stay, West), vec![South, West, West, Stay]),
|
|
|
|
((North, Stay), vec![East, Stay]),
|
|
((North, North), vec![Stay]),
|
|
((North, South), vec![South, Stay]),
|
|
((North, East), vec![South, East, Stay]),
|
|
((North, West), vec![South, West, Stay]),
|
|
|
|
((South, Stay), vec![North, East, Stay]),
|
|
((South, North), vec![North, Stay]),
|
|
((South, South), vec![Stay]),
|
|
((South, East), vec![East, Stay]),
|
|
((South, West), vec![West, Stay]),
|
|
|
|
((East, Stay), vec![North, Stay]),
|
|
((East, North), vec![West, North, Stay]),
|
|
((East, South), vec![West, Stay]),
|
|
((East, East), vec![Stay]),
|
|
((East, West), vec![West, West, Stay]),
|
|
|
|
((West, Stay), vec![East, East, North, Stay]),
|
|
((West, North), vec![East, North, Stay]),
|
|
((West, South), vec![East, Stay]),
|
|
((West, East), vec![East, East, Stay]),
|
|
((West, West), vec![Stay]),
|
|
]);
|
|
|
|
if depth == 0 {
|
|
return 1;
|
|
} else {
|
|
let mut p = Stay;
|
|
let mut len = 0;
|
|
for d in &optimals[&(prev, next)] {
|
|
len += length_calc(p, *d, depth - 1);
|
|
p = *d;
|
|
}
|
|
return len;
|
|
}
|
|
}
|
|
|
|
|
|
fn expandaband(sequence: Vec<Dir>, optimals: &HashMap<(Dir, Dir), Vec<Dir>>) -> Vec<Dir> {
|
|
let mut output = vec![];
|
|
let mut prev = Stay;
|
|
for dir in sequence {
|
|
output.extend(optimals.get(&(prev, dir)).unwrap());
|
|
prev = dir;
|
|
}
|
|
return output;
|
|
}
|
|
fn thru_keypads(code: &str, keypads: i32) -> Vec<Dir> {
|
|
let keypad_door: HashMap<char, Coord> = HashMap::from([
|
|
('7', (0,0)), ('8', (1,0)), ('9', (2,0)),
|
|
('4', (0,1)), ('5', (1,1)), ('6', (2,1)),
|
|
('1', (0,2)), ('2', (1,2)), ('3', (2,2)),
|
|
('0', (1,3)), ('A', (2,3)),
|
|
]);
|
|
let mut prev = 'A';
|
|
let mut output = vec![];
|
|
for c in code.chars() {
|
|
//println!("navigating from {prev} to {c}");
|
|
let shortest = shortest_seq(
|
|
keypad_door[&prev],
|
|
keypad_door[&c],
|
|
keypads-1,
|
|
vec![(0,3)]);
|
|
output.extend(shortest);
|
|
prev = c;
|
|
}
|
|
return output;
|
|
}
|
|
|
|
#[memoize]
|
|
fn shortest_seq(
|
|
start: Coord,
|
|
end: Coord,
|
|
keypads: i32,
|
|
blacklist: Vec<Coord>,
|
|
) -> Vec<Dir> {
|
|
// starting at start what is the shortest sequence to navigate to and press end,
|
|
// through many keypads?
|
|
let keypad: HashMap<Dir, Coord> = HashMap::from([
|
|
(North, (1,0)), (Stay, (2,0)),
|
|
(West, (0,1)), (South, (1,1)), (East, (2,1)),
|
|
]);
|
|
|
|
let paths: Vec<_> = all_paths(start, end)
|
|
.into_iter()
|
|
.filter(|p| traversal_avoids(start, p, &blacklist))
|
|
.collect();
|
|
//println!("with {keypads} left");
|
|
|
|
if keypads == 1 {
|
|
let mut output = vec![];
|
|
output.extend(paths[0].clone());
|
|
output.push(Stay);
|
|
return output;
|
|
} else {
|
|
let mut next_keypad_paths = vec![];
|
|
for path in paths {
|
|
let mut longer = vec![];
|
|
let mut prev = keypad[&Stay];
|
|
for dir in path {
|
|
longer.extend(
|
|
shortest_seq(
|
|
prev,
|
|
keypad[&dir],
|
|
keypads-1,
|
|
vec![(0,0)]
|
|
)
|
|
);
|
|
prev = keypad[&dir];
|
|
}
|
|
longer.extend(
|
|
shortest_seq(
|
|
prev,
|
|
keypad[&Stay],
|
|
keypads-1,
|
|
vec![(0,0)]
|
|
)
|
|
);
|
|
next_keypad_paths.push(longer);
|
|
}
|
|
let mut shortest = &next_keypad_paths[0];
|
|
for path in &next_keypad_paths {
|
|
if path.len() < shortest.len() {
|
|
shortest = path;
|
|
}
|
|
}
|
|
return shortest.to_vec();
|
|
}
|
|
}
|
|
|
|
fn traversal_avoids(start: Coord, path: &Vec<Dir>, blacklist: &Vec<Coord>) -> bool {
|
|
let mut here = start;
|
|
let mut hit = blacklist.contains(&here);
|
|
for d in path {
|
|
here = step(&here, *d, 1);
|
|
hit = hit || blacklist.contains(&here);
|
|
}
|
|
return !hit;
|
|
}
|
|
|
|
fn all_paths(prev: Coord, next: Coord) -> Vec<Vec<Dir>> {
|
|
let mut out = vec![];
|
|
|
|
match (next.0-prev.0, next.1-prev.1) {
|
|
(0, 0) => out.push(vec![]),
|
|
(mut x, 0) => {
|
|
let mut horizontal = vec![];
|
|
while x > 0 { horizontal.push(East); x-=1; }
|
|
while x < 0 { horizontal.push(West); x+=1; }
|
|
out.push(horizontal);
|
|
},
|
|
(0, mut y) => {
|
|
let mut vertical = vec![];
|
|
while y > 0 { vertical.push(South); y-=1; }
|
|
while y < 0 { vertical.push(North); y+=1; }
|
|
out.push(vertical);
|
|
},
|
|
(_x, _y) => {
|
|
for p1 in all_paths(prev, (next.0, prev.1)) {
|
|
for p2 in all_paths((next.0, prev.1), next) {
|
|
let mut p12 = vec![];
|
|
p12.extend(p1.clone());
|
|
p12.extend(p2);
|
|
out.push(p12);
|
|
}
|
|
}
|
|
for p1 in all_paths(prev, (prev.0, next.1)) {
|
|
for p2 in all_paths((prev.0, next.1), next) {
|
|
let mut p12 = vec![];
|
|
p12.extend(p1.clone());
|
|
p12.extend(p2);
|
|
out.push(p12);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return out;
|
|
}
|
|
|
|
fn expand<T: std::cmp::Eq + std::hash::Hash>(
|
|
sequence: &Vec<T>,
|
|
map: &HashMap<Coord, T>,
|
|
keypad: &HashMap<T, Coord>,
|
|
start: T,
|
|
seed: &mut u128
|
|
) -> Vec<Dir> {
|
|
let mut here = keypad[&start];
|
|
let mut output = Vec::new();
|
|
for t in sequence {
|
|
let there = keypad[&t];
|
|
let diff = (there.0 - here.0, there.1 - here.1);
|
|
|
|
output.extend(build_path(diff, seed));
|
|
|
|
here = there;
|
|
}
|
|
return output;
|
|
}
|
|
|
|
fn build_path(diff: Coord, seed: &mut u128) -> Vec<Dir> {
|
|
let mut output = vec![];
|
|
match diff {
|
|
(0, 0) => output.push(Stay),
|
|
(x, 0) => {
|
|
for _ in 0..x.abs() {
|
|
if x < 0 { output.push(West); }
|
|
if x > 0 { output.push(East); }
|
|
}
|
|
output.push(Stay);
|
|
},
|
|
(0, y) => {
|
|
for _ in 0..y.abs() {
|
|
if y < 0 { output.push(North); }
|
|
if y > 0 { output.push(South); }
|
|
}
|
|
output.push(Stay);
|
|
},
|
|
(x, y) => {
|
|
if *seed & 1 == 0 {
|
|
for _ in 0..y.abs() {
|
|
if y < 0 { output.push(North); }
|
|
if y > 0 { output.push(South); }
|
|
}
|
|
for _ in 0..x.abs() {
|
|
if x < 0 { output.push(West); }
|
|
if x > 0 { output.push(East); }
|
|
}
|
|
} else {
|
|
for _ in 0..x.abs() {
|
|
if x < 0 { output.push(West); }
|
|
if x > 0 { output.push(East); }
|
|
}
|
|
for _ in 0..y.abs() {
|
|
if y < 0 { output.push(North); }
|
|
if y > 0 { output.push(South); }
|
|
}
|
|
}
|
|
output.push(Stay);
|
|
*seed = *seed >> 1;
|
|
}
|
|
}
|
|
return output;
|
|
}
|
|
|
|
|
|
|
|
fn sequences<T>(options: Vec<Vec<(Dir, i32)>>, map: &HashMap<Coord, T>, start: Coord) -> Vec<Vec<Dir>> {
|
|
let mut out: Vec<(Vec<Dir>, Coord, Dir)> = vec![(vec![], start, Stay)];
|
|
|
|
for choices in options {
|
|
if choices.len() == 1 {
|
|
let (direction, times) = choices[0];
|
|
for (walk, here, last) in out.iter_mut() {
|
|
for _ in 0..times {
|
|
walk.push(direction);
|
|
*here = step(&here, direction, 1);
|
|
}
|
|
|
|
if direction != Stay {
|
|
*last = direction;
|
|
}
|
|
}
|
|
}
|
|
|
|
else if choices.len() == 2 {
|
|
let mut new_out = vec![];
|
|
|
|
let (d1, t1) = choices[0];
|
|
let (d2, t2) = choices[1];
|
|
|
|
for (walk, here, last) in &mut out {
|
|
let end = step(&step(&here, d1, t1), d2, t2);
|
|
|
|
let candidate_here = step(&here, d1, t1);
|
|
let candidate_there = step(&here, d2, t2);
|
|
|
|
if d1 == *last && map.contains_key(&candidate_here) {
|
|
for _ in 0..t1 { walk.push(d1); }
|
|
for _ in 0..t2 { walk.push(d2); }
|
|
new_out.push((walk.clone(), end, d2));
|
|
continue;
|
|
}
|
|
if d2 == *last && map.contains_key(&candidate_there) {
|
|
for _ in 0..t2 { walk.push(d2); }
|
|
for _ in 0..t1 { walk.push(d1); }
|
|
new_out.push((walk.clone(), end, d1));
|
|
continue;
|
|
}
|
|
|
|
if map.contains_key(&candidate_here) {
|
|
let mut ab_walk = walk.clone();
|
|
|
|
for _ in 0..t1 { ab_walk.push(d1); }
|
|
for _ in 0..t2 { ab_walk.push(d2); }
|
|
|
|
new_out.push((ab_walk, end, d2));
|
|
}
|
|
|
|
if map.contains_key(&candidate_there) {
|
|
let mut ba_walk = walk.clone();
|
|
|
|
for _ in 0..t2 { ba_walk.push(d2); }
|
|
for _ in 0..t1 { ba_walk.push(d1); }
|
|
|
|
new_out.push((ba_walk, end, d1));
|
|
}
|
|
}
|
|
out = new_out;
|
|
} else {
|
|
panic!("encountered a bad set of choices");
|
|
}
|
|
|
|
}
|
|
|
|
let mut final_out = vec![];
|
|
for (walk, _, _) in out {
|
|
final_out.push(walk);
|
|
}
|
|
return final_out;
|
|
}
|
|
|
|
|
|
fn build_options<T: std::hash::Hash + std::cmp::Eq>(
|
|
routes: &HashMap<(Coord, Coord), Vec<(Dir, i32)>>,
|
|
map: &HashMap<T, Coord>,
|
|
sequence: &Vec<T>,
|
|
start: T,
|
|
) -> Vec<Vec<(Dir, i32)>> {
|
|
let mut here = map[&start];
|
|
let mut options: Vec<Vec<(Dir, i32)>> = Vec::new();
|
|
for button in sequence {
|
|
let there = map[&button];
|
|
options.push(routes[&(here, there)].clone());
|
|
options.push(vec![(Stay, 1)]);
|
|
here = there;
|
|
}
|
|
|
|
return options;
|
|
}
|
|
|
|
|
|
fn print_code(code: &Vec<Dir>) {
|
|
for x in code {
|
|
print!("{}", match x {
|
|
North => '^', South => 'v', East => '>', West => '<', Stay => 'A'
|
|
});
|
|
}
|
|
print!("\n");
|
|
}
|
|
|
|
fn print_code_door(code: &Vec<char>) {
|
|
for x in code {
|
|
print!("{}", x);
|
|
}
|
|
print!("\n");
|
|
}
|
|
|
|
|
|
fn execute_code<T: Clone + Copy>(
|
|
code: &Vec<Dir>,
|
|
map: &HashMap<Coord, T>,
|
|
start: Coord
|
|
) -> Vec<T> {
|
|
|
|
let mut out = vec![];
|
|
let mut here = start;
|
|
for d in code {
|
|
if *d == Stay {
|
|
out.push(map[&here]);
|
|
} else {
|
|
here = step(&here, *d, 1);
|
|
}
|
|
|
|
}
|
|
return out;
|
|
|
|
} |