|
|
|
|
(ns dactyl-keyboard.dactyl
|
|
|
|
|
(:refer-clojure :exclude [use import])
|
|
|
|
|
(:require [clojure.core.matrix :refer [array matrix mmul]]
|
|
|
|
|
[scad-clj.scad :refer :all]
|
|
|
|
|
[scad-clj.model :refer :all]
|
|
|
|
|
[unicode-math.core :refer :all]))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
;; Shape parameters ;;
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
|
|
|
|
|
(def nrows 4)
|
|
|
|
|
(def ncols 5)
|
|
|
|
|
|
|
|
|
|
(def α (/ π 12)) ; curvature of the columns
|
|
|
|
|
(def β (/ π (if (= nrows 4) 26 36))) ; curvature of the rows
|
|
|
|
|
(def centerrow (- nrows 3)) ; controls front-back tilt
|
|
|
|
|
(def centercol 3) ; controls left-right tilt / tenting (higher number is more tenting)
|
|
|
|
|
(def orthographic-x (> nrows 5)) ; for larger number of rows don't curve them in as much
|
|
|
|
|
; (def orthographic-x true) ; controls curvature of rowS
|
|
|
|
|
|
|
|
|
|
(defn column-offset [column] (cond
|
|
|
|
|
(= column 2) [0 2.82 -4.5]
|
|
|
|
|
(>= column 4) [0 -5.8 5.64]
|
|
|
|
|
:else [0 0 0]))
|
|
|
|
|
|
|
|
|
|
(def thumb-offsets [6 -3 7])
|
|
|
|
|
|
|
|
|
|
(def keyboard-z-offset 24) ; controls height
|
|
|
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
;; General variables ;;
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
|
|
|
|
|
(def lastrow (dec nrows))
|
|
|
|
|
(def cornerrow (dec lastrow))
|
|
|
|
|
(def lastcol (dec ncols))
|
|
|
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;
|
|
|
|
|
;; 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 ncols))
|
|
|
|
|
(def rows (range 0 nrows))
|
|
|
|
|
|
|
|
|
|
(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))
|
|
|
|
|
(def column-x-delta (+ -1 (- (* column-radius (Math/sin β)))))
|
|
|
|
|
|
|
|
|
|
(defn key-place [column row shape]
|
|
|
|
|
(let [row-placed-shape (->> shape
|
|
|
|
|
(translate [0 0 (- row-radius)])
|
|
|
|
|
(rotate (* α (- centerrow row)) [1 0 0])
|
|
|
|
|
(translate [0 0 row-radius]))
|
|
|
|
|
column-angle (* β (- centercol 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 column)))
|
|
|
|
|
column-z-delta (* column-radius (- 1 (Math/cos column-angle)))
|
|
|
|
|
placed-shape-ortho (->> row-placed-shape
|
|
|
|
|
(rotate column-angle [0 1 0])
|
|
|
|
|
(translate [(- (* (- column centercol) column-x-delta)) 0 column-z-delta])
|
|
|
|
|
(translate (column-offset column)))]
|
|
|
|
|
(->> (if orthographic-x placed-shape-ortho placed-shape)
|
|
|
|
|
(rotate (/ π 12) [0 1 0])
|
|
|
|
|
(translate [0 0 keyboard-z-offset]))))
|
|
|
|
|
|
|
|
|
|
(def key-holes
|
|
|
|
|
(apply union
|
|
|
|
|
(for [column columns
|
|
|
|
|
row rows
|
|
|
|
|
:when (or (.contains [2 3] column)
|
|
|
|
|
(not= row lastrow))]
|
|
|
|
|
(->> single-plate
|
|
|
|
|
(key-place column row)))))
|
|
|
|
|
|
|
|
|
|
(def caps
|
|
|
|
|
(apply union
|
|
|
|
|
(for [column columns
|
|
|
|
|
row rows
|
|
|
|
|
:when (or (.contains [2 3] column)
|
|
|
|
|
(not= row lastrow))]
|
|
|
|
|
(->> (sa-cap (if (= column 5) 1 1))
|
|
|
|
|
(key-place column row)))))
|
|
|
|
|
|
|
|
|
|
(defn rotate-around-x [angle position]
|
|
|
|
|
(mmul
|
|
|
|
|
[[1 0 0]
|
|
|
|
|
[0 (Math/cos angle) (- (Math/sin angle))]
|
|
|
|
|
[0 (Math/sin angle) (Math/cos angle)]]
|
|
|
|
|
position))
|
|
|
|
|
|
|
|
|
|
(defn rotate-around-y [angle position]
|
|
|
|
|
(mmul
|
|
|
|
|
[[(Math/cos angle) 0 (Math/sin angle)]
|
|
|
|
|
[0 1 0]
|
|
|
|
|
[(- (Math/sin angle)) 0 (Math/cos angle)]]
|
|
|
|
|
position))
|
|
|
|
|
|
|
|
|
|
(defn key-position [column row position]
|
|
|
|
|
(let [row-position (->> position
|
|
|
|
|
(map + [0 0 (- row-radius)])
|
|
|
|
|
(rotate-around-x (* α (- centerrow row)))
|
|
|
|
|
(map + [0 0 row-radius]))
|
|
|
|
|
column-angle (* β (- centercol column))
|
|
|
|
|
placed-position (->> row-position
|
|
|
|
|
(map + [0 0 (- column-radius)])
|
|
|
|
|
(rotate-around-y column-angle)
|
|
|
|
|
(map + [0 0 column-radius])
|
|
|
|
|
(map + (column-offset column)))
|
|
|
|
|
column-z-delta (* column-radius (- 1 (Math/cos column-angle)))
|
|
|
|
|
placed-position-ortho (->> row-position
|
|
|
|
|
(rotate-around-y column-angle)
|
|
|
|
|
(map + [(- (* (- column centercol) column-x-delta)) 0 column-z-delta])
|
|
|
|
|
(map + (column-offset column)))]
|
|
|
|
|
(->> (if orthographic-x placed-position-ortho placed-position)
|
|
|
|
|
(rotate-around-y (/ π 12))
|
|
|
|
|
(map + [0 0 24]))))
|
|
|
|
|
|
|
|
|
|
; (pr (rotate-around-y π [10 0 1]))
|
|
|
|
|
; (pr (key-position 1 cornerrow [(/ mount-width 2) (- (/ mount-height 2)) 0]))
|
|
|
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
;; 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 (range 0 (dec ncols))
|
|
|
|
|
row (range 0 lastrow)]
|
|
|
|
|
(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 (range 0 cornerrow)]
|
|
|
|
|
(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 (range 0 (dec ncols))
|
|
|
|
|
row (range 0 cornerrow)]
|
|
|
|
|
(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 ;;
|
|
|
|
|
;;;;;;;;;;;;
|
|
|
|
|
|
|
|
|
|
(def thumborigin
|
|
|
|
|
(map + (key-position 1 cornerrow [(/ mount-width 2) (- (/ mount-height 2)) 0])
|
|
|
|
|
thumb-offsets))
|
|
|
|
|
; (pr thumborigin)
|
|
|
|
|
|
|
|
|
|
(defn deg2rad [degrees]
|
|
|
|
|
(* (/ degrees 180) pi))
|
|
|
|
|
|
|
|
|
|
(defn thumb-tr-place [shape]
|
|
|
|
|
(->> shape
|
|
|
|
|
(rotate (deg2rad 10) [1 0 0])
|
|
|
|
|
(rotate (deg2rad -23) [0 1 0])
|
|
|
|
|
(rotate (deg2rad -3) [0 0 1])
|
|
|
|
|
(translate thumborigin)
|
|
|
|
|
(translate [-10 -16 0])
|
|
|
|
|
))
|
|
|
|
|
(defn thumb-tl-place [shape]
|
|
|
|
|
(->> shape
|
|
|
|
|
(rotate (deg2rad 10) [1 0 0])
|
|
|
|
|
(rotate (deg2rad -23) [0 1 0])
|
|
|
|
|
(rotate (deg2rad -3) [0 0 1])
|
|
|
|
|
(translate thumborigin)
|
|
|
|
|
(translate [-30 -15 -2])))
|
|
|
|
|
(defn thumb-mr-place [shape]
|
|
|
|
|
(->> shape
|
|
|
|
|
(rotate (deg2rad -6) [1 0 0])
|
|
|
|
|
(rotate (deg2rad -34) [0 1 0])
|
|
|
|
|
(rotate (deg2rad 48) [0 0 1])
|
|
|
|
|
(translate thumborigin)
|
|
|
|
|
(translate [-29 -40 -13])
|
|
|
|
|
))
|
|
|
|
|
(defn thumb-ml-place [shape]
|
|
|
|
|
(->> shape
|
|
|
|
|
(rotate (deg2rad 6) [1 0 0])
|
|
|
|
|
(rotate (deg2rad -34) [0 1 0])
|
|
|
|
|
(rotate (deg2rad 40) [0 0 1])
|
|
|
|
|
(translate thumborigin)
|
|
|
|
|
(translate [-51 -25 -12])))
|
|
|
|
|
(defn thumb-br-place [shape]
|
|
|
|
|
(->> shape
|
|
|
|
|
(rotate (deg2rad -16) [1 0 0])
|
|
|
|
|
(rotate (deg2rad -33) [0 1 0])
|
|
|
|
|
(rotate (deg2rad 54) [0 0 1])
|
|
|
|
|
(translate thumborigin)
|
|
|
|
|
(translate [-37.8 -55.3 -25.3])
|
|
|
|
|
))
|
|
|
|
|
(defn thumb-bl-place [shape]
|
|
|
|
|
(->> shape
|
|
|
|
|
(rotate (deg2rad -4) [1 0 0])
|
|
|
|
|
(rotate (deg2rad -35) [0 1 0])
|
|
|
|
|
(rotate (deg2rad 52) [0 0 1])
|
|
|
|
|
(translate thumborigin)
|
|
|
|
|
(translate [-56.3 -43.3 -23.5])
|
|
|
|
|
))
|
|
|
|
|
|
|
|
|
|
(defn thumb-1x-layout [shape]
|
|
|
|
|
(union
|
|
|
|
|
(thumb-mr-place shape)
|
|
|
|
|
(thumb-ml-place shape)
|
|
|
|
|
(thumb-br-place shape)
|
|
|
|
|
(thumb-bl-place shape)))
|
|
|
|
|
|
|
|
|
|
(defn thumb-15x-layout [shape]
|
|
|
|
|
(union
|
|
|
|
|
(thumb-tr-place shape)
|
|
|
|
|
(thumb-tl-place shape)))
|
|
|
|
|
|
|
|
|
|
(def larger-plate
|
|
|
|
|
(let [plate-height (/ (- sa-double-length mount-height) 3)
|
|
|
|
|
top-plate (->> (cube mount-width plate-height web-thickness)
|
|
|
|
|
(translate [0 (/ (+ plate-height mount-height) 2)
|
|
|
|
|
(- plate-thickness (/ web-thickness 2))]))
|
|
|
|
|
]
|
|
|
|
|
(union top-plate (mirror [0 1 0] top-plate))))
|
|
|
|
|
|
|
|
|
|
(def thumbcaps
|
|
|
|
|
(union
|
|
|
|
|
(thumb-1x-layout (sa-cap 1))
|
|
|
|
|
(thumb-15x-layout (rotate (/ π 2) [0 0 1] (sa-cap 1.5)))))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
(def thumb
|
|
|
|
|
(union
|
|
|
|
|
(thumb-1x-layout single-plate)
|
|
|
|
|
(thumb-15x-layout single-plate)
|
|
|
|
|
(thumb-15x-layout larger-plate)
|
|
|
|
|
))
|
|
|
|
|
|
|
|
|
|
(def thumb-post-tr (translate [(- (/ mount-width 2) post-adj) (- (/ mount-height 1.15) post-adj) 0] web-post))
|
|
|
|
|
(def thumb-post-tl (translate [(+ (/ mount-width -2) post-adj) (- (/ mount-height 1.15) post-adj) 0] web-post))
|
|
|
|
|
(def thumb-post-bl (translate [(+ (/ mount-width -2) post-adj) (+ (/ mount-height -1.15) post-adj) 0] web-post))
|
|
|
|
|
(def thumb-post-br (translate [(- (/ mount-width 2) post-adj) (+ (/ mount-height -1.15) post-adj) 0] web-post))
|
|
|
|
|
|
|
|
|
|
(def thumb-connectors
|
|
|
|
|
(union
|
|
|
|
|
(triangle-hulls ; top two
|
|
|
|
|
(thumb-tl-place thumb-post-tr)
|
|
|
|
|
(thumb-tl-place thumb-post-br)
|
|
|
|
|
(thumb-tr-place thumb-post-tl)
|
|
|
|
|
(thumb-tr-place thumb-post-bl))
|
|
|
|
|
(triangle-hulls ; bottom two on the right
|
|
|
|
|
(thumb-br-place web-post-tr)
|
|
|
|
|
(thumb-br-place web-post-br)
|
|
|
|
|
(thumb-mr-place web-post-tl)
|
|
|
|
|
(thumb-mr-place web-post-bl))
|
|
|
|
|
(triangle-hulls ; bottom two on the left
|
|
|
|
|
(thumb-bl-place web-post-tr)
|
|
|
|
|
(thumb-bl-place web-post-br)
|
|
|
|
|
(thumb-ml-place web-post-tl)
|
|
|
|
|
(thumb-ml-place web-post-bl))
|
|
|
|
|
(triangle-hulls ; centers of the bottom four
|
|
|
|
|
(thumb-br-place web-post-tl)
|
|
|
|
|
(thumb-bl-place web-post-bl)
|
|
|
|
|
(thumb-br-place web-post-tr)
|
|
|
|
|
(thumb-bl-place web-post-br)
|
|
|
|
|
(thumb-mr-place web-post-tl)
|
|
|
|
|
(thumb-ml-place web-post-bl)
|
|
|
|
|
(thumb-mr-place web-post-tr)
|
|
|
|
|
(thumb-ml-place web-post-br))
|
|
|
|
|
(triangle-hulls ; top two to the middle two, starting on the left
|
|
|
|
|
(thumb-tl-place thumb-post-tl)
|
|
|
|
|
(thumb-ml-place web-post-tr)
|
|
|
|
|
(thumb-tl-place thumb-post-bl)
|
|
|
|
|
(thumb-ml-place web-post-br)
|
|
|
|
|
(thumb-tl-place thumb-post-br)
|
|
|
|
|
(thumb-mr-place web-post-tr)
|
|
|
|
|
(thumb-tr-place thumb-post-bl)
|
|
|
|
|
(thumb-mr-place web-post-br)
|
|
|
|
|
(thumb-tr-place thumb-post-br))
|
|
|
|
|
(triangle-hulls ; top two to the main keyboard, starting on the left
|
|
|
|
|
(thumb-tl-place thumb-post-tl)
|
|
|
|
|
(key-place 0 cornerrow web-post-bl)
|
|
|
|
|
(thumb-tl-place thumb-post-tr)
|
|
|
|
|
(key-place 0 cornerrow web-post-br)
|
|
|
|
|
(thumb-tr-place thumb-post-tl)
|
|
|
|
|
(key-place 1 cornerrow web-post-bl)
|
|
|
|
|
(thumb-tr-place thumb-post-tr)
|
|
|
|
|
(key-place 1 cornerrow web-post-br)
|
|
|
|
|
(key-place 2 lastrow web-post-tl)
|
|
|
|
|
(key-place 2 lastrow web-post-bl)
|
|
|
|
|
(thumb-tr-place thumb-post-tr)
|
|
|
|
|
(key-place 2 lastrow web-post-bl)
|
|
|
|
|
(thumb-tr-place thumb-post-br)
|
|
|
|
|
(key-place 2 lastrow web-post-br)
|
|
|
|
|
(key-place 3 lastrow web-post-bl)
|
|
|
|
|
(key-place 2 lastrow web-post-tr)
|
|
|
|
|
(key-place 3 lastrow web-post-tl)
|
|
|
|
|
(key-place 3 cornerrow web-post-bl)
|
|
|
|
|
(key-place 3 lastrow web-post-tr)
|
|
|
|
|
(key-place 3 cornerrow web-post-br)
|
|
|
|
|
(key-place 4 cornerrow web-post-bl))
|
|
|
|
|
(triangle-hulls
|
|
|
|
|
(key-place 1 cornerrow web-post-br)
|
|
|
|
|
(key-place 2 lastrow web-post-tl)
|
|
|
|
|
(key-place 2 cornerrow web-post-bl)
|
|
|
|
|
(key-place 2 lastrow web-post-tr)
|
|
|
|
|
(key-place 2 cornerrow web-post-br)
|
|
|
|
|
(key-place 3 cornerrow web-post-bl)
|
|
|
|
|
)
|
|
|
|
|
(triangle-hulls
|
|
|
|
|
(key-place 3 lastrow web-post-tr)
|
|
|
|
|
(key-place 3 lastrow web-post-br)
|
|
|
|
|
(key-place 3 lastrow web-post-tr)
|
|
|
|
|
(key-place 4 cornerrow web-post-bl))
|
|
|
|
|
))
|
|
|
|
|
|
|
|
|
|
;;;;;;;;;;
|
|
|
|
|
;; Case ;;
|
|
|
|
|
;;;;;;;;;;
|
|
|
|
|
|
|
|
|
|
(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 10.001 p)))
|
|
|
|
|
|
|
|
|
|
(def wall-offset -15)
|
|
|
|
|
|
|
|
|
|
(defn wall-brace [place1 dx1 dy1 post1 place2 dx2 dy2 post2]
|
|
|
|
|
(union
|
|
|
|
|
(hull
|
|
|
|
|
(place1 post1)
|
|
|
|
|
(place1 (translate [0 0 wall-offset] post1))
|
|
|
|
|
(place1 (translate [(* dx1 5) (* dy1 5) -4] post1))
|
|
|
|
|
(place1 (translate [(* dx1 5) (* dy1 5) wall-offset] post1))
|
|
|
|
|
(place2 post2)
|
|
|
|
|
(place2 (translate [0 0 wall-offset] post2))
|
|
|
|
|
(place2 (translate [(* dx2 5) (* dy2 5) -4] post2))
|
|
|
|
|
(place2 (translate [(* dx2 5) (* dy2 5) wall-offset] post2)))
|
|
|
|
|
(bottom-hull
|
|
|
|
|
(place1 (translate [(* dx1 5) (* dy1 5) wall-offset] post1))
|
|
|
|
|
(place1 (translate [0 0 wall-offset] post1))
|
|
|
|
|
(place1 (translate [(* dx1 5) (* dy1 5) wall-offset] post1))
|
|
|
|
|
(place2 (translate [0 0 wall-offset] post2))
|
|
|
|
|
(place2 (translate [(* dx2 5) (* dy2 5) wall-offset] post2)))))
|
|
|
|
|
|
|
|
|
|
(defn key-wall-brace [x1 y1 dx1 dy1 post1 x2 y2 dx2 dy2 post2]
|
|
|
|
|
(wall-brace (partial key-place x1 y1) dx1 dy1 post1
|
|
|
|
|
(partial key-place x2 y2) dx2 dy2 post2))
|
|
|
|
|
|
|
|
|
|
(def case-walls
|
|
|
|
|
(union
|
|
|
|
|
; back wall
|
|
|
|
|
(for [x (range 0 ncols)] (key-wall-brace x 0 0 1 web-post-tl x 0 0 1 web-post-tr))
|
|
|
|
|
(for [x (range 1 ncols)] (key-wall-brace x 0 0 1 web-post-tl (dec x) 0 0 1 web-post-tr))
|
|
|
|
|
(key-wall-brace 0 0 0 1 web-post-tl 0 0 -1 0 web-post-tl)
|
|
|
|
|
(key-wall-brace lastcol 0 0 1 web-post-tr lastcol 0 1 0 web-post-tr)
|
|
|
|
|
; right wall
|
|
|
|
|
(for [y (range 0 lastrow)] (key-wall-brace lastcol y 1 0 web-post-tr lastcol y 1 0 web-post-br))
|
|
|
|
|
(for [y (range 1 lastrow)] (key-wall-brace lastcol (dec y) 1 0 web-post-br lastcol y 1 0 web-post-tr))
|
|
|
|
|
(key-wall-brace lastcol cornerrow 0 -1 web-post-br lastcol cornerrow 1 0 web-post-br)
|
|
|
|
|
; left wall
|
|
|
|
|
(for [y (range 0 lastrow)] (key-wall-brace 0 y -1 0 web-post-tl 0 y -1 0 web-post-bl))
|
|
|
|
|
(for [y (range 1 lastrow)] (key-wall-brace 0 (dec y) -1 0 web-post-bl 0 y -1 0 web-post-tl))
|
|
|
|
|
; front wall
|
|
|
|
|
(key-wall-brace 0 0 0 1 web-post-tl 0 0 -1 0 web-post-tl)
|
|
|
|
|
(key-wall-brace lastcol 0 0 1 web-post-tr lastcol 0 1 0 web-post-tr)
|
|
|
|
|
(key-wall-brace 3 lastrow 0 -1 web-post-bl 3 lastrow 0.5 -1 web-post-br)
|
|
|
|
|
(key-wall-brace 3 lastrow 0.5 -1 web-post-br 4 cornerrow 1 -1 web-post-bl)
|
|
|
|
|
(for [x (range 4 ncols)] (key-wall-brace x cornerrow 0 -1 web-post-bl x cornerrow 0 -1 web-post-br))
|
|
|
|
|
(for [x (range 5 ncols)] (key-wall-brace x cornerrow 0 -1 web-post-bl (dec x) cornerrow 0 -1 web-post-br))
|
|
|
|
|
; thumb walls
|
|
|
|
|
(wall-brace thumb-mr-place 0 -1 web-post-br thumb-tr-place 0 -1 thumb-post-br)
|
|
|
|
|
(wall-brace thumb-mr-place 0 -1 web-post-br thumb-mr-place 0 -1 web-post-bl)
|
|
|
|
|
(wall-brace thumb-br-place 0 -1 web-post-br thumb-br-place 0 -1 web-post-bl)
|
|
|
|
|
(wall-brace thumb-ml-place 0 1 web-post-tr thumb-ml-place 0 1 web-post-tl)
|
|
|
|
|
(wall-brace thumb-bl-place 0 1 web-post-tr thumb-bl-place 0 1 web-post-tl)
|
|
|
|
|
(wall-brace thumb-br-place -1 0 web-post-tl thumb-br-place -1 0 web-post-bl)
|
|
|
|
|
(wall-brace thumb-bl-place -1 0 web-post-tl thumb-bl-place -1 0 web-post-bl)
|
|
|
|
|
; thumb corners
|
|
|
|
|
(wall-brace thumb-br-place -1 0 web-post-bl thumb-br-place 0 -1 web-post-bl)
|
|
|
|
|
(wall-brace thumb-bl-place -1 0 web-post-tl thumb-bl-place 0 1 web-post-tl)
|
|
|
|
|
; thumb tweeners
|
|
|
|
|
(wall-brace thumb-mr-place 0 -1 web-post-bl thumb-br-place 0 -1 web-post-br)
|
|
|
|
|
(wall-brace thumb-ml-place 0 1 web-post-tl thumb-bl-place 0 1 web-post-tr)
|
|
|
|
|
(wall-brace thumb-bl-place -1 0 web-post-bl thumb-br-place -1 0 web-post-tl)
|
|
|
|
|
(wall-brace thumb-tr-place 0 -1 thumb-post-br (partial key-place 3 lastrow) 0 -1 web-post-bl)
|
|
|
|
|
; clunky bit on the top left thumb connection
|
|
|
|
|
(wall-brace thumb-ml-place 0 1 web-post-tr thumb-tl-place -2.5 0 thumb-post-tl)
|
|
|
|
|
(wall-brace thumb-tl-place -1 0 thumb-post-tl (partial key-place 0 cornerrow) -1 0 web-post-bl)
|
|
|
|
|
;; another incomplete try to fix this
|
|
|
|
|
; (bottom-hull (thumb-tl-place thumb-post-tl)
|
|
|
|
|
; (thumb-tl-place (translate [-7 0 0] thumb-post-tl))
|
|
|
|
|
; (thumb-ml-place web-post-tr)
|
|
|
|
|
; (thumb-ml-place (translate [0 5 -4] web-post-tr)))
|
|
|
|
|
))
|
|
|
|
|
|
|
|
|
|
(defn on-wall-place [column depth shape]
|
|
|
|
|
(translate [0 0 (- depth)]
|
|
|
|
|
(key-place column 0
|
|
|
|
|
(->> shape
|
|
|
|
|
(rotate (+ (* β (- centercol column)) (/ π 12)) [0 -1 0])
|
|
|
|
|
(rotate (* α centerrow) [-1 0 0])
|
|
|
|
|
(translate [0 (/ mount-height 2) wall-offset])
|
|
|
|
|
))))
|
|
|
|
|
|
|
|
|
|
(def rj9-cube (cube 14.78 13 22.38))
|
|
|
|
|
(def rj9-space (on-wall-place 1 20 rj9-cube))
|
|
|
|
|
(def rj9-holder (on-wall-place 1 20
|
|
|
|
|
(difference rj9-cube
|
|
|
|
|
(union (translate [0 2 0] (cube 10.78 9 18.38))
|
|
|
|
|
(translate [0 0 5] (cube 10.78 13 5))))))
|
|
|
|
|
|
|
|
|
|
(def teensy-width 20) ; was 20
|
|
|
|
|
(def teensy-height 12)
|
|
|
|
|
(def teensy-length 33)
|
|
|
|
|
(def teensy2-length 53)
|
|
|
|
|
(def teensy-pcb-thickness 1.6)
|
|
|
|
|
(def teensy-offset-height 5)
|
|
|
|
|
|
|
|
|
|
(def teensy-holder
|
|
|
|
|
(on-wall-place 0 20
|
|
|
|
|
(translate [-5 0 0]
|
|
|
|
|
(union
|
|
|
|
|
(->> (cube 3 (* 1.2 teensy2-length) (+ 6 teensy-width))
|
|
|
|
|
(translate [-1.5 -30 0]))
|
|
|
|
|
(->> (cube teensy-pcb-thickness (* 1.2 teensy2-length) 3)
|
|
|
|
|
(translate [(/ teensy-pcb-thickness 2) -30 (- -1.5 (/ teensy-width 2))]))
|
|
|
|
|
(->> (cube 4 (* 1.2 teensy2-length) 4)
|
|
|
|
|
(translate [(+ 2 teensy-pcb-thickness) -30 (- -1 (/ teensy-width 2))]))
|
|
|
|
|
(->> (cube teensy-pcb-thickness (* 0.2 teensy2-length) 3)
|
|
|
|
|
(translate [(/ teensy-pcb-thickness 2) (+ (* 0.5 teensy2-length) -30) (+ 1.5 (/ teensy-width 2))]))
|
|
|
|
|
(->> (cube 4 (* 0.2 teensy2-length) 4)
|
|
|
|
|
(translate [(+ 2 teensy-pcb-thickness) (+ (* 0.5 teensy2-length) -30) (+ 1 (/ teensy-width 2))]))
|
|
|
|
|
))))
|
|
|
|
|
|
|
|
|
|
(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])
|
|
|
|
|
(rotate (/ π 2) [0 1 0])
|
|
|
|
|
(on-wall-place 0 20))))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
(defn hex-spacer [column row radius]
|
|
|
|
|
(let [position (key-position column row [0 0 0])
|
|
|
|
|
column-offset (/ mount-width 2)
|
|
|
|
|
row-offset (/ mount-height 2)
|
|
|
|
|
shift-right (= column lastcol)
|
|
|
|
|
shift-left (= column 0)
|
|
|
|
|
shift-up (and (not (or shift-right shift-left)) (= row 0))
|
|
|
|
|
shift-down (and (not (or shift-right shift-left)) (= row lastrow))
|
|
|
|
|
is-vertical (or shift-left shift-right)
|
|
|
|
|
col-angle (+ (* β (- centercol column)) (/ π 12))
|
|
|
|
|
row-angle (* α (- row centerrow))
|
|
|
|
|
height 10]
|
|
|
|
|
(->> (cylinder radius height)
|
|
|
|
|
(rotate (if is-vertical (/ π 6) 0) [0 0 1])
|
|
|
|
|
(translate [(first position) (second position) (/ height 2)])
|
|
|
|
|
; (translate [(* (if shift-right 1 (if shift-left -1 0)) (- column-offset (* wall-offset (Math/abs (Math/sin col-angle)))))
|
|
|
|
|
; (* (if shift-up 1 (if shift-down -1 0)) (- row-offset (* wall-offset (Math/abs (Math/sin row-angle)))))
|
|
|
|
|
; 0])
|
|
|
|
|
(translate [(* (if shift-right 1 (if shift-left -1 0)) column-offset)
|
|
|
|
|
(* (if shift-up 1 (if shift-down -1 0)) row-offset)
|
|
|
|
|
0])
|
|
|
|
|
(translate [(* wall-offset (Math/sin col-angle))
|
|
|
|
|
(* wall-offset (Math/sin row-angle))
|
|
|
|
|
0])
|
|
|
|
|
(with-fn 6))))
|
|
|
|
|
|
|
|
|
|
(defn hex-spacer-shapes [radius]
|
|
|
|
|
(union (hex-spacer 0 0 radius)
|
|
|
|
|
(hex-spacer 0 cornerrow radius)
|
|
|
|
|
(hex-spacer 3 lastrow radius)
|
|
|
|
|
(hex-spacer 2 0 radius)
|
|
|
|
|
(hex-spacer lastcol (dec cornerrow) radius)
|
|
|
|
|
))
|
|
|
|
|
(def hex-spacer-radius (/ 5.42 2))
|
|
|
|
|
(def hex-spacer-holes (hex-spacer-shapes hex-spacer-radius))
|
|
|
|
|
(def hex-spacer-outers (hex-spacer-shapes (+ hex-spacer-radius 1.6)))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;; teensy info
|
|
|
|
|
; base width - 18
|
|
|
|
|
; height - 1.45
|
|
|
|
|
;
|
|
|
|
|
|
|
|
|
|
(spit "things/right.scad"
|
|
|
|
|
(write-scad (union
|
|
|
|
|
key-holes
|
|
|
|
|
connectors
|
|
|
|
|
thumb
|
|
|
|
|
thumb-connectors
|
|
|
|
|
(difference case-walls rj9-space usb-cutout hex-spacer-holes)
|
|
|
|
|
rj9-holder
|
|
|
|
|
(if (= nrows 4) teensy-holder)
|
|
|
|
|
hex-spacer-outers
|
|
|
|
|
; thumbcaps
|
|
|
|
|
; caps
|
|
|
|
|
)))
|
|
|
|
|
|
|
|
|
|
|