diff --git a/day6.peggy b/day6.peggy new file mode 100644 index 0000000..cdcfc7c --- /dev/null +++ b/day6.peggy @@ -0,0 +1 @@ +lines = (@[^\n]+ "\n")* diff --git a/day6.ts b/day6.ts new file mode 100644 index 0000000..3fe8786 --- /dev/null +++ b/day6.ts @@ -0,0 +1,78 @@ +import { Dir, Point, Rect } from './lib/rect.ts'; +import { type MainArgs, parseFile } from './lib/utils.ts'; + +type Parsed = string[][]; + +function path(r: Rect): [Set, Point] { + const [start] = r.filter(val => val === '^'); + let pos = start; + let dir = Dir.N; + const visited = new Set(); + while (true) { + visited.add(pos.toNumber()); + const ahead = pos.inDir(dir); + if (!r.check(ahead)) { + break; + } + const char = r.get(ahead); + if (char === '.' || char === '^') { + pos = ahead; + } else { + dir = (dir + 1) % 4 + } + } + return [visited, start]; +} + +function path2(r: Rect): boolean { + const [start] = r.filter(val => val === '^'); + let pos = start; + let dir = Dir.N; + const visitedWithDir = new Set(); + while (true) { + const pwd = `${pos},${dir}`; + if (visitedWithDir.has(pwd)) { + return true; + } + visitedWithDir.add(pwd); + const ahead = pos.inDir(dir); + if (!r.check(ahead)) { + return false; + } + const char = r.get(ahead); + if (char === '.' || char === '^') { + pos = ahead; + } else { + dir = (dir + 1) % 4 + } + } +} + +function part1(inp: Parsed): number { + const r = new Rect(inp); + const [visited] = path(r) + return visited.size; +} + +function part2(inp: Parsed): number { + const r = new Rect(inp); + const [visited, start] = path(r) + const points = [...visited].map(v => Point.fromNumber(v)) + let tot = 0; + for (const p of points) { + if (p.equals(start)) { + continue; + } + r.set(p, '#'); + if (path2(r)) { + tot++; + } + r.set(p, '.'); + } + return tot; +} + +export default async function main(args: MainArgs): Promise<[number, number]> { + const inp = await parseFile(args); + return [part1(inp), part2(inp)]; +} diff --git a/inputs b/inputs index 545dfbe..1a785f8 160000 --- a/inputs +++ b/inputs @@ -1 +1 @@ -Subproject commit 545dfbe8c1614e19ca6e08fe010b399ffaa355d0 +Subproject commit 1a785f888c1604214aa8fff29383579e32653aab diff --git a/lib/rect.ts b/lib/rect.ts index 30aa188..8da27df 100644 --- a/lib/rect.ts +++ b/lib/rect.ts @@ -145,6 +145,22 @@ export class Point implements PointLike { return `${this.x},${this.y}`; } + toNumber(size = 16): number { + return (this.x << size) | this.y; + } + + static fromNumber(num: number, size = 16): Point { + return new Point(num >> size, num & ((1 << size) - 1)); + } + + static fromString(str: string): Point { + const m = str.match(/(\d+),(\d+)/); + if (!m) { + throw new Error(`Invalid format: "${str}"`); + } + return new Point(Number(m[1]), Number(m[2])); + } + [Symbol.for('Deno.customInspect')](): string { return this.toString(); } diff --git a/test/day6.js b/test/day6.js new file mode 100644 index 0000000..dc70859 --- /dev/null +++ b/test/day6.js @@ -0,0 +1 @@ +export default [4374, 1705];