aboutsummaryrefslogtreecommitdiffstats
path: root/day9/part2.clj
blob: 455cb39655625a239ea81a8fad44bd8b89cf16bc (plain)
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
(def input-map
  (->> (slurp "./in")
       (clojure.string/split-lines)
       (mapv vec)
       (mapv (partial mapv #(- (int %) 48)))
       ))

(defn basin-bottom-single? [a b]
  (if (or (nil? a) (nil? b) (< a b)) 1 0))

(defn basin-continues? [a b]
  (and (some? b) (not= b 9) (> b a)))

(defn basin-bottom? [hmap p adj]
  (= 4
    (apply +
      (map
        #(basin-bottom-single?
           (get-in hmap p)
           (get-in hmap %))
        adj
        ))))

(defn get-adj [y x]
  [[(dec y) x]
   [(inc y) x]
   [y (dec x)]
   [y (inc x)]])

(defn find-basin [hmap y x adj]
  (let [res (reduce
              #(if
                   (basin-continues? (get-in hmap [y x])
                                     (get-in hmap %2))
                 (conj %1 %2)
                 %1)
              []
              adj
              )]
    (if (empty? res)
      []
      (apply
        (partial concat res)
        (map
          #(find-basin hmap (first %) (second %) (apply get-adj %))
          res))
      )
    )
  )

(println
  (apply *
    (take 3
      (sort >
        (filter
          pos?
            (for [y (range 0 (count input-map))
                   x (range 0 (count (first input-map)))]
              (let [adj (get-adj y x)]
                (if (basin-bottom? input-map [y x] adj)
                  (count (distinct (concat [[y x]] (find-basin input-map y x adj))))
                  0
                  ))))))))