File size: 4,226 Bytes
a4da721
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
def parse_input(file):
    with open(file, 'r') as f:
        lines = f.readlines()
    
    # Separate the map and the movement sequence
    map_lines = []
    move_sequence = ""
    for line in lines:
        if line.startswith('#'):
            map_lines.append(line.strip())
        else:
            move_sequence += line.strip()
    
    return map_lines, move_sequence

def find_robot_and_boxes(map_lines):
    robot_pos = None
    boxes = set()
    for r, line in enumerate(map_lines):
        for c, char in enumerate(line):
            if char == '@':
                robot_pos = (r, c)
            elif char == 'O':
                boxes.add((r, c))
    return robot_pos, boxes

def move_robot_and_boxes(map_lines, robot_pos, boxes, move_sequence):
    directions = {'<': (0, -1), '>': (0, 1), '^': (-1, 0), 'v': (1, 0)}
    max_r = len(map_lines)
    max_c = len(map_lines[0])
    
    for move in move_sequence:
        dr, dc = directions[move]
        new_robot_pos = (robot_pos[0] + dr, robot_pos[1] + dc)
        
        if map_lines[new_robot_pos[0]][new_robot_pos[1]] == '#':
            continue  # Robot can't move into a wall
        
        if new_robot_pos in boxes:
            new_box_pos = (new_robot_pos[0] + dr, new_robot_pos[1] + dc)
            if map_lines[new_box_pos[0]][new_box_pos[1]] == '#' or new_box_pos in boxes:
                continue  # Box can't be pushed into a wall or another box
            boxes.remove(new_robot_pos)
            boxes.add(new_box_pos)
        
        robot_pos = new_robot_pos
    
    return robot_pos, boxes

def calculate_gps_sum(boxes, max_r, max_c):
    gps_sum = 0
    for r, c in boxes:
        gps_sum += 100 * r + c
    return gps_sum

def part_one(file):
    map_lines, move_sequence = parse_input(file)
    robot_pos, boxes = find_robot_and_boxes(map_lines)
    robot_pos, boxes = move_robot_and_boxes(map_lines, robot_pos, boxes, move_sequence)
    return calculate_gps_sum(boxes, len(map_lines), len(map_lines[0]))

def scale_map(map_lines):
    scaled_map = []
    for line in map_lines:
        scaled_line = ""
        for char in line:
            if char == '#':
                scaled_line += "##"
            elif char == 'O':
                scaled_line += "[]"
            elif char == '.':
                scaled_line += ".."
            elif char == '@':
                scaled_line += "@."
        scaled_map.append(scaled_line)
    return scaled_map

def find_robot_and_boxes_scaled(map_lines):
    robot_pos = None
    boxes = set()
    for r, line in enumerate(map_lines):
        for c in range(0, len(line), 2):
            char = line[c:c+2]
            if char == '@.':
                robot_pos = (r, c)
            elif char == '[]':
                boxes.add((r, c))
    return robot_pos, boxes

def move_robot_and_boxes_scaled(map_lines, robot_pos, boxes, move_sequence):
    directions = {'<': (0, -2), '>': (0, 2), '^': (-1, 0), 'v': (1, 0)}
    max_r = len(map_lines)
    max_c = len(map_lines[0])
    
    for move in move_sequence:
        dr, dc = directions[move]
        new_robot_pos = (robot_pos[0] + dr, robot_pos[1] + dc)
        
        if map_lines[new_robot_pos[0]][new_robot_pos[1]] == '#' or map_lines[new_robot_pos[0]][new_robot_pos[1]] == '#':
            continue  # Robot can't move into a wall
        
        if new_robot_pos in boxes:
            new_box_pos = (new_robot_pos[0] + dr, new_robot_pos[1] + dc)
            if map_lines[new_box_pos[0]][new_box_pos[1]] == '#' or new_box_pos in boxes:
                continue  # Box can't be pushed into a wall or another box
            boxes.remove(new_robot_pos)
            boxes.add(new_box_pos)
        
        robot_pos = new_robot_pos
    
    return robot_pos, boxes

def part_two(file):
    map_lines, move_sequence = parse_input(file)
    scaled_map_lines = scale_map(map_lines)
    robot_pos, boxes = find_robot_and_boxes_scaled(scaled_map_lines)
    robot_pos, boxes = move_robot_and_boxes_scaled(scaled_map_lines, robot_pos, boxes, move_sequence)
    return calculate_gps_sum(boxes, len(scaled_map_lines), len(scaled_map_lines[0]))

file = "input.txt"
result1 = part_one(file)
print(result1)

result2 = part_two(file)
print(result2)