summaryrefslogtreecommitdiff
path: root/fulmer.cjl.txt
diff options
context:
space:
mode:
Diffstat (limited to 'fulmer.cjl.txt')
-rw-r--r--fulmer.cjl.txt1558
1 files changed, 1558 insertions, 0 deletions
diff --git a/fulmer.cjl.txt b/fulmer.cjl.txt
new file mode 100644
index 0000000..da05e8f
--- /dev/null
+++ b/fulmer.cjl.txt
@@ -0,0 +1,1558 @@
+(ns dactyl-keyboard.dactyl
+ (:refer-clojure :exclude [use import])
+ (:require [scad-clj.scad :refer :all]
+ [scad-clj.model :refer :all]
+ [unicode-math.core :refer :all]
+ [clojure.core.matrix.operators :refer [+ - / *]]))
+
+;;;;;;;;;;;;;;;;;
+;; Switch Hole ;;
+;;;;;;;;;;;;;;;;;
+
+(def plate-thickness 4) ; was 4
+
+(def keyswitch-height 14.4) ;; Was 14.1, then 14.25
+(def keyswitch-width 14.4)
+
+(def key-height 10.4) ; was 12.7, then 10.4
+(def dsa-profile-key-height 7.4)
+(def key-z (+ plate-thickness 3)) ; 3 is pressed, 7 is released
+
+(def mount-width (+ keyswitch-width 3))
+(def mount-height (+ keyswitch-height 3))
+
+(def single-plate
+ (let [top-wall (->> (cube (+ keyswitch-width 3) 1.5 plate-thickness)
+ (translate [0
+ (+ (/ 1.5 2) (/ keyswitch-height 2))
+ (/ plate-thickness 2)]))
+ left-wall (->> (cube 1.5 (+ keyswitch-height 3) plate-thickness)
+ (translate [(+ (/ 1.5 2) (/ keyswitch-width 2))
+ 0
+ (/ plate-thickness 2)]))
+ side-nub (->> (binding [*fn* 30] (cylinder 1 2.75)) ; spec says 5.7
+ (rotate (/ π 2) [1 0 0])
+ (translate [(+ (/ keyswitch-width 2)) 0 1])
+ (hull (->> (cube 1.5 2.75 plate-thickness)
+ (translate [(+ (/ 1.5 2) (/ keyswitch-width 2))
+ 0
+ (/ plate-thickness 2)]))))
+ plate-half (union top-wall left-wall (with-fn 100 side-nub))]
+ (union plate-half
+ (->> plate-half
+ (mirror [1 0 0])
+ (mirror [0 1 0])))))
+
+;;;;;;;;;;;;;;;;
+;; SA Keycaps ;;
+;;;;;;;;;;;;;;;;
+
+(def sa-length 18.5)
+(def sa-double-length 37.5)
+(def sa-cap {1 (let [bl2 (/ sa-length 2)
+ m (/ dsa-profile-key-height 2)
+ key-cap (hull (->> (polygon [[bl2 bl2] [bl2 (- bl2)] [(- bl2) (- bl2)] [(- bl2) bl2]])
+ (extrude-linear {:height 0.1 :twist 0 :convexity 0})
+ (translate [0 0 0.05]))
+ (->> (polygon [[m m] [m (- m)] [(- m) (- m)] [(- m) m]])
+ (extrude-linear {:height 0.1 :twist 0 :convexity 0})
+ (translate [0 0 6]))
+ (->> (polygon [[6 6] [6 -6] [-6 -6] [-6 6]])
+ (extrude-linear {:height 0.1 :twist 0 :convexity 0})
+ (translate [0 0 dsa-profile-key-height])))]
+ (->> key-cap
+ (translate [0 0 key-z])
+ (color [220/255 163/255 163/255 1])))
+ 2 (let [bl2 (/ sa-double-length 2)
+ bw2 (/ 18.25 2)
+ key-cap (hull (->> (polygon [[bw2 bl2] [bw2 (- bl2)] [(- bw2) (- bl2)] [(- bw2) bl2]])
+ (extrude-linear {:height 0.1 :twist 0 :convexity 0})
+ (translate [0 0 0.05]))
+ (->> (polygon [[6 16] [6 -16] [-6 -16] [-6 16]])
+ (extrude-linear {:height 0.1 :twist 0 :convexity 0})
+ (translate [0 0 dsa-profile-key-height])))]
+ (->> key-cap
+ (translate [0 0 key-z])
+ (color [127/255 159/255 127/255 1])))
+ 1.5 (let [bl2 (/ 18.25 2)
+ bw2 (/ 28 2)
+ key-cap (hull (->> (polygon [[bw2 bl2] [bw2 (- bl2)] [(- bw2) (- bl2)] [(- bw2) bl2]])
+ (extrude-linear {:height 0.1 :twist 0 :convexity 0})
+ (translate [0 0 0.05]))
+ (->> (polygon [[11 6] [-11 6] [-11 -6] [11 -6]])
+ (extrude-linear {:height 0.1 :twist 0 :convexity 0})
+ (translate [0 0 dsa-profile-key-height])))]
+ (->> key-cap
+ (translate [0 0 key-z])
+ (color [240/255 223/255 175/255 1])))})
+
+;;;;;;;;;;;;;;;;;;;;;;;;;
+;; Placement Functions ;;
+;;;;;;;;;;;;;;;;;;;;;;;;;
+
+(def columns (range 0 6))
+(def rows (range 0 5))
+
+(def α (/ π 12))
+(def β (/ π 36))
+(def cap-top-height (+ plate-thickness key-height))
+(def row-radius (+ (/ (/ (+ mount-height 1/2) 2)
+ (Math/sin (/ α 2)))
+ cap-top-height))
+(def column-radius (+ (/ (/ (+ mount-width 2.0) 2)
+ (Math/sin (/ β 2)))
+ cap-top-height))
+
+(defn key-place [column row shape]
+ (let [row-placed-shape (->> shape
+ (translate [0 0 (- row-radius)])
+ (rotate (* α (- 2 row)) [1 0 0])
+ (translate [0 0 row-radius]))
+ column-row-offset (cond
+ (= column 2) [0 2.4 -4.5]
+ (= column 4) [0 -5.8 5.64]
+ (and (= column 5) (not= row 4)) [5.2 -5.8 7.01]
+ (and (= column 5) (= row 4)) [0.5 -5.8 5.7]
+ (and (= column 6) (= row 4)) [8.9 -5.8 8] ; extended connector
+ (= column 6) [9 -5.8 8] ; extended connector
+ :else [0 0 0])
+ column-angle (* β (- 2 column))
+ placed-shape (->> row-placed-shape
+ (translate [0 0 (- column-radius)])
+ (rotate column-angle [0 1 0])
+ (translate [0 0 column-radius])
+ (translate column-row-offset))]
+ (->> placed-shape
+ (rotate (/ π 12) [0 1 0])
+ (translate [0 0 14.5]))))
+
+(defn case-place [column row shape]
+ (let [row-placed-shape (->> shape
+ (translate [0 0 (- row-radius)])
+ (rotate (* α (- 2 row)) [1 0 0])
+ (translate [0 0 row-radius]))
+ column-offset (if (= column 6)
+ [-7.25 -5.8 2.1]
+ [0 -3.35 4.9])
+ row-offset (if (= row 0)
+ [0 -2.3 0]
+ [0 0 0])
+ column-row-offset (if (and (= row 0) (= column 6))
+ [0 2.25 0]
+ [0 0 0])
+ column-angle (* β (- 2 column))
+ placed-shape (->> row-placed-shape
+ (translate [0 0 (- column-radius)])
+ (rotate column-angle [0 1 0])
+ (translate [0 0 column-radius])
+ (translate column-row-offset)
+ (translate column-offset)
+ (translate row-offset))]
+ (->> placed-shape
+ (rotate (/ π 12) [0 1 0])
+ (translate [0 0 14.5]))))
+
+(defn bottom-place [column row shape]
+ (let [row-placed-shape (->> shape
+ (translate [0 0 (- row-radius)])
+ (rotate (* α (- 2 row)) [1 0 0])
+ (translate [0 0 row-radius]))
+ column-offset (cond (< column 1.5) [-1 0 0]
+ :else [0 0 0])
+ column-row-offset (if (not= column 6) [0 -4.35 4.8]
+ (if (not= row 4) [-7.25 -5.8 2.1]
+ [-7.89 -5.8 3.6]))
+ column-angle (* β (- 2 column))
+ placed-shape (->> row-placed-shape
+ (translate [0 0 (- column-radius)])
+ (rotate column-angle [0 1 0])
+ (translate [0 0 column-radius])
+ (translate column-row-offset)
+ (translate column-offset))]
+ (->> placed-shape
+ (rotate (/ π 12) [0 1 0])
+ (translate [0 0 13]))))
+
+(def key-holes
+ (apply union
+ (for [column columns
+ row rows
+ :when (or (not= column 0)
+ (not= row 4))]
+ (->> single-plate
+ (key-place column row)))))
+
+(def caps
+ (apply union
+ (for [column columns
+ row rows
+ :when (or (not= column 0)
+ (not= row 4))]
+ (->> (sa-cap (if (and (= column 5) (not= row 4)) 1.5 1))
+ (key-place column row)))))
+
+(defn prism [w l h taper-1 taper-2]
+ (let [t1 taper-1
+ t2 taper-2
+ wt (- w taper-1)
+ lt (- l taper-2)]
+ (polyhedron [[0 0 0]
+ [t1 t1 h]
+ [wt t1 h]
+ [w 0 0]
+ [0 l 0]
+ [t1 lt h]
+ [wt lt h]
+ [w l 0]]
+ [[0 1 2] [2 3 0]
+ [3 2 6] [6 7 3]
+ [7 6 5] [5 4 7]
+ [4 5 1] [1 0 4]
+ [1 5 2] [2 5 6]
+ [4 0 3] [7 4 3]])))
+
+
+;;;;;;;;;;;;;;;;;;;;
+;; Web Connectors ;;
+;;;;;;;;;;;;;;;;;;;;
+
+(def web-thickness plate-thickness) ; was 3.5
+(def post-size 0.1)
+(def web-post (->> (cube post-size post-size web-thickness)
+ (translate [0 0 (+ (/ web-thickness -2)
+ plate-thickness)])))
+
+(def post-adj (/ post-size 2))
+(def web-post-tr (translate [(- (/ mount-width 2) post-adj) (- (/ mount-height 2) post-adj) 0] web-post))
+(def web-post-tl (translate [(+ (/ mount-width -2) post-adj) (- (/ mount-height 2) post-adj) 0] web-post))
+(def web-post-bl (translate [(+ (/ mount-width -2) post-adj) (+ (/ mount-height -2) post-adj) 0] web-post))
+(def web-post-br (translate [(- (/ mount-width 2) post-adj) (+ (/ mount-height -2) post-adj) 0] web-post))
+
+(defn triangle-hulls [& shapes]
+ (apply union
+ (map (partial apply hull)
+ (partition 3 1 shapes))))
+
+(def connectors
+ (apply union
+ (concat
+ ;; Row connections
+ (for [column columns ;(drop-last columns)
+ row rows
+ :when (or (not= column 0)
+ (not= row 4))]
+ (triangle-hulls
+ (key-place (inc column) row web-post-tl)
+ (key-place column row web-post-tr)
+ (key-place (inc column) row web-post-bl)
+ (key-place column row web-post-br)))
+
+ ;; Column connections
+ (for [column columns
+ row (drop-last rows)
+ :when (or (not= column 0)
+ (not= row 3))]
+ (triangle-hulls
+ (key-place column row web-post-bl)
+ (key-place column row web-post-br)
+ (key-place column (inc row) web-post-tl)
+ (key-place column (inc row) web-post-tr)))
+
+ ;; Diagonal connections
+ (for [column columns;(drop-last columns)
+ row (drop-last rows)
+ :when (or (not= column 0)
+ (not= row 3))]
+ (triangle-hulls
+ (key-place column row web-post-br)
+ (key-place column (inc row) web-post-tr)
+ (key-place (inc column) row web-post-bl)
+ (key-place (inc column) (inc row) web-post-tl))))))
+
+;;;;;;;;;;;;
+;; Thumbs ;;
+;;;;;;;;;;;;
+
+(defn thumb-place [column row shape]
+ (let [cap-top-height (+ plate-thickness key-height)
+ α (/ π 12)
+ row-radius (+ (/ (/ (+ mount-height 1) 2)
+ (Math/sin (/ α 2)))
+ cap-top-height)
+ β (/ π 36)
+ column-radius (+ (/ (/ (+ mount-width 2) 2)
+ (Math/sin (/ β 2)))
+ cap-top-height)
+ #_(+ (/ (/ (+ pillar-width 5) 2)
+ (Math/sin (/ β 2)))
+ cap-top-height)]
+ (->> shape
+ (translate [0 0 (- row-radius)])
+ (rotate (* α row) [1 0 0])
+ (translate [0 0 row-radius])
+ (translate [0 0 (- column-radius)])
+ (rotate (* column β) [0 1 0])
+ (translate [0 0 column-radius])
+ (translate [mount-width 0 0])
+ (rotate (* π (- 1/4 3/16)) [0 0 1])
+ (rotate (/ π 12) [1 1 0])
+ (translate [-52 -45 40]))))
+
+(defn thumb-2x-column [shape]
+ (thumb-place 0 -1/2 shape))
+
+(defn thumb-2x+1-column [shape]
+ (union (thumb-place 1 -1/2 shape)
+ (thumb-place 1 1 shape)))
+
+(defn thumb-1x-column [shape]
+ (union (thumb-place 2 -1 shape)
+ (thumb-place 2 0 shape)
+ (thumb-place 2 1 shape)))
+
+(defn thumb-layout [shape]
+ (union
+ (thumb-2x-column shape)
+ (thumb-2x+1-column shape)
+ (thumb-1x-column shape)))
+
+(def double-plates
+ (let [plate-height (/ (- sa-double-length mount-height) 2)
+ top-plate (->> (cube mount-width plate-height web-thickness)
+ (translate [0 (/ (+ plate-height mount-height) 2)
+ (- plate-thickness (/ web-thickness 2))]))
+ ; Costar or WASD stabilizers
+ stabilizer-cutout (union (->> (cube 14.2 3.5 web-thickness)
+ (translate [0.5 12 (- plate-thickness (/ web-thickness 2))])
+ (color [1 0 0 1/2]))
+ (->> (cube 16 3.5 web-thickness)
+ (translate [0.5 12 (- plate-thickness (/ web-thickness 2) 1.4)])
+ (color [1 0 0 1/2])))
+ top-plate (difference top-plate stabilizer-cutout)]
+ (union top-plate (mirror [0 1 0] top-plate))))
+
+(def thumbcaps
+ (union
+ (thumb-2x-column (sa-cap 2))
+ (thumb-place 1 -1/2 (sa-cap 2))
+ (thumb-place 1 1 (sa-cap 1))
+ (thumb-1x-column (sa-cap 1))))
+
+(def thumb-connectors
+ (union
+ (apply union
+ (concat
+ (for [column [2] row [1]]
+ (triangle-hulls (thumb-place column row web-post-br)
+ (thumb-place column row web-post-tr)
+ (thumb-place (dec column) row web-post-bl)
+ (thumb-place (dec column) row web-post-tl)))
+ (for [column [2] row [0 1]]
+ (triangle-hulls
+ (thumb-place column row web-post-bl)
+ (thumb-place column row web-post-br)
+ (thumb-place column (dec row) web-post-tl)
+ (thumb-place column (dec row) web-post-tr)))))
+ (let [plate-height (/ (- sa-double-length mount-height) 2)
+ thumb-tl (->> web-post-tl
+ (translate [0 plate-height 0]))
+ thumb-bl (->> web-post-bl
+ (translate [0 (- plate-height) 0]))
+ thumb-tr (->> web-post-tr
+ (translate [0 plate-height 0]))
+ thumb-br (->> web-post-br
+ (translate [0 (- plate-height) 0]))]
+ (union
+
+ ;;Connecting the two doubles
+ (triangle-hulls (thumb-place 0 -1/2 thumb-tl)
+ (thumb-place 0 -1/2 thumb-bl)
+ (thumb-place 1 -1/2 thumb-tr)
+ (thumb-place 1 -1/2 thumb-br))
+
+ ;;Connecting the double to the one above it
+ (triangle-hulls (thumb-place 1 -1/2 thumb-tr)
+ (thumb-place 1 -1/2 thumb-tl)
+ (thumb-place 1 1 web-post-br)
+ (thumb-place 1 1 web-post-bl))
+
+ ;;Connecting the 4 with the double in the bottom left
+ (triangle-hulls (thumb-place 1 1 web-post-bl)
+ (thumb-place 1 -1/2 thumb-tl)
+ (thumb-place 2 1 web-post-br)
+ (thumb-place 2 0 web-post-tr))
+
+ ;;Connecting the two singles with the middle double
+ (hull (thumb-place 1 -1/2 thumb-tl)
+ (thumb-place 1 -1/2 thumb-bl)
+ (thumb-place 2 0 web-post-br)
+ (thumb-place 2 -1 web-post-tr))
+ (hull (thumb-place 1 -1/2 thumb-tl)
+ (thumb-place 2 0 web-post-tr)
+ (thumb-place 2 0 web-post-br))
+ (hull (thumb-place 1 -1/2 thumb-bl)
+ (thumb-place 2 -1 web-post-tr)
+ (thumb-place 2 -1 web-post-br))
+
+ ;;Connecting the thumb to everything
+ (triangle-hulls (thumb-place 0 -1/2 thumb-br)
+ (key-place 1 4 web-post-bl)
+ (thumb-place 0 -1/2 thumb-tr)
+ (key-place 1 4 web-post-tl)
+ (key-place 1 3 web-post-bl)
+ (thumb-place 0 -1/2 thumb-tr)
+ (key-place 0 3 web-post-br)
+ (key-place 0 3 web-post-bl)
+ (thumb-place 0 -1/2 thumb-tr)
+ (thumb-place 0 -1/2 thumb-tl)
+ (key-place 0 3 web-post-bl)
+ (thumb-place 1 -1/2 thumb-tr)
+ (thumb-place 1 1 web-post-br)
+ (key-place 0 3 web-post-bl)
+ (key-place 0 3 web-post-tl)
+ (thumb-place 1 1 web-post-br)
+ (thumb-place 1 1 web-post-tr))
+ (hull (thumb-place 0 -1/2 web-post-tr)
+ (thumb-place 0 -1/2 thumb-tr)
+ (key-place 1 4 web-post-bl)
+ (key-place 1 4 web-post-tl))))))
+
+(def thumb
+ (union
+ thumb-connectors
+ (thumb-layout (rotate (/ π 2) [0 0 1] single-plate))
+ (thumb-place 0 -1/2 double-plates)
+ (thumb-place 1 -1/2 double-plates)))
+
+;;;;;;;;;;
+;; Case ;;
+;;;;;;;;;;
+
+;; In column units
+(def right-wall-column (+ (last columns) 1.1))
+(def left-wall-column (- (first columns) 1/2))
+(def thumb-back-y 0.93)
+(def thumb-case-z 3)
+(def thumb-right-wall (- -1/2 0.05))
+(def thumb-front-row (+ -1 0.07))
+(def thumb-left-wall-column (+ 5/2 0.05))
+(def back-y 0.02)
+
+(defn range-inclusive [start end step]
+ (concat (range start end step) [end]))
+
+(def wall-step 0.2)
+(def wall-sphere-n 20) ;;20 Sphere resolution, lower for faster renders
+
+(defn wall-cube-at [coords]
+ (->> (cube 3 3 3)
+ (translate coords)))
+
+(defn scale-to-range [start end x]
+ (+ start (* (- end start) x)))
+
+(defn wall-cube-bottom [front-to-back-scale]
+ (wall-cube-at [0
+ (scale-to-range
+ (+ (/ mount-height -2) -3.5)
+ (+ (/ mount-height 2) 5.0)
+ front-to-back-scale)
+ -6])) ; was -6, then 2
+
+(defn wall-cube-top [front-to-back-scale]
+ (wall-cube-at [0
+ (scale-to-range
+ (+ (/ mount-height -2) -3.5)
+ (+ (/ mount-height 2) 3.5)
+ front-to-back-scale)
+ 4])) ; case height
+
+(def wall-cube-top-back (wall-cube-top 1))
+(def wall-cube-bottom-back (wall-cube-bottom 1))
+(def wall-cube-bottom-front (wall-cube-bottom 0))
+(def wall-cube-top-front (wall-cube-top 0))
+
+(defn top-case-cover [place-fn sphere
+ x-start x-end
+ y-start y-end
+ step]
+ (apply union
+ (for [x (range-inclusive x-start (- x-end step) step)
+ y (range-inclusive y-start (- y-end step) step)]
+ (hull (place-fn x y sphere)
+ (place-fn (+ x step) y sphere)
+ (place-fn x (+ y step) sphere)
+ (place-fn (+ x step) (+ y step) sphere)))))
+
+(def case-back-cutout
+ (let [a (+ mount-width 8)]
+ (->> (prism a a a 0 6)
+ (rotate π [0 0 1])
+ (rotate (/ π 15) [1 0 0])
+ (translate [(/ a 2) (+ a 9.3) -15.25])
+ (key-place 2 0))))
+
+(def case-inside-cutout
+ (let [a (+ mount-width 8)
+ b (+ mount-width 3.8)
+ c 6]
+ (union
+ (->> (prism b b c 2 2)
+ (rotate π [1 0 0])
+ (translate [(- (/ b 2)) 10.5 (+ c 4)])
+ (key-place 2 0)))))
+
+(def front-wall
+ (let [step wall-step ;;0.1
+ wall-step 0.05 ;;0.05
+ place case-place
+ top-cover (fn [x-start x-end y-start y-end]
+ (top-case-cover place wall-cube-top-front
+ x-start x-end y-start y-end
+ wall-step))]
+ (union
+ (apply union
+ (for [x (range 2 5)]
+ (union
+ (hull (place (- x 1/2) 4 (translate [0 1 1] wall-cube-bottom-front))
+ (place (+ x 1/2) 4 (translate [0 1 1] wall-cube-bottom-front))
+ (key-place x 4 web-post-bl)
+ (key-place x 4 web-post-br))
+ (hull (place (- x 1/2) 4 (translate [0 1 1] wall-cube-bottom-front))
+ (key-place x 4 web-post-bl)
+ (key-place (- x 1) 4 web-post-br)))))
+ (hull (place right-wall-column 4 (translate [-1 1 1] wall-cube-bottom-front))
+ (place (- right-wall-column 1) 4 (translate [0 1 1] wall-cube-bottom-front))
+ (key-place 5 4 web-post-bl)
+ (key-place 5 4 web-post-br))
+ (hull (place right-wall-column 4 (translate [-1 1 1] wall-cube-bottom-front))
+ (place (- right-wall-column 1) 4 (translate [0 1 1] wall-cube-bottom-front))
+ (key-place 6 4 web-post-bl)
+ (key-place 5 4 web-post-br))
+ (hull (place (+ 4 1/2) 4 (translate [0 1 1] wall-cube-bottom-front))
+ (place (- right-wall-column 1) 4 (translate [0 1 1] wall-cube-bottom-front))
+ (key-place 4 4 web-post-br)
+ (key-place 5 4 web-post-bl))
+ (hull (place 0.721 4 (translate [0 1 -1.4] wall-cube-bottom-front))
+ (place 1.5 4 (translate [0 1 1] wall-cube-bottom-front))
+ (key-place 1 4 web-post-bl)
+ (key-place 1 4 web-post-br)))))
+
+(def back-wall
+ (let [step wall-step
+ wall-cube-top-backtep 0.05
+ place case-place]
+ (difference
+ (union
+ (union
+ (hull (place left-wall-column 0 (translate [1 -1 1] wall-cube-bottom-back))
+ (place (+ left-wall-column 1) 0 (translate [0 -1 1] wall-cube-bottom-back))
+ (key-place 0 0 web-post-tl)
+ (key-place 0 0 web-post-tr))
+
+ (hull (place 5 0 (translate [0 -0.91 1.32] wall-cube-bottom-back))
+ (place right-wall-column 0 (translate [-1.2 -1.52 0.91] wall-cube-bottom-back))
+ (key-place 5 0 web-post-tl)
+ (key-place 5 0 (translate [5.6 0 0.05] web-post-tr)))
+
+ (apply union
+ (for [x (range 1 5)]
+ (union
+ (hull (place (- x 1/2) 0 (translate [0 -1 1] wall-cube-bottom-back))
+ (place (+ x 1/2) 0 (translate [0 -1 1] wall-cube-bottom-back))
+ (key-place x 0 web-post-tl)
+ (key-place x 0 web-post-tr))
+ (hull (place (- x 1/2) 0 (translate [0 -1 1] wall-cube-bottom-back))
+ (key-place x 0 web-post-tl)
+ (key-place (- x 1) 0 web-post-tr))))))
+
+ (hull (place (- 5 1/2) 0 (translate [0 -1 1] wall-cube-bottom-back))
+ (place 5 0 (translate [0 -0.91 1.32] wall-cube-bottom-back))
+ (key-place 4 0 web-post-tr)
+ (key-place 5 0 web-post-tl)))
+ (union case-back-cutout
+ case-inside-cutout))))
+
+(def right-wall
+ (let [place case-place]
+ (union
+ (apply union
+ (concat
+ (for [x (range 0 5)]
+ (union
+ (hull (place right-wall-column x (translate [-1 -1 1] (wall-cube-bottom 1/2)))
+ (place 6 x web-post-br)
+ (place 6 x web-post-tr))))
+ (for [x (range 0 4)]
+ (union
+ (hull (place right-wall-column x (translate [-1 -1 1] (wall-cube-bottom 1/2)))
+ (place right-wall-column (inc x) (translate [-1 -1 1] (wall-cube-bottom 1/2)))
+ (place 6 x web-post-br)
+ (place 6 (inc x) web-post-tr))))
+ [(union
+ (hull (place right-wall-column 0 (translate [-1 -1 1] (wall-cube-bottom 1/2)))
+ (place right-wall-column 0 (translate [-1.2 -1.5 0.9] (wall-cube-bottom 1)))
+ (place 6 0 web-post-tr))
+ (hull (place right-wall-column 4 (translate [-1 -1 1] (wall-cube-bottom 1/2)))
+ (place right-wall-column 4 (translate [-1 1 1] (wall-cube-bottom 0)))
+ (place 6 4 web-post-br)))])))))
+
+(def left-wall
+ (let [place case-place]
+ (union
+ (hull (place left-wall-column 0 (translate [1 -1.5 1] wall-cube-bottom-back))
+ (place left-wall-column 1 (translate [1 0 1] wall-cube-bottom-back))
+ (key-place 0 0 web-post-tl)
+ (key-place 0 0 web-post-bl))
+ (hull (place left-wall-column 1 (translate [1 0 1] wall-cube-bottom-back))
+ (place left-wall-column 2 (translate [1 0 1] wall-cube-bottom-back))
+ (key-place 0 0 web-post-bl)
+ (key-place 0 1 web-post-bl))
+ (hull (place left-wall-column 2 (translate [1 0 1] wall-cube-bottom-back))
+ (place left-wall-column 1.71 (translate [1 0 1] wall-cube-bottom-front))
+ (key-place 0 1 web-post-bl)
+ (key-place 0 2 web-post-bl)))))
+
+(def left-inside-wall
+ (let [place case-place]
+ (union
+ (hull (place left-wall-column 1.71 (translate [1 0 1] wall-cube-bottom-front))
+ (key-place 0 2 web-post-bl)
+ (key-place 0 3 web-post-tl))
+ (hull (place left-wall-column 1.71 (translate [2 0 1] wall-cube-bottom-front))
+ (thumb-place 1 1 web-post-tr)
+ (key-place 0 3 web-post-tl)
+ (place left-wall-column 1.71 (translate [1 0 1] wall-cube-bottom-front))
+ (thumb-place 1 1 web-post-tr)
+ (thumb-place 1/2 thumb-back-y (translate [0 -1.7 thumb-case-z] wall-cube-bottom-back))
+ (thumb-place 1/2 thumb-back-y (translate [1 -1.7 (- thumb-case-z 2.9)] wall-cube-bottom-back))))))
+
+(def thumb-back-wall
+ (let [step wall-step
+ top-step 0.05
+ back-y thumb-back-y]
+ (union
+ (hull
+ (thumb-place 1/2 thumb-back-y (translate [0 -1.7 thumb-case-z] wall-cube-bottom-back))
+ (thumb-place 1 1 web-post-tr)
+ (thumb-place 3/2 thumb-back-y (translate [0 -1.7 thumb-case-z] wall-cube-bottom-back))
+ (thumb-place 1 1 web-post-tl))
+ (hull
+ (thumb-place (+ 5/2 0.05) thumb-back-y (translate [1.5 -1.7 thumb-case-z] wall-cube-bottom-back))
+ (thumb-place 3/2 thumb-back-y (translate [0 -1.7 thumb-case-z] wall-cube-bottom-back))
+ (thumb-place 1 1 web-post-tl)
+ (thumb-place 2 1 web-post-tl)))))
+
+(def thumb-left-wall
+ (let [step wall-step
+ place thumb-place
+ wall (+ thumb-left-wall-column 0.001)]
+ (union
+ (hull
+ (thumb-place wall thumb-back-y (translate [1.5 -1.7 thumb-case-z] wall-cube-bottom-back))
+ (thumb-place wall 0 (translate [1.5 0 thumb-case-z] wall-cube-bottom-back))
+ (thumb-place 2 1 web-post-tl)
+ (thumb-place 2 1 web-post-bl))
+ (hull
+ (thumb-place wall 0 (translate [1.5 0 thumb-case-z] wall-cube-bottom-back))
+ (thumb-place 2 0 web-post-tl)
+ (thumb-place 2 1 web-post-bl))
+ (hull
+ (thumb-place wall 0 (translate [1.5 0 thumb-case-z] wall-cube-bottom-back))
+ (thumb-place wall -1 (translate [1.5 0 thumb-case-z] wall-cube-bottom-back))
+ (thumb-place 2 0 web-post-tl)
+ (thumb-place 2 0 web-post-bl))
+ (hull
+ (thumb-place wall -1 (translate [1.5 0 thumb-case-z] wall-cube-bottom-back))
+ (thumb-place 2 -1 web-post-tl)
+ (thumb-place 2 0 web-post-bl))
+ (hull
+ (thumb-place wall -1 (translate [1.5 0 thumb-case-z] wall-cube-bottom-back))
+ (thumb-place wall (+ -1 0.07) (translate [1.5 0.6 (+ thumb-case-z 0.2)] wall-cube-bottom-front))
+ (thumb-place 2 -1 web-post-tl)
+ (thumb-place 2 -1 web-post-bl)))))
+
+(def thumb-front-wall
+ (let [step wall-step
+ wall-cube-top-front 0.05
+ place thumb-place
+ wall (- thumb-front-row 0.04)
+ plate-height (/ (- sa-double-length mount-height) 2)
+ thumb-tl (->> web-post-tl
+ (translate [0 plate-height 0]))
+ thumb-bl (->> web-post-bl
+ (translate [0 (- plate-height) 0]))
+ thumb-tr (->> web-post-tr
+ (translate [-0 plate-height 0]))
+ thumb-br (->> web-post-br
+ (translate [-0 (- plate-height) 0]))]
+
+ (union
+
+ (hull (place (+ 5/2 0.05) wall (translate [1.5 1.5 thumb-case-z] wall-cube-bottom-front))
+ (place (+ 3/2 0.05) wall (translate [0 1.5 thumb-case-z] wall-cube-bottom-front))
+ (place 2 -1 web-post-bl)
+ (place 2 -1 web-post-br))
+
+ (hull (place thumb-right-wall wall (translate [-1 1.5 thumb-case-z] wall-cube-bottom-front))
+ (place (+ 1/2 0.05) wall (translate [0 1.5 thumb-case-z] wall-cube-bottom-front))
+ (place 0 -1/2 thumb-bl)
+ (place 0 -1/2 thumb-br))
+ (hull (place (+ 1/2 0.05) wall (translate [0 1.5 thumb-case-z] wall-cube-bottom-front))
+ (place (+ 3/2 0.05) wall (translate [0 1.5 thumb-case-z] wall-cube-bottom-front))
+ (place 0 -1/2 thumb-bl)
+ (place 1 -1/2 thumb-bl)
+ (place 1 -1/2 thumb-br)
+ (place 2 -1 web-post-br)))))
+
+(def thumb-inside-wall
+ (let [place thumb-place
+ wall (- thumb-front-row 0.04)
+ plate-height (/ (- sa-double-length mount-height) 2)
+ thumb-bl (->> web-post-bl
+ (translate [0 (- plate-height) 0]))
+ thumb-br (->> web-post-br
+ (translate [-0 (- plate-height) 0]))
+ thumb-bottom (->> (cube 3 3 0.001)
+ (translate [13.6 -15 -8]))
+ thumb-top (->> (cube 1 1 1)
+ (translate [13 -11.7 -5.4]))]
+ (hull (place thumb-right-wall wall (translate [-1 1.5 thumb-case-z] wall-cube-bottom-front))
+ (key-place 1 4 web-post-bl)
+ (place 0 -1/2 thumb-br)
+ (case-place 0 4 thumb-top)
+ (case-place 0 4 (translate [-1 10 0] thumb-bottom)))))
+
+(def new-case
+ (union front-wall
+ right-wall
+ back-wall
+ left-wall
+ left-inside-wall
+ thumb-back-wall
+ thumb-left-wall
+ thumb-inside-wall
+ thumb-front-wall))
+
+;;;;;;;;;;;;
+;; Bottom ;;
+;;;;;;;;;;;;
+
+
+(defn bottom [height & p]
+ (->> (project p)
+ (extrude-linear {:height height :twist 0 :convexity 0})
+ (translate [0 0 (/ height 2)])))
+
+(defn bottom-hull [& p]
+ (hull p (bottom 1 p)))
+
+
+(def bottom-key-guard (->> (cube mount-width mount-height web-thickness)
+ (translate [0 0 (+ (- (/ web-thickness 2)) -5)])))
+(def bottom-front-key-guard (->> (cube mount-width (/ mount-height 2) web-thickness)
+ (translate [0 (/ mount-height 4) (+ (- (/ web-thickness 2)) -5)])))
+
+(defn stand-at [diameter placement]
+ (let [bumper-radius (/ diameter 2)
+ stand-diameter (+ diameter 2)
+ stand-radius (/ stand-diameter 2)]
+ (difference (->> (sphere stand-radius)
+ (translate [0 0 (+ (/ stand-radius -2) -4.5)])
+ placement
+ bottom-hull)
+ (->> (cube stand-diameter stand-diameter stand-radius)
+ (translate [0 0 (/ stand-radius -2)])
+ placement)
+ (->> (sphere bumper-radius)
+ (translate [0 0 (+ (/ stand-radius -2) -4.5)])
+ placement
+ (bottom 1.5)))))
+
+(def bottom-plate
+ (union
+; (apply union
+; (for [column columns
+; row (drop-last rows) ;;
+; :when (or (not= column 0)
+; (not= row 4))]
+; (->> bottom-key-guard
+; (key-place column row))))
+; (thumb-layout (rotate (/ π 2) [0 0 1] bottom-key-guard))
+; (apply union
+; (for [column columns
+; row [(last rows)] ;;
+; :when (or (not= column 0)
+; (not= row 4))]
+; (->> bottom-front-key-guard
+; (key-place column row))))
+ (let [shift #(translate [0 0 (+ (- web-thickness) -5)] %)
+ thumb-ridge-height 1
+ thumb-back-offset -1.28
+ thumb-left-offset 1.13
+ thumb-front-offset 0.56
+ front-offset 0.63
+ left-offset 0.9
+ right-offset -0.73
+ back-offset -0.85
+ web-post-tl (shift web-post-tl)
+ web-post-tr (shift web-post-tr)
+ web-post-br (shift web-post-br)
+ web-post-bl (shift web-post-bl)
+ half-shift-correction #(translate [0 (/ mount-height 2) 0] %)
+ half-post-br (half-shift-correction web-post-br)
+ half-post-bl (half-shift-correction web-post-bl)
+ row-connections (concat
+ (for [column (drop-last columns)
+ row (drop-last rows)
+ :when (or (not= column 0)
+ (not= row 4))]
+ (triangle-hulls
+ (key-place (inc column) row web-post-tl)
+ (key-place column row web-post-tr)
+ (key-place (inc column) row web-post-bl)
+ (key-place column row web-post-br)))
+ (for [column (drop-last columns)
+ row [(last rows)]
+ :when (or (not= column 0)
+ (not= row 4))]
+ (triangle-hulls
+ (key-place (inc column) row web-post-tl)
+ (key-place column row web-post-tr)
+ (key-place (inc column) row half-post-bl)
+ (key-place column row half-post-br))))
+ column-connections (for [column columns
+ row (drop-last rows)
+ :when (or (not= column 0)
+ (not= row 3))]
+ (triangle-hulls
+ (key-place column row web-post-bl)
+ (key-place column row web-post-br)
+ (key-place column (inc row) web-post-tl)
+ (key-place column (inc row) web-post-tr)))
+ diagonal-connections (for [column (drop-last columns)
+ row (drop-last rows)
+ :when (or (not= column 0)
+ (not= row 3))]
+ (triangle-hulls
+ (key-place column row web-post-br)
+ (key-place column (inc row) web-post-tr)
+ (key-place (inc column) row web-post-bl)
+ (key-place (inc column) (inc row) web-post-tl)))
+ main-keys-bottom (concat row-connections
+ column-connections
+ diagonal-connections)
+ front-wall (concat
+ (for [x (range 2 5)]
+ (union
+ (hull (bottom-place (- x 1/2) 4 (translate [0 front-offset 1] wall-cube-bottom-front))
+ (bottom-place (+ x 1/2) 4 (translate [0 front-offset 1] wall-cube-bottom-front))
+ (key-place x 4 half-post-bl)
+ (key-place x 4 half-post-br))
+ (bottom-hull
+ (bottom-place (- x 1/2) 4 (translate [0 front-offset 1] wall-cube-bottom-front))
+ (bottom-place (+ x 1/2) 4 (translate [0 front-offset 1] wall-cube-bottom-front)))
+ (hull (bottom-place (- x 1/2) 4 (translate [0 front-offset 1] wall-cube-bottom-front))
+ (key-place x 4 half-post-bl)
+ (key-place (- x 1) 4 half-post-br))))
+ [(hull (bottom-place right-wall-column 4 (translate [right-offset front-offset 1] wall-cube-bottom-front))
+ (bottom-place (- right-wall-column 1) 4 (translate [0 front-offset 1] wall-cube-bottom-front))
+ (key-place 5 4 half-post-bl)
+ (key-place 5 4 half-post-br))
+ (bottom-hull (bottom-place right-wall-column 4 (translate [right-offset front-offset 1] wall-cube-bottom-front))
+ (bottom-place (- right-wall-column 1) 4 (translate [0 front-offset 1] wall-cube-bottom-front)))
+ (bottom-hull (bottom-place (+ 4 1/2) 4 (translate [0 front-offset 1] wall-cube-bottom-front))
+ (bottom-place (- right-wall-column 1) 4 (translate [0 front-offset 1] wall-cube-bottom-front)))
+ (hull (bottom-place (+ 4 1/2) 4 (translate [0 front-offset 1] wall-cube-bottom-front))
+ (bottom-place (- right-wall-column 1) 4 (translate [0 front-offset 1] wall-cube-bottom-front))
+ (key-place 4 4 half-post-br)
+ (key-place 5 4 half-post-bl))])
+ right-wall (concat
+ (for [x (range 0 4)]
+ (bottom-hull (bottom-place right-wall-column x (translate [right-offset 0 1] (wall-cube-bottom 1/2)))
+ (key-place 5 x web-post-br)
+ (key-place 5 x web-post-tr)))
+ (for [x (range 0 4)]
+ (bottom-hull (bottom-place right-wall-column x (translate [right-offset 0 1] (wall-cube-bottom 1/2)))
+ (bottom-place right-wall-column (inc x) (translate [right-offset 0 1] (wall-cube-bottom 1/2)))
+ (key-place 5 x web-post-br)
+ (key-place 5 (inc x) web-post-tr)))
+ [(union
+ (bottom-hull (bottom-place right-wall-column 0 (translate [right-offset 0 1] (wall-cube-bottom 1/2)))
+ (bottom-place right-wall-column 0.017 (translate [(- right-offset 0.29) -1 1.15] (wall-cube-bottom 1)))
+ (key-place 5 0 web-post-tr))
+ (bottom-hull (bottom-place right-wall-column 4 (translate [right-offset 0 1] (wall-cube-bottom 1/2)))
+ (bottom-place right-wall-column 4 (translate [right-offset front-offset 1] (wall-cube-bottom 0)))
+ (key-place 5 4 half-post-br))
+ (bottom-hull (bottom-place right-wall-column 4 (translate [right-offset 0 1] (wall-cube-bottom 1/2)))
+ (key-place 5 4 half-post-br)
+ (key-place 5 4 web-post-tr)))])
+ back-wall (concat
+ (for [x (range 1 6)]
+ (union
+ ((if (and (>= x 1) (<= x 3)) hull bottom-hull) ; We need a little extra space in the hull on these keys
+ (do (bottom-place (- x 1/2) 0 (translate [0 back-offset 1] wall-cube-bottom-back)))
+ (if (= x 5)
+ (do (bottom-place (+ x 1/2) 0 (translate [10.6 (- back-offset 0.35) 1.30] wall-cube-bottom-back)))
+ (do (bottom-place (+ x 1/2) 0 (translate [0 back-offset 1] wall-cube-bottom-back))))
+ (key-place x 0 web-post-tl)
+ (key-place x 0 web-post-tr))
+ (bottom-hull
+ (key-place x 0 web-post-tl)
+ (key-place x 0 web-post-tr))
+ (bottom-hull
+ (key-place x 0 web-post-tl)
+ (key-place (- x 1) 0 web-post-tr))
+ ((if (= x 4) bottom-hull hull)
+ (bottom-place (- x 1/2) 0 (translate [0 back-offset 1] wall-cube-bottom-back))
+ (key-place x 0 web-post-tl)
+ (key-place (- x 1) 0 web-post-tr))))
+ [(bottom-hull (bottom-place left-wall-column 0 (translate [left-offset back-offset 1.2] wall-cube-bottom-back))
+ (bottom-place (+ left-wall-column 1) 0 (translate [0 back-offset 1.2] wall-cube-bottom-back))
+ (key-place 0 0 web-post-tl)
+ (key-place 0 0 web-post-tr))]
+
+ )
+ left-wall (let [place bottom-place]
+ [
+ (bottom-hull (place left-wall-column 0 (translate [left-offset back-offset 1.2] wall-cube-bottom-back))
+ (place left-wall-column 1 (translate [left-offset 1.5 1/2] wall-cube-bottom-back))
+ (key-place 0 0 web-post-tl)
+ (key-place 0 0 web-post-bl))
+ (bottom-hull (place left-wall-column 1 (translate [left-offset 0 1/2] wall-cube-bottom-back))
+ (place left-wall-column 2 (translate [left-offset 1.5 0.7] wall-cube-bottom-back))
+ (key-place 0 0 web-post-bl)
+ (key-place 0 1 web-post-bl))
+ (bottom-hull (place left-wall-column 2 (translate [left-offset 0 0.7] wall-cube-bottom-back))
+ (place left-wall-column 1.6666 (translate [left-offset 0 1] wall-cube-bottom-front))
+ (key-place 0 1 web-post-bl)
+ (key-place 0 2 web-post-bl))
+ (bottom-hull (place left-wall-column 1.6666 (translate [left-offset 0 1] wall-cube-bottom-front))
+ (key-place 0 2 web-post-bl)
+ (key-place 0 3 web-post-tl))
+
+ ])
+ thumbs [(hull (thumb-place 0 -1/2 web-post-bl)
+ (thumb-place 0 -1/2 web-post-tl)
+ (thumb-place 1 -1/2 web-post-tr)
+ (thumb-place 1 -1/2 web-post-br))
+ (hull (thumb-place 1 -1/2 web-post-tr)
+ (thumb-place 1 -1/2 web-post-tl)
+ (thumb-place 1 1 web-post-bl)
+ (thumb-place 1 1 web-post-br))
+ (hull (thumb-place 2 -1 web-post-tr)
+ (thumb-place 2 -1 web-post-tl)
+ (thumb-place 2 0 web-post-bl)
+ (thumb-place 2 0 web-post-br))
+ (hull (thumb-place 2 0 web-post-tr)
+ (thumb-place 2 0 web-post-tl)
+ (thumb-place 2 1 web-post-bl)
+ (thumb-place 2 1 web-post-br))
+ (triangle-hulls (thumb-place 2 1 web-post-tr)
+ (thumb-place 1 1 web-post-tl)
+ (thumb-place 2 1 web-post-br)
+ (thumb-place 1 1 web-post-bl)
+ (thumb-place 2 0 web-post-tr)
+ (thumb-place 1 -1/2 web-post-tl)
+ (thumb-place 2 0 web-post-br)
+ (thumb-place 1 -1/2 web-post-bl)
+ (thumb-place 2 -1 web-post-tr)
+ (thumb-place 2 -1 web-post-br))
+ (hull (thumb-place 2 -1 web-post-br)
+ (thumb-place 1 -1/2 web-post-bl)
+ (thumb-place 1 -1 web-post-bl))
+ (hull (thumb-place 1 -1/2 web-post-bl)
+ (thumb-place 1 -1 web-post-bl)
+ (thumb-place 1 -1/2 web-post-br)
+ (thumb-place 1 -1 web-post-br))
+ (hull (thumb-place 0 -1/2 web-post-bl)
+ (thumb-place 0 -1 web-post-bl)
+ (thumb-place 0 -1/2 web-post-br)
+ (thumb-place 0 -1 web-post-br))
+ (hull (thumb-place 0 -1/2 web-post-bl)
+ (thumb-place 0 -1 web-post-bl)
+ (thumb-place 1 -1/2 web-post-br)
+ (thumb-place 1 -1 web-post-br))]
+ thumb-back-wall [
+ (bottom-hull
+ (thumb-place 1/2 thumb-back-y (translate [-0.2 thumb-back-offset thumb-ridge-height] wall-cube-bottom-back))
+ (thumb-place 1 1 web-post-tr)
+ (thumb-place 3/2 thumb-back-y (translate [-0.2 thumb-back-offset thumb-ridge-height] wall-cube-bottom-back))
+ (thumb-place 1 1 web-post-tl))
+ (bottom-hull
+ (thumb-place (+ 5/2 0.05) thumb-back-y (translate [thumb-left-offset thumb-back-offset thumb-ridge-height] wall-cube-bottom-back))
+ (thumb-place 3/2 thumb-back-y (translate [-0.2 thumb-back-offset thumb-ridge-height] wall-cube-bottom-back))
+ (thumb-place 1 1 web-post-tl)
+ (thumb-place 2 1 web-post-tl))
+ (bottom-hull
+ (thumb-place 1/2 thumb-back-y (translate [-0.2 thumb-back-offset thumb-ridge-height] wall-cube-bottom-back))
+ (bottom-place left-wall-column 1.6666 (translate [left-offset 0 thumb-ridge-height] wall-cube-bottom-front))
+ (key-place 0 3 web-post-tl)
+ (thumb-place 1 1 web-post-tr))
+ ]
+ thumb-left-wall [(bottom-hull
+ (thumb-place thumb-left-wall-column thumb-back-y (translate [thumb-left-offset thumb-back-offset thumb-ridge-height] wall-cube-bottom-back))
+ (thumb-place thumb-left-wall-column 0 (translate [thumb-left-offset 0 thumb-ridge-height] wall-cube-bottom-back))
+ (thumb-place 2 1 web-post-tl)
+ (thumb-place 2 1 web-post-bl))
+ (bottom-hull
+ (thumb-place thumb-left-wall-column 0 (translate [thumb-left-offset 0 thumb-ridge-height] wall-cube-bottom-back))
+ (thumb-place 2 0 web-post-tl)
+ (thumb-place 2 1 web-post-bl))
+ (bottom-hull
+ (thumb-place thumb-left-wall-column 0 (translate [thumb-left-offset 0 thumb-ridge-height] wall-cube-bottom-back))
+ (thumb-place thumb-left-wall-column -1 (translate [thumb-left-offset 0 thumb-ridge-height] wall-cube-bottom-back))
+ (thumb-place 2 0 web-post-tl)
+ (thumb-place 2 0 web-post-bl))
+ (bottom-hull
+ (thumb-place thumb-left-wall-column -1 (translate [thumb-left-offset 0 thumb-ridge-height] wall-cube-bottom-back))
+ (thumb-place 2 -1 web-post-tl)
+ (thumb-place 2 0 web-post-bl))
+ (bottom-hull
+ (thumb-place thumb-left-wall-column -1 (translate [thumb-left-offset 0 thumb-ridge-height] wall-cube-bottom-back))
+ (thumb-place thumb-left-wall-column (+ -1 0.07) (translate [thumb-left-offset 1 thumb-ridge-height] wall-cube-bottom-front))
+ (thumb-place 2 -1 web-post-tl)
+ (thumb-place 2 -1 web-post-bl))]
+ thumb-front-wall [(bottom-hull (thumb-place (+ 5/2 0.05) thumb-front-row (translate [thumb-left-offset thumb-front-offset thumb-ridge-height] wall-cube-bottom-front))
+ (thumb-place (+ 3/2 0.05) thumb-front-row (translate [0 thumb-front-offset thumb-ridge-height] wall-cube-bottom-front))
+ (thumb-place 2 -1 web-post-bl)
+ (thumb-place 2 -1 web-post-br))
+ (bottom-hull (thumb-place (+ 1/2 0.05) thumb-front-row (translate [0 thumb-front-offset thumb-ridge-height] wall-cube-bottom-front))
+ (thumb-place (+ 3/2 0.05) thumb-front-row (translate [0 thumb-front-offset thumb-ridge-height] wall-cube-bottom-front))
+ (thumb-place 0 -1 web-post-bl)
+ (thumb-place 1 -1 web-post-bl)
+ (thumb-place 1 -1 web-post-br)
+ (thumb-place 2 -1 web-post-br))
+ (bottom-hull (thumb-place thumb-right-wall thumb-front-row (translate [-1.12 thumb-front-offset thumb-ridge-height] wall-cube-bottom-front))
+ (thumb-place (+ 1/2 0.05) thumb-front-row (translate [0 thumb-front-offset thumb-ridge-height] wall-cube-bottom-front))
+ (thumb-place 0 -1 web-post-bl)
+ (thumb-place 0 -1 web-post-br))]
+ thumb-inside [
+; (triangle-hulls
+; (thumb-place 1 1 web-post-tr)
+; (key-place 0 3 web-post-tl)
+; (thumb-place 1 1 web-post-br)
+; (key-place 0 3 web-post-bl)
+; (thumb-place 1 -1/2 web-post-tr)
+; (thumb-place 0 -1/2 web-post-tl)
+; (key-place 0 3 web-post-bl)
+; (thumb-place 0 -1/2 web-post-tr)
+; (key-place 0 3 web-post-br)
+; (key-place 1 3 web-post-bl)
+; (thumb-place 0 -1/2 web-post-tr)
+; (key-place 1 4 web-post-tl)
+; (key-place 1 4 half-post-bl))
+
+; (bottom-hull
+; (thumb-place 0 -1/2 web-post-tr)
+; (thumb-place 0 -1/2 web-post-br)
+; (key-place 1 4 half-post-bl))
+
+ (hull
+ (key-place 1 4 half-post-bl)
+ (key-place 1 4 half-post-br)
+ (bottom-place (- 2 1/2) 4 (translate [0 front-offset 1] wall-cube-bottom-front))
+ (bottom-place 0.75 4 (translate [0 front-offset 0] wall-cube-bottom-front)))
+
+ (bottom-hull
+ (bottom-place (- 2 1/2) 4 (translate [0 front-offset 1] wall-cube-bottom-front))
+ (bottom-place 0.75 4 (translate [0 front-offset 0] wall-cube-bottom-front)))
+
+ (bottom-hull
+ (thumb-place 0 -1 web-post-br)
+ (thumb-place 0 -1/2 web-post-br)
+ (thumb-place thumb-right-wall thumb-front-row (translate [-1.12 thumb-front-offset thumb-ridge-height] wall-cube-bottom-front))
+ (key-place 1 4 (translate [2.5 -2 -0.8] web-post-bl))
+ (key-place 1 4 (translate [1.23 -4 0] web-post-bl))
+ (key-place 1 4 half-post-bl))]
+
+ stands (let [bumper-diameter 9.6] ; 3/8 = 9.6 1/2 = 12.7
+ [(stand-at bumper-diameter #(key-place 0 1 %))
+ (stand-at bumper-diameter #(thumb-place 1 -1/2 %))
+ (stand-at bumper-diameter #(key-place 5 0 %))
+ (stand-at bumper-diameter #(key-place 5 3 %) )])]
+ (apply union
+ (concat
+ ;main-keys-bottom
+ front-wall
+ right-wall
+ back-wall
+ left-wall
+ ;thumbs
+ thumb-back-wall
+ thumb-left-wall
+ thumb-front-wall
+ thumb-inside
+ ;stands)))))
+ )))))
+
+(def screw-hole (->> (cylinder 1.5 60)
+ (translate [0 0 3/2])
+ (with-fn wall-sphere-n)))
+
+(def screw-holes
+ (union
+ (key-place (+ 4.7) 1/2 screw-hole)
+ (key-place (+ 4.7) (+ 3 1/2) screw-hole)
+ (thumb-place 2 -1/2 screw-hole)))
+
+(defn circuit-cover [width length height]
+ (let [cover-sphere-radius 1
+ cover-sphere (->> (sphere cover-sphere-radius)
+ (with-fn 20))
+ cover-sphere-z (+ (- height) (- cover-sphere-radius))
+ cover-sphere-x (+ (/ width 2) cover-sphere-radius)
+ cover-sphere-y (+ (/ length 2) (+ cover-sphere-radius))
+ cover-sphere-tl (->> cover-sphere
+ (translate [(- cover-sphere-x) (- cover-sphere-y) cover-sphere-z])
+ (key-place 1/2 3/2))
+ cover-sphere-tr (->> cover-sphere
+ (translate [cover-sphere-x (- cover-sphere-y) cover-sphere-z])
+ (key-place 1/2 3/2))
+ cover-sphere-br (->> cover-sphere
+ (translate [cover-sphere-x cover-sphere-y cover-sphere-z])
+ (key-place 1/2 3/2))
+ cover-sphere-bl (->> cover-sphere
+ (translate [(- cover-sphere-x) cover-sphere-y cover-sphere-z])
+ (key-place 1/2 3/2))
+
+ lower-to-bottom #(translate [0 0 (+ (- cover-sphere-radius) -5.5)] %)
+ bl (->> cover-sphere lower-to-bottom (key-place 0 1/2))
+ br (->> cover-sphere lower-to-bottom (key-place 1 1/2))
+ tl (->> cover-sphere lower-to-bottom (key-place 0 5/2))
+ tr (->> cover-sphere lower-to-bottom (key-place 1 5/2))
+
+ mlb (->> cover-sphere
+ (translate [(- cover-sphere-x) 0 (+ (- height) -1)])
+ (key-place 1/2 3/2))
+ mrb (->> cover-sphere
+ (translate [cover-sphere-x 0 (+ (- height) -1)])
+ (key-place 1/2 3/2))
+
+ mlt (->> cover-sphere
+ (translate [(+ (- cover-sphere-x) -4) 0 -6])
+ (key-place 1/2 3/2))
+ mrt (->> cover-sphere
+ (translate [(+ cover-sphere-x 4) 0 -6])
+ (key-place 1/2 3/2))]
+ (union
+ (hull cover-sphere-bl cover-sphere-br cover-sphere-tl cover-sphere-tr)
+ (hull cover-sphere-br cover-sphere-bl bl br)
+ (hull cover-sphere-tr cover-sphere-tl tl tr)
+ (hull cover-sphere-tl tl mlb mlt)
+ (hull cover-sphere-bl bl mlb mlt)
+ (hull cover-sphere-tr tr mrb mrt)
+ (hull cover-sphere-br br mrb mrt))))
+
+(def io-exp-width 10)
+(def io-exp-height 8)
+(def io-exp-length 36)
+
+(def teensy-width 20)
+(def teensy-height 12)
+(def teensy-length 33)
+
+(def io-exp-cover (circuit-cover io-exp-width io-exp-length io-exp-height))
+(def teensy-cover (circuit-cover teensy-width teensy-length teensy-height))
+
+(def trrs-diameter 6.6)
+(def trrs-radius (/ trrs-diameter 2))
+(def trrs-hole-depth 10)
+
+(def trrs-hole (->> (union (cylinder trrs-radius trrs-hole-depth)
+ (->> (cube trrs-diameter (+ trrs-radius 5) trrs-hole-depth)
+ (translate [0 (/ (+ trrs-radius 5) 2) 0])))
+ (rotate (/ π 2) [1 0 0])
+ (translate [0 (+ (/ mount-height 2) 4) (- trrs-radius)])
+ (with-fn 50)))
+
+(def trrs-hole-just-circle
+ (->> (cylinder trrs-radius trrs-hole-depth)
+ (rotate (/ π 2) [1 0 0])
+ (translate [0 (+ (/ mount-height 2) 4) (- trrs-radius)])
+ (with-fn 50)
+ (key-place 1/2 0)))
+
+(def trrs-box-hole (->> (cube 14 14 7 )
+ (translate [0 1 -2])))
+
+
+(def trrs-cutout
+ (->> (union trrs-hole
+ trrs-box-hole)
+ (key-place 1/2 0)))
+
+(def teensy-pcb-thickness 1.6)
+(def teensy-offset-height 5)
+
+(def teensy-pcb (->> (cube 18 30.5 teensy-pcb-thickness)
+ (translate [0 0 (+ (/ teensy-pcb-thickness -2) (- teensy-offset-height))])
+ (key-place 1/2 3/2)
+ (color [1 0 0])))
+
+(def teensy-support
+ (difference
+ (union
+ (->> (cube 5 5 9)
+ (translate [0 0 -2])
+ (key-place 1/2 3/2)
+ (color [0 1 0]))
+ (hull (->> (cube 5 6 9)
+ (translate [0 0 -2])
+ (key-place 1/2 2)
+ (color [0 0 1]))
+ (->> (cube 5 5 (+ teensy-pcb-thickness 5))
+ (translate [0 (/ 30.5 -2) (+ (- teensy-offset-height)
+ #_(/ (+ teensy-pcb-thickness 5) -2)
+ )])
+ (key-place 1/2 3/2)
+ (color [0 0 1]))
+
+ ))
+ teensy-pcb
+ (->> (cube 18 31 (+ teensy-pcb-thickness 1))
+ (translate [0 1.5 (+ (/ teensy-pcb-thickness -2) (- teensy-offset-height) -1.5)])
+ (key-place 1/2 3/2)
+ (color [1 0 0]))
+ ))
+
+(def usb-cutout
+ (let [hole-height 5.2
+ side-radius (/ hole-height 2)
+ hole-width 10.75
+ side-cylinder (->> (cylinder side-radius teensy-length)
+ (with-fn 20)
+ (translate [(/ (- hole-width hole-height) 2) 0 0]))]
+ (->> (hull side-cylinder
+ (mirror [-1 0 0] side-cylinder))
+ (rotate (/ π 2) [1 0 0])
+ (translate [-29 50 4]))))
+
+(def hdmi-cutout
+ (let [hole-height 6.2
+ side-radius (/ hole-height 2)
+ hole-width 16
+ side-cylinder (->> (cylinder side-radius teensy-length)
+ (with-fn 20)
+ (translate [(/ (- hole-width hole-height) 2) 0 0]))]
+ (->> (hull side-cylinder
+ (mirror [-1 0 0] side-cylinder))
+ (rotate (/ π 2) [1 0 0])
+ (rotate (/ π 14) [0 0 1])
+ (translate [-70 -15 4]))))
+
+
+;;;;;;;;;;;;;;;;
+;; Tolerances ;;
+;;;;;;;;;;;;;;;;
+
+(def tolerance 0.3)
+
+(defn offset-case-place [offset block]
+ (->> block
+ (translate [0 0 1])
+ (translate offset)))
+
+(def case-tolerance
+ (let [place offset-case-place
+ t tolerance
+ -t (- tolerance)
+ th (/ t 2)
+ -th (/ -t 2)
+ -tq (/ -t 4)
+ t2 (* t 2)
+ -t2 (* -t 2)]
+ (union
+ (place [0 0 0] case-inside-cutout)
+ (place [0 0 0] new-case)
+ (place [0 0 -t] new-case)
+ (place [t2 -t -tq] front-wall)
+ (place [t t -th] front-wall)
+ (place [0 -t -th] front-wall)
+ (place [-t 0 -t] right-wall)
+ (place [t 0 -t] right-wall)
+ (place [0 th -th] back-wall)
+ (place [0 -th -th] back-wall)
+ (place [-45 50 54.3] (rotate (/ π 5.5) [1 0 0] (cube 10 10 10)))
+ (place [80 49 29.83] (rotate (/ π 5.5) [1 0 0] (cube 10 10 10)))
+ (place [-t -t -t] left-wall)
+ (place [t 0 -t2] left-wall)
+ (place [-t 0 0] left-inside-wall)
+ (place [-t 0 -t] left-inside-wall)
+ (place [-t -t -t] left-inside-wall)
+ (place [0 -t -t2] thumb-back-wall)
+ (place [-t t -th] thumb-back-wall)
+ (place [0 t -t] thumb-back-wall)
+ (place [t 0 (* -t 1.5)] thumb-left-wall)
+ (place [-t 0 -t] thumb-left-wall)
+ (place [t t -t] thumb-inside-wall)
+ (place [-t 0 -t] thumb-inside-wall)
+ (place [-t t -t] thumb-front-wall)
+ (place [t 0 -t] thumb-front-wall)
+ (place [0 (* 2 -t) -t] thumb-front-wall))))
+
+
+;;;;;;;;;;;;;;;;
+;; Palm Rests ;;
+;;;;;;;;;;;;;;;;
+
+(defn bezier-conic [p0 p1 p2 steps]
+ (let [step1 (/ (- p1 p0) steps)
+ step2 (/ (- p2 p1) steps)]
+ (for [i (range steps)]
+ (let [point1 (+ p0 (* step1 i))
+ point2 (+ p1 (* step2 i))
+ point3 (+ p0 (* step1 (+ i 1)))
+ point4 (+ p1 (* step2 (+ i 1)))
+ bpoint1 (+ point1 (* (- point2 point1) (/ i steps)))
+ bpoint2 (+ point3 (* (- point4 point3) (/ (+ i 1) steps)))]
+ (polygon [bpoint1 bpoint2 p1])))))
+
+(defn bezier-cone [d h curve steps & {:keys [curve2] :or {curve2 (/ h 2)}}]
+ (let [p0 [(/ d 2) 0]
+ p1 [(+ curve (/ d 4)) curve2]
+ p2 [0 h]]
+ (cond
+ (< (nth p1 0) (/ d 4)) ; concave
+ (do (->> (union (polygon [[0 0] p0 p1 p2 [0 h]])
+ (bezier-conic p0 p1 p2 steps))
+ (extrude-rotate {:fn steps})))
+ (> (nth p1 0) (/ d 4)) ; convex
+ (do (->> (difference (polygon [[0 0] p0 p1 p2 [0 h]])
+ (bezier-conic p0 p1 p2 steps))
+ (extrude-rotate {:fn steps}))))))
+
+(def palm-rest
+ (let [p0 [15 0]
+ p1 [25 14]
+ p2 [7 30]
+ stand-diameter 9.6
+ rest-sphere-n 180 ; 30 for faster renders, 170 for printing
+ profile-sphere-n (* rest-sphere-n 2)
+ floor (->> (cube 300 300 50)
+ (translate [0 0 -25]))
+
+ profile-cyl (->> (cylinder 200 50)
+ (with-fn profile-sphere-n))
+
+ rest-place #(->> % (rotate (/ π 40) [0 1 1])
+ (rotate (/ π 11) [1 0 0])
+ (rotate (/ π 12) [0 1 0])
+ (translate [22 -100 7]))
+
+ front-profile (->> (difference profile-cyl
+ (scale [1.4 0.81 1.1] profile-cyl))
+ (scale [1 1.4 1])
+ (translate [0 -225 55])
+ (rotate (/ π 3.2) [-1 -0.2 -0.2]))
+
+ bottom-profile (->> (cylinder 100 200)
+ (with-fn profile-sphere-n)
+ (rotate (/ π 2.3) [0 1 0])
+ (translate [0 0 -65])
+ (scale [1 1.1 1]))
+
+ base-shape (->> (bezier-cone 80 100 43 rest-sphere-n :curve2 43)
+ (rotate (/ π 2) [-1 0 0])
+ (translate [0 -10 0])
+ (scale [1.1 1 1]))
+
+ base-rest-shape (difference
+ (rest-place
+ (difference base-shape
+ front-profile
+ bottom-profile
+ (scale [0.93 0.93 0.93] base-shape)))
+ floor)
+
+ shape-profile (->> (project base-rest-shape)
+ (extrude-linear {:height 20}))
+
+ side-profile (->> (difference (scale [1.1 1.1 1] shape-profile)
+ (translate [3 -12 0]
+ (scale [0.97 0.97 1.2] shape-profile)))
+ (rotate (/ π 26) [0 1 0])
+ (translate [-2 9 50]))
+
+ rest-shape (difference base-rest-shape
+ side-profile)
+
+ inner-rest #(intersection
+ % (intersection
+ (rest-place base-shape)
+ (->> (project rest-shape)
+ (extrude-linear {:height 100})
+ (translate [0 0 (/ 100 2)]))))
+
+ stand-place #(translate [24 -60 0] %)
+
+ front-rect (->> (prism 15 7 50 5 -2)
+ (rotate (/ π 1) [1 0 0])
+ (rotate (/ π 2) [0 0 1])
+ (rotate (/ π 4) [-1 0 0])
+ (rotate (/ π 6.8) [0 1 0])
+ (translate [17.5 15 35])
+ stand-place)
+
+ front-rect-diff (->> (cube 100 30 30)
+ (rotate (/ π 4) [1 0 0])
+ (rotate (/ π 20) [0 0 -1])
+ (translate [0 -37 31]))
+
+ back-neg-rect (->> (prism 20.3 8 43 3 1)
+ (rotate (/ π 3.5) [1 0 0])
+ (translate [-10.15 -35 -5])
+ stand-place)
+
+ back-neg-rect-diff (->> (cube 30 30 30)
+ (rotate (/ π 6) [-1 0 0])
+ (translate [-7.3 -80.6 28])
+ stand-place)
+
+ back-pos-rect-1 (->> (prism 12 10 60 3 4)
+ (rotate (/ π 2) [0 0 1])
+ (rotate (/ π 9) [-1 0 0])
+ (translate [5 -56 11.8])
+ stand-place
+ inner-rest)
+
+ back-pos-rect-2 (->> (prism 15 5 31 3 -2)
+ (rotate (/ π 6.8) [-1 0 0])
+ (translate [-7.5 -66 23.5])
+ stand-place)
+
+ inner-support (->> (bezier-cone 90 12 -10 rest-sphere-n)
+ (rotate (/ π 1.01) [-1 0 0])
+ (rotate (/ π 20) [0 1 0])
+ (translate [3 -57 54])
+ stand-place
+ inner-rest)
+
+ bottom-rect (->> (cube 30 20 7)
+ (rotate (/ π 2) [0 0 1])
+ (translate [0 -19.3 1])
+ stand-place)
+ stands (difference (union front-rect
+ (translate [48 0 0] (mirror [-1 0 0] front-rect))
+ bottom-rect
+ inner-support
+ back-neg-rect
+ back-pos-rect-1
+ back-pos-rect-2)
+ (translate [0 (- tolerance) 0] bottom-plate)
+ back-neg-rect-diff
+ bottom-plate
+ case-tolerance
+ front-rect-diff
+ floor)]
+ (union stands
+ rest-shape)))
+
+
+(defn rest-alignment-shapes [d h]
+ (let [shape (difference
+ (->> (cylinder d h)
+ (rotate (/ π 2) [0 0 1])
+ (rotate (/ π 4) [-1 0 0])
+ (translate [14.5 0 0])
+ (with-fn 20))
+ (->> (cube 10 10 10)
+ (rotate (/ π 5) [-1 0 0])
+ (translate [14.5 2 7.5])))]
+
+ (translate [24 -51 19]
+ (union
+ (translate [-0.8 -1.75 -1.75] shape)
+ (mirror [-1 0 0] shape)))))
+
+(def case-alignment-male (rest-alignment-shapes 1 2))
+(def case-alignment-female (rest-alignment-shapes (+ tolerance 1) 3))
+
+
+;;;;;;;;;;;;;;;;;;
+;; Final Export ;;
+;;;;;;;;;;;;;;;;;;
+
+(def dactyl-bottom-right
+ (difference
+ bottom-plate
+ case-tolerance
+ usb-cutout
+ hdmi-cutout))
+
+(def dactyl-bottom-left
+ (mirror [-1 0 0]
+ (difference
+ bottom-plate
+ case-tolerance
+ hdmi-cutout)))
+
+(def dactyl-top-right
+ (offset-case-place [0 0 0] (union
+ ; thumbcaps
+ ; caps
+ (difference
+ (union key-holes
+ connectors
+ thumb
+ new-case
+ teensy-support)
+ trrs-hole-just-circle
+ screw-holes))))
+
+(def dactyl-top-left
+ (mirror [-1 0 0]
+ (offset-case-place [0 0 0] (union
+ ;thumbcaps
+ ; caps
+ (difference
+ (union key-holes
+ connectors
+ thumb
+ new-case)
+ trrs-hole-just-circle
+ screw-holes)))))
+
+(def dactyl-rest-left
+ (mirror [-1 0 0]
+ (union palm-rest
+ case-alignment-male)))
+
+
+(def dactyl-rest-right
+ (union palm-rest
+ case-alignment-male))
+
+(def dactyl-combined-left
+ (union dactyl-top-left
+ dactyl-bottom-left
+ dactyl-rest-left))
+
+(def dactyl-combined-right
+ (union dactyl-top-right
+ dactyl-bottom-right
+ dactyl-rest-right))
+
+(spit "things/dactyl-top-right.scad"
+ (write-scad dactyl-top-right))
+
+(spit "things/dactyl-bottom-right.scad"
+ (write-scad dactyl-bottom-right))
+
+(spit "things/dactyl-rest-right.scad"
+ (write-scad dactyl-rest-right))
+
+(spit "things/dactyl-combined-right.scad"
+ (write-scad dactyl-combined-right))
+
+(spit "things/dactyl-top-left.scad"
+ (write-scad dactyl-top-left))
+
+(spit "things/dactyl-bottom-left.scad"
+ (write-scad dactyl-bottom-left))
+
+(spit "things/dactyl-rest-left.scad"
+ (write-scad dactyl-rest-left))
+
+(spit "things/dactyl-combined-left.scad"
+ (write-scad dactyl-combined-left)) \ No newline at end of file