aboutsummaryrefslogtreecommitdiffstats
path: root/year2021/day23/part2.clj
blob: adbefc13def252c430f1b2fd706d75b630c73f1f (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
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
(require '[clojure.core.reducers :as r])

(def init-field
  [nil
   nil
   (seq [:b :d :d :d])
   nil
   (seq [:b :c :b :a])
   nil
   (seq [:c :b :a :a])
   nil
   (seq [:d :a :c :c])
   nil
   nil])

(defn do-slot [[field energy] idx]
  (let [slot (get field idx)]
    (cond
      ; Moving value out of a room
      (and (seq? slot)
           (not (empty? slot))
           (not (every? #(= % ({2 :a 4 :b 6 :c 8 :d} idx)) slot)))
      (let [open-slots
            (filterv
              #(contains? #{0 1 3 5 7 9 10} %)
              (concat
                (for [i (reverse (range 0 idx))
                      :while (or (nil? (get field i)) (seq? (get field i)))] i)
                (for [i (range idx (count field))
                      :while (or (nil? (get field i)) (seq? (get field i)))] i)))]
        (when-not (empty? open-slots)
          (map
            (fn [os]
              [(-> field
                   (assoc os (first slot))
                   (update idx rest))
               (+ energy (* ({:a 1 :b 10 :c 100 :d 1000} (first slot))
                            (+ (inc (- 4 (count slot))) (Math/abs (- os idx)))))])
            open-slots)))
      ; Moving value into a room
      (and (not (seq? slot)) (some? slot))
      (let [our-room ({:a 2 :b 4 :c 6 :d 8} slot)]
        (if (every? #(or (nil? (get field %)) (seq? (get field %)))
                    (range (inc (min our-room idx)) (max our-room idx)))
          (let [room (get field our-room)]
            (when (or (empty? room) (every? #(= slot %) room))
              [(-> field
                   (assoc idx nil)
                   (update our-room conj slot))
               (+ energy (* ({:a 1 :b 10 :c 100 :d 1000} slot)
                            (+ (Math/abs (- idx our-room)) (- 4 (count room)))))])))))))

(defn winner? [[field q s]]
  (= field
    [nil
     nil
     (seq [:a :a :a :a])
     nil
     (seq [:b :b :b :b])
     nil
     (seq [:c :c :c :c])
     nil
     (seq [:d :d :d :d])
     nil
     nil]))

(defn do-turns [fields]
  (into #{}
    (r/fold
      r/cat
      #(if-let [t (apply do-slot %2)]
         (if (seq? t)
           (reduce r/append! %1 t)
           (r/append! %1 t))
         %1)
      (into [] (for [f fields i (range 0 11)] [f i])))))

(defn play-games [turns tc]
  (println "Games:" (count turns) "Turn:" tc)
  (let [new-turns (do-turns turns)
        winners (filter winner? new-turns)]
    (if (seq winners)
      (map second winners)
      (recur new-turns (inc tc)))))

(->> (play-games #{[init-field 0]} 0)
     sort
     first
     ((partial println "Lowest energy:")))