1241 lines
56 KiB
Clojure
1241 lines
56 KiB
Clojure
|
(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]))
|
|||
|
|
|||
|
;;;;;;;;;;;;;;;;;
|
|||
|
;; Switch Hole ;;
|
|||
|
;;;;;;;;;;;;;;;;;
|
|||
|
|
|||
|
(def keyswitch-height 14.4) ;; Was 14.1, then 14.25
|
|||
|
(def keyswitch-width 14.4)
|
|||
|
|
|||
|
(def sa-profile-key-height 12.7)
|
|||
|
|
|||
|
(def plate-thickness 4)
|
|||
|
(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))
|
|||
|
(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.25)
|
|||
|
(def sa-double-length 37.5)
|
|||
|
(def sa-cap {1 (let [bl2 (/ 18.5 2)
|
|||
|
m (/ 17 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 12])))]
|
|||
|
(->> key-cap
|
|||
|
(translate [0 0 (+ 5 plate-thickness)])
|
|||
|
(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 12])))]
|
|||
|
(->> key-cap
|
|||
|
(translate [0 0 (+ 5 plate-thickness)])
|
|||
|
(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 12])))]
|
|||
|
(->> key-cap
|
|||
|
(translate [0 0 (+ 5 plate-thickness)])
|
|||
|
(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 sa-profile-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-offset (cond
|
|||
|
(= column 2) [0 2.82 -4.5]
|
|||
|
(>= column 4) [0 -5.8 5.64]
|
|||
|
: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-offset))]
|
|||
|
(->> placed-shape
|
|||
|
(rotate (/ π 12) [0 1 0])
|
|||
|
(translate [0 0 13]))))
|
|||
|
|
|||
|
(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 [0 -4.35 5.64]
|
|||
|
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-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 (= column 5) 1 1))
|
|||
|
(key-place column row)))))
|
|||
|
|
|||
|
;;;;;;;;;;;;;;;;;;;;
|
|||
|
;; Web Connectors ;;
|
|||
|
;;;;;;;;;;;;;;;;;;;;
|
|||
|
|
|||
|
(def web-thickness 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 (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 (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 sa-profile-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))]))
|
|||
|
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) 0.55))
|
|||
|
(def left-wall-column (- (first columns) 1/2))
|
|||
|
(def thumb-back-y 0.93)
|
|||
|
(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) ;;Sphere resolution, lower for faster renders
|
|||
|
|
|||
|
(defn wall-sphere-at [coords]
|
|||
|
(->> (sphere 1)
|
|||
|
(translate coords)
|
|||
|
(with-fn wall-sphere-n)))
|
|||
|
|
|||
|
(defn scale-to-range [start end x]
|
|||
|
(+ start (* (- end start) x)))
|
|||
|
|
|||
|
(defn wall-sphere-bottom [front-to-back-scale]
|
|||
|
(wall-sphere-at [0
|
|||
|
(scale-to-range
|
|||
|
(+ (/ mount-height -2) -3.5)
|
|||
|
(+ (/ mount-height 2) 5.0)
|
|||
|
front-to-back-scale)
|
|||
|
-6]))
|
|||
|
|
|||
|
(defn wall-sphere-top [front-to-back-scale]
|
|||
|
(wall-sphere-at [0
|
|||
|
(scale-to-range
|
|||
|
(+ (/ mount-height -2) -3.5)
|
|||
|
(+ (/ mount-height 2) 3.5)
|
|||
|
front-to-back-scale)
|
|||
|
10]))
|
|||
|
|
|||
|
(def wall-sphere-top-back (wall-sphere-top 1))
|
|||
|
(def wall-sphere-bottom-back (wall-sphere-bottom 1))
|
|||
|
(def wall-sphere-bottom-front (wall-sphere-bottom 0))
|
|||
|
(def wall-sphere-top-front (wall-sphere-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 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-sphere-top-front
|
|||
|
x-start x-end y-start y-end
|
|||
|
wall-step))]
|
|||
|
(union
|
|||
|
(apply union
|
|||
|
(for [x (range-inclusive 0.7 (- right-wall-column step) step)]
|
|||
|
(hull (place x 4 wall-sphere-top-front)
|
|||
|
(place (+ x step) 4 wall-sphere-top-front)
|
|||
|
(place x 4 wall-sphere-bottom-front)
|
|||
|
(place (+ x step) 4 wall-sphere-bottom-front))))
|
|||
|
(apply union
|
|||
|
(for [x (range-inclusive 0.5 0.7 0.01)]
|
|||
|
(hull (place x 4 wall-sphere-top-front)
|
|||
|
(place (+ x step) 4 wall-sphere-top-front)
|
|||
|
(place 0.7 4 wall-sphere-bottom-front))))
|
|||
|
(top-cover 0.5 1.7 3.6 4)
|
|||
|
(top-cover 1.59 2.41 3.35 4) ;; was 3.32
|
|||
|
(top-cover 2.39 3.41 3.6 4)
|
|||
|
(apply union
|
|||
|
(for [x (range 2 5)]
|
|||
|
(union
|
|||
|
(hull (place (- x 1/2) 4 (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(place (+ x 1/2) 4 (translate [0 1 1] wall-sphere-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-sphere-bottom-front))
|
|||
|
(key-place x 4 web-post-bl)
|
|||
|
(key-place (- x 1) 4 web-post-br)))))
|
|||
|
(hull (place right-wall-column 4 (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(place (- right-wall-column 1) 4 (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(key-place 5 4 web-post-bl)
|
|||
|
(key-place 5 4 web-post-br))
|
|||
|
(hull (place (+ 4 1/2) 4 (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(place (- right-wall-column 1) 4 (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(key-place 4 4 web-post-br)
|
|||
|
(key-place 5 4 web-post-bl))
|
|||
|
(hull (place 0.7 4 (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(place 1.7 4 (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(key-place 1 4 web-post-bl)
|
|||
|
(key-place 1 4 web-post-br)))))
|
|||
|
|
|||
|
(def back-wall
|
|||
|
(let [step wall-step
|
|||
|
wall-sphere-top-backtep 0.05
|
|||
|
place case-place
|
|||
|
front-top-cover (fn [x-start x-end y-start y-end]
|
|||
|
(apply union
|
|||
|
(for [x (range-inclusive x-start (- x-end wall-sphere-top-backtep) wall-sphere-top-backtep)
|
|||
|
y (range-inclusive y-start (- y-end wall-sphere-top-backtep) wall-sphere-top-backtep)]
|
|||
|
(hull (place x y wall-sphere-top-back)
|
|||
|
(place (+ x wall-sphere-top-backtep) y wall-sphere-top-back)
|
|||
|
(place x (+ y wall-sphere-top-backtep) wall-sphere-top-back)
|
|||
|
(place (+ x wall-sphere-top-backtep) (+ y wall-sphere-top-backtep) wall-sphere-top-back)))))]
|
|||
|
(union
|
|||
|
(apply union
|
|||
|
(for [x (range-inclusive left-wall-column (- right-wall-column step) step)]
|
|||
|
(hull (place x back-y wall-sphere-top-back)
|
|||
|
(place (+ x step) back-y wall-sphere-top-back)
|
|||
|
(place x back-y wall-sphere-bottom-back)
|
|||
|
(place (+ x step) back-y wall-sphere-bottom-back))))
|
|||
|
(front-top-cover 1.56 2.44 back-y 0.1)
|
|||
|
(front-top-cover 3.56 4.44 back-y 0.13)
|
|||
|
(front-top-cover 4.3 right-wall-column back-y 0.13)
|
|||
|
|
|||
|
|
|||
|
(hull (place left-wall-column 0 (translate [1 -1 1] wall-sphere-bottom-back))
|
|||
|
(place (+ left-wall-column 1) 0 (translate [0 -1 1] wall-sphere-bottom-back))
|
|||
|
(key-place 0 0 web-post-tl)
|
|||
|
(key-place 0 0 web-post-tr))
|
|||
|
|
|||
|
(hull (place 5 0 (translate [0 -1 1] wall-sphere-bottom-back))
|
|||
|
(place right-wall-column 0 (translate [0 -1 1] wall-sphere-bottom-back))
|
|||
|
(key-place 5 0 web-post-tl)
|
|||
|
(key-place 5 0 web-post-tr))
|
|||
|
|
|||
|
(apply union
|
|||
|
(for [x (range 1 5)]
|
|||
|
(union
|
|||
|
(hull (place (- x 1/2) 0 (translate [0 -1 1] wall-sphere-bottom-back))
|
|||
|
(place (+ x 1/2) 0 (translate [0 -1 1] wall-sphere-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-sphere-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-sphere-bottom-back))
|
|||
|
(place 5 0 (translate [0 -1 1] wall-sphere-bottom-back))
|
|||
|
(key-place 4 0 web-post-tr)
|
|||
|
(key-place 5 0 web-post-tl)))))
|
|||
|
|
|||
|
(def right-wall
|
|||
|
(let [place case-place]
|
|||
|
(union
|
|||
|
(apply union
|
|||
|
(map (partial apply hull)
|
|||
|
(partition 2 1
|
|||
|
(for [scale (range-inclusive 0 1 0.01)]
|
|||
|
(let [x (scale-to-range 4 0.02 scale)]
|
|||
|
(hull (place right-wall-column x (wall-sphere-top scale))
|
|||
|
(place right-wall-column x (wall-sphere-bottom scale))))))))
|
|||
|
|
|||
|
(apply union
|
|||
|
(concat
|
|||
|
(for [x (range 0 5)]
|
|||
|
(union
|
|||
|
(hull (place right-wall-column x (translate [-1 0 1] (wall-sphere-bottom 1/2)))
|
|||
|
(key-place 5 x web-post-br)
|
|||
|
(key-place 5 x web-post-tr))))
|
|||
|
(for [x (range 0 4)]
|
|||
|
(union
|
|||
|
(hull (place right-wall-column x (translate [-1 0 1] (wall-sphere-bottom 1/2)))
|
|||
|
(place right-wall-column (inc x) (translate [-1 0 1] (wall-sphere-bottom 1/2)))
|
|||
|
(key-place 5 x web-post-br)
|
|||
|
(key-place 5 (inc x) web-post-tr))))
|
|||
|
[(union
|
|||
|
(hull (place right-wall-column 0 (translate [-1 0 1] (wall-sphere-bottom 1/2)))
|
|||
|
(place right-wall-column 0.02 (translate [-1 -1 1] (wall-sphere-bottom 1)))
|
|||
|
(key-place 5 0 web-post-tr))
|
|||
|
(hull (place right-wall-column 4 (translate [-1 0 1] (wall-sphere-bottom 1/2)))
|
|||
|
(place right-wall-column 4 (translate [-1 1 1] (wall-sphere-bottom 0)))
|
|||
|
(key-place 5 4 web-post-br)))])))))
|
|||
|
|
|||
|
(def left-wall
|
|||
|
(let [place case-place]
|
|||
|
(union
|
|||
|
(apply union
|
|||
|
(for [x (range-inclusive -1 (- 1.6666 wall-step) wall-step)]
|
|||
|
(hull (place left-wall-column x wall-sphere-top-front)
|
|||
|
(place left-wall-column (+ x wall-step) wall-sphere-top-front)
|
|||
|
(place left-wall-column x wall-sphere-bottom-front)
|
|||
|
(place left-wall-column (+ x wall-step) wall-sphere-bottom-front))))
|
|||
|
(hull (place left-wall-column -1 wall-sphere-top-front)
|
|||
|
(place left-wall-column -1 wall-sphere-bottom-front)
|
|||
|
(place left-wall-column 0.02 wall-sphere-top-back)
|
|||
|
(place left-wall-column 0.02 wall-sphere-bottom-back))
|
|||
|
(hull (place left-wall-column 0 (translate [1 -1 1] wall-sphere-bottom-back))
|
|||
|
(place left-wall-column 1 (translate [1 0 1] wall-sphere-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-sphere-bottom-back))
|
|||
|
(place left-wall-column 2 (translate [1 0 1] wall-sphere-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-sphere-bottom-back))
|
|||
|
(place left-wall-column 1.6666 (translate [1 0 1] wall-sphere-bottom-front))
|
|||
|
(key-place 0 1 web-post-bl)
|
|||
|
(key-place 0 2 web-post-bl))
|
|||
|
(hull (place left-wall-column 1.6666 (translate [1 0 1] wall-sphere-bottom-front))
|
|||
|
(key-place 0 2 web-post-bl)
|
|||
|
(key-place 0 3 web-post-tl))
|
|||
|
(hull (place left-wall-column 1.6666 (translate [1 0 1] wall-sphere-bottom-front))
|
|||
|
(thumb-place 1 1 web-post-tr)
|
|||
|
(key-place 0 3 web-post-tl))
|
|||
|
(hull (place left-wall-column 1.6666 (translate [1 0 1] wall-sphere-bottom-front))
|
|||
|
(thumb-place 1 1 web-post-tr)
|
|||
|
(thumb-place 1/2 thumb-back-y (translate [0 -1 1] wall-sphere-bottom-back))))))
|
|||
|
|
|||
|
(def thumb-back-wall
|
|||
|
(let [step wall-step
|
|||
|
top-step 0.05
|
|||
|
front-top-cover (fn [x-start x-end y-start y-end]
|
|||
|
(apply union
|
|||
|
(for [x (range-inclusive x-start (- x-end top-step) top-step)
|
|||
|
y (range-inclusive y-start (- y-end top-step) top-step)]
|
|||
|
(hull (thumb-place x y wall-sphere-top-back)
|
|||
|
(thumb-place (+ x top-step) y wall-sphere-top-back)
|
|||
|
(thumb-place x (+ y top-step) wall-sphere-top-back)
|
|||
|
(thumb-place (+ x top-step) (+ y top-step) wall-sphere-top-back)))))
|
|||
|
back-y thumb-back-y]
|
|||
|
(union
|
|||
|
(apply union
|
|||
|
(for [x (range-inclusive 1/2 (- (+ 5/2 0.05) step) step)]
|
|||
|
(hull (thumb-place x back-y wall-sphere-top-back)
|
|||
|
(thumb-place (+ x step) back-y wall-sphere-top-back)
|
|||
|
(thumb-place x back-y wall-sphere-bottom-back)
|
|||
|
(thumb-place (+ x step) back-y wall-sphere-bottom-back))))
|
|||
|
(hull (thumb-place 1/2 back-y wall-sphere-top-back)
|
|||
|
(thumb-place 1/2 back-y wall-sphere-bottom-back)
|
|||
|
(case-place left-wall-column 1.6666 wall-sphere-top-front))
|
|||
|
(hull (thumb-place 1/2 back-y wall-sphere-bottom-back)
|
|||
|
(case-place left-wall-column 1.6666 wall-sphere-top-front)
|
|||
|
(case-place left-wall-column 1.6666 wall-sphere-bottom-front))
|
|||
|
(hull
|
|||
|
(thumb-place 1/2 thumb-back-y (translate [0 -1 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place 1 1 web-post-tr)
|
|||
|
(thumb-place 3/2 thumb-back-y (translate [0 -1 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place 1 1 web-post-tl))
|
|||
|
(hull
|
|||
|
(thumb-place (+ 5/2 0.05) thumb-back-y (translate [1 -1 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place 3/2 thumb-back-y (translate [0 -1 1] wall-sphere-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]
|
|||
|
(union
|
|||
|
(apply union
|
|||
|
(for [x (range-inclusive (+ -1 0.07) (- 1.95 step) step)]
|
|||
|
(hull (place thumb-left-wall-column x wall-sphere-top-front)
|
|||
|
(place thumb-left-wall-column (+ x step) wall-sphere-top-front)
|
|||
|
(place thumb-left-wall-column x wall-sphere-bottom-front)
|
|||
|
(place thumb-left-wall-column (+ x step) wall-sphere-bottom-front))))
|
|||
|
(hull (place thumb-left-wall-column 1.95 wall-sphere-top-front)
|
|||
|
(place thumb-left-wall-column 1.95 wall-sphere-bottom-front)
|
|||
|
(place thumb-left-wall-column thumb-back-y wall-sphere-top-back)
|
|||
|
(place thumb-left-wall-column thumb-back-y wall-sphere-bottom-back))
|
|||
|
|
|||
|
(hull
|
|||
|
(thumb-place thumb-left-wall-column thumb-back-y (translate [1 -1 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place thumb-left-wall-column 0 (translate [1 0 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place 2 1 web-post-tl)
|
|||
|
(thumb-place 2 1 web-post-bl))
|
|||
|
(hull
|
|||
|
(thumb-place thumb-left-wall-column 0 (translate [1 0 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place 2 0 web-post-tl)
|
|||
|
(thumb-place 2 1 web-post-bl))
|
|||
|
(hull
|
|||
|
(thumb-place thumb-left-wall-column 0 (translate [1 0 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place thumb-left-wall-column -1 (translate [1 0 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place 2 0 web-post-tl)
|
|||
|
(thumb-place 2 0 web-post-bl))
|
|||
|
(hull
|
|||
|
(thumb-place thumb-left-wall-column -1 (translate [1 0 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place 2 -1 web-post-tl)
|
|||
|
(thumb-place 2 0 web-post-bl))
|
|||
|
(hull
|
|||
|
(thumb-place thumb-left-wall-column -1 (translate [1 0 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place thumb-left-wall-column (+ -1 0.07) (translate [1 1 1] wall-sphere-bottom-front))
|
|||
|
(thumb-place 2 -1 web-post-tl)
|
|||
|
(thumb-place 2 -1 web-post-bl)))))
|
|||
|
|
|||
|
(def thumb-front-wall
|
|||
|
(let [step wall-step ;;0.1
|
|||
|
wall-sphere-top-fronttep 0.05 ;;0.05
|
|||
|
place thumb-place
|
|||
|
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
|
|||
|
(apply union
|
|||
|
(for [x (range-inclusive thumb-right-wall (- (+ 5/2 0.05) step) step)]
|
|||
|
(hull (place x thumb-front-row wall-sphere-top-front)
|
|||
|
(place (+ x step) thumb-front-row wall-sphere-top-front)
|
|||
|
(place x thumb-front-row wall-sphere-bottom-front)
|
|||
|
(place (+ x step) thumb-front-row wall-sphere-bottom-front))))
|
|||
|
|
|||
|
(hull (place thumb-right-wall thumb-front-row wall-sphere-top-front)
|
|||
|
(place thumb-right-wall thumb-front-row wall-sphere-bottom-front)
|
|||
|
(case-place 0.5 4 wall-sphere-top-front))
|
|||
|
(hull (place thumb-right-wall thumb-front-row wall-sphere-bottom-front)
|
|||
|
(case-place 0.5 4 wall-sphere-top-front)
|
|||
|
(case-place 0.7 4 wall-sphere-bottom-front))
|
|||
|
|
|||
|
(hull (place thumb-right-wall thumb-front-row wall-sphere-bottom-front)
|
|||
|
(key-place 1 4 web-post-bl)
|
|||
|
(place 0 -1/2 thumb-br)
|
|||
|
(place 0 -1/2 web-post-br)
|
|||
|
(case-place 0.7 4 wall-sphere-bottom-front))
|
|||
|
|
|||
|
(hull (place (+ 5/2 0.05) thumb-front-row (translate [1 1 1] wall-sphere-bottom-front))
|
|||
|
(place (+ 3/2 0.05) thumb-front-row (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(place 2 -1 web-post-bl)
|
|||
|
(place 2 -1 web-post-br))
|
|||
|
|
|||
|
(hull (place thumb-right-wall thumb-front-row (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(place (+ 1/2 0.05) thumb-front-row (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(place 0 -1/2 thumb-bl)
|
|||
|
(place 0 -1/2 thumb-br))
|
|||
|
(hull (place (+ 1/2 0.05) thumb-front-row (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(place (+ 3/2 0.05) thumb-front-row (translate [0 1 1] wall-sphere-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 new-case
|
|||
|
(union front-wall
|
|||
|
right-wall
|
|||
|
back-wall
|
|||
|
left-wall
|
|||
|
thumb-back-wall
|
|||
|
thumb-left-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)) -4.5)])))
|
|||
|
(def bottom-front-key-guard (->> (cube mount-width (/ mount-height 2) web-thickness)
|
|||
|
(translate [0 (/ mount-height 4) (+ (- (/ web-thickness 2)) -4.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)] %)
|
|||
|
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 (case-place (- x 1/2) 4 (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(case-place (+ x 1/2) 4 (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(key-place x 4 half-post-bl)
|
|||
|
(key-place x 4 half-post-br))
|
|||
|
(hull (case-place (- x 1/2) 4 (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(key-place x 4 half-post-bl)
|
|||
|
(key-place (- x 1) 4 half-post-br))))
|
|||
|
[(hull (case-place right-wall-column 4 (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(case-place (- right-wall-column 1) 4 (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(key-place 5 4 half-post-bl)
|
|||
|
(key-place 5 4 half-post-br))
|
|||
|
(hull (case-place (+ 4 1/2) 4 (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(case-place (- right-wall-column 1) 4 (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(key-place 4 4 half-post-br)
|
|||
|
(key-place 5 4 half-post-bl))])
|
|||
|
right-wall (concat
|
|||
|
(for [x (range 0 4)]
|
|||
|
(hull (case-place right-wall-column x (translate [-1 0 1] (wall-sphere-bottom 1/2)))
|
|||
|
(key-place 5 x web-post-br)
|
|||
|
(key-place 5 x web-post-tr)))
|
|||
|
(for [x (range 0 4)]
|
|||
|
(hull (case-place right-wall-column x (translate [-1 0 1] (wall-sphere-bottom 1/2)))
|
|||
|
(case-place right-wall-column (inc x) (translate [-1 0 1] (wall-sphere-bottom 1/2)))
|
|||
|
(key-place 5 x web-post-br)
|
|||
|
(key-place 5 (inc x) web-post-tr)))
|
|||
|
[(union
|
|||
|
(hull (case-place right-wall-column 0 (translate [-1 0 1] (wall-sphere-bottom 1/2)))
|
|||
|
(case-place right-wall-column 0.02 (translate [-1 -1 1] (wall-sphere-bottom 1)))
|
|||
|
(key-place 5 0 web-post-tr)
|
|||
|
)
|
|||
|
(hull (case-place right-wall-column 4 (translate [-1 0 1] (wall-sphere-bottom 1/2)))
|
|||
|
(case-place right-wall-column 4 (translate [0 1 1] (wall-sphere-bottom 0)))
|
|||
|
(key-place 5 4 half-post-br)
|
|||
|
)
|
|||
|
(hull (case-place right-wall-column 4 (translate [-1 0 1] (wall-sphere-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
|
|||
|
(hull (case-place (- x 1/2) 0 (translate [0 -1 1] wall-sphere-bottom-back))
|
|||
|
(case-place (+ x 1/2) 0 (translate [0 -1 1] wall-sphere-bottom-back))
|
|||
|
(key-place x 0 web-post-tl)
|
|||
|
(key-place x 0 web-post-tr))
|
|||
|
(hull (case-place (- x 1/2) 0 (translate [0 -1 1] wall-sphere-bottom-back))
|
|||
|
(key-place x 0 web-post-tl)
|
|||
|
(key-place (- x 1) 0 web-post-tr))))
|
|||
|
[(hull (case-place left-wall-column 0 (translate [1 -1 1] wall-sphere-bottom-back))
|
|||
|
(case-place (+ left-wall-column 1) 0 (translate [0 -1 1] wall-sphere-bottom-back))
|
|||
|
(key-place 0 0 web-post-tl)
|
|||
|
(key-place 0 0 web-post-tr))])
|
|||
|
left-wall (let [place case-place]
|
|||
|
[(hull (place left-wall-column 0 (translate [1 -1 1] wall-sphere-bottom-back))
|
|||
|
(place left-wall-column 1 (translate [1 0 1] wall-sphere-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-sphere-bottom-back))
|
|||
|
(place left-wall-column 2 (translate [1 0 1] wall-sphere-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-sphere-bottom-back))
|
|||
|
(place left-wall-column 1.6666 (translate [1 0 1] wall-sphere-bottom-front))
|
|||
|
(key-place 0 1 web-post-bl)
|
|||
|
(key-place 0 2 web-post-bl))
|
|||
|
(hull (place left-wall-column 1.6666 (translate [1 0 1] wall-sphere-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 [(hull
|
|||
|
(thumb-place 1/2 thumb-back-y (translate [0 -1 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place 1 1 web-post-tr)
|
|||
|
(thumb-place 3/2 thumb-back-y (translate [0 -1 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place 1 1 web-post-tl))
|
|||
|
|
|||
|
(hull
|
|||
|
(thumb-place (+ 5/2 0.05) thumb-back-y (translate [1 -1 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place 3/2 thumb-back-y (translate [0 -1 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place 1 1 web-post-tl)
|
|||
|
(thumb-place 2 1 web-post-tl))
|
|||
|
(hull
|
|||
|
(thumb-place 1/2 thumb-back-y (translate [0 -1 1] wall-sphere-bottom-back))
|
|||
|
(case-place left-wall-column 1.6666 (translate [1 0 1] wall-sphere-bottom-front))
|
|||
|
(key-place 0 3 web-post-tl)
|
|||
|
(thumb-place 1 1 web-post-tr))
|
|||
|
]
|
|||
|
thumb-left-wall [(hull
|
|||
|
(thumb-place thumb-left-wall-column thumb-back-y (translate [1 -1 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place thumb-left-wall-column 0 (translate [1 0 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place 2 1 web-post-tl)
|
|||
|
(thumb-place 2 1 web-post-bl))
|
|||
|
(hull
|
|||
|
(thumb-place thumb-left-wall-column 0 (translate [1 0 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place 2 0 web-post-tl)
|
|||
|
(thumb-place 2 1 web-post-bl))
|
|||
|
(hull
|
|||
|
(thumb-place thumb-left-wall-column 0 (translate [1 0 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place thumb-left-wall-column -1 (translate [1 0 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place 2 0 web-post-tl)
|
|||
|
(thumb-place 2 0 web-post-bl))
|
|||
|
(hull
|
|||
|
(thumb-place thumb-left-wall-column -1 (translate [1 0 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place 2 -1 web-post-tl)
|
|||
|
(thumb-place 2 0 web-post-bl))
|
|||
|
(hull
|
|||
|
(thumb-place thumb-left-wall-column -1 (translate [1 0 1] wall-sphere-bottom-back))
|
|||
|
(thumb-place thumb-left-wall-column (+ -1 0.07) (translate [1 1 1] wall-sphere-bottom-front))
|
|||
|
(thumb-place 2 -1 web-post-tl)
|
|||
|
(thumb-place 2 -1 web-post-bl))]
|
|||
|
thumb-front-wall [(hull (thumb-place (+ 5/2 0.05) thumb-front-row (translate [1 1 1] wall-sphere-bottom-front))
|
|||
|
(thumb-place (+ 3/2 0.05) thumb-front-row (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(thumb-place 2 -1 web-post-bl)
|
|||
|
(thumb-place 2 -1 web-post-br))
|
|||
|
(hull (thumb-place (+ 1/2 0.05) thumb-front-row (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(thumb-place (+ 3/2 0.05) thumb-front-row (translate [0 1 1] wall-sphere-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))
|
|||
|
(hull (thumb-place thumb-right-wall thumb-front-row (translate [-1 1 1] wall-sphere-bottom-front))
|
|||
|
(thumb-place (+ 1/2 0.05) thumb-front-row (translate [0 1 1] wall-sphere-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))
|
|||
|
|
|||
|
(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)
|
|||
|
(case-place (- 2 1/2) 4 (translate [0 1 1] wall-sphere-bottom-front))
|
|||
|
(case-place 0.7 4 (translate [0 1 1] wall-sphere-bottom-front)))
|
|||
|
|
|||
|
(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 1 1] wall-sphere-bottom-front))
|
|||
|
(key-place 1 4 (translate [0 0 8.5] web-post-bl))
|
|||
|
(key-place 1 4 half-post-bl)
|
|||
|
)]
|
|||
|
stands (let [bumper-diameter 9.6
|
|||
|
bumper-radius (/ bumper-diameter 2)
|
|||
|
stand-diameter (+ bumper-diameter 2)
|
|||
|
stand-radius (/ stand-diameter 2)
|
|||
|
stand-at #(difference (->> (sphere stand-radius)
|
|||
|
(translate [0 0 (+ (/ stand-radius -2) -4.5)])
|
|||
|
%
|
|||
|
(bottom-hull))
|
|||
|
(->> (cube stand-diameter stand-diameter stand-radius)
|
|||
|
(translate [0 0 (/ stand-radius -2)])
|
|||
|
%)
|
|||
|
(->> (sphere bumper-radius)
|
|||
|
(translate [0 0 (+ (/ stand-radius -2) -4.5)])
|
|||
|
%
|
|||
|
(bottom 1.5)))]
|
|||
|
[(stand-at #(key-place 0 1 %))
|
|||
|
(stand-at #(thumb-place 1 -1/2 %))
|
|||
|
(stand-at #(key-place 5 0 %))
|
|||
|
(stand-at #(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 1/2) 1/2 screw-hole)
|
|||
|
(key-place (+ 4 1/2) (+ 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 -3.5])))
|
|||
|
|
|||
|
|
|||
|
(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 3 3 9)
|
|||
|
(translate [0 0 -2])
|
|||
|
(key-place 1/2 3/2)
|
|||
|
(color [0 1 0]))
|
|||
|
(hull (->> (cube 3 6 9)
|
|||
|
(translate [0 0 -2])
|
|||
|
(key-place 1/2 2)
|
|||
|
(color [0 0 1]))
|
|||
|
(->> (cube 3 3 (+ teensy-pcb-thickness 3))
|
|||
|
(translate [0 (/ 30.5 -2) (+ (- teensy-offset-height)
|
|||
|
#_(/ (+ teensy-pcb-thickness 3) -2)
|
|||
|
)])
|
|||
|
(key-place 1/2 3/2)
|
|||
|
(color [0 0 1]))))
|
|||
|
teensy-pcb
|
|||
|
(->> (cube 18 30.5 teensy-pcb-thickness)
|
|||
|
(translate [0 1.5 (+ (/ teensy-pcb-thickness -2) (- teensy-offset-height) -1)])
|
|||
|
(key-place 1/2 3/2)
|
|||
|
(color [1 0 0]))))
|
|||
|
|
|||
|
(def usb-cutout
|
|||
|
(let [hole-height 6.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 [0 (/ teensy-length 2) (- side-radius)])
|
|||
|
(translate [0 0 (- 1)])
|
|||
|
(translate [0 0 (- teensy-offset-height)])
|
|||
|
(key-place 1/2 3/2))))
|
|||
|
|
|||
|
;;;;;;;;;;;;;;;;;;
|
|||
|
;; Final Export ;;
|
|||
|
;;;;;;;;;;;;;;;;;;
|
|||
|
|
|||
|
(def dactyl-bottom-right
|
|||
|
(difference
|
|||
|
(union
|
|||
|
teensy-cover
|
|||
|
(difference
|
|||
|
bottom-plate
|
|||
|
(hull teensy-cover)
|
|||
|
new-case
|
|||
|
teensy-cover
|
|||
|
trrs-cutout
|
|||
|
(->> (cube 1000 1000 10) (translate [0 0 -5]))
|
|||
|
screw-holes))
|
|||
|
usb-cutout))
|
|||
|
|
|||
|
(def dactyl-bottom-left
|
|||
|
(mirror [-1 0 0]
|
|||
|
(union
|
|||
|
io-exp-cover
|
|||
|
(difference
|
|||
|
bottom-plate
|
|||
|
(hull io-exp-cover)
|
|||
|
new-case
|
|||
|
io-exp-cover
|
|||
|
trrs-cutout
|
|||
|
(->> (cube 1000 1000 10) (translate [0 0 -5]))
|
|||
|
screw-holes))))
|
|||
|
|
|||
|
(def dactyl-top-right
|
|||
|
(difference
|
|||
|
(union key-holes
|
|||
|
connectors
|
|||
|
thumb
|
|||
|
new-case
|
|||
|
teensy-support)
|
|||
|
trrs-hole-just-circle
|
|||
|
screw-holes))
|
|||
|
|
|||
|
(def dactyl-top-left
|
|||
|
(mirror [-1 0 0]
|
|||
|
(difference
|
|||
|
(union key-holes
|
|||
|
connectors
|
|||
|
thumb
|
|||
|
new-case)
|
|||
|
trrs-hole-just-circle
|
|||
|
screw-holes)))
|
|||
|
|
|||
|
(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-top-left.scad"
|
|||
|
(write-scad dactyl-top-left))
|
|||
|
|
|||
|
(spit "things/dactyl-bottom-left.scad"
|
|||
|
(write-scad dactyl-bottom-left))
|