2.rkt 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112
  1. #lang racket
  2. (require "../lib/utils.rkt")
  3. (define lines
  4. (call-with-input-file "input"
  5. (λ (fp)
  6. (get-lines fp))))
  7. (define diagram (list->vector lines))
  8. (define height (vector-length diagram))
  9. (define width (string-length (vector-ref diagram 0)))
  10. (define (char-at x y)
  11. (string-ref (vector-ref diagram y) x))
  12. (define (find-start)
  13. (let loop ((x 0) (y 0))
  14. (if (>= y height)
  15. #f
  16. (if (>= x width)
  17. (loop 0 (+ y 1))
  18. (if (char=? #\S (char-at x y))
  19. (cons x y)
  20. (loop (+ x 1) y))))))
  21. (define start (find-start))
  22. (define (valid-pos? pos)
  23. (define x (car pos))
  24. (define y (cdr pos))
  25. (and (and (>= x 0) (< x width))
  26. (and (>= y 0) (< y height))))
  27. (define first-pos
  28. (let ()
  29. (define up (cons (car start) (- (cdr start) 1)))
  30. (define down (cons (car start) (+ (cdr start) 1)))
  31. (define left (cons (- (car start) 1) (cdr start)))
  32. (define right (cons (+ (car start) 1) (cdr start)))
  33. (cond ((and (valid-pos? up)
  34. (member (char-at (car up) (cdr up)) (list #\| #\F #\7)))
  35. up)
  36. ((and (valid-pos? down)
  37. (member (char-at (car down) (cdr down)) (list #\| #\L #\J )))
  38. down)
  39. ((and (valid-pos? left)
  40. (member (char-at (car left) (cdr left)) (list #\- #\F #\L )))
  41. left)
  42. ((and (valid-pos? right)
  43. (member (char-at (car right) (cdr right)) (list #\- #\7 #\J)))
  44. right))))
  45. (define (next cur recent)
  46. (define up (cons (car cur) (- (cdr cur) 1)))
  47. (define down (cons (car cur) (+ (cdr cur) 1)))
  48. (define left (cons (- (car cur) 1) (cdr cur)))
  49. (define right (cons (+ (car cur) 1) (cdr cur)))
  50. (define c (char-at (car cur) (cdr cur)))
  51. (define possible-pos
  52. (cond ((char=? c #\|) (list up down))
  53. ((char=? c #\-) (list left right))
  54. ((char=? c #\F) (list right down))
  55. ((char=? c #\L) (list up right))
  56. ((char=? c #\7) (list left down))
  57. ((char=? c #\J) (list up left))))
  58. (let loop ((cur possible-pos))
  59. (if (not (equal? (car cur) recent))
  60. (car cur)
  61. (loop (cdr cur)))))
  62. (define pipes (make-hash))
  63. (define (simulate)
  64. (let loop ((i 0) (cur first-pos) (recent start))
  65. (hash-set! pipes cur '())
  66. (if (equal? cur start)
  67. i
  68. (let ()
  69. (define next-loc (next cur recent))
  70. (loop (+ i 1) next-loc cur)))))
  71. (define (mark-junk)
  72. (let loop ((x 0) (y 0))
  73. (if (>= y height)
  74. (void)
  75. (if (>= x width)
  76. (loop 0 (+ y 1))
  77. (let ()
  78. (when (and (not (hash-has-key? pipes (cons x y)))
  79. (not (char=? #\. (char-at x y))))
  80. (string-set! (vector-ref diagram y) x #\.))
  81. (loop (+ 1 x) y))))))
  82. (let ()
  83. (simulate)
  84. (mark-junk)
  85. (void))
  86. (define (ray-cast-find s)
  87. (let loop ((i 0) (flag #f) (ret 0))
  88. (if (>= i width)
  89. ret
  90. (let ()
  91. (define char (string-ref s i))
  92. (if (and flag (char=? char #\.))
  93. (loop (+ 1 i) flag (+ 1 ret))
  94. (if (member char (list #\| #\F #\7))
  95. (loop (+ 1 i) (not flag) ret)
  96. (loop (+ 1 i) flag ret)))))))
  97. (apply + (map ray-cast-find (vector->list diagram)))