-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathsqdg.py
138 lines (122 loc) · 5.58 KB
/
sqdg.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
import cadquery as cq, math
from typing import Optional
from dataclasses import dataclass
rot_axis_z = ((0, 0, 0), (0, 0, 1))
@dataclass
class Config:
x: float = 18 # Width of key (not keycap) outline; length for thumbs
y: float = 17 # Length of key (not keycap) outline; width for thumbs
r: float = 60 # Convex surface radius
eh: float = 2 # Top/Bottom edge max height
gap: float = 1 # Gap between cap outlines
cr: float = 60 # Radius of concave/cut shape; only for alphas
crh: Optional[float] = None # Horizontal radius of cut shape
ch: float = 1.5 # Minimum height in the center; only for alphas
sh: Optional[float] = None # Height at which to be sliced, measured from top of stem; only for alphas
ang: float = 30 # Angle of tangent of convex surface hits the top/bottom edge
off: float = 0 # X-offset of cut sphere from the center of cap
soff: Optional[float] = None# X-offset of center of stem from the center of cap
sl: float = 3.5 # Stem length
voff: Optional[float] = None# Y-offset for aligning the edge of thumb keys
tuni: bool = False # Uniform surface for thumbs
def __add__(self, c: 'Config'):
default = vars(Config())
ret = Config()
for k, v in default.items():
if getattr(self, k) != default[k]:
setattr(ret, k, getattr(self, k))
if getattr(c, k) != default[k]:
setattr(ret, k, getattr(c, k))
return ret
class Boxed:
def __init__(self, thing, x, y):
self.thing = thing
self.x = x
self.y = y
def hcat(self, other):
total_x, total_y = self.x + other.x, max(self.y, other.y)
return Boxed(self.thing.translate(((self.x - total_x) / 2, 0, 0))
.union(other.thing.translate(((total_x - other.x) / 2, 0, 0))), total_x, total_y)
def vcat(self, other):
total_x, total_y = max(self.x, other.x), self.y + other.y
return Boxed(self.thing.translate((0, -self.y / 2, 0))
.union(other.thing.translate((0, other.y / 2, 0))), total_x, total_y)
def move(self, x, y):
return Boxed(self.thing.translate((x, y, 0)), self.x, self.y)
def unwrap(self):
return self.thing
def __or__(self, other): return self.hcat(other)
def __add__(self, other): return self.vcat(other)
def add_stem(obj, c: Config, thumb=False):
stem_dist = 5.7
stem = (
cq.Workplane().box(1.2, 3, c.sl, (True, True, False)).faces("<X or |Z").shell(-0.8)
.translate((stem_dist / 2, 0, -c.sl))
)
stems = stem.union(stem.rotate(*rot_axis_z, 180))
if thumb:
stems = stems.rotate(*rot_axis_z, 90)
return (
obj.union(obj.wires("<Z").toPending().extrude(-1, combine=False).faces("|Z").shell(-1))
.union(stems.translate((c.soff if c.soff != None else c.off, 0, 0)))
)
def base(c: Config, pos: str):
base = cq.Workplane().box(c.x - c.gap, c.y - c.gap, 10, (True, True, False)).edges("|Z").chamfer(1)
if pos == "mid":
return add_stem(
base
.intersect(cq.Workplane().transformed(offset=(c.off, 0, 10 - c.r)).sphere(c.r)),
c,
)
y_mul = -1 if pos == "top" else 1
return add_stem(
base
.intersect(cq.Workplane().transformed(offset=(c.off,
y_mul * (c.r * math.sin(math.radians(c.ang)) - c.y / 2),
c.eh - c.r * math.cos(math.radians(c.ang)))).sphere(c.r)),
c,
)
def ellipsoid(base, y, x=None):
if x != None:
return base.sketch().ellipse(x, y).push([(0, y)]).rect(2 * x, 2 * y, mode="s").finalize().revolve(360, axisEnd=(1, 0))
return base.sphere(y)
def col(c: Config):
b = lambda p: base(c, p)
cut_shape = ellipsoid(cq.Workplane().transformed(offset=(c.off, 0, c.cr + c.ch)), c.cr, c.crh)
col = (
b("mid").union(b("top").translate((0, c.y))).union(b("bot").translate((0, -c.y)))
.cut(cut_shape)
)
if c.sh != None:
col = col.transformed(offset=(0, 0, c.sh)).split(keepBottom=True)
return Boxed(col, c.x, c.y * 3)
def thumb(c: Config, pos: str):
x_mul = 0 if not c.tuni else -1 if pos == "top" else 0 if pos == "mid" else 1
voff = c.voff if c.voff != None else 0
cut = cq.Workplane().transformed(offset=(x_mul * c.y,
x_mul * voff + c.r * math.sin(math.radians(c.ang)) - c.x / 2,
c.eh - c.r * math.cos(math.radians(c.ang))))
if c.voff != None:
if not c.tuni:
cut = cut.transformed((0, 0, -math.degrees(-math.atan2(c.voff, c.y))))
# cut = cut.transformed((0, 0, 13.62093739938678))
cut = ellipsoid(cut, c.r, 100)
else:
cut = cut.sphere(c.r)
return add_stem(
cq.Workplane().box(c.y - c.gap, c.x - c.gap, 10, (True, True, False))
.edges("|Z").chamfer(1)
.intersect(cut),
c, thumb=True
)
def thumb_col(c: Config, rotated=True, offset=False):
if rotated:
offset = False
T = lambda p: thumb(c, p)
voff = c.voff if c.voff != None and offset else 0
thumbs = T("mid").union(T("bot").translate((-c.y, -voff, 0))).union(T("top").translate((c.y, voff, 0)))
if rotated:
return Boxed(thumbs.rotate(*rot_axis_z, -90), c.x, c.y * 3)
return Boxed(thumbs, c.y * 3, c.x)
def export_color(obj, name):
cq.Assembly(obj, color=cq.Color("gray")).save(name + ".step")